diff options
-rw-r--r-- | arch/arm64/Kconfig | 2 | ||||
-rw-r--r-- | arch/arm64/include/asm/uaccess.h | 25 | ||||
-rw-r--r-- | arch/arm64/include/asm/word-at-a-time.h | 54 | ||||
-rw-r--r-- | arch/arm64/kernel/arm64ksyms.c | 5 | ||||
-rw-r--r-- | arch/arm64/lib/Makefile | 8 | ||||
-rw-r--r-- | arch/arm64/lib/strncpy_from_user.S | 50 | ||||
-rw-r--r-- | arch/arm64/lib/strnlen_user.S | 47 |
7 files changed, 64 insertions, 127 deletions
diff --git a/arch/arm64/Kconfig b/arch/arm64/Kconfig index 6d4dd22ee4b7..295bf2de7128 100644 --- a/arch/arm64/Kconfig +++ b/arch/arm64/Kconfig | |||
@@ -17,6 +17,8 @@ config ARM64 | |||
17 | select GENERIC_IRQ_SHOW | 17 | select GENERIC_IRQ_SHOW |
18 | select GENERIC_SCHED_CLOCK | 18 | select GENERIC_SCHED_CLOCK |
19 | select GENERIC_SMP_IDLE_THREAD | 19 | select GENERIC_SMP_IDLE_THREAD |
20 | select GENERIC_STRNCPY_FROM_USER | ||
21 | select GENERIC_STRNLEN_USER | ||
20 | select GENERIC_TIME_VSYSCALL | 22 | select GENERIC_TIME_VSYSCALL |
21 | select HARDIRQS_SW_RESEND | 23 | select HARDIRQS_SW_RESEND |
22 | select HAVE_ARCH_TRACEHOOK | 24 | select HAVE_ARCH_TRACEHOOK |
diff --git a/arch/arm64/include/asm/uaccess.h b/arch/arm64/include/asm/uaccess.h index 7ecc2b23882e..6c0f684aca81 100644 --- a/arch/arm64/include/asm/uaccess.h +++ b/arch/arm64/include/asm/uaccess.h | |||
@@ -100,6 +100,7 @@ static inline void set_fs(mm_segment_t fs) | |||
100 | }) | 100 | }) |
101 | 101 | ||
102 | #define access_ok(type, addr, size) __range_ok(addr, size) | 102 | #define access_ok(type, addr, size) __range_ok(addr, size) |
103 | #define user_addr_max get_fs | ||
103 | 104 | ||
104 | /* | 105 | /* |
105 | * The "__xxx" versions of the user access functions do not verify the address | 106 | * The "__xxx" versions of the user access functions do not verify the address |
@@ -240,9 +241,6 @@ extern unsigned long __must_check __copy_to_user(void __user *to, const void *fr | |||
240 | extern unsigned long __must_check __copy_in_user(void __user *to, const void __user *from, unsigned long n); | 241 | extern unsigned long __must_check __copy_in_user(void __user *to, const void __user *from, unsigned long n); |
241 | extern unsigned long __must_check __clear_user(void __user *addr, unsigned long n); | 242 | extern unsigned long __must_check __clear_user(void __user *addr, unsigned long n); |
242 | 243 | ||
243 | extern unsigned long __must_check __strncpy_from_user(char *to, const char __user *from, unsigned long count); | ||
244 | extern unsigned long __must_check __strnlen_user(const char __user *s, long n); | ||
245 | |||
246 | static inline unsigned long __must_check copy_from_user(void *to, const void __user *from, unsigned long n) | 244 | static inline unsigned long __must_check copy_from_user(void *to, const void __user *from, unsigned long n) |
247 | { | 245 | { |
248 | if (access_ok(VERIFY_READ, from, n)) | 246 | if (access_ok(VERIFY_READ, from, n)) |
@@ -276,24 +274,9 @@ static inline unsigned long __must_check clear_user(void __user *to, unsigned lo | |||
276 | return n; | 274 | return n; |
277 | } | 275 | } |
278 | 276 | ||
279 | static inline long __must_check strncpy_from_user(char *dst, const char __user *src, long count) | 277 | extern long strncpy_from_user(char *dest, const char __user *src, long count); |
280 | { | ||
281 | long res = -EFAULT; | ||
282 | if (access_ok(VERIFY_READ, src, 1)) | ||
283 | res = __strncpy_from_user(dst, src, count); | ||
284 | return res; | ||
285 | } | ||
286 | |||
287 | #define strlen_user(s) strnlen_user(s, ~0UL >> 1) | ||
288 | 278 | ||
289 | static inline long __must_check strnlen_user(const char __user *s, long n) | 279 | extern __must_check long strlen_user(const char __user *str); |
290 | { | 280 | extern __must_check long strnlen_user(const char __user *str, long n); |
291 | unsigned long res = 0; | ||
292 | |||
293 | if (__addr_ok(s)) | ||
294 | res = __strnlen_user(s, n); | ||
295 | |||
296 | return res; | ||
297 | } | ||
298 | 281 | ||
299 | #endif /* __ASM_UACCESS_H */ | 282 | #endif /* __ASM_UACCESS_H */ |
diff --git a/arch/arm64/include/asm/word-at-a-time.h b/arch/arm64/include/asm/word-at-a-time.h new file mode 100644 index 000000000000..27a167d044d9 --- /dev/null +++ b/arch/arm64/include/asm/word-at-a-time.h | |||
@@ -0,0 +1,54 @@ | |||
1 | /* | ||
2 | * Copyright (C) 2013 ARM Ltd. | ||
3 | * | ||
4 | * This program is free software; you can redistribute it and/or modify | ||
5 | * it under the terms of the GNU General Public License version 2 as | ||
6 | * published by the Free Software Foundation. | ||
7 | * | ||
8 | * This program is distributed in the hope that it will be useful, | ||
9 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
10 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
11 | * GNU General Public License for more details. | ||
12 | * | ||
13 | * You should have received a copy of the GNU General Public License | ||
14 | * along with this program. If not, see <http://www.gnu.org/licenses/>. | ||
15 | */ | ||
16 | #ifndef __ASM_WORD_AT_A_TIME_H | ||
17 | #define __ASM_WORD_AT_A_TIME_H | ||
18 | |||
19 | #ifndef __AARCH64EB__ | ||
20 | |||
21 | #include <linux/kernel.h> | ||
22 | |||
23 | struct word_at_a_time { | ||
24 | const unsigned long one_bits, high_bits; | ||
25 | }; | ||
26 | |||
27 | #define WORD_AT_A_TIME_CONSTANTS { REPEAT_BYTE(0x01), REPEAT_BYTE(0x80) } | ||
28 | |||
29 | static inline unsigned long has_zero(unsigned long a, unsigned long *bits, | ||
30 | const struct word_at_a_time *c) | ||
31 | { | ||
32 | unsigned long mask = ((a - c->one_bits) & ~a) & c->high_bits; | ||
33 | *bits = mask; | ||
34 | return mask; | ||
35 | } | ||
36 | |||
37 | #define prep_zero_mask(a, bits, c) (bits) | ||
38 | |||
39 | static inline unsigned long create_zero_mask(unsigned long bits) | ||
40 | { | ||
41 | bits = (bits - 1) & ~bits; | ||
42 | return bits >> 7; | ||
43 | } | ||
44 | |||
45 | static inline unsigned long find_zero(unsigned long mask) | ||
46 | { | ||
47 | return fls64(mask) >> 3; | ||
48 | } | ||
49 | |||
50 | #else /* __AARCH64EB__ */ | ||
51 | #include <asm-generic/word-at-a-time.h> | ||
52 | #endif | ||
53 | |||
54 | #endif /* __ASM_WORD_AT_A_TIME_H */ | ||
diff --git a/arch/arm64/kernel/arm64ksyms.c b/arch/arm64/kernel/arm64ksyms.c index e7ee770c0697..338b568cd8ae 100644 --- a/arch/arm64/kernel/arm64ksyms.c +++ b/arch/arm64/kernel/arm64ksyms.c | |||
@@ -29,13 +29,10 @@ | |||
29 | 29 | ||
30 | #include <asm/checksum.h> | 30 | #include <asm/checksum.h> |
31 | 31 | ||
32 | /* user mem (segment) */ | ||
33 | EXPORT_SYMBOL(__strnlen_user); | ||
34 | EXPORT_SYMBOL(__strncpy_from_user); | ||
35 | |||
36 | EXPORT_SYMBOL(copy_page); | 32 | EXPORT_SYMBOL(copy_page); |
37 | EXPORT_SYMBOL(clear_page); | 33 | EXPORT_SYMBOL(clear_page); |
38 | 34 | ||
35 | /* user mem (segment) */ | ||
39 | EXPORT_SYMBOL(__copy_from_user); | 36 | EXPORT_SYMBOL(__copy_from_user); |
40 | EXPORT_SYMBOL(__copy_to_user); | 37 | EXPORT_SYMBOL(__copy_to_user); |
41 | EXPORT_SYMBOL(__clear_user); | 38 | EXPORT_SYMBOL(__clear_user); |
diff --git a/arch/arm64/lib/Makefile b/arch/arm64/lib/Makefile index 59acc0ef0462..328ce1a99daa 100644 --- a/arch/arm64/lib/Makefile +++ b/arch/arm64/lib/Makefile | |||
@@ -1,6 +1,4 @@ | |||
1 | lib-y := bitops.o delay.o \ | 1 | lib-y := bitops.o clear_user.o delay.o copy_from_user.o \ |
2 | strncpy_from_user.o strnlen_user.o clear_user.o \ | 2 | copy_to_user.o copy_in_user.o copy_page.o \ |
3 | copy_from_user.o copy_to_user.o copy_in_user.o \ | 3 | clear_page.o memchr.o memcpy.o memmove.o memset.o \ |
4 | copy_page.o clear_page.o \ | ||
5 | memchr.o memcpy.o memmove.o memset.o \ | ||
6 | strchr.o strrchr.o | 4 | strchr.o strrchr.o |
diff --git a/arch/arm64/lib/strncpy_from_user.S b/arch/arm64/lib/strncpy_from_user.S deleted file mode 100644 index 56e448a831a0..000000000000 --- a/arch/arm64/lib/strncpy_from_user.S +++ /dev/null | |||
@@ -1,50 +0,0 @@ | |||
1 | /* | ||
2 | * Based on arch/arm/lib/strncpy_from_user.S | ||
3 | * | ||
4 | * Copyright (C) 1995-2000 Russell King | ||
5 | * Copyright (C) 2012 ARM Ltd. | ||
6 | * | ||
7 | * This program is free software; you can redistribute it and/or modify | ||
8 | * it under the terms of the GNU General Public License version 2 as | ||
9 | * published by the Free Software Foundation. | ||
10 | * | ||
11 | * This program is distributed in the hope that it will be useful, | ||
12 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
13 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
14 | * GNU General Public License for more details. | ||
15 | * | ||
16 | * You should have received a copy of the GNU General Public License | ||
17 | * along with this program. If not, see <http://www.gnu.org/licenses/>. | ||
18 | */ | ||
19 | |||
20 | #include <linux/linkage.h> | ||
21 | #include <asm/assembler.h> | ||
22 | #include <asm/errno.h> | ||
23 | |||
24 | .text | ||
25 | .align 5 | ||
26 | |||
27 | /* | ||
28 | * Copy a string from user space to kernel space. | ||
29 | * x0 = dst, x1 = src, x2 = byte length | ||
30 | * returns the number of characters copied (strlen of copied string), | ||
31 | * -EFAULT on exception, or "len" if we fill the whole buffer | ||
32 | */ | ||
33 | ENTRY(__strncpy_from_user) | ||
34 | mov x4, x1 | ||
35 | 1: subs x2, x2, #1 | ||
36 | bmi 2f | ||
37 | USER(9f, ldrb w3, [x1], #1 ) | ||
38 | strb w3, [x0], #1 | ||
39 | cbnz w3, 1b | ||
40 | sub x1, x1, #1 // take NUL character out of count | ||
41 | 2: sub x0, x1, x4 | ||
42 | ret | ||
43 | ENDPROC(__strncpy_from_user) | ||
44 | |||
45 | .section .fixup,"ax" | ||
46 | .align 0 | ||
47 | 9: strb wzr, [x0] // null terminate | ||
48 | mov x0, #-EFAULT | ||
49 | ret | ||
50 | .previous | ||
diff --git a/arch/arm64/lib/strnlen_user.S b/arch/arm64/lib/strnlen_user.S deleted file mode 100644 index 7f7b176a5646..000000000000 --- a/arch/arm64/lib/strnlen_user.S +++ /dev/null | |||
@@ -1,47 +0,0 @@ | |||
1 | /* | ||
2 | * Based on arch/arm/lib/strnlen_user.S | ||
3 | * | ||
4 | * Copyright (C) 1995-2000 Russell King | ||
5 | * Copyright (C) 2012 ARM Ltd. | ||
6 | * | ||
7 | * This program is free software; you can redistribute it and/or modify | ||
8 | * it under the terms of the GNU General Public License version 2 as | ||
9 | * published by the Free Software Foundation. | ||
10 | * | ||
11 | * This program is distributed in the hope that it will be useful, | ||
12 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
13 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
14 | * GNU General Public License for more details. | ||
15 | * | ||
16 | * You should have received a copy of the GNU General Public License | ||
17 | * along with this program. If not, see <http://www.gnu.org/licenses/>. | ||
18 | */ | ||
19 | |||
20 | #include <linux/linkage.h> | ||
21 | #include <asm/assembler.h> | ||
22 | #include <asm/errno.h> | ||
23 | |||
24 | .text | ||
25 | .align 5 | ||
26 | |||
27 | /* Prototype: unsigned long __strnlen_user(const char *str, long n) | ||
28 | * Purpose : get length of a string in user memory | ||
29 | * Params : str - address of string in user memory | ||
30 | * Returns : length of string *including terminator* | ||
31 | * or zero on exception, or n if too long | ||
32 | */ | ||
33 | ENTRY(__strnlen_user) | ||
34 | mov x2, x0 | ||
35 | 1: subs x1, x1, #1 | ||
36 | b.mi 2f | ||
37 | USER(9f, ldrb w3, [x0], #1 ) | ||
38 | cbnz w3, 1b | ||
39 | 2: sub x0, x0, x2 | ||
40 | ret | ||
41 | ENDPROC(__strnlen_user) | ||
42 | |||
43 | .section .fixup,"ax" | ||
44 | .align 0 | ||
45 | 9: mov x0, #0 | ||
46 | ret | ||
47 | .previous | ||