diff mbox

[v2,02/13] bpf tools: Extract and collect map names from BPF object file

Message ID 1448614067-197576-3-git-send-email-wangnan0@huawei.com
State New
Headers show

Commit Message

Wang Nan Nov. 27, 2015, 8:47 a.m. UTC
This patch collects name of maps in BPF object files and saves them into
'maps' field in 'struct bpf_object'. 'bpf_object__get_map_by_name' is
introduced to retrive fd and definitions of a map through its name.

Signed-off-by: Wang Nan <wangnan0@huawei.com>

Signed-off-by: He Kuang <hekuang@huawei.com>

Cc: Alexei Starovoitov <ast@kernel.org>
Cc: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Masami Hiramatsu <masami.hiramatsu.pt@hitachi.com>
Cc: Namhyung Kim <namhyung@kernel.org>
Cc: Zefan Li <lizefan@huawei.com>
Cc: pi3orama@163.com
---
 tools/lib/bpf/libbpf.c | 65 +++++++++++++++++++++++++++++++++++++++++++++++---
 tools/lib/bpf/libbpf.h |  3 +++
 2 files changed, 65 insertions(+), 3 deletions(-)

-- 
1.8.3.4

--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/

Comments

Wang Nan Nov. 30, 2015, 5 a.m. UTC | #1
On 2015/11/30 0:14, Namhyung Kim wrote:
> Hi Wang,

>

> On Fri, Nov 27, 2015 at 08:47:36AM +0000, Wang Nan wrote:

>> This patch collects name of maps in BPF object files and saves them into

>> 'maps' field in 'struct bpf_object'. 'bpf_object__get_map_by_name' is

>> introduced to retrive fd and definitions of a map through its name.

>>

>> Signed-off-by: Wang Nan <wangnan0@huawei.com>

>> Signed-off-by: He Kuang <hekuang@huawei.com>

>> Cc: Alexei Starovoitov <ast@kernel.org>

>> Cc: Arnaldo Carvalho de Melo <acme@redhat.com>

>> Cc: Masami Hiramatsu <masami.hiramatsu.pt@hitachi.com>

>> Cc: Namhyung Kim <namhyung@kernel.org>

>> Cc: Zefan Li <lizefan@huawei.com>

>> Cc: pi3orama@163.com

>> ---

>>   tools/lib/bpf/libbpf.c | 65 +++++++++++++++++++++++++++++++++++++++++++++++---

>>   tools/lib/bpf/libbpf.h |  3 +++

>>   2 files changed, 65 insertions(+), 3 deletions(-)

>>

>> diff --git a/tools/lib/bpf/libbpf.c b/tools/lib/bpf/libbpf.c

>> index f509825..a298614 100644

>> --- a/tools/lib/bpf/libbpf.c

>> +++ b/tools/lib/bpf/libbpf.c

>> @@ -165,6 +165,7 @@ struct bpf_program {

>>   

>>   struct bpf_map {

>>   	int fd;

>> +	char *name;

>>   	struct bpf_map_def def;

>>   	void *priv;

>>   	bpf_map_clear_priv_t clear_priv;

>> @@ -526,12 +527,46 @@ bpf_object__init_maps(struct bpf_object *obj, void *data,

>>   	return 0;

>>   }

>>   

>> +static void

>> +bpf_object__init_maps_name(struct bpf_object *obj, int maps_shndx)

>> +{

>> +	int i;

>> +	Elf_Data *symbols = obj->efile.symbols;

>> +

>> +	if (!symbols || maps_shndx < 0)

>> +		return;

>> +

>> +	for (i = 0; i < symbols->d_size / sizeof(GElf_Sym); i++) {

>> +		GElf_Sym sym;

>> +		size_t map_idx;

>> +		const char *map_name;

>> +

>> +		if (!gelf_getsym(symbols, i, &sym))

>> +			continue;

>> +		if (sym.st_shndx != maps_shndx)

>> +			continue;

>> +

>> +		map_name = elf_strptr(obj->efile.elf,

>> +				      obj->efile.ehdr.e_shstrndx,

>> +				      sym.st_name);

> It means that each map name is saved in section header string table?


According to elf format specification:

For an symbol table entry, the st_name field "holds an index
into the object file’s symbol string table, which holds the
character representations of the symbol names. If the value
is non-zero, it represents a string table index that gives
the symbol name. Otherwise, the symbol table entry has no
name."

And so called "object file’s symbol string table" is a
section in the object file which index is stored into
ehdr and be loaded during gelf_getehdr(), and its index
would be set to ehdr->e_shstrndx. So I think for each map
its name should be saved in that string table.

>

>> +		map_idx = sym.st_value / sizeof(struct bpf_map_def);

>> +		if (map_idx >= obj->nr_maps) {

>> +			pr_warning("index of map \"%s\" is buggy: %zu > %zu\n",

>> +				   map_name, map_idx, obj->nr_maps);

>> +			continue;

>> +		}

>> +		obj->maps[map_idx].name = strdup(map_name);

> You need to check the return value.


Will send a patch for it.

Thank you.

--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/
Wang Nan Nov. 30, 2015, 9:27 a.m. UTC | #2
On 2015/11/30 16:51, Namhyung Kim wrote:
> On Mon, Nov 30, 2015 at 01:00:46PM +0800, Wangnan (F) wrote:

>>

>> On 2015/11/30 0:14, Namhyung Kim wrote:

>>> Hi Wang,

>>>

>>> On Fri, Nov 27, 2015 at 08:47:36AM +0000, Wang Nan wrote:

>>>> This patch collects name of maps in BPF object files and saves them into

>>>> 'maps' field in 'struct bpf_object'. 'bpf_object__get_map_by_name' is

>>>> introduced to retrive fd and definitions of a map through its name.

>>>>

>>>> Signed-off-by: Wang Nan <wangnan0@huawei.com>

>>>> Signed-off-by: He Kuang <hekuang@huawei.com>

>>>> Cc: Alexei Starovoitov <ast@kernel.org>

>>>> Cc: Arnaldo Carvalho de Melo <acme@redhat.com>

>>>> Cc: Masami Hiramatsu <masami.hiramatsu.pt@hitachi.com>

>>>> Cc: Namhyung Kim <namhyung@kernel.org>

>>>> Cc: Zefan Li <lizefan@huawei.com>

>>>> Cc: pi3orama@163.com

>>>> ---

>>>>   tools/lib/bpf/libbpf.c | 65 +++++++++++++++++++++++++++++++++++++++++++++++---

>>>>   tools/lib/bpf/libbpf.h |  3 +++

>>>>   2 files changed, 65 insertions(+), 3 deletions(-)

>>>>

>>>> diff --git a/tools/lib/bpf/libbpf.c b/tools/lib/bpf/libbpf.c

>>>> index f509825..a298614 100644

>>>> --- a/tools/lib/bpf/libbpf.c

>>>> +++ b/tools/lib/bpf/libbpf.c

>>>> @@ -165,6 +165,7 @@ struct bpf_program {

>>>>   struct bpf_map {

>>>>   	int fd;

>>>> +	char *name;

>>>>   	struct bpf_map_def def;

>>>>   	void *priv;

>>>>   	bpf_map_clear_priv_t clear_priv;

>>>> @@ -526,12 +527,46 @@ bpf_object__init_maps(struct bpf_object *obj, void *data,

>>>>   	return 0;

>>>>   }

>>>> +static void

>>>> +bpf_object__init_maps_name(struct bpf_object *obj, int maps_shndx)

>>>> +{

>>>> +	int i;

>>>> +	Elf_Data *symbols = obj->efile.symbols;

>>>> +

>>>> +	if (!symbols || maps_shndx < 0)

>>>> +		return;

>>>> +

>>>> +	for (i = 0; i < symbols->d_size / sizeof(GElf_Sym); i++) {

>>>> +		GElf_Sym sym;

>>>> +		size_t map_idx;

>>>> +		const char *map_name;

>>>> +

>>>> +		if (!gelf_getsym(symbols, i, &sym))

>>>> +			continue;

>>>> +		if (sym.st_shndx != maps_shndx)

>>>> +			continue;

>>>> +

>>>> +		map_name = elf_strptr(obj->efile.elf,

>>>> +				      obj->efile.ehdr.e_shstrndx,

>>>> +				      sym.st_name);

>>> It means that each map name is saved in section header string table?

>> According to elf format specification:

>>

>> For an symbol table entry, the st_name field "holds an index

>> into the object file’s symbol string table, which holds the

>> character representations of the symbol names. If the value

>> is non-zero, it represents a string table index that gives

>> the symbol name. Otherwise, the symbol table entry has no

>> name."

>>

>> And so called "object file’s symbol string table" is a

>> section in the object file which index is stored into

>> ehdr and be loaded during gelf_getehdr(), and its index

>> would be set to ehdr->e_shstrndx. So I think for each map

>> its name should be saved in that string table.

> AFAIK there're two symbol string tables in a ELF file.  One for

> section headers (.shstrtab) and another for normal symbols (.strtab).

> And ehdr->e_shstrndx is the index of section header string table so

> your code assumes map names are saved in the section header string

> table, right?

>

> Thanks,

> Namhyung


In case of gcc:

$ echo 'int func() {return 0;}' | gcc -x c -c -o ./temp.o -
$ readelf -h ./temp.o
ELF Header:
   Magic:   7f 45 4c 46 02 01 01 00 00 00 00 00 00 00 00 00
   Class:                             ELF64
   Data:                              2's complement, little endian
   Version:                           1 (current)
   OS/ABI:                            UNIX - System V
   ABI Version:                       0
   Type:                              REL (Relocatable file)
   Machine:                           Advanced Micro Devices X86-64
   Version:                           0x1
   Entry point address:               0x0
   Start of program headers:          0 (bytes into file)
   Start of section headers:          240 (bytes into file)
   Flags:                             0x0
   Size of this header:               64 (bytes)
   Size of program headers:           0 (bytes)
   Number of program headers:         0
   Size of section headers:           64 (bytes)
   Number of section headers:         11
   Section header string table index: 8

Let's see what is section 8:

$ readelf -S ./temp.o
   ...
   [ 8] .shstrtab         STRTAB           0000000000000000  00000098
        0000000000000054  0000000000000000           0     0     1
   ...

Yes, in this case it is .shstrtab.

However, this is what I found when using llvm:

$ echo 'int func() {return 0;}' | x86_64-oe-linux-clang -x c -c -o 
./temp.o -
ELF Header:
   Magic:   7f 45 4c 46 02 01 01 03 00 00 00 00 00 00 00 00
   Class:                             ELF64
   Data:                              2's complement, little endian
   Version:                           1 (current)
   OS/ABI:                            UNIX - GNU
   ABI Version:                       0
   Type:                              REL (Relocatable file)
   Machine:                           Advanced Micro Devices X86-64
   Version:                           0x1
   Entry point address:               0x0
   Start of program headers:          0 (bytes into file)
   Start of section headers:          648 (bytes into file)
   Flags:                             0x0
   Size of this header:               64 (bytes)
   Size of program headers:           0 (bytes)
   Number of program headers:         0
   Size of section headers:           64 (bytes)
   Number of section headers:         10
   Section header string table index: 1

$ readelf -S ./temp.o
There are 10 section headers, starting at offset 0x288:

Section Headers:
   [Nr] Name              Type             Address           Offset
        Size              EntSize          Flags  Link  Info  Align
   [ 0]                   NULL             0000000000000000  00000000
        0000000000000000  0000000000000000           0     0     0
   [ 1] .strtab           STRTAB           0000000000000000  00000230
        0000000000000051  0000000000000000           0     0     1

This time it is strtab.

And here is the content of strtab:

$ readelf -p .strtab ./temp.o

String dump of section '.strtab':
   [     1]  .text
   [     7]  .comment
   [    10]  .bss
   [    15]  .note.GNU-stack
   [    25]  .rela.eh_frame
   [    34]  func
   [    39]  .strtab
   [    41]  .symtab
   [    49]  .data
   [    4f]  -


Note that I don't use BPF backend. This is a normal x86 compiling.

So seems it is the default behavior of LLVM.

Thank you.


--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/
Wang Nan Nov. 30, 2015, 9:48 a.m. UTC | #3
On 2015/11/30 17:43, Namhyung Kim wrote:
> On November 30, 2015 6:27:57 PM GMT+09:00, "Wangnan (F)" <wangnan0@huawei.com> wrote:

>>

>> On 2015/11/30 16:51, Namhyung Kim wrote:

>>> On Mon, Nov 30, 2015 at 01:00:46PM +0800, Wangnan (F) wrote:

>>>> On 2015/11/30 0:14, Namhyung Kim wrote:

>>>>> Hi Wang,

>>>>>

>>>>> On Fri, Nov 27, 2015 at 08:47:36AM +0000, Wang Nan wrote:

>>>>>> This patch collects name of maps in BPF object files and saves

>> them into

>>>>>> 'maps' field in 'struct bpf_object'. 'bpf_object__get_map_by_name'

>> is

>>>>>> introduced to retrive fd and definitions of a map through its

>> name.

>>>>>> Signed-off-by: Wang Nan <wangnan0@huawei.com>

>>>>>> Signed-off-by: He Kuang <hekuang@huawei.com>

>>>>>> Cc: Alexei Starovoitov <ast@kernel.org>

>>>>>> Cc: Arnaldo Carvalho de Melo <acme@redhat.com>

>>>>>> Cc: Masami Hiramatsu <masami.hiramatsu.pt@hitachi.com>

>>>>>> Cc: Namhyung Kim <namhyung@kernel.org>

>>>>>> Cc: Zefan Li <lizefan@huawei.com>

>>>>>> Cc: pi3orama@163.com

>>>>>> ---

>>>>>>    tools/lib/bpf/libbpf.c | 65

>> +++++++++++++++++++++++++++++++++++++++++++++++---

>>>>>>    tools/lib/bpf/libbpf.h |  3 +++

>>>>>>    2 files changed, 65 insertions(+), 3 deletions(-)

>>>>>>

>>>>>> diff --git a/tools/lib/bpf/libbpf.c b/tools/lib/bpf/libbpf.c

>>>>>> index f509825..a298614 100644

>>>>>> --- a/tools/lib/bpf/libbpf.c

>>>>>> +++ b/tools/lib/bpf/libbpf.c

>>>>>> @@ -165,6 +165,7 @@ struct bpf_program {

>>>>>>    struct bpf_map {

>>>>>>    	int fd;

>>>>>> +	char *name;

>>>>>>    	struct bpf_map_def def;

>>>>>>    	void *priv;

>>>>>>    	bpf_map_clear_priv_t clear_priv;

>>>>>> @@ -526,12 +527,46 @@ bpf_object__init_maps(struct bpf_object

>> *obj, void *data,

>>>>>>    	return 0;

>>>>>>    }

>>>>>> +static void

>>>>>> +bpf_object__init_maps_name(struct bpf_object *obj, int

>> maps_shndx)

>>>>>> +{

>>>>>> +	int i;

>>>>>> +	Elf_Data *symbols = obj->efile.symbols;

>>>>>> +

>>>>>> +	if (!symbols || maps_shndx < 0)

>>>>>> +		return;

>>>>>> +

>>>>>> +	for (i = 0; i < symbols->d_size / sizeof(GElf_Sym); i++) {

>>>>>> +		GElf_Sym sym;

>>>>>> +		size_t map_idx;

>>>>>> +		const char *map_name;

>>>>>> +

>>>>>> +		if (!gelf_getsym(symbols, i, &sym))

>>>>>> +			continue;

>>>>>> +		if (sym.st_shndx != maps_shndx)

>>>>>> +			continue;

>>>>>> +

>>>>>> +		map_name = elf_strptr(obj->efile.elf,

>>>>>> +				      obj->efile.ehdr.e_shstrndx,

>>>>>> +				      sym.st_name);

>>>>> It means that each map name is saved in section header string

>> table?

>>>> According to elf format specification:

>>>>

>>>> For an symbol table entry, the st_name field "holds an index

>>>> into the object file’s symbol string table, which holds the

>>>> character representations of the symbol names. If the value

>>>> is non-zero, it represents a string table index that gives

>>>> the symbol name. Otherwise, the symbol table entry has no

>>>> name."

>>>>

>>>> And so called "object file’s symbol string table" is a

>>>> section in the object file which index is stored into

>>>> ehdr and be loaded during gelf_getehdr(), and its index

>>>> would be set to ehdr->e_shstrndx. So I think for each map

>>>> its name should be saved in that string table.

>>> AFAIK there're two symbol string tables in a ELF file.  One for

>>> section headers (.shstrtab) and another for normal symbols (.strtab).

>>> And ehdr->e_shstrndx is the index of section header string table so

>>> your code assumes map names are saved in the section header string

>>> table, right?

>>>

>>> Thanks,

>>> Namhyung

>> In case of gcc:

>>

>> $ echo 'int func() {return 0;}' | gcc -x c -c -o ./temp.o -

>> $ readelf -h ./temp.o

>> ELF Header:

>>    Magic:   7f 45 4c 46 02 01 01 00 00 00 00 00 00 00 00 00

>>    Class:                             ELF64

>>    Data:                              2's complement, little endian

>>    Version:                           1 (current)

>>    OS/ABI:                            UNIX - System V

>>    ABI Version:                       0

>>    Type:                              REL (Relocatable file)

>>    Machine:                           Advanced Micro Devices X86-64

>>    Version:                           0x1

>>    Entry point address:               0x0

>>    Start of program headers:          0 (bytes into file)

>>    Start of section headers:          240 (bytes into file)

>>    Flags:                             0x0

>>    Size of this header:               64 (bytes)

>>    Size of program headers:           0 (bytes)

>>    Number of program headers:         0

>>    Size of section headers:           64 (bytes)

>>    Number of section headers:         11

>>    Section header string table index: 8

>>

>> Let's see what is section 8:

>>

>> $ readelf -S ./temp.o

>>    ...

>>    [ 8] .shstrtab         STRTAB           0000000000000000  00000098

>>         0000000000000054  0000000000000000           0     0     1

>>    ...

>>

>> Yes, in this case it is .shstrtab.

>>

>> However, this is what I found when using llvm:

>>

>> $ echo 'int func() {return 0;}' | x86_64-oe-linux-clang -x c -c -o

>> ./temp.o -

>> ELF Header:

>>    Magic:   7f 45 4c 46 02 01 01 03 00 00 00 00 00 00 00 00

>>    Class:                             ELF64

>>    Data:                              2's complement, little endian

>>    Version:                           1 (current)

>>    OS/ABI:                            UNIX - GNU

>>    ABI Version:                       0

>>    Type:                              REL (Relocatable file)

>>    Machine:                           Advanced Micro Devices X86-64

>>    Version:                           0x1

>>    Entry point address:               0x0

>>    Start of program headers:          0 (bytes into file)

>>    Start of section headers:          648 (bytes into file)

>>    Flags:                             0x0

>>    Size of this header:               64 (bytes)

>>    Size of program headers:           0 (bytes)

>>    Number of program headers:         0

>>    Size of section headers:           64 (bytes)

>>    Number of section headers:         10

>>    Section header string table index: 1

>>

>> $ readelf -S ./temp.o

>> There are 10 section headers, starting at offset 0x288:

>>

>> Section Headers:

>>    [Nr] Name              Type             Address           Offset

>>         Size              EntSize          Flags  Link  Info  Align

>>    [ 0]                   NULL             0000000000000000  00000000

>>         0000000000000000  0000000000000000           0     0     0

>>    [ 1] .strtab           STRTAB           0000000000000000  00000230

>>         0000000000000051  0000000000000000           0     0     1

>>

>> This time it is strtab.

>>

>> And here is the content of strtab:

>>

>> $ readelf -p .strtab ./temp.o

>>

>> String dump of section '.strtab':

>>    [     1]  .text

>>    [     7]  .comment

>>    [    10]  .bss

>>    [    15]  .note.GNU-stack

>>    [    25]  .rela.eh_frame

>>    [    34]  func

>>    [    39]  .strtab

>>    [    41]  .symtab

>>    [    49]  .data

>>    [    4f]  -

>>

>>

>> Note that I don't use BPF backend. This is a normal x86 compiling.

>>

>> So seems it is the default behavior of LLVM.

> Ah, didn't know that. So strtab has section header strings as well as normal symbol strings when compiled with LLVM, right?  It'd be great if you add comment about it.


I think technically speaking you are right, because I haven't see any 
documentation
about it, so I don't know the reason why LLVM behave like this, and 
don't know
whether it would change it in future. And it is also possible that we 
will have
other compiler to compile BPF source file. Now I'm reading readelf's 
code and try to
find a canonical way for it.

Thank you for your review.

> Thanks,

> Namhyung

>



--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/
diff mbox

Patch

diff --git a/tools/lib/bpf/libbpf.c b/tools/lib/bpf/libbpf.c
index f509825..a298614 100644
--- a/tools/lib/bpf/libbpf.c
+++ b/tools/lib/bpf/libbpf.c
@@ -165,6 +165,7 @@  struct bpf_program {
 
 struct bpf_map {
 	int fd;
+	char *name;
 	struct bpf_map_def def;
 	void *priv;
 	bpf_map_clear_priv_t clear_priv;
@@ -526,12 +527,46 @@  bpf_object__init_maps(struct bpf_object *obj, void *data,
 	return 0;
 }
 
+static void
+bpf_object__init_maps_name(struct bpf_object *obj, int maps_shndx)
+{
+	int i;
+	Elf_Data *symbols = obj->efile.symbols;
+
+	if (!symbols || maps_shndx < 0)
+		return;
+
+	for (i = 0; i < symbols->d_size / sizeof(GElf_Sym); i++) {
+		GElf_Sym sym;
+		size_t map_idx;
+		const char *map_name;
+
+		if (!gelf_getsym(symbols, i, &sym))
+			continue;
+		if (sym.st_shndx != maps_shndx)
+			continue;
+
+		map_name = elf_strptr(obj->efile.elf,
+				      obj->efile.ehdr.e_shstrndx,
+				      sym.st_name);
+		map_idx = sym.st_value / sizeof(struct bpf_map_def);
+		if (map_idx >= obj->nr_maps) {
+			pr_warning("index of map \"%s\" is buggy: %zu > %zu\n",
+				   map_name, map_idx, obj->nr_maps);
+			continue;
+		}
+		obj->maps[map_idx].name = strdup(map_name);
+		pr_debug("map %zu is \"%s\"\n", map_idx,
+			 obj->maps[map_idx].name);
+	}
+}
+
 static int bpf_object__elf_collect(struct bpf_object *obj)
 {
 	Elf *elf = obj->efile.elf;
 	GElf_Ehdr *ep = &obj->efile.ehdr;
 	Elf_Scn *scn = NULL;
-	int idx = 0, err = 0;
+	int idx = 0, err = 0, maps_shndx = -1;
 
 	/* Elf is corrupted/truncated, avoid calling elf_strptr. */
 	if (!elf_rawdata(elf_getscn(elf, ep->e_shstrndx), NULL)) {
@@ -581,10 +616,11 @@  static int bpf_object__elf_collect(struct bpf_object *obj)
 			err = bpf_object__init_kversion(obj,
 							data->d_buf,
 							data->d_size);
-		else if (strcmp(name, "maps") == 0)
+		else if (strcmp(name, "maps") == 0) {
 			err = bpf_object__init_maps(obj, data->d_buf,
 						    data->d_size);
-		else if (sh.sh_type == SHT_SYMTAB) {
+			maps_shndx = idx;
+		} else if (sh.sh_type == SHT_SYMTAB) {
 			if (obj->efile.symbols) {
 				pr_warning("bpf: multiple SYMTAB in %s\n",
 					   obj->path);
@@ -625,6 +661,9 @@  static int bpf_object__elf_collect(struct bpf_object *obj)
 		if (err)
 			goto out;
 	}
+
+	if (maps_shndx >= 0)
+		bpf_object__init_maps_name(obj, maps_shndx);
 out:
 	return err;
 }
@@ -1086,6 +1125,7 @@  void bpf_object__close(struct bpf_object *obj)
 	bpf_object__unload(obj);
 
 	for (i = 0; i < obj->nr_maps; i++) {
+		zfree(&obj->maps[i].name);
 		if (obj->maps[i].clear_priv)
 			obj->maps[i].clear_priv(&obj->maps[i],
 						obj->maps[i].priv);
@@ -1266,6 +1306,13 @@  int bpf_map__get_def(struct bpf_map *map, struct bpf_map_def *pdef)
 	return 0;
 }
 
+const char *bpf_map__get_name(struct bpf_map *map)
+{
+	if (!map)
+		return NULL;
+	return map->name;
+}
+
 int bpf_map__set_private(struct bpf_map *map, void *priv,
 			 bpf_map_clear_priv_t clear_priv)
 {
@@ -1318,3 +1365,15 @@  bpf_map__next(struct bpf_map *prev, struct bpf_object *obj)
 		return NULL;
 	return &obj->maps[idx];
 }
+
+struct bpf_map *
+bpf_object__get_map_by_name(struct bpf_object *obj, const char *name)
+{
+	struct bpf_map *pos;
+
+	bpf_map__for_each(pos, obj) {
+		if (strcmp(pos->name, name) == 0)
+			return pos;
+	}
+	return NULL;
+}
diff --git a/tools/lib/bpf/libbpf.h b/tools/lib/bpf/libbpf.h
index ef63125..a51594c 100644
--- a/tools/lib/bpf/libbpf.h
+++ b/tools/lib/bpf/libbpf.h
@@ -170,6 +170,8 @@  struct bpf_map_def {
  * it is not a uapi header so no need to consider name clash.
  */
 struct bpf_map;
+struct bpf_map *
+bpf_object__get_map_by_name(struct bpf_object *obj, const char *name);
 
 struct bpf_map *
 bpf_map__next(struct bpf_map *map, struct bpf_object *obj);
@@ -180,6 +182,7 @@  bpf_map__next(struct bpf_map *map, struct bpf_object *obj);
 
 int bpf_map__get_fd(struct bpf_map *map);
 int bpf_map__get_def(struct bpf_map *map, struct bpf_map_def *pdef);
+const char *bpf_map__get_name(struct bpf_map *map);
 
 typedef void (*bpf_map_clear_priv_t)(struct bpf_map *, void *);
 int bpf_map__set_private(struct bpf_map *map, void *priv,