Commit 6df77494 authored by Alexander Viro's avatar Alexander Viro Committed by Linus Torvalds

[PATCH] sparse alpha: beginning of __user annotation

Copying primitives annotated.
parent cac8e009
...@@ -99,7 +99,7 @@ static inline unsigned short from64to16(unsigned long x) ...@@ -99,7 +99,7 @@ static inline unsigned short from64to16(unsigned long x)
* Ok. This isn't fun, but this is the EASY case. * Ok. This isn't fun, but this is the EASY case.
*/ */
static inline unsigned long static inline unsigned long
csum_partial_cfu_aligned(const unsigned long *src, unsigned long *dst, csum_partial_cfu_aligned(const unsigned long __user *src, unsigned long *dst,
long len, unsigned long checksum, long len, unsigned long checksum,
int *errp) int *errp)
{ {
...@@ -139,7 +139,8 @@ csum_partial_cfu_aligned(const unsigned long *src, unsigned long *dst, ...@@ -139,7 +139,8 @@ csum_partial_cfu_aligned(const unsigned long *src, unsigned long *dst,
* easy. * easy.
*/ */
static inline unsigned long static inline unsigned long
csum_partial_cfu_dest_aligned(const unsigned long *src, unsigned long *dst, csum_partial_cfu_dest_aligned(const unsigned long __user *src,
unsigned long *dst,
unsigned long soff, unsigned long soff,
long len, unsigned long checksum, long len, unsigned long checksum,
int *errp) int *errp)
...@@ -327,7 +328,7 @@ csum_partial_cfu_unaligned(const unsigned long * src, unsigned long * dst, ...@@ -327,7 +328,7 @@ csum_partial_cfu_unaligned(const unsigned long * src, unsigned long * dst,
} }
static unsigned int static unsigned int
do_csum_partial_copy_from_user(const char *src, char *dst, int len, do_csum_partial_copy_from_user(const char __user *src, char *dst, int len,
unsigned int sum, int *errp) unsigned int sum, int *errp)
{ {
unsigned long checksum = (unsigned) sum; unsigned long checksum = (unsigned) sum;
...@@ -338,12 +339,12 @@ do_csum_partial_copy_from_user(const char *src, char *dst, int len, ...@@ -338,12 +339,12 @@ do_csum_partial_copy_from_user(const char *src, char *dst, int len,
if (!doff) { if (!doff) {
if (!soff) if (!soff)
checksum = csum_partial_cfu_aligned( checksum = csum_partial_cfu_aligned(
(const unsigned long *) src, (const unsigned long __user *) src,
(unsigned long *) dst, (unsigned long *) dst,
len-8, checksum, errp); len-8, checksum, errp);
else else
checksum = csum_partial_cfu_dest_aligned( checksum = csum_partial_cfu_dest_aligned(
(const unsigned long *) src, (const unsigned long __user *) src,
(unsigned long *) dst, (unsigned long *) dst,
soff, len-8, checksum, errp); soff, len-8, checksum, errp);
} else { } else {
...@@ -351,13 +352,13 @@ do_csum_partial_copy_from_user(const char *src, char *dst, int len, ...@@ -351,13 +352,13 @@ do_csum_partial_copy_from_user(const char *src, char *dst, int len,
ldq_u(partial_dest, dst); ldq_u(partial_dest, dst);
if (!soff) if (!soff)
checksum = csum_partial_cfu_src_aligned( checksum = csum_partial_cfu_src_aligned(
(const unsigned long *) src, (const unsigned long __user *) src,
(unsigned long *) dst, (unsigned long *) dst,
doff, len-8, checksum, doff, len-8, checksum,
partial_dest, errp); partial_dest, errp);
else else
checksum = csum_partial_cfu_unaligned( checksum = csum_partial_cfu_unaligned(
(const unsigned long *) src, (const unsigned long __user *) src,
(unsigned long *) dst, (unsigned long *) dst,
soff, doff, len-8, checksum, soff, doff, len-8, checksum,
partial_dest, errp); partial_dest, errp);
...@@ -368,7 +369,7 @@ do_csum_partial_copy_from_user(const char *src, char *dst, int len, ...@@ -368,7 +369,7 @@ do_csum_partial_copy_from_user(const char *src, char *dst, int len,
} }
unsigned int unsigned int
csum_partial_copy_from_user(const char *src, char *dst, int len, csum_partial_copy_from_user(const char __user *src, char *dst, int len,
unsigned int sum, int *errp) unsigned int sum, int *errp)
{ {
if (!access_ok(src, len, VERIFY_READ)) { if (!access_ok(src, len, VERIFY_READ)) {
......
...@@ -43,7 +43,7 @@ extern unsigned int csum_partial(const unsigned char * buff, int len, unsigned i ...@@ -43,7 +43,7 @@ extern unsigned int csum_partial(const unsigned char * buff, int len, unsigned i
* here even more important to align src and dst on a 32-bit (or even * here even more important to align src and dst on a 32-bit (or even
* better 64-bit) boundary * better 64-bit) boundary
*/ */
unsigned int csum_partial_copy_from_user(const char *src, char *dst, int len, unsigned int sum, int *errp); unsigned int csum_partial_copy_from_user(const char __user *src, char *dst, int len, unsigned int sum, int *errp);
unsigned int csum_partial_copy_nocheck(const char *src, char *dst, int len, unsigned int sum); unsigned int csum_partial_copy_nocheck(const char *src, char *dst, int len, unsigned int sum);
......
...@@ -29,6 +29,14 @@ ...@@ -29,6 +29,14 @@
#define segment_eq(a,b) ((a).seg == (b).seg) #define segment_eq(a,b) ((a).seg == (b).seg)
#ifdef __CHECKER__
#define CHECK_UPTR(ptr) do { \
__typeof__(*(ptr)) *__dummy_check_uptr = \
(void __user *)&__dummy_check_uptr; \
} while(0)
#else
#define CHECK_UPTR(ptr)
#endif
/* /*
* Is a address valid? This does a straightforward calculation rather * Is a address valid? This does a straightforward calculation rather
...@@ -43,10 +51,13 @@ ...@@ -43,10 +51,13 @@
#define __access_ok(addr,size,segment) \ #define __access_ok(addr,size,segment) \
(((segment).seg & (addr | size | (addr+size))) == 0) (((segment).seg & (addr | size | (addr+size))) == 0)
#define access_ok(type,addr,size) \ #define access_ok(type,addr,size) \
__access_ok(((unsigned long)(addr)),(size),get_fs()) ({ \
CHECK_UPTR(addr); \
__access_ok(((unsigned long)(addr)),(size),get_fs()); \
})
extern inline int verify_area(int type, const void * addr, unsigned long size) extern inline int verify_area(int type, const void __user * addr, unsigned long size)
{ {
return access_ok(type,addr,size) ? 0 : -EFAULT; return access_ok(type,addr,size) ? 0 : -EFAULT;
} }
...@@ -90,6 +101,7 @@ extern void __get_user_unknown(void); ...@@ -90,6 +101,7 @@ extern void __get_user_unknown(void);
#define __get_user_nocheck(x,ptr,size) \ #define __get_user_nocheck(x,ptr,size) \
({ \ ({ \
long __gu_err = 0, __gu_val; \ long __gu_err = 0, __gu_val; \
CHECK_UPTR(ptr); \
switch (size) { \ switch (size) { \
case 1: __get_user_8(ptr); break; \ case 1: __get_user_8(ptr); break; \
case 2: __get_user_16(ptr); break; \ case 2: __get_user_16(ptr); break; \
...@@ -105,6 +117,7 @@ extern void __get_user_unknown(void); ...@@ -105,6 +117,7 @@ extern void __get_user_unknown(void);
({ \ ({ \
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); \
CHECK_UPTR(ptr); \
if (__access_ok((long)__gu_addr,size,segment)) { \ if (__access_ok((long)__gu_addr,size,segment)) { \
__gu_err = 0; \ __gu_err = 0; \
switch (size) { \ switch (size) { \
...@@ -204,6 +217,7 @@ extern void __put_user_unknown(void); ...@@ -204,6 +217,7 @@ extern void __put_user_unknown(void);
#define __put_user_nocheck(x,ptr,size) \ #define __put_user_nocheck(x,ptr,size) \
({ \ ({ \
long __pu_err = 0; \ long __pu_err = 0; \
CHECK_UPTR(ptr); \
switch (size) { \ switch (size) { \
case 1: __put_user_8(x,ptr); break; \ case 1: __put_user_8(x,ptr); break; \
case 2: __put_user_16(x,ptr); break; \ case 2: __put_user_16(x,ptr); break; \
...@@ -218,6 +232,7 @@ extern void __put_user_unknown(void); ...@@ -218,6 +232,7 @@ extern void __put_user_unknown(void);
({ \ ({ \
long __pu_err = -EFAULT; \ long __pu_err = -EFAULT; \
__typeof__(*(ptr)) *__pu_addr = (ptr); \ __typeof__(*(ptr)) *__pu_addr = (ptr); \
CHECK_UPTR(ptr); \
if (__access_ok((long)__pu_addr,size,segment)) { \ if (__access_ok((long)__pu_addr,size,segment)) { \
__pu_err = 0; \ __pu_err = 0; \
switch (size) { \ switch (size) { \
...@@ -371,34 +386,42 @@ __copy_tofrom_user_nocheck(void *to, const void *from, long len) ...@@ -371,34 +386,42 @@ __copy_tofrom_user_nocheck(void *to, const void *from, long len)
} }
extern inline long extern inline long
__copy_tofrom_user(void *to, const void *from, long len, const void *validate) __copy_tofrom_user(void *to, const void *from, long len, const void __user *validate)
{ {
if (__access_ok((long)validate, len, get_fs())) if (__access_ok((long)validate, len, get_fs()))
len = __copy_tofrom_user_nocheck(to, from, len); len = __copy_tofrom_user_nocheck(to, from, len);
return len; return len;
} }
#define __copy_to_user(to,from,n) __copy_tofrom_user_nocheck((to),(from),(n)) #define __copy_to_user(to,from,n) \
#define __copy_from_user(to,from,n) __copy_tofrom_user_nocheck((to),(from),(n)) ({ \
CHECK_UPTR(to); \
__copy_tofrom_user_nocheck((void *)(to),(from),(n)); \
})
#define __copy_from_user(to,from,n) \
({ \
CHECK_UPTR(from); \
__copy_tofrom_user_nocheck((to),(void *)(from),(n)); \
})
extern inline long extern inline long
copy_to_user(void *to, const void *from, long n) copy_to_user(void __user *to, const void *from, long n)
{ {
return __copy_tofrom_user(to, from, n, to); return __copy_tofrom_user((void *)to, from, n, to);
} }
extern inline long extern inline long
copy_from_user(void *to, const void *from, long n) copy_from_user(void *to, const void __user *from, long n)
{ {
return __copy_tofrom_user(to, from, n, from); return __copy_tofrom_user(to, (void *)from, n, from);
} }
extern void __do_clear_user(void); extern void __do_clear_user(void);
extern inline long extern inline long
__clear_user(void *to, long len) __clear_user(void __user *to, long len)
{ {
register void * __cl_to __asm__("$6") = to; register void __user * __cl_to __asm__("$6") = to;
register long __cl_len __asm__("$0") = len; register long __cl_len __asm__("$0") = len;
__asm__ __volatile__( __asm__ __volatile__(
__module_call(28, 2, __do_clear_user) __module_call(28, 2, __do_clear_user)
...@@ -410,7 +433,7 @@ __clear_user(void *to, long len) ...@@ -410,7 +433,7 @@ __clear_user(void *to, long len)
} }
extern inline long extern inline long
clear_user(void *to, long len) clear_user(void __user *to, long len)
{ {
if (__access_ok((long)to, len, get_fs())) if (__access_ok((long)to, len, get_fs()))
len = __clear_user(to, len); len = __clear_user(to, len);
...@@ -423,10 +446,10 @@ clear_user(void *to, long len) ...@@ -423,10 +446,10 @@ clear_user(void *to, long len)
/* Returns: -EFAULT if exception before terminator, N if the entire /* Returns: -EFAULT if exception before terminator, N if the entire
buffer filled, else strlen. */ buffer filled, else strlen. */
extern long __strncpy_from_user(char *__to, const char *__from, long __to_len); extern long __strncpy_from_user(char *__to, const char __user *__from, long __to_len);
extern inline long extern inline long
strncpy_from_user(char *to, const char *from, long n) strncpy_from_user(char *to, const char __user *from, long n)
{ {
long ret = -EFAULT; long ret = -EFAULT;
if (__access_ok((long)from, 0, get_fs())) if (__access_ok((long)from, 0, get_fs()))
...@@ -435,18 +458,18 @@ strncpy_from_user(char *to, const char *from, long n) ...@@ -435,18 +458,18 @@ strncpy_from_user(char *to, const char *from, long n)
} }
/* Returns: 0 if bad, string length+1 (memory size) of string if ok */ /* Returns: 0 if bad, string length+1 (memory size) of string if ok */
extern long __strlen_user(const char *); extern long __strlen_user(const char __user *);
extern inline long strlen_user(const char *str) extern inline long strlen_user(const char __user *str)
{ {
return access_ok(VERIFY_READ,str,0) ? __strlen_user(str) : 0; return access_ok(VERIFY_READ,str,0) ? __strlen_user(str) : 0;
} }
/* Returns: 0 if exception before NUL or reaching the supplied limit (N), /* Returns: 0 if exception before NUL or reaching the supplied limit (N),
* a value greater than N if the limit would be exceeded, else strlen. */ * a value greater than N if the limit would be exceeded, else strlen. */
extern long __strnlen_user(const char *, long); extern long __strnlen_user(const char __user *, long);
extern inline long strnlen_user(const char *str, long n) extern inline long strnlen_user(const char __user *str, long n)
{ {
return access_ok(VERIFY_READ,str,0) ? __strnlen_user(str, n) : 0; return access_ok(VERIFY_READ,str,0) ? __strnlen_user(str, n) : 0;
} }
......
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