diff options
author | Linus Torvalds <torvalds@linux-foundation.org> | 2012-03-29 21:12:23 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2012-03-29 21:12:23 -0400 |
commit | a591afc01d9e48affbacb365558a31e53c85af45 (patch) | |
tree | 9bb91f4eb94ec69fc4706c4944788ec5f3586063 /net/compat.c | |
parent | 820d41cf0cd0e94a5661e093821e2e5c6b36a9d8 (diff) | |
parent | 31796ac4e8f0e88f5c10f1ad6dab8f19bebe44a4 (diff) |
Merge branch 'x86-x32-for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip
Pull x32 support for x86-64 from Ingo Molnar:
"This tree introduces the X32 binary format and execution mode for x86:
32-bit data space binaries using 64-bit instructions and 64-bit kernel
syscalls.
This allows applications whose working set fits into a 32 bits address
space to make use of 64-bit instructions while using a 32-bit address
space with shorter pointers, more compressed data structures, etc."
Fix up trivial context conflicts in arch/x86/{Kconfig,vdso/vma.c}
* 'x86-x32-for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/tip: (71 commits)
x32: Fix alignment fail in struct compat_siginfo
x32: Fix stupid ia32/x32 inversion in the siginfo format
x32: Add ptrace for x32
x32: Switch to a 64-bit clock_t
x32: Provide separate is_ia32_task() and is_x32_task() predicates
x86, mtrr: Use explicit sizing and padding for the 64-bit ioctls
x86/x32: Fix the binutils auto-detect
x32: Warn and disable rather than error if binutils too old
x32: Only clear TIF_X32 flag once
x32: Make sure TS_COMPAT is cleared for x32 tasks
fs: Remove missed ->fds_bits from cessation use of fd_set structs internally
fs: Fix close_on_exec pointer in alloc_fdtable
x32: Drop non-__vdso weak symbols from the x32 VDSO
x32: Fix coding style violations in the x32 VDSO code
x32: Add x32 VDSO support
x32: Allow x32 to be configured
x32: If configured, add x32 system calls to system call tables
x32: Handle process creation
x32: Signal-related system calls
x86: Add #ifdef CONFIG_COMPAT to <asm/sys_ia32.h>
...
Diffstat (limited to 'net/compat.c')
-rw-r--r-- | net/compat.c | 65 |
1 files changed, 40 insertions, 25 deletions
diff --git a/net/compat.c b/net/compat.c index 64b4515a64e6..e055708b8ec9 100644 --- a/net/compat.c +++ b/net/compat.c | |||
@@ -219,8 +219,6 @@ Efault: | |||
219 | 219 | ||
220 | int put_cmsg_compat(struct msghdr *kmsg, int level, int type, int len, void *data) | 220 | int put_cmsg_compat(struct msghdr *kmsg, int level, int type, int len, void *data) |
221 | { | 221 | { |
222 | struct compat_timeval ctv; | ||
223 | struct compat_timespec cts[3]; | ||
224 | struct compat_cmsghdr __user *cm = (struct compat_cmsghdr __user *) kmsg->msg_control; | 222 | struct compat_cmsghdr __user *cm = (struct compat_cmsghdr __user *) kmsg->msg_control; |
225 | struct compat_cmsghdr cmhdr; | 223 | struct compat_cmsghdr cmhdr; |
226 | int cmlen; | 224 | int cmlen; |
@@ -230,24 +228,28 @@ int put_cmsg_compat(struct msghdr *kmsg, int level, int type, int len, void *dat | |||
230 | return 0; /* XXX: return error? check spec. */ | 228 | return 0; /* XXX: return error? check spec. */ |
231 | } | 229 | } |
232 | 230 | ||
233 | if (level == SOL_SOCKET && type == SCM_TIMESTAMP) { | 231 | if (!COMPAT_USE_64BIT_TIME) { |
234 | struct timeval *tv = (struct timeval *)data; | 232 | struct compat_timeval ctv; |
235 | ctv.tv_sec = tv->tv_sec; | 233 | struct compat_timespec cts[3]; |
236 | ctv.tv_usec = tv->tv_usec; | 234 | if (level == SOL_SOCKET && type == SCM_TIMESTAMP) { |
237 | data = &ctv; | 235 | struct timeval *tv = (struct timeval *)data; |
238 | len = sizeof(ctv); | 236 | ctv.tv_sec = tv->tv_sec; |
239 | } | 237 | ctv.tv_usec = tv->tv_usec; |
240 | if (level == SOL_SOCKET && | 238 | data = &ctv; |
241 | (type == SCM_TIMESTAMPNS || type == SCM_TIMESTAMPING)) { | 239 | len = sizeof(ctv); |
242 | int count = type == SCM_TIMESTAMPNS ? 1 : 3; | 240 | } |
243 | int i; | 241 | if (level == SOL_SOCKET && |
244 | struct timespec *ts = (struct timespec *)data; | 242 | (type == SCM_TIMESTAMPNS || type == SCM_TIMESTAMPING)) { |
245 | for (i = 0; i < count; i++) { | 243 | int count = type == SCM_TIMESTAMPNS ? 1 : 3; |
246 | cts[i].tv_sec = ts[i].tv_sec; | 244 | int i; |
247 | cts[i].tv_nsec = ts[i].tv_nsec; | 245 | struct timespec *ts = (struct timespec *)data; |
246 | for (i = 0; i < count; i++) { | ||
247 | cts[i].tv_sec = ts[i].tv_sec; | ||
248 | cts[i].tv_nsec = ts[i].tv_nsec; | ||
249 | } | ||
250 | data = &cts; | ||
251 | len = sizeof(cts[0]) * count; | ||
248 | } | 252 | } |
249 | data = &cts; | ||
250 | len = sizeof(cts[0]) * count; | ||
251 | } | 253 | } |
252 | 254 | ||
253 | cmlen = CMSG_COMPAT_LEN(len); | 255 | cmlen = CMSG_COMPAT_LEN(len); |
@@ -454,11 +456,15 @@ static int compat_sock_getsockopt(struct socket *sock, int level, int optname, | |||
454 | 456 | ||
455 | int compat_sock_get_timestamp(struct sock *sk, struct timeval __user *userstamp) | 457 | int compat_sock_get_timestamp(struct sock *sk, struct timeval __user *userstamp) |
456 | { | 458 | { |
457 | struct compat_timeval __user *ctv = | 459 | struct compat_timeval __user *ctv; |
458 | (struct compat_timeval __user *) userstamp; | 460 | int err; |
459 | int err = -ENOENT; | ||
460 | struct timeval tv; | 461 | struct timeval tv; |
461 | 462 | ||
463 | if (COMPAT_USE_64BIT_TIME) | ||
464 | return sock_get_timestamp(sk, userstamp); | ||
465 | |||
466 | ctv = (struct compat_timeval __user *) userstamp; | ||
467 | err = -ENOENT; | ||
462 | if (!sock_flag(sk, SOCK_TIMESTAMP)) | 468 | if (!sock_flag(sk, SOCK_TIMESTAMP)) |
463 | sock_enable_timestamp(sk, SOCK_TIMESTAMP); | 469 | sock_enable_timestamp(sk, SOCK_TIMESTAMP); |
464 | tv = ktime_to_timeval(sk->sk_stamp); | 470 | tv = ktime_to_timeval(sk->sk_stamp); |
@@ -478,11 +484,15 @@ EXPORT_SYMBOL(compat_sock_get_timestamp); | |||
478 | 484 | ||
479 | int compat_sock_get_timestampns(struct sock *sk, struct timespec __user *userstamp) | 485 | int compat_sock_get_timestampns(struct sock *sk, struct timespec __user *userstamp) |
480 | { | 486 | { |
481 | struct compat_timespec __user *ctv = | 487 | struct compat_timespec __user *ctv; |
482 | (struct compat_timespec __user *) userstamp; | 488 | int err; |
483 | int err = -ENOENT; | ||
484 | struct timespec ts; | 489 | struct timespec ts; |
485 | 490 | ||
491 | if (COMPAT_USE_64BIT_TIME) | ||
492 | return sock_get_timestampns (sk, userstamp); | ||
493 | |||
494 | ctv = (struct compat_timespec __user *) userstamp; | ||
495 | err = -ENOENT; | ||
486 | if (!sock_flag(sk, SOCK_TIMESTAMP)) | 496 | if (!sock_flag(sk, SOCK_TIMESTAMP)) |
487 | sock_enable_timestamp(sk, SOCK_TIMESTAMP); | 497 | sock_enable_timestamp(sk, SOCK_TIMESTAMP); |
488 | ts = ktime_to_timespec(sk->sk_stamp); | 498 | ts = ktime_to_timespec(sk->sk_stamp); |
@@ -767,6 +777,11 @@ asmlinkage long compat_sys_recvmmsg(int fd, struct compat_mmsghdr __user *mmsg, | |||
767 | int datagrams; | 777 | int datagrams; |
768 | struct timespec ktspec; | 778 | struct timespec ktspec; |
769 | 779 | ||
780 | if (COMPAT_USE_64BIT_TIME) | ||
781 | return __sys_recvmmsg(fd, (struct mmsghdr __user *)mmsg, vlen, | ||
782 | flags | MSG_CMSG_COMPAT, | ||
783 | (struct timespec *) timeout); | ||
784 | |||
770 | if (timeout == NULL) | 785 | if (timeout == NULL) |
771 | return __sys_recvmmsg(fd, (struct mmsghdr __user *)mmsg, vlen, | 786 | return __sys_recvmmsg(fd, (struct mmsghdr __user *)mmsg, vlen, |
772 | flags | MSG_CMSG_COMPAT, NULL); | 787 | flags | MSG_CMSG_COMPAT, NULL); |