1*5ac3bc71Schristos /* $NetBSD: libdwarf_ranges.c,v 1.5 2024/03/03 17:37:32 christos Exp $ */
2e81373b4Schristos
39dd9d0cfSchristos /*-
49dd9d0cfSchristos * Copyright (c) 2009 Kai Wang
59dd9d0cfSchristos * All rights reserved.
69dd9d0cfSchristos *
79dd9d0cfSchristos * Redistribution and use in source and binary forms, with or without
89dd9d0cfSchristos * modification, are permitted provided that the following conditions
99dd9d0cfSchristos * are met:
109dd9d0cfSchristos * 1. Redistributions of source code must retain the above copyright
119dd9d0cfSchristos * notice, this list of conditions and the following disclaimer.
129dd9d0cfSchristos * 2. Redistributions in binary form must reproduce the above copyright
139dd9d0cfSchristos * notice, this list of conditions and the following disclaimer in the
149dd9d0cfSchristos * documentation and/or other materials provided with the distribution.
159dd9d0cfSchristos *
169dd9d0cfSchristos * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
179dd9d0cfSchristos * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
189dd9d0cfSchristos * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
199dd9d0cfSchristos * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
209dd9d0cfSchristos * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
219dd9d0cfSchristos * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
229dd9d0cfSchristos * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
239dd9d0cfSchristos * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
249dd9d0cfSchristos * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
259dd9d0cfSchristos * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
269dd9d0cfSchristos * SUCH DAMAGE.
279dd9d0cfSchristos */
289dd9d0cfSchristos
299dd9d0cfSchristos #include "_libdwarf.h"
309dd9d0cfSchristos
31*5ac3bc71Schristos __RCSID("$NetBSD: libdwarf_ranges.c,v 1.5 2024/03/03 17:37:32 christos Exp $");
329dd9d0cfSchristos ELFTC_VCSID("Id: libdwarf_ranges.c 2972 2013-12-23 06:46:04Z kaiwang27");
339dd9d0cfSchristos
349dd9d0cfSchristos static int
_dwarf_ranges_parse(Dwarf_Debug dbg,Dwarf_CU cu,Dwarf_Section * ds,uint64_t off,Dwarf_Ranges * rg,Dwarf_Unsigned * cnt)359dd9d0cfSchristos _dwarf_ranges_parse(Dwarf_Debug dbg, Dwarf_CU cu, Dwarf_Section *ds,
369dd9d0cfSchristos uint64_t off, Dwarf_Ranges *rg, Dwarf_Unsigned *cnt)
379dd9d0cfSchristos {
389dd9d0cfSchristos Dwarf_Unsigned start, end;
399dd9d0cfSchristos int i;
409dd9d0cfSchristos
419dd9d0cfSchristos i = 0;
429dd9d0cfSchristos while (off < ds->ds_size) {
439dd9d0cfSchristos
449dd9d0cfSchristos start = dbg->read(ds->ds_data, &off, cu->cu_pointer_size);
459dd9d0cfSchristos end = dbg->read(ds->ds_data, &off, cu->cu_pointer_size);
469dd9d0cfSchristos
479dd9d0cfSchristos if (rg != NULL) {
489dd9d0cfSchristos rg[i].dwr_addr1 = start;
499dd9d0cfSchristos rg[i].dwr_addr2 = end;
509dd9d0cfSchristos if (start == 0 && end == 0)
519dd9d0cfSchristos rg[i].dwr_type = DW_RANGES_END;
529dd9d0cfSchristos else if ((start == ~0U && cu->cu_pointer_size == 4) ||
539dd9d0cfSchristos (start == ~0ULL && cu->cu_pointer_size == 8))
549dd9d0cfSchristos rg[i].dwr_type = DW_RANGES_ADDRESS_SELECTION;
559dd9d0cfSchristos else
569dd9d0cfSchristos rg[i].dwr_type = DW_RANGES_ENTRY;
579dd9d0cfSchristos }
589dd9d0cfSchristos
599dd9d0cfSchristos i++;
609dd9d0cfSchristos
619dd9d0cfSchristos if (start == 0 && end == 0)
629dd9d0cfSchristos break;
639dd9d0cfSchristos }
649dd9d0cfSchristos
659dd9d0cfSchristos if (cnt != NULL)
669dd9d0cfSchristos *cnt = i;
679dd9d0cfSchristos
689dd9d0cfSchristos return (DW_DLE_NONE);
699dd9d0cfSchristos }
709dd9d0cfSchristos
719dd9d0cfSchristos int
_dwarf_ranges_find(Dwarf_Debug dbg,uint64_t off,Dwarf_Rangelist * ret_rl)729dd9d0cfSchristos _dwarf_ranges_find(Dwarf_Debug dbg, uint64_t off, Dwarf_Rangelist *ret_rl)
739dd9d0cfSchristos {
749dd9d0cfSchristos Dwarf_Rangelist rl;
759dd9d0cfSchristos
769dd9d0cfSchristos STAILQ_FOREACH(rl, &dbg->dbg_rllist, rl_next)
779dd9d0cfSchristos if (rl->rl_offset == off)
789dd9d0cfSchristos break;
799dd9d0cfSchristos
809dd9d0cfSchristos if (rl == NULL)
819dd9d0cfSchristos return (DW_DLE_NO_ENTRY);
829dd9d0cfSchristos
839dd9d0cfSchristos if (ret_rl != NULL)
849dd9d0cfSchristos *ret_rl = rl;
859dd9d0cfSchristos
869dd9d0cfSchristos return (DW_DLE_NONE);
879dd9d0cfSchristos }
889dd9d0cfSchristos
899dd9d0cfSchristos void
_dwarf_ranges_cleanup(Dwarf_Debug dbg)909dd9d0cfSchristos _dwarf_ranges_cleanup(Dwarf_Debug dbg)
919dd9d0cfSchristos {
929dd9d0cfSchristos Dwarf_Rangelist rl, trl;
939dd9d0cfSchristos
949dd9d0cfSchristos if (STAILQ_EMPTY(&dbg->dbg_rllist))
959dd9d0cfSchristos return;
969dd9d0cfSchristos
979dd9d0cfSchristos STAILQ_FOREACH_SAFE(rl, &dbg->dbg_rllist, rl_next, trl) {
989dd9d0cfSchristos STAILQ_REMOVE(&dbg->dbg_rllist, rl, _Dwarf_Rangelist, rl_next);
999dd9d0cfSchristos if (rl->rl_rgarray)
1009dd9d0cfSchristos free(rl->rl_rgarray);
1019dd9d0cfSchristos free(rl);
1029dd9d0cfSchristos }
1039dd9d0cfSchristos }
1049dd9d0cfSchristos
1059dd9d0cfSchristos int
_dwarf_ranges_add(Dwarf_Debug dbg,Dwarf_CU cu,uint64_t off,Dwarf_Rangelist * ret_rl,Dwarf_Error * error)1069dd9d0cfSchristos _dwarf_ranges_add(Dwarf_Debug dbg, Dwarf_CU cu, uint64_t off,
1079dd9d0cfSchristos Dwarf_Rangelist *ret_rl, Dwarf_Error *error)
1089dd9d0cfSchristos {
1099dd9d0cfSchristos Dwarf_Section *ds;
1109dd9d0cfSchristos Dwarf_Rangelist rl;
1119dd9d0cfSchristos Dwarf_Unsigned cnt;
1129dd9d0cfSchristos int ret;
1139dd9d0cfSchristos
1149dd9d0cfSchristos if ((ds = _dwarf_find_section(dbg, ".debug_ranges")) == NULL) {
1159dd9d0cfSchristos DWARF_SET_ERROR(dbg, error, DW_DLE_NO_ENTRY);
1169dd9d0cfSchristos return (DW_DLE_NO_ENTRY);
1179dd9d0cfSchristos }
1189dd9d0cfSchristos
1199dd9d0cfSchristos if ((rl = malloc(sizeof(struct _Dwarf_Rangelist))) == NULL) {
1209dd9d0cfSchristos DWARF_SET_ERROR(dbg, error, DW_DLE_MEMORY);
1219dd9d0cfSchristos return (DW_DLE_MEMORY);
1229dd9d0cfSchristos }
1239dd9d0cfSchristos
1249dd9d0cfSchristos rl->rl_offset = off;
1259dd9d0cfSchristos
1269dd9d0cfSchristos ret = _dwarf_ranges_parse(dbg, cu, ds, off, NULL, &cnt);
1279dd9d0cfSchristos if (ret != DW_DLE_NONE) {
1289dd9d0cfSchristos free(rl);
1299dd9d0cfSchristos return (ret);
1309dd9d0cfSchristos }
1319dd9d0cfSchristos
1329dd9d0cfSchristos rl->rl_rglen = cnt;
1339dd9d0cfSchristos if (cnt != 0) {
1349dd9d0cfSchristos if ((rl->rl_rgarray = calloc(cnt, sizeof(Dwarf_Ranges))) ==
1359dd9d0cfSchristos NULL) {
1369dd9d0cfSchristos free(rl);
1379dd9d0cfSchristos DWARF_SET_ERROR(dbg, error, DW_DLE_MEMORY);
1389dd9d0cfSchristos return (DW_DLE_MEMORY);
1399dd9d0cfSchristos }
1409dd9d0cfSchristos
1419dd9d0cfSchristos ret = _dwarf_ranges_parse(dbg, cu, ds, off, rl->rl_rgarray,
1429dd9d0cfSchristos NULL);
1439dd9d0cfSchristos if (ret != DW_DLE_NONE) {
1449dd9d0cfSchristos free(rl->rl_rgarray);
1459dd9d0cfSchristos free(rl);
1469dd9d0cfSchristos return (ret);
1479dd9d0cfSchristos }
1489dd9d0cfSchristos } else
1499dd9d0cfSchristos rl->rl_rgarray = NULL;
1509dd9d0cfSchristos
1519dd9d0cfSchristos STAILQ_INSERT_TAIL(&dbg->dbg_rllist, rl, rl_next);
1529dd9d0cfSchristos *ret_rl = rl;
1539dd9d0cfSchristos
1549dd9d0cfSchristos return (DW_DLE_NONE);
1559dd9d0cfSchristos }
156