]> err.no Git - linux-2.6/commitdiff
[PATCH] add probe_kernel_address()
authorAndrew Morton <akpm@osdl.org>
Wed, 27 Sep 2006 08:51:14 +0000 (01:51 -0700)
committerLinus Torvalds <torvalds@g5.osdl.org>
Wed, 27 Sep 2006 15:26:20 +0000 (08:26 -0700)
Add a version of __get_user() which is safe to call inside mmap_sem.

Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
include/linux/uaccess.h

index 391e7ed1eb3fc05ea2eeb8548e42381c8aefa7e5..a48d7f11c7be06772327604dc40c3f91b0b57145 100644 (file)
@@ -19,4 +19,26 @@ static inline unsigned long __copy_from_user_nocache(void *to,
 
 #endif         /* ARCH_HAS_NOCACHE_UACCESS */
 
+/**
+ * probe_kernel_address(): safely attempt to read from a location
+ * @addr: address to read from - its type is type typeof(retval)*
+ * @retval: read into this variable
+ *
+ * Safely read from address @addr into variable @revtal.  If a kernel fault
+ * happens, handle that and return -EFAULT.
+ * We ensure that the __get_user() is executed in atomic context so that
+ * do_page_fault() doesn't attempt to take mmap_sem.  This makes
+ * probe_kernel_address() suitable for use within regions where the caller
+ * already holds mmap_sem, or other locks which nest inside mmap_sem.
+ */
+#define probe_kernel_address(addr, retval)             \
+       ({                                              \
+               long ret;                               \
+                                                       \
+               inc_preempt_count();                    \
+               ret = __get_user(retval, addr);         \
+               dec_preempt_count();                    \
+               ret;                                    \
+       })
+
 #endif         /* __LINUX_UACCESS_H__ */