mirror of
https://sourceware.org/git/binutils-gdb.git
synced 2025-01-13 03:23:42 +08:00
3019eac3a2
* symfile.c (default_symfile_relocate): Use sectp->owner instead of objfile->obfd. * symfile.h (dwarf2_debug_sections): New member addr. * dwarf2expr.c (execute_stack_op): New case DW_OP_GNU_addr_index. (ctx_no_get_addr_index): New function. * dwarf2expr.h (dwarf_expr_context_funcs): New member get_addr_index. (ctx_no_get_addr_index): Declare. * dwarf2-frame.c (dwarf2_frame_ctx_funcs): Update. * dwarf2loc.c (dwarf_expr_get_addr_index): New function. (dwarf_expr_ctx_funcs): Update. (needs_get_addr_index): New function. (needs_frame_ctx_funcs): Update. * dwarf2loc.h (dwarf2_read_addr_index): Declare. * dwarf2read.c: #include "gdbcore.h". (dwarf2_per_objfile): New members addr, dwo_files. (dwarf2_elf_names): Add entry for addr. (struct dwo_section_names): New type. (dwo_section_names): New static global. (dwarf2_cu): New members dwo_unit, addr_base, have_addr_base. (dwarf2_per_cu_data): New member is_debug_types, all boolean uses of old debug_types_section member updated to use this. Rename member debug_types_section to info_or_types_section, all uses updated. (signatured_type): Rename member type_offset to type_offset_in_tu, all uses updated. New member type_offset_in_section. (struct dwo_sections): New type. (struct dwo_unit): New type. (struct dwo_file): New type. (die_reader_specs): New member dwo_file. (dwarf2_locate_sections): Watch for .debug_addr. (zlib_decompress_section): Use sectp->owner instead of objfile->obfd. (dwarf2_read_section): Get bfd of section from bfd's asection, instead of objfile. (create_cus_from_index): Initialize the_cu->info_or_types_section. (create_signatured_type_table_from_index): Initialize sig_type->info_or_types_section. (dw2_get_file_names): Statement lists for type units with DWO files live in the DWO file. (create_debug_types_hash_table): New function. (create_all_type_units): Rewrite. (init_cu_die_reader): New arg dwo_file, all callers updated. (init_and_read_dies_worker): Get section from this_cu->info_or_types_section. Set sig_type->type_offset_in_section. Watch for DW_AT_GNU_dwo_name and if present lookup the file and continue reading the CU/TU from there. (init_cutu_and_read_dies_no_follow): New arg dwo_file, all callers updated. Get section from this_cu->info_or_types_section. (create_all_comp_units): Initialize this_cu->info_or_types_section. (skip_one_die): New cases DW_FORM_GNU_addr_index, DW_FORM_GNU_str_index. (hash_dwo_file, eq_dwo_file): New functions. (allocate_dwo_file_hash_table): New function. (hash_dwo_unit, eq_dwo_unit): New functions. (allocate_dwo_unit_table): New function. (dwarf2_locate_dwo_sections): New function. (struct create_dwo_info_table_data): New type. (create_debug_info_hash_table_reader): New function. (create_debug_info_hash_table): New function. (try_open_dwo_file, open_dwo_file, init_dwo_file): New function. (lookup_dwo_file): New function. (lookup_dwo_comp_unit, lookup_dwo_type_unit): New functions. (free_dwo_file, free_dwo_file_cleanup): New functions. (free_dwo_file_from_slot, free_dwo_files): New functions. (dwarf2_get_pc_bounds): Handle DW_FORM_GNU_addr_index. (dwarf2_record_block_ranges): Ditto. (read_partial_die): Ditto. (process_enumeration_scope): Update to use type_offset_in_section. (read_full_die_1): New function. (read_full_die): Rewrite. (read_attribute_value): New cases DW_FORM_GNU_addr_index, DW_FORM_GNU_str_index. (read_addr_index_1, read_addr_index): New functions. (read_addr_index_from_leb128): New function. (struct dwarf2_read_addr_index_data): New type. (dwarf2_read_addr_index_reader): New function. (dwarf2_read_addr_index): New function. (read_str_index): New function. (leb128_size): New function. (dwarf_decode_line_header): Delete arg abfd, all callers updated. If processing a type unit from a DWO file, get the line section from the DWO file. (var_decode_location): Watch for DW_OP_GNU_addr_index. (dwarf2_const_value_attr): New cases DW_FORM_GNU_addr_index, DW_FORM_GNU_str_index. (lookup_die_type): Check whether section offset of type's die is known before looking it up. Remove assert. Condition can legimately happen for inter-cu type references. (dwarf_attr_name): Handle Fission attributes. (dwarf_form_name): Handle Fission forms. (dump_die_shallow): New cases DW_FORM_GNU_addr_index, DW_FORM_GNU_str_index. (follow_die_sig): Update to use type_offset_in_section. (decode_locdesc): New case DW_OP_GNU_addr_index. (skip_form_bytes): New cases DW_FORM_GNU_addr_index, DW_FORM_GNU_str_index. (cu_debug_loc_section): New function. (fill_in_loclist_baton, dwarf2_symbol_mark_computed): Call it. (dwarf2_per_objfile_free): Unmap .debug_addr section. Free DWO files if present. * xcoffread.c (dwarf2_xcoff_names): Add .debug_addr. testsuite/ * gdb.dwarf2/dw2-intercu.S (.Ltype_int2_in_cu2): Renamed from .Ltype_int_in_cu2. Use name "int2" instead of "int". All uses updated. * gdb.dwarf2/dw2-intercu.exp: Add "ptype int2" ahead of "ptype func_cu1" to expand cu2 before cu1.
1569 lines
42 KiB
C
1569 lines
42 KiB
C
/* DWARF 2 Expression Evaluator.
|
||
|
||
Copyright (C) 2001-2003, 2005, 2007-2012 Free Software Foundation,
|
||
Inc.
|
||
|
||
Contributed by Daniel Berlin (dan@dberlin.org)
|
||
|
||
This file is part of GDB.
|
||
|
||
This program is free software; you can redistribute it and/or modify
|
||
it under the terms of the GNU General Public License as published by
|
||
the Free Software Foundation; either version 3 of the License, or
|
||
(at your option) any later version.
|
||
|
||
This program is distributed in the hope that it will be useful,
|
||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
GNU General Public License for more details.
|
||
|
||
You should have received a copy of the GNU General Public License
|
||
along with this program. If not, see <http://www.gnu.org/licenses/>. */
|
||
|
||
#include "defs.h"
|
||
#include "symtab.h"
|
||
#include "gdbtypes.h"
|
||
#include "value.h"
|
||
#include "gdbcore.h"
|
||
#include "dwarf2.h"
|
||
#include "dwarf2expr.h"
|
||
#include "gdb_assert.h"
|
||
|
||
/* Local prototypes. */
|
||
|
||
static void execute_stack_op (struct dwarf_expr_context *,
|
||
const gdb_byte *, const gdb_byte *);
|
||
|
||
/* Cookie for gdbarch data. */
|
||
|
||
static struct gdbarch_data *dwarf_arch_cookie;
|
||
|
||
/* This holds gdbarch-specific types used by the DWARF expression
|
||
evaluator. See comments in execute_stack_op. */
|
||
|
||
struct dwarf_gdbarch_types
|
||
{
|
||
struct type *dw_types[3];
|
||
};
|
||
|
||
/* Allocate and fill in dwarf_gdbarch_types for an arch. */
|
||
|
||
static void *
|
||
dwarf_gdbarch_types_init (struct gdbarch *gdbarch)
|
||
{
|
||
struct dwarf_gdbarch_types *types
|
||
= GDBARCH_OBSTACK_ZALLOC (gdbarch, struct dwarf_gdbarch_types);
|
||
|
||
/* The types themselves are lazily initialized. */
|
||
|
||
return types;
|
||
}
|
||
|
||
/* Return the type used for DWARF operations where the type is
|
||
unspecified in the DWARF spec. Only certain sizes are
|
||
supported. */
|
||
|
||
static struct type *
|
||
dwarf_expr_address_type (struct dwarf_expr_context *ctx)
|
||
{
|
||
struct dwarf_gdbarch_types *types = gdbarch_data (ctx->gdbarch,
|
||
dwarf_arch_cookie);
|
||
int ndx;
|
||
|
||
if (ctx->addr_size == 2)
|
||
ndx = 0;
|
||
else if (ctx->addr_size == 4)
|
||
ndx = 1;
|
||
else if (ctx->addr_size == 8)
|
||
ndx = 2;
|
||
else
|
||
error (_("Unsupported address size in DWARF expressions: %d bits"),
|
||
8 * ctx->addr_size);
|
||
|
||
if (types->dw_types[ndx] == NULL)
|
||
types->dw_types[ndx]
|
||
= arch_integer_type (ctx->gdbarch,
|
||
8 * ctx->addr_size,
|
||
0, "<signed DWARF address type>");
|
||
|
||
return types->dw_types[ndx];
|
||
}
|
||
|
||
/* Create a new context for the expression evaluator. */
|
||
|
||
struct dwarf_expr_context *
|
||
new_dwarf_expr_context (void)
|
||
{
|
||
struct dwarf_expr_context *retval;
|
||
|
||
retval = xcalloc (1, sizeof (struct dwarf_expr_context));
|
||
retval->stack_len = 0;
|
||
retval->stack_allocated = 10;
|
||
retval->stack = xmalloc (retval->stack_allocated
|
||
* sizeof (struct dwarf_stack_value));
|
||
retval->num_pieces = 0;
|
||
retval->pieces = 0;
|
||
retval->max_recursion_depth = 0x100;
|
||
return retval;
|
||
}
|
||
|
||
/* Release the memory allocated to CTX. */
|
||
|
||
void
|
||
free_dwarf_expr_context (struct dwarf_expr_context *ctx)
|
||
{
|
||
xfree (ctx->stack);
|
||
xfree (ctx->pieces);
|
||
xfree (ctx);
|
||
}
|
||
|
||
/* Helper for make_cleanup_free_dwarf_expr_context. */
|
||
|
||
static void
|
||
free_dwarf_expr_context_cleanup (void *arg)
|
||
{
|
||
free_dwarf_expr_context (arg);
|
||
}
|
||
|
||
/* Return a cleanup that calls free_dwarf_expr_context. */
|
||
|
||
struct cleanup *
|
||
make_cleanup_free_dwarf_expr_context (struct dwarf_expr_context *ctx)
|
||
{
|
||
return make_cleanup (free_dwarf_expr_context_cleanup, ctx);
|
||
}
|
||
|
||
/* Expand the memory allocated to CTX's stack to contain at least
|
||
NEED more elements than are currently used. */
|
||
|
||
static void
|
||
dwarf_expr_grow_stack (struct dwarf_expr_context *ctx, size_t need)
|
||
{
|
||
if (ctx->stack_len + need > ctx->stack_allocated)
|
||
{
|
||
size_t newlen = ctx->stack_len + need + 10;
|
||
|
||
ctx->stack = xrealloc (ctx->stack,
|
||
newlen * sizeof (struct dwarf_stack_value));
|
||
ctx->stack_allocated = newlen;
|
||
}
|
||
}
|
||
|
||
/* Push VALUE onto CTX's stack. */
|
||
|
||
static void
|
||
dwarf_expr_push (struct dwarf_expr_context *ctx, struct value *value,
|
||
int in_stack_memory)
|
||
{
|
||
struct dwarf_stack_value *v;
|
||
|
||
dwarf_expr_grow_stack (ctx, 1);
|
||
v = &ctx->stack[ctx->stack_len++];
|
||
v->value = value;
|
||
v->in_stack_memory = in_stack_memory;
|
||
}
|
||
|
||
/* Push VALUE onto CTX's stack. */
|
||
|
||
void
|
||
dwarf_expr_push_address (struct dwarf_expr_context *ctx, CORE_ADDR value,
|
||
int in_stack_memory)
|
||
{
|
||
dwarf_expr_push (ctx,
|
||
value_from_ulongest (dwarf_expr_address_type (ctx), value),
|
||
in_stack_memory);
|
||
}
|
||
|
||
/* Pop the top item off of CTX's stack. */
|
||
|
||
static void
|
||
dwarf_expr_pop (struct dwarf_expr_context *ctx)
|
||
{
|
||
if (ctx->stack_len <= 0)
|
||
error (_("dwarf expression stack underflow"));
|
||
ctx->stack_len--;
|
||
}
|
||
|
||
/* Retrieve the N'th item on CTX's stack. */
|
||
|
||
struct value *
|
||
dwarf_expr_fetch (struct dwarf_expr_context *ctx, int n)
|
||
{
|
||
if (ctx->stack_len <= n)
|
||
error (_("Asked for position %d of stack, "
|
||
"stack only has %d elements on it."),
|
||
n, ctx->stack_len);
|
||
return ctx->stack[ctx->stack_len - (1 + n)].value;
|
||
}
|
||
|
||
/* Require that TYPE be an integral type; throw an exception if not. */
|
||
|
||
static void
|
||
dwarf_require_integral (struct type *type)
|
||
{
|
||
if (TYPE_CODE (type) != TYPE_CODE_INT
|
||
&& TYPE_CODE (type) != TYPE_CODE_CHAR
|
||
&& TYPE_CODE (type) != TYPE_CODE_BOOL)
|
||
error (_("integral type expected in DWARF expression"));
|
||
}
|
||
|
||
/* Return the unsigned form of TYPE. TYPE is necessarily an integral
|
||
type. */
|
||
|
||
static struct type *
|
||
get_unsigned_type (struct gdbarch *gdbarch, struct type *type)
|
||
{
|
||
switch (TYPE_LENGTH (type))
|
||
{
|
||
case 1:
|
||
return builtin_type (gdbarch)->builtin_uint8;
|
||
case 2:
|
||
return builtin_type (gdbarch)->builtin_uint16;
|
||
case 4:
|
||
return builtin_type (gdbarch)->builtin_uint32;
|
||
case 8:
|
||
return builtin_type (gdbarch)->builtin_uint64;
|
||
default:
|
||
error (_("no unsigned variant found for type, while evaluating "
|
||
"DWARF expression"));
|
||
}
|
||
}
|
||
|
||
/* Return the signed form of TYPE. TYPE is necessarily an integral
|
||
type. */
|
||
|
||
static struct type *
|
||
get_signed_type (struct gdbarch *gdbarch, struct type *type)
|
||
{
|
||
switch (TYPE_LENGTH (type))
|
||
{
|
||
case 1:
|
||
return builtin_type (gdbarch)->builtin_int8;
|
||
case 2:
|
||
return builtin_type (gdbarch)->builtin_int16;
|
||
case 4:
|
||
return builtin_type (gdbarch)->builtin_int32;
|
||
case 8:
|
||
return builtin_type (gdbarch)->builtin_int64;
|
||
default:
|
||
error (_("no signed variant found for type, while evaluating "
|
||
"DWARF expression"));
|
||
}
|
||
}
|
||
|
||
/* Retrieve the N'th item on CTX's stack, converted to an address. */
|
||
|
||
CORE_ADDR
|
||
dwarf_expr_fetch_address (struct dwarf_expr_context *ctx, int n)
|
||
{
|
||
struct value *result_val = dwarf_expr_fetch (ctx, n);
|
||
enum bfd_endian byte_order = gdbarch_byte_order (ctx->gdbarch);
|
||
ULONGEST result;
|
||
|
||
dwarf_require_integral (value_type (result_val));
|
||
result = extract_unsigned_integer (value_contents (result_val),
|
||
TYPE_LENGTH (value_type (result_val)),
|
||
byte_order);
|
||
|
||
/* For most architectures, calling extract_unsigned_integer() alone
|
||
is sufficient for extracting an address. However, some
|
||
architectures (e.g. MIPS) use signed addresses and using
|
||
extract_unsigned_integer() will not produce a correct
|
||
result. Make sure we invoke gdbarch_integer_to_address()
|
||
for those architectures which require it. */
|
||
if (gdbarch_integer_to_address_p (ctx->gdbarch))
|
||
{
|
||
gdb_byte *buf = alloca (ctx->addr_size);
|
||
struct type *int_type = get_unsigned_type (ctx->gdbarch,
|
||
value_type (result_val));
|
||
|
||
store_unsigned_integer (buf, ctx->addr_size, byte_order, result);
|
||
return gdbarch_integer_to_address (ctx->gdbarch, int_type, buf);
|
||
}
|
||
|
||
return (CORE_ADDR) result;
|
||
}
|
||
|
||
/* Retrieve the in_stack_memory flag of the N'th item on CTX's stack. */
|
||
|
||
int
|
||
dwarf_expr_fetch_in_stack_memory (struct dwarf_expr_context *ctx, int n)
|
||
{
|
||
if (ctx->stack_len <= n)
|
||
error (_("Asked for position %d of stack, "
|
||
"stack only has %d elements on it."),
|
||
n, ctx->stack_len);
|
||
return ctx->stack[ctx->stack_len - (1 + n)].in_stack_memory;
|
||
}
|
||
|
||
/* Return true if the expression stack is empty. */
|
||
|
||
static int
|
||
dwarf_expr_stack_empty_p (struct dwarf_expr_context *ctx)
|
||
{
|
||
return ctx->stack_len == 0;
|
||
}
|
||
|
||
/* Add a new piece to CTX's piece list. */
|
||
static void
|
||
add_piece (struct dwarf_expr_context *ctx, ULONGEST size, ULONGEST offset)
|
||
{
|
||
struct dwarf_expr_piece *p;
|
||
|
||
ctx->num_pieces++;
|
||
|
||
ctx->pieces = xrealloc (ctx->pieces,
|
||
(ctx->num_pieces
|
||
* sizeof (struct dwarf_expr_piece)));
|
||
|
||
p = &ctx->pieces[ctx->num_pieces - 1];
|
||
p->location = ctx->location;
|
||
p->size = size;
|
||
p->offset = offset;
|
||
|
||
if (p->location == DWARF_VALUE_LITERAL)
|
||
{
|
||
p->v.literal.data = ctx->data;
|
||
p->v.literal.length = ctx->len;
|
||
}
|
||
else if (dwarf_expr_stack_empty_p (ctx))
|
||
{
|
||
p->location = DWARF_VALUE_OPTIMIZED_OUT;
|
||
/* Also reset the context's location, for our callers. This is
|
||
a somewhat strange approach, but this lets us avoid setting
|
||
the location to DWARF_VALUE_MEMORY in all the individual
|
||
cases in the evaluator. */
|
||
ctx->location = DWARF_VALUE_OPTIMIZED_OUT;
|
||
}
|
||
else if (p->location == DWARF_VALUE_MEMORY)
|
||
{
|
||
p->v.mem.addr = dwarf_expr_fetch_address (ctx, 0);
|
||
p->v.mem.in_stack_memory = dwarf_expr_fetch_in_stack_memory (ctx, 0);
|
||
}
|
||
else if (p->location == DWARF_VALUE_IMPLICIT_POINTER)
|
||
{
|
||
p->v.ptr.die.cu_off = ctx->len;
|
||
p->v.ptr.offset = value_as_long (dwarf_expr_fetch (ctx, 0));
|
||
}
|
||
else if (p->location == DWARF_VALUE_REGISTER)
|
||
p->v.regno = value_as_long (dwarf_expr_fetch (ctx, 0));
|
||
else
|
||
{
|
||
p->v.value = dwarf_expr_fetch (ctx, 0);
|
||
}
|
||
}
|
||
|
||
/* Evaluate the expression at ADDR (LEN bytes long) using the context
|
||
CTX. */
|
||
|
||
void
|
||
dwarf_expr_eval (struct dwarf_expr_context *ctx, const gdb_byte *addr,
|
||
size_t len)
|
||
{
|
||
int old_recursion_depth = ctx->recursion_depth;
|
||
|
||
execute_stack_op (ctx, addr, addr + len);
|
||
|
||
/* CTX RECURSION_DEPTH becomes invalid if an exception was thrown here. */
|
||
|
||
gdb_assert (ctx->recursion_depth == old_recursion_depth);
|
||
}
|
||
|
||
/* Decode the unsigned LEB128 constant at BUF into the variable pointed to
|
||
by R, and return the new value of BUF. Verify that it doesn't extend
|
||
past BUF_END. R can be NULL, the constant is then only skipped. */
|
||
|
||
const gdb_byte *
|
||
read_uleb128 (const gdb_byte *buf, const gdb_byte *buf_end, ULONGEST * r)
|
||
{
|
||
unsigned shift = 0;
|
||
ULONGEST result = 0;
|
||
gdb_byte byte;
|
||
|
||
while (1)
|
||
{
|
||
if (buf >= buf_end)
|
||
error (_("read_uleb128: Corrupted DWARF expression."));
|
||
|
||
byte = *buf++;
|
||
result |= ((ULONGEST) (byte & 0x7f)) << shift;
|
||
if ((byte & 0x80) == 0)
|
||
break;
|
||
shift += 7;
|
||
}
|
||
if (r)
|
||
*r = result;
|
||
return buf;
|
||
}
|
||
|
||
/* Decode the signed LEB128 constant at BUF into the variable pointed to
|
||
by R, and return the new value of BUF. Verify that it doesn't extend
|
||
past BUF_END. R can be NULL, the constant is then only skipped. */
|
||
|
||
const gdb_byte *
|
||
read_sleb128 (const gdb_byte *buf, const gdb_byte *buf_end, LONGEST * r)
|
||
{
|
||
unsigned shift = 0;
|
||
LONGEST result = 0;
|
||
gdb_byte byte;
|
||
|
||
while (1)
|
||
{
|
||
if (buf >= buf_end)
|
||
error (_("read_sleb128: Corrupted DWARF expression."));
|
||
|
||
byte = *buf++;
|
||
result |= ((ULONGEST) (byte & 0x7f)) << shift;
|
||
shift += 7;
|
||
if ((byte & 0x80) == 0)
|
||
break;
|
||
}
|
||
if (shift < (sizeof (*r) * 8) && (byte & 0x40) != 0)
|
||
result |= -(((LONGEST) 1) << shift);
|
||
|
||
if (r)
|
||
*r = result;
|
||
return buf;
|
||
}
|
||
|
||
|
||
/* Check that the current operator is either at the end of an
|
||
expression, or that it is followed by a composition operator. */
|
||
|
||
void
|
||
dwarf_expr_require_composition (const gdb_byte *op_ptr, const gdb_byte *op_end,
|
||
const char *op_name)
|
||
{
|
||
/* It seems like DW_OP_GNU_uninit should be handled here. However,
|
||
it doesn't seem to make sense for DW_OP_*_value, and it was not
|
||
checked at the other place that this function is called. */
|
||
if (op_ptr != op_end && *op_ptr != DW_OP_piece && *op_ptr != DW_OP_bit_piece)
|
||
error (_("DWARF-2 expression error: `%s' operations must be "
|
||
"used either alone or in conjunction with DW_OP_piece "
|
||
"or DW_OP_bit_piece."),
|
||
op_name);
|
||
}
|
||
|
||
/* Return true iff the types T1 and T2 are "the same". This only does
|
||
checks that might reasonably be needed to compare DWARF base
|
||
types. */
|
||
|
||
static int
|
||
base_types_equal_p (struct type *t1, struct type *t2)
|
||
{
|
||
if (TYPE_CODE (t1) != TYPE_CODE (t2))
|
||
return 0;
|
||
if (TYPE_UNSIGNED (t1) != TYPE_UNSIGNED (t2))
|
||
return 0;
|
||
return TYPE_LENGTH (t1) == TYPE_LENGTH (t2);
|
||
}
|
||
|
||
/* A convenience function to call get_base_type on CTX and return the
|
||
result. DIE is the DIE whose type we need. SIZE is non-zero if
|
||
this function should verify that the resulting type has the correct
|
||
size. */
|
||
|
||
static struct type *
|
||
dwarf_get_base_type (struct dwarf_expr_context *ctx, cu_offset die, int size)
|
||
{
|
||
struct type *result;
|
||
|
||
if (ctx->funcs->get_base_type)
|
||
{
|
||
result = ctx->funcs->get_base_type (ctx, die);
|
||
if (result == NULL)
|
||
error (_("Could not find type for DW_OP_GNU_const_type"));
|
||
if (size != 0 && TYPE_LENGTH (result) != size)
|
||
error (_("DW_OP_GNU_const_type has different sizes for type and data"));
|
||
}
|
||
else
|
||
/* Anything will do. */
|
||
result = builtin_type (ctx->gdbarch)->builtin_int;
|
||
|
||
return result;
|
||
}
|
||
|
||
/* If <BUF..BUF_END] contains DW_FORM_block* with single DW_OP_reg* return the
|
||
DWARF register number. Otherwise return -1. */
|
||
|
||
int
|
||
dwarf_block_to_dwarf_reg (const gdb_byte *buf, const gdb_byte *buf_end)
|
||
{
|
||
ULONGEST dwarf_reg;
|
||
|
||
if (buf_end <= buf)
|
||
return -1;
|
||
if (*buf >= DW_OP_reg0 && *buf <= DW_OP_reg31)
|
||
{
|
||
if (buf_end - buf != 1)
|
||
return -1;
|
||
return *buf - DW_OP_reg0;
|
||
}
|
||
|
||
if (*buf == DW_OP_GNU_regval_type)
|
||
{
|
||
buf++;
|
||
buf = read_uleb128 (buf, buf_end, &dwarf_reg);
|
||
buf = read_uleb128 (buf, buf_end, NULL);
|
||
}
|
||
else if (*buf == DW_OP_regx)
|
||
{
|
||
buf++;
|
||
buf = read_uleb128 (buf, buf_end, &dwarf_reg);
|
||
}
|
||
else
|
||
return -1;
|
||
if (buf != buf_end || (int) dwarf_reg != dwarf_reg)
|
||
return -1;
|
||
return dwarf_reg;
|
||
}
|
||
|
||
/* If <BUF..BUF_END] contains DW_FORM_block* with just DW_OP_breg*(0) and
|
||
DW_OP_deref* return the DWARF register number. Otherwise return -1.
|
||
DEREF_SIZE_RETURN contains -1 for DW_OP_deref; otherwise it contains the
|
||
size from DW_OP_deref_size. */
|
||
|
||
int
|
||
dwarf_block_to_dwarf_reg_deref (const gdb_byte *buf, const gdb_byte *buf_end,
|
||
CORE_ADDR *deref_size_return)
|
||
{
|
||
ULONGEST dwarf_reg;
|
||
LONGEST offset;
|
||
|
||
if (buf_end <= buf)
|
||
return -1;
|
||
if (*buf >= DW_OP_breg0 && *buf <= DW_OP_breg31)
|
||
{
|
||
dwarf_reg = *buf - DW_OP_breg0;
|
||
buf++;
|
||
}
|
||
else if (*buf == DW_OP_bregx)
|
||
{
|
||
buf++;
|
||
buf = read_uleb128 (buf, buf_end, &dwarf_reg);
|
||
if ((int) dwarf_reg != dwarf_reg)
|
||
return -1;
|
||
}
|
||
else
|
||
return -1;
|
||
|
||
buf = read_sleb128 (buf, buf_end, &offset);
|
||
if (offset != 0)
|
||
return -1;
|
||
|
||
if (buf >= buf_end)
|
||
return -1;
|
||
|
||
if (*buf == DW_OP_deref)
|
||
{
|
||
buf++;
|
||
*deref_size_return = -1;
|
||
}
|
||
else if (*buf == DW_OP_deref_size)
|
||
{
|
||
buf++;
|
||
if (buf >= buf_end)
|
||
return -1;
|
||
*deref_size_return = *buf++;
|
||
}
|
||
else
|
||
return -1;
|
||
|
||
if (buf != buf_end)
|
||
return -1;
|
||
|
||
return dwarf_reg;
|
||
}
|
||
|
||
/* If <BUF..BUF_END] contains DW_FORM_block* with single DW_OP_fbreg(X) fill
|
||
in FB_OFFSET_RETURN with the X offset and return 1. Otherwise return 0. */
|
||
|
||
int
|
||
dwarf_block_to_fb_offset (const gdb_byte *buf, const gdb_byte *buf_end,
|
||
CORE_ADDR *fb_offset_return)
|
||
{
|
||
LONGEST fb_offset;
|
||
|
||
if (buf_end <= buf)
|
||
return 0;
|
||
|
||
if (*buf != DW_OP_fbreg)
|
||
return 0;
|
||
buf++;
|
||
|
||
buf = read_sleb128 (buf, buf_end, &fb_offset);
|
||
*fb_offset_return = fb_offset;
|
||
if (buf != buf_end || fb_offset != (LONGEST) *fb_offset_return)
|
||
return 0;
|
||
|
||
return 1;
|
||
}
|
||
|
||
/* If <BUF..BUF_END] contains DW_FORM_block* with single DW_OP_bregSP(X) fill
|
||
in SP_OFFSET_RETURN with the X offset and return 1. Otherwise return 0.
|
||
The matched SP register number depends on GDBARCH. */
|
||
|
||
int
|
||
dwarf_block_to_sp_offset (struct gdbarch *gdbarch, const gdb_byte *buf,
|
||
const gdb_byte *buf_end, CORE_ADDR *sp_offset_return)
|
||
{
|
||
ULONGEST dwarf_reg;
|
||
LONGEST sp_offset;
|
||
|
||
if (buf_end <= buf)
|
||
return 0;
|
||
if (*buf >= DW_OP_breg0 && *buf <= DW_OP_breg31)
|
||
{
|
||
dwarf_reg = *buf - DW_OP_breg0;
|
||
buf++;
|
||
}
|
||
else
|
||
{
|
||
if (*buf != DW_OP_bregx)
|
||
return 0;
|
||
buf++;
|
||
buf = read_uleb128 (buf, buf_end, &dwarf_reg);
|
||
}
|
||
|
||
if (gdbarch_dwarf2_reg_to_regnum (gdbarch, dwarf_reg)
|
||
!= gdbarch_sp_regnum (gdbarch))
|
||
return 0;
|
||
|
||
buf = read_sleb128 (buf, buf_end, &sp_offset);
|
||
*sp_offset_return = sp_offset;
|
||
if (buf != buf_end || sp_offset != (LONGEST) *sp_offset_return)
|
||
return 0;
|
||
|
||
return 1;
|
||
}
|
||
|
||
/* The engine for the expression evaluator. Using the context in CTX,
|
||
evaluate the expression between OP_PTR and OP_END. */
|
||
|
||
static void
|
||
execute_stack_op (struct dwarf_expr_context *ctx,
|
||
const gdb_byte *op_ptr, const gdb_byte *op_end)
|
||
{
|
||
enum bfd_endian byte_order = gdbarch_byte_order (ctx->gdbarch);
|
||
/* Old-style "untyped" DWARF values need special treatment in a
|
||
couple of places, specifically DW_OP_mod and DW_OP_shr. We need
|
||
a special type for these values so we can distinguish them from
|
||
values that have an explicit type, because explicitly-typed
|
||
values do not need special treatment. This special type must be
|
||
different (in the `==' sense) from any base type coming from the
|
||
CU. */
|
||
struct type *address_type = dwarf_expr_address_type (ctx);
|
||
|
||
ctx->location = DWARF_VALUE_MEMORY;
|
||
ctx->initialized = 1; /* Default is initialized. */
|
||
|
||
if (ctx->recursion_depth > ctx->max_recursion_depth)
|
||
error (_("DWARF-2 expression error: Loop detected (%d)."),
|
||
ctx->recursion_depth);
|
||
ctx->recursion_depth++;
|
||
|
||
while (op_ptr < op_end)
|
||
{
|
||
enum dwarf_location_atom op = *op_ptr++;
|
||
ULONGEST result;
|
||
/* Assume the value is not in stack memory.
|
||
Code that knows otherwise sets this to 1.
|
||
Some arithmetic on stack addresses can probably be assumed to still
|
||
be a stack address, but we skip this complication for now.
|
||
This is just an optimization, so it's always ok to punt
|
||
and leave this as 0. */
|
||
int in_stack_memory = 0;
|
||
ULONGEST uoffset, reg;
|
||
LONGEST offset;
|
||
struct value *result_val = NULL;
|
||
|
||
/* The DWARF expression might have a bug causing an infinite
|
||
loop. In that case, quitting is the only way out. */
|
||
QUIT;
|
||
|
||
switch (op)
|
||
{
|
||
case DW_OP_lit0:
|
||
case DW_OP_lit1:
|
||
case DW_OP_lit2:
|
||
case DW_OP_lit3:
|
||
case DW_OP_lit4:
|
||
case DW_OP_lit5:
|
||
case DW_OP_lit6:
|
||
case DW_OP_lit7:
|
||
case DW_OP_lit8:
|
||
case DW_OP_lit9:
|
||
case DW_OP_lit10:
|
||
case DW_OP_lit11:
|
||
case DW_OP_lit12:
|
||
case DW_OP_lit13:
|
||
case DW_OP_lit14:
|
||
case DW_OP_lit15:
|
||
case DW_OP_lit16:
|
||
case DW_OP_lit17:
|
||
case DW_OP_lit18:
|
||
case DW_OP_lit19:
|
||
case DW_OP_lit20:
|
||
case DW_OP_lit21:
|
||
case DW_OP_lit22:
|
||
case DW_OP_lit23:
|
||
case DW_OP_lit24:
|
||
case DW_OP_lit25:
|
||
case DW_OP_lit26:
|
||
case DW_OP_lit27:
|
||
case DW_OP_lit28:
|
||
case DW_OP_lit29:
|
||
case DW_OP_lit30:
|
||
case DW_OP_lit31:
|
||
result = op - DW_OP_lit0;
|
||
result_val = value_from_ulongest (address_type, result);
|
||
break;
|
||
|
||
case DW_OP_addr:
|
||
result = extract_unsigned_integer (op_ptr,
|
||
ctx->addr_size, byte_order);
|
||
op_ptr += ctx->addr_size;
|
||
/* Some versions of GCC emit DW_OP_addr before
|
||
DW_OP_GNU_push_tls_address. In this case the value is an
|
||
index, not an address. We don't support things like
|
||
branching between the address and the TLS op. */
|
||
if (op_ptr >= op_end || *op_ptr != DW_OP_GNU_push_tls_address)
|
||
result += ctx->offset;
|
||
result_val = value_from_ulongest (address_type, result);
|
||
break;
|
||
|
||
case DW_OP_GNU_addr_index:
|
||
op_ptr = read_uleb128 (op_ptr, op_end, &uoffset);
|
||
result = (ctx->funcs->get_addr_index) (ctx->baton, uoffset);
|
||
result_val = value_from_ulongest (address_type, result);
|
||
break;
|
||
|
||
case DW_OP_const1u:
|
||
result = extract_unsigned_integer (op_ptr, 1, byte_order);
|
||
result_val = value_from_ulongest (address_type, result);
|
||
op_ptr += 1;
|
||
break;
|
||
case DW_OP_const1s:
|
||
result = extract_signed_integer (op_ptr, 1, byte_order);
|
||
result_val = value_from_ulongest (address_type, result);
|
||
op_ptr += 1;
|
||
break;
|
||
case DW_OP_const2u:
|
||
result = extract_unsigned_integer (op_ptr, 2, byte_order);
|
||
result_val = value_from_ulongest (address_type, result);
|
||
op_ptr += 2;
|
||
break;
|
||
case DW_OP_const2s:
|
||
result = extract_signed_integer (op_ptr, 2, byte_order);
|
||
result_val = value_from_ulongest (address_type, result);
|
||
op_ptr += 2;
|
||
break;
|
||
case DW_OP_const4u:
|
||
result = extract_unsigned_integer (op_ptr, 4, byte_order);
|
||
result_val = value_from_ulongest (address_type, result);
|
||
op_ptr += 4;
|
||
break;
|
||
case DW_OP_const4s:
|
||
result = extract_signed_integer (op_ptr, 4, byte_order);
|
||
result_val = value_from_ulongest (address_type, result);
|
||
op_ptr += 4;
|
||
break;
|
||
case DW_OP_const8u:
|
||
result = extract_unsigned_integer (op_ptr, 8, byte_order);
|
||
result_val = value_from_ulongest (address_type, result);
|
||
op_ptr += 8;
|
||
break;
|
||
case DW_OP_const8s:
|
||
result = extract_signed_integer (op_ptr, 8, byte_order);
|
||
result_val = value_from_ulongest (address_type, result);
|
||
op_ptr += 8;
|
||
break;
|
||
case DW_OP_constu:
|
||
op_ptr = read_uleb128 (op_ptr, op_end, &uoffset);
|
||
result = uoffset;
|
||
result_val = value_from_ulongest (address_type, result);
|
||
break;
|
||
case DW_OP_consts:
|
||
op_ptr = read_sleb128 (op_ptr, op_end, &offset);
|
||
result = offset;
|
||
result_val = value_from_ulongest (address_type, result);
|
||
break;
|
||
|
||
/* The DW_OP_reg operations are required to occur alone in
|
||
location expressions. */
|
||
case DW_OP_reg0:
|
||
case DW_OP_reg1:
|
||
case DW_OP_reg2:
|
||
case DW_OP_reg3:
|
||
case DW_OP_reg4:
|
||
case DW_OP_reg5:
|
||
case DW_OP_reg6:
|
||
case DW_OP_reg7:
|
||
case DW_OP_reg8:
|
||
case DW_OP_reg9:
|
||
case DW_OP_reg10:
|
||
case DW_OP_reg11:
|
||
case DW_OP_reg12:
|
||
case DW_OP_reg13:
|
||
case DW_OP_reg14:
|
||
case DW_OP_reg15:
|
||
case DW_OP_reg16:
|
||
case DW_OP_reg17:
|
||
case DW_OP_reg18:
|
||
case DW_OP_reg19:
|
||
case DW_OP_reg20:
|
||
case DW_OP_reg21:
|
||
case DW_OP_reg22:
|
||
case DW_OP_reg23:
|
||
case DW_OP_reg24:
|
||
case DW_OP_reg25:
|
||
case DW_OP_reg26:
|
||
case DW_OP_reg27:
|
||
case DW_OP_reg28:
|
||
case DW_OP_reg29:
|
||
case DW_OP_reg30:
|
||
case DW_OP_reg31:
|
||
if (op_ptr != op_end
|
||
&& *op_ptr != DW_OP_piece
|
||
&& *op_ptr != DW_OP_bit_piece
|
||
&& *op_ptr != DW_OP_GNU_uninit)
|
||
error (_("DWARF-2 expression error: DW_OP_reg operations must be "
|
||
"used either alone or in conjunction with DW_OP_piece "
|
||
"or DW_OP_bit_piece."));
|
||
|
||
result = op - DW_OP_reg0;
|
||
result_val = value_from_ulongest (address_type, result);
|
||
ctx->location = DWARF_VALUE_REGISTER;
|
||
break;
|
||
|
||
case DW_OP_regx:
|
||
op_ptr = read_uleb128 (op_ptr, op_end, ®);
|
||
dwarf_expr_require_composition (op_ptr, op_end, "DW_OP_regx");
|
||
|
||
result = reg;
|
||
result_val = value_from_ulongest (address_type, result);
|
||
ctx->location = DWARF_VALUE_REGISTER;
|
||
break;
|
||
|
||
case DW_OP_implicit_value:
|
||
{
|
||
ULONGEST len;
|
||
|
||
op_ptr = read_uleb128 (op_ptr, op_end, &len);
|
||
if (op_ptr + len > op_end)
|
||
error (_("DW_OP_implicit_value: too few bytes available."));
|
||
ctx->len = len;
|
||
ctx->data = op_ptr;
|
||
ctx->location = DWARF_VALUE_LITERAL;
|
||
op_ptr += len;
|
||
dwarf_expr_require_composition (op_ptr, op_end,
|
||
"DW_OP_implicit_value");
|
||
}
|
||
goto no_push;
|
||
|
||
case DW_OP_stack_value:
|
||
ctx->location = DWARF_VALUE_STACK;
|
||
dwarf_expr_require_composition (op_ptr, op_end, "DW_OP_stack_value");
|
||
goto no_push;
|
||
|
||
case DW_OP_GNU_implicit_pointer:
|
||
{
|
||
ULONGEST die;
|
||
LONGEST len;
|
||
|
||
if (ctx->ref_addr_size == -1)
|
||
error (_("DWARF-2 expression error: DW_OP_GNU_implicit_pointer "
|
||
"is not allowed in frame context"));
|
||
|
||
/* The referred-to DIE of cu_offset kind. */
|
||
ctx->len = extract_unsigned_integer (op_ptr, ctx->ref_addr_size,
|
||
byte_order);
|
||
op_ptr += ctx->ref_addr_size;
|
||
|
||
/* The byte offset into the data. */
|
||
op_ptr = read_sleb128 (op_ptr, op_end, &len);
|
||
result = (ULONGEST) len;
|
||
result_val = value_from_ulongest (address_type, result);
|
||
|
||
ctx->location = DWARF_VALUE_IMPLICIT_POINTER;
|
||
dwarf_expr_require_composition (op_ptr, op_end,
|
||
"DW_OP_GNU_implicit_pointer");
|
||
}
|
||
break;
|
||
|
||
case DW_OP_breg0:
|
||
case DW_OP_breg1:
|
||
case DW_OP_breg2:
|
||
case DW_OP_breg3:
|
||
case DW_OP_breg4:
|
||
case DW_OP_breg5:
|
||
case DW_OP_breg6:
|
||
case DW_OP_breg7:
|
||
case DW_OP_breg8:
|
||
case DW_OP_breg9:
|
||
case DW_OP_breg10:
|
||
case DW_OP_breg11:
|
||
case DW_OP_breg12:
|
||
case DW_OP_breg13:
|
||
case DW_OP_breg14:
|
||
case DW_OP_breg15:
|
||
case DW_OP_breg16:
|
||
case DW_OP_breg17:
|
||
case DW_OP_breg18:
|
||
case DW_OP_breg19:
|
||
case DW_OP_breg20:
|
||
case DW_OP_breg21:
|
||
case DW_OP_breg22:
|
||
case DW_OP_breg23:
|
||
case DW_OP_breg24:
|
||
case DW_OP_breg25:
|
||
case DW_OP_breg26:
|
||
case DW_OP_breg27:
|
||
case DW_OP_breg28:
|
||
case DW_OP_breg29:
|
||
case DW_OP_breg30:
|
||
case DW_OP_breg31:
|
||
{
|
||
op_ptr = read_sleb128 (op_ptr, op_end, &offset);
|
||
result = (ctx->funcs->read_reg) (ctx->baton, op - DW_OP_breg0);
|
||
result += offset;
|
||
result_val = value_from_ulongest (address_type, result);
|
||
}
|
||
break;
|
||
case DW_OP_bregx:
|
||
{
|
||
op_ptr = read_uleb128 (op_ptr, op_end, ®);
|
||
op_ptr = read_sleb128 (op_ptr, op_end, &offset);
|
||
result = (ctx->funcs->read_reg) (ctx->baton, reg);
|
||
result += offset;
|
||
result_val = value_from_ulongest (address_type, result);
|
||
}
|
||
break;
|
||
case DW_OP_fbreg:
|
||
{
|
||
const gdb_byte *datastart;
|
||
size_t datalen;
|
||
unsigned int before_stack_len;
|
||
|
||
op_ptr = read_sleb128 (op_ptr, op_end, &offset);
|
||
/* Rather than create a whole new context, we simply
|
||
record the stack length before execution, then reset it
|
||
afterwards, effectively erasing whatever the recursive
|
||
call put there. */
|
||
before_stack_len = ctx->stack_len;
|
||
/* FIXME: cagney/2003-03-26: This code should be using
|
||
get_frame_base_address(), and then implement a dwarf2
|
||
specific this_base method. */
|
||
(ctx->funcs->get_frame_base) (ctx->baton, &datastart, &datalen);
|
||
dwarf_expr_eval (ctx, datastart, datalen);
|
||
if (ctx->location == DWARF_VALUE_MEMORY)
|
||
result = dwarf_expr_fetch_address (ctx, 0);
|
||
else if (ctx->location == DWARF_VALUE_REGISTER)
|
||
result = (ctx->funcs->read_reg) (ctx->baton,
|
||
value_as_long (dwarf_expr_fetch (ctx, 0)));
|
||
else
|
||
error (_("Not implemented: computing frame "
|
||
"base using explicit value operator"));
|
||
result = result + offset;
|
||
result_val = value_from_ulongest (address_type, result);
|
||
in_stack_memory = 1;
|
||
ctx->stack_len = before_stack_len;
|
||
ctx->location = DWARF_VALUE_MEMORY;
|
||
}
|
||
break;
|
||
|
||
case DW_OP_dup:
|
||
result_val = dwarf_expr_fetch (ctx, 0);
|
||
in_stack_memory = dwarf_expr_fetch_in_stack_memory (ctx, 0);
|
||
break;
|
||
|
||
case DW_OP_drop:
|
||
dwarf_expr_pop (ctx);
|
||
goto no_push;
|
||
|
||
case DW_OP_pick:
|
||
offset = *op_ptr++;
|
||
result_val = dwarf_expr_fetch (ctx, offset);
|
||
in_stack_memory = dwarf_expr_fetch_in_stack_memory (ctx, offset);
|
||
break;
|
||
|
||
case DW_OP_swap:
|
||
{
|
||
struct dwarf_stack_value t1, t2;
|
||
|
||
if (ctx->stack_len < 2)
|
||
error (_("Not enough elements for "
|
||
"DW_OP_swap. Need 2, have %d."),
|
||
ctx->stack_len);
|
||
t1 = ctx->stack[ctx->stack_len - 1];
|
||
t2 = ctx->stack[ctx->stack_len - 2];
|
||
ctx->stack[ctx->stack_len - 1] = t2;
|
||
ctx->stack[ctx->stack_len - 2] = t1;
|
||
goto no_push;
|
||
}
|
||
|
||
case DW_OP_over:
|
||
result_val = dwarf_expr_fetch (ctx, 1);
|
||
in_stack_memory = dwarf_expr_fetch_in_stack_memory (ctx, 1);
|
||
break;
|
||
|
||
case DW_OP_rot:
|
||
{
|
||
struct dwarf_stack_value t1, t2, t3;
|
||
|
||
if (ctx->stack_len < 3)
|
||
error (_("Not enough elements for "
|
||
"DW_OP_rot. Need 3, have %d."),
|
||
ctx->stack_len);
|
||
t1 = ctx->stack[ctx->stack_len - 1];
|
||
t2 = ctx->stack[ctx->stack_len - 2];
|
||
t3 = ctx->stack[ctx->stack_len - 3];
|
||
ctx->stack[ctx->stack_len - 1] = t2;
|
||
ctx->stack[ctx->stack_len - 2] = t3;
|
||
ctx->stack[ctx->stack_len - 3] = t1;
|
||
goto no_push;
|
||
}
|
||
|
||
case DW_OP_deref:
|
||
case DW_OP_deref_size:
|
||
case DW_OP_GNU_deref_type:
|
||
{
|
||
int addr_size = (op == DW_OP_deref ? ctx->addr_size : *op_ptr++);
|
||
gdb_byte *buf = alloca (addr_size);
|
||
CORE_ADDR addr = dwarf_expr_fetch_address (ctx, 0);
|
||
struct type *type;
|
||
|
||
dwarf_expr_pop (ctx);
|
||
|
||
if (op == DW_OP_GNU_deref_type)
|
||
{
|
||
cu_offset type_die;
|
||
|
||
op_ptr = read_uleb128 (op_ptr, op_end, &uoffset);
|
||
type_die.cu_off = uoffset;
|
||
type = dwarf_get_base_type (ctx, type_die, 0);
|
||
}
|
||
else
|
||
type = address_type;
|
||
|
||
(ctx->funcs->read_mem) (ctx->baton, buf, addr, addr_size);
|
||
|
||
/* If the size of the object read from memory is different
|
||
from the type length, we need to zero-extend it. */
|
||
if (TYPE_LENGTH (type) != addr_size)
|
||
{
|
||
ULONGEST result =
|
||
extract_unsigned_integer (buf, addr_size, byte_order);
|
||
|
||
buf = alloca (TYPE_LENGTH (type));
|
||
store_unsigned_integer (buf, TYPE_LENGTH (type),
|
||
byte_order, result);
|
||
}
|
||
|
||
result_val = value_from_contents_and_address (type, buf, addr);
|
||
break;
|
||
}
|
||
|
||
case DW_OP_abs:
|
||
case DW_OP_neg:
|
||
case DW_OP_not:
|
||
case DW_OP_plus_uconst:
|
||
{
|
||
/* Unary operations. */
|
||
result_val = dwarf_expr_fetch (ctx, 0);
|
||
dwarf_expr_pop (ctx);
|
||
|
||
switch (op)
|
||
{
|
||
case DW_OP_abs:
|
||
if (value_less (result_val,
|
||
value_zero (value_type (result_val), not_lval)))
|
||
result_val = value_neg (result_val);
|
||
break;
|
||
case DW_OP_neg:
|
||
result_val = value_neg (result_val);
|
||
break;
|
||
case DW_OP_not:
|
||
dwarf_require_integral (value_type (result_val));
|
||
result_val = value_complement (result_val);
|
||
break;
|
||
case DW_OP_plus_uconst:
|
||
dwarf_require_integral (value_type (result_val));
|
||
result = value_as_long (result_val);
|
||
op_ptr = read_uleb128 (op_ptr, op_end, ®);
|
||
result += reg;
|
||
result_val = value_from_ulongest (address_type, result);
|
||
break;
|
||
}
|
||
}
|
||
break;
|
||
|
||
case DW_OP_and:
|
||
case DW_OP_div:
|
||
case DW_OP_minus:
|
||
case DW_OP_mod:
|
||
case DW_OP_mul:
|
||
case DW_OP_or:
|
||
case DW_OP_plus:
|
||
case DW_OP_shl:
|
||
case DW_OP_shr:
|
||
case DW_OP_shra:
|
||
case DW_OP_xor:
|
||
case DW_OP_le:
|
||
case DW_OP_ge:
|
||
case DW_OP_eq:
|
||
case DW_OP_lt:
|
||
case DW_OP_gt:
|
||
case DW_OP_ne:
|
||
{
|
||
/* Binary operations. */
|
||
struct value *first, *second;
|
||
|
||
second = dwarf_expr_fetch (ctx, 0);
|
||
dwarf_expr_pop (ctx);
|
||
|
||
first = dwarf_expr_fetch (ctx, 0);
|
||
dwarf_expr_pop (ctx);
|
||
|
||
if (! base_types_equal_p (value_type (first), value_type (second)))
|
||
error (_("Incompatible types on DWARF stack"));
|
||
|
||
switch (op)
|
||
{
|
||
case DW_OP_and:
|
||
dwarf_require_integral (value_type (first));
|
||
dwarf_require_integral (value_type (second));
|
||
result_val = value_binop (first, second, BINOP_BITWISE_AND);
|
||
break;
|
||
case DW_OP_div:
|
||
result_val = value_binop (first, second, BINOP_DIV);
|
||
break;
|
||
case DW_OP_minus:
|
||
result_val = value_binop (first, second, BINOP_SUB);
|
||
break;
|
||
case DW_OP_mod:
|
||
{
|
||
int cast_back = 0;
|
||
struct type *orig_type = value_type (first);
|
||
|
||
/* We have to special-case "old-style" untyped values
|
||
-- these must have mod computed using unsigned
|
||
math. */
|
||
if (orig_type == address_type)
|
||
{
|
||
struct type *utype
|
||
= get_unsigned_type (ctx->gdbarch, orig_type);
|
||
|
||
cast_back = 1;
|
||
first = value_cast (utype, first);
|
||
second = value_cast (utype, second);
|
||
}
|
||
/* Note that value_binop doesn't handle float or
|
||
decimal float here. This seems unimportant. */
|
||
result_val = value_binop (first, second, BINOP_MOD);
|
||
if (cast_back)
|
||
result_val = value_cast (orig_type, result_val);
|
||
}
|
||
break;
|
||
case DW_OP_mul:
|
||
result_val = value_binop (first, second, BINOP_MUL);
|
||
break;
|
||
case DW_OP_or:
|
||
dwarf_require_integral (value_type (first));
|
||
dwarf_require_integral (value_type (second));
|
||
result_val = value_binop (first, second, BINOP_BITWISE_IOR);
|
||
break;
|
||
case DW_OP_plus:
|
||
result_val = value_binop (first, second, BINOP_ADD);
|
||
break;
|
||
case DW_OP_shl:
|
||
dwarf_require_integral (value_type (first));
|
||
dwarf_require_integral (value_type (second));
|
||
result_val = value_binop (first, second, BINOP_LSH);
|
||
break;
|
||
case DW_OP_shr:
|
||
dwarf_require_integral (value_type (first));
|
||
dwarf_require_integral (value_type (second));
|
||
if (!TYPE_UNSIGNED (value_type (first)))
|
||
{
|
||
struct type *utype
|
||
= get_unsigned_type (ctx->gdbarch, value_type (first));
|
||
|
||
first = value_cast (utype, first);
|
||
}
|
||
|
||
result_val = value_binop (first, second, BINOP_RSH);
|
||
/* Make sure we wind up with the same type we started
|
||
with. */
|
||
if (value_type (result_val) != value_type (second))
|
||
result_val = value_cast (value_type (second), result_val);
|
||
break;
|
||
case DW_OP_shra:
|
||
dwarf_require_integral (value_type (first));
|
||
dwarf_require_integral (value_type (second));
|
||
if (TYPE_UNSIGNED (value_type (first)))
|
||
{
|
||
struct type *stype
|
||
= get_signed_type (ctx->gdbarch, value_type (first));
|
||
|
||
first = value_cast (stype, first);
|
||
}
|
||
|
||
result_val = value_binop (first, second, BINOP_RSH);
|
||
/* Make sure we wind up with the same type we started
|
||
with. */
|
||
if (value_type (result_val) != value_type (second))
|
||
result_val = value_cast (value_type (second), result_val);
|
||
break;
|
||
case DW_OP_xor:
|
||
dwarf_require_integral (value_type (first));
|
||
dwarf_require_integral (value_type (second));
|
||
result_val = value_binop (first, second, BINOP_BITWISE_XOR);
|
||
break;
|
||
case DW_OP_le:
|
||
/* A <= B is !(B < A). */
|
||
result = ! value_less (second, first);
|
||
result_val = value_from_ulongest (address_type, result);
|
||
break;
|
||
case DW_OP_ge:
|
||
/* A >= B is !(A < B). */
|
||
result = ! value_less (first, second);
|
||
result_val = value_from_ulongest (address_type, result);
|
||
break;
|
||
case DW_OP_eq:
|
||
result = value_equal (first, second);
|
||
result_val = value_from_ulongest (address_type, result);
|
||
break;
|
||
case DW_OP_lt:
|
||
result = value_less (first, second);
|
||
result_val = value_from_ulongest (address_type, result);
|
||
break;
|
||
case DW_OP_gt:
|
||
/* A > B is B < A. */
|
||
result = value_less (second, first);
|
||
result_val = value_from_ulongest (address_type, result);
|
||
break;
|
||
case DW_OP_ne:
|
||
result = ! value_equal (first, second);
|
||
result_val = value_from_ulongest (address_type, result);
|
||
break;
|
||
default:
|
||
internal_error (__FILE__, __LINE__,
|
||
_("Can't be reached."));
|
||
}
|
||
}
|
||
break;
|
||
|
||
case DW_OP_call_frame_cfa:
|
||
result = (ctx->funcs->get_frame_cfa) (ctx->baton);
|
||
result_val = value_from_ulongest (address_type, result);
|
||
in_stack_memory = 1;
|
||
break;
|
||
|
||
case DW_OP_GNU_push_tls_address:
|
||
/* Variable is at a constant offset in the thread-local
|
||
storage block into the objfile for the current thread and
|
||
the dynamic linker module containing this expression. Here
|
||
we return returns the offset from that base. The top of the
|
||
stack has the offset from the beginning of the thread
|
||
control block at which the variable is located. Nothing
|
||
should follow this operator, so the top of stack would be
|
||
returned. */
|
||
result = value_as_long (dwarf_expr_fetch (ctx, 0));
|
||
dwarf_expr_pop (ctx);
|
||
result = (ctx->funcs->get_tls_address) (ctx->baton, result);
|
||
result_val = value_from_ulongest (address_type, result);
|
||
break;
|
||
|
||
case DW_OP_skip:
|
||
offset = extract_signed_integer (op_ptr, 2, byte_order);
|
||
op_ptr += 2;
|
||
op_ptr += offset;
|
||
goto no_push;
|
||
|
||
case DW_OP_bra:
|
||
{
|
||
struct value *val;
|
||
|
||
offset = extract_signed_integer (op_ptr, 2, byte_order);
|
||
op_ptr += 2;
|
||
val = dwarf_expr_fetch (ctx, 0);
|
||
dwarf_require_integral (value_type (val));
|
||
if (value_as_long (val) != 0)
|
||
op_ptr += offset;
|
||
dwarf_expr_pop (ctx);
|
||
}
|
||
goto no_push;
|
||
|
||
case DW_OP_nop:
|
||
goto no_push;
|
||
|
||
case DW_OP_piece:
|
||
{
|
||
ULONGEST size;
|
||
|
||
/* Record the piece. */
|
||
op_ptr = read_uleb128 (op_ptr, op_end, &size);
|
||
add_piece (ctx, 8 * size, 0);
|
||
|
||
/* Pop off the address/regnum, and reset the location
|
||
type. */
|
||
if (ctx->location != DWARF_VALUE_LITERAL
|
||
&& ctx->location != DWARF_VALUE_OPTIMIZED_OUT)
|
||
dwarf_expr_pop (ctx);
|
||
ctx->location = DWARF_VALUE_MEMORY;
|
||
}
|
||
goto no_push;
|
||
|
||
case DW_OP_bit_piece:
|
||
{
|
||
ULONGEST size, offset;
|
||
|
||
/* Record the piece. */
|
||
op_ptr = read_uleb128 (op_ptr, op_end, &size);
|
||
op_ptr = read_uleb128 (op_ptr, op_end, &offset);
|
||
add_piece (ctx, size, offset);
|
||
|
||
/* Pop off the address/regnum, and reset the location
|
||
type. */
|
||
if (ctx->location != DWARF_VALUE_LITERAL
|
||
&& ctx->location != DWARF_VALUE_OPTIMIZED_OUT)
|
||
dwarf_expr_pop (ctx);
|
||
ctx->location = DWARF_VALUE_MEMORY;
|
||
}
|
||
goto no_push;
|
||
|
||
case DW_OP_GNU_uninit:
|
||
if (op_ptr != op_end)
|
||
error (_("DWARF-2 expression error: DW_OP_GNU_uninit must always "
|
||
"be the very last op."));
|
||
|
||
ctx->initialized = 0;
|
||
goto no_push;
|
||
|
||
case DW_OP_call2:
|
||
{
|
||
cu_offset offset;
|
||
|
||
offset.cu_off = extract_unsigned_integer (op_ptr, 2, byte_order);
|
||
op_ptr += 2;
|
||
ctx->funcs->dwarf_call (ctx, offset);
|
||
}
|
||
goto no_push;
|
||
|
||
case DW_OP_call4:
|
||
{
|
||
cu_offset offset;
|
||
|
||
offset.cu_off = extract_unsigned_integer (op_ptr, 4, byte_order);
|
||
op_ptr += 4;
|
||
ctx->funcs->dwarf_call (ctx, offset);
|
||
}
|
||
goto no_push;
|
||
|
||
case DW_OP_GNU_entry_value:
|
||
{
|
||
ULONGEST len;
|
||
int dwarf_reg;
|
||
CORE_ADDR deref_size;
|
||
|
||
op_ptr = read_uleb128 (op_ptr, op_end, &len);
|
||
if (op_ptr + len > op_end)
|
||
error (_("DW_OP_GNU_entry_value: too few bytes available."));
|
||
|
||
dwarf_reg = dwarf_block_to_dwarf_reg (op_ptr, op_ptr + len);
|
||
if (dwarf_reg != -1)
|
||
{
|
||
op_ptr += len;
|
||
ctx->funcs->push_dwarf_reg_entry_value (ctx, dwarf_reg,
|
||
0 /* unused */,
|
||
-1 /* deref_size */);
|
||
goto no_push;
|
||
}
|
||
|
||
dwarf_reg = dwarf_block_to_dwarf_reg_deref (op_ptr, op_ptr + len,
|
||
&deref_size);
|
||
if (dwarf_reg != -1)
|
||
{
|
||
if (deref_size == -1)
|
||
deref_size = ctx->addr_size;
|
||
op_ptr += len;
|
||
ctx->funcs->push_dwarf_reg_entry_value (ctx, dwarf_reg,
|
||
0 /* unused */,
|
||
deref_size);
|
||
goto no_push;
|
||
}
|
||
|
||
error (_("DWARF-2 expression error: DW_OP_GNU_entry_value is "
|
||
"supported only for single DW_OP_reg* "
|
||
"or for DW_OP_breg*(0)+DW_OP_deref*"));
|
||
}
|
||
|
||
case DW_OP_GNU_const_type:
|
||
{
|
||
cu_offset type_die;
|
||
int n;
|
||
const gdb_byte *data;
|
||
struct type *type;
|
||
|
||
op_ptr = read_uleb128 (op_ptr, op_end, &uoffset);
|
||
type_die.cu_off = uoffset;
|
||
n = *op_ptr++;
|
||
data = op_ptr;
|
||
op_ptr += n;
|
||
|
||
type = dwarf_get_base_type (ctx, type_die, n);
|
||
result_val = value_from_contents (type, data);
|
||
}
|
||
break;
|
||
|
||
case DW_OP_GNU_regval_type:
|
||
{
|
||
cu_offset type_die;
|
||
struct type *type;
|
||
|
||
op_ptr = read_uleb128 (op_ptr, op_end, ®);
|
||
op_ptr = read_uleb128 (op_ptr, op_end, &uoffset);
|
||
type_die.cu_off = uoffset;
|
||
|
||
type = dwarf_get_base_type (ctx, type_die, 0);
|
||
result = (ctx->funcs->read_reg) (ctx->baton, reg);
|
||
result_val = value_from_ulongest (address_type, result);
|
||
result_val = value_from_contents (type,
|
||
value_contents_all (result_val));
|
||
}
|
||
break;
|
||
|
||
case DW_OP_GNU_convert:
|
||
case DW_OP_GNU_reinterpret:
|
||
{
|
||
cu_offset type_die;
|
||
struct type *type;
|
||
|
||
op_ptr = read_uleb128 (op_ptr, op_end, &uoffset);
|
||
type_die.cu_off = uoffset;
|
||
|
||
if (type_die.cu_off == 0)
|
||
type = address_type;
|
||
else
|
||
type = dwarf_get_base_type (ctx, type_die, 0);
|
||
|
||
result_val = dwarf_expr_fetch (ctx, 0);
|
||
dwarf_expr_pop (ctx);
|
||
|
||
if (op == DW_OP_GNU_convert)
|
||
result_val = value_cast (type, result_val);
|
||
else if (type == value_type (result_val))
|
||
{
|
||
/* Nothing. */
|
||
}
|
||
else if (TYPE_LENGTH (type)
|
||
!= TYPE_LENGTH (value_type (result_val)))
|
||
error (_("DW_OP_GNU_reinterpret has wrong size"));
|
||
else
|
||
result_val
|
||
= value_from_contents (type,
|
||
value_contents_all (result_val));
|
||
}
|
||
break;
|
||
|
||
default:
|
||
error (_("Unhandled dwarf expression opcode 0x%x"), op);
|
||
}
|
||
|
||
/* Most things push a result value. */
|
||
gdb_assert (result_val != NULL);
|
||
dwarf_expr_push (ctx, result_val, in_stack_memory);
|
||
no_push:
|
||
;
|
||
}
|
||
|
||
/* To simplify our main caller, if the result is an implicit
|
||
pointer, then make a pieced value. This is ok because we can't
|
||
have implicit pointers in contexts where pieces are invalid. */
|
||
if (ctx->location == DWARF_VALUE_IMPLICIT_POINTER)
|
||
add_piece (ctx, 8 * ctx->addr_size, 0);
|
||
|
||
abort_expression:
|
||
ctx->recursion_depth--;
|
||
gdb_assert (ctx->recursion_depth >= 0);
|
||
}
|
||
|
||
/* Stub dwarf_expr_context_funcs.get_frame_base implementation. */
|
||
|
||
void
|
||
ctx_no_get_frame_base (void *baton, const gdb_byte **start, size_t *length)
|
||
{
|
||
error (_("%s is invalid in this context"), "DW_OP_fbreg");
|
||
}
|
||
|
||
/* Stub dwarf_expr_context_funcs.get_frame_cfa implementation. */
|
||
|
||
CORE_ADDR
|
||
ctx_no_get_frame_cfa (void *baton)
|
||
{
|
||
error (_("%s is invalid in this context"), "DW_OP_call_frame_cfa");
|
||
}
|
||
|
||
/* Stub dwarf_expr_context_funcs.get_frame_pc implementation. */
|
||
|
||
CORE_ADDR
|
||
ctx_no_get_frame_pc (void *baton)
|
||
{
|
||
error (_("%s is invalid in this context"), "DW_OP_GNU_implicit_pointer");
|
||
}
|
||
|
||
/* Stub dwarf_expr_context_funcs.get_tls_address implementation. */
|
||
|
||
CORE_ADDR
|
||
ctx_no_get_tls_address (void *baton, CORE_ADDR offset)
|
||
{
|
||
error (_("%s is invalid in this context"), "DW_OP_GNU_push_tls_address");
|
||
}
|
||
|
||
/* Stub dwarf_expr_context_funcs.dwarf_call implementation. */
|
||
|
||
void
|
||
ctx_no_dwarf_call (struct dwarf_expr_context *ctx, cu_offset die_offset)
|
||
{
|
||
error (_("%s is invalid in this context"), "DW_OP_call*");
|
||
}
|
||
|
||
/* Stub dwarf_expr_context_funcs.get_base_type implementation. */
|
||
|
||
struct type *
|
||
ctx_no_get_base_type (struct dwarf_expr_context *ctx, cu_offset die)
|
||
{
|
||
error (_("Support for typed DWARF is not supported in this context"));
|
||
}
|
||
|
||
/* Stub dwarf_expr_context_funcs.push_dwarf_block_entry_value
|
||
implementation. */
|
||
|
||
void
|
||
ctx_no_push_dwarf_reg_entry_value (struct dwarf_expr_context *ctx,
|
||
int dwarf_reg, CORE_ADDR fb_offset,
|
||
int deref_size)
|
||
{
|
||
internal_error (__FILE__, __LINE__,
|
||
_("Support for DW_OP_GNU_entry_value is unimplemented"));
|
||
}
|
||
|
||
/* Stub dwarf_expr_context_funcs.get_addr_index implementation. */
|
||
|
||
CORE_ADDR
|
||
ctx_no_get_addr_index (void *baton, unsigned int index)
|
||
{
|
||
error (_("%s is invalid in this context"), "DW_OP_GNU_addr_index");
|
||
}
|
||
|
||
/* Provide a prototype to silence -Wmissing-prototypes. */
|
||
extern initialize_file_ftype _initialize_dwarf2expr;
|
||
|
||
void
|
||
_initialize_dwarf2expr (void)
|
||
{
|
||
dwarf_arch_cookie
|
||
= gdbarch_data_register_post_init (dwarf_gdbarch_types_init);
|
||
}
|