187 lines
5.5 KiB
C
187 lines
5.5 KiB
C
/* Get abbreviation at given offset.
|
|
Copyright (C) 2003, 2004, 2005, 2006, 2014, 2017 Red Hat, Inc.
|
|
This file is part of elfutils.
|
|
Written by Ulrich Drepper <drepper@redhat.com>, 2003.
|
|
|
|
This file is free software; you can redistribute it and/or modify
|
|
it under the terms of either
|
|
|
|
* the GNU Lesser General Public License as published by the Free
|
|
Software Foundation; either version 3 of the License, or (at
|
|
your option) any later version
|
|
|
|
or
|
|
|
|
* the GNU General Public License as published by the Free
|
|
Software Foundation; either version 2 of the License, or (at
|
|
your option) any later version
|
|
|
|
or both in parallel, as here.
|
|
|
|
elfutils is distributed in the hope that it will be useful, but
|
|
WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
General Public License for more details.
|
|
|
|
You should have received copies of the GNU General Public License and
|
|
the GNU Lesser General Public License along with this program. If
|
|
not, see <http://www.gnu.org/licenses/>. */
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
# include <config.h>
|
|
#endif
|
|
|
|
#include <dwarf.h>
|
|
#include "libdwP.h"
|
|
|
|
|
|
Dwarf_Abbrev *
|
|
internal_function
|
|
__libdw_getabbrev (Dwarf *dbg, struct Dwarf_CU *cu, Dwarf_Off offset,
|
|
size_t *lengthp, Dwarf_Abbrev *result)
|
|
{
|
|
/* Don't fail if there is not .debug_abbrev section. */
|
|
if (dbg->sectiondata[IDX_debug_abbrev] == NULL)
|
|
return NULL;
|
|
|
|
if (offset >= dbg->sectiondata[IDX_debug_abbrev]->d_size)
|
|
{
|
|
__libdw_seterrno (DWARF_E_INVALID_OFFSET);
|
|
return NULL;
|
|
}
|
|
|
|
const unsigned char *abbrevp
|
|
= (unsigned char *) dbg->sectiondata[IDX_debug_abbrev]->d_buf + offset;
|
|
|
|
if (*abbrevp == '\0')
|
|
/* We are past the last entry. */
|
|
return DWARF_END_ABBREV;
|
|
|
|
/* 7.5.3 Abbreviations Tables
|
|
|
|
[...] Each declaration begins with an unsigned LEB128 number
|
|
representing the abbreviation code itself. [...] The
|
|
abbreviation code is followed by another unsigned LEB128
|
|
number that encodes the entry's tag. [...]
|
|
|
|
[...] Following the tag encoding is a 1-byte value that
|
|
determines whether a debugging information entry using this
|
|
abbreviation has child entries or not. [...]
|
|
|
|
[...] Finally, the child encoding is followed by a series of
|
|
attribute specifications. Each attribute specification
|
|
consists of two parts. The first part is an unsigned LEB128
|
|
number representing the attribute's name. The second part is
|
|
an unsigned LEB128 number representing the attribute's form. */
|
|
const unsigned char *end = (dbg->sectiondata[IDX_debug_abbrev]->d_buf
|
|
+ dbg->sectiondata[IDX_debug_abbrev]->d_size);
|
|
const unsigned char *start_abbrevp = abbrevp;
|
|
unsigned int code;
|
|
get_uleb128 (code, abbrevp, end);
|
|
|
|
/* Check whether this code is already in the hash table. */
|
|
bool foundit = false;
|
|
Dwarf_Abbrev *abb = NULL;
|
|
if (cu == NULL
|
|
|| (abb = Dwarf_Abbrev_Hash_find (&cu->abbrev_hash, code)) == NULL)
|
|
{
|
|
if (result == NULL)
|
|
abb = libdw_typed_alloc (dbg, Dwarf_Abbrev);
|
|
else
|
|
abb = result;
|
|
}
|
|
else
|
|
{
|
|
foundit = true;
|
|
|
|
if (unlikely (abb->offset != offset))
|
|
{
|
|
/* A duplicate abbrev code at a different offset,
|
|
that should never happen. */
|
|
invalid:
|
|
if (! foundit)
|
|
libdw_typed_unalloc (dbg, Dwarf_Abbrev);
|
|
__libdw_seterrno (DWARF_E_INVALID_DWARF);
|
|
return NULL;
|
|
}
|
|
|
|
/* If the caller doesn't need the length we are done. */
|
|
if (lengthp == NULL)
|
|
goto out;
|
|
}
|
|
|
|
/* If there is already a value in the hash table we are going to
|
|
overwrite its content. This must not be a problem, since the
|
|
content better be the same. */
|
|
abb->code = code;
|
|
if (abbrevp >= end)
|
|
goto invalid;
|
|
get_uleb128 (abb->tag, abbrevp, end);
|
|
if (abbrevp + 1 >= end)
|
|
goto invalid;
|
|
abb->has_children = *abbrevp++ == DW_CHILDREN_yes;
|
|
abb->attrp = (unsigned char *) abbrevp;
|
|
abb->offset = offset;
|
|
|
|
/* Skip over all the attributes and check rest of the abbrev is valid. */
|
|
unsigned int attrname;
|
|
unsigned int attrform;
|
|
do
|
|
{
|
|
if (abbrevp >= end)
|
|
goto invalid;
|
|
get_uleb128 (attrname, abbrevp, end);
|
|
if (abbrevp >= end)
|
|
goto invalid;
|
|
get_uleb128 (attrform, abbrevp, end);
|
|
if (attrform == DW_FORM_implicit_const)
|
|
{
|
|
int64_t formval __attribute__((__unused__));
|
|
if (abbrevp >= end)
|
|
goto invalid;
|
|
get_sleb128 (formval, abbrevp, end);
|
|
}
|
|
}
|
|
while (attrname != 0 || attrform != 0);
|
|
|
|
/* Return the length to the caller if she asked for it. */
|
|
if (lengthp != NULL)
|
|
*lengthp = abbrevp - start_abbrevp;
|
|
|
|
/* Add the entry to the hash table. */
|
|
if (cu != NULL && ! foundit)
|
|
if (Dwarf_Abbrev_Hash_insert (&cu->abbrev_hash, abb->code, abb) == -1)
|
|
{
|
|
/* The entry was already in the table, remove the one we just
|
|
created and get the one already inserted. */
|
|
libdw_typed_unalloc (dbg, Dwarf_Abbrev);
|
|
abb = Dwarf_Abbrev_Hash_find (&cu->abbrev_hash, code);
|
|
}
|
|
|
|
out:
|
|
return abb;
|
|
}
|
|
|
|
|
|
Dwarf_Abbrev *
|
|
dwarf_getabbrev (Dwarf_Die *die, Dwarf_Off offset, size_t *lengthp)
|
|
{
|
|
if (die == NULL || die->cu == NULL)
|
|
return NULL;
|
|
|
|
Dwarf_CU *cu = die->cu;
|
|
Dwarf *dbg = cu->dbg;
|
|
Dwarf_Off abbrev_offset = cu->orig_abbrev_offset;
|
|
Elf_Data *data = dbg->sectiondata[IDX_debug_abbrev];
|
|
if (data == NULL)
|
|
return NULL;
|
|
|
|
if (offset >= data->d_size - abbrev_offset)
|
|
{
|
|
__libdw_seterrno (DWARF_E_INVALID_OFFSET);
|
|
return NULL;
|
|
}
|
|
|
|
return __libdw_getabbrev (dbg, cu, abbrev_offset + offset, lengthp, NULL);
|
|
}
|