Re: [PATCH v3] btf_encoder: Teach pahole to store percpu variables in vmlinux BTF.

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

 



Em Tue, Jun 09, 2020 at 11:29:40AM -0300, Arnaldo Carvalho de Melo escreveu:
> Em Mon, Jun 08, 2020 at 10:34:03AM -0700, Hao Luo escreveu:
> > On SMP systems, the global percpu variables are placed in a special
> > '.data..percpu' section, which is stored in a segment whose initial
> > address is set to 0, the addresses of per-CPU variables are relative
> > positive addresses [1].
> > 
> > This patch extracts these variables from vmlinux and places them with
> > their type information in BTF. More specifically, when BTF is encoded,
> > we find the index of the '.data..percpu' section and then traverse
> > the symbol table to find those global objects which are in this section.
> > For each of these objects, we push a BTF_KIND_VAR into the types buffer,
> > and a BTF_VAR_SECINFO into another buffer, percpu_secinfo. When all the
> > CUs have finished processing, we push a BTF_KIND_DATASEC into the
> > btfe->types buffer, followed by the percpu_secinfo's content.
> > 
> > In a v5.7-rc7 linux kernel, I was able to extract 291 such variables.
> > The build time overhead is small and the space overhead is also small.
> 
> Looks good, I'm doing some testing on it now, Andrii, can you provide an
> Acked-by or Reviewed-by?

So, I see these (anon) variables, what are these? for an 5.7 vmlinux:

[acme@five pahole]$ bpftool btf dump file vmlinux | grep -w VAR | tail
[67381] VAR '(anon)' type_id=67175, linkage=static
[67495] VAR 'rt_cache_stat' type_id=67417, linkage=static
[67496] VAR 'rt_uncached_list' type_id=67416, linkage=static
[67857] VAR 'tcp_md5sig_pool' type_id=67788, linkage=static
[67993] VAR 'tsq_tasklet' type_id=67927, linkage=static
[69524] VAR 'xfrm_trans_tasklet' type_id=69502, linkage=static
[70055] VAR 'rt6_uncached_list' type_id=67416, linkage=static
[70609] VAR '(anon)' type_id=1713, linkage=static
[70634] VAR 'hmac_ring' type_id=1909, linkage=static
[71591] VAR 'xskmap_flush_list' type_id=85, linkage=static
[acme@five pahole]$

[acme@five pahole]$ bpftool btf dump file vmlinux | grep -w 1713
[1713] FUNC 'memset' type_id=1712
[7235] VAR '(anon)' type_id=1713, linkage=static
[7236] VAR '(anon)' type_id=1713, linkage=static
[8832] VAR '(anon)' type_id=1713, linkage=static
[14346] VAR '(anon)' type_id=1713, linkage=static
[14347] VAR '(anon)' type_id=1713, linkage=static
[14348] VAR '(anon)' type_id=1713, linkage=static
[14349] VAR '(anon)' type_id=1713, linkage=static
[14350] VAR '(anon)' type_id=1713, linkage=static
[14351] VAR '(anon)' type_id=1713, linkage=static
[14352] VAR '(anon)' type_id=1713, linkage=static
[18903] VAR '(anon)' type_id=1713, linkage=static
[18904] VAR '(anon)' type_id=1713, linkage=static
[23180] VAR '(anon)' type_id=1713, linkage=static
[44605] VAR '(anon)' type_id=1713, linkage=static
[60638] VAR '(anon)' type_id=1713, linkage=static
[60639] VAR '(anon)' type_id=1713, linkage=static
[63869] VAR '(anon)' type_id=1713, linkage=static
[70609] VAR '(anon)' type_id=1713, linkage=static
[acme@five pahole]$

[acme@five pahole]$ bpftool btf dump file vmlinux | grep -m1 -w 1713 -B9
[1710] FUNC_PROTO '(anon)' ret_type_id=96 vlen=3
	'p' type_id=96
	'q' type_id=97
	'size' type_id=49
[1711] FUNC 'memcpy' type_id=1710
[1712] FUNC_PROTO '(anon)' ret_type_id=96 vlen=3
	'p' type_id=96
	'c' type_id=22
	'size' type_id=49
[1713] FUNC 'memset' type_id=1712
[acme@five pahole]$

[acme@five pahole]$ bpftool btf dump file vmlinux | grep -m10 -w 7235 -B5 -A5
	'prec' type_id=22
[7232] FUNC 'arch_show_interrupts' type_id=7231
[7233] FUNC_PROTO '(anon)' ret_type_id=0 vlen=1
	'irq' type_id=10
[7234] FUNC 'ack_bad_irq' type_id=7233
[7235] VAR '(anon)' type_id=1713, linkage=static
[7236] VAR '(anon)' type_id=1713, linkage=static
[7237] ARRAY '(anon)' type_id=233 index_type_id=1 nr_elems=4
[7238] FUNC 'irq_init_percpu_irqstack' type_id=6732
[7239] VAR 'irq_stack_backing_store' type_id=412, linkage=global-alloc
[7240] STRUCT 'estack_pages' size=8 vlen=3
--
	type_id=6739 offset=98096 size=16
	type_id=6737 offset=98112 size=16
	type_id=6736 offset=98128 size=16
	type_id=6764 offset=98144 size=16
	type_id=6763 offset=98160 size=16
	type_id=7235 offset=98176 size=0
	type_id=7330 offset=98192 size=4
	type_id=7329 offset=98200 size=8
	type_id=7328 offset=98208 size=4
	type_id=7325 offset=98216 size=8
	type_id=7327 offset=98224 size=1
[acme@five pahole]$

[acme@five pahole]$ readelf  -wi vmlinux  | grep -m 2 DW_AT_producer
    <1c>   DW_AT_producer    : (indirect string, offset: 0x49): GNU AS 2.32
    <2e>   DW_AT_producer    : (indirect string, offset: 0x1358): GNU C89 9.3.1 20200408 (Red Hat 9.3.1-2) -mno-sse -mno-mmx -mno-sse2 -mno-3dnow -mno-avx -m64 -mno-80387 -mno-fp-ret-in-387 -mpreferred-stack-boundary=3 -mskip-rax-setup -mtune=generic -mno-red-zone -mcmodel=kernel -mindirect-branch=thunk-extern -mindirect-branch-register -mrecord-mcount -mfentry -march=x86-64 -g -O2 -std=gnu90 -fno-strict-aliasing -fno-common -fshort-wchar -fno-PIE -falign-jumps=1 -falign-loops=1 -fno-asynchronous-unwind-tables -fno-jump-tables -fno-delete-null-pointer-checks -fstack-protector-strong -fno-var-tracking-assignments -fno-strict-overflow -fno-merge-all-constants -fmerge-constants -fstack-check=no -fconserve-stack -fcf-protection=none --param allow-store-data-races=0
[acme@five pahole]$
 
> Thanks,
> 
> - Arnaldo
>  
> > Testing:
> > 
> > Before:
> >  $ readelf -SW vmlinux | grep BTF
> >  [25] .BTF              PROGBITS        ffffffff821a905c 13a905c 2d2bf8 00   A  0   0  1
> > 
> > After:
> >  $ pahole -J vmlinux
> >  $ readelf -SW vmlinux  | grep BTF
> >  [25] .BTF              PROGBITS        ffffffff821a905c 13a905c 2d5bca 00   A  0   0  1
> > 
> > Common percpu vars can be found in the BTF section.
> > 
> >  $ bpftool btf dump file vmlinux | grep runqueues
> >  [14098] VAR 'runqueues' type_id=13725, linkage=global-alloc
> > 
> >  $ bpftool btf dump file vmlinux | grep 'cpu_stopper'
> >  [17592] STRUCT 'cpu_stopper' size=72 vlen=5
> >  [17612] VAR 'cpu_stopper' type_id=17592, linkage=static
> > 
> >  $ bpftool btf dump file vmlinux | grep ' DATASEC '
> >  [63652] DATASEC '.data..percpu' size=0 vlen=294
> > 
> > References:
> >  [1] https://lwn.net/Articles/531148/
> > Signed-off-by: Hao Luo <haoluo@xxxxxxxxxx>
> > ---
> > Changelog since v2:
> > - Move finding percpu_shndx and extracting symtab into btfe creation,
> >   so we don't have to allocate a new symtab for each CU.
> > - More debug msg by logging the vars encoded in 'verbose' mode. We
> >   probably don't want to log the symbols that are _not_ encoded,
> >   since that would be too verbose.
> > - Calculate var offsets using 'addr - shdr.sh_addr', so it could be
> >   generalized to other sections in future.
> > - Filter out the symbols that are not STT_OBJECT.
> > - Sort var_secinfos in the DATASEC by their offsets.
> > - Free 'persec_secinfo' buffer and 'symtab' in btfe deletion.
> > - Replace the string ".data..percpu" with a constant PERCPU_SECTION.
> > 
> > Changelog since v1:
> > - Add a ".data..percpu" DATASEC that encodes the found VARs.
> > - Use percpu section's shndx to find the symbols that are percpu variables.
> > - Use the correct type to set VAR's linkage.
> > 
> >  btf_encoder.c | 119 ++++++++++++++++++++++++++++++++++++++++++++++++
> >  dwarves.c     |   6 +++
> >  dwarves.h     |   2 +
> >  libbtf.c      | 123 ++++++++++++++++++++++++++++++++++++++++++++++++++
> >  libbtf.h      |  12 +++++
> >  pahole.c      |   1 +
> >  6 files changed, 263 insertions(+)
> > 
> > diff --git a/btf_encoder.c b/btf_encoder.c
> > index df16ba0..2f98f48 100644
> > --- a/btf_encoder.c
> > +++ b/btf_encoder.c
> > @@ -168,6 +168,27 @@ int btf_encoder__encode()
> >  	return err;
> >  }
> >  
> > +
> > +#define HASHADDR__BITS 8
> > +#define HASHADDR__SIZE (1UL << HASHADDR__BITS)
> > +#define hashaddr__fn(key) hash_64(key, HASHADDR__BITS)
> > +
> > +static struct variable *hashaddr__find_variable(const struct hlist_head hashtable[],
> > +						const uint64_t addr)
> > +{
> > +	struct variable *variable;
> > +	struct hlist_node *pos;
> > +	uint16_t bucket = hashaddr__fn(addr);
> > +	const struct hlist_head *head = &hashtable[bucket];
> > +
> > +	hlist_for_each_entry(variable, pos, head, tool_hnode) {
> > +		if (variable->ip.addr == addr)
> > +			return variable;
> > +	}
> > +
> > +	return NULL;
> > +}
> > +
> >  int cu__encode_btf(struct cu *cu, int verbose)
> >  {
> >  	bool add_index_type = false;
> > @@ -176,6 +197,10 @@ int cu__encode_btf(struct cu *cu, int verbose)
> >  	struct function *fn;
> >  	struct tag *pos;
> >  	int err = 0;
> > +	struct hlist_head hash_addr[HASHADDR__SIZE];
> > +	struct variable *var;
> > +	bool has_global_var = false;
> > +	GElf_Sym sym;
> >  
> >  	if (btfe && strcmp(btfe->filename, cu->filename)) {
> >  		err = btf_encoder__encode();
> > @@ -241,6 +266,100 @@ int cu__encode_btf(struct cu *cu, int verbose)
> >  		}
> >  	}
> >  
> > +
> > +	if (btfe->percpu_shndx == 0 || !btfe->symtab)
> > +		goto out;
> > +
> > +	if (verbose)
> > +		printf("search cu '%s' for percpu global variables.\n", cu->name);
> > +
> > +	/* cache variables' addresses, preparing for searching in symtab. */
> > +	for (core_id = 0; core_id < HASHADDR__SIZE; ++core_id)
> > +		INIT_HLIST_HEAD(&hash_addr[core_id]);
> > +
> > +	cu__for_each_variable(cu, core_id, pos) {
> > +		struct hlist_head *head;
> > +
> > +		var = tag__variable(pos);
> > +		if (var->declaration)
> > +			continue;
> > +		/* percpu variables are allocated in global space */
> > +		if (variable__scope(var) != VSCOPE_GLOBAL)
> > +			continue;
> > +		has_global_var = true;
> > +		head = &hash_addr[hashaddr__fn(var->ip.addr)];
> > +		hlist_add_head(&var->tool_hnode, head);
> > +	}
> > +	if (!has_global_var) {
> > +		if (verbose)
> > +			printf("cu has no global variable defined, skip.\n");
> > +		goto out;
> > +	}
> > +
> > +	/* search within symtab for percpu variables */
> > +	elf_symtab__for_each_symbol(btfe->symtab, core_id, sym) {
> > +		uint32_t linkage, type, size, offset;
> > +		int32_t btf_var_id, btf_var_secinfo_id;
> > +		uint64_t addr;
> > +
> > +		/* compare a symbol's shndx to determine if it's a percpu variable */
> > +		if (elf_sym__section(&sym) != btfe->percpu_shndx)
> > +			continue;
> > +		if (elf_sym__type(&sym) != STT_OBJECT)
> > +			continue;
> > +
> > +		addr = elf_sym__value(&sym);
> > +		/*
> > +		 * Store only those symbols that have allocated space in the percpu section.
> > +		 * This excludes the following three types of symbols:
> > +		 *
> > +		 *  1. __ADDRESSABLE(sym), which are forcely emitted as symbols.
> > +		 *  2. __UNIQUE_ID(prefix), which are introduced to generate unique ids.
> > +		 *  3. __exitcall(fn), functions which are labeled as exit calls.
> > +		 *
> > +		 * In addition, the variables defined using DEFINE_PERCPU_FIRST are
> > +		 * also not included, which currently includes:
> > +		 *
> > +		 *  1. fixed_percpu_data
> > +		 */
> > +		if (!addr)
> > +			continue;
> > +		var = hashaddr__find_variable(hash_addr, addr);
> > +		if (var == NULL)
> > +			continue;
> > +
> > +		if (verbose)
> > +			printf("symbol '%s' of address 0x%lx encoded\n",
> > +			       elf_sym__name(&sym, btfe->symtab), addr);
> > +
> > +		/* add a BTF_KIND_VAR in btfe->types */
> > +		linkage = var->external ? BTF_VAR_GLOBAL_ALLOCATED : BTF_VAR_STATIC;
> > +		type = var->ip.tag.type + type_id_off;
> > +		btf_var_id = btf_elf__add_var_type(btfe, type, var->name, linkage);
> > +		if (btf_var_id < 0) {
> > +			err = -1;
> > +			printf("error: failed to encode variable '%s'\n",
> > +			       variable__name(var, cu));
> > +			break;
> > +		}
> > +
> > +		/*
> > +		 * add a BTF_VAR_SECINFO in btfe->percpu_secinfo, which will be added into
> > +		 * btfe->types later when we add BTF_VAR_DATASEC.
> > +		 */
> > +		size = variable__type_size(var, cu);
> > +		type = btf_var_id;
> > +		offset = addr - btfe->percpu_base_addr;
> > +		btf_var_secinfo_id = btf_elf__add_var_secinfo(&btfe->percpu_secinfo,
> > +							      type, offset, size);
> > +		if (btf_var_secinfo_id < 0) {
> > +			err = -1;
> > +			printf("error: failed to encode var secinfo '%s'\n",
> > +			       variable__name(var, cu));
> > +			break;
> > +		}
> > +	}
> > +
> >  out:
> >  	if (err)
> >  		btf_elf__delete(btfe);
> > diff --git a/dwarves.c b/dwarves.c
> > index eb7885f..141f688 100644
> > --- a/dwarves.c
> > +++ b/dwarves.c
> > @@ -978,6 +978,12 @@ const char *variable__type_name(const struct variable *var,
> >  	return tag != NULL ? tag__name(tag, cu, bf, len, NULL) : NULL;
> >  }
> >  
> > +uint32_t variable__type_size(const struct variable *var, const struct cu *cu)
> > +{
> > +	const struct tag *tag = cu__type(cu, var->ip.tag.type);
> > +	return tag != NULL ? tag__size(tag, cu) : 0;
> > +}
> > +
> >  void class_member__delete(struct class_member *member, struct cu *cu)
> >  {
> >  	obstack_free(&cu->obstack, member);
> > diff --git a/dwarves.h b/dwarves.h
> > index a772e57..e4b0255 100644
> > --- a/dwarves.h
> > +++ b/dwarves.h
> > @@ -672,6 +672,8 @@ const char *variable__name(const struct variable *var, const struct cu *cu);
> >  const char *variable__type_name(const struct variable *var,
> >  				const struct cu *cu, char *bf, size_t len);
> >  
> > +uint32_t variable__type_size(const struct variable *var, const struct cu *cu);
> > +
> >  struct lexblock {
> >  	struct ip_tag	 ip;
> >  	struct list_head tags;
> > diff --git a/libbtf.c b/libbtf.c
> > index 2fbce40..a7dc5fd 100644
> > --- a/libbtf.c
> > +++ b/libbtf.c
> > @@ -25,6 +25,7 @@
> >  #include "dutil.h"
> >  #include "gobuffer.h"
> >  #include "dwarves.h"
> > +#include "elf_symtab.h"
> >  
> >  #define BTF_INFO_ENCODE(kind, kind_flag, vlen)				\
> >  	((!!(kind_flag) << 31) | ((kind) << 24) | ((vlen) & BTF_MAX_VLEN))
> > @@ -46,8 +47,21 @@ struct btf_array_type {
> >  	struct btf_array array;
> >  };
> >  
> > +struct btf_var_type {
> > +	struct btf_type type;
> > +	struct btf_var var;
> > +};
> > +
> >  uint8_t btf_elf__verbose;
> >  
> > +static int btf_var_secinfo_cmp(const void *a, const void *b)
> > +{
> > +	const struct btf_var_secinfo *av = a;
> > +	const struct btf_var_secinfo *bv = b;
> > +
> > +	return av->offset - bv->offset;
> > +}
> > +
> >  uint32_t btf_elf__get32(struct btf_elf *btfe, uint32_t *p)
> >  {
> >  	uint32_t val = *p;
> > @@ -137,6 +151,8 @@ out:
> >  struct btf_elf *btf_elf__new(const char *filename, Elf *elf)
> >  {
> >  	struct btf_elf *btfe = zalloc(sizeof(*btfe));
> > +	GElf_Shdr shdr;
> > +	Elf_Scn *sec;
> >  
> >  	if (!btfe)
> >  		return NULL;
> > @@ -193,6 +209,26 @@ struct btf_elf *btf_elf__new(const char *filename, Elf *elf)
> >  	default:	 btfe->wordsize = 0; break;
> >  	}
> >  
> > +	btfe->symtab = elf_symtab__new(NULL, btfe->elf, &btfe->ehdr);
> > +	if (!btfe->symtab) {
> > +		if (btf_elf__verbose)
> > +			printf("%s: '%s' doesn't have symtab.\n", __func__,
> > +			       btfe->filename);
> > +		return btfe;
> > +	}
> > +
> > +	/* find percpu section's shndx */
> > +	sec = elf_section_by_name(btfe->elf, &btfe->ehdr, &shdr, PERCPU_SECTION,
> > +				  NULL);
> > +	if (!sec) {
> > +		if (btf_elf__verbose)
> > +			printf("%s: '%s' doesn't have '%s' section\n", __func__,
> > +			       btfe->filename, PERCPU_SECTION);
> > +		return btfe;
> > +	}
> > +	btfe->percpu_shndx = elf_ndxscn(sec);
> > +	btfe->percpu_base_addr = shdr.sh_addr;
> > +
> >  	return btfe;
> >  
> >  errout:
> > @@ -211,7 +247,10 @@ void btf_elf__delete(struct btf_elf *btfe)
> >  			elf_end(btfe->elf);
> >  	}
> >  
> > +	elf_symtab__delete(btfe->symtab);
> > +
> >  	__gobuffer__delete(&btfe->types);
> > +	__gobuffer__delete(&btfe->percpu_secinfo);
> >  	free(btfe->filename);
> >  	free(btfe->data);
> >  	free(btfe);
> > @@ -613,6 +652,86 @@ int32_t btf_elf__add_func_proto(struct btf_elf *btfe, struct ftype *ftype, uint3
> >  	return type_id;
> >  }
> >  
> > +int32_t btf_elf__add_var_type(struct btf_elf *btfe, uint32_t type, uint32_t name_off,
> > +			      uint32_t linkage)
> > +{
> > +	struct btf_var_type t;
> > +
> > +	t.type.name_off = name_off;
> > +	t.type.info = BTF_INFO_ENCODE(BTF_KIND_VAR, 0, 0);
> > +	t.type.type = type;
> > +
> > +	t.var.linkage = linkage;
> > +
> > +	++btfe->type_index;
> > +	if (gobuffer__add(&btfe->types, &t.type, sizeof(t)) < 0) {
> > +		btf_elf__log_type(btfe, &t.type, true, true,
> > +				  "type=%u name=%s Error in adding gobuffer",
> > +				  t.type.type, btf_elf__name_in_gobuf(btfe, t.type.name_off));
> > +		return -1;
> > +	}
> > +
> > +	btf_elf__log_type(btfe, &t.type, false, false, "type=%u name=%s",
> > +			  t.type.type, btf_elf__name_in_gobuf(btfe, t.type.name_off));
> > +
> > +	return btfe->type_index;
> > +}
> > +
> > +int32_t btf_elf__add_var_secinfo(struct gobuffer *buf, uint32_t type,
> > +				 uint32_t offset, uint32_t size)
> > +{
> > +	struct btf_var_secinfo si = {
> > +		.type = type,
> > +		.offset = offset,
> > +		.size = size,
> > +	};
> > +	return gobuffer__add(buf, &si, sizeof(si));
> > +}
> > +
> > +extern struct strings *strings;
> > +
> > +int32_t btf_elf__add_datasec_type(struct btf_elf *btfe, const char *section_name,
> > +				  struct gobuffer *var_secinfo_buf)
> > +{
> > +	struct btf_type type;
> > +	size_t sz = gobuffer__size(var_secinfo_buf);
> > +	uint16_t nr_var_secinfo = sz / sizeof(struct btf_var_secinfo);
> > +	uint32_t name_off;
> > +
> > +	/*
> > +	 * dwarves doesn't store section names in its string table,
> > +	 * so we have to add it by ourselves.
> > +	 */
> > +	name_off = strings__add(strings, section_name);
> > +
> > +	type.name_off = name_off;
> > +	type.info = BTF_INFO_ENCODE(BTF_KIND_DATASEC, 0, nr_var_secinfo);
> > +	type.size = 0;
> > +
> > +	++btfe->type_index;
> > +	if (gobuffer__add(&btfe->types, &type, sizeof(type)) < 0) {
> > +		btf_elf__log_type(btfe, &type, true, true,
> > +				  "name=%s vlen=%u Error in adding datasec",
> > +				  btf_elf__name_in_gobuf(btfe, type.name_off),
> > +				  nr_var_secinfo);
> > +		return -1;
> > +	}
> > +	qsort(var_secinfo_buf->entries, nr_var_secinfo,
> > +	      sizeof(struct btf_var_secinfo), btf_var_secinfo_cmp);
> > +	if (gobuffer__add(&btfe->types, var_secinfo_buf->entries, sz) < 0) {
> > +		btf_elf__log_type(btfe, &type, true, true,
> > +				  "name=%s vlen=%u Error in adding var_secinfo",
> > +				  btf_elf__name_in_gobuf(btfe, type.name_off),
> > +				  nr_var_secinfo);
> > +		return -1;
> > +	}
> > +
> > +	btf_elf__log_type(btfe, &type, false, false, "type=datasec name=%s",
> > +			  btf_elf__name_in_gobuf(btfe, type.name_off));
> > +
> > +	return btfe->type_index;
> > +}
> > +
> >  static int btf_elf__write(const char *filename, struct btf *btf)
> >  {
> >  	GElf_Shdr shdr_mem, *shdr;
> > @@ -727,6 +846,10 @@ int btf_elf__encode(struct btf_elf *btfe, uint8_t flags)
> >  	if (gobuffer__size(&btfe->types) == 0)
> >  		return 0;
> >  
> > +	if (gobuffer__size(&btfe->percpu_secinfo) != 0)
> > +		btf_elf__add_datasec_type(btfe, PERCPU_SECTION,
> > +					  &btfe->percpu_secinfo);
> > +
> >  	btfe->size = sizeof(*hdr) + (gobuffer__size(&btfe->types) + gobuffer__size(btfe->strings));
> >  	btfe->data = zalloc(btfe->size);
> >  
> > diff --git a/libbtf.h b/libbtf.h
> > index f3c8500..be06480 100644
> > --- a/libbtf.h
> > +++ b/libbtf.h
> > @@ -20,8 +20,10 @@ struct btf_elf {
> >  	void		  *priv;
> >  	Elf		  *elf;
> >  	GElf_Ehdr	  ehdr;
> > +	struct elf_symtab *symtab;
> >  	struct gobuffer	  types;
> >  	struct gobuffer   *strings;
> > +	struct gobuffer   percpu_secinfo;
> >  	char		  *filename;
> >  	size_t		  size;
> >  	int		  swapped;
> > @@ -30,11 +32,15 @@ struct btf_elf {
> >  	bool		  is_big_endian;
> >  	bool		  raw_btf; // "/sys/kernel/btf/vmlinux"
> >  	uint32_t	  type_index;
> > +	uint32_t	  percpu_shndx;
> > +	uint64_t	  percpu_base_addr;
> >  };
> >  
> >  extern uint8_t btf_elf__verbose;
> >  #define btf_elf__verbose_log(fmt, ...) { if (btf_elf__verbose) printf(fmt, __VA_ARGS__); }
> >  
> > +#define PERCPU_SECTION ".data..percpu"
> > +
> >  struct base_type;
> >  struct ftype;
> >  
> > @@ -55,6 +61,12 @@ int32_t btf_elf__add_enum(struct btf_elf *btf, uint32_t name, uint32_t size,
> >  int btf_elf__add_enum_val(struct btf_elf *btf, uint32_t name, int32_t value);
> >  int32_t btf_elf__add_func_proto(struct btf_elf *btf, struct ftype *ftype,
> >  				uint32_t type_id_off);
> > +int32_t btf_elf__add_var_type(struct btf_elf *btfe, uint32_t type, uint32_t name_off,
> > +			      uint32_t linkage);
> > +int32_t btf_elf__add_var_secinfo(struct gobuffer *buf, uint32_t type,
> > +				 uint32_t offset, uint32_t size);
> > +int32_t btf_elf__add_datasec_type(struct btf_elf *btfe, const char *section_name,
> > +				  struct gobuffer *var_secinfo_buf);
> >  void btf_elf__set_strings(struct btf_elf *btf, struct gobuffer *strings);
> >  int  btf_elf__encode(struct btf_elf *btf, uint8_t flags);
> >  
> > diff --git a/pahole.c b/pahole.c
> > index e2a081b..8407db9 100644
> > --- a/pahole.c
> > +++ b/pahole.c
> > @@ -1084,6 +1084,7 @@ static error_t pahole__options_parser(int key, char *arg,
> >  	case 'i': find_containers = 1;
> >  		  class_name = arg;			break;
> >  	case 'J': btf_encode = 1;
> > +		  conf_load.get_addr_info = true;
> >  		  no_bitfield_type_recode = true;	break;
> >  	case 'l': conf.show_first_biggest_size_base_type_member = 1;	break;
> >  	case 'M': conf.show_only_data_members = 1;	break;
> > -- 
> > 2.27.0.278.ge193c7cf3a9-goog
> > 
> 
> -- 
> 
> - Arnaldo

-- 

- Arnaldo



[Index of Archives]     [Linux USB Devel]     [Linux Audio Users]     [Yosemite News]     [Linux Kernel]     [Linux SCSI]

  Powered by Linux