xref: /openbmc/linux/include/linux/kasan.h (revision d57a964e)
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
33a064cb00SAndrey Konovalov /* This matches KASAN_TAG_INVALID. */
34a064cb00SAndrey Konovalov #define KASAN_SHADOW_INIT 0xFE
35d5750edfSAndrey Konovalov #else
36d5750edfSAndrey Konovalov #define KASAN_SHADOW_INIT 0
37d5750edfSAndrey Konovalov #endif
38d5750edfSAndrey Konovalov 
3929970dc2SHailong Liu #ifndef PTE_HWTABLE_PTRS
4029970dc2SHailong Liu #define PTE_HWTABLE_PTRS 0
4129970dc2SHailong Liu #endif
4229970dc2SHailong Liu 
439577dd74SAndrey Konovalov extern unsigned char kasan_early_shadow_page[PAGE_SIZE];
4429970dc2SHailong Liu extern pte_t kasan_early_shadow_pte[PTRS_PER_PTE + PTE_HWTABLE_PTRS];
459577dd74SAndrey Konovalov extern pmd_t kasan_early_shadow_pmd[PTRS_PER_PMD];
469577dd74SAndrey Konovalov extern pud_t kasan_early_shadow_pud[PTRS_PER_PUD];
479577dd74SAndrey Konovalov extern p4d_t kasan_early_shadow_p4d[MAX_PTRS_PER_P4D];
4869786cdbSAndrey Ryabinin 
499577dd74SAndrey Konovalov int kasan_populate_early_shadow(const void *shadow_start,
5069786cdbSAndrey Ryabinin 				const void *shadow_end);
5169786cdbSAndrey Ryabinin 
520b24beccSAndrey Ryabinin static inline void *kasan_mem_to_shadow(const void *addr)
530b24beccSAndrey Ryabinin {
540b24beccSAndrey Ryabinin 	return (void *)((unsigned long)addr >> KASAN_SHADOW_SCALE_SHIFT)
550b24beccSAndrey Ryabinin 		+ KASAN_SHADOW_OFFSET;
560b24beccSAndrey Ryabinin }
570b24beccSAndrey Ryabinin 
58d5750edfSAndrey Konovalov int kasan_add_zero_shadow(void *start, unsigned long size);
59d5750edfSAndrey Konovalov void kasan_remove_zero_shadow(void *start, unsigned long size);
60d5750edfSAndrey Konovalov 
61d73b4936SAndrey Konovalov /* Enable reporting bugs after kasan_disable_current() */
62d73b4936SAndrey Konovalov extern void kasan_enable_current(void);
63d73b4936SAndrey Konovalov 
64d73b4936SAndrey Konovalov /* Disable reporting bugs for current task */
65d73b4936SAndrey Konovalov extern void kasan_disable_current(void);
66d73b4936SAndrey Konovalov 
67d5750edfSAndrey Konovalov #else /* CONFIG_KASAN_GENERIC || CONFIG_KASAN_SW_TAGS */
68d5750edfSAndrey Konovalov 
69d5750edfSAndrey Konovalov static inline int kasan_add_zero_shadow(void *start, unsigned long size)
70d5750edfSAndrey Konovalov {
71d5750edfSAndrey Konovalov 	return 0;
72d5750edfSAndrey Konovalov }
73d5750edfSAndrey Konovalov static inline void kasan_remove_zero_shadow(void *start,
74d5750edfSAndrey Konovalov 					unsigned long size)
75d5750edfSAndrey Konovalov {}
76d5750edfSAndrey Konovalov 
77d73b4936SAndrey Konovalov static inline void kasan_enable_current(void) {}
78d73b4936SAndrey Konovalov static inline void kasan_disable_current(void) {}
79d73b4936SAndrey Konovalov 
80d5750edfSAndrey Konovalov #endif /* CONFIG_KASAN_GENERIC || CONFIG_KASAN_SW_TAGS */
81d5750edfSAndrey Konovalov 
82d5750edfSAndrey Konovalov #ifdef CONFIG_KASAN
83d5750edfSAndrey Konovalov 
847ed2f9e6SAlexander Potapenko struct kasan_cache {
857ed2f9e6SAlexander Potapenko 	int alloc_meta_offset;
867ed2f9e6SAlexander Potapenko 	int free_meta_offset;
8792850134SAndrey Konovalov 	bool is_kmalloc;
887ed2f9e6SAlexander Potapenko };
897ed2f9e6SAlexander Potapenko 
9034303244SAndrey Konovalov #ifdef CONFIG_KASAN_HW_TAGS
91e86f8b09SAndrey Konovalov 
9234303244SAndrey Konovalov DECLARE_STATIC_KEY_FALSE(kasan_flag_enabled);
93e86f8b09SAndrey Konovalov 
9434303244SAndrey Konovalov static __always_inline bool kasan_enabled(void)
9534303244SAndrey Konovalov {
9634303244SAndrey Konovalov 	return static_branch_likely(&kasan_flag_enabled);
9734303244SAndrey Konovalov }
98e86f8b09SAndrey Konovalov 
991bb5eab3SAndrey Konovalov static inline bool kasan_has_integrated_init(void)
1001bb5eab3SAndrey Konovalov {
1011bb5eab3SAndrey Konovalov 	return kasan_enabled();
1021bb5eab3SAndrey Konovalov }
1031bb5eab3SAndrey Konovalov 
104e86f8b09SAndrey Konovalov #else /* CONFIG_KASAN_HW_TAGS */
105e86f8b09SAndrey Konovalov 
10634303244SAndrey Konovalov static inline bool kasan_enabled(void)
10734303244SAndrey Konovalov {
10834303244SAndrey Konovalov 	return true;
10934303244SAndrey Konovalov }
110e86f8b09SAndrey Konovalov 
1111bb5eab3SAndrey Konovalov static inline bool kasan_has_integrated_init(void)
1121bb5eab3SAndrey Konovalov {
1131bb5eab3SAndrey Konovalov 	return false;
1141bb5eab3SAndrey Konovalov }
1151bb5eab3SAndrey Konovalov 
116e86f8b09SAndrey Konovalov #endif /* CONFIG_KASAN_HW_TAGS */
117e86f8b09SAndrey Konovalov 
118e86f8b09SAndrey Konovalov slab_flags_t __kasan_never_merge(void);
119e86f8b09SAndrey Konovalov static __always_inline slab_flags_t kasan_never_merge(void)
120e86f8b09SAndrey Konovalov {
121e86f8b09SAndrey Konovalov 	if (kasan_enabled())
122e86f8b09SAndrey Konovalov 		return __kasan_never_merge();
123e86f8b09SAndrey Konovalov 	return 0;
124e86f8b09SAndrey Konovalov }
12534303244SAndrey Konovalov 
12634303244SAndrey Konovalov void __kasan_unpoison_range(const void *addr, size_t size);
12734303244SAndrey Konovalov static __always_inline void kasan_unpoison_range(const void *addr, size_t size)
12834303244SAndrey Konovalov {
12934303244SAndrey Konovalov 	if (kasan_enabled())
13034303244SAndrey Konovalov 		__kasan_unpoison_range(addr, size);
13134303244SAndrey Konovalov }
13234303244SAndrey Konovalov 
1331bb5eab3SAndrey Konovalov void __kasan_alloc_pages(struct page *page, unsigned int order, bool init);
13434303244SAndrey Konovalov static __always_inline void kasan_alloc_pages(struct page *page,
1351bb5eab3SAndrey Konovalov 						unsigned int order, bool init)
13634303244SAndrey Konovalov {
13734303244SAndrey Konovalov 	if (kasan_enabled())
1381bb5eab3SAndrey Konovalov 		__kasan_alloc_pages(page, order, init);
13934303244SAndrey Konovalov }
14034303244SAndrey Konovalov 
1411bb5eab3SAndrey Konovalov void __kasan_free_pages(struct page *page, unsigned int order, bool init);
14234303244SAndrey Konovalov static __always_inline void kasan_free_pages(struct page *page,
1431bb5eab3SAndrey Konovalov 						unsigned int order, bool init)
14434303244SAndrey Konovalov {
14534303244SAndrey Konovalov 	if (kasan_enabled())
1461bb5eab3SAndrey Konovalov 		__kasan_free_pages(page, order, init);
14734303244SAndrey Konovalov }
14834303244SAndrey Konovalov 
14934303244SAndrey Konovalov void __kasan_cache_create(struct kmem_cache *cache, unsigned int *size,
15034303244SAndrey Konovalov 				slab_flags_t *flags);
15134303244SAndrey Konovalov static __always_inline void kasan_cache_create(struct kmem_cache *cache,
15234303244SAndrey Konovalov 				unsigned int *size, slab_flags_t *flags)
15334303244SAndrey Konovalov {
15434303244SAndrey Konovalov 	if (kasan_enabled())
15534303244SAndrey Konovalov 		__kasan_cache_create(cache, size, flags);
15634303244SAndrey Konovalov }
15734303244SAndrey Konovalov 
15892850134SAndrey Konovalov void __kasan_cache_create_kmalloc(struct kmem_cache *cache);
15992850134SAndrey Konovalov static __always_inline void kasan_cache_create_kmalloc(struct kmem_cache *cache)
16092850134SAndrey Konovalov {
16192850134SAndrey Konovalov 	if (kasan_enabled())
16292850134SAndrey Konovalov 		__kasan_cache_create_kmalloc(cache);
16392850134SAndrey Konovalov }
16492850134SAndrey Konovalov 
16534303244SAndrey Konovalov size_t __kasan_metadata_size(struct kmem_cache *cache);
16634303244SAndrey Konovalov static __always_inline size_t kasan_metadata_size(struct kmem_cache *cache)
16734303244SAndrey Konovalov {
16834303244SAndrey Konovalov 	if (kasan_enabled())
16934303244SAndrey Konovalov 		return __kasan_metadata_size(cache);
17034303244SAndrey Konovalov 	return 0;
17134303244SAndrey Konovalov }
17234303244SAndrey Konovalov 
17334303244SAndrey Konovalov void __kasan_poison_slab(struct page *page);
17434303244SAndrey Konovalov static __always_inline void kasan_poison_slab(struct page *page)
17534303244SAndrey Konovalov {
17634303244SAndrey Konovalov 	if (kasan_enabled())
17734303244SAndrey Konovalov 		__kasan_poison_slab(page);
17834303244SAndrey Konovalov }
17934303244SAndrey Konovalov 
18034303244SAndrey Konovalov void __kasan_unpoison_object_data(struct kmem_cache *cache, void *object);
18134303244SAndrey Konovalov static __always_inline void kasan_unpoison_object_data(struct kmem_cache *cache,
18234303244SAndrey Konovalov 							void *object)
18334303244SAndrey Konovalov {
18434303244SAndrey Konovalov 	if (kasan_enabled())
18534303244SAndrey Konovalov 		__kasan_unpoison_object_data(cache, object);
18634303244SAndrey Konovalov }
18734303244SAndrey Konovalov 
18834303244SAndrey Konovalov void __kasan_poison_object_data(struct kmem_cache *cache, void *object);
18934303244SAndrey Konovalov static __always_inline void kasan_poison_object_data(struct kmem_cache *cache,
19034303244SAndrey Konovalov 							void *object)
19134303244SAndrey Konovalov {
19234303244SAndrey Konovalov 	if (kasan_enabled())
19334303244SAndrey Konovalov 		__kasan_poison_object_data(cache, object);
19434303244SAndrey Konovalov }
19534303244SAndrey Konovalov 
19634303244SAndrey Konovalov void * __must_check __kasan_init_slab_obj(struct kmem_cache *cache,
19734303244SAndrey Konovalov 					  const void *object);
19834303244SAndrey Konovalov static __always_inline void * __must_check kasan_init_slab_obj(
19934303244SAndrey Konovalov 				struct kmem_cache *cache, const void *object)
20034303244SAndrey Konovalov {
20134303244SAndrey Konovalov 	if (kasan_enabled())
20234303244SAndrey Konovalov 		return __kasan_init_slab_obj(cache, object);
20334303244SAndrey Konovalov 	return (void *)object;
20434303244SAndrey Konovalov }
20534303244SAndrey Konovalov 
206*d57a964eSAndrey Konovalov bool __kasan_slab_free(struct kmem_cache *s, void *object,
207*d57a964eSAndrey Konovalov 			unsigned long ip, bool init);
208*d57a964eSAndrey Konovalov static __always_inline bool kasan_slab_free(struct kmem_cache *s,
209*d57a964eSAndrey Konovalov 						void *object, bool init)
21034303244SAndrey Konovalov {
21134303244SAndrey Konovalov 	if (kasan_enabled())
212*d57a964eSAndrey Konovalov 		return __kasan_slab_free(s, object, _RET_IP_, init);
21334303244SAndrey Konovalov 	return false;
21434303244SAndrey Konovalov }
21534303244SAndrey Konovalov 
216200072ceSAndrey Konovalov void __kasan_kfree_large(void *ptr, unsigned long ip);
217200072ceSAndrey Konovalov static __always_inline void kasan_kfree_large(void *ptr)
218200072ceSAndrey Konovalov {
219200072ceSAndrey Konovalov 	if (kasan_enabled())
220200072ceSAndrey Konovalov 		__kasan_kfree_large(ptr, _RET_IP_);
221200072ceSAndrey Konovalov }
222200072ceSAndrey Konovalov 
223eeb3160cSAndrey Konovalov void __kasan_slab_free_mempool(void *ptr, unsigned long ip);
224027b37b5SAndrey Konovalov static __always_inline void kasan_slab_free_mempool(void *ptr)
225eeb3160cSAndrey Konovalov {
226eeb3160cSAndrey Konovalov 	if (kasan_enabled())
227027b37b5SAndrey Konovalov 		__kasan_slab_free_mempool(ptr, _RET_IP_);
228eeb3160cSAndrey Konovalov }
229eeb3160cSAndrey Konovalov 
23034303244SAndrey Konovalov void * __must_check __kasan_slab_alloc(struct kmem_cache *s,
231da844b78SAndrey Konovalov 				       void *object, gfp_t flags, bool init);
23234303244SAndrey Konovalov static __always_inline void * __must_check kasan_slab_alloc(
233da844b78SAndrey Konovalov 		struct kmem_cache *s, void *object, gfp_t flags, bool init)
23434303244SAndrey Konovalov {
23534303244SAndrey Konovalov 	if (kasan_enabled())
236da844b78SAndrey Konovalov 		return __kasan_slab_alloc(s, object, flags, init);
23734303244SAndrey Konovalov 	return object;
23834303244SAndrey Konovalov }
23934303244SAndrey Konovalov 
24034303244SAndrey Konovalov void * __must_check __kasan_kmalloc(struct kmem_cache *s, const void *object,
24134303244SAndrey Konovalov 				    size_t size, gfp_t flags);
24234303244SAndrey Konovalov static __always_inline void * __must_check kasan_kmalloc(struct kmem_cache *s,
24334303244SAndrey Konovalov 				const void *object, size_t size, gfp_t flags)
24434303244SAndrey Konovalov {
24534303244SAndrey Konovalov 	if (kasan_enabled())
24634303244SAndrey Konovalov 		return __kasan_kmalloc(s, object, size, flags);
24734303244SAndrey Konovalov 	return (void *)object;
24834303244SAndrey Konovalov }
24934303244SAndrey Konovalov 
25034303244SAndrey Konovalov void * __must_check __kasan_kmalloc_large(const void *ptr,
25134303244SAndrey Konovalov 					  size_t size, gfp_t flags);
25234303244SAndrey Konovalov static __always_inline void * __must_check kasan_kmalloc_large(const void *ptr,
25334303244SAndrey Konovalov 						      size_t size, gfp_t flags)
25434303244SAndrey Konovalov {
25534303244SAndrey Konovalov 	if (kasan_enabled())
25634303244SAndrey Konovalov 		return __kasan_kmalloc_large(ptr, size, flags);
25734303244SAndrey Konovalov 	return (void *)ptr;
25834303244SAndrey Konovalov }
25934303244SAndrey Konovalov 
26034303244SAndrey Konovalov void * __must_check __kasan_krealloc(const void *object,
26134303244SAndrey Konovalov 				     size_t new_size, gfp_t flags);
26234303244SAndrey Konovalov static __always_inline void * __must_check kasan_krealloc(const void *object,
26334303244SAndrey Konovalov 						 size_t new_size, gfp_t flags)
26434303244SAndrey Konovalov {
26534303244SAndrey Konovalov 	if (kasan_enabled())
26634303244SAndrey Konovalov 		return __kasan_krealloc(object, new_size, flags);
26734303244SAndrey Konovalov 	return (void *)object;
26834303244SAndrey Konovalov }
26934303244SAndrey Konovalov 
270611806b4SAndrey Konovalov /*
271611806b4SAndrey Konovalov  * Unlike kasan_check_read/write(), kasan_check_byte() is performed even for
272611806b4SAndrey Konovalov  * the hardware tag-based mode that doesn't rely on compiler instrumentation.
273611806b4SAndrey Konovalov  */
274611806b4SAndrey Konovalov bool __kasan_check_byte(const void *addr, unsigned long ip);
275611806b4SAndrey Konovalov static __always_inline bool kasan_check_byte(const void *addr)
276611806b4SAndrey Konovalov {
277611806b4SAndrey Konovalov 	if (kasan_enabled())
278611806b4SAndrey Konovalov 		return __kasan_check_byte(addr, _RET_IP_);
279611806b4SAndrey Konovalov 	return true;
280611806b4SAndrey Konovalov }
281611806b4SAndrey Konovalov 
282611806b4SAndrey Konovalov 
283b0845ce5SMark Rutland bool kasan_save_enable_multi_shot(void);
284b0845ce5SMark Rutland void kasan_restore_multi_shot(bool enabled);
285b0845ce5SMark Rutland 
2860b24beccSAndrey Ryabinin #else /* CONFIG_KASAN */
2870b24beccSAndrey Ryabinin 
28834303244SAndrey Konovalov static inline bool kasan_enabled(void)
28934303244SAndrey Konovalov {
29034303244SAndrey Konovalov 	return false;
29134303244SAndrey Konovalov }
2921bb5eab3SAndrey Konovalov static inline bool kasan_has_integrated_init(void)
2931bb5eab3SAndrey Konovalov {
2941bb5eab3SAndrey Konovalov 	return false;
2951bb5eab3SAndrey Konovalov }
296e86f8b09SAndrey Konovalov static inline slab_flags_t kasan_never_merge(void)
297e86f8b09SAndrey Konovalov {
298e86f8b09SAndrey Konovalov 	return 0;
299e86f8b09SAndrey Konovalov }
300cebd0eb2SAndrey Konovalov static inline void kasan_unpoison_range(const void *address, size_t size) {}
3011bb5eab3SAndrey Konovalov static inline void kasan_alloc_pages(struct page *page, unsigned int order, bool init) {}
3021bb5eab3SAndrey Konovalov static inline void kasan_free_pages(struct page *page, unsigned int order, bool init) {}
3037ed2f9e6SAlexander Potapenko static inline void kasan_cache_create(struct kmem_cache *cache,
304be4a7988SAlexey Dobriyan 				      unsigned int *size,
305d50112edSAlexey Dobriyan 				      slab_flags_t *flags) {}
30692850134SAndrey Konovalov static inline void kasan_cache_create_kmalloc(struct kmem_cache *cache) {}
30734303244SAndrey Konovalov static inline size_t kasan_metadata_size(struct kmem_cache *cache) { return 0; }
3080316bec2SAndrey Ryabinin static inline void kasan_poison_slab(struct page *page) {}
3090316bec2SAndrey Ryabinin static inline void kasan_unpoison_object_data(struct kmem_cache *cache,
3100316bec2SAndrey Ryabinin 					void *object) {}
3110316bec2SAndrey Ryabinin static inline void kasan_poison_object_data(struct kmem_cache *cache,
3120316bec2SAndrey Ryabinin 					void *object) {}
3130116523cSAndrey Konovalov static inline void *kasan_init_slab_obj(struct kmem_cache *cache,
3140116523cSAndrey Konovalov 				const void *object)
3150116523cSAndrey Konovalov {
3160116523cSAndrey Konovalov 	return (void *)object;
3170116523cSAndrey Konovalov }
318*d57a964eSAndrey Konovalov static inline bool kasan_slab_free(struct kmem_cache *s, void *object, bool init)
3190116523cSAndrey Konovalov {
32034303244SAndrey Konovalov 	return false;
3210116523cSAndrey Konovalov }
322200072ceSAndrey Konovalov static inline void kasan_kfree_large(void *ptr) {}
323027b37b5SAndrey Konovalov static inline void kasan_slab_free_mempool(void *ptr) {}
32434303244SAndrey Konovalov static inline void *kasan_slab_alloc(struct kmem_cache *s, void *object,
325da844b78SAndrey Konovalov 				   gfp_t flags, bool init)
32634303244SAndrey Konovalov {
32734303244SAndrey Konovalov 	return object;
32834303244SAndrey Konovalov }
3290116523cSAndrey Konovalov static inline void *kasan_kmalloc(struct kmem_cache *s, const void *object,
3300116523cSAndrey Konovalov 				size_t size, gfp_t flags)
3310116523cSAndrey Konovalov {
3320116523cSAndrey Konovalov 	return (void *)object;
3330116523cSAndrey Konovalov }
33434303244SAndrey Konovalov static inline void *kasan_kmalloc_large(const void *ptr, size_t size, gfp_t flags)
33534303244SAndrey Konovalov {
33634303244SAndrey Konovalov 	return (void *)ptr;
33734303244SAndrey Konovalov }
3380116523cSAndrey Konovalov static inline void *kasan_krealloc(const void *object, size_t new_size,
3390116523cSAndrey Konovalov 				 gfp_t flags)
3400116523cSAndrey Konovalov {
3410116523cSAndrey Konovalov 	return (void *)object;
3420116523cSAndrey Konovalov }
343611806b4SAndrey Konovalov static inline bool kasan_check_byte(const void *address)
344611806b4SAndrey Konovalov {
345611806b4SAndrey Konovalov 	return true;
346611806b4SAndrey Konovalov }
3479b75a867SAndrey Ryabinin 
3480b24beccSAndrey Ryabinin #endif /* CONFIG_KASAN */
3490b24beccSAndrey Ryabinin 
35002c58773SWalter Wu #if defined(CONFIG_KASAN) && defined(CONFIG_KASAN_STACK)
351d56a9ef8SAndrey Konovalov void kasan_unpoison_task_stack(struct task_struct *task);
352d56a9ef8SAndrey Konovalov #else
353d56a9ef8SAndrey Konovalov static inline void kasan_unpoison_task_stack(struct task_struct *task) {}
354d56a9ef8SAndrey Konovalov #endif
355d56a9ef8SAndrey Konovalov 
3562bd926b4SAndrey Konovalov #ifdef CONFIG_KASAN_GENERIC
3572bd926b4SAndrey Konovalov 
3582bd926b4SAndrey Konovalov void kasan_cache_shrink(struct kmem_cache *cache);
3592bd926b4SAndrey Konovalov void kasan_cache_shutdown(struct kmem_cache *cache);
36026e760c9SWalter Wu void kasan_record_aux_stack(void *ptr);
3612bd926b4SAndrey Konovalov 
3622bd926b4SAndrey Konovalov #else /* CONFIG_KASAN_GENERIC */
3632bd926b4SAndrey Konovalov 
3642bd926b4SAndrey Konovalov static inline void kasan_cache_shrink(struct kmem_cache *cache) {}
3652bd926b4SAndrey Konovalov static inline void kasan_cache_shutdown(struct kmem_cache *cache) {}
36626e760c9SWalter Wu static inline void kasan_record_aux_stack(void *ptr) {}
3672bd926b4SAndrey Konovalov 
3682bd926b4SAndrey Konovalov #endif /* CONFIG_KASAN_GENERIC */
3692bd926b4SAndrey Konovalov 
3702e903b91SAndrey Konovalov #if defined(CONFIG_KASAN_SW_TAGS) || defined(CONFIG_KASAN_HW_TAGS)
3713c9e3aa1SAndrey Konovalov 
372c0054c56SAndrey Konovalov static inline void *kasan_reset_tag(const void *addr)
373c0054c56SAndrey Konovalov {
374c0054c56SAndrey Konovalov 	return (void *)arch_kasan_reset_tag(addr);
375c0054c56SAndrey Konovalov }
3763c9e3aa1SAndrey Konovalov 
37749c6631dSVincenzo Frascino /**
37849c6631dSVincenzo Frascino  * kasan_report - print a report about a bad memory access detected by KASAN
37949c6631dSVincenzo Frascino  * @addr: address of the bad access
38049c6631dSVincenzo Frascino  * @size: size of the bad access
38149c6631dSVincenzo Frascino  * @is_write: whether the bad access is a write or a read
38249c6631dSVincenzo Frascino  * @ip: instruction pointer for the accessibility check or the bad access itself
38349c6631dSVincenzo Frascino  */
3848cceeff4SWalter Wu bool kasan_report(unsigned long addr, size_t size,
38541eea9cdSAndrey Konovalov 		bool is_write, unsigned long ip);
38641eea9cdSAndrey Konovalov 
3872e903b91SAndrey Konovalov #else /* CONFIG_KASAN_SW_TAGS || CONFIG_KASAN_HW_TAGS */
3883c9e3aa1SAndrey Konovalov 
3893c9e3aa1SAndrey Konovalov static inline void *kasan_reset_tag(const void *addr)
3903c9e3aa1SAndrey Konovalov {
3913c9e3aa1SAndrey Konovalov 	return (void *)addr;
3923c9e3aa1SAndrey Konovalov }
3933c9e3aa1SAndrey Konovalov 
3942e903b91SAndrey Konovalov #endif /* CONFIG_KASAN_SW_TAGS || CONFIG_KASAN_HW_TAGS*/
3952e903b91SAndrey Konovalov 
3968f7b5054SVincenzo Frascino #ifdef CONFIG_KASAN_HW_TAGS
3978f7b5054SVincenzo Frascino 
3988f7b5054SVincenzo Frascino void kasan_report_async(void);
3998f7b5054SVincenzo Frascino 
4008f7b5054SVincenzo Frascino #endif /* CONFIG_KASAN_HW_TAGS */
4018f7b5054SVincenzo Frascino 
4022e903b91SAndrey Konovalov #ifdef CONFIG_KASAN_SW_TAGS
4032e903b91SAndrey Konovalov void __init kasan_init_sw_tags(void);
4042e903b91SAndrey Konovalov #else
4052e903b91SAndrey Konovalov static inline void kasan_init_sw_tags(void) { }
4062e903b91SAndrey Konovalov #endif
4072e903b91SAndrey Konovalov 
4082e903b91SAndrey Konovalov #ifdef CONFIG_KASAN_HW_TAGS
4092e903b91SAndrey Konovalov void kasan_init_hw_tags_cpu(void);
4102e903b91SAndrey Konovalov void __init kasan_init_hw_tags(void);
4112e903b91SAndrey Konovalov #else
4122e903b91SAndrey Konovalov static inline void kasan_init_hw_tags_cpu(void) { }
4132e903b91SAndrey Konovalov static inline void kasan_init_hw_tags(void) { }
4142e903b91SAndrey Konovalov #endif
415080eb83fSAndrey Konovalov 
4163c5c3cfbSDaniel Axtens #ifdef CONFIG_KASAN_VMALLOC
4173b1a4a86SAndrey Konovalov 
418d98c9e83SAndrey Ryabinin int kasan_populate_vmalloc(unsigned long addr, unsigned long size);
419d98c9e83SAndrey Ryabinin void kasan_poison_vmalloc(const void *start, unsigned long size);
420d98c9e83SAndrey Ryabinin void kasan_unpoison_vmalloc(const void *start, unsigned long size);
4213c5c3cfbSDaniel Axtens void kasan_release_vmalloc(unsigned long start, unsigned long end,
4223c5c3cfbSDaniel Axtens 			   unsigned long free_region_start,
4233c5c3cfbSDaniel Axtens 			   unsigned long free_region_end);
4243b1a4a86SAndrey Konovalov 
4253b1a4a86SAndrey Konovalov #else /* CONFIG_KASAN_VMALLOC */
4263b1a4a86SAndrey Konovalov 
427d98c9e83SAndrey Ryabinin static inline int kasan_populate_vmalloc(unsigned long start,
428d98c9e83SAndrey Ryabinin 					unsigned long size)
4293c5c3cfbSDaniel Axtens {
4303c5c3cfbSDaniel Axtens 	return 0;
4313c5c3cfbSDaniel Axtens }
4323c5c3cfbSDaniel Axtens 
433d98c9e83SAndrey Ryabinin static inline void kasan_poison_vmalloc(const void *start, unsigned long size)
434d98c9e83SAndrey Ryabinin { }
435d98c9e83SAndrey Ryabinin static inline void kasan_unpoison_vmalloc(const void *start, unsigned long size)
436d98c9e83SAndrey Ryabinin { }
4373c5c3cfbSDaniel Axtens static inline void kasan_release_vmalloc(unsigned long start,
4383c5c3cfbSDaniel Axtens 					 unsigned long end,
4393c5c3cfbSDaniel Axtens 					 unsigned long free_region_start,
4403c5c3cfbSDaniel Axtens 					 unsigned long free_region_end) {}
4413b1a4a86SAndrey Konovalov 
4423b1a4a86SAndrey Konovalov #endif /* CONFIG_KASAN_VMALLOC */
4433b1a4a86SAndrey Konovalov 
4440fea6e9aSAndrey Konovalov #if (defined(CONFIG_KASAN_GENERIC) || defined(CONFIG_KASAN_SW_TAGS)) && \
4450fea6e9aSAndrey Konovalov 		!defined(CONFIG_KASAN_VMALLOC)
4463b1a4a86SAndrey Konovalov 
4473b1a4a86SAndrey Konovalov /*
4483b1a4a86SAndrey Konovalov  * These functions provide a special case to support backing module
4493b1a4a86SAndrey Konovalov  * allocations with real shadow memory. With KASAN vmalloc, the special
4503b1a4a86SAndrey Konovalov  * case is unnecessary, as the work is handled in the generic case.
4513b1a4a86SAndrey Konovalov  */
4523b1a4a86SAndrey Konovalov int kasan_module_alloc(void *addr, size_t size);
4533b1a4a86SAndrey Konovalov void kasan_free_shadow(const struct vm_struct *vm);
4543b1a4a86SAndrey Konovalov 
4550fea6e9aSAndrey Konovalov #else /* (CONFIG_KASAN_GENERIC || CONFIG_KASAN_SW_TAGS) && !CONFIG_KASAN_VMALLOC */
4563b1a4a86SAndrey Konovalov 
4573b1a4a86SAndrey Konovalov static inline int kasan_module_alloc(void *addr, size_t size) { return 0; }
4583b1a4a86SAndrey Konovalov static inline void kasan_free_shadow(const struct vm_struct *vm) {}
4593b1a4a86SAndrey Konovalov 
4600fea6e9aSAndrey Konovalov #endif /* (CONFIG_KASAN_GENERIC || CONFIG_KASAN_SW_TAGS) && !CONFIG_KASAN_VMALLOC */
4613c5c3cfbSDaniel Axtens 
4622f004eeaSJann Horn #ifdef CONFIG_KASAN_INLINE
4632f004eeaSJann Horn void kasan_non_canonical_hook(unsigned long addr);
4642f004eeaSJann Horn #else /* CONFIG_KASAN_INLINE */
4652f004eeaSJann Horn static inline void kasan_non_canonical_hook(unsigned long addr) { }
4662f004eeaSJann Horn #endif /* CONFIG_KASAN_INLINE */
4672f004eeaSJann Horn 
4680b24beccSAndrey Ryabinin #endif /* LINUX_KASAN_H */
469