Commit e9213c78 authored by Anton Blanchard's avatar Anton Blanchard

ppc64: rework user access functions

parent 2d38a917
...@@ -200,6 +200,7 @@ _GLOBAL(__clear_user) ...@@ -200,6 +200,7 @@ _GLOBAL(__clear_user)
.llong 8b,92b .llong 8b,92b
.text .text
/* r3 = dst, r4 = src, r5 = count */
_GLOBAL(__strncpy_from_user) _GLOBAL(__strncpy_from_user)
addi r6,r3,-1 addi r6,r3,-1
addi r4,r4,-1 addi r4,r4,-1
...@@ -222,14 +223,10 @@ _GLOBAL(__strncpy_from_user) ...@@ -222,14 +223,10 @@ _GLOBAL(__strncpy_from_user)
.llong 1b,99b .llong 1b,99b
.text .text
/* r3 = str, r4 = len (> 0), r5 = top (highest addr) */ /* r3 = str, r4 = len (> 0) */
_GLOBAL(__strnlen_user) _GLOBAL(__strnlen_user)
addi r7,r3,-1 addi r7,r3,-1
subf r6,r7,r5 /* top+1 - str */ mtctr r4 /* ctr = len */
cmplw 0,r4,r6
bge 0f
mr r6,r4
0: mtctr r6 /* ctr = min(len, top - str) */
1: lbzu r0,1(r7) /* get next byte */ 1: lbzu r0,1(r7) /* get next byte */
cmpwi 0,r0,0 cmpwi 0,r0,0
bdnzf 2,1b /* loop if --ctr != 0 && byte != 0 */ bdnzf 2,1b /* loop if --ctr != 0 && byte != 0 */
......
...@@ -24,8 +24,10 @@ ...@@ -24,8 +24,10 @@
* For historical reasons, these macros are grossly misnamed. * For historical reasons, these macros are grossly misnamed.
*/ */
#define KERNEL_DS ((mm_segment_t) { 0 }) #define MAKE_MM_SEG(s) ((mm_segment_t) { (s) })
#define USER_DS ((mm_segment_t) { 1 })
#define KERNEL_DS MAKE_MM_SEG(0UL)
#define USER_DS MAKE_MM_SEG(0xf000000000000000UL)
#define get_ds() (KERNEL_DS) #define get_ds() (KERNEL_DS)
#define get_fs() (current->thread.fs) #define get_fs() (current->thread.fs)
...@@ -33,12 +35,25 @@ ...@@ -33,12 +35,25 @@
#define segment_eq(a,b) ((a).seg == (b).seg) #define segment_eq(a,b) ((a).seg == (b).seg)
#define __kernel_ok (segment_eq(get_fs(), KERNEL_DS)) /*
#define __user_ok(addr,size) (((size) <= TASK_SIZE)&&((addr) <= TASK_SIZE-(size))) * Use the alpha trick for checking ranges:
#define __access_ok(addr,size) (__kernel_ok || __user_ok((addr),(size))) *
#define access_ok(type,addr,size) __access_ok((unsigned long)(addr),(size)) * Is a address valid? This does a straightforward calculation rather
* than tests.
*
* Address valid if:
* - "addr" doesn't have any high-bits set
* - AND "size" doesn't have any high-bits set
* - AND "addr+size" doesn't have any high-bits set
* - OR we are in kernel mode.
*/
#define __access_ok(addr,size,segment) \
(((segment).seg & (addr | size | (addr+size))) == 0)
#define access_ok(type,addr,size) \
__access_ok(((unsigned long)(addr)),(size),get_fs())
static inline int verify_area(int type, const void * addr, unsigned long size) static inline int verify_area(int type, const void *addr, unsigned long size)
{ {
return access_ok(type,addr,size) ? 0 : -EFAULT; return access_ok(type,addr,size) ? 0 : -EFAULT;
} }
...@@ -96,32 +111,32 @@ extern void sort_exception_table(void); ...@@ -96,32 +111,32 @@ extern void sort_exception_table(void);
extern long __put_user_bad(void); extern long __put_user_bad(void);
#define __put_user_nocheck(x,ptr,size) \ #define __put_user_nocheck(x,ptr,size) \
({ \
long __pu_err; \
__put_user_size((x),(ptr),(size),__pu_err); \
__pu_err; \
})
#define __put_user_check(x,ptr,size) \
({ \ ({ \
long __pu_err = -EFAULT; \ long __pu_err; \
__typeof__(*(ptr)) *__pu_addr = (ptr); \ __put_user_size((x),(ptr),(size),__pu_err,-EFAULT); \
if (access_ok(VERIFY_WRITE,__pu_addr,size)) \
__put_user_size((x),__pu_addr,(size),__pu_err); \
__pu_err; \ __pu_err; \
}) })
#define __put_user_size(x,ptr,size,retval) \ #define __put_user_check(x,ptr,size) \
do { \ ({ \
retval = 0; \ long __pu_err = -EFAULT; \
switch (size) { \ __typeof__(*(ptr)) *__pu_addr = (ptr); \
case 1: __put_user_asm(x,ptr,retval,"stb"); break; \ if (access_ok(VERIFY_WRITE,__pu_addr,size)) \
case 2: __put_user_asm(x,ptr,retval,"sth"); break; \ __put_user_size((x),__pu_addr,(size),__pu_err,-EFAULT); \
case 4: __put_user_asm(x,ptr,retval,"stw"); break; \ __pu_err; \
case 8: __put_user_asm(x,ptr,retval,"std"); break; \ })
default: __put_user_bad(); \
} \ #define __put_user_size(x,ptr,size,retval,errret) \
do { \
retval = 0; \
switch (size) { \
case 1: __put_user_asm(x,ptr,retval,"stb",errret); break; \
case 2: __put_user_asm(x,ptr,retval,"sth",errret); break; \
case 4: __put_user_asm(x,ptr,retval,"stw",errret); break; \
case 8: __put_user_asm(x,ptr,retval,"std",errret); break; \
default: __put_user_bad(); \
} \
} while (0) } while (0)
/* /*
...@@ -129,7 +144,7 @@ do { \ ...@@ -129,7 +144,7 @@ do { \
* because we do not write to any memory gcc knows about, so there * because we do not write to any memory gcc knows about, so there
* are no aliasing issues. * are no aliasing issues.
*/ */
#define __put_user_asm(x, addr, err, op) \ #define __put_user_asm(x, addr, err, op, errret) \
__asm__ __volatile__( \ __asm__ __volatile__( \
"1: "op" %1,0(%2) # put_user\n" \ "1: "op" %1,0(%2) # put_user\n" \
"2:\n" \ "2:\n" \
...@@ -142,13 +157,13 @@ do { \ ...@@ -142,13 +157,13 @@ do { \
" .llong 1b,3b\n" \ " .llong 1b,3b\n" \
".previous" \ ".previous" \
: "=r"(err) \ : "=r"(err) \
: "r"(x), "b"(addr), "i"(-EFAULT), "0"(err)) : "r"(x), "b"(addr), "i"(errret), "0"(err))
#define __get_user_nocheck(x,ptr,size) \ #define __get_user_nocheck(x,ptr,size) \
({ \ ({ \
long __gu_err, __gu_val; \ long __gu_err, __gu_val; \
__get_user_size(__gu_val,(ptr),(size),__gu_err); \ __get_user_size(__gu_val,(ptr),(size),__gu_err,-EFAULT);\
(x) = (__typeof__(*(ptr)))__gu_val; \ (x) = (__typeof__(*(ptr)))__gu_val; \
__gu_err; \ __gu_err; \
}) })
...@@ -158,26 +173,26 @@ do { \ ...@@ -158,26 +173,26 @@ do { \
long __gu_err = -EFAULT, __gu_val = 0; \ long __gu_err = -EFAULT, __gu_val = 0; \
const __typeof__(*(ptr)) *__gu_addr = (ptr); \ const __typeof__(*(ptr)) *__gu_addr = (ptr); \
if (access_ok(VERIFY_READ,__gu_addr,size)) \ if (access_ok(VERIFY_READ,__gu_addr,size)) \
__get_user_size(__gu_val,__gu_addr,(size),__gu_err); \ __get_user_size(__gu_val,__gu_addr,(size),__gu_err,-EFAULT);\
(x) = (__typeof__(*(ptr)))__gu_val; \ (x) = (__typeof__(*(ptr)))__gu_val; \
__gu_err; \ __gu_err; \
}) })
extern long __get_user_bad(void); extern long __get_user_bad(void);
#define __get_user_size(x,ptr,size,retval) \ #define __get_user_size(x,ptr,size,retval,errret) \
do { \ do { \
retval = 0; \ retval = 0; \
switch (size) { \ switch (size) { \
case 1: __get_user_asm(x,ptr,retval,"lbz"); break; \ case 1: __get_user_asm(x,ptr,retval,"lbz",errret); break; \
case 2: __get_user_asm(x,ptr,retval,"lhz"); break; \ case 2: __get_user_asm(x,ptr,retval,"lhz",errret); break; \
case 4: __get_user_asm(x,ptr,retval,"lwz"); break; \ case 4: __get_user_asm(x,ptr,retval,"lwz",errret); break; \
case 8: __get_user_asm(x,ptr,retval,"ld"); break; \ case 8: __get_user_asm(x,ptr,retval,"ld",errret); break; \
default: (x) = __get_user_bad(); \ default: (x) = __get_user_bad(); \
} \ } \
} while (0) } while (0)
#define __get_user_asm(x, addr, err, op) \ #define __get_user_asm(x, addr, err, op, errret) \
__asm__ __volatile__( \ __asm__ __volatile__( \
"1: "op" %1,0(%2) # get_user\n" \ "1: "op" %1,0(%2) # get_user\n" \
"2:\n" \ "2:\n" \
...@@ -191,69 +206,97 @@ do { \ ...@@ -191,69 +206,97 @@ do { \
" .llong 1b,3b\n" \ " .llong 1b,3b\n" \
".previous" \ ".previous" \
: "=r"(err), "=r"(x) \ : "=r"(err), "=r"(x) \
: "b"(addr), "i"(-EFAULT), "0"(err)) : "b"(addr), "i"(errret), "0"(err))
/* more complex routines */ /* more complex routines */
extern unsigned long __copy_tofrom_user(void *to, const void *from, unsigned long size); extern unsigned long __copy_tofrom_user(void *to, const void *from,
unsigned long size);
/* XXX should zero destination if fault happened */
static inline unsigned long static inline unsigned long
copy_from_user(void *to, const void *from, unsigned long n) __copy_from_user(void *to, const void __user *from, unsigned long n)
{
if (__builtin_constant_p(n)) {
unsigned long ret;
switch (n) {
case 1:
__get_user_size(*(u8 *)to, from, 1, ret, 1);
return ret;
case 2:
__get_user_size(*(u16 *)to, from, 2, ret, 2);
return ret;
case 4:
__get_user_size(*(u32 *)to, from, 4, ret, 4);
return ret;
case 8:
__get_user_size(*(u64 *)to, from, 8, ret, 8);
return ret;
}
}
return __copy_tofrom_user(to, from, n);
}
static inline unsigned long
__copy_to_user(void __user *to, const void *from, unsigned long n)
{ {
unsigned long over; if (__builtin_constant_p(n)) {
unsigned long ret;
if (access_ok(VERIFY_READ, from, n))
return __copy_tofrom_user(to, from, n); switch (n) {
if ((unsigned long)from < TASK_SIZE) { case 1:
over = (unsigned long)from + n - TASK_SIZE; __put_user_size(*(u8 *)from, (u8 *)to, 1, ret, 1);
memset(to + over, 0, over); return ret;
return __copy_tofrom_user(to, from, n - over) + over; case 2:
__put_user_size(*(u16 *)from, (u16 *)to, 2, ret, 2);
return ret;
case 4:
__put_user_size(*(u32 *)from, (u32 *)to, 4, ret, 4);
return ret;
case 8:
__put_user_size(*(u64 *)from, (u64 *)to, 8, ret, 8);
return ret;
}
} }
return __copy_tofrom_user(to, from, n);
}
#define __copy_in_user(to, from, size) \
__copy_tofrom_user((to), (from), (size))
static inline unsigned long
copy_from_user(void *to, const void *from, unsigned long n)
{
if (likely(access_ok(VERIFY_READ, from, n)))
n = __copy_from_user(to, from, n);
return n; return n;
} }
static inline unsigned long static inline unsigned long
copy_to_user(void *to, const void *from, unsigned long n) copy_to_user(void *to, const void *from, unsigned long n)
{ {
unsigned long over; if (likely(access_ok(VERIFY_WRITE, to, n)))
n = __copy_to_user(to, from, n);
if (access_ok(VERIFY_WRITE, to, n))
return __copy_tofrom_user(to, from, n);
if ((unsigned long)to < TASK_SIZE) {
over = (unsigned long)to + n - TASK_SIZE;
return __copy_tofrom_user(to, from, n - over) + over;
}
return n; return n;
} }
static inline unsigned long static inline unsigned long
copy_in_user(void *to, const void *from, unsigned long n) copy_in_user(void *to, const void *from, unsigned long n)
{ {
if (!access_ok(VERIFY_READ, from, n) || if (likely(access_ok(VERIFY_READ, from, n) &&
!access_ok(VERIFY_WRITE, to, n)) access_ok(VERIFY_WRITE, to, n)))
return n; n =__copy_tofrom_user(to, from, n);
return n;
return __copy_tofrom_user(to, from, n);
} }
#define __copy_from_user(to, from, size) \
__copy_tofrom_user((to), (from), (size))
#define __copy_to_user(to, from, size) \
__copy_tofrom_user((to), (from), (size))
#define __copy_in_user(to, from, size) \
__copy_tofrom_user((to), (from), (size))
extern unsigned long __clear_user(void *addr, unsigned long size); extern unsigned long __clear_user(void *addr, unsigned long size);
static inline unsigned long static inline unsigned long
clear_user(void *addr, unsigned long size) clear_user(void *addr, unsigned long size)
{ {
if (access_ok(VERIFY_WRITE, addr, size)) if (likely(access_ok(VERIFY_WRITE, addr, size)))
return __clear_user(addr, size); size = __clear_user(addr, size);
if ((unsigned long)addr < TASK_SIZE) {
unsigned long over = (unsigned long)addr + size - TASK_SIZE;
return __clear_user(addr, size - over) + over;
}
return size; return size;
} }
...@@ -262,7 +305,7 @@ extern int __strncpy_from_user(char *dst, const char *src, long count); ...@@ -262,7 +305,7 @@ extern int __strncpy_from_user(char *dst, const char *src, long count);
static inline long static inline long
strncpy_from_user(char *dst, const char *src, long count) strncpy_from_user(char *dst, const char *src, long count)
{ {
if (access_ok(VERIFY_READ, src, 1)) if (likely(access_ok(VERIFY_READ, src, 1)))
return __strncpy_from_user(dst, src, count); return __strncpy_from_user(dst, src, count);
return -EFAULT; return -EFAULT;
} }
...@@ -272,24 +315,18 @@ strncpy_from_user(char *dst, const char *src, long count) ...@@ -272,24 +315,18 @@ strncpy_from_user(char *dst, const char *src, long count)
* *
* Return 0 for error * Return 0 for error
*/ */
extern int __strnlen_user(const char *str, long len);
extern int __strnlen_user(const char *str, long len, unsigned long top);
/* /*
* Returns the length of the string at str (including the null byte), * Returns the length of the string at str (including the null byte),
* or 0 if we hit a page we can't access, * or 0 if we hit a page we can't access,
* or something > len if we didn't find a null byte. * or something > len if we didn't find a null byte.
*
* The `top' parameter to __strnlen_user is to make sure that
* we can never overflow from the user area into kernel space.
*/ */
static inline int strnlen_user(const char *str, long len) static inline int strnlen_user(const char *str, long len)
{ {
unsigned long top = __kernel_ok? ~0UL: TASK_SIZE - 1; if (likely(access_ok(VERIFY_READ, str, 1)))
return __strnlen_user(str, len);
if ((unsigned long)str > top) return 0;
return 0;
return __strnlen_user(str, len, top);
} }
#define strlen_user(str) strnlen_user((str), 0x7ffffffe) #define strlen_user(str) strnlen_user((str), 0x7ffffffe)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment