vserver 1.9.5.x5
[linux-2.6.git] / include / asm-mips / uaccess.h
index adbb24b..95c067a 100644 (file)
@@ -10,7 +10,7 @@
 #define _ASM_UACCESS_H
 
 #include <linux/config.h>
-#include <linux/compiler.h>
+#include <linux/kernel.h>
 #include <linux/errno.h>
 #include <linux/thread_info.h>
 #include <asm-generic/uaccess.h>
@@ -150,7 +150,7 @@ static inline int verify_area(int type, const void * addr, unsigned long size)
  * Returns zero on success, or -EFAULT on error.
  */
 #define put_user(x,ptr)        \
-       __put_user_check((__typeof__(*(ptr)))(x),(ptr),sizeof(*(ptr)))
+       __put_user_check((x),(ptr),sizeof(*(ptr)))
 
 /*
  * get_user: - Get a simple variable from user space.
@@ -170,7 +170,7 @@ static inline int verify_area(int type, const void * addr, unsigned long size)
  * On error, the variable @x is set to zero.
  */
 #define get_user(x,ptr) \
-       __get_user_check((__typeof__(*(ptr)))(x),(ptr),sizeof(*(ptr)))
+       __get_user_check((x),(ptr),sizeof(*(ptr)))
 
 /*
  * __put_user: - Write a simple value into user space, with less checking.
@@ -192,7 +192,7 @@ static inline int verify_area(int type, const void * addr, unsigned long size)
  * Returns zero on success, or -EFAULT on error.
  */
 #define __put_user(x,ptr) \
-       __put_user_nocheck((__typeof__(*(ptr)))(x),(ptr),sizeof(*(ptr)))
+       __put_user_nocheck((x),(ptr),sizeof(*(ptr)))
 
 /*
  * __get_user: - Get a simple variable from user space, with less checking.
@@ -215,7 +215,7 @@ static inline int verify_area(int type, const void * addr, unsigned long size)
  * On error, the variable @x is set to zero.
  */
 #define __get_user(x,ptr) \
-       __get_user_nocheck((__typeof__(*(ptr)))(x),(ptr),sizeof(*(ptr)))
+       __get_user_nocheck((x),(ptr),sizeof(*(ptr)))
 
 struct __large_struct { unsigned long buf[100]; };
 #define __m(x) (*(struct __large_struct *)(x))
@@ -225,89 +225,91 @@ struct __large_struct { unsigned long buf[100]; };
  * for 32 bit mode and old iron.
  */
 #ifdef __mips64
-#define __GET_USER_DW __get_user_asm("ld")
+#define __GET_USER_DW(__gu_err) __get_user_asm("ld", __gu_err)
 #else
-#define __GET_USER_DW __get_user_asm_ll32
+#define __GET_USER_DW(__gu_err) __get_user_asm_ll32(__gu_err)
 #endif
 
-#define __get_user_nocheck(x,ptr,size)                         \
-({                                                             \
-       long __gu_err;                                          \
-       __typeof(*(ptr)) __gu_val;                              \
-       long __gu_addr;                                         \
-       might_sleep();                                          \
-       __asm__("":"=r" (__gu_val));                            \
-       __gu_addr = (long) (ptr);                               \
-       __asm__("":"=r" (__gu_err));                            \
-       switch (size) {                                         \
-       case 1: __get_user_asm("lb"); break;                    \
-       case 2: __get_user_asm("lh"); break;                    \
-       case 4: __get_user_asm("lw"); break;                    \
-       case 8: __GET_USER_DW; break;                           \
-       default: __get_user_unknown(); break;                   \
-       } x = (__typeof__(*(ptr))) __gu_val; __gu_err;          \
+#define __get_user_nocheck(x,ptr,size)                                 \
+({                                                                     \
+       __typeof(*(ptr)) __gu_val = 0;                                  \
+       long __gu_addr;                                                 \
+       long __gu_err = 0;                                              \
+                                                                       \
+       might_sleep();                                                  \
+       __gu_addr = (long) (ptr);                                       \
+       switch (size) {                                                 \
+       case 1: __get_user_asm("lb", __gu_err); break;                  \
+       case 2: __get_user_asm("lh", __gu_err); break;                  \
+       case 4: __get_user_asm("lw", __gu_err); break;                  \
+       case 8: __GET_USER_DW(__gu_err); break;                         \
+       default: __get_user_unknown(); break;                           \
+       }                                                               \
+       x = (__typeof__(*(ptr))) __gu_val;                              \
+       __gu_err;                                                       \
 })
 
 #define __get_user_check(x,ptr,size)                                   \
 ({                                                                     \
-       long __gu_err;                                                  \
-       __typeof__(*(ptr)) __gu_val;                                    \
+       __typeof__(*(ptr)) __gu_val = 0;                                \
        long __gu_addr;                                                 \
+       long __gu_err;                                                  \
+                                                                       \
        might_sleep();                                                  \
-       __asm__("":"=r" (__gu_val));                                    \
        __gu_addr = (long) (ptr);                                       \
-       __asm__("":"=r" (__gu_err));                                    \
-       if (access_ok(VERIFY_READ,__gu_addr,size)) {                    \
+       __gu_err = verify_area(VERIFY_READ, (void *) __gu_addr, size);  \
+                                                                       \
+       if (likely(!__gu_err)) {                                        \
                switch (size) {                                         \
-               case 1: __get_user_asm("lb"); break;                    \
-               case 2: __get_user_asm("lh"); break;                    \
-               case 4: __get_user_asm("lw"); break;                    \
-               case 8: __GET_USER_DW; break;                           \
+               case 1: __get_user_asm("lb", __gu_err); break;          \
+               case 2: __get_user_asm("lh", __gu_err); break;          \
+               case 4: __get_user_asm("lw", __gu_err); break;          \
+               case 8: __GET_USER_DW(__gu_err); break;                 \
                default: __get_user_unknown(); break;                   \
                }                                                       \
-       } x = (__typeof__(*(ptr))) __gu_val; __gu_err;                  \
+       }                                                               \
+       x = (__typeof__(*(ptr))) __gu_val;                              \
+       __gu_err;                                                       \
 })
 
-#define __get_user_asm(insn)                                           \
+#define __get_user_asm(insn,__gu_err)                                  \
 ({                                                                     \
        __asm__ __volatile__(                                           \
-       "1:\t" insn "\t%1,%2\n\t"                                       \
-       "move\t%0,$0\n"                                                 \
-       "2:\n\t"                                                        \
-       ".section\t.fixup,\"ax\"\n"                                     \
-       "3:\tli\t%0,%3\n\t"                                             \
-       "move\t%1,$0\n\t"                                               \
-       "j\t2b\n\t"                                                     \
-       ".previous\n\t"                                                 \
-       ".section\t__ex_table,\"a\"\n\t"                                \
-       __UA_ADDR "\t1b,3b\n\t"                                         \
-       ".previous"                                                     \
-       :"=r" (__gu_err), "=r" (__gu_val)                               \
-       :"o" (__m(__gu_addr)), "i" (-EFAULT));                          \
+       "1:     " insn "        %1, %3                          \n"     \
+       "2:                                                     \n"     \
+       "       .section .fixup,\"ax\"                          \n"     \
+       "3:     li      %0, %4                                  \n"     \
+       "       j       2b                                      \n"     \
+       "       .previous                                       \n"     \
+       "       .section __ex_table,\"a\"                       \n"     \
+       "       "__UA_ADDR "\t1b, 3b                            \n"     \
+       "       .previous                                       \n"     \
+       : "=r" (__gu_err), "=r" (__gu_val)                              \
+       : "0" (__gu_err), "o" (__m(__gu_addr)), "i" (-EFAULT));         \
 })
 
 /*
  * Get a long long 64 using 32 bit registers.
  */
-#define __get_user_asm_ll32                                            \
+#define __get_user_asm_ll32(__gu_err)                                  \
 ({                                                                     \
        __asm__ __volatile__(                                           \
-       "1:\tlw\t%1,%2\n"                                               \
-       "2:\tlw\t%D1,%3\n\t"                                            \
-       "move\t%0,$0\n"                                                 \
-       "3:\t.section\t.fixup,\"ax\"\n"                                 \
-       "4:\tli\t%0,%4\n\t"                                             \
-       "move\t%1,$0\n\t"                                               \
-       "move\t%D1,$0\n\t"                                              \
-       "j\t3b\n\t"                                                     \
-       ".previous\n\t"                                                 \
-       ".section\t__ex_table,\"a\"\n\t"                                \
-       __UA_ADDR "\t1b,4b\n\t"                                         \
-       __UA_ADDR "\t2b,4b\n\t"                                         \
-       ".previous"                                                     \
-       :"=r" (__gu_err), "=&r" (__gu_val)                              \
-       :"o" (__m(__gu_addr)), "o" (__m(__gu_addr + 4)),                \
-        "i" (-EFAULT));                                                \
+       "1:     lw      %1, %3                                  \n"     \
+       "2:     lw      %D1, %4                                 \n"     \
+       "       move    %0, $0                                  \n"     \
+       "3:     .section        .fixup,\"ax\"                   \n"     \
+       "4:     li      %0, %5                                  \n"     \
+       "       move    %1, $0                                  \n"     \
+       "       move    %D1, $0                                 \n"     \
+       "       j       3b                                      \n"     \
+       "       .previous                                       \n"     \
+       "       .section        __ex_table,\"a\"                \n"     \
+       "       " __UA_ADDR "   1b, 4b                          \n"     \
+       "       " __UA_ADDR "   2b, 4b                          \n"     \
+       "       .previous                                       \n"     \
+       : "=r" (__gu_err), "=&r" (__gu_val)                             \
+       : "0" (__gu_err), "o" (__m(__gu_addr)),                         \
+         "o" (__m(__gu_addr + 4)), "i" (-EFAULT));                     \
 })
 
 extern void __get_user_unknown(void);
@@ -317,25 +319,25 @@ extern void __get_user_unknown(void);
  * for 32 bit mode and old iron.
  */
 #ifdef __mips64
-#define __PUT_USER_DW __put_user_asm("sd")
+#define __PUT_USER_DW(__pu_val) __put_user_asm("sd", __pu_val)
 #else
-#define __PUT_USER_DW __put_user_asm_ll32
+#define __PUT_USER_DW(__pu_val) __put_user_asm_ll32(__pu_val)
 #endif
 
 #define __put_user_nocheck(x,ptr,size)                                 \
 ({                                                                     \
-       long __pu_err;                                                  \
        __typeof__(*(ptr)) __pu_val;                                    \
        long __pu_addr;                                                 \
+       long __pu_err = 0;                                              \
+                                                                       \
        might_sleep();                                                  \
        __pu_val = (x);                                                 \
        __pu_addr = (long) (ptr);                                       \
-       __asm__("":"=r" (__pu_err));                                    \
        switch (size) {                                                 \
-       case 1: __put_user_asm("sb"); break;                            \
-       case 2: __put_user_asm("sh"); break;                            \
-       case 4: __put_user_asm("sw"); break;                            \
-       case 8: __PUT_USER_DW; break;                                   \
+       case 1: __put_user_asm("sb", __pu_val); break;                  \
+       case 2: __put_user_asm("sh", __pu_val); break;                  \
+       case 4: __put_user_asm("sw", __pu_val); break;                  \
+       case 8: __PUT_USER_DW(__pu_val); break;                         \
        default: __put_user_unknown(); break;                           \
        }                                                               \
        __pu_err;                                                       \
@@ -343,60 +345,61 @@ extern void __get_user_unknown(void);
 
 #define __put_user_check(x,ptr,size)                                   \
 ({                                                                     \
-       long __pu_err;                                                  \
        __typeof__(*(ptr)) __pu_val;                                    \
        long __pu_addr;                                                 \
+       long __pu_err;                                                  \
+                                                                       \
        might_sleep();                                                  \
        __pu_val = (x);                                                 \
        __pu_addr = (long) (ptr);                                       \
-       __asm__("":"=r" (__pu_err));                                    \
-       if (access_ok(VERIFY_WRITE, __pu_addr, size)) {                 \
+       __pu_err = verify_area(VERIFY_WRITE, (void *) __pu_addr, size); \
+                                                                       \
+       if (likely(!__pu_err)) {                                        \
                switch (size) {                                         \
-               case 1: __put_user_asm("sb"); break;                    \
-               case 2: __put_user_asm("sh"); break;                    \
-               case 4: __put_user_asm("sw"); break;                    \
-               case 8: __PUT_USER_DW; break;                           \
+               case 1: __put_user_asm("sb", __pu_val); break;          \
+               case 2: __put_user_asm("sh", __pu_val); break;          \
+               case 4: __put_user_asm("sw", __pu_val); break;          \
+               case 8: __PUT_USER_DW(__pu_val); break;                 \
                default: __put_user_unknown(); break;                   \
                }                                                       \
        }                                                               \
        __pu_err;                                                       \
 })
 
-#define __put_user_asm(insn)                                           \
+#define __put_user_asm(insn, __pu_val)                                 \
 ({                                                                     \
        __asm__ __volatile__(                                           \
-       "1:\t" insn "\t%z1, %2\t\t\t# __put_user_asm\n\t"               \
-       "move\t%0, $0\n"                                                \
-       "2:\n\t"                                                        \
-       ".section\t.fixup,\"ax\"\n"                                     \
-       "3:\tli\t%0,%3\n\t"                                             \
-       "j\t2b\n\t"                                                     \
-       ".previous\n\t"                                                 \
-       ".section\t__ex_table,\"a\"\n\t"                                \
-       __UA_ADDR "\t1b,3b\n\t"                                         \
-       ".previous"                                                     \
-       :"=r" (__pu_err)                                                \
-       :"Jr" (__pu_val), "o" (__m(__pu_addr)), "i" (-EFAULT));         \
+       "1:     " insn "        %z2, %3         # __put_user_asm\n"     \
+       "2:                                                     \n"     \
+       "       .section        .fixup,\"ax\"                   \n"     \
+       "3:     li      %0, %4                                  \n"     \
+       "       j       2b                                      \n"     \
+       "       .previous                                       \n"     \
+       "       .section        __ex_table,\"a\"                \n"     \
+       "       " __UA_ADDR "   1b, 3b                          \n"     \
+       "       .previous                                       \n"     \
+       : "=r" (__pu_err)                                               \
+       : "0" (__pu_err), "Jr" (__pu_val), "o" (__m(__pu_addr)),        \
+         "i" (-EFAULT));                                               \
 })
 
-#define __put_user_asm_ll32                                            \
+#define __put_user_asm_ll32(__pu_val)                                  \
 ({                                                                     \
        __asm__ __volatile__(                                           \
-       "1:\tsw\t%1, %2\t\t\t# __put_user_asm_ll32\n\t"                 \
-       "2:\tsw\t%D1, %3\n"                                             \
-       "move\t%0, $0\n"                                                \
-       "3:\n\t"                                                        \
-       ".section\t.fixup,\"ax\"\n"                                     \
-       "4:\tli\t%0,%4\n\t"                                             \
-       "j\t3b\n\t"                                                     \
-       ".previous\n\t"                                                 \
-       ".section\t__ex_table,\"a\"\n\t"                                \
-       __UA_ADDR "\t1b,4b\n\t"                                         \
-       __UA_ADDR "\t2b,4b\n\t"                                         \
-       ".previous"                                                     \
-       :"=r" (__pu_err)                                                \
-       :"r" (__pu_val), "o" (__m(__pu_addr)),                          \
-        "o" (__m(__pu_addr + 4)), "i" (-EFAULT));                      \
+       "1:     sw      %2, %3          # __put_user_asm_ll32   \n"     \
+       "2:     sw      %D2, %4                                 \n"     \
+       "3:                                                     \n"     \
+       "       .section        .fixup,\"ax\"                   \n"     \
+       "4:     li      %0, %5                                  \n"     \
+       "       j       3b                                      \n"     \
+       "       .previous                                       \n"     \
+       "       .section        __ex_table,\"a\"                \n"     \
+       "       " __UA_ADDR "   1b, 4b                          \n"     \
+       "       " __UA_ADDR "   2b, 4b                          \n"     \
+       "       .previous"                                              \
+       : "=r" (__pu_err)                                               \
+       : "0" (__pu_err), "r" (__pu_val), "o" (__m(__pu_addr)),         \
+         "o" (__m(__pu_addr + 4)), "i" (-EFAULT));                     \
 })
 
 extern void __put_user_unknown(void);
@@ -511,8 +514,7 @@ extern size_t __copy_user(void *__to, const void *__from, size_t __n);
        ".set\tnoat\n\t"                                                \
        __UA_ADDU "\t$1, %1, %2\n\t"                                    \
        ".set\tat\n\t"                                                  \
-       ".set\treorder\n\t"                                             \
-       "move\t%0, $6"          /* XXX */                               \
+       ".set\treorder"                                                 \
        : "+r" (__cu_to_r), "+r" (__cu_from_r), "+r" (__cu_len_r)       \
        :                                                               \
        : "$8", "$9", "$10", "$11", "$12", "$15", "$24", "$31",         \