xtensa-specific fixes for linux-user:
- fix flushing registers for signal processing in call8 and call12 frames; - fix PC value for restarted syscalls; - fix sysv IPC structures; - fix fadvise64 syscall; generic fixes for linux-user: - fix QEMU assertion in multithreaded application by calling cpu_copy under clone_lock; - fix mq_getsetattr implementation; - fix error propagation in clock_gettime; - implement clock_settime. -----BEGIN PGP SIGNATURE----- iQJHBAABCAAxFiEEK2eFS5jlMn3N6xfYUfnMkfg/oEQFAlrCYzwTHGpjbXZia2Jj QGdtYWlsLmNvbQAKCRBR+cyR+D+gRD1qD/9ZYl4iEoNJPzRAafwIpJf6hNv6SU0i 2QBDSVWuyFjIZWZ2i2he6N3PZPTJ2YuCkG7sr/nhd8uWgFuULCUqedbgoheW+Jwf Wfn2F/d8qNhLIQv6XvNxUoIsKtsx+/359VegToz1XoxgD0dQYfpCtODK+kVfHMay 1OXvY0Yy0p5e0G0et/uS6dl2dB3wibWtaaxMFcIX/nVbhOrwZLhOIAz58NhHAqvI Tq+pLqHheKRpsvNl2cOs8nhQ1X9m8nlhVjrIuv/V21P6fRZc0bIfINqHx7g8FRiQ O+B4MBGp0wgXbHKjTkOUUUn0FQBVqnv9xlfeLolu67usnX8hN5NuNYiJVVtFmy6R ChqvM5zDzykpo+wiPIn5nBzZvDF0H3rnRpGiZaboH0jdDJb6ZyUVU8PWIj6KbTLB uDaJ0Rn7MLZnBiqnNtME7cpLm+hDlIIHXu8PDFrSz8NqnoPQbE+OZwlAEnos05uL mB1roPALIeEAHvA64b+M2REy4/qpm+rZdjDXU6iGVXrdjN6bizykOIzyzhXikBpP Kpf5HuKFHLIHnFavZCqg3tJGkXB+0AmJV7p1XntXnZ4JWOP0OStbg/f21XOrZVVd RDpP2nmSzyP6EwHAIxKIDQPVEVeWbXnpAWz7qm++eOBXRUIcbtQVormXfmupR9kI /H8C+5fGf2LZeA== =NiR+ -----END PGP SIGNATURE----- Merge remote-tracking branch 'remotes/xtensa/tags/20180402-xtensa' into staging xtensa-specific fixes for linux-user: - fix flushing registers for signal processing in call8 and call12 frames; - fix PC value for restarted syscalls; - fix sysv IPC structures; - fix fadvise64 syscall; generic fixes for linux-user: - fix QEMU assertion in multithreaded application by calling cpu_copy under clone_lock; - fix mq_getsetattr implementation; - fix error propagation in clock_gettime; - implement clock_settime. # gpg: Signature made Mon 02 Apr 2018 18:07:08 BST # gpg: using RSA key 51F9CC91F83FA044 # gpg: Good signature from "Max Filippov <filippov@cadence.com>" # gpg: aka "Max Filippov <max.filippov@cogentembedded.com>" # gpg: aka "Max Filippov <jcmvbkbc@gmail.com>" # Primary key fingerprint: 2B67 854B 98E5 327D CDEB 17D8 51F9 CC91 F83F A044 * remotes/xtensa/tags/20180402-xtensa: target/xtensa: linux-user: fix fadvise64 call linux-user: implement clock_settime linux-user: fix error propagation in clock_gettime target/xtensa: linux-user: fix sysv IPC structures linux-user: fix mq_getsetattr implementation linux-user: call cpu_copy under clone_lock target/xtensa: linux-user: rewind pc for restarted syscall target/xtensa: fix flush_window_regs Signed-off-by: Peter Maydell <peter.maydell@linaro.org>
This commit is contained in:
commit
9abfc88af3
@ -4006,6 +4006,9 @@ void cpu_loop(CPUXtensaState *env)
|
||||
break;
|
||||
|
||||
case -TARGET_ERESTARTSYS:
|
||||
env->pc -= 3;
|
||||
break;
|
||||
|
||||
case -TARGET_QEMU_ESIGRETURN:
|
||||
break;
|
||||
}
|
||||
|
@ -7094,52 +7094,45 @@ static abi_ulong get_sigframe(struct target_sigaction *sa,
|
||||
|
||||
static int flush_window_regs(CPUXtensaState *env)
|
||||
{
|
||||
const uint32_t nareg_mask = env->config->nareg - 1;
|
||||
uint32_t wb = env->sregs[WINDOW_BASE];
|
||||
uint32_t ws = (xtensa_replicate_windowstart(env) >> (wb + 1)) &
|
||||
((1 << env->config->nareg / 4) - 1);
|
||||
uint32_t d = ctz32(ws) + 1;
|
||||
uint32_t sp;
|
||||
abi_long ret = 0;
|
||||
uint32_t ws = xtensa_replicate_windowstart(env) >> (wb + 1);
|
||||
unsigned d = ctz32(ws) + 1;
|
||||
unsigned i;
|
||||
int ret = 0;
|
||||
|
||||
wb += d;
|
||||
ws >>= d;
|
||||
for (i = d; i < env->config->nareg / 4; i += d) {
|
||||
uint32_t ssp, osp;
|
||||
unsigned j;
|
||||
|
||||
xtensa_sync_phys_from_window(env);
|
||||
sp = env->phys_regs[(wb * 4 + 1) & nareg_mask];
|
||||
|
||||
while (ws && ret == 0) {
|
||||
int d;
|
||||
int i;
|
||||
int idx;
|
||||
ws >>= d;
|
||||
xtensa_rotate_window(env, d);
|
||||
|
||||
if (ws & 0x1) {
|
||||
ws >>= 1;
|
||||
ssp = env->regs[5];
|
||||
d = 1;
|
||||
} else if (ws & 0x2) {
|
||||
ws >>= 2;
|
||||
ssp = env->regs[9];
|
||||
ret |= get_user_ual(osp, env->regs[1] - 12);
|
||||
osp -= 32;
|
||||
d = 2;
|
||||
for (i = 0; i < 4; ++i) {
|
||||
idx = (wb * 4 + 4 + i) & nareg_mask;
|
||||
ret |= put_user_ual(env->phys_regs[idx], sp + (i - 12) * 4);
|
||||
}
|
||||
} else if (ws & 0x4) {
|
||||
ws >>= 3;
|
||||
ssp = env->regs[13];
|
||||
ret |= get_user_ual(osp, env->regs[1] - 12);
|
||||
osp -= 48;
|
||||
d = 3;
|
||||
for (i = 0; i < 8; ++i) {
|
||||
idx = (wb * 4 + 4 + i) & nareg_mask;
|
||||
ret |= put_user_ual(env->phys_regs[idx], sp + (i - 16) * 4);
|
||||
}
|
||||
} else {
|
||||
g_assert_not_reached();
|
||||
}
|
||||
sp = env->phys_regs[((wb + d) * 4 + 1) & nareg_mask];
|
||||
for (i = 0; i < 4; ++i) {
|
||||
idx = (wb * 4 + i) & nareg_mask;
|
||||
ret |= put_user_ual(env->phys_regs[idx], sp + (i - 4) * 4);
|
||||
|
||||
for (j = 0; j < 4; ++j) {
|
||||
ret |= put_user_ual(env->regs[j], ssp - 16 + j * 4);
|
||||
}
|
||||
for (j = 4; j < d * 4; ++j) {
|
||||
ret |= put_user_ual(env->regs[j], osp - 16 + j * 4);
|
||||
}
|
||||
wb += d;
|
||||
}
|
||||
xtensa_rotate_window(env, d);
|
||||
g_assert(env->sregs[WINDOW_BASE] == wb);
|
||||
return ret == 0;
|
||||
}
|
||||
|
||||
|
@ -6346,6 +6346,10 @@ static int do_fork(CPUArchState *env, unsigned int flags, abi_ulong newsp,
|
||||
|
||||
ts = g_new0(TaskState, 1);
|
||||
init_task_state(ts);
|
||||
|
||||
/* Grab a mutex so that thread setup appears atomic. */
|
||||
pthread_mutex_lock(&clone_lock);
|
||||
|
||||
/* we create a new CPU instance. */
|
||||
new_env = cpu_copy(env);
|
||||
/* Init regs that differ from the parent. */
|
||||
@ -6364,9 +6368,6 @@ static int do_fork(CPUArchState *env, unsigned int flags, abi_ulong newsp,
|
||||
cpu_set_tls (new_env, newtls);
|
||||
}
|
||||
|
||||
/* Grab a mutex so that thread setup appears atomic. */
|
||||
pthread_mutex_lock(&clone_lock);
|
||||
|
||||
memset(&info, 0, sizeof(info));
|
||||
pthread_mutex_init(&info.mutex, NULL);
|
||||
pthread_mutex_lock(&info.mutex);
|
||||
@ -11508,7 +11509,7 @@ abi_long do_syscall(void *cpu_env, int num, abi_long arg1,
|
||||
|
||||
#ifdef TARGET_NR_fadvise64_64
|
||||
case TARGET_NR_fadvise64_64:
|
||||
#if defined(TARGET_PPC)
|
||||
#if defined(TARGET_PPC) || defined(TARGET_XTENSA)
|
||||
/* 6 args: fd, advice, offset (high, low), len (high, low) */
|
||||
ret = arg2;
|
||||
arg2 = arg3;
|
||||
@ -11877,13 +11878,25 @@ abi_long do_syscall(void *cpu_env, int num, abi_long arg1,
|
||||
goto unimplemented_nowarn;
|
||||
#endif
|
||||
|
||||
#ifdef TARGET_NR_clock_settime
|
||||
case TARGET_NR_clock_settime:
|
||||
{
|
||||
struct timespec ts;
|
||||
|
||||
ret = target_to_host_timespec(&ts, arg2);
|
||||
if (!is_error(ret)) {
|
||||
ret = get_errno(clock_settime(arg1, &ts));
|
||||
}
|
||||
break;
|
||||
}
|
||||
#endif
|
||||
#ifdef TARGET_NR_clock_gettime
|
||||
case TARGET_NR_clock_gettime:
|
||||
{
|
||||
struct timespec ts;
|
||||
ret = get_errno(clock_gettime(arg1, &ts));
|
||||
if (!is_error(ret)) {
|
||||
host_to_target_timespec(arg2, &ts);
|
||||
ret = host_to_target_timespec(arg2, &ts);
|
||||
}
|
||||
break;
|
||||
}
|
||||
@ -12091,15 +12104,16 @@ abi_long do_syscall(void *cpu_env, int num, abi_long arg1,
|
||||
{
|
||||
struct mq_attr posix_mq_attr_in, posix_mq_attr_out;
|
||||
ret = 0;
|
||||
if (arg3 != 0) {
|
||||
ret = mq_getattr(arg1, &posix_mq_attr_out);
|
||||
copy_to_user_mq_attr(arg3, &posix_mq_attr_out);
|
||||
}
|
||||
if (arg2 != 0) {
|
||||
copy_from_user_mq_attr(&posix_mq_attr_in, arg2);
|
||||
ret |= mq_setattr(arg1, &posix_mq_attr_in, &posix_mq_attr_out);
|
||||
ret = get_errno(mq_setattr(arg1, &posix_mq_attr_in,
|
||||
&posix_mq_attr_out));
|
||||
} else if (arg3 != 0) {
|
||||
ret = get_errno(mq_getattr(arg1, &posix_mq_attr_out));
|
||||
}
|
||||
if (ret == 0 && arg3 != 0) {
|
||||
copy_to_user_mq_attr(arg3, &posix_mq_attr_out);
|
||||
}
|
||||
|
||||
}
|
||||
break;
|
||||
#endif
|
||||
|
@ -8,21 +8,44 @@ struct target_ipc_perm {
|
||||
abi_uint cuid; /* Creator's user ID. */
|
||||
abi_uint cgid; /* Creator's group ID. */
|
||||
abi_uint mode; /* Read/write permission. */
|
||||
abi_ushort __seq; /* Sequence number. */
|
||||
abi_ulong __seq; /* Sequence number. */
|
||||
abi_ulong __unused1;
|
||||
abi_ulong __unused2;
|
||||
};
|
||||
|
||||
struct target_semid64_ds {
|
||||
struct target_ipc_perm sem_perm;
|
||||
#ifdef TARGET_WORDS_BIGENDIAN
|
||||
abi_ulong __unused1;
|
||||
abi_ulong sem_otime;
|
||||
abi_ulong __unused2;
|
||||
abi_ulong sem_ctime;
|
||||
#else
|
||||
abi_ulong sem_otime;
|
||||
abi_ulong __unused1;
|
||||
abi_ulong sem_ctime;
|
||||
abi_ulong __unused2;
|
||||
#endif
|
||||
abi_ulong sem_nsems;
|
||||
abi_ulong __unused3;
|
||||
abi_ulong __unused4;
|
||||
};
|
||||
#define TARGET_SEMID64_DS
|
||||
|
||||
struct target_shmid_ds {
|
||||
struct target_ipc_perm shm_perm; /* operation permission struct */
|
||||
abi_int shm_segsz; /* size of segment in bytes */
|
||||
abi_long shm_segsz; /* size of segment in bytes */
|
||||
abi_long shm_atime; /* time of last shmat() */
|
||||
abi_ulong __unused1;
|
||||
abi_long shm_dtime; /* time of last shmdt() */
|
||||
abi_long shm_ctime; /* time of last change by shmctl() */
|
||||
abi_ushort shm_cpid; /* pid of creator */
|
||||
abi_ushort shm_lpid; /* pid of last shmop */
|
||||
abi_ushort shm_nattch; /* number of current attaches */
|
||||
abi_ushort shm_unused; /* compatibility */
|
||||
abi_ulong __unused2;
|
||||
abi_long shm_ctime; /* time of last change by shmctl() */
|
||||
abi_ulong __unused3;
|
||||
abi_uint shm_cpid; /* pid of creator */
|
||||
abi_uint shm_lpid; /* pid of last shmop */
|
||||
abi_ulong shm_nattch; /* number of current attaches */
|
||||
abi_ulong __unused4;
|
||||
abi_ulong __unused5;
|
||||
};
|
||||
|
||||
#endif
|
||||
|
Loading…
Reference in New Issue
Block a user