2012-03-05 19:49:32 +08:00
|
|
|
/*
|
|
|
|
* Based on arch/arm/include/asm/uaccess.h
|
|
|
|
*
|
|
|
|
* Copyright (C) 2012 ARM Ltd.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License version 2 as
|
|
|
|
* published by the Free Software Foundation.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
#ifndef __ASM_UACCESS_H
|
|
|
|
#define __ASM_UACCESS_H
|
|
|
|
|
2016-07-01 21:58:21 +08:00
|
|
|
#include <asm/alternative.h>
|
2016-07-01 23:53:00 +08:00
|
|
|
#include <asm/kernel-pgtable.h>
|
2016-07-01 21:58:21 +08:00
|
|
|
#include <asm/sysreg.h>
|
|
|
|
|
2012-03-05 19:49:32 +08:00
|
|
|
/*
|
|
|
|
* User space memory access functions
|
|
|
|
*/
|
2016-10-19 21:40:54 +08:00
|
|
|
#include <linux/bitops.h>
|
2016-06-09 05:40:56 +08:00
|
|
|
#include <linux/kasan-checks.h>
|
2012-03-05 19:49:32 +08:00
|
|
|
#include <linux/string.h>
|
|
|
|
|
2015-07-23 02:05:54 +08:00
|
|
|
#include <asm/cpufeature.h>
|
2012-03-05 19:49:32 +08:00
|
|
|
#include <asm/ptrace.h>
|
|
|
|
#include <asm/memory.h>
|
|
|
|
#include <asm/compiler.h>
|
2016-12-26 03:00:03 +08:00
|
|
|
#include <asm/extable.h>
|
2012-03-05 19:49:32 +08:00
|
|
|
|
|
|
|
#define KERNEL_DS (-1UL)
|
|
|
|
#define get_ds() (KERNEL_DS)
|
|
|
|
|
|
|
|
#define USER_DS TASK_SIZE_64
|
|
|
|
#define get_fs() (current_thread_info()->addr_limit)
|
|
|
|
|
|
|
|
static inline void set_fs(mm_segment_t fs)
|
|
|
|
{
|
|
|
|
current_thread_info()->addr_limit = fs;
|
2016-02-05 22:58:48 +08:00
|
|
|
|
2017-06-15 09:12:03 +08:00
|
|
|
/* On user-mode return, check fs is correct */
|
|
|
|
set_thread_flag(TIF_FSCHECK);
|
|
|
|
|
2016-02-05 22:58:48 +08:00
|
|
|
/*
|
|
|
|
* Enable/disable UAO so that copy_to_user() etc can access
|
|
|
|
* kernel memory with the unprivileged instructions.
|
|
|
|
*/
|
|
|
|
if (IS_ENABLED(CONFIG_ARM64_UAO) && fs == KERNEL_DS)
|
|
|
|
asm(ALTERNATIVE("nop", SET_PSTATE_UAO(1), ARM64_HAS_UAO));
|
|
|
|
else
|
|
|
|
asm(ALTERNATIVE("nop", SET_PSTATE_UAO(0), ARM64_HAS_UAO,
|
|
|
|
CONFIG_ARM64_UAO));
|
2012-03-05 19:49:32 +08:00
|
|
|
}
|
|
|
|
|
2015-01-06 21:11:13 +08:00
|
|
|
#define segment_eq(a, b) ((a) == (b))
|
2012-03-05 19:49:32 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Test whether a block of memory is a valid user space address.
|
|
|
|
* Returns 1 if the range is valid, 0 otherwise.
|
|
|
|
*
|
|
|
|
* This is equivalent to the following test:
|
2014-03-20 00:29:37 +08:00
|
|
|
* (u65)addr + (u65)size <= current->addr_limit
|
2012-03-05 19:49:32 +08:00
|
|
|
*
|
|
|
|
* This needs 65-bit arithmetic.
|
|
|
|
*/
|
|
|
|
#define __range_ok(addr, size) \
|
|
|
|
({ \
|
2017-07-03 21:37:46 +08:00
|
|
|
unsigned long __addr = (unsigned long)(addr); \
|
2012-03-05 19:49:32 +08:00
|
|
|
unsigned long flag, roksum; \
|
|
|
|
__chk_user_ptr(addr); \
|
2014-03-20 00:29:37 +08:00
|
|
|
asm("adds %1, %1, %3; ccmp %1, %4, #2, cc; cset %0, ls" \
|
2012-03-05 19:49:32 +08:00
|
|
|
: "=&r" (flag), "=&r" (roksum) \
|
2017-05-03 23:09:35 +08:00
|
|
|
: "1" (__addr), "Ir" (size), \
|
2012-03-05 19:49:32 +08:00
|
|
|
"r" (current_thread_info()->addr_limit) \
|
|
|
|
: "cc"); \
|
|
|
|
flag; \
|
|
|
|
})
|
|
|
|
|
2016-10-19 21:40:54 +08:00
|
|
|
/*
|
2017-05-03 23:37:46 +08:00
|
|
|
* When dealing with data aborts, watchpoints, or instruction traps we may end
|
|
|
|
* up with a tagged userland pointer. Clear the tag to get a sane pointer to
|
|
|
|
* pass on to access_ok(), for instance.
|
2016-10-19 21:40:54 +08:00
|
|
|
*/
|
|
|
|
#define untagged_addr(addr) sign_extend64(addr, 55)
|
|
|
|
|
2012-03-05 19:49:32 +08:00
|
|
|
#define access_ok(type, addr, size) __range_ok(addr, size)
|
2013-11-07 01:20:22 +08:00
|
|
|
#define user_addr_max get_fs
|
2012-03-05 19:49:32 +08:00
|
|
|
|
2016-01-01 22:02:12 +08:00
|
|
|
#define _ASM_EXTABLE(from, to) \
|
|
|
|
" .pushsection __ex_table, \"a\"\n" \
|
|
|
|
" .align 3\n" \
|
|
|
|
" .long (" #from " - .), (" #to " - .)\n" \
|
|
|
|
" .popsection\n"
|
|
|
|
|
2016-07-01 21:58:21 +08:00
|
|
|
/*
|
|
|
|
* User access enabling/disabling.
|
|
|
|
*/
|
2016-07-01 23:53:00 +08:00
|
|
|
#ifdef CONFIG_ARM64_SW_TTBR0_PAN
|
|
|
|
static inline void __uaccess_ttbr0_disable(void)
|
|
|
|
{
|
|
|
|
unsigned long ttbr;
|
|
|
|
|
2017-08-10 20:58:16 +08:00
|
|
|
ttbr = read_sysreg(ttbr1_el1);
|
2016-07-01 23:53:00 +08:00
|
|
|
/* reserved_ttbr0 placed at the end of swapper_pg_dir */
|
2017-08-10 20:58:16 +08:00
|
|
|
write_sysreg(ttbr + SWAPPER_DIR_SIZE, ttbr0_el1);
|
|
|
|
isb();
|
|
|
|
/* Set reserved ASID */
|
2017-12-02 01:33:48 +08:00
|
|
|
ttbr &= ~TTBR_ASID_MASK;
|
2017-08-10 20:58:16 +08:00
|
|
|
write_sysreg(ttbr, ttbr1_el1);
|
2016-07-01 23:53:00 +08:00
|
|
|
isb();
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void __uaccess_ttbr0_enable(void)
|
|
|
|
{
|
2017-08-10 20:58:16 +08:00
|
|
|
unsigned long flags, ttbr0, ttbr1;
|
2016-07-01 23:53:00 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Disable interrupts to avoid preemption between reading the 'ttbr0'
|
|
|
|
* variable and the MSR. A context switch could trigger an ASID
|
|
|
|
* roll-over and an update of 'ttbr0'.
|
|
|
|
*/
|
|
|
|
local_irq_save(flags);
|
2017-08-10 20:58:16 +08:00
|
|
|
ttbr0 = current_thread_info()->ttbr0;
|
|
|
|
|
|
|
|
/* Restore active ASID */
|
|
|
|
ttbr1 = read_sysreg(ttbr1_el1);
|
2017-12-02 01:33:48 +08:00
|
|
|
ttbr1 |= ttbr0 & TTBR_ASID_MASK;
|
2017-08-10 20:58:16 +08:00
|
|
|
write_sysreg(ttbr1, ttbr1_el1);
|
|
|
|
isb();
|
|
|
|
|
|
|
|
/* Restore user page table */
|
|
|
|
write_sysreg(ttbr0, ttbr0_el1);
|
2016-07-01 23:53:00 +08:00
|
|
|
isb();
|
|
|
|
local_irq_restore(flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool uaccess_ttbr0_disable(void)
|
|
|
|
{
|
|
|
|
if (!system_uses_ttbr0_pan())
|
|
|
|
return false;
|
|
|
|
__uaccess_ttbr0_disable();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool uaccess_ttbr0_enable(void)
|
|
|
|
{
|
|
|
|
if (!system_uses_ttbr0_pan())
|
|
|
|
return false;
|
|
|
|
__uaccess_ttbr0_enable();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
static inline bool uaccess_ttbr0_disable(void)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline bool uaccess_ttbr0_enable(void)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-01-08 23:38:11 +08:00
|
|
|
static inline void __uaccess_disable_hw_pan(void)
|
|
|
|
{
|
|
|
|
asm(ALTERNATIVE("nop", SET_PSTATE_PAN(0), ARM64_HAS_PAN,
|
|
|
|
CONFIG_ARM64_PAN));
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void __uaccess_enable_hw_pan(void)
|
|
|
|
{
|
|
|
|
asm(ALTERNATIVE("nop", SET_PSTATE_PAN(1), ARM64_HAS_PAN,
|
|
|
|
CONFIG_ARM64_PAN));
|
|
|
|
}
|
|
|
|
|
2016-07-01 21:58:21 +08:00
|
|
|
#define __uaccess_disable(alt) \
|
|
|
|
do { \
|
2016-07-01 23:53:00 +08:00
|
|
|
if (!uaccess_ttbr0_disable()) \
|
|
|
|
asm(ALTERNATIVE("nop", SET_PSTATE_PAN(1), alt, \
|
|
|
|
CONFIG_ARM64_PAN)); \
|
2016-07-01 21:58:21 +08:00
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define __uaccess_enable(alt) \
|
|
|
|
do { \
|
2016-12-12 21:50:26 +08:00
|
|
|
if (!uaccess_ttbr0_enable()) \
|
2016-07-01 23:53:00 +08:00
|
|
|
asm(ALTERNATIVE("nop", SET_PSTATE_PAN(0), alt, \
|
|
|
|
CONFIG_ARM64_PAN)); \
|
2016-07-01 21:58:21 +08:00
|
|
|
} while (0)
|
|
|
|
|
|
|
|
static inline void uaccess_disable(void)
|
|
|
|
{
|
|
|
|
__uaccess_disable(ARM64_HAS_PAN);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void uaccess_enable(void)
|
|
|
|
{
|
|
|
|
__uaccess_enable(ARM64_HAS_PAN);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* These functions are no-ops when UAO is present.
|
|
|
|
*/
|
|
|
|
static inline void uaccess_disable_not_uao(void)
|
|
|
|
{
|
|
|
|
__uaccess_disable(ARM64_ALT_PAN_NOT_UAO);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void uaccess_enable_not_uao(void)
|
|
|
|
{
|
|
|
|
__uaccess_enable(ARM64_ALT_PAN_NOT_UAO);
|
|
|
|
}
|
|
|
|
|
2012-03-05 19:49:32 +08:00
|
|
|
/*
|
|
|
|
* The "__xxx" versions of the user access functions do not verify the address
|
|
|
|
* space - it must have been done previously with a separate "access_ok()"
|
|
|
|
* call.
|
|
|
|
*
|
|
|
|
* The "__xxx_error" versions set the third argument to -EFAULT if an error
|
|
|
|
* occurs, and leave it unchanged on success.
|
|
|
|
*/
|
2016-02-05 22:58:48 +08:00
|
|
|
#define __get_user_asm(instr, alt_instr, reg, x, addr, err, feature) \
|
2012-03-05 19:49:32 +08:00
|
|
|
asm volatile( \
|
2016-02-05 22:58:48 +08:00
|
|
|
"1:"ALTERNATIVE(instr " " reg "1, [%2]\n", \
|
|
|
|
alt_instr " " reg "1, [%2]\n", feature) \
|
2012-03-05 19:49:32 +08:00
|
|
|
"2:\n" \
|
|
|
|
" .section .fixup, \"ax\"\n" \
|
|
|
|
" .align 2\n" \
|
|
|
|
"3: mov %w0, %3\n" \
|
|
|
|
" mov %1, #0\n" \
|
|
|
|
" b 2b\n" \
|
|
|
|
" .previous\n" \
|
2016-01-01 22:02:12 +08:00
|
|
|
_ASM_EXTABLE(1b, 3b) \
|
2012-03-05 19:49:32 +08:00
|
|
|
: "+r" (err), "=&r" (x) \
|
|
|
|
: "r" (addr), "i" (-EFAULT))
|
|
|
|
|
|
|
|
#define __get_user_err(x, ptr, err) \
|
|
|
|
do { \
|
|
|
|
unsigned long __gu_val; \
|
|
|
|
__chk_user_ptr(ptr); \
|
2016-07-01 21:58:21 +08:00
|
|
|
uaccess_enable_not_uao(); \
|
2012-03-05 19:49:32 +08:00
|
|
|
switch (sizeof(*(ptr))) { \
|
|
|
|
case 1: \
|
2016-02-05 22:58:48 +08:00
|
|
|
__get_user_asm("ldrb", "ldtrb", "%w", __gu_val, (ptr), \
|
|
|
|
(err), ARM64_HAS_UAO); \
|
2012-03-05 19:49:32 +08:00
|
|
|
break; \
|
|
|
|
case 2: \
|
2016-02-05 22:58:48 +08:00
|
|
|
__get_user_asm("ldrh", "ldtrh", "%w", __gu_val, (ptr), \
|
|
|
|
(err), ARM64_HAS_UAO); \
|
2012-03-05 19:49:32 +08:00
|
|
|
break; \
|
|
|
|
case 4: \
|
2016-02-05 22:58:48 +08:00
|
|
|
__get_user_asm("ldr", "ldtr", "%w", __gu_val, (ptr), \
|
|
|
|
(err), ARM64_HAS_UAO); \
|
2012-03-05 19:49:32 +08:00
|
|
|
break; \
|
|
|
|
case 8: \
|
arm64: uaccess: suppress spurious clang warning
Clang tries to warn when there's a mismatch between an operand's size,
and the size of the register it is held in, as this may indicate a bug.
Specifically, clang warns when the operand's type is less than 64 bits
wide, and the register is used unqualified (i.e. %N rather than %xN or
%wN).
Unfortunately clang can generate these warnings for unreachable code.
For example, for code like:
do { \
typeof(*(ptr)) __v = (v); \
switch(sizeof(*(ptr))) { \
case 1: \
// assume __v is 1 byte wide \
asm ("{op}b %w0" : : "r" (v)); \
break; \
case 8: \
// assume __v is 8 bytes wide \
asm ("{op} %0" : : "r" (v)); \
break; \
}
while (0)
... if op() were passed a char value and pointer to char, clang may
produce a warning for the unreachable case where sizeof(*(ptr)) is 8.
For the same reasons, clang produces warnings when __put_user_err() is
used for types that are less than 64 bits wide.
We could avoid this with a cast to a fixed-width type in each of the
cases. However, GCC will then warn that pointer types are being cast to
mismatched integer sizes (in unreachable paths).
Another option would be to use the same union trickery as we do for
__smp_store_release() and __smp_load_acquire(), but this is fairly
invasive.
Instead, this patch suppresses the clang warning by using an x modifier
in the assembly for the 8 byte case of __put_user_err(). No additional
work is necessary as the value has been cast to typeof(*(ptr)), so the
compiler will have performed any necessary extension for the reachable
case.
For consistency, __get_user_err() is also updated to use the x modifier
for its 8 byte case.
Acked-by: Will Deacon <will.deacon@arm.com>
Signed-off-by: Mark Rutland <mark.rutland@arm.com>
Reported-by: Matthias Kaehlcke <mka@chromium.org>
Signed-off-by: Catalin Marinas <catalin.marinas@arm.com>
2017-05-03 23:09:38 +08:00
|
|
|
__get_user_asm("ldr", "ldtr", "%x", __gu_val, (ptr), \
|
2016-02-05 22:58:48 +08:00
|
|
|
(err), ARM64_HAS_UAO); \
|
2012-03-05 19:49:32 +08:00
|
|
|
break; \
|
|
|
|
default: \
|
|
|
|
BUILD_BUG(); \
|
|
|
|
} \
|
2016-07-01 21:58:21 +08:00
|
|
|
uaccess_disable_not_uao(); \
|
2014-12-12 07:56:04 +08:00
|
|
|
(x) = (__force __typeof__(*(ptr)))__gu_val; \
|
2012-03-05 19:49:32 +08:00
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define __get_user(x, ptr) \
|
|
|
|
({ \
|
|
|
|
int __gu_err = 0; \
|
|
|
|
__get_user_err((x), (ptr), __gu_err); \
|
|
|
|
__gu_err; \
|
|
|
|
})
|
|
|
|
|
|
|
|
#define __get_user_error(x, ptr, err) \
|
|
|
|
({ \
|
|
|
|
__get_user_err((x), (ptr), (err)); \
|
|
|
|
(void)0; \
|
|
|
|
})
|
|
|
|
|
|
|
|
#define get_user(x, ptr) \
|
|
|
|
({ \
|
2013-09-24 17:00:50 +08:00
|
|
|
__typeof__(*(ptr)) __user *__p = (ptr); \
|
2013-05-26 22:30:42 +08:00
|
|
|
might_fault(); \
|
2013-09-24 17:00:50 +08:00
|
|
|
access_ok(VERIFY_READ, __p, sizeof(*__p)) ? \
|
|
|
|
__get_user((x), __p) : \
|
2012-03-05 19:49:32 +08:00
|
|
|
((x) = 0, -EFAULT); \
|
|
|
|
})
|
|
|
|
|
2016-02-05 22:58:48 +08:00
|
|
|
#define __put_user_asm(instr, alt_instr, reg, x, addr, err, feature) \
|
2012-03-05 19:49:32 +08:00
|
|
|
asm volatile( \
|
2016-02-05 22:58:48 +08:00
|
|
|
"1:"ALTERNATIVE(instr " " reg "1, [%2]\n", \
|
|
|
|
alt_instr " " reg "1, [%2]\n", feature) \
|
2012-03-05 19:49:32 +08:00
|
|
|
"2:\n" \
|
|
|
|
" .section .fixup,\"ax\"\n" \
|
|
|
|
" .align 2\n" \
|
|
|
|
"3: mov %w0, %3\n" \
|
|
|
|
" b 2b\n" \
|
|
|
|
" .previous\n" \
|
2016-01-01 22:02:12 +08:00
|
|
|
_ASM_EXTABLE(1b, 3b) \
|
2012-03-05 19:49:32 +08:00
|
|
|
: "+r" (err) \
|
|
|
|
: "r" (x), "r" (addr), "i" (-EFAULT))
|
|
|
|
|
|
|
|
#define __put_user_err(x, ptr, err) \
|
|
|
|
do { \
|
|
|
|
__typeof__(*(ptr)) __pu_val = (x); \
|
|
|
|
__chk_user_ptr(ptr); \
|
2016-07-01 21:58:21 +08:00
|
|
|
uaccess_enable_not_uao(); \
|
2012-03-05 19:49:32 +08:00
|
|
|
switch (sizeof(*(ptr))) { \
|
|
|
|
case 1: \
|
2016-02-05 22:58:48 +08:00
|
|
|
__put_user_asm("strb", "sttrb", "%w", __pu_val, (ptr), \
|
|
|
|
(err), ARM64_HAS_UAO); \
|
2012-03-05 19:49:32 +08:00
|
|
|
break; \
|
|
|
|
case 2: \
|
2016-02-05 22:58:48 +08:00
|
|
|
__put_user_asm("strh", "sttrh", "%w", __pu_val, (ptr), \
|
|
|
|
(err), ARM64_HAS_UAO); \
|
2012-03-05 19:49:32 +08:00
|
|
|
break; \
|
|
|
|
case 4: \
|
2016-02-05 22:58:48 +08:00
|
|
|
__put_user_asm("str", "sttr", "%w", __pu_val, (ptr), \
|
|
|
|
(err), ARM64_HAS_UAO); \
|
2012-03-05 19:49:32 +08:00
|
|
|
break; \
|
|
|
|
case 8: \
|
arm64: uaccess: suppress spurious clang warning
Clang tries to warn when there's a mismatch between an operand's size,
and the size of the register it is held in, as this may indicate a bug.
Specifically, clang warns when the operand's type is less than 64 bits
wide, and the register is used unqualified (i.e. %N rather than %xN or
%wN).
Unfortunately clang can generate these warnings for unreachable code.
For example, for code like:
do { \
typeof(*(ptr)) __v = (v); \
switch(sizeof(*(ptr))) { \
case 1: \
// assume __v is 1 byte wide \
asm ("{op}b %w0" : : "r" (v)); \
break; \
case 8: \
// assume __v is 8 bytes wide \
asm ("{op} %0" : : "r" (v)); \
break; \
}
while (0)
... if op() were passed a char value and pointer to char, clang may
produce a warning for the unreachable case where sizeof(*(ptr)) is 8.
For the same reasons, clang produces warnings when __put_user_err() is
used for types that are less than 64 bits wide.
We could avoid this with a cast to a fixed-width type in each of the
cases. However, GCC will then warn that pointer types are being cast to
mismatched integer sizes (in unreachable paths).
Another option would be to use the same union trickery as we do for
__smp_store_release() and __smp_load_acquire(), but this is fairly
invasive.
Instead, this patch suppresses the clang warning by using an x modifier
in the assembly for the 8 byte case of __put_user_err(). No additional
work is necessary as the value has been cast to typeof(*(ptr)), so the
compiler will have performed any necessary extension for the reachable
case.
For consistency, __get_user_err() is also updated to use the x modifier
for its 8 byte case.
Acked-by: Will Deacon <will.deacon@arm.com>
Signed-off-by: Mark Rutland <mark.rutland@arm.com>
Reported-by: Matthias Kaehlcke <mka@chromium.org>
Signed-off-by: Catalin Marinas <catalin.marinas@arm.com>
2017-05-03 23:09:38 +08:00
|
|
|
__put_user_asm("str", "sttr", "%x", __pu_val, (ptr), \
|
2016-02-05 22:58:48 +08:00
|
|
|
(err), ARM64_HAS_UAO); \
|
2012-03-05 19:49:32 +08:00
|
|
|
break; \
|
|
|
|
default: \
|
|
|
|
BUILD_BUG(); \
|
|
|
|
} \
|
2016-07-01 21:58:21 +08:00
|
|
|
uaccess_disable_not_uao(); \
|
2012-03-05 19:49:32 +08:00
|
|
|
} while (0)
|
|
|
|
|
|
|
|
#define __put_user(x, ptr) \
|
|
|
|
({ \
|
|
|
|
int __pu_err = 0; \
|
|
|
|
__put_user_err((x), (ptr), __pu_err); \
|
|
|
|
__pu_err; \
|
|
|
|
})
|
|
|
|
|
|
|
|
#define __put_user_error(x, ptr, err) \
|
|
|
|
({ \
|
|
|
|
__put_user_err((x), (ptr), (err)); \
|
|
|
|
(void)0; \
|
|
|
|
})
|
|
|
|
|
|
|
|
#define put_user(x, ptr) \
|
|
|
|
({ \
|
2013-09-24 17:00:50 +08:00
|
|
|
__typeof__(*(ptr)) __user *__p = (ptr); \
|
2013-05-26 22:30:42 +08:00
|
|
|
might_fault(); \
|
2013-09-24 17:00:50 +08:00
|
|
|
access_ok(VERIFY_WRITE, __p, sizeof(*__p)) ? \
|
|
|
|
__put_user((x), __p) : \
|
2012-03-05 19:49:32 +08:00
|
|
|
-EFAULT; \
|
|
|
|
})
|
|
|
|
|
2016-06-09 05:40:56 +08:00
|
|
|
extern unsigned long __must_check __arch_copy_from_user(void *to, const void __user *from, unsigned long n);
|
2017-03-21 20:40:57 +08:00
|
|
|
#define raw_copy_from_user __arch_copy_from_user
|
2016-06-09 05:40:56 +08:00
|
|
|
extern unsigned long __must_check __arch_copy_to_user(void __user *to, const void *from, unsigned long n);
|
2017-03-21 20:40:57 +08:00
|
|
|
#define raw_copy_to_user __arch_copy_to_user
|
|
|
|
extern unsigned long __must_check raw_copy_in_user(void __user *to, const void __user *from, unsigned long n);
|
2012-03-05 19:49:32 +08:00
|
|
|
extern unsigned long __must_check __clear_user(void __user *addr, unsigned long n);
|
2017-03-21 20:40:57 +08:00
|
|
|
#define INLINE_COPY_TO_USER
|
|
|
|
#define INLINE_COPY_FROM_USER
|
2012-03-05 19:49:32 +08:00
|
|
|
|
|
|
|
static inline unsigned long __must_check clear_user(void __user *to, unsigned long n)
|
|
|
|
{
|
|
|
|
if (access_ok(VERIFY_WRITE, to, n))
|
|
|
|
n = __clear_user(to, n);
|
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
2013-11-07 01:20:22 +08:00
|
|
|
extern long strncpy_from_user(char *dest, const char __user *src, long count);
|
2012-03-05 19:49:32 +08:00
|
|
|
|
2013-11-07 01:20:22 +08:00
|
|
|
extern __must_check long strnlen_user(const char __user *str, long n);
|
2012-03-05 19:49:32 +08:00
|
|
|
|
2017-07-25 18:55:43 +08:00
|
|
|
#ifdef CONFIG_ARCH_HAS_UACCESS_FLUSHCACHE
|
|
|
|
struct page;
|
|
|
|
void memcpy_page_flushcache(char *to, struct page *page, size_t offset, size_t len);
|
|
|
|
extern unsigned long __must_check __copy_user_flushcache(void *to, const void __user *from, unsigned long n);
|
|
|
|
|
|
|
|
static inline int __copy_from_user_flushcache(void *dst, const void __user *src, unsigned size)
|
|
|
|
{
|
|
|
|
kasan_check_write(dst, size);
|
|
|
|
return __copy_user_flushcache(dst, src, size);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2012-03-05 19:49:32 +08:00
|
|
|
#endif /* __ASM_UACCESS_H */
|