drgn/libdrgn/symbol.c
Stephen Brennan 7970a60818 Add methods to return multiple matching symbols
Currently we can lookup symbols by name or address, but this will only
return one symbol, prioritizing the global symbols. However, symbols may
share the same name, and symbols may also overlap address ranges, so
it's possible for searches to return multiple results. Add functions
which can return a list of multiple matching symbols.

Signed-off-by: Stephen Brennan <stephen@brennan.io>
2022-01-15 11:44:33 -08:00

76 lines
1.7 KiB
C

// Copyright (c) Meta Platforms, Inc. and affiliates.
// SPDX-License-Identifier: GPL-3.0-or-later
#include <elf.h>
#include <stdlib.h>
#include <string.h>
#include "drgn.h"
#include "symbol.h"
#include "util.h"
LIBDRGN_PUBLIC void drgn_symbol_destroy(struct drgn_symbol *sym)
{
free(sym);
}
LIBDRGN_PUBLIC void drgn_symbols_destroy(struct drgn_symbol **syms,
size_t count)
{
for (size_t i = 0; i < count; i++)
drgn_symbol_destroy(syms[i]);
free(syms);
}
void drgn_symbol_from_elf(const char *name, uint64_t address,
const GElf_Sym *elf_sym, struct drgn_symbol *ret)
{
ret->name = name;
ret->address = address;
ret->size = elf_sym->st_size;
int binding = GELF_ST_BIND(elf_sym->st_info);
if (binding <= STB_WEAK || binding == STB_GNU_UNIQUE)
ret->binding = binding + 1;
else
ret->binding = DRGN_SYMBOL_BINDING_UNKNOWN;
int type = GELF_ST_TYPE(elf_sym->st_info);
if (type <= STT_TLS || type == STT_GNU_IFUNC)
ret->kind = type;
else
ret->kind = DRGN_SYMBOL_KIND_UNKNOWN;
}
LIBDRGN_PUBLIC const char *drgn_symbol_name(struct drgn_symbol *sym)
{
return sym->name;
}
LIBDRGN_PUBLIC uint64_t drgn_symbol_address(struct drgn_symbol *sym)
{
return sym->address;
}
LIBDRGN_PUBLIC uint64_t drgn_symbol_size(struct drgn_symbol *sym)
{
return sym->size;
}
LIBDRGN_PUBLIC enum drgn_symbol_binding
drgn_symbol_binding(struct drgn_symbol *sym)
{
return sym->binding;
}
LIBDRGN_PUBLIC enum drgn_symbol_kind drgn_symbol_kind(struct drgn_symbol *sym)
{
return sym->kind;
}
LIBDRGN_PUBLIC bool drgn_symbol_eq(struct drgn_symbol *a, struct drgn_symbol *b)
{
return (strcmp(a->name, b->name) == 0 && a->address == b->address &&
a->size == b->size && a->binding == b->binding &&
a->kind == b->kind);
}