linux-user pull request for June 2016
-----BEGIN PGP SIGNATURE----- Version: GnuPG v1 iQIVAwUAV1gdMrRIkN7ePJvAAQhLcg/+Kby99taEuewItrA1yDs75jxOlLqaJopd cVzo4LFRFPhIn4UEKqRQS0CGoIeU/DYOmObvuUzJxs2LyUoHoqmQOwEm5obC2a85 JrHo/NOppYBbyvvIEAAXzZDCZo0KZKVclrlT+AX5obpOSNSvAnKvEuLWq1aQ9WGN n4AzHuFEl885cd4nFd8VK/xth89bqz6U/z8CjgIuw3mczp1XNrK5IJJwAy5epHay GCBr9XHooW3SU971WS20RTRS0D33tKPHgCU3ZeZ3rKh4g3JNj6/ixdVgzi9NqFsQ 5DzAj/iBGhN1LtCOednRS6tUt32Bhy8G/g4O3GiXdejagAmNe2wz31cveNJ8S3W5 DK8SZAnJlz06zN5uIpOVQgDOqfXZkCp7ndq779QJoHOAnuOjJBcUbhw1myz2R3eR 6208tStWl3R0+ATEK8CZ7ejg1cUHvdzyqGJA+1nC2HaFUrBWipxN8jf2fz9vO/wG G7zNbahvVgyJWO7bPNK4mxkb6qkWCETnCnLJsq2ZbmtPEMcINjD8vNWLNvFGVG8b 2HbinDrzh0Z9Zik5gLZfiVyP5HFaWSrJn9QRVIgaUjuIH9n3/25sl9OvW/JLjxJ+ h2P17CLnAK6dhUYc4R3wQTx2X/N2FvO4DD8iMYOcgDY6fhZ2b6EEyE9yBgQrIDbF gU1AlC/CX+M= =AXqa -----END PGP SIGNATURE----- Merge remote-tracking branch 'remotes/riku/tags/pull-linux-user-20160608' into staging linux-user pull request for June 2016 # gpg: Signature made Wed 08 Jun 2016 14:27:14 BST # gpg: using RSA key 0xB44890DEDE3C9BC0 # gpg: Good signature from "Riku Voipio <riku.voipio@iki.fi>" # gpg: aka "Riku Voipio <riku.voipio@linaro.org>" * remotes/riku/tags/pull-linux-user-20160608: (44 commits) linux-user: In fork_end(), remove correct CPUs from CPU list linux-user: Special-case ERESTARTSYS in target_strerror() linux-user: Make target_strerror() return 'const char *' linux-user: Correct signedness of target_flock l_start and l_len fields linux-user: Use safe_syscall wrapper for ioctl linux-user: Use safe_syscall wrapper for accept and accept4 syscalls linux-user: Use safe_syscall wrapper for semop linux-user: Use safe_syscall wrapper for epoll_wait syscalls linux-user: Use safe_syscall wrapper for poll and ppoll syscalls linux-user: Use safe_syscall wrapper for sleep syscalls linux-user: Use safe_syscall wrapper for rt_sigtimedwait syscall linux-user: Use safe_syscall wrapper for flock linux-user: Use safe_syscall wrapper for mq_timedsend and mq_timedreceive linux-user: Use safe_syscall wrapper for msgsnd and msgrcv linux-user: Use safe_syscall wrapper for send* and recv* syscalls linux-user: Use safe_syscall wrapper for connect syscall linux-user: Use safe_syscall wrapper for readv and writev syscalls linux-user: Fix error conversion in 64-bit fadvise syscall linux-user: Fix NR_fadvise64 and NR_fadvise64_64 for 32-bit guests linux-user: Fix handling of arm_fadvise64_64 syscall ... Signed-off-by: Peter Maydell <peter.maydell@linaro.org> Conflicts: configure scripts/qemu-binfmt-conf.sh
This commit is contained in:
commit
b66e10e4c9
38
configure
vendored
38
configure
vendored
@ -3800,8 +3800,8 @@ if compile_prog "" "" ; then
|
||||
epoll=yes
|
||||
fi
|
||||
|
||||
# epoll_create1 and epoll_pwait are later additions
|
||||
# so we must check separately for their presence
|
||||
# epoll_create1 is a later addition
|
||||
# so we must check separately for its presence
|
||||
epoll_create1=no
|
||||
cat > $TMPC << EOF
|
||||
#include <sys/epoll.h>
|
||||
@ -3823,20 +3823,6 @@ if compile_prog "" "" ; then
|
||||
epoll_create1=yes
|
||||
fi
|
||||
|
||||
epoll_pwait=no
|
||||
cat > $TMPC << EOF
|
||||
#include <sys/epoll.h>
|
||||
|
||||
int main(void)
|
||||
{
|
||||
epoll_pwait(0, 0, 0, 0, 0);
|
||||
return 0;
|
||||
}
|
||||
EOF
|
||||
if compile_prog "" "" ; then
|
||||
epoll_pwait=yes
|
||||
fi
|
||||
|
||||
# check for sendfile support
|
||||
sendfile=no
|
||||
cat > $TMPC << EOF
|
||||
@ -4528,6 +4514,19 @@ if compile_prog "" "" ; then
|
||||
have_fsxattr=yes
|
||||
fi
|
||||
|
||||
##########################################
|
||||
# check if rtnetlink.h exists and is useful
|
||||
have_rtnetlink=no
|
||||
cat > $TMPC << EOF
|
||||
#include <linux/rtnetlink.h>
|
||||
int main(void) {
|
||||
return IFLA_PROTO_DOWN;
|
||||
}
|
||||
EOF
|
||||
if compile_prog "" "" ; then
|
||||
have_rtnetlink=yes
|
||||
fi
|
||||
|
||||
#################################################
|
||||
# Sparc implicitly links with --relax, which is
|
||||
# incompatible with -r, so --no-relax should be
|
||||
@ -5135,9 +5134,6 @@ fi
|
||||
if test "$epoll_create1" = "yes" ; then
|
||||
echo "CONFIG_EPOLL_CREATE1=y" >> $config_host_mak
|
||||
fi
|
||||
if test "$epoll_pwait" = "yes" ; then
|
||||
echo "CONFIG_EPOLL_PWAIT=y" >> $config_host_mak
|
||||
fi
|
||||
if test "$sendfile" = "yes" ; then
|
||||
echo "CONFIG_SENDFILE=y" >> $config_host_mak
|
||||
fi
|
||||
@ -5482,6 +5478,10 @@ if test "$rdma" = "yes" ; then
|
||||
echo "CONFIG_RDMA=y" >> $config_host_mak
|
||||
fi
|
||||
|
||||
if test "$have_rtnetlink" = "yes" ; then
|
||||
echo "CONFIG_RTNETLINK=y" >> $config_host_mak
|
||||
fi
|
||||
|
||||
# Hold two types of flag:
|
||||
# CONFIG_THREAD_SETNAME_BYTHREAD - we've got a way of setting the name on
|
||||
# a thread we have a handle to
|
||||
|
13
gdbstub.c
13
gdbstub.c
@ -1493,19 +1493,6 @@ void gdb_exit(CPUArchState *env, int code)
|
||||
}
|
||||
|
||||
#ifdef CONFIG_USER_ONLY
|
||||
int
|
||||
gdb_queuesig (void)
|
||||
{
|
||||
GDBState *s;
|
||||
|
||||
s = gdbserver_state;
|
||||
|
||||
if (gdbserver_fd < 0 || s->fd < 0)
|
||||
return 0;
|
||||
else
|
||||
return 1;
|
||||
}
|
||||
|
||||
int
|
||||
gdb_handlesig(CPUState *cpu, int sig)
|
||||
{
|
||||
|
@ -48,7 +48,6 @@ int use_gdb_syscalls(void);
|
||||
void gdb_set_stop_cpu(CPUState *cpu);
|
||||
void gdb_exit(CPUArchState *, int);
|
||||
#ifdef CONFIG_USER_ONLY
|
||||
int gdb_queuesig (void);
|
||||
int gdb_handlesig(CPUState *, int);
|
||||
void gdb_signalled(CPUArchState *, int);
|
||||
void gdbserver_fork(CPUState *);
|
||||
|
@ -24,6 +24,7 @@
|
||||
* -1-and-errno-set convention is done by the calling wrapper.
|
||||
*/
|
||||
safe_syscall_base:
|
||||
.cfi_startproc
|
||||
/* This saves a frame pointer and aligns the stack for the syscall.
|
||||
* (It's unclear if the syscall ABI has the same stack alignment
|
||||
* requirements as the userspace function call ABI, but better safe than
|
||||
@ -31,6 +32,8 @@ safe_syscall_base:
|
||||
* does not list any ABI differences regarding stack alignment.)
|
||||
*/
|
||||
push %rbp
|
||||
.cfi_adjust_cfa_offset 8
|
||||
.cfi_rel_offset rbp, 0
|
||||
|
||||
/* The syscall calling convention isn't the same as the
|
||||
* C one:
|
||||
@ -70,12 +73,19 @@ safe_syscall_start:
|
||||
safe_syscall_end:
|
||||
/* code path for having successfully executed the syscall */
|
||||
pop %rbp
|
||||
.cfi_remember_state
|
||||
.cfi_def_cfa_offset 8
|
||||
.cfi_restore rbp
|
||||
ret
|
||||
|
||||
return_ERESTARTSYS:
|
||||
/* code path when we didn't execute the syscall */
|
||||
.cfi_restore_state
|
||||
mov $-TARGET_ERESTARTSYS, %rax
|
||||
pop %rbp
|
||||
.cfi_def_cfa_offset 8
|
||||
.cfi_restore rbp
|
||||
ret
|
||||
.cfi_endproc
|
||||
|
||||
.size safe_syscall_base, .-safe_syscall_base
|
||||
|
@ -131,7 +131,7 @@ void fork_end(int child)
|
||||
Discard information about the parent threads. */
|
||||
CPU_FOREACH_SAFE(cpu, next_cpu) {
|
||||
if (cpu != thread_cpu) {
|
||||
QTAILQ_REMOVE(&cpus, thread_cpu, node);
|
||||
QTAILQ_REMOVE(&cpus, cpu, node);
|
||||
}
|
||||
}
|
||||
pending_cpus = 0;
|
||||
@ -3795,14 +3795,7 @@ void stop_all_tasks(void)
|
||||
/* Assumes contents are already zeroed. */
|
||||
void init_task_state(TaskState *ts)
|
||||
{
|
||||
int i;
|
||||
|
||||
ts->used = 1;
|
||||
ts->first_free = ts->sigqueue_table;
|
||||
for (i = 0; i < MAX_SIGQUEUE_SIZE - 1; i++) {
|
||||
ts->sigqueue_table[i].next = &ts->sigqueue_table[i + 1];
|
||||
}
|
||||
ts->sigqueue_table[i].next = NULL;
|
||||
}
|
||||
|
||||
CPUArchState *cpu_copy(CPUArchState *env)
|
||||
|
@ -78,16 +78,9 @@ struct vm86_saved_state {
|
||||
|
||||
#define MAX_SIGQUEUE_SIZE 1024
|
||||
|
||||
struct sigqueue {
|
||||
struct sigqueue *next;
|
||||
target_siginfo_t info;
|
||||
};
|
||||
|
||||
struct emulated_sigtable {
|
||||
int pending; /* true if signal is pending */
|
||||
struct sigqueue *first;
|
||||
struct sigqueue info; /* in order to always have memory for the
|
||||
first signal, we put it here */
|
||||
target_siginfo_t info;
|
||||
};
|
||||
|
||||
/* NOTE: we force a big alignment so that the stack stored after is
|
||||
@ -123,14 +116,32 @@ typedef struct TaskState {
|
||||
#endif
|
||||
uint32_t stack_base;
|
||||
int used; /* non zero if used */
|
||||
bool sigsegv_blocked; /* SIGSEGV blocked by guest */
|
||||
struct image_info *info;
|
||||
struct linux_binprm *bprm;
|
||||
|
||||
struct emulated_sigtable sync_signal;
|
||||
struct emulated_sigtable sigtab[TARGET_NSIG];
|
||||
struct sigqueue sigqueue_table[MAX_SIGQUEUE_SIZE]; /* siginfo queue */
|
||||
struct sigqueue *first_free; /* first free siginfo queue entry */
|
||||
int signal_pending; /* non zero if a signal may be pending */
|
||||
/* This thread's signal mask, as requested by the guest program.
|
||||
* The actual signal mask of this thread may differ:
|
||||
* + we don't let SIGSEGV and SIGBUS be blocked while running guest code
|
||||
* + sometimes we block all signals to avoid races
|
||||
*/
|
||||
sigset_t signal_mask;
|
||||
/* The signal mask imposed by a guest sigsuspend syscall, if we are
|
||||
* currently in the middle of such a syscall
|
||||
*/
|
||||
sigset_t sigsuspend_mask;
|
||||
/* Nonzero if we're leaving a sigsuspend and sigsuspend_mask is valid. */
|
||||
int in_sigsuspend;
|
||||
|
||||
/* Nonzero if process_pending_signals() needs to do something (either
|
||||
* handle a pending signal or unblock signals).
|
||||
* This flag is written from a signal handler so should be accessed via
|
||||
* the atomic_read() and atomic_write() functions. (It is not accessed
|
||||
* from multiple threads.)
|
||||
*/
|
||||
int signal_pending;
|
||||
|
||||
} __attribute__((aligned(16))) TaskState;
|
||||
|
||||
extern char *exec_path;
|
||||
@ -184,7 +195,7 @@ abi_long do_syscall(void *cpu_env, int num, abi_long arg1,
|
||||
void gemu_log(const char *fmt, ...) GCC_FMT_ATTR(1, 2);
|
||||
extern THREAD CPUState *thread_cpu;
|
||||
void cpu_loop(CPUArchState *env);
|
||||
char *target_strerror(int err);
|
||||
const char *target_strerror(int err);
|
||||
int get_osversion(void);
|
||||
void init_qemu_uname_release(void);
|
||||
void fork_start(void);
|
||||
@ -235,6 +246,12 @@ unsigned long init_guest_space(unsigned long host_start,
|
||||
* It's also OK to implement these with safe_syscall, though it will be
|
||||
* a little less efficient if a signal is delivered at the 'wrong' moment.
|
||||
*
|
||||
* Some non-interruptible syscalls need to be handled using block_signals()
|
||||
* to block signals for the duration of the syscall. This mainly applies
|
||||
* to code which needs to modify the data structures used by the
|
||||
* host_signal_handler() function and the functions it calls, including
|
||||
* all syscalls which change the thread's signal mask.
|
||||
*
|
||||
* (2) Interruptible syscalls
|
||||
*
|
||||
* These are guest syscalls that can be interrupted by signals and
|
||||
@ -266,6 +283,8 @@ unsigned long init_guest_space(unsigned long host_start,
|
||||
* you make in the implementation returns either -TARGET_ERESTARTSYS or
|
||||
* EINTR though.)
|
||||
*
|
||||
* block_signals() cannot be used for interruptible syscalls.
|
||||
*
|
||||
*
|
||||
* How and why the safe_syscall implementation works:
|
||||
*
|
||||
@ -352,6 +371,25 @@ long do_sigreturn(CPUArchState *env);
|
||||
long do_rt_sigreturn(CPUArchState *env);
|
||||
abi_long do_sigaltstack(abi_ulong uss_addr, abi_ulong uoss_addr, abi_ulong sp);
|
||||
int do_sigprocmask(int how, const sigset_t *set, sigset_t *oldset);
|
||||
/**
|
||||
* block_signals: block all signals while handling this guest syscall
|
||||
*
|
||||
* Block all signals, and arrange that the signal mask is returned to
|
||||
* its correct value for the guest before we resume execution of guest code.
|
||||
* If this function returns non-zero, then the caller should immediately
|
||||
* return -TARGET_ERESTARTSYS to the main loop, which will take the pending
|
||||
* signal and restart execution of the syscall.
|
||||
* If block_signals() returns zero, then the caller can continue with
|
||||
* emulation of the system call knowing that no signals can be taken
|
||||
* (and therefore that no race conditions will result).
|
||||
* This should only be called once, because if it is called a second time
|
||||
* it will always return non-zero. (Think of it like a mutex that can't
|
||||
* be recursively locked.)
|
||||
* Signals will be unblocked again by process_pending_signals().
|
||||
*
|
||||
* Return value: non-zero if there was a pending signal, zero if not.
|
||||
*/
|
||||
int block_signals(void); /* Returns non zero if signal pending */
|
||||
|
||||
#ifdef TARGET_I386
|
||||
/* vm86.c */
|
||||
|
@ -17,6 +17,7 @@
|
||||
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
#include "qemu/osdep.h"
|
||||
#include "qemu/bitops.h"
|
||||
#include <sys/ucontext.h>
|
||||
#include <sys/resource.h>
|
||||
|
||||
@ -190,125 +191,213 @@ void target_to_host_old_sigset(sigset_t *sigset,
|
||||
target_to_host_sigset(sigset, &d);
|
||||
}
|
||||
|
||||
int block_signals(void)
|
||||
{
|
||||
TaskState *ts = (TaskState *)thread_cpu->opaque;
|
||||
sigset_t set;
|
||||
int pending;
|
||||
|
||||
/* It's OK to block everything including SIGSEGV, because we won't
|
||||
* run any further guest code before unblocking signals in
|
||||
* process_pending_signals().
|
||||
*/
|
||||
sigfillset(&set);
|
||||
sigprocmask(SIG_SETMASK, &set, 0);
|
||||
|
||||
pending = atomic_xchg(&ts->signal_pending, 1);
|
||||
|
||||
return pending;
|
||||
}
|
||||
|
||||
/* Wrapper for sigprocmask function
|
||||
* Emulates a sigprocmask in a safe way for the guest. Note that set and oldset
|
||||
* are host signal set, not guest ones. This wraps the sigprocmask host calls
|
||||
* that should be protected (calls originated from guest)
|
||||
* are host signal set, not guest ones. Returns -TARGET_ERESTARTSYS if
|
||||
* a signal was already pending and the syscall must be restarted, or
|
||||
* 0 on success.
|
||||
* If set is NULL, this is guaranteed not to fail.
|
||||
*/
|
||||
int do_sigprocmask(int how, const sigset_t *set, sigset_t *oldset)
|
||||
{
|
||||
int ret;
|
||||
sigset_t val;
|
||||
sigset_t *temp = NULL;
|
||||
CPUState *cpu = thread_cpu;
|
||||
TaskState *ts = (TaskState *)cpu->opaque;
|
||||
bool segv_was_blocked = ts->sigsegv_blocked;
|
||||
TaskState *ts = (TaskState *)thread_cpu->opaque;
|
||||
|
||||
if (oldset) {
|
||||
*oldset = ts->signal_mask;
|
||||
}
|
||||
|
||||
if (set) {
|
||||
bool has_sigsegv = sigismember(set, SIGSEGV);
|
||||
val = *set;
|
||||
temp = &val;
|
||||
int i;
|
||||
|
||||
sigdelset(temp, SIGSEGV);
|
||||
if (block_signals()) {
|
||||
return -TARGET_ERESTARTSYS;
|
||||
}
|
||||
|
||||
switch (how) {
|
||||
case SIG_BLOCK:
|
||||
if (has_sigsegv) {
|
||||
ts->sigsegv_blocked = true;
|
||||
}
|
||||
sigorset(&ts->signal_mask, &ts->signal_mask, set);
|
||||
break;
|
||||
case SIG_UNBLOCK:
|
||||
if (has_sigsegv) {
|
||||
ts->sigsegv_blocked = false;
|
||||
for (i = 1; i <= NSIG; ++i) {
|
||||
if (sigismember(set, i)) {
|
||||
sigdelset(&ts->signal_mask, i);
|
||||
}
|
||||
}
|
||||
break;
|
||||
case SIG_SETMASK:
|
||||
ts->sigsegv_blocked = has_sigsegv;
|
||||
ts->signal_mask = *set;
|
||||
break;
|
||||
default:
|
||||
g_assert_not_reached();
|
||||
}
|
||||
|
||||
/* Silently ignore attempts to change blocking status of KILL or STOP */
|
||||
sigdelset(&ts->signal_mask, SIGKILL);
|
||||
sigdelset(&ts->signal_mask, SIGSTOP);
|
||||
}
|
||||
|
||||
ret = sigprocmask(how, temp, oldset);
|
||||
|
||||
if (oldset && segv_was_blocked) {
|
||||
sigaddset(oldset, SIGSEGV);
|
||||
}
|
||||
|
||||
return ret;
|
||||
return 0;
|
||||
}
|
||||
|
||||
#if !defined(TARGET_OPENRISC) && !defined(TARGET_UNICORE32) && \
|
||||
!defined(TARGET_X86_64)
|
||||
/* Just set the guest's signal mask to the specified value; the
|
||||
* caller is assumed to have called block_signals() already.
|
||||
*/
|
||||
static void set_sigmask(const sigset_t *set)
|
||||
{
|
||||
TaskState *ts = (TaskState *)thread_cpu->opaque;
|
||||
|
||||
ts->signal_mask = *set;
|
||||
}
|
||||
#endif
|
||||
|
||||
/* siginfo conversion */
|
||||
|
||||
static inline void host_to_target_siginfo_noswap(target_siginfo_t *tinfo,
|
||||
const siginfo_t *info)
|
||||
{
|
||||
int sig = host_to_target_signal(info->si_signo);
|
||||
int si_code = info->si_code;
|
||||
int si_type;
|
||||
tinfo->si_signo = sig;
|
||||
tinfo->si_errno = 0;
|
||||
tinfo->si_code = info->si_code;
|
||||
|
||||
if (sig == TARGET_SIGILL || sig == TARGET_SIGFPE || sig == TARGET_SIGSEGV
|
||||
|| sig == TARGET_SIGBUS || sig == TARGET_SIGTRAP) {
|
||||
/* Should never come here, but who knows. The information for
|
||||
the target is irrelevant. */
|
||||
tinfo->_sifields._sigfault._addr = 0;
|
||||
} else if (sig == TARGET_SIGIO) {
|
||||
tinfo->_sifields._sigpoll._band = info->si_band;
|
||||
tinfo->_sifields._sigpoll._fd = info->si_fd;
|
||||
} else if (sig == TARGET_SIGCHLD) {
|
||||
tinfo->_sifields._sigchld._pid = info->si_pid;
|
||||
tinfo->_sifields._sigchld._uid = info->si_uid;
|
||||
tinfo->_sifields._sigchld._status
|
||||
/* This is awkward, because we have to use a combination of
|
||||
* the si_code and si_signo to figure out which of the union's
|
||||
* members are valid. (Within the host kernel it is always possible
|
||||
* to tell, but the kernel carefully avoids giving userspace the
|
||||
* high 16 bits of si_code, so we don't have the information to
|
||||
* do this the easy way...) We therefore make our best guess,
|
||||
* bearing in mind that a guest can spoof most of the si_codes
|
||||
* via rt_sigqueueinfo() if it likes.
|
||||
*
|
||||
* Once we have made our guess, we record it in the top 16 bits of
|
||||
* the si_code, so that tswap_siginfo() later can use it.
|
||||
* tswap_siginfo() will strip these top bits out before writing
|
||||
* si_code to the guest (sign-extending the lower bits).
|
||||
*/
|
||||
|
||||
switch (si_code) {
|
||||
case SI_USER:
|
||||
case SI_TKILL:
|
||||
case SI_KERNEL:
|
||||
/* Sent via kill(), tkill() or tgkill(), or direct from the kernel.
|
||||
* These are the only unspoofable si_code values.
|
||||
*/
|
||||
tinfo->_sifields._kill._pid = info->si_pid;
|
||||
tinfo->_sifields._kill._uid = info->si_uid;
|
||||
si_type = QEMU_SI_KILL;
|
||||
break;
|
||||
default:
|
||||
/* Everything else is spoofable. Make best guess based on signal */
|
||||
switch (sig) {
|
||||
case TARGET_SIGCHLD:
|
||||
tinfo->_sifields._sigchld._pid = info->si_pid;
|
||||
tinfo->_sifields._sigchld._uid = info->si_uid;
|
||||
tinfo->_sifields._sigchld._status
|
||||
= host_to_target_waitstatus(info->si_status);
|
||||
tinfo->_sifields._sigchld._utime = info->si_utime;
|
||||
tinfo->_sifields._sigchld._stime = info->si_stime;
|
||||
} else if (sig >= TARGET_SIGRTMIN) {
|
||||
tinfo->_sifields._rt._pid = info->si_pid;
|
||||
tinfo->_sifields._rt._uid = info->si_uid;
|
||||
/* XXX: potential problem if 64 bit */
|
||||
tinfo->_sifields._rt._sigval.sival_ptr
|
||||
tinfo->_sifields._sigchld._utime = info->si_utime;
|
||||
tinfo->_sifields._sigchld._stime = info->si_stime;
|
||||
si_type = QEMU_SI_CHLD;
|
||||
break;
|
||||
case TARGET_SIGIO:
|
||||
tinfo->_sifields._sigpoll._band = info->si_band;
|
||||
tinfo->_sifields._sigpoll._fd = info->si_fd;
|
||||
si_type = QEMU_SI_POLL;
|
||||
break;
|
||||
default:
|
||||
/* Assume a sigqueue()/mq_notify()/rt_sigqueueinfo() source. */
|
||||
tinfo->_sifields._rt._pid = info->si_pid;
|
||||
tinfo->_sifields._rt._uid = info->si_uid;
|
||||
/* XXX: potential problem if 64 bit */
|
||||
tinfo->_sifields._rt._sigval.sival_ptr
|
||||
= (abi_ulong)(unsigned long)info->si_value.sival_ptr;
|
||||
si_type = QEMU_SI_RT;
|
||||
break;
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
tinfo->si_code = deposit32(si_code, 16, 16, si_type);
|
||||
}
|
||||
|
||||
static void tswap_siginfo(target_siginfo_t *tinfo,
|
||||
const target_siginfo_t *info)
|
||||
{
|
||||
int sig = info->si_signo;
|
||||
tinfo->si_signo = tswap32(sig);
|
||||
tinfo->si_errno = tswap32(info->si_errno);
|
||||
tinfo->si_code = tswap32(info->si_code);
|
||||
int si_type = extract32(info->si_code, 16, 16);
|
||||
int si_code = sextract32(info->si_code, 0, 16);
|
||||
|
||||
if (sig == TARGET_SIGILL || sig == TARGET_SIGFPE || sig == TARGET_SIGSEGV
|
||||
|| sig == TARGET_SIGBUS || sig == TARGET_SIGTRAP) {
|
||||
tinfo->_sifields._sigfault._addr
|
||||
= tswapal(info->_sifields._sigfault._addr);
|
||||
} else if (sig == TARGET_SIGIO) {
|
||||
tinfo->_sifields._sigpoll._band
|
||||
= tswap32(info->_sifields._sigpoll._band);
|
||||
tinfo->_sifields._sigpoll._fd = tswap32(info->_sifields._sigpoll._fd);
|
||||
} else if (sig == TARGET_SIGCHLD) {
|
||||
tinfo->_sifields._sigchld._pid
|
||||
= tswap32(info->_sifields._sigchld._pid);
|
||||
tinfo->_sifields._sigchld._uid
|
||||
= tswap32(info->_sifields._sigchld._uid);
|
||||
tinfo->_sifields._sigchld._status
|
||||
= tswap32(info->_sifields._sigchld._status);
|
||||
tinfo->_sifields._sigchld._utime
|
||||
= tswapal(info->_sifields._sigchld._utime);
|
||||
tinfo->_sifields._sigchld._stime
|
||||
= tswapal(info->_sifields._sigchld._stime);
|
||||
} else if (sig >= TARGET_SIGRTMIN) {
|
||||
tinfo->_sifields._rt._pid = tswap32(info->_sifields._rt._pid);
|
||||
tinfo->_sifields._rt._uid = tswap32(info->_sifields._rt._uid);
|
||||
tinfo->_sifields._rt._sigval.sival_ptr
|
||||
= tswapal(info->_sifields._rt._sigval.sival_ptr);
|
||||
__put_user(info->si_signo, &tinfo->si_signo);
|
||||
__put_user(info->si_errno, &tinfo->si_errno);
|
||||
__put_user(si_code, &tinfo->si_code);
|
||||
|
||||
/* We can use our internal marker of which fields in the structure
|
||||
* are valid, rather than duplicating the guesswork of
|
||||
* host_to_target_siginfo_noswap() here.
|
||||
*/
|
||||
switch (si_type) {
|
||||
case QEMU_SI_KILL:
|
||||
__put_user(info->_sifields._kill._pid, &tinfo->_sifields._kill._pid);
|
||||
__put_user(info->_sifields._kill._uid, &tinfo->_sifields._kill._uid);
|
||||
break;
|
||||
case QEMU_SI_TIMER:
|
||||
__put_user(info->_sifields._timer._timer1,
|
||||
&tinfo->_sifields._timer._timer1);
|
||||
__put_user(info->_sifields._timer._timer2,
|
||||
&tinfo->_sifields._timer._timer2);
|
||||
break;
|
||||
case QEMU_SI_POLL:
|
||||
__put_user(info->_sifields._sigpoll._band,
|
||||
&tinfo->_sifields._sigpoll._band);
|
||||
__put_user(info->_sifields._sigpoll._fd,
|
||||
&tinfo->_sifields._sigpoll._fd);
|
||||
break;
|
||||
case QEMU_SI_FAULT:
|
||||
__put_user(info->_sifields._sigfault._addr,
|
||||
&tinfo->_sifields._sigfault._addr);
|
||||
break;
|
||||
case QEMU_SI_CHLD:
|
||||
__put_user(info->_sifields._sigchld._pid,
|
||||
&tinfo->_sifields._sigchld._pid);
|
||||
__put_user(info->_sifields._sigchld._uid,
|
||||
&tinfo->_sifields._sigchld._uid);
|
||||
__put_user(info->_sifields._sigchld._status,
|
||||
&tinfo->_sifields._sigchld._status);
|
||||
__put_user(info->_sifields._sigchld._utime,
|
||||
&tinfo->_sifields._sigchld._utime);
|
||||
__put_user(info->_sifields._sigchld._stime,
|
||||
&tinfo->_sifields._sigchld._stime);
|
||||
break;
|
||||
case QEMU_SI_RT:
|
||||
__put_user(info->_sifields._rt._pid, &tinfo->_sifields._rt._pid);
|
||||
__put_user(info->_sifields._rt._uid, &tinfo->_sifields._rt._uid);
|
||||
__put_user(info->_sifields._rt._sigval.sival_ptr,
|
||||
&tinfo->_sifields._rt._sigval.sival_ptr);
|
||||
break;
|
||||
default:
|
||||
g_assert_not_reached();
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
void host_to_target_siginfo(target_siginfo_t *tinfo, const siginfo_t *info)
|
||||
{
|
||||
host_to_target_siginfo_noswap(tinfo, info);
|
||||
@ -319,13 +408,18 @@ void host_to_target_siginfo(target_siginfo_t *tinfo, const siginfo_t *info)
|
||||
/* XXX: find a solution for 64 bit (additional malloced data is needed) */
|
||||
void target_to_host_siginfo(siginfo_t *info, const target_siginfo_t *tinfo)
|
||||
{
|
||||
info->si_signo = tswap32(tinfo->si_signo);
|
||||
info->si_errno = tswap32(tinfo->si_errno);
|
||||
info->si_code = tswap32(tinfo->si_code);
|
||||
info->si_pid = tswap32(tinfo->_sifields._rt._pid);
|
||||
info->si_uid = tswap32(tinfo->_sifields._rt._uid);
|
||||
info->si_value.sival_ptr =
|
||||
(void *)(long)tswapal(tinfo->_sifields._rt._sigval.sival_ptr);
|
||||
/* This conversion is used only for the rt_sigqueueinfo syscall,
|
||||
* and so we know that the _rt fields are the valid ones.
|
||||
*/
|
||||
abi_ulong sival_ptr;
|
||||
|
||||
__get_user(info->si_signo, &tinfo->si_signo);
|
||||
__get_user(info->si_errno, &tinfo->si_errno);
|
||||
__get_user(info->si_code, &tinfo->si_code);
|
||||
__get_user(info->si_pid, &tinfo->_sifields._rt._pid);
|
||||
__get_user(info->si_uid, &tinfo->_sifields._rt._uid);
|
||||
__get_user(sival_ptr, &tinfo->_sifields._rt._sigval.sival_ptr);
|
||||
info->si_value.sival_ptr = (void *)(long)sival_ptr;
|
||||
}
|
||||
|
||||
static int fatal_signal (int sig)
|
||||
@ -367,6 +461,7 @@ static int core_dump_signal(int sig)
|
||||
|
||||
void signal_init(void)
|
||||
{
|
||||
TaskState *ts = (TaskState *)thread_cpu->opaque;
|
||||
struct sigaction act;
|
||||
struct sigaction oact;
|
||||
int i, j;
|
||||
@ -382,6 +477,9 @@ void signal_init(void)
|
||||
target_to_host_signal_table[j] = i;
|
||||
}
|
||||
|
||||
/* Set the signal mask from the host mask. */
|
||||
sigprocmask(0, 0, &ts->signal_mask);
|
||||
|
||||
/* set all host signal handlers. ALL signals are blocked during
|
||||
the handlers to serialize them. */
|
||||
memset(sigact_table, 0, sizeof(sigact_table));
|
||||
@ -408,27 +506,6 @@ void signal_init(void)
|
||||
}
|
||||
}
|
||||
|
||||
/* signal queue handling */
|
||||
|
||||
static inline struct sigqueue *alloc_sigqueue(CPUArchState *env)
|
||||
{
|
||||
CPUState *cpu = ENV_GET_CPU(env);
|
||||
TaskState *ts = cpu->opaque;
|
||||
struct sigqueue *q = ts->first_free;
|
||||
if (!q)
|
||||
return NULL;
|
||||
ts->first_free = q->next;
|
||||
return q;
|
||||
}
|
||||
|
||||
static inline void free_sigqueue(CPUArchState *env, struct sigqueue *q)
|
||||
{
|
||||
CPUState *cpu = ENV_GET_CPU(env);
|
||||
TaskState *ts = cpu->opaque;
|
||||
|
||||
q->next = ts->first_free;
|
||||
ts->first_free = q;
|
||||
}
|
||||
|
||||
/* abort execution with signal */
|
||||
static void QEMU_NORETURN force_sig(int target_sig)
|
||||
@ -490,75 +567,21 @@ int queue_signal(CPUArchState *env, int sig, target_siginfo_t *info)
|
||||
{
|
||||
CPUState *cpu = ENV_GET_CPU(env);
|
||||
TaskState *ts = cpu->opaque;
|
||||
struct emulated_sigtable *k;
|
||||
struct sigqueue *q, **pq;
|
||||
abi_ulong handler;
|
||||
int queue;
|
||||
|
||||
trace_user_queue_signal(env, sig);
|
||||
k = &ts->sigtab[sig - 1];
|
||||
queue = gdb_queuesig ();
|
||||
handler = sigact_table[sig - 1]._sa_handler;
|
||||
|
||||
if (ts->sigsegv_blocked && sig == TARGET_SIGSEGV) {
|
||||
/* Guest has blocked SIGSEGV but we got one anyway. Assume this
|
||||
* is a forced SIGSEGV (ie one the kernel handles via force_sig_info
|
||||
* because it got a real MMU fault). A blocked SIGSEGV in that
|
||||
* situation is treated as if using the default handler. This is
|
||||
* not correct if some other process has randomly sent us a SIGSEGV
|
||||
* via kill(), but that is not easy to distinguish at this point,
|
||||
* so we assume it doesn't happen.
|
||||
*/
|
||||
handler = TARGET_SIG_DFL;
|
||||
}
|
||||
/* Currently all callers define siginfo structures which
|
||||
* use the _sifields._sigfault union member, so we can
|
||||
* set the type here. If that changes we should push this
|
||||
* out so the si_type is passed in by callers.
|
||||
*/
|
||||
info->si_code = deposit32(info->si_code, 16, 16, QEMU_SI_FAULT);
|
||||
|
||||
if (!queue && handler == TARGET_SIG_DFL) {
|
||||
if (sig == TARGET_SIGTSTP || sig == TARGET_SIGTTIN || sig == TARGET_SIGTTOU) {
|
||||
kill(getpid(),SIGSTOP);
|
||||
return 0;
|
||||
} else
|
||||
/* default handler : ignore some signal. The other are fatal */
|
||||
if (sig != TARGET_SIGCHLD &&
|
||||
sig != TARGET_SIGURG &&
|
||||
sig != TARGET_SIGWINCH &&
|
||||
sig != TARGET_SIGCONT) {
|
||||
force_sig(sig);
|
||||
} else {
|
||||
return 0; /* indicate ignored */
|
||||
}
|
||||
} else if (!queue && handler == TARGET_SIG_IGN) {
|
||||
/* ignore signal */
|
||||
return 0;
|
||||
} else if (!queue && handler == TARGET_SIG_ERR) {
|
||||
force_sig(sig);
|
||||
} else {
|
||||
pq = &k->first;
|
||||
if (sig < TARGET_SIGRTMIN) {
|
||||
/* if non real time signal, we queue exactly one signal */
|
||||
if (!k->pending)
|
||||
q = &k->info;
|
||||
else
|
||||
return 0;
|
||||
} else {
|
||||
if (!k->pending) {
|
||||
/* first signal */
|
||||
q = &k->info;
|
||||
} else {
|
||||
q = alloc_sigqueue(env);
|
||||
if (!q)
|
||||
return -EAGAIN;
|
||||
while (*pq != NULL)
|
||||
pq = &(*pq)->next;
|
||||
}
|
||||
}
|
||||
*pq = q;
|
||||
q->info = *info;
|
||||
q->next = NULL;
|
||||
k->pending = 1;
|
||||
/* signal that a new signal is pending */
|
||||
ts->signal_pending = 1;
|
||||
return 1; /* indicates that the signal was queued */
|
||||
}
|
||||
ts->sync_signal.info = *info;
|
||||
ts->sync_signal.pending = sig;
|
||||
/* signal that a new signal is pending */
|
||||
atomic_set(&ts->signal_pending, 1);
|
||||
return 1; /* indicates that the signal was queued */
|
||||
}
|
||||
|
||||
#ifndef HAVE_SAFE_SYSCALL
|
||||
@ -572,8 +595,13 @@ static void host_signal_handler(int host_signum, siginfo_t *info,
|
||||
void *puc)
|
||||
{
|
||||
CPUArchState *env = thread_cpu->env_ptr;
|
||||
CPUState *cpu = ENV_GET_CPU(env);
|
||||
TaskState *ts = cpu->opaque;
|
||||
|
||||
int sig;
|
||||
target_siginfo_t tinfo;
|
||||
ucontext_t *uc = puc;
|
||||
struct emulated_sigtable *k;
|
||||
|
||||
/* the CPU emulator uses some host signals to detect exceptions,
|
||||
we forward to it some signals */
|
||||
@ -592,10 +620,23 @@ static void host_signal_handler(int host_signum, siginfo_t *info,
|
||||
rewind_if_in_safe_syscall(puc);
|
||||
|
||||
host_to_target_siginfo_noswap(&tinfo, info);
|
||||
if (queue_signal(env, sig, &tinfo) == 1) {
|
||||
/* interrupt the virtual CPU as soon as possible */
|
||||
cpu_exit(thread_cpu);
|
||||
}
|
||||
k = &ts->sigtab[sig - 1];
|
||||
k->info = tinfo;
|
||||
k->pending = sig;
|
||||
ts->signal_pending = 1;
|
||||
|
||||
/* Block host signals until target signal handler entered. We
|
||||
* can't block SIGSEGV or SIGBUS while we're executing guest
|
||||
* code in case the guest code provokes one in the window between
|
||||
* now and it getting out to the main loop. Signals will be
|
||||
* unblocked again in process_pending_signals().
|
||||
*/
|
||||
sigfillset(&uc->uc_sigmask);
|
||||
sigdelset(&uc->uc_sigmask, SIGSEGV);
|
||||
sigdelset(&uc->uc_sigmask, SIGBUS);
|
||||
|
||||
/* interrupt the virtual CPU as soon as possible */
|
||||
cpu_exit(thread_cpu);
|
||||
}
|
||||
|
||||
/* do_sigaltstack() returns target values and errnos. */
|
||||
@ -671,7 +712,7 @@ out:
|
||||
return ret;
|
||||
}
|
||||
|
||||
/* do_sigaction() return host values and errnos */
|
||||
/* do_sigaction() return target values and host errnos */
|
||||
int do_sigaction(int sig, const struct target_sigaction *act,
|
||||
struct target_sigaction *oact)
|
||||
{
|
||||
@ -680,8 +721,14 @@ int do_sigaction(int sig, const struct target_sigaction *act,
|
||||
int host_sig;
|
||||
int ret = 0;
|
||||
|
||||
if (sig < 1 || sig > TARGET_NSIG || sig == TARGET_SIGKILL || sig == TARGET_SIGSTOP)
|
||||
return -EINVAL;
|
||||
if (sig < 1 || sig > TARGET_NSIG || sig == TARGET_SIGKILL || sig == TARGET_SIGSTOP) {
|
||||
return -TARGET_EINVAL;
|
||||
}
|
||||
|
||||
if (block_signals()) {
|
||||
return -TARGET_ERESTARTSYS;
|
||||
}
|
||||
|
||||
k = &sigact_table[sig - 1];
|
||||
if (oact) {
|
||||
__put_user(k->_sa_handler, &oact->_sa_handler);
|
||||
@ -1093,7 +1140,7 @@ long do_sigreturn(CPUX86State *env)
|
||||
}
|
||||
|
||||
target_to_host_sigset_internal(&set, &target_set);
|
||||
do_sigprocmask(SIG_SETMASK, &set, NULL);
|
||||
set_sigmask(&set);
|
||||
|
||||
/* restore registers */
|
||||
if (restore_sigcontext(env, &frame->sc))
|
||||
@ -1118,7 +1165,7 @@ long do_rt_sigreturn(CPUX86State *env)
|
||||
if (!lock_user_struct(VERIFY_READ, frame, frame_addr, 1))
|
||||
goto badframe;
|
||||
target_to_host_sigset(&set, &frame->uc.tuc_sigmask);
|
||||
do_sigprocmask(SIG_SETMASK, &set, NULL);
|
||||
set_sigmask(&set);
|
||||
|
||||
if (restore_sigcontext(env, &frame->uc.tuc_mcontext)) {
|
||||
goto badframe;
|
||||
@ -1258,7 +1305,7 @@ static int target_restore_sigframe(CPUARMState *env,
|
||||
uint64_t pstate;
|
||||
|
||||
target_to_host_sigset(&set, &sf->uc.tuc_sigmask);
|
||||
do_sigprocmask(SIG_SETMASK, &set, NULL);
|
||||
set_sigmask(&set);
|
||||
|
||||
for (i = 0; i < 31; i++) {
|
||||
__get_user(env->xregs[i], &sf->uc.tuc_mcontext.regs[i]);
|
||||
@ -1900,7 +1947,7 @@ static long do_sigreturn_v1(CPUARMState *env)
|
||||
}
|
||||
|
||||
target_to_host_sigset_internal(&host_set, &set);
|
||||
do_sigprocmask(SIG_SETMASK, &host_set, NULL);
|
||||
set_sigmask(&host_set);
|
||||
|
||||
if (restore_sigcontext(env, &frame->sc)) {
|
||||
goto badframe;
|
||||
@ -1981,7 +2028,7 @@ static int do_sigframe_return_v2(CPUARMState *env, target_ulong frame_addr,
|
||||
abi_ulong *regspace;
|
||||
|
||||
target_to_host_sigset(&host_set, &uc->tuc_sigmask);
|
||||
do_sigprocmask(SIG_SETMASK, &host_set, NULL);
|
||||
set_sigmask(&host_set);
|
||||
|
||||
if (restore_sigcontext(env, &uc->tuc_mcontext))
|
||||
return 1;
|
||||
@ -2077,7 +2124,7 @@ static long do_rt_sigreturn_v1(CPUARMState *env)
|
||||
}
|
||||
|
||||
target_to_host_sigset(&host_set, &frame->uc.tuc_sigmask);
|
||||
do_sigprocmask(SIG_SETMASK, &host_set, NULL);
|
||||
set_sigmask(&host_set);
|
||||
|
||||
if (restore_sigcontext(env, &frame->uc.tuc_mcontext)) {
|
||||
goto badframe;
|
||||
@ -2453,7 +2500,7 @@ long do_sigreturn(CPUSPARCState *env)
|
||||
}
|
||||
|
||||
target_to_host_sigset_internal(&host_set, &set);
|
||||
do_sigprocmask(SIG_SETMASK, &host_set, NULL);
|
||||
set_sigmask(&host_set);
|
||||
|
||||
if (err) {
|
||||
goto segv_and_exit;
|
||||
@ -2576,7 +2623,7 @@ void sparc64_set_context(CPUSPARCState *env)
|
||||
}
|
||||
}
|
||||
target_to_host_sigset_internal(&set, &target_set);
|
||||
do_sigprocmask(SIG_SETMASK, &set, NULL);
|
||||
set_sigmask(&set);
|
||||
}
|
||||
env->pc = pc;
|
||||
env->npc = npc;
|
||||
@ -2664,9 +2711,13 @@ void sparc64_get_context(CPUSPARCState *env)
|
||||
env->pc = env->npc;
|
||||
env->npc += 4;
|
||||
|
||||
err = 0;
|
||||
|
||||
do_sigprocmask(0, NULL, &set);
|
||||
/* If we're only reading the signal mask then do_sigprocmask()
|
||||
* is guaranteed not to fail, which is important because we don't
|
||||
* have any way to signal a failure or restart this operation since
|
||||
* this is not a normal syscall.
|
||||
*/
|
||||
err = do_sigprocmask(0, NULL, &set);
|
||||
assert(err == 0);
|
||||
host_to_target_sigset_internal(&target_set, &set);
|
||||
if (TARGET_NSIG_WORDS == 1) {
|
||||
__put_user(target_set.sig[0],
|
||||
@ -2993,7 +3044,7 @@ long do_sigreturn(CPUMIPSState *regs)
|
||||
}
|
||||
|
||||
target_to_host_sigset_internal(&blocked, &target_set);
|
||||
do_sigprocmask(SIG_SETMASK, &blocked, NULL);
|
||||
set_sigmask(&blocked);
|
||||
|
||||
restore_sigcontext(regs, &frame->sf_sc);
|
||||
|
||||
@ -3097,7 +3148,7 @@ long do_rt_sigreturn(CPUMIPSState *env)
|
||||
}
|
||||
|
||||
target_to_host_sigset(&blocked, &frame->rs_uc.tuc_sigmask);
|
||||
do_sigprocmask(SIG_SETMASK, &blocked, NULL);
|
||||
set_sigmask(&blocked);
|
||||
|
||||
restore_sigcontext(env, &frame->rs_uc.tuc_mcontext);
|
||||
|
||||
@ -3371,7 +3422,7 @@ long do_sigreturn(CPUSH4State *regs)
|
||||
goto badframe;
|
||||
|
||||
target_to_host_sigset_internal(&blocked, &target_set);
|
||||
do_sigprocmask(SIG_SETMASK, &blocked, NULL);
|
||||
set_sigmask(&blocked);
|
||||
|
||||
restore_sigcontext(regs, &frame->sc);
|
||||
|
||||
@ -3397,7 +3448,7 @@ long do_rt_sigreturn(CPUSH4State *regs)
|
||||
}
|
||||
|
||||
target_to_host_sigset(&blocked, &frame->uc.tuc_sigmask);
|
||||
do_sigprocmask(SIG_SETMASK, &blocked, NULL);
|
||||
set_sigmask(&blocked);
|
||||
|
||||
restore_sigcontext(regs, &frame->uc.tuc_mcontext);
|
||||
|
||||
@ -3621,7 +3672,7 @@ long do_sigreturn(CPUMBState *env)
|
||||
__get_user(target_set.sig[i], &frame->extramask[i - 1]);
|
||||
}
|
||||
target_to_host_sigset_internal(&set, &target_set);
|
||||
do_sigprocmask(SIG_SETMASK, &set, NULL);
|
||||
set_sigmask(&set);
|
||||
|
||||
restore_sigcontext(&frame->uc.tuc_mcontext, env);
|
||||
/* We got here through a sigreturn syscall, our path back is via an
|
||||
@ -3792,7 +3843,7 @@ long do_sigreturn(CPUCRISState *env)
|
||||
__get_user(target_set.sig[i], &frame->extramask[i - 1]);
|
||||
}
|
||||
target_to_host_sigset_internal(&set, &target_set);
|
||||
do_sigprocmask(SIG_SETMASK, &set, NULL);
|
||||
set_sigmask(&set);
|
||||
|
||||
restore_sigcontext(&frame->sc, env);
|
||||
unlock_user_struct(frame, frame_addr, 0);
|
||||
@ -4284,7 +4335,7 @@ long do_sigreturn(CPUS390XState *env)
|
||||
__get_user(target_set.sig[0], &frame->sc.oldmask[0]);
|
||||
|
||||
target_to_host_sigset_internal(&set, &target_set);
|
||||
do_sigprocmask(SIG_SETMASK, &set, NULL); /* ~_BLOCKABLE? */
|
||||
set_sigmask(&set); /* ~_BLOCKABLE? */
|
||||
|
||||
if (restore_sigregs(env, &frame->sregs)) {
|
||||
goto badframe;
|
||||
@ -4310,7 +4361,7 @@ long do_rt_sigreturn(CPUS390XState *env)
|
||||
}
|
||||
target_to_host_sigset(&set, &frame->uc.tuc_sigmask);
|
||||
|
||||
do_sigprocmask(SIG_SETMASK, &set, NULL); /* ~_BLOCKABLE? */
|
||||
set_sigmask(&set); /* ~_BLOCKABLE? */
|
||||
|
||||
if (restore_sigregs(env, &frame->uc.tuc_mcontext)) {
|
||||
goto badframe;
|
||||
@ -4872,7 +4923,7 @@ long do_sigreturn(CPUPPCState *env)
|
||||
__get_user(set.sig[1], &sc->_unused[3]);
|
||||
#endif
|
||||
target_to_host_sigset_internal(&blocked, &set);
|
||||
do_sigprocmask(SIG_SETMASK, &blocked, NULL);
|
||||
set_sigmask(&blocked);
|
||||
|
||||
__get_user(sr_addr, &sc->regs);
|
||||
if (!lock_user_struct(VERIFY_READ, sr, sr_addr, 1))
|
||||
@ -4913,7 +4964,7 @@ static int do_setcontext(struct target_ucontext *ucp, CPUPPCState *env, int sig)
|
||||
return 1;
|
||||
|
||||
target_to_host_sigset_internal(&blocked, &set);
|
||||
do_sigprocmask(SIG_SETMASK, &blocked, NULL);
|
||||
set_sigmask(&blocked);
|
||||
restore_user_regs(env, mcp, sig);
|
||||
|
||||
unlock_user_struct(mcp, mcp_addr, 1);
|
||||
@ -5261,7 +5312,7 @@ long do_sigreturn(CPUM68KState *env)
|
||||
}
|
||||
|
||||
target_to_host_sigset_internal(&set, &target_set);
|
||||
do_sigprocmask(SIG_SETMASK, &set, NULL);
|
||||
set_sigmask(&set);
|
||||
|
||||
/* restore registers */
|
||||
|
||||
@ -5287,7 +5338,7 @@ long do_rt_sigreturn(CPUM68KState *env)
|
||||
goto badframe;
|
||||
|
||||
target_to_host_sigset_internal(&set, &target_set);
|
||||
do_sigprocmask(SIG_SETMASK, &set, NULL);
|
||||
set_sigmask(&set);
|
||||
|
||||
/* restore registers */
|
||||
|
||||
@ -5530,7 +5581,7 @@ long do_sigreturn(CPUAlphaState *env)
|
||||
__get_user(target_set.sig[0], &sc->sc_mask);
|
||||
|
||||
target_to_host_sigset_internal(&set, &target_set);
|
||||
do_sigprocmask(SIG_SETMASK, &set, NULL);
|
||||
set_sigmask(&set);
|
||||
|
||||
restore_sigcontext(env, sc);
|
||||
unlock_user_struct(sc, sc_addr, 0);
|
||||
@ -5551,7 +5602,7 @@ long do_rt_sigreturn(CPUAlphaState *env)
|
||||
goto badframe;
|
||||
}
|
||||
target_to_host_sigset(&set, &frame->uc.tuc_sigmask);
|
||||
do_sigprocmask(SIG_SETMASK, &set, NULL);
|
||||
set_sigmask(&set);
|
||||
|
||||
restore_sigcontext(env, &frame->uc.tuc_mcontext);
|
||||
if (do_sigaltstack(frame_addr + offsetof(struct target_rt_sigframe,
|
||||
@ -5718,7 +5769,7 @@ long do_rt_sigreturn(CPUTLGState *env)
|
||||
goto badframe;
|
||||
}
|
||||
target_to_host_sigset(&set, &frame->uc.tuc_sigmask);
|
||||
do_sigprocmask(SIG_SETMASK, &set, NULL);
|
||||
set_sigmask(&set);
|
||||
|
||||
restore_sigcontext(env, &frame->uc.tuc_mcontext);
|
||||
if (do_sigaltstack(frame_addr + offsetof(struct target_rt_sigframe,
|
||||
@ -5765,39 +5816,19 @@ long do_rt_sigreturn(CPUArchState *env)
|
||||
|
||||
#endif
|
||||
|
||||
void process_pending_signals(CPUArchState *cpu_env)
|
||||
static void handle_pending_signal(CPUArchState *cpu_env, int sig)
|
||||
{
|
||||
CPUState *cpu = ENV_GET_CPU(cpu_env);
|
||||
int sig;
|
||||
abi_ulong handler;
|
||||
sigset_t set, old_set;
|
||||
sigset_t set;
|
||||
target_sigset_t target_old_set;
|
||||
struct emulated_sigtable *k;
|
||||
struct target_sigaction *sa;
|
||||
struct sigqueue *q;
|
||||
TaskState *ts = cpu->opaque;
|
||||
struct emulated_sigtable *k = &ts->sigtab[sig - 1];
|
||||
|
||||
if (!ts->signal_pending)
|
||||
return;
|
||||
|
||||
/* FIXME: This is not threadsafe. */
|
||||
k = ts->sigtab;
|
||||
for(sig = 1; sig <= TARGET_NSIG; sig++) {
|
||||
if (k->pending)
|
||||
goto handle_signal;
|
||||
k++;
|
||||
}
|
||||
/* if no signal is pending, just return */
|
||||
ts->signal_pending = 0;
|
||||
return;
|
||||
|
||||
handle_signal:
|
||||
trace_user_handle_signal(cpu_env, sig);
|
||||
/* dequeue signal */
|
||||
q = k->first;
|
||||
k->first = q->next;
|
||||
if (!k->first)
|
||||
k->pending = 0;
|
||||
k->pending = 0;
|
||||
|
||||
sig = gdb_handlesig(cpu, sig);
|
||||
if (!sig) {
|
||||
@ -5808,14 +5839,6 @@ void process_pending_signals(CPUArchState *cpu_env)
|
||||
handler = sa->_sa_handler;
|
||||
}
|
||||
|
||||
if (ts->sigsegv_blocked && sig == TARGET_SIGSEGV) {
|
||||
/* Guest has blocked SIGSEGV but we got one anyway. Assume this
|
||||
* is a forced SIGSEGV (ie one the kernel handles via force_sig_info
|
||||
* because it got a real MMU fault), and treat as if default handler.
|
||||
*/
|
||||
handler = TARGET_SIG_DFL;
|
||||
}
|
||||
|
||||
if (handler == TARGET_SIG_DFL) {
|
||||
/* default handler : ignore some signal. The other are job control or fatal */
|
||||
if (sig == TARGET_SIGTSTP || sig == TARGET_SIGTTIN || sig == TARGET_SIGTTOU) {
|
||||
@ -5832,17 +5855,23 @@ void process_pending_signals(CPUArchState *cpu_env)
|
||||
force_sig(sig);
|
||||
} else {
|
||||
/* compute the blocked signals during the handler execution */
|
||||
sigset_t *blocked_set;
|
||||
|
||||
target_to_host_sigset(&set, &sa->sa_mask);
|
||||
/* SA_NODEFER indicates that the current signal should not be
|
||||
blocked during the handler */
|
||||
if (!(sa->sa_flags & TARGET_SA_NODEFER))
|
||||
sigaddset(&set, target_to_host_signal(sig));
|
||||
|
||||
/* block signals in the handler using Linux */
|
||||
do_sigprocmask(SIG_BLOCK, &set, &old_set);
|
||||
/* save the previous blocked signal state to restore it at the
|
||||
end of the signal execution (see do_sigreturn) */
|
||||
host_to_target_sigset_internal(&target_old_set, &old_set);
|
||||
host_to_target_sigset_internal(&target_old_set, &ts->signal_mask);
|
||||
|
||||
/* block signals in the handler */
|
||||
blocked_set = ts->in_sigsuspend ?
|
||||
&ts->sigsuspend_mask : &ts->signal_mask;
|
||||
sigorset(&ts->signal_mask, blocked_set, &set);
|
||||
ts->in_sigsuspend = 0;
|
||||
|
||||
/* if the CPU is in VM86 mode, we restore the 32 bit values */
|
||||
#if defined(TARGET_I386) && !defined(TARGET_X86_64)
|
||||
@ -5856,16 +5885,74 @@ void process_pending_signals(CPUArchState *cpu_env)
|
||||
#if defined(TARGET_ABI_MIPSN32) || defined(TARGET_ABI_MIPSN64) \
|
||||
|| defined(TARGET_OPENRISC) || defined(TARGET_TILEGX)
|
||||
/* These targets do not have traditional signals. */
|
||||
setup_rt_frame(sig, sa, &q->info, &target_old_set, cpu_env);
|
||||
setup_rt_frame(sig, sa, &k->info, &target_old_set, cpu_env);
|
||||
#else
|
||||
if (sa->sa_flags & TARGET_SA_SIGINFO)
|
||||
setup_rt_frame(sig, sa, &q->info, &target_old_set, cpu_env);
|
||||
setup_rt_frame(sig, sa, &k->info, &target_old_set, cpu_env);
|
||||
else
|
||||
setup_frame(sig, sa, &target_old_set, cpu_env);
|
||||
#endif
|
||||
if (sa->sa_flags & TARGET_SA_RESETHAND)
|
||||
if (sa->sa_flags & TARGET_SA_RESETHAND) {
|
||||
sa->_sa_handler = TARGET_SIG_DFL;
|
||||
}
|
||||
}
|
||||
if (q != &k->info)
|
||||
free_sigqueue(cpu_env, q);
|
||||
}
|
||||
|
||||
void process_pending_signals(CPUArchState *cpu_env)
|
||||
{
|
||||
CPUState *cpu = ENV_GET_CPU(cpu_env);
|
||||
int sig;
|
||||
TaskState *ts = cpu->opaque;
|
||||
sigset_t set;
|
||||
sigset_t *blocked_set;
|
||||
|
||||
while (atomic_read(&ts->signal_pending)) {
|
||||
/* FIXME: This is not threadsafe. */
|
||||
sigfillset(&set);
|
||||
sigprocmask(SIG_SETMASK, &set, 0);
|
||||
|
||||
sig = ts->sync_signal.pending;
|
||||
if (sig) {
|
||||
/* Synchronous signals are forced,
|
||||
* see force_sig_info() and callers in Linux
|
||||
* Note that not all of our queue_signal() calls in QEMU correspond
|
||||
* to force_sig_info() calls in Linux (some are send_sig_info()).
|
||||
* However it seems like a kernel bug to me to allow the process
|
||||
* to block a synchronous signal since it could then just end up
|
||||
* looping round and round indefinitely.
|
||||
*/
|
||||
if (sigismember(&ts->signal_mask, target_to_host_signal_table[sig])
|
||||
|| sigact_table[sig - 1]._sa_handler == TARGET_SIG_IGN) {
|
||||
sigdelset(&ts->signal_mask, target_to_host_signal_table[sig]);
|
||||
sigact_table[sig - 1]._sa_handler = TARGET_SIG_DFL;
|
||||
}
|
||||
|
||||
handle_pending_signal(cpu_env, sig);
|
||||
}
|
||||
|
||||
for (sig = 1; sig <= TARGET_NSIG; sig++) {
|
||||
blocked_set = ts->in_sigsuspend ?
|
||||
&ts->sigsuspend_mask : &ts->signal_mask;
|
||||
|
||||
if (ts->sigtab[sig - 1].pending &&
|
||||
(!sigismember(blocked_set,
|
||||
target_to_host_signal_table[sig]))) {
|
||||
handle_pending_signal(cpu_env, sig);
|
||||
/* Restart scan from the beginning */
|
||||
sig = 1;
|
||||
}
|
||||
}
|
||||
|
||||
/* if no signal is pending, unblock signals and recheck (the act
|
||||
* of unblocking might cause us to take another host signal which
|
||||
* will set signal_pending again).
|
||||
*/
|
||||
atomic_set(&ts->signal_pending, 0);
|
||||
ts->in_sigsuspend = 0;
|
||||
set = ts->signal_mask;
|
||||
sigdelset(&set, SIGSEGV);
|
||||
sigdelset(&set, SIGBUS);
|
||||
sigprocmask(SIG_SETMASK, &set, 0);
|
||||
}
|
||||
ts->in_sigsuspend = 0;
|
||||
}
|
||||
|
@ -281,7 +281,7 @@ print_ipc(const struct syscallname *name,
|
||||
static void
|
||||
print_syscall_ret_addr(const struct syscallname *name, abi_long ret)
|
||||
{
|
||||
char *errstr = NULL;
|
||||
const char *errstr = NULL;
|
||||
|
||||
if (ret < 0) {
|
||||
errstr = target_strerror(-ret);
|
||||
@ -1594,7 +1594,7 @@ void
|
||||
print_syscall_ret(int num, abi_long ret)
|
||||
{
|
||||
int i;
|
||||
char *errstr = NULL;
|
||||
const char *errstr = NULL;
|
||||
|
||||
for(i=0;i<nsyscalls;i++)
|
||||
if( scnames[i].nr == num ) {
|
||||
|
1076
linux-user/syscall.c
1076
linux-user/syscall.c
File diff suppressed because it is too large
Load Diff
@ -673,6 +673,21 @@ typedef struct {
|
||||
|
||||
#define TARGET_SI_PAD_SIZE ((TARGET_SI_MAX_SIZE - TARGET_SI_PREAMBLE_SIZE) / sizeof(int))
|
||||
|
||||
/* Within QEMU the top 16 bits of si_code indicate which of the parts of
|
||||
* the union in target_siginfo is valid. This only applies between
|
||||
* host_to_target_siginfo_noswap() and tswap_siginfo(); it does not
|
||||
* appear either within host siginfo_t or in target_siginfo structures
|
||||
* which we get from the guest userspace program. (The Linux kernel
|
||||
* does a similar thing with using the top bits for its own internal
|
||||
* purposes but not letting them be visible to userspace.)
|
||||
*/
|
||||
#define QEMU_SI_KILL 0
|
||||
#define QEMU_SI_TIMER 1
|
||||
#define QEMU_SI_POLL 2
|
||||
#define QEMU_SI_FAULT 3
|
||||
#define QEMU_SI_CHLD 4
|
||||
#define QEMU_SI_RT 5
|
||||
|
||||
typedef struct target_siginfo {
|
||||
#ifdef TARGET_MIPS
|
||||
int si_signo;
|
||||
@ -2274,34 +2289,34 @@ struct target_statfs64 {
|
||||
#endif
|
||||
|
||||
struct target_flock {
|
||||
short l_type;
|
||||
short l_whence;
|
||||
abi_ulong l_start;
|
||||
abi_ulong l_len;
|
||||
int l_pid;
|
||||
short l_type;
|
||||
short l_whence;
|
||||
abi_long l_start;
|
||||
abi_long l_len;
|
||||
int l_pid;
|
||||
};
|
||||
|
||||
struct target_flock64 {
|
||||
short l_type;
|
||||
short l_whence;
|
||||
short l_type;
|
||||
short l_whence;
|
||||
#if defined(TARGET_PPC) || defined(TARGET_X86_64) || defined(TARGET_MIPS) \
|
||||
|| defined(TARGET_SPARC) || defined(TARGET_HPPA) \
|
||||
|| defined(TARGET_MICROBLAZE) || defined(TARGET_TILEGX)
|
||||
int __pad;
|
||||
int __pad;
|
||||
#endif
|
||||
unsigned long long l_start;
|
||||
unsigned long long l_len;
|
||||
int l_pid;
|
||||
abi_llong l_start;
|
||||
abi_llong l_len;
|
||||
int l_pid;
|
||||
} QEMU_PACKED;
|
||||
|
||||
#ifdef TARGET_ARM
|
||||
struct target_eabi_flock64 {
|
||||
short l_type;
|
||||
short l_whence;
|
||||
int __pad;
|
||||
unsigned long long l_start;
|
||||
unsigned long long l_len;
|
||||
int l_pid;
|
||||
short l_type;
|
||||
short l_whence;
|
||||
int __pad;
|
||||
abi_llong l_start;
|
||||
abi_llong l_len;
|
||||
int l_pid;
|
||||
} QEMU_PACKED;
|
||||
#endif
|
||||
|
||||
|
385
scripts/qemu-binfmt-conf.sh
Normal file → Executable file
385
scripts/qemu-binfmt-conf.sh
Normal file → Executable file
@ -1,72 +1,323 @@
|
||||
#!/bin/sh
|
||||
# enable automatic i386/ARM/M68K/MIPS/SPARC/PPC/s390 program execution by the kernel
|
||||
|
||||
# load the binfmt_misc module
|
||||
if [ ! -d /proc/sys/fs/binfmt_misc ]; then
|
||||
/sbin/modprobe binfmt_misc
|
||||
fi
|
||||
if [ ! -f /proc/sys/fs/binfmt_misc/register ]; then
|
||||
mount binfmt_misc -t binfmt_misc /proc/sys/fs/binfmt_misc
|
||||
fi
|
||||
qemu_target_list="i386 i486 alpha arm sparc32plus ppc ppc64 ppc64le m68k \
|
||||
mips mipsel mipsn32 mipsn32el mips64 mips64el \
|
||||
sh4 sh4eb s390x aarch64"
|
||||
|
||||
# probe cpu type
|
||||
cpu=$(uname -m)
|
||||
case "$cpu" in
|
||||
i386|i486|i586|i686|i86pc|BePC|x86_64)
|
||||
cpu="i386"
|
||||
;;
|
||||
m68k)
|
||||
cpu="m68k"
|
||||
;;
|
||||
mips*)
|
||||
cpu="mips"
|
||||
;;
|
||||
"Power Macintosh"|ppc|ppc64)
|
||||
cpu="ppc"
|
||||
;;
|
||||
armv[4-9]*)
|
||||
cpu="arm"
|
||||
;;
|
||||
esac
|
||||
i386_magic='\x7fELF\x01\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x03\x00'
|
||||
i386_mask='\xff\xff\xff\xff\xff\xfe\xfe\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff'
|
||||
i386_family=i386
|
||||
|
||||
# register the interpreter for each cpu except for the native one
|
||||
if [ $cpu != "i386" ] ; then
|
||||
echo ':i386:M::\x7fELF\x01\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x03\x00:\xff\xff\xff\xff\xff\xfe\xfe\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff:/usr/local/bin/qemu-i386:' > /proc/sys/fs/binfmt_misc/register
|
||||
echo ':i486:M::\x7fELF\x01\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x06\x00:\xff\xff\xff\xff\xff\xfe\xfe\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff:/usr/local/bin/qemu-i386:' > /proc/sys/fs/binfmt_misc/register
|
||||
fi
|
||||
if [ $cpu != "alpha" ] ; then
|
||||
echo ':alpha:M::\x7fELF\x02\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x26\x90:\xff\xff\xff\xff\xff\xfe\xfe\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff:/usr/local/bin/qemu-alpha:' > /proc/sys/fs/binfmt_misc/register
|
||||
fi
|
||||
if [ $cpu != "arm" ] ; then
|
||||
echo ':arm:M::\x7fELF\x01\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x28\x00:\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff:/usr/local/bin/qemu-arm:' > /proc/sys/fs/binfmt_misc/register
|
||||
echo ':armeb:M::\x7fELF\x01\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x28:\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff:/usr/local/bin/qemu-armeb:' > /proc/sys/fs/binfmt_misc/register
|
||||
fi
|
||||
if [ $cpu != "aarch64" ] ; then
|
||||
echo ':aarch64:M::\x7fELF\x02\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\xb7\x00:\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff:/usr/local/bin/qemu-aarch64:' > /proc/sys/fs/binfmt_misc/register
|
||||
fi
|
||||
if [ $cpu != "sparc" ] ; then
|
||||
echo ':sparc:M::\x7fELF\x01\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x02:\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff:/usr/local/bin/qemu-sparc:' > /proc/sys/fs/binfmt_misc/register
|
||||
fi
|
||||
if [ $cpu != "ppc" ] ; then
|
||||
echo ':ppc:M::\x7fELF\x01\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x14:\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff:/usr/local/bin/qemu-ppc:' > /proc/sys/fs/binfmt_misc/register
|
||||
fi
|
||||
if [ $cpu != "m68k" ] ; then
|
||||
echo 'Please check cpu value and header information for m68k!'
|
||||
echo ':m68k:M::\x7fELF\x01\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x04:\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff:/usr/local/bin/qemu-m68k:' > /proc/sys/fs/binfmt_misc/register
|
||||
fi
|
||||
if [ $cpu != "mips" ] ; then
|
||||
# FIXME: We could use the other endianness on a MIPS host.
|
||||
echo ':mips:M::\x7fELF\x01\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x08:\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff:/usr/local/bin/qemu-mips:' > /proc/sys/fs/binfmt_misc/register
|
||||
echo ':mipsel:M::\x7fELF\x01\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x08\x00:\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff:/usr/local/bin/qemu-mipsel:' > /proc/sys/fs/binfmt_misc/register
|
||||
echo ':mipsn32:M::\x7fELF\x01\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x08:\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff:/usr/local/bin/qemu-mipsn32:' > /proc/sys/fs/binfmt_misc/register
|
||||
echo ':mipsn32el:M::\x7fELF\x01\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x08\x00:\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff:/usr/local/bin/qemu-mipsn32el:' > /proc/sys/fs/binfmt_misc/register
|
||||
echo ':mips64:M::\x7fELF\x02\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x08:\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff:/usr/local/bin/qemu-mips64:' > /proc/sys/fs/binfmt_misc/register
|
||||
echo ':mips64el:M::\x7fELF\x02\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x08\x00:\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff:/usr/local/bin/qemu-mips64el:' > /proc/sys/fs/binfmt_misc/register
|
||||
fi
|
||||
if [ $cpu != "sh" ] ; then
|
||||
echo ':sh4:M::\x7fELF\x01\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x2a\x00:\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff:/usr/local/bin/qemu-sh4:' > /proc/sys/fs/binfmt_misc/register
|
||||
echo ':sh4eb:M::\x7fELF\x01\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x2a:\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff:/usr/local/bin/qemu-sh4eb:' > /proc/sys/fs/binfmt_misc/register
|
||||
fi
|
||||
if [ $cpu != "s390x" ] ; then
|
||||
echo ':s390x:M::\x7fELF\x02\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x16:\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff:/usr/local/bin/qemu-s390x:' > /proc/sys/fs/binfmt_misc/register
|
||||
fi
|
||||
i486_magic='\x7fELF\x01\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x06\x00'
|
||||
i486_mask='\xff\xff\xff\xff\xff\xfe\xfe\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff'
|
||||
i486_family=i386
|
||||
|
||||
alpha_magic='\x7fELF\x02\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x26\x90'
|
||||
alpha_mask='\xff\xff\xff\xff\xff\xfe\xfe\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff'
|
||||
alpha_family=alpha
|
||||
|
||||
arm_magic='\x7fELF\x01\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x28\x00'
|
||||
arm_mask='\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff'
|
||||
arm_family=arm
|
||||
|
||||
armeb_magic='\x7fELF\x01\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x28'
|
||||
armeb_mask='\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff'
|
||||
armeb_family=arm
|
||||
|
||||
sparc_magic='\x7fELF\x01\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x02'
|
||||
sparc_mask='\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff'
|
||||
sparc_family=sparc
|
||||
|
||||
sparc32plus_magic='\x7fELF\x01\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x12'
|
||||
sparc32plus_mask='\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff'
|
||||
sparc32plus_family=sparc
|
||||
|
||||
ppc_magic='\x7fELF\x01\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x14'
|
||||
ppc_mask='\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff'
|
||||
ppc_family=ppc
|
||||
|
||||
ppc64_magic='\x7fELF\x02\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x15'
|
||||
ppc64_mask='\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff'
|
||||
ppc64_family=ppc
|
||||
|
||||
ppc64le_magic='\x7fELF\x02\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x15\x00'
|
||||
ppc64le_mask='\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\x00'
|
||||
ppc64le_family=ppcle
|
||||
|
||||
m68k_magic='\x7fELF\x01\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x04'
|
||||
m68k_mask='\xff\xff\xff\xff\xff\xff\xfe\xfe\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff'
|
||||
m68k_family=m68k
|
||||
|
||||
# FIXME: We could use the other endianness on a MIPS host.
|
||||
|
||||
mips_magic='\x7fELF\x01\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x08'
|
||||
mips_mask='\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff'
|
||||
mips_family=mips
|
||||
|
||||
mipsel_magic='\x7fELF\x01\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x08\x00'
|
||||
mipsel_mask='\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff'
|
||||
mipsel_family=mips
|
||||
|
||||
mipsn32_magic='\x7fELF\x01\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x08'
|
||||
mipsn32_mask='\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff'
|
||||
mipsn32_family=mips
|
||||
|
||||
mipsn32el_magic='\x7fELF\x01\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x08\x00'
|
||||
mipsn32el_mask='\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff'
|
||||
mipsn32el_family=mips
|
||||
|
||||
mips64_magic='\x7fELF\x02\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x08'
|
||||
mips64_mask='\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff'
|
||||
mips64_family=mips
|
||||
|
||||
mips64el_magic='\x7fELF\x02\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x08\x00'
|
||||
mips64el_mask='\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff'
|
||||
mips64el_family=mips
|
||||
|
||||
sh4_magic='\x7fELF\x01\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x2a\x00'
|
||||
sh4_mask='\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff'
|
||||
sh4_family=sh4
|
||||
|
||||
sh4eb_magic='\x7fELF\x01\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x2a'
|
||||
sh4eb_mask='\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff'
|
||||
sh4eb_family=sh4
|
||||
|
||||
s390x_magic='\x7fELF\x02\x02\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\x16'
|
||||
s390x_mask='\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff'
|
||||
s390x_family=s390x
|
||||
|
||||
aarch64_magic='\x7fELF\x02\x01\x01\x00\x00\x00\x00\x00\x00\x00\x00\x00\x02\x00\xb7\x00'
|
||||
aarch64_mask='\xff\xff\xff\xff\xff\xff\xff\x00\xff\xff\xff\xff\xff\xff\xff\xff\xfe\xff\xff\xff'
|
||||
aarch64_family=arm
|
||||
|
||||
qemu_get_family() {
|
||||
cpu=${HOST_ARCH:-$(uname -m)}
|
||||
case "$cpu" in
|
||||
amd64|i386|i486|i586|i686|i86pc|BePC|x86_64)
|
||||
echo "i386"
|
||||
;;
|
||||
mips*)
|
||||
echo "mips"
|
||||
;;
|
||||
"Power Macintosh"|ppc64|powerpc|ppc)
|
||||
echo "ppc"
|
||||
;;
|
||||
ppc64el|ppc64le)
|
||||
echo "ppcle"
|
||||
;;
|
||||
arm|armel|armhf|arm64|armv[4-9]*)
|
||||
echo "arm"
|
||||
;;
|
||||
sparc*)
|
||||
echo "sparc"
|
||||
;;
|
||||
*)
|
||||
echo "$cpu"
|
||||
;;
|
||||
esac
|
||||
}
|
||||
|
||||
usage() {
|
||||
cat <<EOF
|
||||
Usage: qemu-binfmt-conf.sh [--qemu-path PATH][--debian][--systemd CPU]
|
||||
[--help][--credential yes|no][--exportdir PATH]
|
||||
|
||||
Configure binfmt_misc to use qemu interpreter
|
||||
|
||||
--help: display this usage
|
||||
--qemu-path: set path to qemu interpreter ($QEMU_PATH)
|
||||
--debian: don't write into /proc,
|
||||
instead generate update-binfmts templates
|
||||
--systemd: don't write into /proc,
|
||||
instead generate file for systemd-binfmt.service
|
||||
for the given CPU
|
||||
--exportdir: define where to write configuration files
|
||||
(default: $SYSTEMDDIR or $DEBIANDIR)
|
||||
--credential: if yes, credential and security tokens are
|
||||
calculated according to the binary to interpret
|
||||
|
||||
To import templates with update-binfmts, use :
|
||||
|
||||
sudo update-binfmts --importdir ${EXPORTDIR:-$DEBIANDIR} --import qemu-CPU
|
||||
|
||||
To remove interpreter, use :
|
||||
|
||||
sudo update-binfmts --package qemu-CPU --remove qemu-CPU $QEMU_PATH
|
||||
|
||||
With systemd, binfmt files are loaded by systemd-binfmt.service
|
||||
|
||||
The environment variable HOST_ARCH allows to override 'uname' to generate
|
||||
configuration files for a different architecture than the current one.
|
||||
|
||||
where CPU is one of:
|
||||
|
||||
$qemu_target_list
|
||||
|
||||
EOF
|
||||
}
|
||||
|
||||
qemu_check_access() {
|
||||
if [ ! -w "$1" ] ; then
|
||||
echo "ERROR: cannot write to $1" 1>&2
|
||||
exit 1
|
||||
fi
|
||||
}
|
||||
|
||||
qemu_check_bintfmt_misc() {
|
||||
# load the binfmt_misc module
|
||||
if [ ! -d /proc/sys/fs/binfmt_misc ]; then
|
||||
if ! /sbin/modprobe binfmt_misc ; then
|
||||
exit 1
|
||||
fi
|
||||
fi
|
||||
if [ ! -f /proc/sys/fs/binfmt_misc/register ]; then
|
||||
if ! mount binfmt_misc -t binfmt_misc /proc/sys/fs/binfmt_misc ; then
|
||||
exit 1
|
||||
fi
|
||||
fi
|
||||
|
||||
qemu_check_access /proc/sys/fs/binfmt_misc/register
|
||||
}
|
||||
|
||||
installed_dpkg() {
|
||||
dpkg --status "$1" > /dev/null 2>&1
|
||||
}
|
||||
|
||||
qemu_check_debian() {
|
||||
if [ ! -e /etc/debian_version ] ; then
|
||||
echo "WARNING: your system is not a Debian based distro" 1>&2
|
||||
elif ! installed_dpkg binfmt-support ; then
|
||||
echo "WARNING: package binfmt-support is needed" 1>&2
|
||||
fi
|
||||
qemu_check_access "$EXPORTDIR"
|
||||
}
|
||||
|
||||
qemu_check_systemd() {
|
||||
if ! systemctl -q is-enabled systemd-binfmt.service ; then
|
||||
echo "WARNING: systemd-binfmt.service is missing or disabled" 1>&2
|
||||
fi
|
||||
qemu_check_access "$EXPORTDIR"
|
||||
}
|
||||
|
||||
qemu_generate_register() {
|
||||
echo ":qemu-$cpu:M::$magic:$mask:$qemu:$FLAGS"
|
||||
}
|
||||
|
||||
qemu_register_interpreter() {
|
||||
echo "Setting $qemu as binfmt interpreter for $cpu"
|
||||
qemu_generate_register > /proc/sys/fs/binfmt_misc/register
|
||||
}
|
||||
|
||||
qemu_generate_systemd() {
|
||||
echo "Setting $qemu as binfmt interpreter for $cpu for systemd-binfmt.service"
|
||||
qemu_generate_register > "$EXPORTDIR/qemu-$cpu.conf"
|
||||
}
|
||||
|
||||
qemu_generate_debian() {
|
||||
cat > "$EXPORTDIR/qemu-$cpu" <<EOF
|
||||
package qemu-$cpu
|
||||
interpreter $qemu
|
||||
magic $magic
|
||||
mask $mask
|
||||
EOF
|
||||
if [ "$FLAGS" = "OC" ] ; then
|
||||
echo "credentials yes" >> "$EXPORTDIR/qemu-$cpu"
|
||||
fi
|
||||
}
|
||||
|
||||
qemu_set_binfmts() {
|
||||
# probe cpu type
|
||||
host_family=$(qemu_get_family)
|
||||
|
||||
# register the interpreter for each cpu except for the native one
|
||||
|
||||
for cpu in ${qemu_target_list} ; do
|
||||
magic=$(eval echo \$${cpu}_magic)
|
||||
mask=$(eval echo \$${cpu}_mask)
|
||||
family=$(eval echo \$${cpu}_family)
|
||||
|
||||
if [ "$magic" = "" ] || [ "$mask" = "" ] || [ "$family" = "" ] ; then
|
||||
echo "INTERNAL ERROR: unknown cpu $cpu" 1>&2
|
||||
continue
|
||||
fi
|
||||
|
||||
qemu="$QEMU_PATH/qemu-$cpu"
|
||||
if [ "$cpu" = "i486" ] ; then
|
||||
qemu="$QEMU_PATH/qemu-i386"
|
||||
fi
|
||||
|
||||
if [ "$host_family" != "$family" ] ; then
|
||||
$BINFMT_SET
|
||||
fi
|
||||
done
|
||||
}
|
||||
|
||||
CHECK=qemu_check_bintfmt_misc
|
||||
BINFMT_SET=qemu_register_interpreter
|
||||
|
||||
SYSTEMDDIR="/etc/binfmt.d"
|
||||
DEBIANDIR="/usr/share/binfmts"
|
||||
|
||||
QEMU_PATH=/usr/local/bin
|
||||
FLAGS=""
|
||||
|
||||
options=$(getopt -o ds:Q:e:hc: -l debian,systemd:,qemu-path:,exportdir:,help,credential: -- "$@")
|
||||
eval set -- "$options"
|
||||
|
||||
while true ; do
|
||||
case "$1" in
|
||||
-d|--debian)
|
||||
CHECK=qemu_check_debian
|
||||
BINFMT_SET=qemu_generate_debian
|
||||
EXPORTDIR=${EXPORTDIR:-$DEBIANDIR}
|
||||
;;
|
||||
-s|--systemd)
|
||||
CHECK=qemu_check_systemd
|
||||
BINFMT_SET=qemu_generate_systemd
|
||||
EXPORTDIR=${EXPORTDIR:-$SYSTEMDDIR}
|
||||
shift
|
||||
# check given cpu is in the supported CPU list
|
||||
for cpu in ${qemu_target_list} ; do
|
||||
if [ "$cpu" == "$1" ] ; then
|
||||
break
|
||||
fi
|
||||
done
|
||||
|
||||
if [ "$cpu" == "$1" ] ; then
|
||||
qemu_target_list="$1"
|
||||
else
|
||||
echo "ERROR: unknown CPU \"$1\"" 1>&2
|
||||
usage
|
||||
exit 1
|
||||
fi
|
||||
;;
|
||||
-Q|--qemu-path)
|
||||
shift
|
||||
QEMU_PATH="$1"
|
||||
;;
|
||||
-e|--exportdir)
|
||||
shift
|
||||
EXPORTDIR="$1"
|
||||
;;
|
||||
-h|--help)
|
||||
usage
|
||||
exit 1
|
||||
;;
|
||||
-c|--credential)
|
||||
shift
|
||||
if [ "$1" = "yes" ] ; then
|
||||
FLAGS="OC"
|
||||
else
|
||||
FLAGS=""
|
||||
fi
|
||||
;;
|
||||
*)
|
||||
break
|
||||
;;
|
||||
esac
|
||||
shift
|
||||
done
|
||||
|
||||
$CHECK
|
||||
qemu_set_binfmts
|
||||
|
Loading…
Reference in New Issue
Block a user