xref: /aosp_15_r20/external/elfutils/libdw/dwarf_frame_register.c (revision 7304104da70ce23c86437a01be71edd1a2d7f37e)
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