|
@@ -196,3 +196,61 @@ asm volatile ("\n" \
|
|
|
break; \
|
|
|
} \
|
|
|
__gu_err; \
|
|
|
+})
|
|
|
+#define get_user(x, ptr) __get_user(x, ptr)
|
|
|
+
|
|
|
+unsigned long __generic_copy_from_user(void *to, const void __user *from, unsigned long n);
|
|
|
+unsigned long __generic_copy_to_user(void __user *to, const void *from, unsigned long n);
|
|
|
+
|
|
|
+#define __constant_copy_from_user_asm(res, to, from, tmp, n, s1, s2, s3)\
|
|
|
+ asm volatile ("\n" \
|
|
|
+ "1: "MOVES"."#s1" (%2)+,%3\n" \
|
|
|
+ " move."#s1" %3,(%1)+\n" \
|
|
|
+ "2: "MOVES"."#s2" (%2)+,%3\n" \
|
|
|
+ " move."#s2" %3,(%1)+\n" \
|
|
|
+ " .ifnc \""#s3"\",\"\"\n" \
|
|
|
+ "3: "MOVES"."#s3" (%2)+,%3\n" \
|
|
|
+ " move."#s3" %3,(%1)+\n" \
|
|
|
+ " .endif\n" \
|
|
|
+ "4:\n" \
|
|
|
+ " .section __ex_table,\"a\"\n" \
|
|
|
+ " .align 4\n" \
|
|
|
+ " .long 1b,10f\n" \
|
|
|
+ " .long 2b,20f\n" \
|
|
|
+ " .ifnc \""#s3"\",\"\"\n" \
|
|
|
+ " .long 3b,30f\n" \
|
|
|
+ " .endif\n" \
|
|
|
+ " .previous\n" \
|
|
|
+ "\n" \
|
|
|
+ " .section .fixup,\"ax\"\n" \
|
|
|
+ " .even\n" \
|
|
|
+ "10: clr."#s1" (%1)+\n" \
|
|
|
+ "20: clr."#s2" (%1)+\n" \
|
|
|
+ " .ifnc \""#s3"\",\"\"\n" \
|
|
|
+ "30: clr."#s3" (%1)+\n" \
|
|
|
+ " .endif\n" \
|
|
|
+ " moveq.l #"#n",%0\n" \
|
|
|
+ " jra 4b\n" \
|
|
|
+ " .previous\n" \
|
|
|
+ : "+d" (res), "+&a" (to), "+a" (from), "=&d" (tmp) \
|
|
|
+ : : "memory")
|
|
|
+
|
|
|
+static __always_inline unsigned long
|
|
|
+__constant_copy_from_user(void *to, const void __user *from, unsigned long n)
|
|
|
+{
|
|
|
+ unsigned long res = 0, tmp;
|
|
|
+
|
|
|
+ switch (n) {
|
|
|
+ case 1:
|
|
|
+ __get_user_asm(res, *(u8 *)to, (u8 __user *)from, u8, b, d, 1);
|
|
|
+ break;
|
|
|
+ case 2:
|
|
|
+ __get_user_asm(res, *(u16 *)to, (u16 __user *)from, u16, w, d, 2);
|
|
|
+ break;
|
|
|
+ case 3:
|
|
|
+ __constant_copy_from_user_asm(res, to, from, tmp, 3, w, b,);
|
|
|
+ break;
|
|
|
+ case 4:
|
|
|
+ __get_user_asm(res, *(u32 *)to, (u32 __user *)from, u32, l, r, 4);
|
|
|
+ break;
|
|
|
+ case 5:
|