1*7304104dSAndroid Build Coastguard Worker /* Get register location expression for frame.
2*7304104dSAndroid Build Coastguard Worker Copyright (C) 2009-2010, 2014 Red Hat, Inc.
3*7304104dSAndroid Build Coastguard Worker This file is part of elfutils.
4*7304104dSAndroid Build Coastguard Worker
5*7304104dSAndroid Build Coastguard Worker This file is free software; you can redistribute it and/or modify
6*7304104dSAndroid Build Coastguard Worker it under the terms of either
7*7304104dSAndroid Build Coastguard Worker
8*7304104dSAndroid Build Coastguard Worker * the GNU Lesser General Public License as published by the Free
9*7304104dSAndroid Build Coastguard Worker Software Foundation; either version 3 of the License, or (at
10*7304104dSAndroid Build Coastguard Worker your option) any later version
11*7304104dSAndroid Build Coastguard Worker
12*7304104dSAndroid Build Coastguard Worker or
13*7304104dSAndroid Build Coastguard Worker
14*7304104dSAndroid Build Coastguard Worker * the GNU General Public License as published by the Free
15*7304104dSAndroid Build Coastguard Worker Software Foundation; either version 2 of the License, or (at
16*7304104dSAndroid Build Coastguard Worker your option) any later version
17*7304104dSAndroid Build Coastguard Worker
18*7304104dSAndroid Build Coastguard Worker or both in parallel, as here.
19*7304104dSAndroid Build Coastguard Worker
20*7304104dSAndroid Build Coastguard Worker elfutils is distributed in the hope that it will be useful, but
21*7304104dSAndroid Build Coastguard Worker WITHOUT ANY WARRANTY; without even the implied warranty of
22*7304104dSAndroid Build Coastguard Worker MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
23*7304104dSAndroid Build Coastguard Worker General Public License for more details.
24*7304104dSAndroid Build Coastguard Worker
25*7304104dSAndroid Build Coastguard Worker You should have received copies of the GNU General Public License and
26*7304104dSAndroid Build Coastguard Worker the GNU Lesser General Public License along with this program. If
27*7304104dSAndroid Build Coastguard Worker not, see <http://www.gnu.org/licenses/>. */
28*7304104dSAndroid Build Coastguard Worker
29*7304104dSAndroid Build Coastguard Worker #ifdef HAVE_CONFIG_H
30*7304104dSAndroid Build Coastguard Worker # include <config.h>
31*7304104dSAndroid Build Coastguard Worker #endif
32*7304104dSAndroid Build Coastguard Worker
33*7304104dSAndroid Build Coastguard Worker #include "cfi.h"
34*7304104dSAndroid Build Coastguard Worker #include <dwarf.h>
35*7304104dSAndroid Build Coastguard Worker
36*7304104dSAndroid Build Coastguard Worker int
dwarf_frame_register(Dwarf_Frame * fs,int regno,Dwarf_Op ops_mem[3],Dwarf_Op ** ops,size_t * nops)37*7304104dSAndroid Build Coastguard Worker dwarf_frame_register (Dwarf_Frame *fs, int regno, Dwarf_Op ops_mem[3],
38*7304104dSAndroid Build Coastguard Worker Dwarf_Op **ops, size_t *nops)
39*7304104dSAndroid Build Coastguard Worker {
40*7304104dSAndroid Build Coastguard Worker /* Maybe there was a previous error. */
41*7304104dSAndroid Build Coastguard Worker if (fs == NULL)
42*7304104dSAndroid Build Coastguard Worker return -1;
43*7304104dSAndroid Build Coastguard Worker
44*7304104dSAndroid Build Coastguard Worker if (unlikely (regno < 0))
45*7304104dSAndroid Build Coastguard Worker {
46*7304104dSAndroid Build Coastguard Worker __libdw_seterrno (DWARF_E_INVALID_ACCESS);
47*7304104dSAndroid Build Coastguard Worker return -1;
48*7304104dSAndroid Build Coastguard Worker }
49*7304104dSAndroid Build Coastguard Worker
50*7304104dSAndroid Build Coastguard Worker *ops = ops_mem;
51*7304104dSAndroid Build Coastguard Worker *nops = 0;
52*7304104dSAndroid Build Coastguard Worker
53*7304104dSAndroid Build Coastguard Worker if (unlikely ((size_t) regno >= fs->nregs))
54*7304104dSAndroid Build Coastguard Worker goto default_rule;
55*7304104dSAndroid Build Coastguard Worker
56*7304104dSAndroid Build Coastguard Worker const struct dwarf_frame_register *reg = &fs->regs[regno];
57*7304104dSAndroid Build Coastguard Worker
58*7304104dSAndroid Build Coastguard Worker switch (reg->rule)
59*7304104dSAndroid Build Coastguard Worker {
60*7304104dSAndroid Build Coastguard Worker case reg_unspecified:
61*7304104dSAndroid Build Coastguard Worker default_rule:
62*7304104dSAndroid Build Coastguard Worker /* Use the default rule for registers not yet mentioned in CFI. */
63*7304104dSAndroid Build Coastguard Worker if (fs->cache->default_same_value)
64*7304104dSAndroid Build Coastguard Worker goto same_value;
65*7304104dSAndroid Build Coastguard Worker FALLTHROUGH;
66*7304104dSAndroid Build Coastguard Worker case reg_undefined:
67*7304104dSAndroid Build Coastguard Worker /* The value is known to be unavailable. */
68*7304104dSAndroid Build Coastguard Worker break;
69*7304104dSAndroid Build Coastguard Worker
70*7304104dSAndroid Build Coastguard Worker case reg_same_value:
71*7304104dSAndroid Build Coastguard Worker same_value:
72*7304104dSAndroid Build Coastguard Worker /* The location is not known here, but the caller might know it. */
73*7304104dSAndroid Build Coastguard Worker *ops = NULL;
74*7304104dSAndroid Build Coastguard Worker break;
75*7304104dSAndroid Build Coastguard Worker
76*7304104dSAndroid Build Coastguard Worker case reg_offset:
77*7304104dSAndroid Build Coastguard Worker case reg_val_offset:
78*7304104dSAndroid Build Coastguard Worker ops_mem[(*nops)++] = (Dwarf_Op) { .atom = DW_OP_call_frame_cfa };
79*7304104dSAndroid Build Coastguard Worker if (reg->value != 0)
80*7304104dSAndroid Build Coastguard Worker ops_mem[(*nops)++] = (Dwarf_Op) { .atom = DW_OP_plus_uconst,
81*7304104dSAndroid Build Coastguard Worker .number = reg->value };
82*7304104dSAndroid Build Coastguard Worker if (reg->rule == reg_val_offset)
83*7304104dSAndroid Build Coastguard Worker /* A value, not a location. */
84*7304104dSAndroid Build Coastguard Worker ops_mem[(*nops)++] = (Dwarf_Op) { .atom = DW_OP_stack_value };
85*7304104dSAndroid Build Coastguard Worker *ops = ops_mem;
86*7304104dSAndroid Build Coastguard Worker break;
87*7304104dSAndroid Build Coastguard Worker
88*7304104dSAndroid Build Coastguard Worker case reg_register:
89*7304104dSAndroid Build Coastguard Worker ops_mem[(*nops)++] = (Dwarf_Op) { .atom = DW_OP_regx,
90*7304104dSAndroid Build Coastguard Worker .number = reg->value };
91*7304104dSAndroid Build Coastguard Worker break;
92*7304104dSAndroid Build Coastguard Worker
93*7304104dSAndroid Build Coastguard Worker case reg_val_expression:
94*7304104dSAndroid Build Coastguard Worker case reg_expression:
95*7304104dSAndroid Build Coastguard Worker {
96*7304104dSAndroid Build Coastguard Worker unsigned int address_size = (fs->cache->e_ident[EI_CLASS] == ELFCLASS32
97*7304104dSAndroid Build Coastguard Worker ? 4 : 8);
98*7304104dSAndroid Build Coastguard Worker
99*7304104dSAndroid Build Coastguard Worker Dwarf_Block block;
100*7304104dSAndroid Build Coastguard Worker const uint8_t *p = fs->cache->data->d.d_buf + reg->value;
101*7304104dSAndroid Build Coastguard Worker const uint8_t *end = (fs->cache->data->d.d_buf
102*7304104dSAndroid Build Coastguard Worker + fs->cache->data->d.d_size);
103*7304104dSAndroid Build Coastguard Worker if (p >= end)
104*7304104dSAndroid Build Coastguard Worker {
105*7304104dSAndroid Build Coastguard Worker __libdw_seterrno (DWARF_E_INVALID_DWARF);
106*7304104dSAndroid Build Coastguard Worker return -1;
107*7304104dSAndroid Build Coastguard Worker }
108*7304104dSAndroid Build Coastguard Worker get_uleb128 (block.length, p, end);
109*7304104dSAndroid Build Coastguard Worker block.data = (void *) p;
110*7304104dSAndroid Build Coastguard Worker
111*7304104dSAndroid Build Coastguard Worker /* Parse the expression into internal form. */
112*7304104dSAndroid Build Coastguard Worker if (__libdw_intern_expression (NULL,
113*7304104dSAndroid Build Coastguard Worker fs->cache->other_byte_order,
114*7304104dSAndroid Build Coastguard Worker address_size, 4,
115*7304104dSAndroid Build Coastguard Worker &fs->cache->expr_tree, &block,
116*7304104dSAndroid Build Coastguard Worker true, reg->rule == reg_val_expression,
117*7304104dSAndroid Build Coastguard Worker ops, nops, IDX_debug_frame) < 0)
118*7304104dSAndroid Build Coastguard Worker return -1;
119*7304104dSAndroid Build Coastguard Worker break;
120*7304104dSAndroid Build Coastguard Worker }
121*7304104dSAndroid Build Coastguard Worker }
122*7304104dSAndroid Build Coastguard Worker
123*7304104dSAndroid Build Coastguard Worker return 0;
124*7304104dSAndroid Build Coastguard Worker }
125