xref: /netbsd-src/external/bsd/elftoolchain/dist/libdwarf/libdwarf_ranges.c (revision 5ac3bc719ce6e70593039505b491894133237d12)
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