diff options
author | Michal Simek <michal.simek@xilinx.com> | 2013-08-13 19:00:53 -0400 |
---|---|---|
committer | Greg Kroah-Hartman <gregkh@linuxfoundation.org> | 2013-08-20 11:43:02 -0400 |
commit | 4f01c72ef36d3305d6273fe7f1f6670c52745c3d (patch) | |
tree | ab71ffb7a755c453f84a2aaf37fe285d0bcdcd31 | |
parent | 0dcf19b4fb41449de4d1f953f86aa6a90accdff5 (diff) |
microblaze: fix clone syscall
commit dfa9771a7c4784bafd0673bc7abcee3813088b77 upstream.
Fix inadvertent breakage in the clone syscall ABI for Microblaze that
was introduced in commit f3268edbe6fe ("microblaze: switch to generic
fork/vfork/clone").
The Microblaze syscall ABI for clone takes the parent tid address in the
4th argument; the third argument slot is used for the stack size. The
incorrectly-used CLONE_BACKWARDS type assigned parent tid to the 3rd
slot.
This commit restores the original ABI so that existing userspace libc
code will work correctly.
All kernel versions from v3.8-rc1 were affected.
Signed-off-by: Michal Simek <michal.simek@xilinx.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
-rw-r--r-- | arch/Kconfig | 6 | ||||
-rw-r--r-- | arch/microblaze/Kconfig | 2 | ||||
-rw-r--r-- | include/linux/syscalls.h | 5 | ||||
-rw-r--r-- | kernel/fork.c | 6 |
4 files changed, 18 insertions, 1 deletions
diff --git a/arch/Kconfig b/arch/Kconfig index a4429bcd609e..00e3702ec79b 100644 --- a/arch/Kconfig +++ b/arch/Kconfig | |||
@@ -404,6 +404,12 @@ config CLONE_BACKWARDS2 | |||
404 | help | 404 | help |
405 | Architecture has the first two arguments of clone(2) swapped. | 405 | Architecture has the first two arguments of clone(2) swapped. |
406 | 406 | ||
407 | config CLONE_BACKWARDS3 | ||
408 | bool | ||
409 | help | ||
410 | Architecture has tls passed as the 3rd argument of clone(2), | ||
411 | not the 5th one. | ||
412 | |||
407 | config ODD_RT_SIGACTION | 413 | config ODD_RT_SIGACTION |
408 | bool | 414 | bool |
409 | help | 415 | help |
diff --git a/arch/microblaze/Kconfig b/arch/microblaze/Kconfig index d22a4ecffff4..4fab52294d98 100644 --- a/arch/microblaze/Kconfig +++ b/arch/microblaze/Kconfig | |||
@@ -28,7 +28,7 @@ config MICROBLAZE | |||
28 | select GENERIC_CLOCKEVENTS | 28 | select GENERIC_CLOCKEVENTS |
29 | select GENERIC_IDLE_POLL_SETUP | 29 | select GENERIC_IDLE_POLL_SETUP |
30 | select MODULES_USE_ELF_RELA | 30 | select MODULES_USE_ELF_RELA |
31 | select CLONE_BACKWARDS | 31 | select CLONE_BACKWARDS3 |
32 | 32 | ||
33 | config SWAP | 33 | config SWAP |
34 | def_bool n | 34 | def_bool n |
diff --git a/include/linux/syscalls.h b/include/linux/syscalls.h index 4147d700a293..84662ecc7b51 100644 --- a/include/linux/syscalls.h +++ b/include/linux/syscalls.h | |||
@@ -802,9 +802,14 @@ asmlinkage long sys_vfork(void); | |||
802 | asmlinkage long sys_clone(unsigned long, unsigned long, int __user *, int, | 802 | asmlinkage long sys_clone(unsigned long, unsigned long, int __user *, int, |
803 | int __user *); | 803 | int __user *); |
804 | #else | 804 | #else |
805 | #ifdef CONFIG_CLONE_BACKWARDS3 | ||
806 | asmlinkage long sys_clone(unsigned long, unsigned long, int, int __user *, | ||
807 | int __user *, int); | ||
808 | #else | ||
805 | asmlinkage long sys_clone(unsigned long, unsigned long, int __user *, | 809 | asmlinkage long sys_clone(unsigned long, unsigned long, int __user *, |
806 | int __user *, int); | 810 | int __user *, int); |
807 | #endif | 811 | #endif |
812 | #endif | ||
808 | 813 | ||
809 | asmlinkage long sys_execve(const char __user *filename, | 814 | asmlinkage long sys_execve(const char __user *filename, |
810 | const char __user *const __user *argv, | 815 | const char __user *const __user *argv, |
diff --git a/kernel/fork.c b/kernel/fork.c index 987b28a1f01b..ffbc0904794e 100644 --- a/kernel/fork.c +++ b/kernel/fork.c | |||
@@ -1675,6 +1675,12 @@ SYSCALL_DEFINE5(clone, unsigned long, newsp, unsigned long, clone_flags, | |||
1675 | int __user *, parent_tidptr, | 1675 | int __user *, parent_tidptr, |
1676 | int __user *, child_tidptr, | 1676 | int __user *, child_tidptr, |
1677 | int, tls_val) | 1677 | int, tls_val) |
1678 | #elif defined(CONFIG_CLONE_BACKWARDS3) | ||
1679 | SYSCALL_DEFINE6(clone, unsigned long, clone_flags, unsigned long, newsp, | ||
1680 | int, stack_size, | ||
1681 | int __user *, parent_tidptr, | ||
1682 | int __user *, child_tidptr, | ||
1683 | int, tls_val) | ||
1678 | #else | 1684 | #else |
1679 | SYSCALL_DEFINE5(clone, unsigned long, clone_flags, unsigned long, newsp, | 1685 | SYSCALL_DEFINE5(clone, unsigned long, clone_flags, unsigned long, newsp, |
1680 | int __user *, parent_tidptr, | 1686 | int __user *, parent_tidptr, |