/*- * Copyright (c) 2010 Kai Wang * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * * $Id: die_traverse.c 2084 2011-10-27 04:48:12Z jkoshy $ */ #include #include #include #include "driver.h" static int die_cnt; /* * DIE traverse function shared by test cases. */ static void _die_traverse_recursive(Dwarf_Debug dbg, Dwarf_Die die, void (*die_callback)(Dwarf_Die die)) { Dwarf_Die die0; Dwarf_Off offset; Dwarf_Half tag; Dwarf_Error de; const char *tagname; int r; assert(dbg != NULL && die != NULL && die_callback != NULL); if (dwarf_tag(die, &tag, &de) != DW_DLV_OK) { tet_printf("dwarf_tag failed: %s\n", dwarf_errmsg(de)); result = TET_FAIL; } tagname = NULL; if (dwarf_get_TAG_name(tag, &tagname) != DW_DLV_OK) { tet_infoline("dwarf_get_TAG_name failed"); result = TET_FAIL; } offset = 0; if (dwarf_dieoffset(die, &offset, &de) != DW_DLV_OK) { tet_printf("dwarf_dieoffset failed: %s\n", dwarf_errmsg(de)); result = TET_FAIL; } tet_printf("DIE #%d (%s) [%#x]\n", die_cnt++, tagname, offset); die_callback(die); /* Search children. */ r = dwarf_child(die, &die0, &de); if (r == DW_DLV_ERROR) tet_printf("%s: dwarf_child failed: %s", __func__, dwarf_errmsg(de)); else if (r == DW_DLV_OK) _die_traverse_recursive(dbg, die0, die_callback); /* Search sibling. */ r = dwarf_siblingof(dbg, die, &die0, &de); if (r == DW_DLV_ERROR) tet_printf("%s: dwarf_siblingof failed: %s", __func__, dwarf_errmsg(de)); else if (r == DW_DLV_OK) _die_traverse_recursive(dbg, die0, die_callback); } static void _die_traverse(Dwarf_Debug dbg, void (*die_callback)(Dwarf_Die die)) { Dwarf_Die die; Dwarf_Error de; Dwarf_Unsigned cu_next_offset; assert(dbg != NULL && die_callback != NULL); die_cnt = 0; TS_DWARF_CU_FOREACH(dbg, cu_next_offset, de) { if (dwarf_siblingof(dbg, NULL, &die, &de) != DW_DLV_OK) break; _die_traverse_recursive(dbg, die, die_callback); } }