import less(1)
[unleashed/tickless.git] / usr / src / lib / libdwarf / common / pro_section.c
blob6503c2cf09b7884e3f3af52645662acdc25b136f
1 /*
3 Copyright (C) 2000,2004,2006 Silicon Graphics, Inc. All Rights Reserved.
4 Portions Copyright (C) 2007-2010 David Anderson. All Rights Reserved.
5 Portions Copyright 2002-2010 Sun Microsystems, Inc. All rights reserved.
7 This program is free software; you can redistribute it and/or modify it
8 under the terms of version 2.1 of the GNU Lesser General Public License
9 as published by the Free Software Foundation.
11 This program is distributed in the hope that it would be useful, but
12 WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
15 Further, this software is distributed without any warranty that it is
16 free of the rightful claim of any third person regarding infringement
17 or the like. Any license provided herein, whether implied or
18 otherwise, applies only to this software file. Patent licenses, if
19 any, provided herein do not apply to combinations of this program with
20 other software, or any other product whatsoever.
22 You should have received a copy of the GNU Lesser General Public
23 License along with this program; if not, write the Free Software
24 Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston MA 02110-1301,
25 USA.
27 Contact information: Silicon Graphics, Inc., 1500 Crittenden Lane,
28 Mountain View, CA 94043, or:
30 http://www.sgi.com
32 For further information regarding this notice, see:
34 http://oss.sgi.com/projects/GenInfo/NoticeExplan
37 /*
38 SGI has moved from the Crittenden Lane address.
45 #include "config.h"
46 #include "libdwarfdefs.h"
47 #include <stdio.h>
48 #include <string.h>
49 #ifdef HAVE_ELFACCESS_H
50 #include <elfaccess.h>
51 #endif
52 #include "pro_incl.h"
53 #include "pro_section.h"
54 #include "pro_line.h"
55 #include "pro_frame.h"
56 #include "pro_die.h"
57 #include "pro_macinfo.h"
58 #include "pro_types.h"
60 #ifndef SHF_MIPS_NOSTRIP
61 /* if this is not defined, we probably don't need it: just use 0 */
62 #define SHF_MIPS_NOSTRIP 0
63 #endif
64 #ifndef R_MIPS_NONE
65 #define R_MIPS_NONE 0
66 #endif
68 #ifndef TRUE
69 #define TRUE 1
70 #endif
71 #ifndef FALSE
72 #define FALSE 0
73 #endif
75 /* must match up with pro_section.h defines of DEBUG_INFO etc
76 and sectnames (below). REL_SEC_PREFIX is either ".rel" or ".rela"
77 see pro_incl.h
79 char *_dwarf_rel_section_names[] = {
80 REL_SEC_PREFIX ".debug_info",
81 REL_SEC_PREFIX ".debug_line",
82 REL_SEC_PREFIX ".debug_abbrev", /* no relocations on this, really */
83 REL_SEC_PREFIX ".debug_frame",
84 REL_SEC_PREFIX ".debug_aranges",
85 REL_SEC_PREFIX ".debug_pubnames",
86 REL_SEC_PREFIX ".debug_str",
87 REL_SEC_PREFIX ".debug_funcnames", /* sgi extension */
88 REL_SEC_PREFIX ".debug_typenames", /* sgi extension */
89 REL_SEC_PREFIX ".debug_varnames", /* sgi extension */
90 REL_SEC_PREFIX ".debug_weaknames", /* sgi extension */
91 REL_SEC_PREFIX ".debug_macinfo",
92 REL_SEC_PREFIX ".debug_loc"
95 /* names of sections. Ensure that it matches the defines
96 in pro_section.h, in the same order
97 Must match also _dwarf_rel_section_names above
99 char *_dwarf_sectnames[] = {
100 ".debug_info",
101 ".debug_line",
102 ".debug_abbrev",
103 ".debug_frame",
104 ".debug_aranges",
105 ".debug_pubnames",
106 ".debug_str",
107 ".debug_funcnames", /* sgi extension */
108 ".debug_typenames", /* sgi extension */
109 ".debug_varnames", /* sgi extension */
110 ".debug_weaknames", /* sgi extension */
111 ".debug_macinfo",
112 ".debug_loc"
118 static Dwarf_Ubyte std_opcode_len[] = { 0, /* DW_LNS_copy */
119 1, /* DW_LNS_advance_pc */
120 1, /* DW_LNS_advance_line */
121 1, /* DW_LNS_set_file */
122 1, /* DW_LNS_set_column */
123 0, /* DW_LNS_negate_stmt */
124 0, /* DW_LNS_set_basic_block */
125 0, /* DW_LNS_const_add_pc */
126 1, /* DW_LNS_fixed_advance_pc */
129 /* struct to hold relocation entries. Its mantained as a linked
130 list of relocation structs, and will then be written at as a
131 whole into the relocation section. Whether its 32 bit or
132 64 bit will be obtained from Dwarf_Debug pointer.
135 typedef struct Dwarf_P_Rel_s *Dwarf_P_Rel;
136 struct Dwarf_P_Rel_s {
137 Dwarf_P_Rel dr_next;
138 void *dr_rel_datap;
140 typedef struct Dwarf_P_Rel_Head_s *Dwarf_P_Rel_Head;
141 struct Dwarf_P_Rel_Head_s {
142 struct Dwarf_P_Rel_s *drh_head;
143 struct Dwarf_P_Rel_s *drh_tail;
146 static int _dwarf_pro_generate_debugline(Dwarf_P_Debug dbg,
147 Dwarf_Error * error);
148 static int _dwarf_pro_generate_debugframe(Dwarf_P_Debug dbg,
149 Dwarf_Error * error);
150 static int _dwarf_pro_generate_debuginfo(Dwarf_P_Debug dbg,
151 Dwarf_Error * error);
152 static Dwarf_P_Abbrev _dwarf_pro_getabbrev(Dwarf_P_Die, Dwarf_P_Abbrev);
153 static int _dwarf_pro_match_attr
154 (Dwarf_P_Attribute, Dwarf_P_Abbrev, int no_attr);
156 /* these macros used as return value for below functions */
157 #define OPC_INCS_ZERO -1
158 #define OPC_OUT_OF_RANGE -2
159 #define LINE_OUT_OF_RANGE -3
160 static int _dwarf_pro_get_opc(Dwarf_Unsigned addr_adv, int line_adv);
163 /* BEGIN_LEN_SIZE is the size of the 'length' field in total.
164 Which may be 4,8, or 12 bytes!
165 4 is standard DWARF2.
166 8 is non-standard MIPS-IRIX 64-bit.
167 12 is standard DWARF3 for 64 bit offsets.
168 Used in various routines: local variable names
169 must match the names here.
171 #define BEGIN_LEN_SIZE (uwordb_size + extension_size)
174 Return TRUE if we need the section, FALSE otherwise
176 If any of the 'line-data-related' calls were made
177 including file or directory entries,
178 produce .debug_line .
181 static int
182 dwarf_need_debug_line_section(Dwarf_P_Debug dbg)
184 if (dbg->de_lines == NULL && dbg->de_file_entries == NULL
185 && dbg->de_inc_dirs == NULL) {
186 return FALSE;
188 return TRUE;
192 Convert debug information to a format such that
193 it can be written on disk.
194 Called exactly once per execution.
196 Dwarf_Signed
197 dwarf_transform_to_disk_form(Dwarf_P_Debug dbg, Dwarf_Error * error)
200 Section data in written out in a number of buffers. Each
201 _generate_*() function returns a cumulative count of buffers for
202 all the sections. get_section_bytes() returns pointers to these
203 buffers one at a time. */
204 int nbufs = 0;
205 int sect = 0;
206 int err = 0;
207 Dwarf_Unsigned du = 0;
209 if (dbg->de_version_magic_number != PRO_VERSION_MAGIC) {
210 DWARF_P_DBG_ERROR(dbg, DW_DLE_IA, DW_DLV_NOCOUNT);
213 /* Create dwarf section headers */
214 for (sect = 0; sect < NUM_DEBUG_SECTIONS; sect++) {
215 long flags = 0;
217 switch (sect) {
219 case DEBUG_INFO:
220 if (dbg->de_dies == NULL)
221 continue;
222 break;
224 case DEBUG_LINE:
225 if (dwarf_need_debug_line_section(dbg) == FALSE) {
226 continue;
228 break;
230 case DEBUG_ABBREV:
231 if (dbg->de_dies == NULL)
232 continue;
233 break;
235 case DEBUG_FRAME:
236 if (dbg->de_frame_cies == NULL)
237 continue;
238 flags = SHF_MIPS_NOSTRIP;
239 break;
241 case DEBUG_ARANGES:
242 if (dbg->de_arange == NULL)
243 continue;
244 break;
246 case DEBUG_PUBNAMES:
247 if (dbg->de_simple_name_headers[dwarf_snk_pubname].
248 sn_head == NULL)
249 continue;
250 break;
252 case DEBUG_STR:
253 if (dbg->de_strings == NULL)
254 continue;
255 break;
257 case DEBUG_FUNCNAMES:
258 if (dbg->de_simple_name_headers[dwarf_snk_funcname].
259 sn_head == NULL)
260 continue;
261 break;
263 case DEBUG_TYPENAMES:
264 if (dbg->de_simple_name_headers[dwarf_snk_typename].
265 sn_head == NULL)
266 continue;
267 break;
269 case DEBUG_VARNAMES:
270 if (dbg->de_simple_name_headers[dwarf_snk_varname].
271 sn_head == NULL)
272 continue;
273 break;
275 case DEBUG_WEAKNAMES:
276 if (dbg->de_simple_name_headers[dwarf_snk_weakname].
277 sn_head == NULL)
278 continue;
279 break;
281 case DEBUG_MACINFO:
282 if (dbg->de_first_macinfo == NULL)
283 continue;
284 break;
285 case DEBUG_LOC:
286 /* not handled yet */
287 continue;
288 default:
289 /* logic error: missing a case */
290 DWARF_P_DBG_ERROR(dbg, DW_DLE_ELF_SECT_ERR, DW_DLV_NOCOUNT);
293 int new_base_elf_sect;
295 if (dbg->de_callback_func_b) {
296 new_base_elf_sect =
297 dbg->de_callback_func_b(_dwarf_sectnames[sect],
298 /* rec size */ 1,
299 SECTION_TYPE,
300 flags, SHN_UNDEF, 0, &du, &err);
302 } else {
303 int name_idx = 0;
304 new_base_elf_sect = dbg->de_callback_func(
305 _dwarf_sectnames[sect],
306 dbg->de_relocation_record_size,
307 SECTION_TYPE, flags,
308 SHN_UNDEF, 0,
309 &name_idx, &err);
310 du = name_idx;
312 if (new_base_elf_sect == -1) {
313 DWARF_P_DBG_ERROR(dbg, DW_DLE_ELF_SECT_ERR,
314 DW_DLV_NOCOUNT);
316 dbg->de_elf_sects[sect] = new_base_elf_sect;
318 dbg->de_sect_name_idx[sect] = du;
322 nbufs = 0;
325 Changing the order in which the sections are generated may cause
326 problems because of relocations. */
328 if (dwarf_need_debug_line_section(dbg) == TRUE) {
329 nbufs = _dwarf_pro_generate_debugline(dbg, error);
330 if (nbufs < 0) {
331 DWARF_P_DBG_ERROR(dbg, DW_DLE_DEBUGLINE_ERROR,
332 DW_DLV_NOCOUNT);
336 if (dbg->de_frame_cies) {
337 nbufs = _dwarf_pro_generate_debugframe(dbg, error);
338 if (nbufs < 0) {
339 DWARF_P_DBG_ERROR(dbg, DW_DLE_DEBUGFRAME_ERROR,
340 DW_DLV_NOCOUNT);
343 if (dbg->de_first_macinfo) {
344 nbufs = _dwarf_pro_transform_macro_info_to_disk(dbg, error);
345 if (nbufs < 0) {
346 DWARF_P_DBG_ERROR(dbg, DW_DLE_DEBUGMACINFO_ERROR,
347 DW_DLV_NOCOUNT);
351 if (dbg->de_dies) {
352 nbufs = _dwarf_pro_generate_debuginfo(dbg, error);
353 if (nbufs < 0) {
354 DWARF_P_DBG_ERROR(dbg, DW_DLE_DEBUGINFO_ERROR,
355 DW_DLV_NOCOUNT);
359 if (dbg->de_arange) {
360 nbufs = _dwarf_transform_arange_to_disk(dbg, error);
361 if (nbufs < 0) {
362 DWARF_P_DBG_ERROR(dbg, DW_DLE_DEBUGINFO_ERROR,
363 DW_DLV_NOCOUNT);
367 if (dbg->de_simple_name_headers[dwarf_snk_pubname].sn_head) {
368 nbufs = _dwarf_transform_simplename_to_disk(dbg,
369 dwarf_snk_pubname,
370 DEBUG_PUBNAMES,
371 error);
374 if (nbufs < 0) {
375 DWARF_P_DBG_ERROR(dbg, DW_DLE_DEBUGINFO_ERROR,
376 DW_DLV_NOCOUNT);
380 if (dbg->de_simple_name_headers[dwarf_snk_funcname].sn_head) {
381 nbufs = _dwarf_transform_simplename_to_disk(dbg,
382 dwarf_snk_funcname,
383 DEBUG_FUNCNAMES,
384 error);
385 if (nbufs < 0) {
386 DWARF_P_DBG_ERROR(dbg, DW_DLE_DEBUGINFO_ERROR,
387 DW_DLV_NOCOUNT);
391 if (dbg->de_simple_name_headers[dwarf_snk_typename].sn_head) {
392 nbufs = _dwarf_transform_simplename_to_disk(dbg,
393 dwarf_snk_typename,
394 DEBUG_TYPENAMES,
395 error);
396 if (nbufs < 0) {
397 DWARF_P_DBG_ERROR(dbg, DW_DLE_DEBUGINFO_ERROR,
398 DW_DLV_NOCOUNT);
402 if (dbg->de_simple_name_headers[dwarf_snk_varname].sn_head) {
403 nbufs = _dwarf_transform_simplename_to_disk(dbg,
404 dwarf_snk_varname,
405 DEBUG_VARNAMES,
406 error);
408 if (nbufs < 0) {
409 DWARF_P_DBG_ERROR(dbg, DW_DLE_DEBUGINFO_ERROR,
410 DW_DLV_NOCOUNT);
414 if (dbg->de_simple_name_headers[dwarf_snk_weakname].sn_head) {
415 nbufs = _dwarf_transform_simplename_to_disk(dbg,
416 dwarf_snk_weakname, DEBUG_WEAKNAMES, error);
417 if (nbufs < 0) {
418 DWARF_P_DBG_ERROR(dbg, DW_DLE_DEBUGINFO_ERROR,
419 DW_DLV_NOCOUNT);
424 Dwarf_Signed new_secs = 0;
425 int res = 0;
427 res = dbg->de_transform_relocs_to_disk(dbg, &new_secs);
428 if (res != DW_DLV_OK) {
429 DWARF_P_DBG_ERROR(dbg, DW_DLE_DEBUGINFO_ERROR,
430 DW_DLV_NOCOUNT);
432 nbufs += new_secs;
434 return nbufs;
438 /*---------------------------------------------------------------
439 Generate debug_line section
440 ---------------------------------------------------------------*/
441 static int
442 _dwarf_pro_generate_debugline(Dwarf_P_Debug dbg, Dwarf_Error * error)
444 Dwarf_P_Inc_Dir curdir = 0;
445 Dwarf_P_F_Entry curentry = 0;
446 Dwarf_P_Line curline = 0;
447 Dwarf_P_Line prevline = 0;
449 /* all data named cur* are used to loop thru linked lists */
451 int sum_bytes = 0;
452 int prolog_size = 0;
453 unsigned char *data = 0; /* holds disk form data */
454 int elfsectno = 0;
455 unsigned char *start_line_sec = 0; /* pointer to the buffer at
456 section start */
457 /* temps for memcpy */
458 Dwarf_Unsigned du = 0;
459 Dwarf_Ubyte db = 0;
460 Dwarf_Half dh = 0;
461 int res = 0;
462 int uwordb_size = dbg->de_offset_size;
463 int extension_size = dbg->de_64bit_extension ? 4 : 0;
464 int upointer_size = dbg->de_pointer_size;
465 char buff1[ENCODE_SPACE_NEEDED];
469 sum_bytes = 0;
471 elfsectno = dbg->de_elf_sects[DEBUG_LINE];
473 /* include directories */
474 curdir = dbg->de_inc_dirs;
475 while (curdir) {
476 prolog_size += strlen(curdir->did_name) + 1;
477 curdir = curdir->did_next;
479 prolog_size++; /* last null following last directory
480 entry. */
482 /* file entries */
483 curentry = dbg->de_file_entries;
484 while (curentry) {
485 prolog_size +=
486 strlen(curentry->dfe_name) + 1 + curentry->dfe_nbytes;
487 curentry = curentry->dfe_next;
489 prolog_size++; /* last null byte */
492 prolog_size += BEGIN_LEN_SIZE + sizeof_uhalf(dbg) + /* version # */
493 uwordb_size + /* header length */
494 sizeof_ubyte(dbg) + /* min_instr length */
495 sizeof_ubyte(dbg) + /* default is_stmt */
496 sizeof_ubyte(dbg) + /* linebase */
497 sizeof_ubyte(dbg) + /* linerange */
498 sizeof_ubyte(dbg); /* opcode base */
500 /* length of table specifying # of opnds */
501 prolog_size += sizeof(std_opcode_len);
503 GET_CHUNK(dbg, elfsectno, data, prolog_size, error);
504 start_line_sec = data;
506 /* copy over the data */
507 /* total_length */
508 du = 0;
509 if (extension_size) {
510 Dwarf_Word x = DISTINGUISHED_VALUE;
512 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &x,
513 sizeof(x), extension_size);
514 data += extension_size;
517 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &du,
518 sizeof(du), uwordb_size);
519 data += uwordb_size;
521 dh = VERSION;
522 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &dh,
523 sizeof(dh), sizeof(Dwarf_Half));
524 data += sizeof(Dwarf_Half);
526 /* header length */
527 du = prolog_size - (BEGIN_LEN_SIZE + sizeof(Dwarf_Half) +
528 uwordb_size);
530 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &du,
531 sizeof(du), uwordb_size);
532 data += uwordb_size;
534 db = MIN_INST_LENGTH;
535 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &db,
536 sizeof(db), sizeof(Dwarf_Ubyte));
537 data += sizeof(Dwarf_Ubyte);
538 db = DEFAULT_IS_STMT;
539 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &db,
540 sizeof(db), sizeof(Dwarf_Ubyte));
541 data += sizeof(Dwarf_Ubyte);
542 db = (Dwarf_Ubyte) LINE_BASE;
543 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &db,
544 sizeof(db), sizeof(Dwarf_Ubyte));
545 data += sizeof(Dwarf_Ubyte);
546 db = LINE_RANGE;
547 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &db,
548 sizeof(db), sizeof(Dwarf_Ubyte));
549 data += sizeof(Dwarf_Ubyte);
550 db = OPCODE_BASE;
551 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &db,
552 sizeof(db), sizeof(Dwarf_Ubyte));
553 data += sizeof(Dwarf_Ubyte);
554 WRITE_UNALIGNED(dbg, (void *) data, (const void *) std_opcode_len,
555 sizeof(std_opcode_len), sizeof(std_opcode_len));
556 data += sizeof(std_opcode_len);
558 /* copy over include directories */
559 curdir = dbg->de_inc_dirs;
560 while (curdir) {
561 strcpy((char *) data, curdir->did_name);
562 data += strlen(curdir->did_name) + 1;
563 curdir = curdir->did_next;
565 *data = '\0'; /* last null */
566 data++;
568 /* copy file entries */
569 curentry = dbg->de_file_entries;
570 while (curentry) {
571 strcpy((char *) data, curentry->dfe_name);
572 data += strlen(curentry->dfe_name) + 1;
573 /* copies of leb numbers, no endian issues */
574 memcpy((void *) data,
575 (const void *) curentry->dfe_args, curentry->dfe_nbytes);
576 data += curentry->dfe_nbytes;
577 curentry = curentry->dfe_next;
579 *data = '\0';
580 data++;
582 sum_bytes += prolog_size;
584 curline = dbg->de_lines;
585 prevline = (Dwarf_P_Line)
586 _dwarf_p_get_alloc(dbg, sizeof(struct Dwarf_P_Line_s));
587 if (prevline == NULL) {
588 DWARF_P_DBG_ERROR(dbg, DW_DLE_LINE_ALLOC, -1);
590 _dwarf_pro_reg_init(prevline);
591 /* generate opcodes for line numbers */
592 while (curline) {
593 int nbytes;
594 char *arg;
595 int opc;
596 int no_lns_copy; /* if lns copy opcode doesnt need to be
597 generated, if special opcode or end
598 sequence */
599 Dwarf_Unsigned addr_adv;
600 int line_adv; /* supposed to be a reasonably small
601 number, so the size should not be a
602 problem. ? */
604 no_lns_copy = 0;
605 if (curline->dpl_opc != 0) {
606 int inst_bytes; /* no of bytes in extended opcode */
607 char *str; /* hold leb encoded inst_bytes */
608 int str_nbytes; /* no of bytes in str */
610 switch (curline->dpl_opc) {
611 case DW_LNE_end_sequence:
613 /* Advance pc to end of text section. */
614 addr_adv = curline->dpl_address - prevline->dpl_address;
615 if (addr_adv > 0) {
616 db = DW_LNS_advance_pc;
617 res =
618 _dwarf_pro_encode_leb128_nm(addr_adv /
619 MIN_INST_LENGTH,
620 &nbytes, buff1,
621 sizeof(buff1));
622 if (res != DW_DLV_OK) {
623 DWARF_P_DBG_ERROR(dbg, DW_DLE_CHUNK_ALLOC, -1);
625 GET_CHUNK(dbg, elfsectno, data,
626 nbytes + sizeof(Dwarf_Ubyte), error);
627 WRITE_UNALIGNED(dbg, (void *) data,
628 (const void *) &db, sizeof(db),
629 sizeof(Dwarf_Ubyte));
630 data += sizeof(Dwarf_Ubyte);
631 /* leb, no endianness issue */
632 memcpy((void *) data, (const void *) buff1, nbytes);
633 data += nbytes + sizeof(Dwarf_Ubyte);
634 sum_bytes += nbytes + sizeof(Dwarf_Ubyte);
635 prevline->dpl_address = curline->dpl_address;
638 /* first null byte */
639 db = 0;
640 GET_CHUNK(dbg, elfsectno, data, sizeof(Dwarf_Ubyte),
641 error);
642 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &db,
643 sizeof(db), sizeof(Dwarf_Ubyte));
644 data += sizeof(Dwarf_Ubyte);
645 sum_bytes += sizeof(Dwarf_Ubyte);
647 /* write length of extended opcode */
648 inst_bytes = sizeof(Dwarf_Ubyte);
649 res =
650 _dwarf_pro_encode_leb128_nm(inst_bytes, &str_nbytes,
651 buff1, sizeof(buff1));
652 if (res != DW_DLV_OK) {
653 DWARF_P_DBG_ERROR(dbg, DW_DLE_CHUNK_ALLOC, -1);
655 GET_CHUNK(dbg, elfsectno, data, str_nbytes, error);
656 memcpy((void *) data, (const void *) buff1, str_nbytes);
657 data += str_nbytes;
658 sum_bytes += str_nbytes;
660 /* write extended opcode */
661 db = DW_LNE_end_sequence;
662 GET_CHUNK(dbg, elfsectno, data, sizeof(Dwarf_Ubyte),
663 error);
664 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &db,
665 sizeof(db), sizeof(Dwarf_Ubyte));
666 data += sizeof(Dwarf_Ubyte);
667 sum_bytes += sizeof(Dwarf_Ubyte);
668 /* reset value to original values */
669 _dwarf_pro_reg_init(prevline);
670 no_lns_copy = 1;
671 /* this is set only for end_sequence, so that a
672 dw_lns_copy is not generated */
673 break;
675 case DW_LNE_set_address:
677 /* first null byte */
678 db = 0;
679 GET_CHUNK(dbg, elfsectno, data, sizeof(Dwarf_Ubyte),
680 error);
681 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &db,
682 sizeof(db), sizeof(Dwarf_Ubyte));
683 data += sizeof(Dwarf_Ubyte);
684 sum_bytes += sizeof(Dwarf_Ubyte);
686 /* write length of extended opcode */
687 inst_bytes = sizeof(Dwarf_Ubyte) + upointer_size;
688 res =
689 _dwarf_pro_encode_leb128_nm(inst_bytes, &str_nbytes,
690 buff1, sizeof(buff1));
691 if (res != DW_DLV_OK) {
692 DWARF_P_DBG_ERROR(dbg, DW_DLE_CHUNK_ALLOC, -1);
694 GET_CHUNK(dbg, elfsectno, data, str_nbytes, error);
695 str = buff1;
696 /* leb number, no endian issue */
697 memcpy((void *) data, (const void *) str, str_nbytes);
698 data += str_nbytes;
699 sum_bytes += str_nbytes;
701 /* write extended opcode */
702 db = DW_LNE_set_address;
703 GET_CHUNK(dbg, elfsectno, data, upointer_size +
704 sizeof(Dwarf_Ubyte), error);
705 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &db,
706 sizeof(db), sizeof(Dwarf_Ubyte));
707 data += sizeof(Dwarf_Ubyte);
708 sum_bytes += sizeof(Dwarf_Ubyte);
710 /* reloc for address */
711 res = dbg->de_reloc_name(dbg, DEBUG_LINE,
712 sum_bytes, /* r_offset */
713 curline->dpl_r_symidx,
714 dwarf_drt_data_reloc,
715 uwordb_size);
716 if (res != DW_DLV_OK) {
717 DWARF_P_DBG_ERROR(dbg, DW_DLE_CHUNK_ALLOC, -1);
720 /* write offset (address) */
721 du = curline->dpl_address;
722 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &du,
723 sizeof(du), upointer_size);
724 data += upointer_size;
725 sum_bytes += upointer_size;
726 prevline->dpl_address = curline->dpl_address;
727 no_lns_copy = 1;
728 break;
730 } else {
731 if (curline->dpl_file != prevline->dpl_file) {
732 db = DW_LNS_set_file;
733 res =
734 _dwarf_pro_encode_leb128_nm(curline->dpl_file,
735 &nbytes, buff1,
736 sizeof(buff1));
737 if (res != DW_DLV_OK) {
738 DWARF_P_DBG_ERROR(dbg, DW_DLE_CHUNK_ALLOC, -1);
740 arg = buff1;
741 GET_CHUNK(dbg, elfsectno, data,
742 nbytes + sizeof(Dwarf_Ubyte), error);
743 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &db,
744 sizeof(db), sizeof(Dwarf_Ubyte));
745 data += sizeof(Dwarf_Ubyte);
746 memcpy((void *) data, (const void *) arg, nbytes);
747 data += nbytes;
748 sum_bytes += nbytes + sizeof(Dwarf_Ubyte);
749 prevline->dpl_file = curline->dpl_file;
751 if (curline->dpl_column != prevline->dpl_column) {
752 db = DW_LNS_set_column;
753 res = _dwarf_pro_encode_leb128_nm(curline->dpl_column,
754 &nbytes,
755 buff1, sizeof(buff1));
756 if (res != DW_DLV_OK) {
757 DWARF_P_DBG_ERROR(dbg, DW_DLE_CHUNK_ALLOC, -1);
760 arg = buff1;
761 GET_CHUNK(dbg, elfsectno, data,
762 nbytes + sizeof(Dwarf_Ubyte), error);
763 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &db,
764 sizeof(db), sizeof(Dwarf_Ubyte));
765 data += sizeof(Dwarf_Ubyte);
766 memcpy((void *) data, (const void *) arg, nbytes);
767 data += nbytes;
768 sum_bytes += nbytes + sizeof(Dwarf_Ubyte);
769 prevline->dpl_column = curline->dpl_column;
771 if (curline->dpl_is_stmt != prevline->dpl_is_stmt) {
772 db = DW_LNS_negate_stmt;
773 GET_CHUNK(dbg, elfsectno, data, sizeof(Dwarf_Ubyte),
774 error);
775 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &db,
776 sizeof(db), sizeof(Dwarf_Ubyte));
777 data += sizeof(Dwarf_Ubyte);
778 sum_bytes += sizeof(Dwarf_Ubyte);
779 prevline->dpl_is_stmt = curline->dpl_is_stmt;
781 if (curline->dpl_basic_block == true &&
782 prevline->dpl_basic_block == false) {
783 db = DW_LNS_set_basic_block;
784 GET_CHUNK(dbg, elfsectno, data, sizeof(Dwarf_Ubyte),
785 error);
786 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &db,
787 sizeof(db), sizeof(Dwarf_Ubyte));
788 data += sizeof(Dwarf_Ubyte);
789 sum_bytes += sizeof(Dwarf_Ubyte);
790 prevline->dpl_basic_block = curline->dpl_basic_block;
792 addr_adv = curline->dpl_address - prevline->dpl_address;
794 line_adv = (int) (curline->dpl_line - prevline->dpl_line);
795 if ((addr_adv % MIN_INST_LENGTH) != 0) {
796 DWARF_P_DBG_ERROR(dbg, DW_DLE_WRONG_ADDRESS, -1);
798 if ((opc = _dwarf_pro_get_opc(addr_adv, line_adv)) > 0) {
799 no_lns_copy = 1;
800 db = opc;
801 GET_CHUNK(dbg, elfsectno, data, sizeof(Dwarf_Ubyte),
802 error);
803 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &db,
804 sizeof(db), sizeof(Dwarf_Ubyte));
805 data += sizeof(Dwarf_Ubyte);
806 sum_bytes += sizeof(Dwarf_Ubyte);
807 prevline->dpl_basic_block = false;
808 prevline->dpl_address = curline->dpl_address;
809 prevline->dpl_line = curline->dpl_line;
810 } else {
811 if (addr_adv > 0) {
812 db = DW_LNS_advance_pc;
813 res =
814 _dwarf_pro_encode_leb128_nm(addr_adv /
815 MIN_INST_LENGTH,
816 &nbytes, buff1,
817 sizeof(buff1));
818 if (res != DW_DLV_OK) {
819 DWARF_P_DBG_ERROR(dbg, DW_DLE_CHUNK_ALLOC, -1);
822 arg = buff1;
823 GET_CHUNK(dbg, elfsectno, data,
824 nbytes + sizeof(Dwarf_Ubyte), error);
825 WRITE_UNALIGNED(dbg, (void *) data,
826 (const void *) &db,
827 sizeof(db), sizeof(Dwarf_Ubyte));
828 data += sizeof(Dwarf_Ubyte);
829 memcpy((void *) data, (const void *) arg, nbytes);
830 data += nbytes + sizeof(Dwarf_Ubyte);
831 sum_bytes += nbytes + sizeof(Dwarf_Ubyte);
832 prevline->dpl_basic_block = false;
833 prevline->dpl_address = curline->dpl_address;
835 if (line_adv != 0) {
836 db = DW_LNS_advance_line;
837 res = _dwarf_pro_encode_signed_leb128_nm(line_adv,
838 &nbytes,
839 buff1,
840 sizeof
841 (buff1));
842 if (res != DW_DLV_OK) {
843 DWARF_P_DBG_ERROR(dbg, DW_DLE_CHUNK_ALLOC, -1);
846 arg = buff1;
847 GET_CHUNK(dbg, elfsectno, data,
848 nbytes + sizeof(Dwarf_Ubyte), error);
849 WRITE_UNALIGNED(dbg, (void *) data,
850 (const void *) &db, sizeof(db),
851 sizeof(Dwarf_Ubyte));
852 data += sizeof(Dwarf_Ubyte);
853 memcpy((void *) data, (const void *) arg, nbytes);
854 data += nbytes + sizeof(Dwarf_Ubyte);
855 sum_bytes += nbytes + sizeof(Dwarf_Ubyte);
856 prevline->dpl_basic_block = false;
857 prevline->dpl_line = curline->dpl_line;
860 } /* ends else for opc != 0 */
861 if (no_lns_copy == 0) { /* if not a special or dw_lne_end_seq
862 generate a matrix line */
863 db = DW_LNS_copy;
864 GET_CHUNK(dbg, elfsectno, data, sizeof(Dwarf_Ubyte), error);
865 WRITE_UNALIGNED(dbg, (void *) data,
866 (const void *) &db,
867 sizeof(db), sizeof(Dwarf_Ubyte));
868 data += sizeof(Dwarf_Ubyte);
869 sum_bytes += sizeof(Dwarf_Ubyte);
870 prevline->dpl_basic_block = false;
872 curline = curline->dpl_next;
875 /* write total length field */
876 du = sum_bytes - BEGIN_LEN_SIZE;
878 start_line_sec += extension_size;
879 WRITE_UNALIGNED(dbg, (void *) start_line_sec,
880 (const void *) &du, sizeof(du), uwordb_size);
883 return (int) dbg->de_n_debug_sect;
886 /*---------------------------------------------------------------
887 Generate debug_frame section
888 ---------------------------------------------------------------*/
889 static int
890 _dwarf_pro_generate_debugframe(Dwarf_P_Debug dbg, Dwarf_Error * error)
892 int elfsectno = 0;
893 int i = 0;
894 int firsttime = 1;
895 int pad = 0; /* Pad for padding to align cies and fdes */
896 Dwarf_P_Cie curcie = 0;
897 Dwarf_P_Fde curfde = 0;
898 unsigned char *data = 0;
899 Dwarf_sfixed dsw = 0;
900 Dwarf_Unsigned du = 0;
901 Dwarf_Ubyte db = 0;
902 long *cie_offs = 0; /* Holds byte offsets for links to fde's */
903 unsigned long cie_length = 0;
904 int cie_no = 0;
905 int uwordb_size = dbg->de_offset_size;
906 int extension_size = dbg->de_64bit_extension ? 4 : 0;
907 int upointer_size = dbg->de_pointer_size;
908 Dwarf_Unsigned cur_off = 0; /* current offset of written data, held
909 for relocation info */
911 elfsectno = dbg->de_elf_sects[DEBUG_FRAME];
913 curcie = dbg->de_frame_cies;
914 cie_length = 0;
915 cur_off = 0;
916 cie_offs = (long *)
917 _dwarf_p_get_alloc(dbg, sizeof(long) * dbg->de_n_cie);
918 if (cie_offs == NULL) {
919 DWARF_P_DBG_ERROR(dbg, DW_DLE_CIE_OFFS_ALLOC, -1);
921 /* Generate cie number as we go along. This writes
922 all CIEs first before any FDEs, which is rather
923 different from the order a compiler might like (which
924 might be each CIE followed by its FDEs then the next CIE, and
925 so on). */
926 cie_no = 1;
927 while (curcie) {
928 char *code_al = 0;
929 int c_bytes = 0;
930 char *data_al = 0;
931 int d_bytes = 0;
932 int res = 0;
933 char buff1[ENCODE_SPACE_NEEDED];
934 char buff2[ENCODE_SPACE_NEEDED];
935 char buff3[ENCODE_SPACE_NEEDED];
936 char *augmentation = 0;
937 char *augmented_al = 0;
938 long augmented_fields_length = 0;
939 int a_bytes = 0;
941 res = _dwarf_pro_encode_leb128_nm(curcie->cie_code_align,
942 &c_bytes,
943 buff1, sizeof(buff1));
944 if (res != DW_DLV_OK) {
945 DWARF_P_DBG_ERROR(dbg, DW_DLE_CIE_OFFS_ALLOC, -1);
947 /* Before April 1999, the following was using an unsigned
948 encode. That worked ok even though the decoder used the
949 correct signed leb read, but doing the encode correctly
950 (according to the dwarf spec) saves space in the output file
951 and is completely compatible.
953 Note the actual stored amount on MIPS was 10 bytes (!) to
954 store the value -4. (hex)fc ffffffff ffffffff 01 The
955 libdwarf consumer consumed all 10 bytes too!
957 old version res =
958 _dwarf_pro_encode_leb128_nm(curcie->cie_data_align,
960 below is corrected signed version. */
961 res = _dwarf_pro_encode_signed_leb128_nm(curcie->cie_data_align,
962 &d_bytes,
963 buff2, sizeof(buff2));
964 if (res != DW_DLV_OK) {
965 DWARF_P_DBG_ERROR(dbg, DW_DLE_CIE_OFFS_ALLOC, -1);
967 code_al = buff1;
968 data_al = buff2;
970 /* get the correct offset */
971 if (firsttime) {
972 cie_offs[cie_no - 1] = 0;
973 firsttime = 0;
974 } else {
975 cie_offs[cie_no - 1] = cie_offs[cie_no - 2] +
976 (long) cie_length + BEGIN_LEN_SIZE;
978 cie_no++;
979 augmentation = curcie->cie_aug;
980 if (strcmp(augmentation, DW_CIE_AUGMENTER_STRING_V0) == 0) {
981 augmented_fields_length = 0;
982 res = _dwarf_pro_encode_leb128_nm(augmented_fields_length,
983 &a_bytes, buff3,
984 sizeof(buff3));
985 augmented_al = buff3;
986 if (res != DW_DLV_OK) {
987 DWARF_P_DBG_ERROR(dbg, DW_DLE_CIE_OFFS_ALLOC, -1);
989 cie_length = uwordb_size + /* cie_id */
990 sizeof(Dwarf_Ubyte) + /* cie version */
991 strlen(curcie->cie_aug) + 1 + /* augmentation */
992 c_bytes + /* code alignment factor */
993 d_bytes + /* data alignment factor */
994 sizeof(Dwarf_Ubyte) + /* return reg address */
995 a_bytes + /* augmentation length */
996 curcie->cie_inst_bytes;
997 } else {
998 cie_length = uwordb_size + /* cie_id */
999 sizeof(Dwarf_Ubyte) + /* cie version */
1000 strlen(curcie->cie_aug) + 1 + /* augmentation */
1001 c_bytes + d_bytes + sizeof(Dwarf_Ubyte) + /* return
1003 address
1005 curcie->cie_inst_bytes;
1007 pad = (int) PADDING(cie_length, upointer_size);
1008 cie_length += pad;
1009 GET_CHUNK(dbg, elfsectno, data, cie_length +
1010 BEGIN_LEN_SIZE, error);
1011 if (extension_size) {
1012 Dwarf_Unsigned x = DISTINGUISHED_VALUE;
1014 WRITE_UNALIGNED(dbg, (void *) data,
1015 (const void *) &x,
1016 sizeof(x), extension_size);
1017 data += extension_size;
1020 du = cie_length;
1021 /* total length of cie */
1022 WRITE_UNALIGNED(dbg, (void *) data,
1023 (const void *) &du, sizeof(du), uwordb_size);
1024 data += uwordb_size;
1026 /* cie-id is a special value. */
1027 du = DW_CIE_ID;
1028 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &du,
1029 sizeof(du), uwordb_size);
1030 data += uwordb_size;
1032 db = curcie->cie_version;
1033 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &db,
1034 sizeof(db), sizeof(Dwarf_Ubyte));
1035 data += sizeof(Dwarf_Ubyte);
1036 strcpy((char *) data, curcie->cie_aug);
1037 data += strlen(curcie->cie_aug) + 1;
1038 memcpy((void *) data, (const void *) code_al, c_bytes);
1039 data += c_bytes;
1040 memcpy((void *) data, (const void *) data_al, d_bytes);
1041 data += d_bytes;
1042 db = curcie->cie_ret_reg;
1043 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &db,
1044 sizeof(db), sizeof(Dwarf_Ubyte));
1045 data += sizeof(Dwarf_Ubyte);
1047 if (strcmp(augmentation, DW_CIE_AUGMENTER_STRING_V0) == 0) {
1048 memcpy((void *) data, (const void *) augmented_al, a_bytes);
1049 data += a_bytes;
1051 memcpy((void *) data, (const void *) curcie->cie_inst,
1052 curcie->cie_inst_bytes);
1053 data += curcie->cie_inst_bytes;
1054 for (i = 0; i < pad; i++) {
1055 *data = DW_CFA_nop;
1056 data++;
1058 curcie = curcie->cie_next;
1060 /* calculate current offset */
1061 cur_off = cie_offs[cie_no - 2] + cie_length + BEGIN_LEN_SIZE;
1063 /* write out fde's */
1064 curfde = dbg->de_frame_fdes;
1065 while (curfde) {
1066 Dwarf_P_Frame_Pgm curinst = 0;
1067 long fde_length = 0;
1068 int pad = 0;
1069 Dwarf_P_Cie cie_ptr = 0;
1070 Dwarf_Word cie_index = 0;
1071 Dwarf_Word index = 0;
1072 int oet_length = 0;
1073 int afl_length = 0;
1074 int res = 0;
1075 int v0_augmentation = 0;
1076 #if 0
1077 unsigned char *fde_start_point = 0;
1078 #endif
1079 char afl_buff[ENCODE_SPACE_NEEDED];
1081 /* Find the CIE associated with this fde. */
1082 cie_ptr = dbg->de_frame_cies;
1083 cie_index = curfde->fde_cie;
1084 index = 1; /* The cie_index of the first cie is 1,
1085 not 0. */
1086 while (cie_ptr && index < cie_index) {
1087 cie_ptr = cie_ptr->cie_next;
1088 index++;
1090 if (cie_ptr == NULL) {
1091 DWARF_P_DBG_ERROR(dbg, DW_DLE_CIE_NULL, -1);
1094 if (strcmp(cie_ptr->cie_aug, DW_CIE_AUGMENTER_STRING_V0) == 0) {
1095 v0_augmentation = 1;
1096 oet_length = sizeof(Dwarf_sfixed);
1097 /* encode the length of augmented fields. */
1098 res = _dwarf_pro_encode_leb128_nm(oet_length,
1099 &afl_length, afl_buff,
1100 sizeof(afl_buff));
1101 if (res != DW_DLV_OK) {
1102 DWARF_P_DBG_ERROR(dbg, DW_DLE_CIE_OFFS_ALLOC, -1);
1105 fde_length = curfde->fde_n_bytes + BEGIN_LEN_SIZE + /* cie
1106 pointer
1108 upointer_size + /* initial loc */
1109 upointer_size + /* address range */
1110 afl_length + /* augmented field length */
1111 oet_length; /* exception_table offset */
1112 } else {
1113 fde_length = curfde->fde_n_bytes + BEGIN_LEN_SIZE + /* cie
1114 pointer
1116 upointer_size + /* initial loc */
1117 upointer_size; /* address range */
1121 if (curfde->fde_die) {
1122 /* IRIX/MIPS extension:
1123 Using fde offset, generate DW_AT_MIPS_fde attribute for the
1124 die corresponding to this fde. */
1125 if(_dwarf_pro_add_AT_fde(dbg, curfde->fde_die, cur_off,
1126 error) < 0) {
1127 return -1;
1131 /* store relocation for cie pointer */
1132 res = dbg->de_reloc_name(dbg, DEBUG_FRAME, cur_off +
1133 BEGIN_LEN_SIZE /* r_offset */,
1134 dbg->de_sect_name_idx[DEBUG_FRAME],
1135 dwarf_drt_data_reloc, uwordb_size);
1136 if (res != DW_DLV_OK) {
1137 DWARF_P_DBG_ERROR(dbg, DW_DLE_CHUNK_ALLOC, -1);
1140 /* store relocation information for initial location */
1141 res = dbg->de_reloc_name(dbg, DEBUG_FRAME,
1142 cur_off + BEGIN_LEN_SIZE +
1143 upointer_size /* r_offset */,
1144 curfde->fde_r_symidx,
1145 dwarf_drt_data_reloc, upointer_size);
1146 if (res != DW_DLV_OK) {
1147 DWARF_P_DBG_ERROR(dbg, DW_DLE_CHUNK_ALLOC, -1);
1149 /* Store the relocation information for the
1150 offset_into_exception_info field, if the offset is valid (0
1151 is a valid offset). */
1152 if (v0_augmentation &&
1153 curfde->fde_offset_into_exception_tables >= 0) {
1155 res = dbg->de_reloc_name(dbg, DEBUG_FRAME,
1156 /* r_offset, where in cie this
1157 field starts */
1158 cur_off + BEGIN_LEN_SIZE +
1159 uwordb_size + 2 * upointer_size +
1160 afl_length,
1161 curfde->fde_exception_table_symbol,
1162 dwarf_drt_segment_rel,
1163 sizeof(Dwarf_sfixed));
1164 if (res != DW_DLV_OK) {
1165 DWARF_P_DBG_ERROR(dbg, DW_DLE_CHUNK_ALLOC, -1);
1169 /* adjust for padding */
1170 pad = (int) PADDING(fde_length, upointer_size);
1171 fde_length += pad;
1174 /* write out fde */
1175 GET_CHUNK(dbg, elfsectno, data, fde_length + BEGIN_LEN_SIZE,
1176 error);
1177 #if 0
1178 fde_start_point = data;
1179 #endif
1180 du = fde_length;
1182 if (extension_size) {
1183 Dwarf_Word x = DISTINGUISHED_VALUE;
1185 WRITE_UNALIGNED(dbg, (void *) data,
1186 (const void *) &x,
1187 sizeof(x), extension_size);
1188 data += extension_size;
1190 /* length */
1191 WRITE_UNALIGNED(dbg, (void *) data,
1192 (const void *) &du,
1193 sizeof(du), uwordb_size);
1194 data += uwordb_size;
1196 /* offset to cie */
1197 du = cie_offs[curfde->fde_cie - 1];
1198 WRITE_UNALIGNED(dbg, (void *) data,
1199 (const void *) &du,
1200 sizeof(du), uwordb_size);
1201 data += uwordb_size;
1203 du = curfde->fde_initloc;
1204 WRITE_UNALIGNED(dbg, (void *) data,
1205 (const void *) &du,
1206 sizeof(du), upointer_size);
1207 data += upointer_size;
1209 if (dbg->de_reloc_pair &&
1210 curfde->fde_end_symbol != 0 &&
1211 curfde->fde_addr_range == 0) {
1212 /* symbolic reloc, need reloc for length What if we
1213 really know the length? If so, should use the other
1214 part of 'if'. */
1215 Dwarf_Unsigned val;
1217 res = dbg->de_reloc_pair(dbg,
1218 /* DEBUG_ARANGES, */
1219 DEBUG_FRAME, cur_off + 2 * uwordb_size + upointer_size, /* r_offset
1221 curfde->fde_r_symidx,
1222 curfde->fde_end_symbol,
1223 dwarf_drt_first_of_length_pair,
1224 upointer_size);
1225 if (res != DW_DLV_OK) {
1227 _dwarf_p_error(dbg, error, DW_DLE_ALLOC_FAIL);
1228 return (0);
1232 /* arrange pre-calc so assem text can do .word end -
1233 begin + val (gets val from stream) */
1234 val = curfde->fde_end_symbol_offset -
1235 curfde->fde_initloc;
1236 WRITE_UNALIGNED(dbg, data,
1237 (const void *) &val,
1238 sizeof(val), upointer_size);
1239 data += upointer_size;
1240 } else {
1242 du = curfde->fde_addr_range;
1243 WRITE_UNALIGNED(dbg, (void *) data,
1244 (const void *) &du,
1245 sizeof(du), upointer_size);
1246 data += upointer_size;
1250 if (v0_augmentation) {
1251 /* write the encoded augmented field length. */
1252 memcpy((void *) data, (const void *) afl_buff, afl_length);
1253 data += afl_length;
1254 /* write the offset_into_exception_tables field. */
1255 dsw =
1256 (Dwarf_sfixed) curfde->fde_offset_into_exception_tables;
1257 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &dsw,
1258 sizeof(dsw), sizeof(Dwarf_sfixed));
1259 data += sizeof(Dwarf_sfixed);
1262 curinst = curfde->fde_inst;
1263 if(curfde->fde_block) {
1264 unsigned long size = curfde->fde_inst_block_size;
1265 memcpy((void *) data, (const void *) curfde->fde_block, size);
1266 data += size;
1267 } else {
1268 while (curinst) {
1269 db = curinst->dfp_opcode;
1270 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &db,
1271 sizeof(db), sizeof(Dwarf_Ubyte));
1272 data += sizeof(Dwarf_Ubyte);
1273 #if 0
1274 if (curinst->dfp_sym_index) {
1275 int res = dbg->de_reloc_name(dbg,
1276 DEBUG_FRAME,
1277 /* r_offset = */
1278 (data - fde_start_point) + cur_off + uwordb_size,
1279 curinst->dfp_sym_index,
1280 dwarf_drt_data_reloc,
1281 upointer_size);
1282 if (res != DW_DLV_OK) {
1283 _dwarf_p_error(dbg, error, DW_DLE_ALLOC_FAIL);
1284 return (0);
1287 #endif
1288 memcpy((void *) data,
1289 (const void *) curinst->dfp_args,
1290 curinst->dfp_nbytes);
1291 data += curinst->dfp_nbytes;
1292 curinst = curinst->dfp_next;
1295 /* padding */
1296 for (i = 0; i < pad; i++) {
1297 *data = DW_CFA_nop;
1298 data++;
1300 cur_off += fde_length + uwordb_size;
1301 curfde = curfde->fde_next;
1305 return (int) dbg->de_n_debug_sect;
1309 These functions remember all the markers we see along
1310 with the right offset in the .debug_info section so that
1311 we can dump them all back to the user with the section info.
1314 static int
1315 marker_init(Dwarf_P_Debug dbg,
1316 unsigned count)
1318 dbg->de_marker_n_alloc = count;
1319 dbg->de_markers = NULL;
1320 if (count > 0) {
1321 dbg->de_markers = _dwarf_p_get_alloc(dbg, sizeof(struct Dwarf_P_Marker_s) *
1322 dbg->de_marker_n_alloc);
1323 if (dbg->de_markers == NULL) {
1324 dbg->de_marker_n_alloc = 0;
1325 return -1;
1328 return 0;
1331 static int
1332 marker_add(Dwarf_P_Debug dbg,
1333 Dwarf_Unsigned offset,
1334 Dwarf_Unsigned marker)
1336 if (dbg->de_marker_n_alloc >= (dbg->de_marker_n_used + 1)) {
1337 unsigned n = dbg->de_marker_n_used++;
1338 dbg->de_markers[n].ma_offset = offset;
1339 dbg->de_markers[n].ma_marker = marker;
1340 return 0;
1343 return -1;
1346 Dwarf_Signed
1347 dwarf_get_die_markers(Dwarf_P_Debug dbg,
1348 Dwarf_P_Marker * marker_list, /* pointer to a pointer */
1349 Dwarf_Unsigned * marker_count,
1350 Dwarf_Error * error)
1352 if (marker_list == NULL || marker_count == NULL) {
1353 DWARF_P_DBG_ERROR(dbg, DW_DLE_IA, DW_DLV_BADADDR);
1355 if (dbg->de_marker_n_used != dbg->de_marker_n_alloc) {
1356 DWARF_P_DBG_ERROR(dbg, DW_DLE_MAF, DW_DLV_BADADDR);
1359 *marker_list = dbg->de_markers;
1360 *marker_count = dbg->de_marker_n_used;
1361 return DW_DLV_OK;
1364 /* These functions provide the offsets of DW_FORM_string
1365 attributes in the section section_index. These information
1366 will enable a producer app that is generating assembly
1367 text output to easily emit those attributes in ascii form
1368 without having to decode the byte stream.
1370 static int
1371 string_attr_init (Dwarf_P_Debug dbg,
1372 Dwarf_Signed section_index,
1373 unsigned count)
1375 Dwarf_P_Per_Sect_String_Attrs sect_sa = &dbg->de_sect_string_attr[section_index];
1377 sect_sa->sect_sa_n_alloc = count;
1378 sect_sa->sect_sa_list = NULL;
1379 if (count > 0) {
1380 sect_sa->sect_sa_section_number = section_index;
1381 sect_sa->sect_sa_list = _dwarf_p_get_alloc(dbg,
1382 sizeof(struct Dwarf_P_String_Attr_s)
1383 * sect_sa->sect_sa_n_alloc);
1384 if (sect_sa->sect_sa_list == NULL) {
1385 sect_sa->sect_sa_n_alloc = 0;
1386 return -1;
1389 return 0;
1392 static int
1393 string_attr_add (Dwarf_P_Debug dbg,
1394 Dwarf_Signed section_index,
1395 Dwarf_Unsigned offset,
1396 Dwarf_P_Attribute attr)
1398 Dwarf_P_Per_Sect_String_Attrs sect_sa = &dbg->de_sect_string_attr[section_index];
1399 if (sect_sa->sect_sa_n_alloc >= (sect_sa->sect_sa_n_used + 1)) {
1400 unsigned n = sect_sa->sect_sa_n_used++;
1401 sect_sa->sect_sa_list[n].sa_offset = offset;
1402 sect_sa->sect_sa_list[n].sa_nbytes = attr->ar_nbytes;
1403 return 0;
1406 return -1;
1410 dwarf_get_string_attributes_count(Dwarf_P_Debug dbg,
1411 Dwarf_Unsigned *
1412 count_of_sa_sections,
1413 int *drd_buffer_version,
1414 Dwarf_Error *error)
1416 int i;
1417 unsigned int count = 0;
1419 for (i = 0; i < NUM_DEBUG_SECTIONS; ++i) {
1420 if (dbg->de_sect_string_attr[i].sect_sa_n_used > 0) {
1421 ++count;
1424 *count_of_sa_sections = (Dwarf_Unsigned) count;
1425 *drd_buffer_version = DWARF_DRD_BUFFER_VERSION;
1427 return DW_DLV_OK;
1430 int
1431 dwarf_get_string_attributes_info(Dwarf_P_Debug dbg,
1432 Dwarf_Signed *elf_section_index,
1433 Dwarf_Unsigned *sect_sa_buffer_count,
1434 Dwarf_P_String_Attr *sect_sa_buffer,
1435 Dwarf_Error *error)
1437 int i;
1438 int next = dbg->de_sect_sa_next_to_return;
1440 for (i = next; i < NUM_DEBUG_SECTIONS; ++i) {
1441 Dwarf_P_Per_Sect_String_Attrs sect_sa = &dbg->de_sect_string_attr[i];
1442 if (sect_sa->sect_sa_n_used > 0) {
1443 dbg->de_sect_sa_next_to_return = i + 1;
1444 *elf_section_index = sect_sa->sect_sa_section_number;
1445 *sect_sa_buffer_count = sect_sa->sect_sa_n_used;
1446 *sect_sa_buffer = sect_sa->sect_sa_list;
1447 return DW_DLV_OK;
1450 return DW_DLV_NO_ENTRY;
1455 /*---------------------------------------------------------------
1456 Generate debug_info and debug_abbrev sections
1457 ---------------------------------------------------------------*/
1458 static int
1459 _dwarf_pro_generate_debuginfo(Dwarf_P_Debug dbg, Dwarf_Error * error)
1461 int elfsectno_of_debug_info = 0;
1462 int abbrevsectno = 0;
1463 unsigned char *data = 0;
1464 int cu_header_size = 0;
1465 Dwarf_P_Abbrev curabbrev = 0;
1466 Dwarf_P_Abbrev abbrev_head = 0;
1467 Dwarf_P_Abbrev abbrev_tail = 0;
1468 Dwarf_P_Die curdie = 0;
1469 Dwarf_P_Die first_child = 0;
1470 Dwarf_Word dw = 0;
1471 Dwarf_Unsigned du = 0;
1472 Dwarf_Half dh = 0;
1473 Dwarf_Ubyte db = 0;
1474 Dwarf_Half version = 0; /* Need 2 byte quantity. */
1475 Dwarf_Unsigned die_off = 0; /* Offset of die in debug_info. */
1476 int n_abbrevs = 0;
1477 int res = 0;
1478 unsigned marker_count = 0;
1479 unsigned string_attr_count = 0;
1480 unsigned string_attr_offset = 0;
1482 Dwarf_Small *start_info_sec = 0;
1484 int uwordb_size = dbg->de_offset_size;
1485 int extension_size = dbg->de_64bit_extension ? 4 : 0;
1487 abbrev_head = abbrev_tail = NULL;
1488 elfsectno_of_debug_info = dbg->de_elf_sects[DEBUG_INFO];
1490 /* write cu header */
1491 cu_header_size = BEGIN_LEN_SIZE + sizeof(Dwarf_Half) + /* version
1492 stamp
1494 uwordb_size + /* offset into abbrev table */
1495 sizeof(Dwarf_Ubyte); /* size of target address */
1496 GET_CHUNK(dbg, elfsectno_of_debug_info, data, cu_header_size,
1497 error);
1498 start_info_sec = data;
1499 if (extension_size) {
1500 du = DISTINGUISHED_VALUE;
1501 WRITE_UNALIGNED(dbg, (void *) data,
1502 (const void *) &du, sizeof(du), extension_size);
1503 data += extension_size;
1505 du = 0; /* length of debug_info, not counting
1506 this field itself (unknown at this
1507 point). */
1508 WRITE_UNALIGNED(dbg, (void *) data,
1509 (const void *) &du, sizeof(du), uwordb_size);
1510 data += uwordb_size;
1512 version = CURRENT_VERSION_STAMP; /* assume this length will not
1513 change */
1514 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &version,
1515 sizeof(version), sizeof(Dwarf_Half));
1516 data += sizeof(Dwarf_Half);
1518 du = 0; /* offset into abbrev table, not yet
1519 known. */
1520 WRITE_UNALIGNED(dbg, (void *) data,
1521 (const void *) &du, sizeof(du), uwordb_size);
1522 data += uwordb_size;
1525 db = dbg->de_pointer_size;
1527 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &db,
1528 sizeof(db), 1);
1530 /* We have filled the chunk we got with GET_CHUNK. At this point we
1531 no longer dare use "data" or "start_info_sec" as a pointer any
1532 longer except to refer to that first small chunk for the cu
1533 header. */
1535 curdie = dbg->de_dies;
1537 /* create AT_macro_info if appropriate */
1538 if (dbg->de_first_macinfo != NULL) {
1539 if (_dwarf_pro_add_AT_macro_info(dbg, curdie, 0, error) < 0)
1540 return -1;
1543 /* create AT_stmt_list attribute if necessary */
1544 if (dwarf_need_debug_line_section(dbg) == TRUE)
1545 if (_dwarf_pro_add_AT_stmt_list(dbg, curdie, error) < 0)
1546 return -1;
1548 die_off = cu_header_size;
1551 Relocation for abbrev offset in cu header store relocation
1552 record in linked list */
1553 res = dbg->de_reloc_name(dbg, DEBUG_INFO, BEGIN_LEN_SIZE +
1554 sizeof(Dwarf_Half),
1555 /* r_offset */
1556 dbg->de_sect_name_idx[DEBUG_ABBREV],
1557 dwarf_drt_data_reloc, uwordb_size);
1558 if (res != DW_DLV_OK) {
1559 DWARF_P_DBG_ERROR(dbg, DW_DLE_REL_ALLOC, -1);
1562 /* pass 0: only top level dies, add at_sibling attribute to those
1563 dies with children */
1564 first_child = curdie->di_child;
1565 while (first_child && first_child->di_right) {
1566 if (first_child->di_child)
1567 dwarf_add_AT_reference(dbg,
1568 first_child,
1569 DW_AT_sibling,
1570 first_child->di_right, error);
1571 first_child = first_child->di_right;
1574 /* pass 1: create abbrev info, get die offsets, calc relocations */
1575 marker_count = 0;
1576 string_attr_count = 0;
1577 while (curdie != NULL) {
1578 int nbytes = 0;
1579 Dwarf_P_Attribute curattr;
1580 Dwarf_P_Attribute new_first_attr;
1581 Dwarf_P_Attribute new_last_attr;
1582 char *space = 0;
1583 int res = 0;
1584 char buff1[ENCODE_SPACE_NEEDED];
1585 int i = 0;
1587 curdie->di_offset = die_off;
1589 if (curdie->di_marker != 0)
1590 marker_count++;
1592 curabbrev = _dwarf_pro_getabbrev(curdie, abbrev_head);
1593 if (curabbrev == NULL) {
1594 DWARF_P_DBG_ERROR(dbg, DW_DLE_ABBREV_ALLOC, -1);
1596 if (abbrev_head == NULL) {
1597 n_abbrevs = 1;
1598 curabbrev->abb_idx = n_abbrevs;
1599 abbrev_tail = abbrev_head = curabbrev;
1600 } else {
1601 /* check if its a new abbreviation, if yes, add to tail */
1602 if (curabbrev->abb_idx == 0) {
1603 n_abbrevs++;
1604 curabbrev->abb_idx = n_abbrevs;
1605 abbrev_tail->abb_next = curabbrev;
1606 abbrev_tail = curabbrev;
1609 res = _dwarf_pro_encode_leb128_nm(curabbrev->abb_idx,
1610 &nbytes,
1611 buff1, sizeof(buff1));
1612 if (res != DW_DLV_OK) {
1613 DWARF_P_DBG_ERROR(dbg, DW_DLE_ABBREV_ALLOC, -1);
1615 space = _dwarf_p_get_alloc(dbg, nbytes);
1616 if (space == NULL) {
1617 DWARF_P_DBG_ERROR(dbg, DW_DLE_ABBREV_ALLOC, -1);
1619 memcpy(space, buff1, nbytes);
1620 curdie->di_abbrev = space;
1621 curdie->di_abbrev_nbytes = nbytes;
1622 die_off += nbytes;
1624 /* Resorting the attributes!! */
1625 new_first_attr = new_last_attr = NULL;
1626 curattr = curdie->di_attrs;
1627 for (i = 0; i < (int)curabbrev->abb_n_attr; i++) {
1628 Dwarf_P_Attribute ca;
1629 Dwarf_P_Attribute cl;
1631 /* The following should always find an attribute! */
1632 for (ca = cl = curattr;
1633 ca && curabbrev->abb_attrs[i] != ca->ar_attribute;
1634 cl = ca, ca = ca->ar_next)
1638 if (!ca) {
1639 DWARF_P_DBG_ERROR(dbg,DW_DLE_ABBREV_ALLOC, -1);
1642 /* Remove the attribute from the old list. */
1643 if (ca == curattr) {
1644 curattr = ca->ar_next;
1645 } else {
1646 cl->ar_next = ca->ar_next;
1649 ca->ar_next = NULL;
1651 /* Add the attribute to the new list. */
1652 if (new_first_attr == NULL) {
1653 new_first_attr = new_last_attr = ca;
1654 } else {
1655 new_last_attr->ar_next = ca;
1656 new_last_attr = ca;
1660 curdie->di_attrs = new_first_attr;
1662 curattr = curdie->di_attrs;
1664 while (curattr) {
1665 if (curattr->ar_rel_type != R_MIPS_NONE) {
1666 switch (curattr->ar_attribute) {
1667 case DW_AT_stmt_list:
1668 curattr->ar_rel_symidx =
1669 dbg->de_sect_name_idx[DEBUG_LINE];
1670 break;
1671 case DW_AT_MIPS_fde:
1672 curattr->ar_rel_symidx =
1673 dbg->de_sect_name_idx[DEBUG_FRAME];
1674 break;
1675 case DW_AT_macro_info:
1676 curattr->ar_rel_symidx =
1677 dbg->de_sect_name_idx[DEBUG_MACINFO];
1678 break;
1679 default:
1680 break;
1682 res = dbg->de_reloc_name(dbg, DEBUG_INFO, die_off + curattr->ar_rel_offset, /* r_offset
1684 curattr->ar_rel_symidx,
1685 dwarf_drt_data_reloc,
1686 curattr->ar_reloc_len);
1688 if (res != DW_DLV_OK) {
1689 DWARF_P_DBG_ERROR(dbg, DW_DLE_REL_ALLOC, -1);
1693 if (curattr->ar_attribute_form == DW_FORM_string) {
1694 string_attr_count++;
1696 die_off += curattr->ar_nbytes;
1697 curattr = curattr->ar_next;
1700 /* depth first search */
1701 if (curdie->di_child)
1702 curdie = curdie->di_child;
1703 else {
1704 while (curdie != NULL && curdie->di_right == NULL) {
1705 curdie = curdie->di_parent;
1706 die_off++; /* since we are writing a null die at
1707 the end of each sibling chain */
1709 if (curdie != NULL)
1710 curdie = curdie->di_right;
1713 } /* end while (curdie != NULL) */
1715 res = marker_init(dbg, marker_count);
1716 if (res == -1) {
1717 DWARF_P_DBG_ERROR(dbg, DW_DLE_REL_ALLOC, -1);
1719 res = string_attr_init(dbg, DEBUG_INFO, string_attr_count);
1720 if (res == -1) {
1721 DWARF_P_DBG_ERROR(dbg, DW_DLE_REL_ALLOC, -1);
1724 /* Pass 2: Write out the die information Here 'data' is a
1725 temporary, one block for each GET_CHUNK. 'data' is overused. */
1726 curdie = dbg->de_dies;
1727 while (curdie != NULL) {
1728 Dwarf_P_Attribute curattr;
1730 if (curdie->di_marker != 0) {
1731 res = marker_add(dbg, curdie->di_offset, curdie->di_marker);
1732 if (res == -1) {
1733 DWARF_P_DBG_ERROR(dbg, DW_DLE_REL_ALLOC, -1);
1737 /* index to abbreviation table */
1738 GET_CHUNK(dbg, elfsectno_of_debug_info,
1739 data, curdie->di_abbrev_nbytes, error);
1741 memcpy((void *) data,
1742 (const void *) curdie->di_abbrev,
1743 curdie->di_abbrev_nbytes);
1745 /* Attribute values - need to fill in all form attributes */
1746 curattr = curdie->di_attrs;
1747 string_attr_offset = curdie->di_offset + curdie->di_abbrev_nbytes;
1749 while (curattr) {
1750 GET_CHUNK(dbg, elfsectno_of_debug_info, data,
1751 (unsigned long) curattr->ar_nbytes, error);
1752 switch (curattr->ar_attribute_form) {
1753 case DW_FORM_ref1:
1755 if (curattr->ar_ref_die->di_offset >
1756 (unsigned) 0xff) {
1757 DWARF_P_DBG_ERROR(dbg, DW_DLE_OFFSET_UFLW, -1);
1759 db = curattr->ar_ref_die->di_offset;
1760 WRITE_UNALIGNED(dbg, (void *) data,
1761 (const void *) &db,
1762 sizeof(db), sizeof(Dwarf_Ubyte));
1763 break;
1765 case DW_FORM_ref2:
1767 if (curattr->ar_ref_die->di_offset >
1768 (unsigned) 0xffff) {
1769 DWARF_P_DBG_ERROR(dbg, DW_DLE_OFFSET_UFLW, -1);
1771 dh = curattr->ar_ref_die->di_offset;
1772 WRITE_UNALIGNED(dbg, (void *) data,
1773 (const void *) &dh,
1774 sizeof(dh), sizeof(Dwarf_Half));
1775 break;
1777 case DW_FORM_ref_addr:
1779 /* curattr->ar_ref_die == NULL!
1781 * ref_addr doesn't take a CU-offset.
1782 * This is different than other refs.
1783 * This value will be set by the user of the
1784 * producer library using a relocation.
1785 * No need to set a value here.
1787 #if 0
1788 du = curattr->ar_ref_die->di_offset;
1790 /* ref to offset of die */
1791 WRITE_UNALIGNED(dbg, (void *) data,
1792 (const void *) &du,
1793 sizeof(du), uwordb_size);
1795 #endif
1796 break;
1799 case DW_FORM_ref4:
1801 if (curattr->ar_ref_die->di_offset >
1802 (unsigned) 0xffffffff) {
1803 DWARF_P_DBG_ERROR(dbg, DW_DLE_OFFSET_UFLW, -1);
1805 dw = (Dwarf_Word) curattr->ar_ref_die->di_offset;
1806 WRITE_UNALIGNED(dbg, (void *) data,
1807 (const void *) &dw,
1808 sizeof(dw), sizeof(Dwarf_ufixed));
1809 break;
1811 case DW_FORM_ref8:
1812 du = curattr->ar_ref_die->di_offset;
1813 WRITE_UNALIGNED(dbg, (void *) data,
1814 (const void *) &du,
1815 sizeof(du), sizeof(Dwarf_Unsigned));
1816 break;
1817 case DW_FORM_ref_udata:
1818 { /* unsigned leb128 offset */
1820 int nbytes;
1821 char buff1[ENCODE_SPACE_NEEDED];
1823 res =
1824 _dwarf_pro_encode_leb128_nm(curattr->
1825 ar_ref_die->
1826 di_offset, &nbytes,
1827 buff1,
1828 sizeof(buff1));
1829 if (res != DW_DLV_OK) {
1830 DWARF_P_DBG_ERROR(dbg, DW_DLE_ABBREV_ALLOC, -1);
1833 memcpy(data, buff1, nbytes);
1834 break;
1836 default:
1837 memcpy((void *) data,
1838 (const void *) curattr->ar_data,
1839 curattr->ar_nbytes);
1840 break;
1842 if (curattr->ar_attribute_form == DW_FORM_string) {
1843 string_attr_add(dbg, DEBUG_INFO, string_attr_offset, curattr);
1845 string_attr_offset += curattr->ar_nbytes;
1846 curattr = curattr->ar_next;
1849 /* depth first search */
1850 if (curdie->di_child)
1851 curdie = curdie->di_child;
1852 else {
1853 while (curdie != NULL && curdie->di_right == NULL) {
1854 GET_CHUNK(dbg, elfsectno_of_debug_info, data, 1, error);
1855 *data = '\0';
1856 curdie = curdie->di_parent;
1858 if (curdie != NULL)
1859 curdie = curdie->di_right;
1861 } /* end while (curdir != NULL) */
1863 /* Write out debug_info size */
1864 /* Dont include length field or extension bytes */
1865 du = die_off - BEGIN_LEN_SIZE;
1866 WRITE_UNALIGNED(dbg, (void *) (start_info_sec + extension_size),
1867 (const void *) &du, sizeof(du), uwordb_size);
1870 data = 0; /* Emphasise not usable now */
1872 /* Write out debug_abbrev section */
1873 abbrevsectno = dbg->de_elf_sects[DEBUG_ABBREV];
1875 curabbrev = abbrev_head;
1876 while (curabbrev) {
1877 char *val;
1878 int nbytes;
1879 int idx;
1880 int res;
1881 char buff1[ENCODE_SPACE_NEEDED];
1883 res = _dwarf_pro_encode_leb128_nm(curabbrev->abb_idx, &nbytes,
1884 buff1, sizeof(buff1));
1885 if (res != DW_DLV_OK) {
1886 DWARF_P_DBG_ERROR(dbg, DW_DLE_ABBREV_ALLOC, -1);
1889 GET_CHUNK(dbg, abbrevsectno, data, nbytes, error);
1890 val = buff1;
1891 memcpy((void *) data, (const void *) val, nbytes);
1892 res = _dwarf_pro_encode_leb128_nm(curabbrev->abb_tag, &nbytes,
1893 buff1, sizeof(buff1));
1894 if (res != DW_DLV_OK) {
1895 DWARF_P_DBG_ERROR(dbg, DW_DLE_ABBREV_ALLOC, -1);
1897 val = buff1;
1898 GET_CHUNK(dbg, abbrevsectno, data, nbytes, error);
1899 memcpy((void *) data, (const void *) val, nbytes);
1900 db = curabbrev->abb_children;
1901 GET_CHUNK(dbg, abbrevsectno, data, sizeof(Dwarf_Ubyte), error);
1902 WRITE_UNALIGNED(dbg, (void *) data, (const void *) &db,
1903 sizeof(db), sizeof(Dwarf_Ubyte));
1905 /* add attributes and forms */
1906 for (idx = 0; idx < curabbrev->abb_n_attr; idx++) {
1907 res = _dwarf_pro_encode_leb128_nm(curabbrev->abb_attrs[idx],
1908 &nbytes,
1909 buff1, sizeof(buff1));
1910 if (res != DW_DLV_OK) {
1911 DWARF_P_DBG_ERROR(dbg, DW_DLE_ABBREV_ALLOC, -1);
1913 val = buff1;
1914 GET_CHUNK(dbg, abbrevsectno, data, nbytes, error);
1915 memcpy((void *) data, (const void *) val, nbytes);
1916 res = _dwarf_pro_encode_leb128_nm(curabbrev->abb_forms[idx],
1917 &nbytes,
1918 buff1, sizeof(buff1));
1919 if (res != DW_DLV_OK) {
1920 DWARF_P_DBG_ERROR(dbg, DW_DLE_ABBREV_ALLOC, -1);
1922 val = buff1;
1923 GET_CHUNK(dbg, abbrevsectno, data, nbytes, error);
1924 memcpy((void *) data, (const void *) val, nbytes);
1926 GET_CHUNK(dbg, abbrevsectno, data, 2, error); /* two zeros,
1927 for last
1928 entry, see
1929 dwarf2 sec
1930 7.5.3 */
1931 *data = 0;
1932 data++;
1933 *data = 0;
1935 curabbrev = curabbrev->abb_next;
1938 GET_CHUNK(dbg, abbrevsectno, data, 1, error); /* one zero,
1939 for end of
1940 cu, see
1941 dwarf2 sec
1942 7.5.3 */
1943 *data = 0;
1946 return (int) dbg->de_n_debug_sect;
1950 /*---------------------------------------------------------------------
1951 Get a buffer of section data.
1952 section_idx is the elf-section number that this data applies to.
1953 length shows length of returned data
1954 ----------------------------------------------------------------------*/
1955 /*ARGSUSED*/ /* pretend all args used */
1956 Dwarf_Ptr
1957 dwarf_get_section_bytes(Dwarf_P_Debug dbg,
1958 Dwarf_Signed dwarf_section,
1959 Dwarf_Signed * section_idx,
1960 Dwarf_Unsigned * length, Dwarf_Error * error)
1962 Dwarf_Ptr buf;
1964 if (dbg->de_version_magic_number != PRO_VERSION_MAGIC) {
1965 DWARF_P_DBG_ERROR(dbg, DW_DLE_IA, NULL);
1968 if (dbg->de_debug_sects == 0) {
1969 /* no more data !! */
1970 return NULL;
1972 if (dbg->de_debug_sects->ds_elf_sect_no == MAGIC_SECT_NO) {
1973 /* no data ever entered !! */
1974 return NULL;
1976 *section_idx = dbg->de_debug_sects->ds_elf_sect_no;
1977 *length = dbg->de_debug_sects->ds_nbytes;
1979 buf = (Dwarf_Ptr *) dbg->de_debug_sects->ds_data;
1981 dbg->de_debug_sects = dbg->de_debug_sects->ds_next;
1983 /* We may want to call the section stuff more than once: see
1984 dwarf_reset_section_bytes() do not do: dbg->de_n_debug_sect--; */
1986 return buf;
1990 No errors possible.
1992 void
1993 dwarf_reset_section_bytes(Dwarf_P_Debug dbg)
1995 dbg->de_debug_sects = dbg->de_first_debug_sect;
1996 /* No need to reset; commented out decrement. dbg->de_n_debug_sect
1997 = ???; */
1998 dbg->de_reloc_next_to_return = 0;
1999 dbg->de_sect_sa_next_to_return = 0;
2003 Storage handler. Gets either a new chunk of memory, or
2004 a pointer in existing memory, from the linked list attached
2005 to dbg at de_debug_sects, depending on size of nbytes
2007 Assume dbg not null, checked in top level routine
2009 Returns a pointer to the allocated buffer space for the
2010 lib to fill in, predincrements next-to-use count so the
2011 space requested is already counted 'used'
2012 when this returns (ie, reserved).
2015 Dwarf_Small *
2016 _dwarf_pro_buffer(Dwarf_P_Debug dbg,
2017 int elfsectno, unsigned long nbytes)
2019 Dwarf_P_Section_Data cursect;
2022 cursect = dbg->de_current_active_section;
2023 /* By using MAGIC_SECT_NO we allow the following MAGIC_SECT_NO must
2024 not match any legit section number. test to have just two
2025 clauses (no NULL pointer test) See dwarf_producer_init(). */
2026 if ((cursect->ds_elf_sect_no != elfsectno) ||
2027 ((cursect->ds_nbytes + nbytes) > cursect->ds_orig_alloc)
2030 /* Either the elf section has changed or there is not enough
2031 space in the current section.
2033 Create a new Dwarf_P_Section_Data_s for the chunk. and have
2034 space 'on the end' for the buffer itself so we just do one
2035 malloc (not two).
2038 unsigned long space = nbytes;
2040 if (nbytes < CHUNK_SIZE)
2041 space = CHUNK_SIZE;
2043 cursect = (Dwarf_P_Section_Data)
2044 _dwarf_p_get_alloc(dbg,
2045 sizeof(struct Dwarf_P_Section_Data_s)
2046 + space);
2049 if (cursect == NULL)
2050 return (NULL);
2052 /* _dwarf_p_get_alloc zeroes the space... */
2054 cursect->ds_data = (char *) cursect +
2055 sizeof(struct Dwarf_P_Section_Data_s);
2056 cursect->ds_orig_alloc = space;
2057 cursect->ds_elf_sect_no = elfsectno;
2058 cursect->ds_nbytes = nbytes; /* reserve this number of bytes
2059 of space for caller to fill
2060 in */
2062 /* Now link on the end of the list, and mark this one as the
2063 current one */
2065 if (dbg->de_debug_sects->ds_elf_sect_no == MAGIC_SECT_NO) {
2066 /* the only entry is the special one for 'no entry' so
2067 delete that phony one while adding this initial real
2068 one. */
2069 dbg->de_debug_sects = cursect;
2070 dbg->de_current_active_section = cursect;
2071 dbg->de_first_debug_sect = cursect;
2072 } else {
2073 dbg->de_current_active_section->ds_next = cursect;
2074 dbg->de_current_active_section = cursect;
2076 dbg->de_n_debug_sect++;
2078 return ((Dwarf_Small *) cursect->ds_data);
2081 /* There is enough space in the current buffer */
2083 Dwarf_Small *space_for_caller = (Dwarf_Small *)
2084 (cursect->ds_data + cursect->ds_nbytes);
2086 cursect->ds_nbytes += nbytes;
2087 return space_for_caller;
2092 /*------------------------------------------------------------
2093 Given address advance and line advance, it gives
2094 either special opcode, or a number < 0
2095 ------------------------------------------------------------*/
2096 static int
2097 _dwarf_pro_get_opc(Dwarf_Unsigned addr_adv, int line_adv)
2099 int opc;
2101 addr_adv = addr_adv / MIN_INST_LENGTH;
2102 if (line_adv == 0 && addr_adv == 0)
2103 return OPC_INCS_ZERO;
2104 if (line_adv >= LINE_BASE && line_adv < LINE_BASE + LINE_RANGE) {
2105 opc =
2106 (line_adv - LINE_BASE) + (addr_adv * LINE_RANGE) +
2107 OPCODE_BASE;
2108 if (opc > 255)
2109 return OPC_OUT_OF_RANGE;
2110 return opc;
2111 } else
2112 return LINE_OUT_OF_RANGE;
2115 /*-----------------------------------------------------------------------
2116 Handles abbreviations. It takes a die, searches through
2117 current list of abbreviations for matching one. If it
2118 finds one, it returns a pointer to it, and if it doesnt,
2119 it returns a new one. Upto the user of this function to
2120 link it up to the abbreviation head. If its a new one,
2121 abb_idx has 0.
2122 -----------------------------------------------------------------------*/
2123 static Dwarf_P_Abbrev
2124 _dwarf_pro_getabbrev(Dwarf_P_Die die, Dwarf_P_Abbrev head)
2126 Dwarf_P_Abbrev curabbrev;
2127 Dwarf_P_Attribute curattr;
2128 int res1;
2129 int nattrs;
2130 int match;
2131 Dwarf_ufixed *forms = 0;
2132 Dwarf_ufixed *attrs = 0;
2134 curabbrev = head;
2135 while (curabbrev) {
2136 if ((die->di_tag == curabbrev->abb_tag) &&
2137 ((die->di_child != NULL &&
2138 curabbrev->abb_children == DW_CHILDREN_yes) ||
2139 (die->di_child == NULL &&
2140 curabbrev->abb_children == DW_CHILDREN_no)) &&
2141 (die->di_n_attr == curabbrev->abb_n_attr)) {
2143 /* There is a chance of a match. */
2144 curattr = die->di_attrs;
2145 match = 1; /* Assume match found. */
2146 while (match && curattr) {
2147 res1 = _dwarf_pro_match_attr(curattr,
2148 curabbrev,
2149 (int) curabbrev->
2150 abb_n_attr);
2151 if (res1 == 0)
2152 match = 0;
2153 curattr = curattr->ar_next;
2155 if (match == 1)
2156 return curabbrev;
2158 curabbrev = curabbrev->abb_next;
2161 /* no match, create new abbreviation */
2162 if (die->di_n_attr != 0) {
2163 forms = (Dwarf_ufixed *)
2164 _dwarf_p_get_alloc(die->di_dbg,
2165 sizeof(Dwarf_ufixed) * die->di_n_attr);
2166 if (forms == NULL)
2167 return NULL;
2168 attrs = (Dwarf_ufixed *)
2169 _dwarf_p_get_alloc(die->di_dbg,
2170 sizeof(Dwarf_ufixed) * die->di_n_attr);
2171 if (attrs == NULL)
2172 return NULL;
2174 nattrs = 0;
2175 curattr = die->di_attrs;
2176 while (curattr) {
2177 attrs[nattrs] = curattr->ar_attribute;
2178 forms[nattrs] = curattr->ar_attribute_form;
2179 nattrs++;
2180 curattr = curattr->ar_next;
2183 curabbrev = (Dwarf_P_Abbrev)
2184 _dwarf_p_get_alloc(die->di_dbg, sizeof(struct Dwarf_P_Abbrev_s));
2185 if (curabbrev == NULL)
2186 return NULL;
2188 if (die->di_child == NULL)
2189 curabbrev->abb_children = DW_CHILDREN_no;
2190 else
2191 curabbrev->abb_children = DW_CHILDREN_yes;
2192 curabbrev->abb_tag = die->di_tag;
2193 curabbrev->abb_attrs = attrs;
2194 curabbrev->abb_forms = forms;
2195 curabbrev->abb_n_attr = die->di_n_attr;
2196 curabbrev->abb_idx = 0;
2197 curabbrev->abb_next = NULL;
2199 return curabbrev;
2202 /*------------------------------------------------------------------
2203 Tries to see if given attribute and form combination
2204 exists in the given abbreviation
2205 -------------------------------------------------------------------*/
2206 static int
2207 _dwarf_pro_match_attr(Dwarf_P_Attribute attr,
2208 Dwarf_P_Abbrev abbrev, int no_attr)
2210 int i;
2211 int found = 0;
2213 for (i = 0; i < no_attr; i++) {
2214 if (attr->ar_attribute == abbrev->abb_attrs[i] &&
2215 attr->ar_attribute_form == abbrev->abb_forms[i]) {
2216 found = 1;
2217 break;
2220 return found;