xref: /linux-6.15/include/linux/alloc_tag.h (revision 12ca42c2)
122d407b1SSuren Baghdasaryan /* SPDX-License-Identifier: GPL-2.0 */
222d407b1SSuren Baghdasaryan /*
322d407b1SSuren Baghdasaryan  * allocation tagging
422d407b1SSuren Baghdasaryan  */
522d407b1SSuren Baghdasaryan #ifndef _LINUX_ALLOC_TAG_H
622d407b1SSuren Baghdasaryan #define _LINUX_ALLOC_TAG_H
722d407b1SSuren Baghdasaryan 
822d407b1SSuren Baghdasaryan #include <linux/bug.h>
922d407b1SSuren Baghdasaryan #include <linux/codetag.h>
1022d407b1SSuren Baghdasaryan #include <linux/container_of.h>
1122d407b1SSuren Baghdasaryan #include <linux/preempt.h>
1222d407b1SSuren Baghdasaryan #include <asm/percpu.h>
1322d407b1SSuren Baghdasaryan #include <linux/cpumask.h>
149b0abe79SKent Overstreet #include <linux/smp.h>
1522d407b1SSuren Baghdasaryan #include <linux/static_key.h>
1622d407b1SSuren Baghdasaryan #include <linux/irqflags.h>
1722d407b1SSuren Baghdasaryan 
1822d407b1SSuren Baghdasaryan struct alloc_tag_counters {
1922d407b1SSuren Baghdasaryan 	u64 bytes;
2022d407b1SSuren Baghdasaryan 	u64 calls;
2122d407b1SSuren Baghdasaryan };
2222d407b1SSuren Baghdasaryan 
2322d407b1SSuren Baghdasaryan /*
2422d407b1SSuren Baghdasaryan  * An instance of this structure is created in a special ELF section at every
2522d407b1SSuren Baghdasaryan  * allocation callsite. At runtime, the special section is treated as
2622d407b1SSuren Baghdasaryan  * an array of these. Embedded codetag utilizes codetag framework.
2722d407b1SSuren Baghdasaryan  */
2822d407b1SSuren Baghdasaryan struct alloc_tag {
2922d407b1SSuren Baghdasaryan 	struct codetag			ct;
3022d407b1SSuren Baghdasaryan 	struct alloc_tag_counters __percpu	*counters;
3122d407b1SSuren Baghdasaryan } __aligned(8);
3222d407b1SSuren Baghdasaryan 
334835f747SSuren Baghdasaryan struct alloc_tag_kernel_section {
344835f747SSuren Baghdasaryan 	struct alloc_tag *first_tag;
354835f747SSuren Baghdasaryan 	unsigned long count;
364835f747SSuren Baghdasaryan };
374835f747SSuren Baghdasaryan 
380db6f8d7SSuren Baghdasaryan struct alloc_tag_module_section {
394835f747SSuren Baghdasaryan 	union {
400db6f8d7SSuren Baghdasaryan 		unsigned long start_addr;
414835f747SSuren Baghdasaryan 		struct alloc_tag *first_tag;
424835f747SSuren Baghdasaryan 	};
430db6f8d7SSuren Baghdasaryan 	unsigned long end_addr;
440db6f8d7SSuren Baghdasaryan 	/* used size */
450db6f8d7SSuren Baghdasaryan 	unsigned long size;
460db6f8d7SSuren Baghdasaryan };
470db6f8d7SSuren Baghdasaryan 
48239d6c96SSuren Baghdasaryan #ifdef CONFIG_MEM_ALLOC_PROFILING_DEBUG
49239d6c96SSuren Baghdasaryan 
50239d6c96SSuren Baghdasaryan #define CODETAG_EMPTY	((void *)1)
51239d6c96SSuren Baghdasaryan 
is_codetag_empty(union codetag_ref * ref)52239d6c96SSuren Baghdasaryan static inline bool is_codetag_empty(union codetag_ref *ref)
53239d6c96SSuren Baghdasaryan {
54239d6c96SSuren Baghdasaryan 	return ref->ct == CODETAG_EMPTY;
55239d6c96SSuren Baghdasaryan }
56239d6c96SSuren Baghdasaryan 
set_codetag_empty(union codetag_ref * ref)57239d6c96SSuren Baghdasaryan static inline void set_codetag_empty(union codetag_ref *ref)
58239d6c96SSuren Baghdasaryan {
59239d6c96SSuren Baghdasaryan 	if (ref)
60239d6c96SSuren Baghdasaryan 		ref->ct = CODETAG_EMPTY;
61239d6c96SSuren Baghdasaryan }
62239d6c96SSuren Baghdasaryan 
63239d6c96SSuren Baghdasaryan #else /* CONFIG_MEM_ALLOC_PROFILING_DEBUG */
64239d6c96SSuren Baghdasaryan 
is_codetag_empty(union codetag_ref * ref)65239d6c96SSuren Baghdasaryan static inline bool is_codetag_empty(union codetag_ref *ref) { return false; }
6660da7445SSuren Baghdasaryan 
set_codetag_empty(union codetag_ref * ref)6760da7445SSuren Baghdasaryan static inline void set_codetag_empty(union codetag_ref *ref)
6860da7445SSuren Baghdasaryan {
6960da7445SSuren Baghdasaryan 	if (ref)
7060da7445SSuren Baghdasaryan 		ref->ct = NULL;
7160da7445SSuren Baghdasaryan }
72239d6c96SSuren Baghdasaryan 
73239d6c96SSuren Baghdasaryan #endif /* CONFIG_MEM_ALLOC_PROFILING_DEBUG */
74239d6c96SSuren Baghdasaryan 
7522d407b1SSuren Baghdasaryan #ifdef CONFIG_MEM_ALLOC_PROFILING
7622d407b1SSuren Baghdasaryan 
770db6f8d7SSuren Baghdasaryan #define ALLOC_TAG_SECTION_NAME	"alloc_tags"
780db6f8d7SSuren Baghdasaryan 
791438d349SSuren Baghdasaryan struct codetag_bytes {
801438d349SSuren Baghdasaryan 	struct codetag *ct;
811438d349SSuren Baghdasaryan 	s64 bytes;
821438d349SSuren Baghdasaryan };
831438d349SSuren Baghdasaryan 
841438d349SSuren Baghdasaryan size_t alloc_tag_top_users(struct codetag_bytes *tags, size_t count, bool can_sleep);
851438d349SSuren Baghdasaryan 
ct_to_alloc_tag(struct codetag * ct)8622d407b1SSuren Baghdasaryan static inline struct alloc_tag *ct_to_alloc_tag(struct codetag *ct)
8722d407b1SSuren Baghdasaryan {
8822d407b1SSuren Baghdasaryan 	return container_of(ct, struct alloc_tag, ct);
8922d407b1SSuren Baghdasaryan }
9022d407b1SSuren Baghdasaryan 
9122d407b1SSuren Baghdasaryan #ifdef ARCH_NEEDS_WEAK_PER_CPU
9222d407b1SSuren Baghdasaryan /*
9322d407b1SSuren Baghdasaryan  * When percpu variables are required to be defined as weak, static percpu
9422d407b1SSuren Baghdasaryan  * variables can't be used inside a function (see comments for DECLARE_PER_CPU_SECTION).
95eebc0f48SYu Zhao  * Instead we will account all module allocations to a single counter.
9622d407b1SSuren Baghdasaryan  */
9722d407b1SSuren Baghdasaryan DECLARE_PER_CPU(struct alloc_tag_counters, _shared_alloc_tag);
9822d407b1SSuren Baghdasaryan 
9922d407b1SSuren Baghdasaryan #define DEFINE_ALLOC_TAG(_alloc_tag)						\
10022d407b1SSuren Baghdasaryan 	static struct alloc_tag _alloc_tag __used __aligned(8)			\
1010db6f8d7SSuren Baghdasaryan 	__section(ALLOC_TAG_SECTION_NAME) = {					\
10222d407b1SSuren Baghdasaryan 		.ct = CODE_TAG_INIT,						\
10322d407b1SSuren Baghdasaryan 		.counters = &_shared_alloc_tag };
10422d407b1SSuren Baghdasaryan 
10522d407b1SSuren Baghdasaryan #else /* ARCH_NEEDS_WEAK_PER_CPU */
10622d407b1SSuren Baghdasaryan 
107*12ca42c2SSuren Baghdasaryan #ifdef MODULE
108*12ca42c2SSuren Baghdasaryan 
109*12ca42c2SSuren Baghdasaryan #define DEFINE_ALLOC_TAG(_alloc_tag)						\
110*12ca42c2SSuren Baghdasaryan 	static struct alloc_tag _alloc_tag __used __aligned(8)			\
111*12ca42c2SSuren Baghdasaryan 	__section(ALLOC_TAG_SECTION_NAME) = {					\
112*12ca42c2SSuren Baghdasaryan 		.ct = CODE_TAG_INIT,						\
113*12ca42c2SSuren Baghdasaryan 		.counters = NULL };
114*12ca42c2SSuren Baghdasaryan 
115*12ca42c2SSuren Baghdasaryan #else  /* MODULE */
116*12ca42c2SSuren Baghdasaryan 
11722d407b1SSuren Baghdasaryan #define DEFINE_ALLOC_TAG(_alloc_tag)						\
11822d407b1SSuren Baghdasaryan 	static DEFINE_PER_CPU(struct alloc_tag_counters, _alloc_tag_cntr);	\
11922d407b1SSuren Baghdasaryan 	static struct alloc_tag _alloc_tag __used __aligned(8)			\
1200db6f8d7SSuren Baghdasaryan 	__section(ALLOC_TAG_SECTION_NAME) = {					\
12122d407b1SSuren Baghdasaryan 		.ct = CODE_TAG_INIT,						\
12222d407b1SSuren Baghdasaryan 		.counters = &_alloc_tag_cntr };
12322d407b1SSuren Baghdasaryan 
124*12ca42c2SSuren Baghdasaryan #endif /* MODULE */
125*12ca42c2SSuren Baghdasaryan 
12622d407b1SSuren Baghdasaryan #endif /* ARCH_NEEDS_WEAK_PER_CPU */
12722d407b1SSuren Baghdasaryan 
12822d407b1SSuren Baghdasaryan DECLARE_STATIC_KEY_MAYBE(CONFIG_MEM_ALLOC_PROFILING_ENABLED_BY_DEFAULT,
12922d407b1SSuren Baghdasaryan 			mem_alloc_profiling_key);
13022d407b1SSuren Baghdasaryan 
mem_alloc_profiling_enabled(void)13122d407b1SSuren Baghdasaryan static inline bool mem_alloc_profiling_enabled(void)
13222d407b1SSuren Baghdasaryan {
13322d407b1SSuren Baghdasaryan 	return static_branch_maybe(CONFIG_MEM_ALLOC_PROFILING_ENABLED_BY_DEFAULT,
13422d407b1SSuren Baghdasaryan 				   &mem_alloc_profiling_key);
13522d407b1SSuren Baghdasaryan }
13622d407b1SSuren Baghdasaryan 
alloc_tag_read(struct alloc_tag * tag)13722d407b1SSuren Baghdasaryan static inline struct alloc_tag_counters alloc_tag_read(struct alloc_tag *tag)
13822d407b1SSuren Baghdasaryan {
13922d407b1SSuren Baghdasaryan 	struct alloc_tag_counters v = { 0, 0 };
14022d407b1SSuren Baghdasaryan 	struct alloc_tag_counters *counter;
14122d407b1SSuren Baghdasaryan 	int cpu;
14222d407b1SSuren Baghdasaryan 
14322d407b1SSuren Baghdasaryan 	for_each_possible_cpu(cpu) {
14422d407b1SSuren Baghdasaryan 		counter = per_cpu_ptr(tag->counters, cpu);
14522d407b1SSuren Baghdasaryan 		v.bytes += counter->bytes;
14622d407b1SSuren Baghdasaryan 		v.calls += counter->calls;
14722d407b1SSuren Baghdasaryan 	}
14822d407b1SSuren Baghdasaryan 
14922d407b1SSuren Baghdasaryan 	return v;
15022d407b1SSuren Baghdasaryan }
15122d407b1SSuren Baghdasaryan 
15222d407b1SSuren Baghdasaryan #ifdef CONFIG_MEM_ALLOC_PROFILING_DEBUG
alloc_tag_add_check(union codetag_ref * ref,struct alloc_tag * tag)15322d407b1SSuren Baghdasaryan static inline void alloc_tag_add_check(union codetag_ref *ref, struct alloc_tag *tag)
15422d407b1SSuren Baghdasaryan {
155640a6039SDavid Wang 	WARN_ONCE(ref && ref->ct && !is_codetag_empty(ref),
15622d407b1SSuren Baghdasaryan 		  "alloc_tag was not cleared (got tag for %s:%u)\n",
15722d407b1SSuren Baghdasaryan 		  ref->ct->filename, ref->ct->lineno);
15822d407b1SSuren Baghdasaryan 
1594810a82cSSuren Baghdasaryan 	WARN_ONCE(!tag, "current->alloc_tag not set\n");
16022d407b1SSuren Baghdasaryan }
16122d407b1SSuren Baghdasaryan 
alloc_tag_sub_check(union codetag_ref * ref)16222d407b1SSuren Baghdasaryan static inline void alloc_tag_sub_check(union codetag_ref *ref)
16322d407b1SSuren Baghdasaryan {
16422d407b1SSuren Baghdasaryan 	WARN_ONCE(ref && !ref->ct, "alloc_tag was not set\n");
16522d407b1SSuren Baghdasaryan }
16622d407b1SSuren Baghdasaryan #else
alloc_tag_add_check(union codetag_ref * ref,struct alloc_tag * tag)16722d407b1SSuren Baghdasaryan static inline void alloc_tag_add_check(union codetag_ref *ref, struct alloc_tag *tag) {}
alloc_tag_sub_check(union codetag_ref * ref)16822d407b1SSuren Baghdasaryan static inline void alloc_tag_sub_check(union codetag_ref *ref) {}
16922d407b1SSuren Baghdasaryan #endif
17022d407b1SSuren Baghdasaryan 
17122d407b1SSuren Baghdasaryan /* Caller should verify both ref and tag to be valid */
__alloc_tag_ref_set(union codetag_ref * ref,struct alloc_tag * tag)172f4657e16SHao Ge static inline bool __alloc_tag_ref_set(union codetag_ref *ref, struct alloc_tag *tag)
17322d407b1SSuren Baghdasaryan {
174e0a955bfSYu Zhao 	alloc_tag_add_check(ref, tag);
175e0a955bfSYu Zhao 	if (!ref || !tag)
176f4657e16SHao Ge 		return false;
177e0a955bfSYu Zhao 
17822d407b1SSuren Baghdasaryan 	ref->ct = &tag->ct;
179f4657e16SHao Ge 	return true;
180e0a955bfSYu Zhao }
181e0a955bfSYu Zhao 
alloc_tag_ref_set(union codetag_ref * ref,struct alloc_tag * tag)182f4657e16SHao Ge static inline bool alloc_tag_ref_set(union codetag_ref *ref, struct alloc_tag *tag)
183e0a955bfSYu Zhao {
184f4657e16SHao Ge 	if (unlikely(!__alloc_tag_ref_set(ref, tag)))
185f4657e16SHao Ge 		return false;
186f4657e16SHao Ge 
18722d407b1SSuren Baghdasaryan 	/*
18822d407b1SSuren Baghdasaryan 	 * We need in increment the call counter every time we have a new
18922d407b1SSuren Baghdasaryan 	 * allocation or when we split a large allocation into smaller ones.
19022d407b1SSuren Baghdasaryan 	 * Each new reference for every sub-allocation needs to increment call
19122d407b1SSuren Baghdasaryan 	 * counter because when we free each part the counter will be decremented.
19222d407b1SSuren Baghdasaryan 	 */
19322d407b1SSuren Baghdasaryan 	this_cpu_inc(tag->counters->calls);
194f4657e16SHao Ge 	return true;
19522d407b1SSuren Baghdasaryan }
19622d407b1SSuren Baghdasaryan 
alloc_tag_add(union codetag_ref * ref,struct alloc_tag * tag,size_t bytes)19722d407b1SSuren Baghdasaryan static inline void alloc_tag_add(union codetag_ref *ref, struct alloc_tag *tag, size_t bytes)
19822d407b1SSuren Baghdasaryan {
199f4657e16SHao Ge 	if (likely(alloc_tag_ref_set(ref, tag)))
20022d407b1SSuren Baghdasaryan 		this_cpu_add(tag->counters->bytes, bytes);
20122d407b1SSuren Baghdasaryan }
20222d407b1SSuren Baghdasaryan 
alloc_tag_sub(union codetag_ref * ref,size_t bytes)20322d407b1SSuren Baghdasaryan static inline void alloc_tag_sub(union codetag_ref *ref, size_t bytes)
20422d407b1SSuren Baghdasaryan {
20522d407b1SSuren Baghdasaryan 	struct alloc_tag *tag;
20622d407b1SSuren Baghdasaryan 
20722d407b1SSuren Baghdasaryan 	alloc_tag_sub_check(ref);
20822d407b1SSuren Baghdasaryan 	if (!ref || !ref->ct)
20922d407b1SSuren Baghdasaryan 		return;
21022d407b1SSuren Baghdasaryan 
211239d6c96SSuren Baghdasaryan 	if (is_codetag_empty(ref)) {
212239d6c96SSuren Baghdasaryan 		ref->ct = NULL;
213239d6c96SSuren Baghdasaryan 		return;
214239d6c96SSuren Baghdasaryan 	}
215239d6c96SSuren Baghdasaryan 
21622d407b1SSuren Baghdasaryan 	tag = ct_to_alloc_tag(ref->ct);
21722d407b1SSuren Baghdasaryan 
21822d407b1SSuren Baghdasaryan 	this_cpu_sub(tag->counters->bytes, bytes);
21922d407b1SSuren Baghdasaryan 	this_cpu_dec(tag->counters->calls);
22022d407b1SSuren Baghdasaryan 
22122d407b1SSuren Baghdasaryan 	ref->ct = NULL;
22222d407b1SSuren Baghdasaryan }
22322d407b1SSuren Baghdasaryan 
2249e54dd8bSKent Overstreet #define alloc_tag_record(p)	((p) = current->alloc_tag)
2259e54dd8bSKent Overstreet 
22622d407b1SSuren Baghdasaryan #else /* CONFIG_MEM_ALLOC_PROFILING */
22722d407b1SSuren Baghdasaryan 
22822d407b1SSuren Baghdasaryan #define DEFINE_ALLOC_TAG(_alloc_tag)
mem_alloc_profiling_enabled(void)22922d407b1SSuren Baghdasaryan static inline bool mem_alloc_profiling_enabled(void) { return false; }
alloc_tag_add(union codetag_ref * ref,struct alloc_tag * tag,size_t bytes)23022d407b1SSuren Baghdasaryan static inline void alloc_tag_add(union codetag_ref *ref, struct alloc_tag *tag,
23122d407b1SSuren Baghdasaryan 				 size_t bytes) {}
alloc_tag_sub(union codetag_ref * ref,size_t bytes)23222d407b1SSuren Baghdasaryan static inline void alloc_tag_sub(union codetag_ref *ref, size_t bytes) {}
2339e54dd8bSKent Overstreet #define alloc_tag_record(p)	do {} while (0)
23422d407b1SSuren Baghdasaryan 
23522d407b1SSuren Baghdasaryan #endif /* CONFIG_MEM_ALLOC_PROFILING */
23622d407b1SSuren Baghdasaryan 
237b951aaffSSuren Baghdasaryan #define alloc_hooks_tag(_tag, _do_alloc)				\
238b951aaffSSuren Baghdasaryan ({									\
23907438779SSuren Baghdasaryan 	typeof(_do_alloc) _res;						\
24007438779SSuren Baghdasaryan 	if (mem_alloc_profiling_enabled()) {				\
24107438779SSuren Baghdasaryan 		struct alloc_tag * __maybe_unused _old;			\
24207438779SSuren Baghdasaryan 		_old = alloc_tag_save(_tag);				\
24307438779SSuren Baghdasaryan 		_res = _do_alloc;					\
244b951aaffSSuren Baghdasaryan 		alloc_tag_restore(_tag, _old);				\
24507438779SSuren Baghdasaryan 	} else								\
24607438779SSuren Baghdasaryan 		_res = _do_alloc;					\
247b951aaffSSuren Baghdasaryan 	_res;								\
248b951aaffSSuren Baghdasaryan })
249b951aaffSSuren Baghdasaryan 
250b951aaffSSuren Baghdasaryan #define alloc_hooks(_do_alloc)						\
251b951aaffSSuren Baghdasaryan ({									\
252b951aaffSSuren Baghdasaryan 	DEFINE_ALLOC_TAG(_alloc_tag);					\
253b951aaffSSuren Baghdasaryan 	alloc_hooks_tag(&_alloc_tag, _do_alloc);			\
254b951aaffSSuren Baghdasaryan })
255b951aaffSSuren Baghdasaryan 
25622d407b1SSuren Baghdasaryan #endif /* _LINUX_ALLOC_TAG_H */
257