1*7304104dSAndroid Build Coastguard Worker /* Inspect nvidia extended linemap with dwarf_next_lines.
2*7304104dSAndroid Build Coastguard Worker Copyright (C) 2002, 2004, 2018 Red Hat, Inc.
3*7304104dSAndroid Build Coastguard Worker This file is part of elfutils.
4*7304104dSAndroid Build Coastguard Worker
5*7304104dSAndroid Build Coastguard Worker This file is free software; you can redistribute it and/or modify
6*7304104dSAndroid Build Coastguard Worker it under the terms of the GNU General Public License as published by
7*7304104dSAndroid Build Coastguard Worker the Free Software Foundation; either version 3 of the License, or
8*7304104dSAndroid Build Coastguard Worker (at your option) any later version.
9*7304104dSAndroid Build Coastguard Worker
10*7304104dSAndroid Build Coastguard Worker elfutils is distributed in the hope that it will be useful, but
11*7304104dSAndroid Build Coastguard Worker WITHOUT ANY WARRANTY; without even the implied warranty of
12*7304104dSAndroid Build Coastguard Worker MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13*7304104dSAndroid Build Coastguard Worker GNU General Public License for more details.
14*7304104dSAndroid Build Coastguard Worker
15*7304104dSAndroid Build Coastguard Worker You should have received a copy of the GNU General Public License
16*7304104dSAndroid Build Coastguard Worker along with this program. If not, see <http://www.gnu.org/licenses/>. */
17*7304104dSAndroid Build Coastguard Worker
18*7304104dSAndroid Build Coastguard Worker #ifdef HAVE_CONFIG_H
19*7304104dSAndroid Build Coastguard Worker # include <config.h>
20*7304104dSAndroid Build Coastguard Worker #endif
21*7304104dSAndroid Build Coastguard Worker
22*7304104dSAndroid Build Coastguard Worker #include <fcntl.h>
23*7304104dSAndroid Build Coastguard Worker #include <inttypes.h>
24*7304104dSAndroid Build Coastguard Worker #include <libelf.h>
25*7304104dSAndroid Build Coastguard Worker #include ELFUTILS_HEADER(dw)
26*7304104dSAndroid Build Coastguard Worker #include <stdio.h>
27*7304104dSAndroid Build Coastguard Worker #include <string.h>
28*7304104dSAndroid Build Coastguard Worker #include <unistd.h>
29*7304104dSAndroid Build Coastguard Worker
30*7304104dSAndroid Build Coastguard Worker
31*7304104dSAndroid Build Coastguard Worker int
main(int argc,char * argv[])32*7304104dSAndroid Build Coastguard Worker main (int argc, char *argv[])
33*7304104dSAndroid Build Coastguard Worker {
34*7304104dSAndroid Build Coastguard Worker int result = 0;
35*7304104dSAndroid Build Coastguard Worker int cnt;
36*7304104dSAndroid Build Coastguard Worker
37*7304104dSAndroid Build Coastguard Worker for (cnt = 1; cnt < argc; ++cnt)
38*7304104dSAndroid Build Coastguard Worker {
39*7304104dSAndroid Build Coastguard Worker int fd = open (argv[cnt], O_RDONLY);
40*7304104dSAndroid Build Coastguard Worker
41*7304104dSAndroid Build Coastguard Worker Dwarf *dbg = dwarf_begin (fd, DWARF_C_READ);
42*7304104dSAndroid Build Coastguard Worker if (dbg == NULL)
43*7304104dSAndroid Build Coastguard Worker {
44*7304104dSAndroid Build Coastguard Worker printf ("%s not usable: %s\n", argv[cnt], dwarf_errmsg (-1));
45*7304104dSAndroid Build Coastguard Worker close (fd);
46*7304104dSAndroid Build Coastguard Worker continue;
47*7304104dSAndroid Build Coastguard Worker }
48*7304104dSAndroid Build Coastguard Worker
49*7304104dSAndroid Build Coastguard Worker Dwarf_Off off;
50*7304104dSAndroid Build Coastguard Worker Dwarf_Off next_off = 0;
51*7304104dSAndroid Build Coastguard Worker Dwarf_CU *cu = NULL;
52*7304104dSAndroid Build Coastguard Worker Dwarf_Lines *lb;
53*7304104dSAndroid Build Coastguard Worker size_t nlb;
54*7304104dSAndroid Build Coastguard Worker int res;
55*7304104dSAndroid Build Coastguard Worker while ((res = dwarf_next_lines (dbg, off = next_off, &next_off, &cu,
56*7304104dSAndroid Build Coastguard Worker NULL, NULL, &lb, &nlb)) == 0)
57*7304104dSAndroid Build Coastguard Worker {
58*7304104dSAndroid Build Coastguard Worker printf ("off = %" PRIu64 "\n", off);
59*7304104dSAndroid Build Coastguard Worker printf (" %zu lines\n", nlb);
60*7304104dSAndroid Build Coastguard Worker
61*7304104dSAndroid Build Coastguard Worker for (size_t i = 0; i < nlb; ++i)
62*7304104dSAndroid Build Coastguard Worker {
63*7304104dSAndroid Build Coastguard Worker Dwarf_Line *l = dwarf_onesrcline (lb, i);
64*7304104dSAndroid Build Coastguard Worker if (l == NULL)
65*7304104dSAndroid Build Coastguard Worker {
66*7304104dSAndroid Build Coastguard Worker printf ("%s: cannot get individual line\n", argv[cnt]);
67*7304104dSAndroid Build Coastguard Worker result = 1;
68*7304104dSAndroid Build Coastguard Worker break;
69*7304104dSAndroid Build Coastguard Worker }
70*7304104dSAndroid Build Coastguard Worker
71*7304104dSAndroid Build Coastguard Worker Dwarf_Addr addr;
72*7304104dSAndroid Build Coastguard Worker if (dwarf_lineaddr (l, &addr) != 0)
73*7304104dSAndroid Build Coastguard Worker addr = 0;
74*7304104dSAndroid Build Coastguard Worker const char *file = dwarf_linesrc (l, NULL, NULL);
75*7304104dSAndroid Build Coastguard Worker int line;
76*7304104dSAndroid Build Coastguard Worker if (dwarf_lineno (l, &line) != 0)
77*7304104dSAndroid Build Coastguard Worker line = 0;
78*7304104dSAndroid Build Coastguard Worker
79*7304104dSAndroid Build Coastguard Worker printf ("%" PRIx64 ": %s:%d:", (uint64_t) addr,
80*7304104dSAndroid Build Coastguard Worker file ?: "???", line);
81*7304104dSAndroid Build Coastguard Worker
82*7304104dSAndroid Build Coastguard Worker /* Getting the file path through the Dwarf_Files should
83*7304104dSAndroid Build Coastguard Worker result in the same path. */
84*7304104dSAndroid Build Coastguard Worker Dwarf_Files *files;
85*7304104dSAndroid Build Coastguard Worker size_t idx;
86*7304104dSAndroid Build Coastguard Worker if (dwarf_line_file (l, &files, &idx) != 0)
87*7304104dSAndroid Build Coastguard Worker {
88*7304104dSAndroid Build Coastguard Worker printf ("%s: cannot get file from line (%zd): %s\n",
89*7304104dSAndroid Build Coastguard Worker argv[cnt], i, dwarf_errmsg (-1));
90*7304104dSAndroid Build Coastguard Worker result = 1;
91*7304104dSAndroid Build Coastguard Worker break;
92*7304104dSAndroid Build Coastguard Worker }
93*7304104dSAndroid Build Coastguard Worker const char *path = dwarf_filesrc (files, idx, NULL, NULL);
94*7304104dSAndroid Build Coastguard Worker if ((path == NULL && file != NULL)
95*7304104dSAndroid Build Coastguard Worker || (path != NULL && file == NULL)
96*7304104dSAndroid Build Coastguard Worker || (strcmp (file, path) != 0))
97*7304104dSAndroid Build Coastguard Worker {
98*7304104dSAndroid Build Coastguard Worker printf ("%s: line %zd srcline (%s) != file srcline (%s)\n",
99*7304104dSAndroid Build Coastguard Worker argv[cnt], i, file ?: "???", path ?: "???");
100*7304104dSAndroid Build Coastguard Worker result = 1;
101*7304104dSAndroid Build Coastguard Worker break;
102*7304104dSAndroid Build Coastguard Worker }
103*7304104dSAndroid Build Coastguard Worker
104*7304104dSAndroid Build Coastguard Worker int column;
105*7304104dSAndroid Build Coastguard Worker if (dwarf_linecol (l, &column) != 0)
106*7304104dSAndroid Build Coastguard Worker column = 0;
107*7304104dSAndroid Build Coastguard Worker if (column >= 0)
108*7304104dSAndroid Build Coastguard Worker printf ("%d:", column);
109*7304104dSAndroid Build Coastguard Worker
110*7304104dSAndroid Build Coastguard Worker bool is_stmt;
111*7304104dSAndroid Build Coastguard Worker if (dwarf_linebeginstatement (l, &is_stmt) != 0)
112*7304104dSAndroid Build Coastguard Worker is_stmt = false;
113*7304104dSAndroid Build Coastguard Worker bool end_sequence;
114*7304104dSAndroid Build Coastguard Worker if (dwarf_lineendsequence (l, &end_sequence) != 0)
115*7304104dSAndroid Build Coastguard Worker end_sequence = false;
116*7304104dSAndroid Build Coastguard Worker bool basic_block;
117*7304104dSAndroid Build Coastguard Worker if (dwarf_lineblock (l, &basic_block) != 0)
118*7304104dSAndroid Build Coastguard Worker basic_block = false;
119*7304104dSAndroid Build Coastguard Worker bool prologue_end;
120*7304104dSAndroid Build Coastguard Worker if (dwarf_lineprologueend (l, &prologue_end) != 0)
121*7304104dSAndroid Build Coastguard Worker prologue_end = false;
122*7304104dSAndroid Build Coastguard Worker bool epilogue_begin;
123*7304104dSAndroid Build Coastguard Worker if (dwarf_lineepiloguebegin (l, &epilogue_begin) != 0)
124*7304104dSAndroid Build Coastguard Worker epilogue_begin = false;
125*7304104dSAndroid Build Coastguard Worker printf (" is_stmt:%s, end_seq:%s, bb:%s, prologue:%s, epilogue:%s\n",
126*7304104dSAndroid Build Coastguard Worker is_stmt ? "yes" : "no", end_sequence ? "yes" : "no",
127*7304104dSAndroid Build Coastguard Worker basic_block ? "yes" : "no", prologue_end ? "yes" : "no",
128*7304104dSAndroid Build Coastguard Worker epilogue_begin ? "yes" : "no");
129*7304104dSAndroid Build Coastguard Worker
130*7304104dSAndroid Build Coastguard Worker Dwarf_Line* callee_context = l;
131*7304104dSAndroid Build Coastguard Worker Dwarf_Line* caller_context = dwarf_linecontext (lb, l);
132*7304104dSAndroid Build Coastguard Worker unsigned int depth = 0;
133*7304104dSAndroid Build Coastguard Worker while (caller_context != NULL)
134*7304104dSAndroid Build Coastguard Worker {
135*7304104dSAndroid Build Coastguard Worker depth++;
136*7304104dSAndroid Build Coastguard Worker for (unsigned int x = 0; x < depth; x++)
137*7304104dSAndroid Build Coastguard Worker printf (" ");
138*7304104dSAndroid Build Coastguard Worker
139*7304104dSAndroid Build Coastguard Worker const char *inlined_file = dwarf_linesrc (caller_context,
140*7304104dSAndroid Build Coastguard Worker NULL, NULL);
141*7304104dSAndroid Build Coastguard Worker int inlined_line;
142*7304104dSAndroid Build Coastguard Worker if (dwarf_lineno (caller_context, &inlined_line) != 0)
143*7304104dSAndroid Build Coastguard Worker inlined_line = 0;
144*7304104dSAndroid Build Coastguard Worker
145*7304104dSAndroid Build Coastguard Worker printf ("%s inlined at %s:%d\n",
146*7304104dSAndroid Build Coastguard Worker dwarf_linefunctionname(dbg, callee_context),
147*7304104dSAndroid Build Coastguard Worker inlined_file ?: "???", inlined_line);
148*7304104dSAndroid Build Coastguard Worker
149*7304104dSAndroid Build Coastguard Worker callee_context = caller_context;
150*7304104dSAndroid Build Coastguard Worker caller_context = dwarf_linecontext (lb, callee_context);
151*7304104dSAndroid Build Coastguard Worker }
152*7304104dSAndroid Build Coastguard Worker }
153*7304104dSAndroid Build Coastguard Worker }
154*7304104dSAndroid Build Coastguard Worker
155*7304104dSAndroid Build Coastguard Worker if (res < 0)
156*7304104dSAndroid Build Coastguard Worker {
157*7304104dSAndroid Build Coastguard Worker printf ("dwarf_next_lines failed: %s\n", dwarf_errmsg (-1));
158*7304104dSAndroid Build Coastguard Worker result = 1;
159*7304104dSAndroid Build Coastguard Worker }
160*7304104dSAndroid Build Coastguard Worker
161*7304104dSAndroid Build Coastguard Worker dwarf_end (dbg);
162*7304104dSAndroid Build Coastguard Worker close (fd);
163*7304104dSAndroid Build Coastguard Worker }
164*7304104dSAndroid Build Coastguard Worker
165*7304104dSAndroid Build Coastguard Worker return result;
166*7304104dSAndroid Build Coastguard Worker }
167