import less(1)
[unleashed/tickless.git] / usr / src / lib / libdwarf / common / dwarf_print_lines.c
blob30c4889ee54246e6cd1fdf1f5bc4f51999fa392a
1 /*
3 Copyright (C) 2000,2002,2004,2005,2006 Silicon Graphics, Inc. All Rights Reserved.
4 Portions Copyright (C) 2007-2010 David Anderson. All Rights Reserved.
6 This program is free software; you can redistribute it and/or modify it
7 under the terms of version 2.1 of the GNU Lesser General Public License
8 as published by the Free Software Foundation.
10 This program is distributed in the hope that it would be useful, but
11 WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
14 Further, this software is distributed without any warranty that it is
15 free of the rightful claim of any third person regarding infringement
16 or the like. Any license provided herein, whether implied or
17 otherwise, applies only to this software file. Patent licenses, if
18 any, provided herein do not apply to combinations of this program with
19 other software, or any other product whatsoever.
21 You should have received a copy of the GNU Lesser General Public
22 License along with this program; if not, write the Free Software
23 Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston MA 02110-1301,
24 USA.
26 Contact information: Silicon Graphics, Inc., 1500 Crittenden Lane,
27 Mountain View, CA 94043, or:
29 http://www.sgi.com
31 For further information regarding this notice, see:
33 http://oss.sgi.com/projects/GenInfo/NoticeExplan
36 /* The address of the Free Software Foundation is
37 Free Software Foundation, Inc., 51 Franklin St, Fifth Floor,
38 Boston, MA 02110-1301, USA.
39 SGI has moved from the Crittenden Lane address.
45 #include "config.h"
46 #include "dwarf_incl.h"
47 #include <stdio.h>
48 #include <time.h>
49 #include "dwarf_line.h"
51 /* FIXME Need to add prologue_end epilogue_begin isa fields. */
52 static void
53 print_line_header(void)
55 printf
56 (" s b e\n"
57 " t l s\n"
58 " m c e\n"
59 " section op col t k q\n"
60 " offset code address file line umn ? ? ?\n");
63 /* FIXME: print new line values: prologue_end epilogue_begin isa */
64 static void
65 print_line_detail(char *prefix,
66 int opcode,
67 Dwarf_Unsigned address,
68 unsigned long file,
69 unsigned long line,
70 unsigned long column,
71 int is_stmt, int basic_block, int end_sequence,
72 int prologue_end, int epilogue_begin, int isa)
74 printf("%-15s %2d 0x%08" DW_PR_DUx " "
75 "%2lu %4lu %2lu %1d %1d %1d\n",
76 prefix,
77 (int) opcode,
78 (Dwarf_Unsigned) address,
79 (unsigned long) file,
80 (unsigned long) line,
81 (unsigned long) column,
82 (int) is_stmt, (int) basic_block, (int) end_sequence);
88 return DW_DLV_OK if ok. else DW_DLV_NO_ENTRY or DW_DLV_ERROR
89 If err_count_out is non-NULL, this is a special 'check'
90 call.
92 int
93 _dwarf_internal_printlines(Dwarf_Die die, Dwarf_Error * error,
94 int * err_count_out, int only_line_header)
96 /*
97 This pointer is used to scan the portion of the .debug_line
98 section for the current cu. */
99 Dwarf_Small *line_ptr = 0;
100 Dwarf_Small *orig_line_ptr = 0;
103 This points to the last byte of the .debug_line portion for the
104 current cu. */
105 Dwarf_Small *line_ptr_end = 0;
108 Pointer to a DW_AT_stmt_list attribute in case it exists in the
109 die. */
110 Dwarf_Attribute stmt_list_attr = 0;
112 /* Pointer to DW_AT_comp_dir attribute in die. */
113 Dwarf_Attribute comp_dir_attr = 0;
115 /* Pointer to name of compilation directory. */
116 Dwarf_Small *comp_dir = NULL;
119 Offset into .debug_line specified by a DW_AT_stmt_list
120 attribute. */
121 Dwarf_Unsigned line_offset = 0;
123 struct Line_Table_Prefix_s prefix;
126 /* These are the state machine state variables. */
127 Dwarf_Addr address = 0;
128 Dwarf_Word file = 1;
129 Dwarf_Word line = 1;
130 Dwarf_Word column = 0;
131 Dwarf_Bool is_stmt = false;
132 Dwarf_Bool basic_block = false;
133 Dwarf_Bool end_sequence = false;
134 Dwarf_Bool prologue_end = false;
135 Dwarf_Bool epilogue_begin = false;
136 Dwarf_Small isa = 0;
139 Dwarf_Sword i=0;
142 This is the current opcode read from the statement program. */
143 Dwarf_Small opcode=0;
147 These variables are used to decode leb128 numbers. Leb128_num
148 holds the decoded number, and leb128_length is its length in
149 bytes. */
150 Dwarf_Word leb128_num=0;
151 Dwarf_Word leb128_length=0;
152 Dwarf_Sword advance_line=0;
153 Dwarf_Half attrform = 0;
155 This is the operand of the latest fixed_advance_pc extended
156 opcode. */
157 Dwarf_Half fixed_advance_pc=0;
159 /* In case there are wierd bytes 'after' the line table
160 * prologue this lets us print something. This is a gcc
161 * compiler bug and we expect the bytes count to be 12.
163 Dwarf_Small* bogus_bytes_ptr = 0;
164 Dwarf_Unsigned bogus_bytes_count = 0;
167 /* The Dwarf_Debug this die belongs to. */
168 Dwarf_Debug dbg=0;
169 int resattr = DW_DLV_ERROR;
170 int lres = DW_DLV_ERROR;
171 int res = DW_DLV_ERROR;
173 /* ***** BEGIN CODE ***** */
175 if (error != NULL) {
176 *error = NULL;
179 CHECK_DIE(die, DW_DLV_ERROR);
180 dbg = die->di_cu_context->cc_dbg;
182 res = _dwarf_load_section(dbg, &dbg->de_debug_line,error);
183 if (res != DW_DLV_OK) {
184 return res;
187 resattr = dwarf_attr(die, DW_AT_stmt_list, &stmt_list_attr, error);
188 if (resattr != DW_DLV_OK) {
189 return resattr;
193 /* The list of relevant FORMs is small.
194 DW_FORM_data4, DW_FORM_data8, DW_FORM_sec_offset
196 lres = dwarf_whatform(stmt_list_attr,&attrform,error);
197 if (lres != DW_DLV_OK) {
198 return lres;
200 if (attrform != DW_FORM_data4 && attrform != DW_FORM_data8 &&
201 attrform != DW_FORM_sec_offset ) {
202 _dwarf_error(dbg, error, DW_DLE_LINE_OFFSET_BAD);
203 return (DW_DLV_ERROR);
205 lres = dwarf_global_formref(stmt_list_attr, &line_offset, error);
206 if (lres != DW_DLV_OK) {
207 return lres;
210 if (line_offset >= dbg->de_debug_line.dss_size) {
211 _dwarf_error(dbg, error, DW_DLE_LINE_OFFSET_BAD);
212 return (DW_DLV_ERROR);
214 orig_line_ptr = dbg->de_debug_line.dss_data;
215 line_ptr = dbg->de_debug_line.dss_data + line_offset;
216 dwarf_dealloc(dbg, stmt_list_attr, DW_DLA_ATTR);
219 If die has DW_AT_comp_dir attribute, get the string that names
220 the compilation directory. */
221 resattr = dwarf_attr(die, DW_AT_comp_dir, &comp_dir_attr, error);
222 if (resattr == DW_DLV_ERROR) {
223 return resattr;
225 if (resattr == DW_DLV_OK) {
226 int cres = DW_DLV_ERROR;
227 char *cdir = 0;
229 cres = dwarf_formstring(comp_dir_attr, &cdir, error);
230 if (cres == DW_DLV_ERROR) {
231 return cres;
232 } else if (cres == DW_DLV_OK) {
233 comp_dir = (Dwarf_Small *) cdir;
236 if (resattr == DW_DLV_OK) {
237 dwarf_dealloc(dbg, comp_dir_attr, DW_DLA_ATTR);
240 dwarf_init_line_table_prefix(&prefix);
242 Dwarf_Small *line_ptr_out = 0;
243 int dres = dwarf_read_line_table_prefix(dbg,
244 line_ptr,dbg->de_debug_line.dss_size - line_offset,
245 &line_ptr_out,
246 &prefix,
247 &bogus_bytes_ptr,
248 &bogus_bytes_count,
249 error,
250 err_count_out);
251 if (dres == DW_DLV_ERROR) {
252 dwarf_free_line_table_prefix(&prefix);
253 return dres;
255 if (dres == DW_DLV_NO_ENTRY) {
256 dwarf_free_line_table_prefix(&prefix);
257 return dres;
259 line_ptr_end = prefix.pf_line_ptr_end;
260 line_ptr = line_ptr_out;
262 if(only_line_header) {
263 /* Just checking for header errors, nothing more here.*/
264 dwarf_free_line_table_prefix(&prefix);
265 return DW_DLV_OK;
269 printf("total line info length %ld bytes, "
270 "line offset 0x%" DW_PR_DUx " %" DW_PR_DSd "\n",
271 (long) prefix.pf_total_length,
272 (Dwarf_Unsigned) line_offset,
273 (Dwarf_Signed) line_offset);
274 printf("line table version %d\n",(int) prefix.pf_version);
275 printf("line table length field length %d prologue length %d\n",
276 (int)prefix.pf_length_field_length,
277 (int)prefix.pf_prologue_length);
278 printf("compilation_directory %s\n",
279 comp_dir ? ((char *) comp_dir) : "");
281 printf(" min instruction length %d\n",
282 (int) prefix.pf_minimum_instruction_length);
283 printf(" default is stmt %d\n", (int)
284 prefix.pf_default_is_stmt);
285 printf(" line base %d\n", (int)
286 prefix.pf_line_base);
287 printf(" line_range %d\n", (int)
288 prefix.pf_line_range);
289 printf(" opcode base %d\n", (int)
290 prefix.pf_opcode_base);
291 printf(" standard opcode count %d\n", (int)
292 prefix.pf_std_op_count);
294 for (i = 1; i < prefix.pf_opcode_base; i++) {
295 printf(" opcode[%2d] length %d\n", (int) i,
296 (int) prefix.pf_opcode_length_table[i - 1]);
298 printf(" include directories count %d\n", (int)
299 prefix.pf_include_directories_count);
302 for (i = 0; i < prefix.pf_include_directories_count; ++i) {
303 printf(" include dir[%d] %s\n",
304 (int) i, prefix.pf_include_directories[i]);
306 printf(" files count %d\n", (int)
307 prefix.pf_files_count);
309 for (i = 0; i < prefix.pf_files_count; ++i) {
310 struct Line_Table_File_Entry_s *lfile =
311 prefix.pf_line_table_file_entries + i;
313 Dwarf_Unsigned tlm2 = lfile->lte_last_modification_time;
314 Dwarf_Unsigned di = lfile->lte_directory_index;
315 Dwarf_Unsigned fl = lfile->lte_length_of_file;
317 printf(" file[%d] %s (file-number: %d) \n",
318 (int) i, (char *) lfile->lte_filename,
319 (int)(i+1));
321 printf(" dir index %d\n", (int) di);
323 time_t tt = (time_t) tlm2;
325 printf(" last time 0x%x %s", /* ctime supplies
326 newline */
327 (unsigned) tlm2, ctime(&tt));
329 printf(" file length %ld 0x%lx\n",
330 (long) fl, (unsigned long) fl);
337 Dwarf_Unsigned offset = 0;
338 if(bogus_bytes_count > 0) {
339 Dwarf_Unsigned wcount = bogus_bytes_count;
340 Dwarf_Unsigned boffset = bogus_bytes_ptr - orig_line_ptr;
341 printf("*** DWARF CHECK: the line table prologue header_length "
342 " is %" DW_PR_DUu " too high, we pretend it is smaller."
343 "Section offset: %" DW_PR_DUu " (0x%" DW_PR_DUx ") ***\n",
344 wcount, boffset,boffset);
345 *err_count_out += 1;
347 offset = line_ptr - orig_line_ptr;
349 printf(" statement prog offset in section: %" DW_PR_DUu " 0x%" DW_PR_DUx "\n",
350 offset, offset);
353 /* Initialize the part of the state machine dependent on the
354 prefix. */
355 is_stmt = prefix.pf_default_is_stmt;
358 print_line_header();
359 /* Start of statement program. */
360 while (line_ptr < line_ptr_end) {
361 int type = 0;
363 printf(" [0x%06" DW_PR_DSx "] ",
364 (Dwarf_Signed) (line_ptr - orig_line_ptr));
365 opcode = *(Dwarf_Small *) line_ptr;
366 line_ptr++;
367 /* 'type' is the output */
368 WHAT_IS_OPCODE(type, opcode, prefix.pf_opcode_base,
369 prefix.pf_opcode_length_table, line_ptr,
370 prefix.pf_std_op_count);
372 if (type == LOP_DISCARD) {
373 int oc;
374 int opcnt = prefix.pf_opcode_length_table[opcode];
376 printf("*** DWARF CHECK: DISCARD standard opcode %d "
377 "with %d operands: "
378 "not understood.", opcode, opcnt);
379 *err_count_out += 1;
380 for (oc = 0; oc < opcnt; oc++) {
382 * Read and discard operands we don't
383 * understand.
384 * Arbitrary choice of unsigned read.
385 * Signed read would work as well.
387 Dwarf_Unsigned utmp2;
389 DECODE_LEB128_UWORD(line_ptr, utmp2);
390 printf(" %" DW_PR_DUu " (0x%" DW_PR_DUx ")",
391 (Dwarf_Unsigned) utmp2,
392 (Dwarf_Unsigned) utmp2);
395 printf("***\n");
396 /* do nothing, necessary ops done */
397 } else if (type == LOP_SPECIAL) {
398 /* This op code is a special op in the object, no matter
399 that it might fall into the standard op range in this
400 compile Thatis, these are special opcodes between
401 special_opcode_base and MAX_LINE_OP_CODE. (including
402 special_opcode_base and MAX_LINE_OP_CODE) */
403 char special[50];
404 unsigned origop = opcode;
406 opcode = opcode - prefix.pf_opcode_base;
407 address = address + prefix.pf_minimum_instruction_length *
408 (opcode / prefix.pf_line_range);
409 line =
410 line + prefix.pf_line_base +
411 opcode % prefix.pf_line_range;
413 sprintf(special, "Specialop %3u", origop);
414 print_line_detail(special,
415 opcode, address, (int) file, line, column,
416 is_stmt, basic_block, end_sequence,
417 prologue_end, epilogue_begin, isa);
419 basic_block = false;
421 } else if (type == LOP_STANDARD) {
422 switch (opcode) {
424 case DW_LNS_copy:{
426 print_line_detail("DW_LNS_copy",
427 opcode, address, file, line,
428 column, is_stmt, basic_block,
429 end_sequence, prologue_end,
430 epilogue_begin, isa);
432 basic_block = false;
433 break;
436 case DW_LNS_advance_pc:{
437 Dwarf_Unsigned utmp2;
440 DECODE_LEB128_UWORD(line_ptr, utmp2);
441 printf("DW_LNS_advance_pc val %" DW_PR_DSd " 0x%" DW_PR_DUx "\n",
442 (Dwarf_Signed) (Dwarf_Word) utmp2,
443 (Dwarf_Unsigned) (Dwarf_Word) utmp2);
444 leb128_num = (Dwarf_Word) utmp2;
445 address =
446 address +
447 prefix.pf_minimum_instruction_length *
448 leb128_num;
449 break;
452 case DW_LNS_advance_line:{
453 Dwarf_Signed stmp;
456 DECODE_LEB128_SWORD(line_ptr, stmp);
457 advance_line = (Dwarf_Sword) stmp;
458 printf("DW_LNS_advance_line val %" DW_PR_DSd " 0x%" DW_PR_DSx "\n",
459 (Dwarf_Signed) advance_line,
460 (Dwarf_Signed) advance_line);
461 line = line + advance_line;
462 break;
465 case DW_LNS_set_file:{
466 Dwarf_Unsigned utmp2;
469 DECODE_LEB128_UWORD(line_ptr, utmp2);
470 file = (Dwarf_Word) utmp2;
471 printf("DW_LNS_set_file %ld\n", (long) file);
472 break;
475 case DW_LNS_set_column:{
476 Dwarf_Unsigned utmp2;
479 DECODE_LEB128_UWORD(line_ptr, utmp2);
480 column = (Dwarf_Word) utmp2;
481 printf("DW_LNS_set_column val %" DW_PR_DSd " 0x%" DW_PR_DSx "\n",
482 (Dwarf_Signed) column, (Dwarf_Signed) column);
483 break;
486 case DW_LNS_negate_stmt:{
487 is_stmt = !is_stmt;
488 printf("DW_LNS_negate_stmt\n");
489 break;
492 case DW_LNS_set_basic_block:{
494 printf("DW_LNS_set_basic_block\n");
495 basic_block = true;
496 break;
499 case DW_LNS_const_add_pc:{
500 opcode = MAX_LINE_OP_CODE - prefix.pf_opcode_base;
501 address =
502 address +
503 prefix.pf_minimum_instruction_length * (opcode /
504 prefix.
505 pf_line_range);
507 printf("DW_LNS_const_add_pc new address 0x%" DW_PR_DSx "\n",
508 (Dwarf_Signed) address);
509 break;
512 case DW_LNS_fixed_advance_pc:{
514 READ_UNALIGNED(dbg, fixed_advance_pc, Dwarf_Half,
515 line_ptr, sizeof(Dwarf_Half));
516 line_ptr += sizeof(Dwarf_Half);
517 address = address + fixed_advance_pc;
518 printf("DW_LNS_fixed_advance_pc val %" DW_PR_DSd
519 " 0x%" DW_PR_DSx " new address 0x%" DW_PR_DSx "\n",
520 (Dwarf_Signed) fixed_advance_pc,
521 (Dwarf_Signed) fixed_advance_pc,
522 (Dwarf_Signed) address);
523 break;
525 case DW_LNS_set_prologue_end:{
527 prologue_end = true;
528 printf("DW_LNS_set_prologue_end set true.\n");
529 break;
533 /* New in DWARF3 */
534 case DW_LNS_set_epilogue_begin:{
535 epilogue_begin = true;
536 printf("DW_LNS_set_epilogue_begin set true.\n");
537 break;
540 /* New in DWARF3 */
541 case DW_LNS_set_isa:{
542 Dwarf_Unsigned utmp2;
544 DECODE_LEB128_UWORD(line_ptr, utmp2);
545 isa = utmp2;
546 printf("DW_LNS_set_isa new value 0x%" DW_PR_DUx ".\n",
547 (Dwarf_Unsigned) utmp2);
548 if (isa != utmp2) {
549 /* The value of the isa did not fit in our
550 local so we record it wrong. declare an
551 error. */
552 dwarf_free_line_table_prefix(&prefix);
554 _dwarf_error(dbg, error,
555 DW_DLE_LINE_NUM_OPERANDS_BAD);
556 return (DW_DLV_ERROR);
558 break;
563 } else if (type == LOP_EXTENDED) {
564 Dwarf_Unsigned utmp3 = 0;
565 Dwarf_Word instr_length = 0;
566 Dwarf_Small ext_opcode = 0;
568 DECODE_LEB128_UWORD(line_ptr, utmp3);
569 instr_length = (Dwarf_Word) utmp3;
570 ext_opcode = *(Dwarf_Small *) line_ptr;
571 line_ptr++;
572 switch (ext_opcode) {
574 case DW_LNE_end_sequence:{
575 end_sequence = true;
577 print_line_detail("DW_LNE_end_sequence extended",
578 opcode, address, file, line,
579 column, is_stmt, basic_block,
580 end_sequence, prologue_end,
581 epilogue_begin, isa);
583 address = 0;
584 file = 1;
585 line = 1;
586 column = 0;
587 is_stmt = prefix.pf_default_is_stmt;
588 basic_block = false;
589 end_sequence = false;
590 prologue_end = false;
591 epilogue_begin = false;
594 break;
597 case DW_LNE_set_address:{
599 READ_UNALIGNED(dbg, address, Dwarf_Addr,
600 line_ptr,
601 die->di_cu_context->cc_address_size);
603 line_ptr += die->di_cu_context->cc_address_size;
604 printf("DW_LNE_set_address address 0x%" DW_PR_DUx "\n",
605 (Dwarf_Unsigned) address);
608 break;
611 case DW_LNE_define_file:{
612 Dwarf_Unsigned di = 0;
613 Dwarf_Unsigned tlm = 0;
614 Dwarf_Unsigned fl = 0;
616 Dwarf_Small *fn = (Dwarf_Small *) line_ptr;
617 line_ptr = line_ptr + strlen((char *) line_ptr) + 1;
619 di = _dwarf_decode_u_leb128(line_ptr,
620 &leb128_length);
621 line_ptr = line_ptr + leb128_length;
623 tlm = _dwarf_decode_u_leb128(line_ptr,
624 &leb128_length);
625 line_ptr = line_ptr + leb128_length;
627 fl = _dwarf_decode_u_leb128(line_ptr,
628 &leb128_length);
629 line_ptr = line_ptr + leb128_length;
632 printf("DW_LNE_define_file %s \n", fn);
633 printf(" dir index %d\n", (int) di);
635 time_t tt3 = (time_t) tlm;
637 /* ctime supplies newline */
638 printf(" last time 0x%x %s",
639 (unsigned) tlm, ctime(&tt3));
641 printf(" file length %ld 0x%lx\n",
642 (long) fl, (unsigned long) fl);
644 break;
647 default:{
648 /* This is an extended op code we do not know about,
649 other than we know now many bytes it is
650 (and the op code and the bytes of operand). */
652 Dwarf_Unsigned remaining_bytes = instr_length -1;
653 if(instr_length < 1 || remaining_bytes > DW_LNE_LEN_MAX) {
654 dwarf_free_line_table_prefix(&prefix);
655 _dwarf_error(dbg, error,
656 DW_DLE_LINE_EXT_OPCODE_BAD);
657 return (DW_DLV_ERROR);
659 printf("DW_LNE extended op 0x%x ",ext_opcode);
660 printf("Bytecount: " DW_PR_DUu , instr_length);
661 if(remaining_bytes > 0) {
662 printf(" linedata: 0x");
663 while (remaining_bytes > 0) {
664 printf("%02x",(unsigned char)(*(line_ptr)));
665 line_ptr++;
666 remaining_bytes--;
669 printf("\n");
671 break;
677 dwarf_free_line_table_prefix(&prefix);
678 return (DW_DLV_OK);
682 This is support for dwarfdump: making it possible
683 for clients wanting line detail info on stdout
684 to get that detail without including internal libdwarf
685 header information.
686 Caller passes in compilation unit DIE.
687 The _dwarf_ version is obsolete (though supported for
688 compatibility).
689 The dwarf_ version is preferred.
690 The functions are intentionally identical: having
691 _dwarf_print_lines call dwarf_print_lines might
692 better emphasize they are intentionally identical, but
693 that seemed slightly silly given how short the functions are.
694 Interface adds error_count (output value) February 2009.
697 dwarf_print_lines(Dwarf_Die die, Dwarf_Error * error,int *error_count)
699 int only_line_header = 0;
700 int res = _dwarf_internal_printlines(die, error,
701 error_count,
702 only_line_header);
703 if (res != DW_DLV_OK) {
704 return res;
706 return res;
709 _dwarf_print_lines(Dwarf_Die die, Dwarf_Error * error)
711 int only_line_header = 0;
712 int err_count = 0;
713 int res = _dwarf_internal_printlines(die, error,
714 &err_count,
715 only_line_header);
716 /* No way to get error count back in this interface */
717 if (res != DW_DLV_OK) {
718 return res;
720 return res;
723 /* The check is in case we are not printing full line data,
724 this gets some of the issues noted with .debug_line,
725 but not all. Call dwarf_print_lines() to get all issues.
726 Intended for apps like dwarfdump.
728 void
729 dwarf_check_lineheader(Dwarf_Die die, int *err_count_out)
731 Dwarf_Error err;
732 int only_line_header = 1;
733 _dwarf_internal_printlines(die, &err,err_count_out,
734 only_line_header);
735 return;