uaccess_64.h 8.0 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
#ifndef __X86_64_UACCESS_H
#define __X86_64_UACCESS_H

/*
 * User space memory access functions
 */
#include <linux/compiler.h>
#include <linux/errno.h>
#include <linux/prefetch.h>
#include <asm/page.h>

#define ARCH_HAS_SEARCH_EXTABLE

extern void __put_user_1(void);
extern void __put_user_2(void);
extern void __put_user_4(void);
extern void __put_user_8(void);
extern void __put_user_bad(void);

20 21 22
#define __put_user_x(size, ret, x, ptr)					\
	asm volatile("call __put_user_" #size				\
		     :"=a" (ret)					\
23
		     :"c" (ptr),"a" (x)					\
24
		     :"ebx")
L
Linus Torvalds 已提交
25

26 27
#define put_user(x, ptr)						\
	__put_user_check((__typeof__(*(ptr)))(x), (ptr), sizeof(*(ptr)))
L
Linus Torvalds 已提交
28

29 30 31 32
#define __get_user(x, ptr)						\
	__get_user_nocheck((x), (ptr), sizeof(*(ptr)))
#define __put_user(x, ptr)						\
	__put_user_nocheck((__typeof__(*(ptr)))(x), (ptr), sizeof(*(ptr)))
L
Linus Torvalds 已提交
33 34 35 36

#define __get_user_unaligned __get_user
#define __put_user_unaligned __put_user

37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57
#define __put_user_check(x, ptr, size)				\
({								\
	int __pu_err;						\
	typeof(*(ptr)) __user *__pu_addr = (ptr);		\
	switch (size) {						\
	case 1:							\
		__put_user_x(1, __pu_err, x, __pu_addr);	\
		break;						\
	case 2:							\
		__put_user_x(2, __pu_err, x, __pu_addr);	\
		break;						\
	case 4:							\
		__put_user_x(4, __pu_err, x, __pu_addr);	\
		break;						\
	case 8:							\
		__put_user_x(8, __pu_err, x, __pu_addr);	\
		break;						\
	default:						\
		__put_user_bad();				\
	}							\
	__pu_err;						\
L
Linus Torvalds 已提交
58 59
})

60
#define __get_user_nocheck(x, ptr, size)			\
L
Linus Torvalds 已提交
61 62 63
({								\
	int __gu_err;						\
	unsigned long __gu_val;					\
64
	__get_user_size(__gu_val, (ptr), (size), __gu_err);	\
A
Al Viro 已提交
65
	(x) = (__force typeof(*(ptr)))__gu_val;			\
L
Linus Torvalds 已提交
66 67 68
	__gu_err;						\
})

69
#define __get_user_size(x, ptr, size, retval)				\
L
Linus Torvalds 已提交
70 71 72 73
do {									\
	retval = 0;							\
	__chk_user_ptr(ptr);						\
	switch (size) {							\
74 75 76 77 78 79 80 81 82 83 84 85 86 87
	case 1:								\
		__get_user_asm(x, ptr, retval, "b", "b", "=q", -EFAULT);\
		break;							\
	case 2:								\
		__get_user_asm(x, ptr, retval, "w", "w", "=r", -EFAULT);\
		break;							\
	case 4:								\
		__get_user_asm(x, ptr, retval, "l", "k", "=r", -EFAULT);\
		break;							\
	case 8:								\
		__get_user_asm(x, ptr, retval, "q", "", "=r", -EFAULT);	\
		break;							\
	default:							\
		(x) = __get_user_bad();					\
L
Linus Torvalds 已提交
88 89 90 91
	}								\
} while (0)

#define __get_user_asm(x, addr, err, itype, rtype, ltype, errno)	\
92 93 94 95 96 97 98 99 100 101
	asm volatile("1:	mov"itype" %2,%"rtype"1\n"		\
		     "2:\n"						\
		     ".section .fixup, \"ax\"\n"			\
		     "3:	mov %3,%0\n"				\
		     "	xor"itype" %"rtype"1,%"rtype"1\n"		\
		     "	jmp 2b\n"					\
		     ".previous\n"					\
		     _ASM_EXTABLE(1b, 3b)				\
		     : "=r" (err), ltype (x)				\
		     : "m" (__m(addr)), "i"(errno), "0"(err))
L
Linus Torvalds 已提交
102 103 104 105 106 107

/*
 * Copy To/From Userspace
 */

/* Handles exceptions in both to and from, but doesn't do access_ok */
108 109 110 111 112 113 114 115 116 117 118 119
__must_check unsigned long
copy_user_generic(void *to, const void *from, unsigned len);

__must_check unsigned long
copy_to_user(void __user *to, const void *from, unsigned len);
__must_check unsigned long
copy_from_user(void *to, const void __user *from, unsigned len);
__must_check unsigned long
copy_in_user(void __user *to, const void __user *from, unsigned len);

static __always_inline __must_check
int __copy_from_user(void *dst, const void __user *src, unsigned size)
120
{
121
	int ret = 0;
L
Linus Torvalds 已提交
122
	if (!__builtin_constant_p(size))
123 124 125 126
		return copy_user_generic(dst, (__force void *)src, size);
	switch (size) {
	case 1:__get_user_asm(*(u8 *)dst, (u8 __user *)src,
			      ret, "b", "b", "=q", 1);
L
Linus Torvalds 已提交
127
		return ret;
128 129
	case 2:__get_user_asm(*(u16 *)dst, (u16 __user *)src,
			      ret, "w", "w", "=r", 2);
L
Linus Torvalds 已提交
130
		return ret;
131 132 133 134 135
	case 4:__get_user_asm(*(u32 *)dst, (u32 __user *)src,
			      ret, "l", "k", "=r", 4);
		return ret;
	case 8:__get_user_asm(*(u64 *)dst, (u64 __user *)src,
			      ret, "q", "", "=r", 8);
L
Linus Torvalds 已提交
136 137
		return ret;
	case 10:
138 139 140 141 142 143 144 145
		__get_user_asm(*(u64 *)dst, (u64 __user *)src,
			       ret, "q", "", "=r", 16);
		if (unlikely(ret))
			return ret;
		__get_user_asm(*(u16 *)(8 + (char *)dst),
			       (u16 __user *)(8 + (char __user *)src),
			       ret, "w", "w", "=r", 2);
		return ret;
L
Linus Torvalds 已提交
146
	case 16:
147 148 149 150 151 152 153 154
		__get_user_asm(*(u64 *)dst, (u64 __user *)src,
			       ret, "q", "", "=r", 16);
		if (unlikely(ret))
			return ret;
		__get_user_asm(*(u64 *)(8 + (char *)dst),
			       (u64 __user *)(8 + (char __user *)src),
			       ret, "q", "", "=r", 8);
		return ret;
L
Linus Torvalds 已提交
155
	default:
156
		return copy_user_generic(dst, (__force void *)src, size);
L
Linus Torvalds 已提交
157
	}
158
}
L
Linus Torvalds 已提交
159

160 161
static __always_inline __must_check
int __copy_to_user(void __user *dst, const void *src, unsigned size)
162
{
163
	int ret = 0;
L
Linus Torvalds 已提交
164
	if (!__builtin_constant_p(size))
165 166 167 168
		return copy_user_generic((__force void *)dst, src, size);
	switch (size) {
	case 1:__put_user_asm(*(u8 *)src, (u8 __user *)dst,
			      ret, "b", "b", "iq", 1);
L
Linus Torvalds 已提交
169
		return ret;
170 171
	case 2:__put_user_asm(*(u16 *)src, (u16 __user *)dst,
			      ret, "w", "w", "ir", 2);
L
Linus Torvalds 已提交
172
		return ret;
173 174 175 176 177
	case 4:__put_user_asm(*(u32 *)src, (u32 __user *)dst,
			      ret, "l", "k", "ir", 4);
		return ret;
	case 8:__put_user_asm(*(u64 *)src, (u64 __user *)dst,
			      ret, "q", "", "ir", 8);
L
Linus Torvalds 已提交
178 179
		return ret;
	case 10:
180 181 182 183
		__put_user_asm(*(u64 *)src, (u64 __user *)dst,
			       ret, "q", "", "ir", 10);
		if (unlikely(ret))
			return ret;
L
Linus Torvalds 已提交
184
		asm("":::"memory");
185 186 187
		__put_user_asm(4[(u16 *)src], 4 + (u16 __user *)dst,
			       ret, "w", "w", "ir", 2);
		return ret;
L
Linus Torvalds 已提交
188
	case 16:
189 190 191 192
		__put_user_asm(*(u64 *)src, (u64 __user *)dst,
			       ret, "q", "", "ir", 16);
		if (unlikely(ret))
			return ret;
L
Linus Torvalds 已提交
193
		asm("":::"memory");
194 195 196
		__put_user_asm(1[(u64 *)src], 1 + (u64 __user *)dst,
			       ret, "q", "", "ir", 8);
		return ret;
L
Linus Torvalds 已提交
197
	default:
198
		return copy_user_generic((__force void *)dst, src, size);
L
Linus Torvalds 已提交
199
	}
200
}
L
Linus Torvalds 已提交
201

202 203
static __always_inline __must_check
int __copy_in_user(void __user *dst, const void __user *src, unsigned size)
204
{
205
	int ret = 0;
L
Linus Torvalds 已提交
206
	if (!__builtin_constant_p(size))
207 208 209 210
		return copy_user_generic((__force void *)dst,
					 (__force void *)src, size);
	switch (size) {
	case 1: {
L
Linus Torvalds 已提交
211
		u8 tmp;
212 213
		__get_user_asm(tmp, (u8 __user *)src,
			       ret, "b", "b", "=q", 1);
L
Linus Torvalds 已提交
214
		if (likely(!ret))
215 216
			__put_user_asm(tmp, (u8 __user *)dst,
				       ret, "b", "b", "iq", 1);
L
Linus Torvalds 已提交
217 218
		return ret;
	}
219
	case 2: {
L
Linus Torvalds 已提交
220
		u16 tmp;
221 222
		__get_user_asm(tmp, (u16 __user *)src,
			       ret, "w", "w", "=r", 2);
L
Linus Torvalds 已提交
223
		if (likely(!ret))
224 225
			__put_user_asm(tmp, (u16 __user *)dst,
				       ret, "w", "w", "ir", 2);
L
Linus Torvalds 已提交
226 227 228
		return ret;
	}

229
	case 4: {
L
Linus Torvalds 已提交
230
		u32 tmp;
231 232
		__get_user_asm(tmp, (u32 __user *)src,
			       ret, "l", "k", "=r", 4);
L
Linus Torvalds 已提交
233
		if (likely(!ret))
234 235
			__put_user_asm(tmp, (u32 __user *)dst,
				       ret, "l", "k", "ir", 4);
L
Linus Torvalds 已提交
236 237
		return ret;
	}
238
	case 8: {
L
Linus Torvalds 已提交
239
		u64 tmp;
240 241
		__get_user_asm(tmp, (u64 __user *)src,
			       ret, "q", "", "=r", 8);
L
Linus Torvalds 已提交
242
		if (likely(!ret))
243 244
			__put_user_asm(tmp, (u64 __user *)dst,
				       ret, "q", "", "ir", 8);
L
Linus Torvalds 已提交
245 246 247
		return ret;
	}
	default:
248 249
		return copy_user_generic((__force void *)dst,
					 (__force void *)src, size);
L
Linus Torvalds 已提交
250
	}
251
}
L
Linus Torvalds 已提交
252

253
__must_check long
254
strncpy_from_user(char *dst, const char __user *src, long count);
255
__must_check long
256 257 258 259 260 261 262
__strncpy_from_user(char *dst, const char __user *src, long count);
__must_check long strnlen_user(const char __user *str, long n);
__must_check long __strnlen_user(const char __user *str, long n);
__must_check long strlen_user(const char __user *str);
__must_check unsigned long clear_user(void __user *mem, unsigned long len);
__must_check unsigned long __clear_user(void __user *mem, unsigned long len);

263 264
__must_check long __copy_from_user_inatomic(void *dst, const void __user *src,
					    unsigned size);
265 266 267 268 269 270

static __must_check __always_inline int
__copy_to_user_inatomic(void __user *dst, const void *src, unsigned size)
{
	return copy_user_generic((__force void *)dst, src, size);
}
L
Linus Torvalds 已提交
271

272
#define ARCH_HAS_NOCACHE_UACCESS 1
273 274
extern long __copy_user_nocache(void *dst, const void __user *src,
				unsigned size, int zerorest);
275

276 277
static inline int __copy_from_user_nocache(void *dst, const void __user *src,
					   unsigned size)
278 279
{
	might_sleep();
280
	return __copy_user_nocache(dst, src, size, 1);
281 282
}

283 284 285
static inline int __copy_from_user_inatomic_nocache(void *dst,
						    const void __user *src,
						    unsigned size)
286
{
287
	return __copy_user_nocache(dst, src, size, 0);
288 289
}

L
Linus Torvalds 已提交
290
#endif /* __X86_64_UACCESS_H */