author | Rich Felker <dalias@aerifal.cx> | 2018年06月20日 00:07:09 -0400 |
---|---|---|
committer | Rich Felker <dalias@aerifal.cx> | 2018年06月20日 00:07:09 -0400 |
commit | 0cd2be231481d68d244662bde25ad9cadbd7221d (patch) | |
tree | 586cb81d3f4ffae60276626b8a9b5a1b3ef59a34 /src/ipc/semctl.c | |
parent | 7ea235b1be38c57c49b164c9762cf90be02dbc05 (diff) | |
download | musl-0cd2be231481d68d244662bde25ad9cadbd7221d.tar.gz |
-rw-r--r-- | src/ipc/semctl.c | 26 |
diff --git a/src/ipc/semctl.c b/src/ipc/semctl.c index 673a9a8c..941e2813 100644 --- a/src/ipc/semctl.c +++ b/src/ipc/semctl.c @@ -1,8 +1,13 @@ #include <sys/sem.h> #include <stdarg.h> +#include <endian.h> #include "syscall.h" #include "ipc.h" +#if __BYTE_ORDER != __BIG_ENDIAN +#undef SYSCALL_IPC_BROKEN_MODE +#endif + union semun { int val; struct semid_ds *buf; @@ -20,9 +25,26 @@ int semctl(int id, int num, int cmd, ...) arg = va_arg(ap, union semun); va_end(ap); } +#ifdef SYSCALL_IPC_BROKEN_MODE + struct semid_ds tmp; + if (cmd == IPC_SET) { + tmp = *arg.buf; + tmp.sem_perm.mode *= 0x10000U; + arg.buf = &tmp; + } +#endif #ifdef SYS_semctl - return syscall(SYS_semctl, id, num, cmd | IPC_64, arg.buf); + int r = __syscall(SYS_semctl, id, num, cmd | IPC_64, arg.buf); #else - return syscall(SYS_ipc, IPCOP_semctl, id, num, cmd | IPC_64, &arg.buf); + int r = __syscall(SYS_ipc, IPCOP_semctl, id, num, cmd | IPC_64, &arg.buf); +#endif +#ifdef SYSCALL_IPC_BROKEN_MODE + if (r >= 0) switch (cmd) { + case IPC_STAT: + case SEM_STAT: + case SEM_STAT_ANY: + arg.buf->sem_perm.mode >>= 16; + } #endif + return __syscall_ret(r); } |