import less(1)
[unleashed/tickless.git] / usr / src / lib / libdwarf / common / pro_expr.c
blobad40eb762a20941b2ecc8479952bcb04638a1e12
1 /*
3 Copyright (C) 2000,2004,2006 Silicon Graphics, Inc. All Rights Reserved.
4 Portions Copyright 2007-2010 Sun Microsystems, Inc. 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
39 #include "config.h"
40 #include "libdwarfdefs.h"
41 #include <stdio.h>
42 #include <string.h>
43 #include <sys/types.h>
44 #include "pro_incl.h"
45 #include "pro_expr.h"
48 This function creates a new expression
49 struct that can be used to build up a
50 location expression.
52 Dwarf_P_Expr
53 dwarf_new_expr(Dwarf_P_Debug dbg, Dwarf_Error * error)
55 Dwarf_P_Expr ret_expr;
57 if (dbg == NULL) {
58 _dwarf_p_error(NULL, error, DW_DLE_DBG_NULL);
59 return (NULL);
62 ret_expr = (Dwarf_P_Expr)
63 _dwarf_p_get_alloc(dbg, sizeof(struct Dwarf_P_Expr_s));
64 if (ret_expr == NULL) {
65 _dwarf_p_error(dbg, error, DW_DLE_ALLOC_FAIL);
66 return (NULL);
69 ret_expr->ex_dbg = dbg;
71 return (ret_expr);
75 Dwarf_Unsigned
76 dwarf_add_expr_gen(Dwarf_P_Expr expr,
77 Dwarf_Small opcode,
78 Dwarf_Unsigned val1,
79 Dwarf_Unsigned val2, Dwarf_Error * error)
81 char encode_buffer[2 * ENCODE_SPACE_NEEDED]; /* 2* since
82 used to
83 concatenate
84 2 leb's
85 below */
86 char encode_buffer2[ENCODE_SPACE_NEEDED];
87 int res;
88 Dwarf_P_Debug dbg = expr->ex_dbg;
90 /*
91 Give the buffer where the operands are first going to be
92 assembled the largest alignment. */
93 Dwarf_Unsigned operand_buffer[10];
95 /*
96 Size of the byte stream buffer that needs to be memcpy-ed. */
97 int operand_size;
99 /*
100 Points to the byte stream for the first operand, and finally to
101 the buffer that is memcp-ed into the Dwarf_P_Expr_s struct. */
102 Dwarf_Small *operand;
104 /* Size of the byte stream for second operand. */
105 int operand2_size;
107 /* Points to next byte to be written in Dwarf_P_Expr_s struct. */
108 Dwarf_Small *next_byte_ptr;
110 /* Offset past the last byte written into Dwarf_P_Expr_s. */
111 int next_byte_offset;
113 /* ***** BEGIN CODE ***** */
115 if (expr == NULL) {
116 _dwarf_p_error(NULL, error, DW_DLE_EXPR_NULL);
117 return (DW_DLV_NOCOUNT);
120 if (expr->ex_dbg == NULL) {
121 _dwarf_p_error(NULL, error, DW_DLE_DBG_NULL);
122 return (DW_DLV_NOCOUNT);
125 operand = NULL;
126 operand_size = 0;
128 switch (opcode) {
129 case DW_OP_reg0:
130 case DW_OP_reg1:
131 case DW_OP_reg2:
132 case DW_OP_reg3:
133 case DW_OP_reg4:
134 case DW_OP_reg5:
135 case DW_OP_reg6:
136 case DW_OP_reg7:
137 case DW_OP_reg8:
138 case DW_OP_reg9:
139 case DW_OP_reg10:
140 case DW_OP_reg11:
141 case DW_OP_reg12:
142 case DW_OP_reg13:
143 case DW_OP_reg14:
144 case DW_OP_reg15:
145 case DW_OP_reg16:
146 case DW_OP_reg17:
147 case DW_OP_reg18:
148 case DW_OP_reg19:
149 case DW_OP_reg20:
150 case DW_OP_reg21:
151 case DW_OP_reg22:
152 case DW_OP_reg23:
153 case DW_OP_reg24:
154 case DW_OP_reg25:
155 case DW_OP_reg26:
156 case DW_OP_reg27:
157 case DW_OP_reg28:
158 case DW_OP_reg29:
159 case DW_OP_reg30:
160 case DW_OP_reg31:
161 break;
163 case DW_OP_breg0:
164 case DW_OP_breg1:
165 case DW_OP_breg2:
166 case DW_OP_breg3:
167 case DW_OP_breg4:
168 case DW_OP_breg5:
169 case DW_OP_breg6:
170 case DW_OP_breg7:
171 case DW_OP_breg8:
172 case DW_OP_breg9:
173 case DW_OP_breg10:
174 case DW_OP_breg11:
175 case DW_OP_breg12:
176 case DW_OP_breg13:
177 case DW_OP_breg14:
178 case DW_OP_breg15:
179 case DW_OP_breg16:
180 case DW_OP_breg17:
181 case DW_OP_breg18:
182 case DW_OP_breg19:
183 case DW_OP_breg20:
184 case DW_OP_breg21:
185 case DW_OP_breg22:
186 case DW_OP_breg23:
187 case DW_OP_breg24:
188 case DW_OP_breg25:
189 case DW_OP_breg26:
190 case DW_OP_breg27:
191 case DW_OP_breg28:
192 case DW_OP_breg29:
193 case DW_OP_breg30:
194 case DW_OP_breg31:
195 res = _dwarf_pro_encode_signed_leb128_nm(val1,
196 &operand_size,
197 encode_buffer,
198 sizeof(encode_buffer));
199 if (res != DW_DLV_OK) {
200 _dwarf_p_error(expr->ex_dbg, error, DW_DLE_EXPR_LENGTH_BAD);
201 return (DW_DLV_NOCOUNT);
203 operand = (Dwarf_Small *) encode_buffer;
204 break;
206 case DW_OP_regx:
207 res = _dwarf_pro_encode_leb128_nm(val1, &operand_size,
208 encode_buffer,
209 sizeof(encode_buffer));
210 if (res != DW_DLV_OK) {
211 _dwarf_p_error(expr->ex_dbg, error, DW_DLE_EXPR_LENGTH_BAD);
212 return (DW_DLV_NOCOUNT);
214 operand = (Dwarf_Small *) encode_buffer;
215 break;
217 case DW_OP_lit0:
218 case DW_OP_lit1:
219 case DW_OP_lit2:
220 case DW_OP_lit3:
221 case DW_OP_lit4:
222 case DW_OP_lit5:
223 case DW_OP_lit6:
224 case DW_OP_lit7:
225 case DW_OP_lit8:
226 case DW_OP_lit9:
227 case DW_OP_lit10:
228 case DW_OP_lit11:
229 case DW_OP_lit12:
230 case DW_OP_lit13:
231 case DW_OP_lit14:
232 case DW_OP_lit15:
233 case DW_OP_lit16:
234 case DW_OP_lit17:
235 case DW_OP_lit18:
236 case DW_OP_lit19:
237 case DW_OP_lit20:
238 case DW_OP_lit21:
239 case DW_OP_lit22:
240 case DW_OP_lit23:
241 case DW_OP_lit24:
242 case DW_OP_lit25:
243 case DW_OP_lit26:
244 case DW_OP_lit27:
245 case DW_OP_lit28:
246 case DW_OP_lit29:
247 case DW_OP_lit30:
248 case DW_OP_lit31:
249 break;
251 case DW_OP_addr:
252 _dwarf_p_error(expr->ex_dbg, error, DW_DLE_BAD_EXPR_OPCODE);
253 return (DW_DLV_NOCOUNT);
255 case DW_OP_const1u:
256 case DW_OP_const1s:
257 operand = (Dwarf_Small *) & operand_buffer[0];
258 WRITE_UNALIGNED(dbg, operand, &val1, sizeof(val1), 1);
259 operand_size = 1;
260 break;
262 case DW_OP_const2u:
263 case DW_OP_const2s:
264 operand = (Dwarf_Small *) & operand_buffer[0];
265 WRITE_UNALIGNED(dbg, operand, &val1, sizeof(val1), 2);
266 operand_size = 2;
267 break;
269 case DW_OP_const4u:
270 case DW_OP_const4s:
271 operand = (Dwarf_Small *) & operand_buffer[0];
272 WRITE_UNALIGNED(dbg, operand, &val1, sizeof(val1), 4);
273 operand_size = 4;
274 break;
276 case DW_OP_const8u:
277 case DW_OP_const8s:
278 operand = (Dwarf_Small *) & operand_buffer[0];
279 WRITE_UNALIGNED(dbg, operand, &val1, sizeof(val1), 8);
280 operand_size = 8;
281 break;
283 case DW_OP_constu:
284 res = _dwarf_pro_encode_leb128_nm(val1,
285 &operand_size,
286 encode_buffer,
287 sizeof(encode_buffer));
288 if (res != DW_DLV_OK) {
289 _dwarf_p_error(expr->ex_dbg, error, DW_DLE_EXPR_LENGTH_BAD);
290 return (DW_DLV_NOCOUNT);
292 operand = (Dwarf_Small *) encode_buffer;
293 break;
295 case DW_OP_consts:
296 res = _dwarf_pro_encode_signed_leb128_nm(val1,
297 &operand_size,
298 encode_buffer,
299 sizeof(encode_buffer));
300 if (res != DW_DLV_OK) {
301 _dwarf_p_error(expr->ex_dbg, error, DW_DLE_EXPR_LENGTH_BAD);
302 return (DW_DLV_NOCOUNT);
304 operand = (Dwarf_Small *) encode_buffer;
305 break;
307 case DW_OP_fbreg:
308 res = _dwarf_pro_encode_signed_leb128_nm(val1,
309 &operand_size,
310 encode_buffer,
311 sizeof(encode_buffer));
312 if (res != DW_DLV_OK) {
313 _dwarf_p_error(expr->ex_dbg, error, DW_DLE_EXPR_LENGTH_BAD);
314 return (DW_DLV_NOCOUNT);
316 operand = (Dwarf_Small *) encode_buffer;
317 break;
319 case DW_OP_bregx:
320 res = _dwarf_pro_encode_leb128_nm(val1, &operand_size,
321 encode_buffer,
322 sizeof(encode_buffer));
323 if (res != DW_DLV_OK) {
324 _dwarf_p_error(expr->ex_dbg, error, DW_DLE_EXPR_LENGTH_BAD);
325 return (DW_DLV_NOCOUNT);
327 operand = (Dwarf_Small *) encode_buffer;
328 /* put this one directly into 'operand' at tail of prev value */
329 res = _dwarf_pro_encode_signed_leb128_nm(val2, &operand2_size,
330 ((char *) operand) +
331 operand_size,
332 sizeof
333 (encode_buffer2));
334 if (res != DW_DLV_OK) {
335 _dwarf_p_error(expr->ex_dbg, error, DW_DLE_EXPR_LENGTH_BAD);
336 return (DW_DLV_NOCOUNT);
338 operand_size += operand2_size;
340 case DW_OP_dup:
341 case DW_OP_drop:
342 break;
344 case DW_OP_pick:
345 operand = (Dwarf_Small *) & operand_buffer[0];
346 WRITE_UNALIGNED(dbg, operand, (const void *) &val1,
347 sizeof(val1), 1);
348 operand_size = 1;
349 break;
351 case DW_OP_over:
352 case DW_OP_swap:
353 case DW_OP_rot:
354 case DW_OP_deref:
355 case DW_OP_xderef:
356 break;
358 case DW_OP_deref_size:
359 case DW_OP_xderef_size:
360 operand = (Dwarf_Small *) & operand_buffer[0];
361 WRITE_UNALIGNED(dbg, operand, (const void *) &val1,
362 sizeof(val1), 1);
363 operand_size = 1;
364 break;
366 case DW_OP_abs:
367 case DW_OP_and:
368 case DW_OP_div:
369 case DW_OP_minus:
370 case DW_OP_mod:
371 case DW_OP_mul:
372 case DW_OP_neg:
373 case DW_OP_not:
374 case DW_OP_or:
375 case DW_OP_plus:
376 break;
378 case DW_OP_plus_uconst:
379 res = _dwarf_pro_encode_leb128_nm(val1, &operand_size,
380 encode_buffer,
381 sizeof(encode_buffer));
382 if (res != DW_DLV_OK) {
383 _dwarf_p_error(expr->ex_dbg, error, DW_DLE_EXPR_LENGTH_BAD);
384 return (DW_DLV_NOCOUNT);
386 operand = (Dwarf_Small *) encode_buffer;
387 break;
389 case DW_OP_shl:
390 case DW_OP_shr:
391 case DW_OP_shra:
392 case DW_OP_xor:
393 break;
395 case DW_OP_le:
396 case DW_OP_ge:
397 case DW_OP_eq:
398 case DW_OP_lt:
399 case DW_OP_gt:
400 case DW_OP_ne:
401 break;
403 case DW_OP_skip:
404 case DW_OP_bra:
405 /* FIX: unhandled! OP_bra, OP_skip! */
406 _dwarf_p_error(expr->ex_dbg, error, DW_DLE_BAD_EXPR_OPCODE);
407 return (DW_DLV_NOCOUNT);
409 case DW_OP_piece:
410 res = _dwarf_pro_encode_leb128_nm(val1, &operand_size,
411 encode_buffer,
412 sizeof(encode_buffer));
413 if (res != DW_DLV_OK) {
414 _dwarf_p_error(expr->ex_dbg, error, DW_DLE_EXPR_LENGTH_BAD);
415 return (DW_DLV_NOCOUNT);
417 operand = (Dwarf_Small *) encode_buffer;
418 break;
420 case DW_OP_nop:
421 break;
422 case DW_OP_push_object_address: /* DWARF3 */
423 break;
424 case DW_OP_call2: /* DWARF3 */
425 operand = (Dwarf_Small *) & operand_buffer[0];
426 WRITE_UNALIGNED(dbg, operand, &val1, sizeof(val1), 2);
427 operand_size = 2;
428 break;
430 case DW_OP_call4: /* DWARF3 */
431 operand = (Dwarf_Small *) & operand_buffer[0];
432 WRITE_UNALIGNED(dbg, operand, &val1, sizeof(val1), 4);
433 operand_size = 4;
434 break;
436 case DW_OP_call_ref: /* DWARF3 */
437 operand = (Dwarf_Small *) & operand_buffer[0];
438 WRITE_UNALIGNED(dbg, operand, &val1, sizeof(val1),
439 dbg->de_offset_size);
440 operand_size = dbg->de_offset_size;
441 break;
442 case DW_OP_form_tls_address: /* DWARF3f */
443 break;
444 case DW_OP_call_frame_cfa: /* DWARF3f */
445 break;
446 case DW_OP_bit_piece: /* DWARF3f */
447 res = _dwarf_pro_encode_leb128_nm(val1, &operand_size,
448 encode_buffer,
449 sizeof(encode_buffer));
450 if (res != DW_DLV_OK) {
451 _dwarf_p_error(expr->ex_dbg, error, DW_DLE_EXPR_LENGTH_BAD);
452 return (DW_DLV_NOCOUNT);
454 operand = (Dwarf_Small *) encode_buffer;
455 /* put this one directly into 'operand' at tail of prev value */
456 res = _dwarf_pro_encode_leb128_nm(val2, &operand2_size,
457 ((char *) operand) +
458 operand_size,
459 sizeof(encode_buffer2));
460 if (res != DW_DLV_OK) {
461 _dwarf_p_error(expr->ex_dbg, error, DW_DLE_EXPR_LENGTH_BAD);
462 return (DW_DLV_NOCOUNT);
464 operand_size += operand2_size;
465 break;
467 default:
468 _dwarf_p_error(expr->ex_dbg, error, DW_DLE_BAD_EXPR_OPCODE);
469 return (DW_DLV_NOCOUNT);
472 next_byte_offset = expr->ex_next_byte_offset + operand_size + 1;
474 if (next_byte_offset > MAXIMUM_LOC_EXPR_LENGTH) {
475 _dwarf_p_error(expr->ex_dbg, error, DW_DLE_EXPR_LENGTH_BAD);
476 return (DW_DLV_NOCOUNT);
479 next_byte_ptr =
480 &(expr->ex_byte_stream[0]) + expr->ex_next_byte_offset;
482 *next_byte_ptr = opcode;
483 next_byte_ptr++;
484 memcpy(next_byte_ptr, operand, operand_size);
486 expr->ex_next_byte_offset = next_byte_offset;
487 return (next_byte_offset);
490 Dwarf_Unsigned
491 dwarf_add_expr_addr_b(Dwarf_P_Expr expr,
492 Dwarf_Unsigned addr,
493 Dwarf_Unsigned sym_index, Dwarf_Error * error)
495 Dwarf_P_Debug dbg;
496 Dwarf_Small *next_byte_ptr;
497 Dwarf_Unsigned next_byte_offset;
498 int upointer_size;
500 if (expr == NULL) {
501 _dwarf_p_error(NULL, error, DW_DLE_EXPR_NULL);
502 return (DW_DLV_NOCOUNT);
505 dbg = expr->ex_dbg;
506 if (dbg == NULL) {
507 _dwarf_p_error(NULL, error, DW_DLE_DBG_NULL);
508 return (DW_DLV_NOCOUNT);
511 upointer_size = dbg->de_pointer_size;
512 next_byte_offset = expr->ex_next_byte_offset + upointer_size + 1;
513 if (next_byte_offset > MAXIMUM_LOC_EXPR_LENGTH) {
514 _dwarf_p_error(dbg, error, DW_DLE_EXPR_LENGTH_BAD);
515 return (DW_DLV_NOCOUNT);
518 next_byte_ptr =
519 &(expr->ex_byte_stream[0]) + expr->ex_next_byte_offset;
521 *next_byte_ptr = DW_OP_addr;
522 next_byte_ptr++;
523 WRITE_UNALIGNED(dbg, next_byte_ptr, (const void *) &addr,
524 sizeof(addr), upointer_size);
526 if (expr->ex_reloc_offset != 0) {
527 _dwarf_p_error(dbg, error, DW_DLE_MULTIPLE_RELOC_IN_EXPR);
528 return (DW_DLV_NOCOUNT);
531 expr->ex_reloc_sym_index = sym_index;
532 expr->ex_reloc_offset = expr->ex_next_byte_offset + 1;
534 expr->ex_next_byte_offset = next_byte_offset;
535 return (next_byte_offset);
538 Dwarf_Unsigned
539 dwarf_add_expr_addr(Dwarf_P_Expr expr,
540 Dwarf_Unsigned addr,
541 Dwarf_Signed sym_index, Dwarf_Error * error)
543 return
544 dwarf_add_expr_addr_b(expr, addr, (Dwarf_Unsigned) sym_index,
545 error);
549 Dwarf_Unsigned
550 dwarf_expr_current_offset(Dwarf_P_Expr expr, Dwarf_Error * error)
552 if (expr == NULL) {
553 _dwarf_p_error(NULL, error, DW_DLE_EXPR_NULL);
554 return (DW_DLV_NOCOUNT);
557 if (expr->ex_dbg == NULL) {
558 _dwarf_p_error(NULL, error, DW_DLE_DBG_NULL);
559 return (DW_DLV_NOCOUNT);
562 return (expr->ex_next_byte_offset);
565 void
566 dwarf_expr_reset(Dwarf_P_Expr expr, Dwarf_Error * error)
568 if (expr == NULL) {
569 _dwarf_p_error(NULL, error, DW_DLE_EXPR_NULL);
570 return;
572 expr->ex_next_byte_offset=0;
576 Dwarf_Addr
577 dwarf_expr_into_block(Dwarf_P_Expr expr,
578 Dwarf_Unsigned * length, Dwarf_Error * error)
580 if (expr == NULL) {
581 _dwarf_p_error(NULL, error, DW_DLE_EXPR_NULL);
582 return (DW_DLV_BADADDR);
585 if (expr->ex_dbg == NULL) {
586 _dwarf_p_error(NULL, error, DW_DLE_DBG_NULL);
587 return (DW_DLV_BADADDR);
590 if (length != NULL)
591 *length = expr->ex_next_byte_offset;
592 /* The following cast from pointer to integer is ok as long as
593 Dwarf_Addr is at least as large as a pointer. Which is a
594 requirement of libdwarf so must be satisfied (some compilers
595 emit a warning about the following line). */
596 return ((Dwarf_Addr)(uintptr_t) &(expr->ex_byte_stream[0]));