/* Test dwarf_cu_info properties. Copyright (C) 2018 Red Hat, Inc. This file is part of elfutils. This file is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 3 of the License, or (at your option) any later version. 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 a copy of the GNU General Public License along with this program. If not, see . */ #ifdef HAVE_CONFIG_H # include #endif #include #include ELFUTILS_HEADER(dw) #include #include #include #include #include #include /* Yeah, lazy, 16K CUs should be enough for everybody... */ #define MAX_UNITS 16384 struct info { int dietag; int subtag; Dwarf_Half version; uint8_t unit_type; uint64_t id; uint8_t addr_size; uint8_t off_size; }; static struct info unit_info[MAX_UNITS]; int main (int argc, char *argv[]) { for (int i = 1; i < argc; i++) { printf ("file: %s\n", argv[i]); int fd = open (argv[i], O_RDONLY); Dwarf *dbg = dwarf_begin (fd, DWARF_C_READ); if (dbg == NULL) { printf ("%s not usable: %s\n", argv[i], dwarf_errmsg (-1)); return -1; } Dwarf_CU *cu = NULL; Dwarf_Half version; Dwarf_Die cudie, subdie; uint8_t unit_type; size_t u, units; u = units = 0; printf ("Iterate getting all info, compare with dwarf_cu_info.\n"); while (dwarf_get_units (dbg, cu, &cu, &version, &unit_type, &cudie, &subdie) == 0) { int dietag = dwarf_tag (&cudie); int subtag = dwarf_tag (&subdie); unit_info[u].dietag = dietag; unit_info[u].subtag = subtag; unit_info[u].version = version; unit_info[u].unit_type = unit_type; printf ("%zu cu dietag: %x, subtag: %x, version %" PRIx32 ", unit_type %" PRIx8 "\n", u, dietag, subtag, version, unit_type); uint64_t unit_id; uint8_t addr_size, off_size; if (dwarf_cu_info (cu, &version, &unit_type, &cudie, &subdie, &unit_id, &addr_size, &off_size) != 0) { printf ("Invalid dwarf_cu_info: %s\n", dwarf_errmsg (-1)); return -1; } dietag = dwarf_tag (&cudie); subtag = dwarf_tag (&subdie); if (unit_info[u].dietag != dietag) { printf("Unequal dietags\n"); return -1; } if (unit_info[u].subtag != subtag) { printf("Unequal subtags\n"); return -1; } if (unit_info[u].version != version) { printf("Unequal versions\n"); return -1; } if (unit_info[u].unit_type != unit_type) { printf("Unequal unit_types\n"); return -1; } unit_info[u].id = unit_id; unit_info[u].addr_size = addr_size; unit_info[u].off_size = off_size; if (unit_type == DW_UT_skeleton) { if (dwarf_cu_info (subdie.cu, &version, &unit_type, &cudie, &subdie, &unit_id, &addr_size, &off_size) != 0) { printf ("Invalid subdie dwarf_cu_info: %s\n", dwarf_errmsg (-1)); return -1; } dietag = dwarf_tag (&cudie); subtag = dwarf_tag (&subdie); printf ("%zu subdietag: %x, subtag: %x, version %" PRIx32 ", unit_type %" PRIx8 "\n", u, dietag, subtag, version, unit_type); /* subdie is now cudie. */ if (unit_info[u].subtag != dietag) { printf ("Inconsistent subdie tag\n"); return -1; } if (unit_info[u].id != unit_id) { printf ("Unequal subdie ids\n"); return -1; } if (unit_info[u].addr_size != addr_size) { printf ("Unequal subdie addr_size\n"); return -1; } if (unit_info[u].off_size != off_size) { printf ("Unequal subdie off_size\n"); return -1; } } if (u >= MAX_UNITS) { printf ("Oops, more than 16K units...\n"); return -1; } u = ++units; } dwarf_end (dbg); close (fd); /* And again... */ printf ("rechecking: %s\n", argv[i]); fd = open (argv[i], O_RDONLY); dbg = dwarf_begin (fd, DWARF_C_READ); if (dbg == NULL) { printf ("%s not usable: %s\n", argv[i], dwarf_errmsg (-1)); return -1; } cu = NULL; u = 0; printf ("Iterate no info, compare recorded info with dwarf_cu_info.\n"); while (dwarf_get_units (dbg, cu, &cu, NULL, NULL, NULL, NULL) == 0) { if (u > units) { printf ("Got too many units???\n"); return -1; } uint64_t unit_id; uint8_t addr_size, off_size; if (dwarf_cu_info (cu, &version, &unit_type, &cudie, &subdie, &unit_id, &addr_size, &off_size) != 0) { printf ("Invalid dwarf_cu_info: %s\n", dwarf_errmsg (-1)); return -1; } int dietag = dwarf_tag (&cudie); int subtag = dwarf_tag (&subdie); printf ("%zu re dietag: %x, subtag: %x, version %" PRIx32 ", unit_type %" PRIx8 "\n", u, dietag, subtag, version, unit_type); if (unit_info[u].dietag != dietag) { printf("Unequal dietags %x != %x\n", unit_info[u].dietag, dietag); return -1; } if (unit_info[u].subtag != subtag) { printf("Unequal subtags\n"); return -1; } if (unit_info[u].version != version) { printf("Unequal versions\n"); return -1; } if (unit_info[u].unit_type != unit_type) { printf("Unequal unit_types\n"); return -1; } if (unit_info[u].id != unit_id) { printf ("Unequal subdie ids\n"); return -1; } if (unit_info[u].addr_size != addr_size) { printf ("Unequal subdie addr_size\n"); return -1; } if (unit_info[u].off_size != off_size) { printf ("Unequal subdie off_size\n"); return -1; } if (unit_type == DW_UT_skeleton) { if (dwarf_cu_info (subdie.cu, &version, &unit_type, &cudie, &subdie, &unit_id, &addr_size, &off_size) != 0) { printf ("Invalid subdie dwarf_cu_info: %s\n", dwarf_errmsg (-1)); return -1; } dietag = dwarf_tag (&cudie); subtag = dwarf_tag (&subdie); printf ("%zu subdietag: %x, subtag: %x, version %" PRIx32 ", unit_type %" PRIx8 "\n", u, dietag, subtag, version, unit_type); /* subdie is now cudie. */ subtag = dwarf_tag (&cudie); if (unit_info[u].subtag != subtag) { printf ("Inconsistent subdie tag\n"); return -1; } if (unit_info[u].id != unit_id) { printf ("Unequal subdie ids\n"); return -1; } if (unit_info[u].addr_size != addr_size) { printf ("Unequal subdie addr_size\n"); return -1; } if (unit_info[u].off_size != off_size) { printf ("Unequal subdie off_size\n"); return -1; } } if (u >= MAX_UNITS) { printf ("Oops, more than 16K units...\n"); return -1; } u++; } if (u != units) { printf ("Got not enough units???\n"); return -1; } dwarf_end (dbg); close (fd); printf ("\n"); } return 0; }