mirror of
git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-08-05 16:54:27 +00:00
riscv: uaccess: use 'asm_goto_output' for get_user()
With 'asm goto' we don't need to test the error etc, the exception just jumps to the error handling directly. Unlike put_user(), get_user() must work around GCC bugs [1] when using output clobbers in an asm goto statement. Link: https://gcc.gnu.org/bugzilla/show_bug.cgi?id=113921 # 1 Signed-off-by: Jisheng Zhang <jszhang@kernel.org> [Cyril Bur: Rewritten commit message] Signed-off-by: Cyril Bur <cyrilbur@tenstorrent.com> Reviewed-by: Alexandre Ghiti <alexghiti@rivosinc.com> Link: https://lore.kernel.org/r/20250410070526.3160847-6-cyrilbur@tenstorrent.com Signed-off-by: Palmer Dabbelt <palmer@rivosinc.com>
This commit is contained in:
parent
cdf647e817
commit
f6bff7827a
1 changed files with 68 additions and 27 deletions
|
@ -96,27 +96,58 @@ static inline unsigned long __untagged_addr_remote(struct mm_struct *mm, unsigne
|
||||||
* call.
|
* call.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#define __get_user_asm(insn, x, ptr, err) \
|
#ifdef CONFIG_CC_HAS_ASM_GOTO_OUTPUT
|
||||||
|
#define __get_user_asm(insn, x, ptr, label) \
|
||||||
|
asm_goto_output( \
|
||||||
|
"1:\n" \
|
||||||
|
" " insn " %0, %1\n" \
|
||||||
|
_ASM_EXTABLE_UACCESS_ERR(1b, %l2, %0) \
|
||||||
|
: "=&r" (x) \
|
||||||
|
: "m" (*(ptr)) : : label)
|
||||||
|
#else /* !CONFIG_CC_HAS_ASM_GOTO_OUTPUT */
|
||||||
|
#define __get_user_asm(insn, x, ptr, label) \
|
||||||
do { \
|
do { \
|
||||||
__typeof__(x) __x; \
|
long __gua_err = 0; \
|
||||||
__asm__ __volatile__ ( \
|
__asm__ __volatile__ ( \
|
||||||
"1:\n" \
|
"1:\n" \
|
||||||
" " insn " %1, %2\n" \
|
" " insn " %1, %2\n" \
|
||||||
"2:\n" \
|
"2:\n" \
|
||||||
_ASM_EXTABLE_UACCESS_ERR_ZERO(1b, 2b, %0, %1) \
|
_ASM_EXTABLE_UACCESS_ERR_ZERO(1b, 2b, %0, %1) \
|
||||||
: "+r" (err), "=&r" (__x) \
|
: "+r" (__gua_err), "=&r" (x) \
|
||||||
: "m" (*(ptr))); \
|
: "m" (*(ptr))); \
|
||||||
(x) = __x; \
|
if (__gua_err) \
|
||||||
|
goto label; \
|
||||||
} while (0)
|
} while (0)
|
||||||
|
#endif /* CONFIG_CC_HAS_ASM_GOTO_OUTPUT */
|
||||||
|
|
||||||
#ifdef CONFIG_64BIT
|
#ifdef CONFIG_64BIT
|
||||||
#define __get_user_8(x, ptr, err) \
|
#define __get_user_8(x, ptr, label) \
|
||||||
__get_user_asm("ld", x, ptr, err)
|
__get_user_asm("ld", x, ptr, label)
|
||||||
#else /* !CONFIG_64BIT */
|
#else /* !CONFIG_64BIT */
|
||||||
#define __get_user_8(x, ptr, err) \
|
|
||||||
|
#ifdef CONFIG_CC_HAS_ASM_GOTO_OUTPUT
|
||||||
|
#define __get_user_8(x, ptr, label) \
|
||||||
|
u32 __user *__ptr = (u32 __user *)(ptr); \
|
||||||
|
u32 __lo, __hi; \
|
||||||
|
asm_goto_output( \
|
||||||
|
"1:\n" \
|
||||||
|
" lw %0, %2\n" \
|
||||||
|
"2:\n" \
|
||||||
|
" lw %1, %3\n" \
|
||||||
|
_ASM_EXTABLE_UACCESS_ERR(1b, %l4, %0) \
|
||||||
|
_ASM_EXTABLE_UACCESS_ERR(2b, %l4, %0) \
|
||||||
|
: "=&r" (__lo), "=r" (__hi) \
|
||||||
|
: "m" (__ptr[__LSW]), "m" (__ptr[__MSW]) \
|
||||||
|
: : label); \
|
||||||
|
(x) = (__typeof__(x))((__typeof__((x) - (x)))( \
|
||||||
|
(((u64)__hi << 32) | __lo))); \
|
||||||
|
|
||||||
|
#else /* !CONFIG_CC_HAS_ASM_GOTO_OUTPUT */
|
||||||
|
#define __get_user_8(x, ptr, label) \
|
||||||
do { \
|
do { \
|
||||||
u32 __user *__ptr = (u32 __user *)(ptr); \
|
u32 __user *__ptr = (u32 __user *)(ptr); \
|
||||||
u32 __lo, __hi; \
|
u32 __lo, __hi; \
|
||||||
|
long __gu8_err = 0; \
|
||||||
__asm__ __volatile__ ( \
|
__asm__ __volatile__ ( \
|
||||||
"1:\n" \
|
"1:\n" \
|
||||||
" lw %1, %3\n" \
|
" lw %1, %3\n" \
|
||||||
|
@ -125,35 +156,51 @@ do { \
|
||||||
"3:\n" \
|
"3:\n" \
|
||||||
_ASM_EXTABLE_UACCESS_ERR_ZERO(1b, 3b, %0, %1) \
|
_ASM_EXTABLE_UACCESS_ERR_ZERO(1b, 3b, %0, %1) \
|
||||||
_ASM_EXTABLE_UACCESS_ERR_ZERO(2b, 3b, %0, %1) \
|
_ASM_EXTABLE_UACCESS_ERR_ZERO(2b, 3b, %0, %1) \
|
||||||
: "+r" (err), "=&r" (__lo), "=r" (__hi) \
|
: "+r" (__gu8_err), "=&r" (__lo), "=r" (__hi) \
|
||||||
: "m" (__ptr[__LSW]), "m" (__ptr[__MSW])); \
|
: "m" (__ptr[__LSW]), "m" (__ptr[__MSW])); \
|
||||||
if (err) \
|
if (__gu8_err) { \
|
||||||
__hi = 0; \
|
__hi = 0; \
|
||||||
(x) = (__typeof__(x))((__typeof__((x)-(x)))( \
|
goto label; \
|
||||||
|
} \
|
||||||
|
(x) = (__typeof__(x))((__typeof__((x) - (x)))( \
|
||||||
(((u64)__hi << 32) | __lo))); \
|
(((u64)__hi << 32) | __lo))); \
|
||||||
} while (0)
|
} while (0)
|
||||||
|
#endif /* CONFIG_CC_HAS_ASM_GOTO_OUTPUT */
|
||||||
|
|
||||||
#endif /* CONFIG_64BIT */
|
#endif /* CONFIG_64BIT */
|
||||||
|
|
||||||
#define __get_user_nocheck(x, __gu_ptr, __gu_err) \
|
#define __get_user_nocheck(x, __gu_ptr, label) \
|
||||||
do { \
|
do { \
|
||||||
switch (sizeof(*__gu_ptr)) { \
|
switch (sizeof(*__gu_ptr)) { \
|
||||||
case 1: \
|
case 1: \
|
||||||
__get_user_asm("lb", (x), __gu_ptr, __gu_err); \
|
__get_user_asm("lb", (x), __gu_ptr, label); \
|
||||||
break; \
|
break; \
|
||||||
case 2: \
|
case 2: \
|
||||||
__get_user_asm("lh", (x), __gu_ptr, __gu_err); \
|
__get_user_asm("lh", (x), __gu_ptr, label); \
|
||||||
break; \
|
break; \
|
||||||
case 4: \
|
case 4: \
|
||||||
__get_user_asm("lw", (x), __gu_ptr, __gu_err); \
|
__get_user_asm("lw", (x), __gu_ptr, label); \
|
||||||
break; \
|
break; \
|
||||||
case 8: \
|
case 8: \
|
||||||
__get_user_8((x), __gu_ptr, __gu_err); \
|
__get_user_8((x), __gu_ptr, label); \
|
||||||
break; \
|
break; \
|
||||||
default: \
|
default: \
|
||||||
BUILD_BUG(); \
|
BUILD_BUG(); \
|
||||||
} \
|
} \
|
||||||
} while (0)
|
} while (0)
|
||||||
|
|
||||||
|
#define __get_user_error(x, ptr, err) \
|
||||||
|
do { \
|
||||||
|
__label__ __gu_failed; \
|
||||||
|
\
|
||||||
|
__get_user_nocheck(x, ptr, __gu_failed); \
|
||||||
|
err = 0; \
|
||||||
|
break; \
|
||||||
|
__gu_failed: \
|
||||||
|
x = 0; \
|
||||||
|
err = -EFAULT; \
|
||||||
|
} while (0)
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* __get_user: - Get a simple variable from user space, with less checking.
|
* __get_user: - Get a simple variable from user space, with less checking.
|
||||||
* @x: Variable to store result.
|
* @x: Variable to store result.
|
||||||
|
@ -178,13 +225,16 @@ do { \
|
||||||
({ \
|
({ \
|
||||||
const __typeof__(*(ptr)) __user *__gu_ptr = untagged_addr(ptr); \
|
const __typeof__(*(ptr)) __user *__gu_ptr = untagged_addr(ptr); \
|
||||||
long __gu_err = 0; \
|
long __gu_err = 0; \
|
||||||
|
__typeof__(x) __gu_val; \
|
||||||
\
|
\
|
||||||
__chk_user_ptr(__gu_ptr); \
|
__chk_user_ptr(__gu_ptr); \
|
||||||
\
|
\
|
||||||
__enable_user_access(); \
|
__enable_user_access(); \
|
||||||
__get_user_nocheck(x, __gu_ptr, __gu_err); \
|
__get_user_error(__gu_val, __gu_ptr, __gu_err); \
|
||||||
__disable_user_access(); \
|
__disable_user_access(); \
|
||||||
\
|
\
|
||||||
|
(x) = __gu_val; \
|
||||||
|
\
|
||||||
__gu_err; \
|
__gu_err; \
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -369,13 +419,7 @@ unsigned long __must_check clear_user(void __user *to, unsigned long n)
|
||||||
}
|
}
|
||||||
|
|
||||||
#define __get_kernel_nofault(dst, src, type, err_label) \
|
#define __get_kernel_nofault(dst, src, type, err_label) \
|
||||||
do { \
|
__get_user_nocheck(*((type *)(dst)), (type *)(src), err_label)
|
||||||
long __kr_err = 0; \
|
|
||||||
\
|
|
||||||
__get_user_nocheck(*((type *)(dst)), (type *)(src), __kr_err); \
|
|
||||||
if (unlikely(__kr_err)) \
|
|
||||||
goto err_label; \
|
|
||||||
} while (0)
|
|
||||||
|
|
||||||
#define __put_kernel_nofault(dst, src, type, err_label) \
|
#define __put_kernel_nofault(dst, src, type, err_label) \
|
||||||
__put_user_nocheck(*((type *)(src)), (type *)(dst), err_label)
|
__put_user_nocheck(*((type *)(src)), (type *)(dst), err_label)
|
||||||
|
@ -401,12 +445,9 @@ static inline void user_access_restore(unsigned long enabled) { }
|
||||||
__put_user_nocheck(x, (ptr), label)
|
__put_user_nocheck(x, (ptr), label)
|
||||||
|
|
||||||
#define unsafe_get_user(x, ptr, label) do { \
|
#define unsafe_get_user(x, ptr, label) do { \
|
||||||
long __err = 0; \
|
|
||||||
__inttype(*(ptr)) __gu_val; \
|
__inttype(*(ptr)) __gu_val; \
|
||||||
__get_user_nocheck(__gu_val, (ptr), __err); \
|
__get_user_nocheck(__gu_val, (ptr), label); \
|
||||||
(x) = (__force __typeof__(*(ptr)))__gu_val; \
|
(x) = (__force __typeof__(*(ptr)))__gu_val; \
|
||||||
if (__err) \
|
|
||||||
goto label; \
|
|
||||||
} while (0)
|
} while (0)
|
||||||
|
|
||||||
#define unsafe_copy_loop(dst, src, len, type, op, label) \
|
#define unsafe_copy_loop(dst, src, len, type, op, label) \
|
||||||
|
|
Loading…
Add table
Reference in a new issue