提交 6df9d38d 编写于 作者: P Peter Maydell 提交者: Riku Voipio

linux-user: Use safe_syscall for pselect, select syscalls

Use the safe_syscall wrapper for the pselect and select syscalls.
Since not every architecture has the select syscall, we now
have to implement select in terms of pselect, which means doing
timeval<->timespec conversion.

(Five years on from the initial patch that added pselect support
to QEMU and a decade after pselect6 went into the kernel, it seems
safe to not try to support hosts with header files which don't
define __NR_pselect6.)
Signed-off-by: NPeter Maydell <peter.maydell@linaro.org>
Signed-off-by: NRiku Voipio <riku.voipio@linaro.org>
上级 ffdcbe22
...@@ -430,15 +430,6 @@ _syscall5(int, sys_ppoll, struct pollfd *, fds, nfds_t, nfds, ...@@ -430,15 +430,6 @@ _syscall5(int, sys_ppoll, struct pollfd *, fds, nfds_t, nfds,
size_t, sigsetsize) size_t, sigsetsize)
#endif #endif
#if defined(TARGET_NR_pselect6)
#ifndef __NR_pselect6
# define __NR_pselect6 -1
#endif
#define __NR_sys_pselect6 __NR_pselect6
_syscall6(int, sys_pselect6, int, nfds, fd_set *, readfds, fd_set *, writefds,
fd_set *, exceptfds, struct timespec *, timeout, void *, sig);
#endif
#if defined(TARGET_NR_prlimit64) #if defined(TARGET_NR_prlimit64)
#ifndef __NR_prlimit64 #ifndef __NR_prlimit64
# define __NR_prlimit64 -1 # define __NR_prlimit64 -1
...@@ -704,6 +695,8 @@ safe_syscall4(pid_t, wait4, pid_t, pid, int *, status, int, options, \ ...@@ -704,6 +695,8 @@ safe_syscall4(pid_t, wait4, pid_t, pid, int *, status, int, options, \
safe_syscall5(int, waitid, idtype_t, idtype, id_t, id, siginfo_t *, infop, \ safe_syscall5(int, waitid, idtype_t, idtype, id_t, id, siginfo_t *, infop, \
int, options, struct rusage *, rusage) int, options, struct rusage *, rusage)
safe_syscall3(int, execve, const char *, filename, char **, argv, char **, envp) safe_syscall3(int, execve, const char *, filename, char **, argv, char **, envp)
safe_syscall6(int, pselect6, int, nfds, fd_set *, readfds, fd_set *, writefds, \
fd_set *, exceptfds, struct timespec *, timeout, void *, sig)
static inline int host_to_target_sock_type(int host_type) static inline int host_to_target_sock_type(int host_type)
{ {
...@@ -1115,7 +1108,8 @@ static abi_long do_select(int n, ...@@ -1115,7 +1108,8 @@ static abi_long do_select(int n,
{ {
fd_set rfds, wfds, efds; fd_set rfds, wfds, efds;
fd_set *rfds_ptr, *wfds_ptr, *efds_ptr; fd_set *rfds_ptr, *wfds_ptr, *efds_ptr;
struct timeval tv, *tv_ptr; struct timeval tv;
struct timespec ts, *ts_ptr;
abi_long ret; abi_long ret;
ret = copy_from_user_fdset_ptr(&rfds, &rfds_ptr, rfd_addr, n); ret = copy_from_user_fdset_ptr(&rfds, &rfds_ptr, rfd_addr, n);
...@@ -1134,12 +1128,15 @@ static abi_long do_select(int n, ...@@ -1134,12 +1128,15 @@ static abi_long do_select(int n,
if (target_tv_addr) { if (target_tv_addr) {
if (copy_from_user_timeval(&tv, target_tv_addr)) if (copy_from_user_timeval(&tv, target_tv_addr))
return -TARGET_EFAULT; return -TARGET_EFAULT;
tv_ptr = &tv; ts.tv_sec = tv.tv_sec;
ts.tv_nsec = tv.tv_usec * 1000;
ts_ptr = &ts;
} else { } else {
tv_ptr = NULL; ts_ptr = NULL;
} }
ret = get_errno(select(n, rfds_ptr, wfds_ptr, efds_ptr, tv_ptr)); ret = get_errno(safe_pselect6(n, rfds_ptr, wfds_ptr, efds_ptr,
ts_ptr, NULL));
if (!is_error(ret)) { if (!is_error(ret)) {
if (rfd_addr && copy_to_user_fdset(rfd_addr, &rfds, n)) if (rfd_addr && copy_to_user_fdset(rfd_addr, &rfds, n))
...@@ -1149,8 +1146,13 @@ static abi_long do_select(int n, ...@@ -1149,8 +1146,13 @@ static abi_long do_select(int n,
if (efd_addr && copy_to_user_fdset(efd_addr, &efds, n)) if (efd_addr && copy_to_user_fdset(efd_addr, &efds, n))
return -TARGET_EFAULT; return -TARGET_EFAULT;
if (target_tv_addr && copy_to_user_timeval(target_tv_addr, &tv)) if (target_tv_addr) {
return -TARGET_EFAULT; tv.tv_sec = ts.tv_sec;
tv.tv_usec = ts.tv_nsec / 1000;
if (copy_to_user_timeval(target_tv_addr, &tv)) {
return -TARGET_EFAULT;
}
}
} }
return ret; return ret;
...@@ -7206,8 +7208,8 @@ abi_long do_syscall(void *cpu_env, int num, abi_long arg1, ...@@ -7206,8 +7208,8 @@ abi_long do_syscall(void *cpu_env, int num, abi_long arg1,
sig_ptr = NULL; sig_ptr = NULL;
} }
ret = get_errno(sys_pselect6(n, rfds_ptr, wfds_ptr, efds_ptr, ret = get_errno(safe_pselect6(n, rfds_ptr, wfds_ptr, efds_ptr,
ts_ptr, sig_ptr)); ts_ptr, sig_ptr));
if (!is_error(ret)) { if (!is_error(ret)) {
if (rfd_addr && copy_to_user_fdset(rfd_addr, &rfds, n)) if (rfd_addr && copy_to_user_fdset(rfd_addr, &rfds, n))
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册