blob: 3a31d643b420c0fee4044cec62f746c8e9dbe2f5 [file] [log] [blame]
Arnaldo Carvalho de Melo0954d752009-03-24 16:58:44 -03001/*
Domenico Andreolie714d2e2019-01-15 18:28:24 +01002 SPDX-License-Identifier: GPL-2.0-only
3
Arnaldo Carvalho de Melo0954d752009-03-24 16:58:44 -03004 Copyright (C) 2009 Red Hat Inc.
5 Copyright (C) 2009 Arnaldo Carvalho de Melo <acme@redhat.com>
Arnaldo Carvalho de Melo0954d752009-03-24 16:58:44 -03006*/
7
8#include <malloc.h>
9#include <stdio.h>
10#include <string.h>
11
12#include "dutil.h"
13#include "elf_symtab.h"
14
15#define HASHSYMS__BITS 8
16#define HASHSYMS__SIZE (1UL << HASHSYMS__BITS)
17
Arnaldo Carvalho de Melo24404192021-08-05 16:05:45 -030018struct elf_symtab *elf_symtab__new(const char *name, Elf *elf)
Arnaldo Carvalho de Melo0954d752009-03-24 16:58:44 -030019{
Jiri Olsa1bb49892021-01-24 23:15:19 +010020 size_t symtab_index;
21
Arnaldo Carvalho de Meloba478902009-03-25 18:02:48 -030022 if (name == NULL)
23 name = ".symtab";
24
Arnaldo Carvalho de Melo0954d752009-03-24 16:58:44 -030025 GElf_Shdr shdr;
Arnaldo Carvalho de Melo74c20782021-08-05 10:21:15 -030026 Elf_Scn *sec = elf_section_by_name(elf, &shdr, name, &symtab_index);
Arnaldo Carvalho de Melo0954d752009-03-24 16:58:44 -030027
28 if (sec == NULL)
29 return NULL;
30
31 if (gelf_getshdr(sec, &shdr) == NULL)
32 return NULL;
33
Jiri Olsa1bb49892021-01-24 23:15:19 +010034 struct elf_symtab *symtab = zalloc(sizeof(*symtab));
Arnaldo Carvalho de Meloa54515f2012-08-17 18:47:15 -030035 if (symtab == NULL)
Arnaldo Carvalho de Melo0954d752009-03-24 16:58:44 -030036 return NULL;
37
Arnaldo Carvalho de Meloa54515f2012-08-17 18:47:15 -030038 symtab->name = strdup(name);
39 if (symtab->name == NULL)
Arnaldo Carvalho de Meloba478902009-03-25 18:02:48 -030040 goto out_delete;
41
Arnaldo Carvalho de Meloa54515f2012-08-17 18:47:15 -030042 symtab->syms = elf_getdata(sec, NULL);
43 if (symtab->syms == NULL)
Arnaldo Carvalho de Meloba478902009-03-25 18:02:48 -030044 goto out_free_name;
Arnaldo Carvalho de Melo0954d752009-03-24 16:58:44 -030045
Jiri Olsa1bb49892021-01-24 23:15:19 +010046 /*
47 * This returns extended section index table's
48 * section index, if it exists.
49 */
50 int symtab_xindex = elf_scnshndx(sec);
51
Arnaldo Carvalho de Melo0954d752009-03-24 16:58:44 -030052 sec = elf_getscn(elf, shdr.sh_link);
53 if (sec == NULL)
Arnaldo Carvalho de Meloba478902009-03-25 18:02:48 -030054 goto out_free_name;
Arnaldo Carvalho de Melo0954d752009-03-24 16:58:44 -030055
Arnaldo Carvalho de Meloa54515f2012-08-17 18:47:15 -030056 symtab->symstrs = elf_getdata(sec, NULL);
57 if (symtab->symstrs == NULL)
Arnaldo Carvalho de Meloba478902009-03-25 18:02:48 -030058 goto out_free_name;
Arnaldo Carvalho de Melo0954d752009-03-24 16:58:44 -030059
Jiri Olsa1bb49892021-01-24 23:15:19 +010060 /*
61 * The .symtab section has optional extended section index
62 * table, load its data so it can be used to resolve symbol's
63 * section index.
64 **/
65 if (symtab_xindex > 0) {
66 GElf_Shdr shdr_xindex;
67 Elf_Scn *sec_xindex;
68
69 sec_xindex = elf_getscn(elf, symtab_xindex);
70 if (sec_xindex == NULL)
71 goto out_free_name;
72
73 if (gelf_getshdr(sec_xindex, &shdr_xindex) == NULL)
74 goto out_free_name;
75
76 /* Extra check to verify it's correct type */
77 if (shdr_xindex.sh_type != SHT_SYMTAB_SHNDX)
78 goto out_free_name;
79
80 /* Extra check to verify it belongs to the .symtab */
81 if (symtab_index != shdr_xindex.sh_link)
82 goto out_free_name;
83
84 symtab->syms_sec_idx_table = elf_getdata(elf_getscn(elf, symtab_xindex), NULL);
85 if (symtab->syms_sec_idx_table == NULL)
86 goto out_free_name;
87 }
88
Arnaldo Carvalho de Meloa54515f2012-08-17 18:47:15 -030089 symtab->nr_syms = shdr.sh_size / shdr.sh_entsize;
Arnaldo Carvalho de Melo0954d752009-03-24 16:58:44 -030090
Arnaldo Carvalho de Meloa54515f2012-08-17 18:47:15 -030091 return symtab;
Arnaldo Carvalho de Meloba478902009-03-25 18:02:48 -030092out_free_name:
Arnaldo Carvalho de Melo1105b7d2021-05-27 10:35:14 -030093 zfree(&symtab->name);
Arnaldo Carvalho de Melo0954d752009-03-24 16:58:44 -030094out_delete:
Arnaldo Carvalho de Meloa54515f2012-08-17 18:47:15 -030095 free(symtab);
Arnaldo Carvalho de Melo0954d752009-03-24 16:58:44 -030096 return NULL;
97}
98
Arnaldo Carvalho de Meloa54515f2012-08-17 18:47:15 -030099void elf_symtab__delete(struct elf_symtab *symtab)
Arnaldo Carvalho de Melo0954d752009-03-24 16:58:44 -0300100{
Arnaldo Carvalho de Meloa54515f2012-08-17 18:47:15 -0300101 if (symtab == NULL)
Arnaldo Carvalho de Melo33ed25d2009-04-03 09:41:09 -0300102 return;
Arnaldo Carvalho de Melo1105b7d2021-05-27 10:35:14 -0300103 zfree(&symtab->name);
Arnaldo Carvalho de Meloa54515f2012-08-17 18:47:15 -0300104 free(symtab);
Arnaldo Carvalho de Melo0954d752009-03-24 16:58:44 -0300105}