diff options
Diffstat (limited to 'include/linux/highmem.h')
| -rw-r--r-- | include/linux/highmem.h | 65 |
1 files changed, 41 insertions, 24 deletions
diff --git a/include/linux/highmem.h b/include/linux/highmem.h index e3060ef85b6d..8a85ec109a3a 100644 --- a/include/linux/highmem.h +++ b/include/linux/highmem.h | |||
| @@ -28,18 +28,6 @@ static inline void invalidate_kernel_vmap_range(void *vaddr, int size) | |||
| 28 | 28 | ||
| 29 | #include <asm/kmap_types.h> | 29 | #include <asm/kmap_types.h> |
| 30 | 30 | ||
| 31 | #ifdef CONFIG_DEBUG_HIGHMEM | ||
| 32 | |||
| 33 | void debug_kmap_atomic(enum km_type type); | ||
| 34 | |||
| 35 | #else | ||
| 36 | |||
| 37 | static inline void debug_kmap_atomic(enum km_type type) | ||
| 38 | { | ||
| 39 | } | ||
| 40 | |||
| 41 | #endif | ||
| 42 | |||
| 43 | #ifdef CONFIG_HIGHMEM | 31 | #ifdef CONFIG_HIGHMEM |
| 44 | #include <asm/highmem.h> | 32 | #include <asm/highmem.h> |
| 45 | 33 | ||
| @@ -49,6 +37,27 @@ extern unsigned long totalhigh_pages; | |||
| 49 | 37 | ||
| 50 | void kmap_flush_unused(void); | 38 | void kmap_flush_unused(void); |
| 51 | 39 | ||
| 40 | DECLARE_PER_CPU(int, __kmap_atomic_idx); | ||
| 41 | |||
| 42 | static inline int kmap_atomic_idx_push(void) | ||
| 43 | { | ||
| 44 | int idx = __get_cpu_var(__kmap_atomic_idx)++; | ||
| 45 | #ifdef CONFIG_DEBUG_HIGHMEM | ||
| 46 | WARN_ON_ONCE(in_irq() && !irqs_disabled()); | ||
| 47 | BUG_ON(idx > KM_TYPE_NR); | ||
| 48 | #endif | ||
| 49 | return idx; | ||
| 50 | } | ||
| 51 | |||
| 52 | static inline int kmap_atomic_idx_pop(void) | ||
| 53 | { | ||
| 54 | int idx = --__get_cpu_var(__kmap_atomic_idx); | ||
| 55 | #ifdef CONFIG_DEBUG_HIGHMEM | ||
| 56 | BUG_ON(idx < 0); | ||
| 57 | #endif | ||
| 58 | return idx; | ||
| 59 | } | ||
| 60 | |||
| 52 | #else /* CONFIG_HIGHMEM */ | 61 | #else /* CONFIG_HIGHMEM */ |
| 53 | 62 | ||
| 54 | static inline unsigned int nr_free_highpages(void) { return 0; } | 63 | static inline unsigned int nr_free_highpages(void) { return 0; } |
| @@ -66,19 +75,19 @@ static inline void kunmap(struct page *page) | |||
| 66 | { | 75 | { |
| 67 | } | 76 | } |
| 68 | 77 | ||
| 69 | static inline void *kmap_atomic(struct page *page, enum km_type idx) | 78 | static inline void *__kmap_atomic(struct page *page) |
| 70 | { | 79 | { |
| 71 | pagefault_disable(); | 80 | pagefault_disable(); |
| 72 | return page_address(page); | 81 | return page_address(page); |
| 73 | } | 82 | } |
| 74 | #define kmap_atomic_prot(page, idx, prot) kmap_atomic(page, idx) | 83 | #define kmap_atomic_prot(page, prot) __kmap_atomic(page) |
| 75 | 84 | ||
| 76 | static inline void kunmap_atomic_notypecheck(void *addr, enum km_type idx) | 85 | static inline void __kunmap_atomic(void *addr) |
| 77 | { | 86 | { |
| 78 | pagefault_enable(); | 87 | pagefault_enable(); |
| 79 | } | 88 | } |
| 80 | 89 | ||
| 81 | #define kmap_atomic_pfn(pfn, idx) kmap_atomic(pfn_to_page(pfn), (idx)) | 90 | #define kmap_atomic_pfn(pfn) kmap_atomic(pfn_to_page(pfn)) |
| 82 | #define kmap_atomic_to_page(ptr) virt_to_page(ptr) | 91 | #define kmap_atomic_to_page(ptr) virt_to_page(ptr) |
| 83 | 92 | ||
| 84 | #define kmap_flush_unused() do {} while(0) | 93 | #define kmap_flush_unused() do {} while(0) |
| @@ -86,12 +95,20 @@ static inline void kunmap_atomic_notypecheck(void *addr, enum km_type idx) | |||
| 86 | 95 | ||
| 87 | #endif /* CONFIG_HIGHMEM */ | 96 | #endif /* CONFIG_HIGHMEM */ |
| 88 | 97 | ||
| 89 | /* Prevent people trying to call kunmap_atomic() as if it were kunmap() */ | 98 | /* |
| 90 | /* kunmap_atomic() should get the return value of kmap_atomic, not the page. */ | 99 | * Make both: kmap_atomic(page, idx) and kmap_atomic(page) work. |
| 91 | #define kunmap_atomic(addr, idx) do { \ | 100 | */ |
| 92 | BUILD_BUG_ON(__same_type((addr), struct page *)); \ | 101 | #define kmap_atomic(page, args...) __kmap_atomic(page) |
| 93 | kunmap_atomic_notypecheck((addr), (idx)); \ | 102 | |
| 94 | } while (0) | 103 | /* |
| 104 | * Prevent people trying to call kunmap_atomic() as if it were kunmap() | ||
| 105 | * kunmap_atomic() should get the return value of kmap_atomic, not the page. | ||
| 106 | */ | ||
| 107 | #define kunmap_atomic(addr, args...) \ | ||
| 108 | do { \ | ||
| 109 | BUILD_BUG_ON(__same_type((addr), struct page *)); \ | ||
| 110 | __kunmap_atomic(addr); \ | ||
| 111 | } while (0) | ||
| 95 | 112 | ||
| 96 | /* when CONFIG_HIGHMEM is not set these will be plain clear/copy_page */ | 113 | /* when CONFIG_HIGHMEM is not set these will be plain clear/copy_page */ |
| 97 | #ifndef clear_user_highpage | 114 | #ifndef clear_user_highpage |
| @@ -201,8 +218,8 @@ static inline void copy_user_highpage(struct page *to, struct page *from, | |||
| 201 | vfrom = kmap_atomic(from, KM_USER0); | 218 | vfrom = kmap_atomic(from, KM_USER0); |
| 202 | vto = kmap_atomic(to, KM_USER1); | 219 | vto = kmap_atomic(to, KM_USER1); |
| 203 | copy_user_page(vto, vfrom, vaddr, to); | 220 | copy_user_page(vto, vfrom, vaddr, to); |
| 204 | kunmap_atomic(vfrom, KM_USER0); | ||
| 205 | kunmap_atomic(vto, KM_USER1); | 221 | kunmap_atomic(vto, KM_USER1); |
| 222 | kunmap_atomic(vfrom, KM_USER0); | ||
| 206 | } | 223 | } |
| 207 | 224 | ||
| 208 | #endif | 225 | #endif |
| @@ -214,8 +231,8 @@ static inline void copy_highpage(struct page *to, struct page *from) | |||
| 214 | vfrom = kmap_atomic(from, KM_USER0); | 231 | vfrom = kmap_atomic(from, KM_USER0); |
| 215 | vto = kmap_atomic(to, KM_USER1); | 232 | vto = kmap_atomic(to, KM_USER1); |
| 216 | copy_page(vto, vfrom); | 233 | copy_page(vto, vfrom); |
| 217 | kunmap_atomic(vfrom, KM_USER0); | ||
| 218 | kunmap_atomic(vto, KM_USER1); | 234 | kunmap_atomic(vto, KM_USER1); |
| 235 | kunmap_atomic(vfrom, KM_USER0); | ||
| 219 | } | 236 | } |
| 220 | 237 | ||
| 221 | #endif /* _LINUX_HIGHMEM_H */ | 238 | #endif /* _LINUX_HIGHMEM_H */ |
