xref: /openbmc/linux/include/linux/kasan.h (revision 49c6631d)
1b2441318SGreg Kroah-Hartman /* SPDX-License-Identifier: GPL-2.0 */
20b24beccSAndrey Ryabinin #ifndef _LINUX_KASAN_H
30b24beccSAndrey Ryabinin #define _LINUX_KASAN_H
40b24beccSAndrey Ryabinin 
534303244SAndrey Konovalov #include <linux/static_key.h>
60b24beccSAndrey Ryabinin #include <linux/types.h>
70b24beccSAndrey Ryabinin 
80b24beccSAndrey Ryabinin struct kmem_cache;
90b24beccSAndrey Ryabinin struct page;
10a5af5aa8SAndrey Ryabinin struct vm_struct;
115be9b730SMasami Hiramatsu struct task_struct;
120b24beccSAndrey Ryabinin 
130b24beccSAndrey Ryabinin #ifdef CONFIG_KASAN
140b24beccSAndrey Ryabinin 
15d5750edfSAndrey Konovalov #include <linux/linkage.h>
1665fddcfcSMike Rapoport #include <asm/kasan.h>
170b24beccSAndrey Ryabinin 
1883c4e7a0SPatricia Alfonso /* kasan_data struct is used in KUnit tests for KASAN expected failures */
1983c4e7a0SPatricia Alfonso struct kunit_kasan_expectation {
2083c4e7a0SPatricia Alfonso 	bool report_expected;
2183c4e7a0SPatricia Alfonso 	bool report_found;
2283c4e7a0SPatricia Alfonso };
2383c4e7a0SPatricia Alfonso 
24d5750edfSAndrey Konovalov #endif
25d5750edfSAndrey Konovalov 
26d5750edfSAndrey Konovalov #if defined(CONFIG_KASAN_GENERIC) || defined(CONFIG_KASAN_SW_TAGS)
27d5750edfSAndrey Konovalov 
28d5750edfSAndrey Konovalov #include <linux/pgtable.h>
29d5750edfSAndrey Konovalov 
30d5750edfSAndrey Konovalov /* Software KASAN implementations use shadow memory. */
31d5750edfSAndrey Konovalov 
32d5750edfSAndrey Konovalov #ifdef CONFIG_KASAN_SW_TAGS
33d5750edfSAndrey Konovalov #define KASAN_SHADOW_INIT 0xFF
34d5750edfSAndrey Konovalov #else
35d5750edfSAndrey Konovalov #define KASAN_SHADOW_INIT 0
36d5750edfSAndrey Konovalov #endif
37d5750edfSAndrey Konovalov 
3829970dc2SHailong Liu #ifndef PTE_HWTABLE_PTRS
3929970dc2SHailong Liu #define PTE_HWTABLE_PTRS 0
4029970dc2SHailong Liu #endif
4129970dc2SHailong Liu 
429577dd74SAndrey Konovalov extern unsigned char kasan_early_shadow_page[PAGE_SIZE];
4329970dc2SHailong Liu extern pte_t kasan_early_shadow_pte[PTRS_PER_PTE + PTE_HWTABLE_PTRS];
449577dd74SAndrey Konovalov extern pmd_t kasan_early_shadow_pmd[PTRS_PER_PMD];
459577dd74SAndrey Konovalov extern pud_t kasan_early_shadow_pud[PTRS_PER_PUD];
469577dd74SAndrey Konovalov extern p4d_t kasan_early_shadow_p4d[MAX_PTRS_PER_P4D];
4769786cdbSAndrey Ryabinin 
489577dd74SAndrey Konovalov int kasan_populate_early_shadow(const void *shadow_start,
4969786cdbSAndrey Ryabinin 				const void *shadow_end);
5069786cdbSAndrey Ryabinin 
510b24beccSAndrey Ryabinin static inline void *kasan_mem_to_shadow(const void *addr)
520b24beccSAndrey Ryabinin {
530b24beccSAndrey Ryabinin 	return (void *)((unsigned long)addr >> KASAN_SHADOW_SCALE_SHIFT)
540b24beccSAndrey Ryabinin 		+ KASAN_SHADOW_OFFSET;
550b24beccSAndrey Ryabinin }
560b24beccSAndrey Ryabinin 
57d5750edfSAndrey Konovalov int kasan_add_zero_shadow(void *start, unsigned long size);
58d5750edfSAndrey Konovalov void kasan_remove_zero_shadow(void *start, unsigned long size);
59d5750edfSAndrey Konovalov 
60d73b4936SAndrey Konovalov /* Enable reporting bugs after kasan_disable_current() */
61d73b4936SAndrey Konovalov extern void kasan_enable_current(void);
62d73b4936SAndrey Konovalov 
63d73b4936SAndrey Konovalov /* Disable reporting bugs for current task */
64d73b4936SAndrey Konovalov extern void kasan_disable_current(void);
65d73b4936SAndrey Konovalov 
66d5750edfSAndrey Konovalov #else /* CONFIG_KASAN_GENERIC || CONFIG_KASAN_SW_TAGS */
67d5750edfSAndrey Konovalov 
68d5750edfSAndrey Konovalov static inline int kasan_add_zero_shadow(void *start, unsigned long size)
69d5750edfSAndrey Konovalov {
70d5750edfSAndrey Konovalov 	return 0;
71d5750edfSAndrey Konovalov }
72d5750edfSAndrey Konovalov static inline void kasan_remove_zero_shadow(void *start,
73d5750edfSAndrey Konovalov 					unsigned long size)
74d5750edfSAndrey Konovalov {}
75d5750edfSAndrey Konovalov 
76d73b4936SAndrey Konovalov static inline void kasan_enable_current(void) {}
77d73b4936SAndrey Konovalov static inline void kasan_disable_current(void) {}
78d73b4936SAndrey Konovalov 
79d5750edfSAndrey Konovalov #endif /* CONFIG_KASAN_GENERIC || CONFIG_KASAN_SW_TAGS */
80d5750edfSAndrey Konovalov 
81d5750edfSAndrey Konovalov #ifdef CONFIG_KASAN
82d5750edfSAndrey Konovalov 
837ed2f9e6SAlexander Potapenko struct kasan_cache {
847ed2f9e6SAlexander Potapenko 	int alloc_meta_offset;
857ed2f9e6SAlexander Potapenko 	int free_meta_offset;
867ed2f9e6SAlexander Potapenko };
877ed2f9e6SAlexander Potapenko 
8834303244SAndrey Konovalov #ifdef CONFIG_KASAN_HW_TAGS
89e86f8b09SAndrey Konovalov 
9034303244SAndrey Konovalov DECLARE_STATIC_KEY_FALSE(kasan_flag_enabled);
91e86f8b09SAndrey Konovalov 
9234303244SAndrey Konovalov static __always_inline bool kasan_enabled(void)
9334303244SAndrey Konovalov {
9434303244SAndrey Konovalov 	return static_branch_likely(&kasan_flag_enabled);
9534303244SAndrey Konovalov }
96e86f8b09SAndrey Konovalov 
97e86f8b09SAndrey Konovalov #else /* CONFIG_KASAN_HW_TAGS */
98e86f8b09SAndrey Konovalov 
9934303244SAndrey Konovalov static inline bool kasan_enabled(void)
10034303244SAndrey Konovalov {
10134303244SAndrey Konovalov 	return true;
10234303244SAndrey Konovalov }
103e86f8b09SAndrey Konovalov 
104e86f8b09SAndrey Konovalov #endif /* CONFIG_KASAN_HW_TAGS */
105e86f8b09SAndrey Konovalov 
106e86f8b09SAndrey Konovalov slab_flags_t __kasan_never_merge(void);
107e86f8b09SAndrey Konovalov static __always_inline slab_flags_t kasan_never_merge(void)
108e86f8b09SAndrey Konovalov {
109e86f8b09SAndrey Konovalov 	if (kasan_enabled())
110e86f8b09SAndrey Konovalov 		return __kasan_never_merge();
111e86f8b09SAndrey Konovalov 	return 0;
112e86f8b09SAndrey Konovalov }
11334303244SAndrey Konovalov 
11434303244SAndrey Konovalov void __kasan_unpoison_range(const void *addr, size_t size);
11534303244SAndrey Konovalov static __always_inline void kasan_unpoison_range(const void *addr, size_t size)
11634303244SAndrey Konovalov {
11734303244SAndrey Konovalov 	if (kasan_enabled())
11834303244SAndrey Konovalov 		__kasan_unpoison_range(addr, size);
11934303244SAndrey Konovalov }
12034303244SAndrey Konovalov 
12134303244SAndrey Konovalov void __kasan_alloc_pages(struct page *page, unsigned int order);
12234303244SAndrey Konovalov static __always_inline void kasan_alloc_pages(struct page *page,
12334303244SAndrey Konovalov 						unsigned int order)
12434303244SAndrey Konovalov {
12534303244SAndrey Konovalov 	if (kasan_enabled())
12634303244SAndrey Konovalov 		__kasan_alloc_pages(page, order);
12734303244SAndrey Konovalov }
12834303244SAndrey Konovalov 
12934303244SAndrey Konovalov void __kasan_free_pages(struct page *page, unsigned int order);
13034303244SAndrey Konovalov static __always_inline void kasan_free_pages(struct page *page,
13134303244SAndrey Konovalov 						unsigned int order)
13234303244SAndrey Konovalov {
13334303244SAndrey Konovalov 	if (kasan_enabled())
13434303244SAndrey Konovalov 		__kasan_free_pages(page, order);
13534303244SAndrey Konovalov }
13634303244SAndrey Konovalov 
13734303244SAndrey Konovalov void __kasan_cache_create(struct kmem_cache *cache, unsigned int *size,
13834303244SAndrey Konovalov 				slab_flags_t *flags);
13934303244SAndrey Konovalov static __always_inline void kasan_cache_create(struct kmem_cache *cache,
14034303244SAndrey Konovalov 				unsigned int *size, slab_flags_t *flags)
14134303244SAndrey Konovalov {
14234303244SAndrey Konovalov 	if (kasan_enabled())
14334303244SAndrey Konovalov 		__kasan_cache_create(cache, size, flags);
14434303244SAndrey Konovalov }
14534303244SAndrey Konovalov 
14634303244SAndrey Konovalov size_t __kasan_metadata_size(struct kmem_cache *cache);
14734303244SAndrey Konovalov static __always_inline size_t kasan_metadata_size(struct kmem_cache *cache)
14834303244SAndrey Konovalov {
14934303244SAndrey Konovalov 	if (kasan_enabled())
15034303244SAndrey Konovalov 		return __kasan_metadata_size(cache);
15134303244SAndrey Konovalov 	return 0;
15234303244SAndrey Konovalov }
15334303244SAndrey Konovalov 
15434303244SAndrey Konovalov void __kasan_poison_slab(struct page *page);
15534303244SAndrey Konovalov static __always_inline void kasan_poison_slab(struct page *page)
15634303244SAndrey Konovalov {
15734303244SAndrey Konovalov 	if (kasan_enabled())
15834303244SAndrey Konovalov 		__kasan_poison_slab(page);
15934303244SAndrey Konovalov }
16034303244SAndrey Konovalov 
16134303244SAndrey Konovalov void __kasan_unpoison_object_data(struct kmem_cache *cache, void *object);
16234303244SAndrey Konovalov static __always_inline void kasan_unpoison_object_data(struct kmem_cache *cache,
16334303244SAndrey Konovalov 							void *object)
16434303244SAndrey Konovalov {
16534303244SAndrey Konovalov 	if (kasan_enabled())
16634303244SAndrey Konovalov 		__kasan_unpoison_object_data(cache, object);
16734303244SAndrey Konovalov }
16834303244SAndrey Konovalov 
16934303244SAndrey Konovalov void __kasan_poison_object_data(struct kmem_cache *cache, void *object);
17034303244SAndrey Konovalov static __always_inline void kasan_poison_object_data(struct kmem_cache *cache,
17134303244SAndrey Konovalov 							void *object)
17234303244SAndrey Konovalov {
17334303244SAndrey Konovalov 	if (kasan_enabled())
17434303244SAndrey Konovalov 		__kasan_poison_object_data(cache, object);
17534303244SAndrey Konovalov }
17634303244SAndrey Konovalov 
17734303244SAndrey Konovalov void * __must_check __kasan_init_slab_obj(struct kmem_cache *cache,
17834303244SAndrey Konovalov 					  const void *object);
17934303244SAndrey Konovalov static __always_inline void * __must_check kasan_init_slab_obj(
18034303244SAndrey Konovalov 				struct kmem_cache *cache, const void *object)
18134303244SAndrey Konovalov {
18234303244SAndrey Konovalov 	if (kasan_enabled())
18334303244SAndrey Konovalov 		return __kasan_init_slab_obj(cache, object);
18434303244SAndrey Konovalov 	return (void *)object;
18534303244SAndrey Konovalov }
18634303244SAndrey Konovalov 
18734303244SAndrey Konovalov bool __kasan_slab_free(struct kmem_cache *s, void *object, unsigned long ip);
18834303244SAndrey Konovalov static __always_inline bool kasan_slab_free(struct kmem_cache *s, void *object,
18934303244SAndrey Konovalov 						unsigned long ip)
19034303244SAndrey Konovalov {
19134303244SAndrey Konovalov 	if (kasan_enabled())
19234303244SAndrey Konovalov 		return __kasan_slab_free(s, object, ip);
19334303244SAndrey Konovalov 	return false;
19434303244SAndrey Konovalov }
19534303244SAndrey Konovalov 
196eeb3160cSAndrey Konovalov void __kasan_slab_free_mempool(void *ptr, unsigned long ip);
197eeb3160cSAndrey Konovalov static __always_inline void kasan_slab_free_mempool(void *ptr, unsigned long ip)
198eeb3160cSAndrey Konovalov {
199eeb3160cSAndrey Konovalov 	if (kasan_enabled())
200eeb3160cSAndrey Konovalov 		__kasan_slab_free_mempool(ptr, ip);
201eeb3160cSAndrey Konovalov }
202eeb3160cSAndrey Konovalov 
20334303244SAndrey Konovalov void * __must_check __kasan_slab_alloc(struct kmem_cache *s,
20434303244SAndrey Konovalov 				       void *object, gfp_t flags);
20534303244SAndrey Konovalov static __always_inline void * __must_check kasan_slab_alloc(
20634303244SAndrey Konovalov 				struct kmem_cache *s, void *object, gfp_t flags)
20734303244SAndrey Konovalov {
20834303244SAndrey Konovalov 	if (kasan_enabled())
20934303244SAndrey Konovalov 		return __kasan_slab_alloc(s, object, flags);
21034303244SAndrey Konovalov 	return object;
21134303244SAndrey Konovalov }
21234303244SAndrey Konovalov 
21334303244SAndrey Konovalov void * __must_check __kasan_kmalloc(struct kmem_cache *s, const void *object,
21434303244SAndrey Konovalov 				    size_t size, gfp_t flags);
21534303244SAndrey Konovalov static __always_inline void * __must_check kasan_kmalloc(struct kmem_cache *s,
21634303244SAndrey Konovalov 				const void *object, size_t size, gfp_t flags)
21734303244SAndrey Konovalov {
21834303244SAndrey Konovalov 	if (kasan_enabled())
21934303244SAndrey Konovalov 		return __kasan_kmalloc(s, object, size, flags);
22034303244SAndrey Konovalov 	return (void *)object;
22134303244SAndrey Konovalov }
22234303244SAndrey Konovalov 
22334303244SAndrey Konovalov void * __must_check __kasan_kmalloc_large(const void *ptr,
22434303244SAndrey Konovalov 					  size_t size, gfp_t flags);
22534303244SAndrey Konovalov static __always_inline void * __must_check kasan_kmalloc_large(const void *ptr,
22634303244SAndrey Konovalov 						      size_t size, gfp_t flags)
22734303244SAndrey Konovalov {
22834303244SAndrey Konovalov 	if (kasan_enabled())
22934303244SAndrey Konovalov 		return __kasan_kmalloc_large(ptr, size, flags);
23034303244SAndrey Konovalov 	return (void *)ptr;
23134303244SAndrey Konovalov }
23234303244SAndrey Konovalov 
23334303244SAndrey Konovalov void * __must_check __kasan_krealloc(const void *object,
23434303244SAndrey Konovalov 				     size_t new_size, gfp_t flags);
23534303244SAndrey Konovalov static __always_inline void * __must_check kasan_krealloc(const void *object,
23634303244SAndrey Konovalov 						 size_t new_size, gfp_t flags)
23734303244SAndrey Konovalov {
23834303244SAndrey Konovalov 	if (kasan_enabled())
23934303244SAndrey Konovalov 		return __kasan_krealloc(object, new_size, flags);
24034303244SAndrey Konovalov 	return (void *)object;
24134303244SAndrey Konovalov }
24234303244SAndrey Konovalov 
24334303244SAndrey Konovalov void __kasan_kfree_large(void *ptr, unsigned long ip);
24434303244SAndrey Konovalov static __always_inline void kasan_kfree_large(void *ptr, unsigned long ip)
24534303244SAndrey Konovalov {
24634303244SAndrey Konovalov 	if (kasan_enabled())
24734303244SAndrey Konovalov 		__kasan_kfree_large(ptr, ip);
24834303244SAndrey Konovalov }
2499b75a867SAndrey Ryabinin 
250b0845ce5SMark Rutland bool kasan_save_enable_multi_shot(void);
251b0845ce5SMark Rutland void kasan_restore_multi_shot(bool enabled);
252b0845ce5SMark Rutland 
2530b24beccSAndrey Ryabinin #else /* CONFIG_KASAN */
2540b24beccSAndrey Ryabinin 
25534303244SAndrey Konovalov static inline bool kasan_enabled(void)
25634303244SAndrey Konovalov {
25734303244SAndrey Konovalov 	return false;
25834303244SAndrey Konovalov }
259e86f8b09SAndrey Konovalov static inline slab_flags_t kasan_never_merge(void)
260e86f8b09SAndrey Konovalov {
261e86f8b09SAndrey Konovalov 	return 0;
262e86f8b09SAndrey Konovalov }
263cebd0eb2SAndrey Konovalov static inline void kasan_unpoison_range(const void *address, size_t size) {}
264b8c73fc2SAndrey Ryabinin static inline void kasan_alloc_pages(struct page *page, unsigned int order) {}
265b8c73fc2SAndrey Ryabinin static inline void kasan_free_pages(struct page *page, unsigned int order) {}
2667ed2f9e6SAlexander Potapenko static inline void kasan_cache_create(struct kmem_cache *cache,
267be4a7988SAlexey Dobriyan 				      unsigned int *size,
268d50112edSAlexey Dobriyan 				      slab_flags_t *flags) {}
26934303244SAndrey Konovalov static inline size_t kasan_metadata_size(struct kmem_cache *cache) { return 0; }
2700316bec2SAndrey Ryabinin static inline void kasan_poison_slab(struct page *page) {}
2710316bec2SAndrey Ryabinin static inline void kasan_unpoison_object_data(struct kmem_cache *cache,
2720316bec2SAndrey Ryabinin 					void *object) {}
2730316bec2SAndrey Ryabinin static inline void kasan_poison_object_data(struct kmem_cache *cache,
2740316bec2SAndrey Ryabinin 					void *object) {}
2750116523cSAndrey Konovalov static inline void *kasan_init_slab_obj(struct kmem_cache *cache,
2760116523cSAndrey Konovalov 				const void *object)
2770116523cSAndrey Konovalov {
2780116523cSAndrey Konovalov 	return (void *)object;
2790116523cSAndrey Konovalov }
28034303244SAndrey Konovalov static inline bool kasan_slab_free(struct kmem_cache *s, void *object,
28134303244SAndrey Konovalov 				   unsigned long ip)
2820116523cSAndrey Konovalov {
28334303244SAndrey Konovalov 	return false;
2840116523cSAndrey Konovalov }
285eeb3160cSAndrey Konovalov static inline void kasan_slab_free_mempool(void *ptr, unsigned long ip) {}
28634303244SAndrey Konovalov static inline void *kasan_slab_alloc(struct kmem_cache *s, void *object,
28734303244SAndrey Konovalov 				   gfp_t flags)
28834303244SAndrey Konovalov {
28934303244SAndrey Konovalov 	return object;
29034303244SAndrey Konovalov }
2910116523cSAndrey Konovalov static inline void *kasan_kmalloc(struct kmem_cache *s, const void *object,
2920116523cSAndrey Konovalov 				size_t size, gfp_t flags)
2930116523cSAndrey Konovalov {
2940116523cSAndrey Konovalov 	return (void *)object;
2950116523cSAndrey Konovalov }
29634303244SAndrey Konovalov static inline void *kasan_kmalloc_large(const void *ptr, size_t size, gfp_t flags)
29734303244SAndrey Konovalov {
29834303244SAndrey Konovalov 	return (void *)ptr;
29934303244SAndrey Konovalov }
3000116523cSAndrey Konovalov static inline void *kasan_krealloc(const void *object, size_t new_size,
3010116523cSAndrey Konovalov 				 gfp_t flags)
3020116523cSAndrey Konovalov {
3030116523cSAndrey Konovalov 	return (void *)object;
3040116523cSAndrey Konovalov }
30534303244SAndrey Konovalov static inline void kasan_kfree_large(void *ptr, unsigned long ip) {}
3069b75a867SAndrey Ryabinin 
3070b24beccSAndrey Ryabinin #endif /* CONFIG_KASAN */
3080b24beccSAndrey Ryabinin 
309d56a9ef8SAndrey Konovalov #if defined(CONFIG_KASAN) && CONFIG_KASAN_STACK
310d56a9ef8SAndrey Konovalov void kasan_unpoison_task_stack(struct task_struct *task);
311d56a9ef8SAndrey Konovalov #else
312d56a9ef8SAndrey Konovalov static inline void kasan_unpoison_task_stack(struct task_struct *task) {}
313d56a9ef8SAndrey Konovalov #endif
314d56a9ef8SAndrey Konovalov 
3152bd926b4SAndrey Konovalov #ifdef CONFIG_KASAN_GENERIC
3162bd926b4SAndrey Konovalov 
3172bd926b4SAndrey Konovalov void kasan_cache_shrink(struct kmem_cache *cache);
3182bd926b4SAndrey Konovalov void kasan_cache_shutdown(struct kmem_cache *cache);
31926e760c9SWalter Wu void kasan_record_aux_stack(void *ptr);
3202bd926b4SAndrey Konovalov 
3212bd926b4SAndrey Konovalov #else /* CONFIG_KASAN_GENERIC */
3222bd926b4SAndrey Konovalov 
3232bd926b4SAndrey Konovalov static inline void kasan_cache_shrink(struct kmem_cache *cache) {}
3242bd926b4SAndrey Konovalov static inline void kasan_cache_shutdown(struct kmem_cache *cache) {}
32526e760c9SWalter Wu static inline void kasan_record_aux_stack(void *ptr) {}
3262bd926b4SAndrey Konovalov 
3272bd926b4SAndrey Konovalov #endif /* CONFIG_KASAN_GENERIC */
3282bd926b4SAndrey Konovalov 
3292e903b91SAndrey Konovalov #if defined(CONFIG_KASAN_SW_TAGS) || defined(CONFIG_KASAN_HW_TAGS)
3303c9e3aa1SAndrey Konovalov 
331c0054c56SAndrey Konovalov static inline void *kasan_reset_tag(const void *addr)
332c0054c56SAndrey Konovalov {
333c0054c56SAndrey Konovalov 	return (void *)arch_kasan_reset_tag(addr);
334c0054c56SAndrey Konovalov }
3353c9e3aa1SAndrey Konovalov 
336*49c6631dSVincenzo Frascino /**
337*49c6631dSVincenzo Frascino  * kasan_report - print a report about a bad memory access detected by KASAN
338*49c6631dSVincenzo Frascino  * @addr: address of the bad access
339*49c6631dSVincenzo Frascino  * @size: size of the bad access
340*49c6631dSVincenzo Frascino  * @is_write: whether the bad access is a write or a read
341*49c6631dSVincenzo Frascino  * @ip: instruction pointer for the accessibility check or the bad access itself
342*49c6631dSVincenzo Frascino  */
3438cceeff4SWalter Wu bool kasan_report(unsigned long addr, size_t size,
34441eea9cdSAndrey Konovalov 		bool is_write, unsigned long ip);
34541eea9cdSAndrey Konovalov 
3462e903b91SAndrey Konovalov #else /* CONFIG_KASAN_SW_TAGS || CONFIG_KASAN_HW_TAGS */
3473c9e3aa1SAndrey Konovalov 
3483c9e3aa1SAndrey Konovalov static inline void *kasan_reset_tag(const void *addr)
3493c9e3aa1SAndrey Konovalov {
3503c9e3aa1SAndrey Konovalov 	return (void *)addr;
3513c9e3aa1SAndrey Konovalov }
3523c9e3aa1SAndrey Konovalov 
3532e903b91SAndrey Konovalov #endif /* CONFIG_KASAN_SW_TAGS || CONFIG_KASAN_HW_TAGS*/
3542e903b91SAndrey Konovalov 
3552e903b91SAndrey Konovalov #ifdef CONFIG_KASAN_SW_TAGS
3562e903b91SAndrey Konovalov void __init kasan_init_sw_tags(void);
3572e903b91SAndrey Konovalov #else
3582e903b91SAndrey Konovalov static inline void kasan_init_sw_tags(void) { }
3592e903b91SAndrey Konovalov #endif
3602e903b91SAndrey Konovalov 
3612e903b91SAndrey Konovalov #ifdef CONFIG_KASAN_HW_TAGS
3622e903b91SAndrey Konovalov void kasan_init_hw_tags_cpu(void);
3632e903b91SAndrey Konovalov void __init kasan_init_hw_tags(void);
3642e903b91SAndrey Konovalov #else
3652e903b91SAndrey Konovalov static inline void kasan_init_hw_tags_cpu(void) { }
3662e903b91SAndrey Konovalov static inline void kasan_init_hw_tags(void) { }
3672e903b91SAndrey Konovalov #endif
368080eb83fSAndrey Konovalov 
3693c5c3cfbSDaniel Axtens #ifdef CONFIG_KASAN_VMALLOC
3703b1a4a86SAndrey Konovalov 
371d98c9e83SAndrey Ryabinin int kasan_populate_vmalloc(unsigned long addr, unsigned long size);
372d98c9e83SAndrey Ryabinin void kasan_poison_vmalloc(const void *start, unsigned long size);
373d98c9e83SAndrey Ryabinin void kasan_unpoison_vmalloc(const void *start, unsigned long size);
3743c5c3cfbSDaniel Axtens void kasan_release_vmalloc(unsigned long start, unsigned long end,
3753c5c3cfbSDaniel Axtens 			   unsigned long free_region_start,
3763c5c3cfbSDaniel Axtens 			   unsigned long free_region_end);
3773b1a4a86SAndrey Konovalov 
3783b1a4a86SAndrey Konovalov #else /* CONFIG_KASAN_VMALLOC */
3793b1a4a86SAndrey Konovalov 
380d98c9e83SAndrey Ryabinin static inline int kasan_populate_vmalloc(unsigned long start,
381d98c9e83SAndrey Ryabinin 					unsigned long size)
3823c5c3cfbSDaniel Axtens {
3833c5c3cfbSDaniel Axtens 	return 0;
3843c5c3cfbSDaniel Axtens }
3853c5c3cfbSDaniel Axtens 
386d98c9e83SAndrey Ryabinin static inline void kasan_poison_vmalloc(const void *start, unsigned long size)
387d98c9e83SAndrey Ryabinin { }
388d98c9e83SAndrey Ryabinin static inline void kasan_unpoison_vmalloc(const void *start, unsigned long size)
389d98c9e83SAndrey Ryabinin { }
3903c5c3cfbSDaniel Axtens static inline void kasan_release_vmalloc(unsigned long start,
3913c5c3cfbSDaniel Axtens 					 unsigned long end,
3923c5c3cfbSDaniel Axtens 					 unsigned long free_region_start,
3933c5c3cfbSDaniel Axtens 					 unsigned long free_region_end) {}
3943b1a4a86SAndrey Konovalov 
3953b1a4a86SAndrey Konovalov #endif /* CONFIG_KASAN_VMALLOC */
3963b1a4a86SAndrey Konovalov 
3970fea6e9aSAndrey Konovalov #if (defined(CONFIG_KASAN_GENERIC) || defined(CONFIG_KASAN_SW_TAGS)) && \
3980fea6e9aSAndrey Konovalov 		!defined(CONFIG_KASAN_VMALLOC)
3993b1a4a86SAndrey Konovalov 
4003b1a4a86SAndrey Konovalov /*
4013b1a4a86SAndrey Konovalov  * These functions provide a special case to support backing module
4023b1a4a86SAndrey Konovalov  * allocations with real shadow memory. With KASAN vmalloc, the special
4033b1a4a86SAndrey Konovalov  * case is unnecessary, as the work is handled in the generic case.
4043b1a4a86SAndrey Konovalov  */
4053b1a4a86SAndrey Konovalov int kasan_module_alloc(void *addr, size_t size);
4063b1a4a86SAndrey Konovalov void kasan_free_shadow(const struct vm_struct *vm);
4073b1a4a86SAndrey Konovalov 
4080fea6e9aSAndrey Konovalov #else /* (CONFIG_KASAN_GENERIC || CONFIG_KASAN_SW_TAGS) && !CONFIG_KASAN_VMALLOC */
4093b1a4a86SAndrey Konovalov 
4103b1a4a86SAndrey Konovalov static inline int kasan_module_alloc(void *addr, size_t size) { return 0; }
4113b1a4a86SAndrey Konovalov static inline void kasan_free_shadow(const struct vm_struct *vm) {}
4123b1a4a86SAndrey Konovalov 
4130fea6e9aSAndrey Konovalov #endif /* (CONFIG_KASAN_GENERIC || CONFIG_KASAN_SW_TAGS) && !CONFIG_KASAN_VMALLOC */
4143c5c3cfbSDaniel Axtens 
4152f004eeaSJann Horn #ifdef CONFIG_KASAN_INLINE
4162f004eeaSJann Horn void kasan_non_canonical_hook(unsigned long addr);
4172f004eeaSJann Horn #else /* CONFIG_KASAN_INLINE */
4182f004eeaSJann Horn static inline void kasan_non_canonical_hook(unsigned long addr) { }
4192f004eeaSJann Horn #endif /* CONFIG_KASAN_INLINE */
4202f004eeaSJann Horn 
4210b24beccSAndrey Ryabinin #endif /* LINUX_KASAN_H */
422