import less(1)
[unleashed/tickless.git] / usr / src / lib / libdwarf / common / pro_frame.c
blobbd1ef6a637d037e6b23f59be92f6427a1d56da15
1 /*
3 Copyright (C) 2000,2004 Silicon Graphics, Inc. All Rights Reserved.
5 This program is free software; you can redistribute it and/or modify it
6 under the terms of version 2.1 of the GNU Lesser General Public License
7 as published by the Free Software Foundation.
9 This program is distributed in the hope that it would be useful, but
10 WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
13 Further, this software is distributed without any warranty that it is
14 free of the rightful claim of any third person regarding infringement
15 or the like. Any license provided herein, whether implied or
16 otherwise, applies only to this software file. Patent licenses, if
17 any, provided herein do not apply to combinations of this program with
18 other software, or any other product whatsoever.
20 You should have received a copy of the GNU Lesser General Public
21 License along with this program; if not, write the Free Software
22 Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston MA 02110-1301,
23 USA.
25 Contact information: Silicon Graphics, Inc., 1500 Crittenden Lane,
26 Mountain View, CA 94043, or:
28 http://www.sgi.com
30 For further information regarding this notice, see:
32 http://oss.sgi.com/projects/GenInfo/NoticeExplan
38 #include "config.h"
39 #include "libdwarfdefs.h"
40 #include <stdio.h>
41 #include <string.h>
42 #include <limits.h>
43 #include "pro_incl.h"
44 #include "pro_frame.h"
46 static void _dwarf_pro_add_to_fde(Dwarf_P_Fde fde,
47 Dwarf_P_Frame_Pgm inst);
49 /*-------------------------------------------------------------------------
50 This function adds a cie struct to the debug pointer. Its in the
51 form of a linked list.
52 augmenter: string reps augmentation (implementation defined)
53 code_align: alignment of code
54 data_align: alignment of data
55 init_bytes: byts having initial instructions
56 init_n_bytes: number of bytes of initial instructions
57 --------------------------------------------------------------------------*/
58 Dwarf_Unsigned
59 dwarf_add_frame_cie(Dwarf_P_Debug dbg,
60 char *augmenter,
61 Dwarf_Small code_align,
62 Dwarf_Small data_align,
63 Dwarf_Small return_reg,
64 Dwarf_Ptr init_bytes,
65 Dwarf_Unsigned init_n_bytes, Dwarf_Error * error)
67 Dwarf_P_Cie curcie;
69 if (dbg->de_frame_cies == NULL) {
70 dbg->de_frame_cies = (Dwarf_P_Cie)
71 _dwarf_p_get_alloc(dbg, sizeof(struct Dwarf_P_Cie_s));
72 if (dbg->de_frame_cies == NULL) {
73 DWARF_P_DBG_ERROR(dbg, DW_DLE_CIE_ALLOC, DW_DLV_NOCOUNT);
75 curcie = dbg->de_frame_cies;
76 dbg->de_n_cie = 1;
77 dbg->de_last_cie = curcie;
78 } else {
79 curcie = dbg->de_last_cie;
80 curcie->cie_next = (Dwarf_P_Cie)
81 _dwarf_p_get_alloc(dbg, sizeof(struct Dwarf_P_Cie_s));
82 if (curcie->cie_next == NULL) {
83 DWARF_P_DBG_ERROR(dbg, DW_DLE_CIE_ALLOC, DW_DLV_NOCOUNT);
85 curcie = curcie->cie_next;
86 dbg->de_n_cie++;
87 dbg->de_last_cie = curcie;
89 curcie->cie_version = DW_CIE_VERSION;
90 curcie->cie_aug = augmenter;
91 curcie->cie_code_align = code_align;
92 curcie->cie_data_align = data_align;
93 curcie->cie_ret_reg = return_reg;
94 curcie->cie_inst = (char *) init_bytes;
95 curcie->cie_inst_bytes = (long) init_n_bytes;
96 curcie->cie_next = NULL;
97 return dbg->de_n_cie;
101 /*-------------------------------------------------------------------------
102 This functions adds a fde struct to the debug pointer. Its in the
103 form of a linked list.
104 die: subprogram/function die corresponding to this fde
105 cie: cie referred to by this fde, obtained from call to
106 add_frame_cie() routine.
107 virt_addr: beginning address
108 code_len: length of code reps by the fde
109 --------------------------------------------------------------------------*/
110 /*ARGSUSED*/ /* pretend all args used */
111 Dwarf_Unsigned
112 dwarf_add_frame_fde(Dwarf_P_Debug dbg,
113 Dwarf_P_Fde fde,
114 Dwarf_P_Die die,
115 Dwarf_Unsigned cie,
116 Dwarf_Unsigned virt_addr,
117 Dwarf_Unsigned code_len,
118 Dwarf_Unsigned symidx, Dwarf_Error * error)
120 return dwarf_add_frame_fde_b(dbg, fde, die, cie, virt_addr,
121 code_len, symidx, 0, 0, error);
124 /*ARGSUSED10*/
125 Dwarf_Unsigned
126 dwarf_add_frame_fde_b(Dwarf_P_Debug dbg,
127 Dwarf_P_Fde fde,
128 Dwarf_P_Die die,
129 Dwarf_Unsigned cie,
130 Dwarf_Unsigned virt_addr,
131 Dwarf_Unsigned code_len,
132 Dwarf_Unsigned symidx,
133 Dwarf_Unsigned symidx_of_end,
134 Dwarf_Addr offset_from_end_sym,
135 Dwarf_Error * error)
137 Dwarf_P_Fde curfde;
139 fde->fde_die = die;
140 fde->fde_cie = (long) cie;
141 fde->fde_initloc = virt_addr;
142 fde->fde_r_symidx = symidx;
143 fde->fde_addr_range = code_len;
144 fde->fde_offset_into_exception_tables = DW_DLX_NO_EH_OFFSET;
145 fde->fde_exception_table_symbol = 0;
146 fde->fde_end_symbol_offset = offset_from_end_sym;
147 fde->fde_end_symbol = symidx_of_end;
148 fde->fde_dbg = dbg;
150 curfde = dbg->de_last_fde;
151 if (curfde == NULL) {
152 dbg->de_frame_fdes = fde;
153 dbg->de_last_fde = fde;
154 dbg->de_n_fde = 1;
155 } else {
156 curfde->fde_next = fde;
157 dbg->de_last_fde = fde;
158 dbg->de_n_fde++;
160 return dbg->de_n_fde;
163 /*-------------------------------------------------------------------------
164 This functions adds information to an fde. The fde is
165 linked into the linked list of fde's maintained in the Dwarf_P_Debug
166 structure.
167 dbg: The debug descriptor.
168 fde: The fde to be added.
169 die: subprogram/function die corresponding to this fde
170 cie: cie referred to by this fde, obtained from call to
171 add_frame_cie() routine.
172 virt_addr: beginning address
173 code_len: length of code reps by the fde
174 symidx: The symbol id of the symbol wrt to which relocation needs
175 to be performed for 'virt_addr'.
176 offset_into_exception_tables: The start of exception tables for
177 this function (indicated as an offset into the exception
178 tables). A value of -1 indicates that there is no exception
179 table entries associated with this function.
180 exception_table_symbol: The symbol id of the section for exception
181 tables wrt to which the offset_into_exception_tables will
182 be relocated.
183 --------------------------------------------------------------------------*/
184 Dwarf_Unsigned
185 dwarf_add_frame_info(Dwarf_P_Debug dbg,
186 Dwarf_P_Fde fde,
187 Dwarf_P_Die die,
188 Dwarf_Unsigned cie,
189 Dwarf_Unsigned virt_addr,
190 Dwarf_Unsigned code_len,
191 Dwarf_Unsigned symidx,
192 Dwarf_Signed offset_into_exception_tables,
193 Dwarf_Unsigned exception_table_symbol,
194 Dwarf_Error * error)
197 return dwarf_add_frame_info_b(dbg, fde, die, cie, virt_addr,
198 code_len, symidx,
199 /* end_symbol */ 0,
200 /* offset_from_end */ 0,
201 offset_into_exception_tables,
202 exception_table_symbol, error);
206 /*ARGSUSED*/ /* pretend all args used */
207 Dwarf_Unsigned
208 dwarf_add_frame_info_b(Dwarf_P_Debug dbg,
209 Dwarf_P_Fde fde,
210 Dwarf_P_Die die,
211 Dwarf_Unsigned cie,
212 Dwarf_Unsigned virt_addr,
213 Dwarf_Unsigned code_len,
214 Dwarf_Unsigned symidx,
215 Dwarf_Unsigned end_symidx,
216 Dwarf_Unsigned offset_from_end_symbol,
217 Dwarf_Signed offset_into_exception_tables,
218 Dwarf_Unsigned exception_table_symbol,
219 Dwarf_Error * error)
221 Dwarf_P_Fde curfde;
223 fde->fde_die = die;
224 fde->fde_cie = (long) cie;
225 fde->fde_initloc = virt_addr;
226 fde->fde_r_symidx = symidx;
227 fde->fde_addr_range = code_len;
228 fde->fde_offset_into_exception_tables =
229 offset_into_exception_tables;
230 fde->fde_exception_table_symbol = exception_table_symbol;
231 fde->fde_end_symbol_offset = offset_from_end_symbol;
232 fde->fde_end_symbol = end_symidx;
233 fde->fde_dbg = dbg;
235 curfde = dbg->de_last_fde;
236 if (curfde == NULL) {
237 dbg->de_frame_fdes = fde;
238 dbg->de_last_fde = fde;
239 dbg->de_n_fde = 1;
240 } else {
241 curfde->fde_next = fde;
242 dbg->de_last_fde = fde;
243 dbg->de_n_fde++;
245 return dbg->de_n_fde;
248 /* This is an alternate to inserting frame instructions
249 one instruction at a time. But use either this
250 or instruction level, not both in one fde. */
252 dwarf_insert_fde_inst_bytes(Dwarf_P_Debug dbg,
253 Dwarf_P_Fde fde,Dwarf_Unsigned len, Dwarf_Ptr ibytes,
254 Dwarf_Error *error)
256 if( len == 0) {
257 return DW_DLV_OK;
259 if(fde->fde_block || fde->fde_inst) {
260 DWARF_P_DBG_ERROR(dbg, DW_DLE_DUPLICATE_INST_BLOCK,
261 (int)DW_DLV_BADADDR);
263 fde->fde_block = (Dwarf_Ptr)_dwarf_p_get_alloc(dbg, len);
264 memcpy(fde->fde_block,ibytes,len);
265 fde->fde_inst_block_size = len;
266 fde->fde_n_bytes += len;
267 return DW_DLV_OK;
272 /*-------------------------------------------------------------------
273 Create a new fde.
274 ---------------------------------------------------------------------*/
275 Dwarf_P_Fde
276 dwarf_new_fde(Dwarf_P_Debug dbg, Dwarf_Error * error)
278 Dwarf_P_Fde fde;
280 fde = (Dwarf_P_Fde)
281 _dwarf_p_get_alloc(dbg, sizeof(struct Dwarf_P_Fde_s));
282 if (fde == NULL) {
283 DWARF_P_DBG_ERROR(dbg, DW_DLE_FDE_ALLOC,
284 (Dwarf_P_Fde) DW_DLV_BADADDR);
287 fde->fde_uwordb_size = dbg->de_offset_size;
289 return fde;
293 /*------------------------------------------------------------------------
294 Add a cfe_offset instruction to the fde passed in.
295 -------------------------------------------------------------------------*/
296 Dwarf_P_Fde
297 dwarf_fde_cfa_offset(Dwarf_P_Fde fde,
298 Dwarf_Unsigned reg,
299 Dwarf_Signed offset, Dwarf_Error * error)
301 Dwarf_Ubyte opc, regno;
302 char *ptr;
303 Dwarf_P_Frame_Pgm curinst;
304 int nbytes;
305 int res;
306 char buff1[ENCODE_SPACE_NEEDED];
307 Dwarf_P_Debug dbg = fde->fde_dbg;
309 curinst = (Dwarf_P_Frame_Pgm)
310 _dwarf_p_get_alloc(dbg, sizeof(struct Dwarf_P_Frame_Pgm_s));
311 if (curinst == NULL) {
312 DWARF_P_DBG_ERROR(dbg, DW_DLE_FPGM_ALLOC,
313 (Dwarf_P_Fde) DW_DLV_BADADDR);
315 opc = DW_CFA_offset;
316 regno = reg;
317 if (regno & 0xc0) {
318 DWARF_P_DBG_ERROR(dbg, DW_DLE_REGNO_OVFL,
319 (Dwarf_P_Fde) DW_DLV_BADADDR);
321 opc = opc | regno; /* lower 6 bits are register number */
322 curinst->dfp_opcode = opc;
323 res = _dwarf_pro_encode_leb128_nm(offset, &nbytes,
324 buff1, sizeof(buff1));
325 if (res != DW_DLV_OK) {
326 _dwarf_p_error(dbg, error, DW_DLE_STRING_ALLOC);
327 return ((Dwarf_P_Fde) DW_DLV_BADADDR);
329 ptr = (char *) _dwarf_p_get_alloc(dbg, nbytes);
330 if (ptr == NULL) {
331 _dwarf_p_error(dbg, error, DW_DLE_STRING_ALLOC);
332 return ((Dwarf_P_Fde) DW_DLV_BADADDR);
334 memcpy(ptr, buff1, nbytes);
336 curinst->dfp_args = ptr;
337 curinst->dfp_nbytes = nbytes;
338 curinst->dfp_next = NULL;
340 _dwarf_pro_add_to_fde(fde, curinst);
341 return fde;
345 Generic routine to add opcode to fde instructions. val1 and
346 val2 are parameters whose interpretation depends on the 'op'.
348 This does not work properly for DW_DLC_SYMBOLIC_RELOCATIONS
349 for DW_CFA_set_loc or DW_DVA_advance_loc* 'op', as
350 these ops normally are addresses or (DW_CFA_set_loc)
351 or code lengths (DW_DVA_advance_loc*) and such must be
352 represented with relocations and symbol indices for
353 DW_DLC_SYMBOLIC_RELOCATIONS.
355 This does not treat all DW_CFA instructions yet.
357 For certain operations a val? value must be
358 signed (though passed in as unsigned here).
360 Currently this does not check that the frame
361 version is 3(for dwarf3) or 4 (for dwarf4)
362 when applying operations that are only valid for
363 dwarf3 or dwarf4.
366 Dwarf_P_Fde
367 dwarf_add_fde_inst(Dwarf_P_Fde fde,
368 Dwarf_Small op,
369 Dwarf_Unsigned val1,
370 Dwarf_Unsigned val2, Dwarf_Error * error)
372 Dwarf_P_Frame_Pgm curinst;
373 int nbytes, nbytes1, nbytes2;
374 Dwarf_Ubyte db;
375 Dwarf_Half dh;
376 Dwarf_Word dw;
377 Dwarf_Unsigned du;
378 char *ptr;
379 int res;
380 char buff1[ENCODE_SPACE_NEEDED];
381 char buff2[ENCODE_SPACE_NEEDED];
382 Dwarf_P_Debug dbg = fde->fde_dbg;
383 /* This is a hack telling the code when to transform
384 a value to a signed leb number. */
385 int signed_second = 0;
386 int signed_first = 0;
389 nbytes = 0;
390 ptr = NULL;
391 curinst = (Dwarf_P_Frame_Pgm)
392 _dwarf_p_get_alloc(dbg, sizeof(struct Dwarf_P_Frame_Pgm_s));
393 if (curinst == NULL) {
394 _dwarf_p_error(dbg, error, DW_DLE_FPGM_ALLOC);
395 return ((Dwarf_P_Fde) DW_DLV_BADADDR);
398 switch (op) {
400 case DW_CFA_advance_loc:
401 if (val1 <= 0x3f) {
402 db = val1;
403 op |= db;
405 /* test not portable FIX */
406 else if (val1 <= UCHAR_MAX) {
407 op = DW_CFA_advance_loc1;
408 db = val1;
409 ptr = (char *) _dwarf_p_get_alloc(dbg, 1);
410 if (ptr == NULL) {
411 _dwarf_p_error(dbg, error, DW_DLE_STRING_ALLOC);
412 return ((Dwarf_P_Fde) DW_DLV_BADADDR);
414 memcpy((void *) ptr, (const void *) &db, 1);
415 nbytes = 1;
417 /* test not portable FIX */
418 else if (val1 <= USHRT_MAX) {
419 op = DW_CFA_advance_loc2;
420 dh = val1;
421 ptr = (char *) _dwarf_p_get_alloc(dbg, 2);
422 if (ptr == NULL) {
423 _dwarf_p_error(dbg, error, DW_DLE_STRING_ALLOC);
424 return ((Dwarf_P_Fde) DW_DLV_BADADDR);
426 memcpy((void *) ptr, (const void *) &dh, 2);
427 nbytes = 2;
429 /* test not portable FIX */
430 else if (val1 <= ULONG_MAX) {
431 op = DW_CFA_advance_loc4;
432 dw = (Dwarf_Word) val1;
433 ptr = (char *) _dwarf_p_get_alloc(dbg, 4);
434 if (ptr == NULL) {
435 _dwarf_p_error(dbg, error, DW_DLE_STRING_ALLOC);
436 return ((Dwarf_P_Fde) DW_DLV_BADADDR);
438 memcpy((void *) ptr, (const void *) &dw, 4);
439 nbytes = 4;
440 } else {
441 op = DW_CFA_MIPS_advance_loc8;
442 du = val1;
443 ptr =
444 (char *) _dwarf_p_get_alloc(dbg,
445 sizeof(Dwarf_Unsigned));
446 if (ptr == NULL) {
447 _dwarf_p_error(dbg, error, DW_DLE_STRING_ALLOC);
448 return ((Dwarf_P_Fde) DW_DLV_BADADDR);
450 memcpy((void *) ptr, (const void *) &du, 8);
451 nbytes = 8;
453 break;
455 case DW_CFA_offset:
456 if (val1 <= MAX_6_BIT_VALUE) {
457 db = val1;
458 op |= db;
459 res = _dwarf_pro_encode_leb128_nm(val2, &nbytes,
460 buff1, sizeof(buff1));
461 if (res != DW_DLV_OK) {
462 _dwarf_p_error(dbg, error, DW_DLE_STRING_ALLOC);
463 return ((Dwarf_P_Fde) DW_DLV_BADADDR);
465 ptr = (char *) _dwarf_p_get_alloc(dbg, nbytes);
466 if (ptr == NULL) {
467 _dwarf_p_error(dbg, error, DW_DLE_STRING_ALLOC);
468 return ((Dwarf_P_Fde) DW_DLV_BADADDR);
470 memcpy(ptr, buff1, nbytes);
472 } else {
473 op = DW_CFA_offset_extended;
474 goto two_leb;
476 break;
477 case DW_CFA_offset_extended_sf: /* DWARF3 */
478 signed_second = 1;
479 goto two_leb;
480 case DW_CFA_offset_extended:
481 goto two_leb;
483 case DW_CFA_undefined:
484 case DW_CFA_same_value:
485 goto one_leb;
487 case DW_CFA_val_offset:
488 goto two_leb;
489 case DW_CFA_val_offset_sf:
490 signed_second = 1;
491 goto two_leb;
492 case DW_CFA_def_cfa_sf:
493 signed_second = 1;
494 goto two_leb;
495 case DW_CFA_register:
496 case DW_CFA_def_cfa:
497 two_leb:
498 res = _dwarf_pro_encode_leb128_nm(val1, &nbytes1,
499 buff1, sizeof(buff1));
500 if (res != DW_DLV_OK) {
501 _dwarf_p_error(dbg, error, DW_DLE_STRING_ALLOC);
502 return ((Dwarf_P_Fde) DW_DLV_BADADDR);
504 if (!signed_second) {
505 res = _dwarf_pro_encode_leb128_nm(val2, &nbytes2,
506 buff2, sizeof(buff2));
507 } else {
508 Dwarf_Signed val2s = val2;
509 res = _dwarf_pro_encode_signed_leb128_nm(val2s, &nbytes2,
510 buff2, sizeof(buff2));
513 res = _dwarf_pro_encode_leb128_nm(val2, &nbytes2,
514 buff2, sizeof(buff2));
515 if (res != DW_DLV_OK) {
516 _dwarf_p_error(dbg, error, DW_DLE_STRING_ALLOC);
517 return ((Dwarf_P_Fde) DW_DLV_BADADDR);
520 ptr = (char *) _dwarf_p_get_alloc(dbg, nbytes1 + nbytes2);
521 if (ptr == NULL) {
522 _dwarf_p_error(dbg, error, DW_DLE_STRING_ALLOC);
523 return ((Dwarf_P_Fde) DW_DLV_BADADDR);
525 memcpy(ptr, buff1, nbytes1);
526 memcpy(ptr + nbytes1, buff2, nbytes2);
527 nbytes = nbytes1 + nbytes2;
528 break;
530 case DW_CFA_def_cfa_offset_sf: /* DWARF3 */
531 signed_first = 1;
532 goto one_leb;
533 case DW_CFA_def_cfa_register:
534 case DW_CFA_def_cfa_offset:
535 one_leb:
536 if(!signed_first) {
537 res = _dwarf_pro_encode_leb128_nm(val1, &nbytes,
538 buff1, sizeof(buff1));
539 } else {
540 Dwarf_Signed val1s = val1;
541 res = _dwarf_pro_encode_signed_leb128_nm(val1s, &nbytes,
542 buff1, sizeof(buff1));
544 if (res != DW_DLV_OK) {
545 _dwarf_p_error(dbg, error, DW_DLE_STRING_ALLOC);
546 return ((Dwarf_P_Fde) DW_DLV_BADADDR);
548 ptr = (char *) _dwarf_p_get_alloc(dbg, nbytes);
549 if (ptr == NULL) {
550 _dwarf_p_error(dbg, error, DW_DLE_STRING_ALLOC);
551 return ((Dwarf_P_Fde) DW_DLV_BADADDR);
553 memcpy(ptr, buff1, nbytes);
554 break;
555 case DW_CFA_def_cfa_expression: /* DWARF3 */
556 /* FIXME: argument is dwarf expr, not handled yet. */
557 case DW_CFA_expression: /* DWARF3 */
558 /* First arg: ULEB reg num. 2nd arg dwarf expr in form block.
559 FIXME: not handled yet. */
560 case DW_CFA_val_expression: /* DWARF3f */
561 /* First arg: ULEB reg num. 2nd arg dwarf expr in form block.
562 FIXME: not handled yet. */
563 default:
564 _dwarf_p_error(dbg, error, DW_DLE_DEBUGFRAME_ERROR);
565 return ((Dwarf_P_Fde) DW_DLV_BADADDR);
568 curinst->dfp_opcode = op;
569 curinst->dfp_args = ptr;
570 curinst->dfp_nbytes = nbytes;
571 curinst->dfp_next = NULL;
573 _dwarf_pro_add_to_fde(fde, curinst);
574 return fde;
578 /*------------------------------------------------------------------------
579 Instructions are added to an fde in the form of a linked
580 list. This function manages the linked list.
581 -------------------------------------------------------------------------*/
582 void
583 _dwarf_pro_add_to_fde(Dwarf_P_Fde fde, Dwarf_P_Frame_Pgm curinst)
585 if (fde->fde_last_inst) {
586 fde->fde_last_inst->dfp_next = curinst;
587 fde->fde_last_inst = curinst;
588 fde->fde_n_inst++;
589 fde->fde_n_bytes +=
590 (long) (curinst->dfp_nbytes + sizeof(Dwarf_Ubyte));
591 } else {
592 fde->fde_last_inst = curinst;
593 fde->fde_inst = curinst;
594 fde->fde_n_inst = 1;
595 fde->fde_n_bytes =
596 (long) (curinst->dfp_nbytes + sizeof(Dwarf_Ubyte));