/* Find CU for given offset. Copyright (C) 2003, 2004 Red Hat, Inc. Written by Ulrich Drepper <drepper@redhat.com>, 2003. This program is Open Source software; you can redistribute it and/or modify it under the terms of the Open Software License version 1.0 as published by the Open Source Initiative. You should have received a copy of the Open Software License along with this program; if not, you may obtain a copy of the Open Software License version 1.0 from http://www.opensource.org/licenses/osl.php or by writing the Open Source Initiative c/o Lawrence Rosen, Esq., 3001 King Ranch Road, Ukiah, CA 95482. */ #ifdef HAVE_CONFIG_H # include <config.h> #endif #include <assert.h> #include <search.h> #include "libdwP.h" static int findcu_cb (const void *arg1, const void *arg2) { struct Dwarf_CU *cu1 = (struct Dwarf_CU *) arg1; struct Dwarf_CU *cu2 = (struct Dwarf_CU *) arg2; /* Find out which of the two arguments is the search value. It has end offset 0. */ if (cu1->end == 0) { if (cu1->start < cu2->start) return -1; if (cu1->start >= cu2->end) return 1; } else { if (cu2->start < cu1->start) return 1; if (cu2->start >= cu1->end) return -1; } return 0; } struct Dwarf_CU * __libdw_findcu (dbg, start) Dwarf *dbg; Dwarf_Off start; { /* Maybe we already know that CU. */ struct Dwarf_CU fake = { .start = start, .end = 0 }; struct Dwarf_CU **found = tfind (&fake, &dbg->cu_tree, findcu_cb); if (found != NULL) return *found; if (start < dbg->next_cu_offset) { __libdw_seterrno (DWARF_E_INVALID_DWARF); return NULL; } /* No. Then read more CUs. */ while (1) { Dwarf_Off oldoff = dbg->next_cu_offset; uint8_t address_size; uint8_t offset_size; Dwarf_Off abbrev_offset; if (dwarf_nextcu (dbg, oldoff, &dbg->next_cu_offset, NULL, &abbrev_offset, &address_size, &offset_size) != 0) /* No more entries. */ return NULL; /* Create an entry for this CU. */ struct Dwarf_CU *newp = libdw_typed_alloc (dbg, struct Dwarf_CU); newp->dbg = dbg; newp->start = oldoff; newp->end = dbg->next_cu_offset; newp->address_size = address_size; newp->offset_size = offset_size; Dwarf_Abbrev_Hash_init (&newp->abbrev_hash, 41); newp->orig_abbrev_offset = newp->last_abbrev_offset = abbrev_offset; newp->lines = NULL; newp->locs = NULL; /* Add the new entry to the search tree. */ if (tsearch (newp, &dbg->cu_tree, findcu_cb) == NULL) { /* Something went wrong. Unfo the operation. */ dbg->next_cu_offset = oldoff; __libdw_seterrno (DWARF_E_NOMEM); return NULL; } /* Is this the one we are looking for? */ if (start < dbg->next_cu_offset) // XXX Match exact offset. return newp; } /* NOTREACHED */ }