xref: /aosp_15_r20/external/elfutils/libdw/libdw_form.c (revision 7304104da70ce23c86437a01be71edd1a2d7f37e)
1*7304104dSAndroid Build Coastguard Worker /* Helper functions for form handling.
2*7304104dSAndroid Build Coastguard Worker    Copyright (C) 2003-2009, 2014 Red Hat, Inc.
3*7304104dSAndroid Build Coastguard Worker    This file is part of elfutils.
4*7304104dSAndroid Build Coastguard Worker    Written by Ulrich Drepper <[email protected]>, 2003.
5*7304104dSAndroid Build Coastguard Worker 
6*7304104dSAndroid Build Coastguard Worker    This file is free software; you can redistribute it and/or modify
7*7304104dSAndroid Build Coastguard Worker    it under the terms of either
8*7304104dSAndroid Build Coastguard Worker 
9*7304104dSAndroid Build Coastguard Worker      * the GNU Lesser General Public License as published by the Free
10*7304104dSAndroid Build Coastguard Worker        Software Foundation; either version 3 of the License, or (at
11*7304104dSAndroid Build Coastguard Worker        your option) any later version
12*7304104dSAndroid Build Coastguard Worker 
13*7304104dSAndroid Build Coastguard Worker    or
14*7304104dSAndroid Build Coastguard Worker 
15*7304104dSAndroid Build Coastguard Worker      * the GNU General Public License as published by the Free
16*7304104dSAndroid Build Coastguard Worker        Software Foundation; either version 2 of the License, or (at
17*7304104dSAndroid Build Coastguard Worker        your option) any later version
18*7304104dSAndroid Build Coastguard Worker 
19*7304104dSAndroid Build Coastguard Worker    or both in parallel, as here.
20*7304104dSAndroid Build Coastguard Worker 
21*7304104dSAndroid Build Coastguard Worker    elfutils is distributed in the hope that it will be useful, but
22*7304104dSAndroid Build Coastguard Worker    WITHOUT ANY WARRANTY; without even the implied warranty of
23*7304104dSAndroid Build Coastguard Worker    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
24*7304104dSAndroid Build Coastguard Worker    General Public License for more details.
25*7304104dSAndroid Build Coastguard Worker 
26*7304104dSAndroid Build Coastguard Worker    You should have received copies of the GNU General Public License and
27*7304104dSAndroid Build Coastguard Worker    the GNU Lesser General Public License along with this program.  If
28*7304104dSAndroid Build Coastguard Worker    not, see <http://www.gnu.org/licenses/>.  */
29*7304104dSAndroid Build Coastguard Worker 
30*7304104dSAndroid Build Coastguard Worker #ifdef HAVE_CONFIG_H
31*7304104dSAndroid Build Coastguard Worker # include <config.h>
32*7304104dSAndroid Build Coastguard Worker #endif
33*7304104dSAndroid Build Coastguard Worker 
34*7304104dSAndroid Build Coastguard Worker #include <dwarf.h>
35*7304104dSAndroid Build Coastguard Worker #include <string.h>
36*7304104dSAndroid Build Coastguard Worker 
37*7304104dSAndroid Build Coastguard Worker #include "libdwP.h"
38*7304104dSAndroid Build Coastguard Worker 
39*7304104dSAndroid Build Coastguard Worker 
40*7304104dSAndroid Build Coastguard Worker size_t
41*7304104dSAndroid Build Coastguard Worker internal_function
__libdw_form_val_compute_len(struct Dwarf_CU * cu,unsigned int form,const unsigned char * valp)42*7304104dSAndroid Build Coastguard Worker __libdw_form_val_compute_len (struct Dwarf_CU *cu, unsigned int form,
43*7304104dSAndroid Build Coastguard Worker 			      const unsigned char *valp)
44*7304104dSAndroid Build Coastguard Worker {
45*7304104dSAndroid Build Coastguard Worker   const unsigned char *startp = valp;
46*7304104dSAndroid Build Coastguard Worker   const unsigned char *endp = cu->endp;
47*7304104dSAndroid Build Coastguard Worker   Dwarf_Word u128;
48*7304104dSAndroid Build Coastguard Worker   size_t result;
49*7304104dSAndroid Build Coastguard Worker 
50*7304104dSAndroid Build Coastguard Worker   /* NB: This doesn't cover constant form lengths, which are
51*7304104dSAndroid Build Coastguard Worker      already handled by the inlined __libdw_form_val_len.  */
52*7304104dSAndroid Build Coastguard Worker   switch (form)
53*7304104dSAndroid Build Coastguard Worker     {
54*7304104dSAndroid Build Coastguard Worker     case DW_FORM_addr:
55*7304104dSAndroid Build Coastguard Worker       result = cu->address_size;
56*7304104dSAndroid Build Coastguard Worker       break;
57*7304104dSAndroid Build Coastguard Worker 
58*7304104dSAndroid Build Coastguard Worker     case DW_FORM_ref_addr:
59*7304104dSAndroid Build Coastguard Worker       result = cu->version == 2 ? cu->address_size : cu->offset_size;
60*7304104dSAndroid Build Coastguard Worker       break;
61*7304104dSAndroid Build Coastguard Worker 
62*7304104dSAndroid Build Coastguard Worker     case DW_FORM_strp:
63*7304104dSAndroid Build Coastguard Worker     case DW_FORM_strp_sup:
64*7304104dSAndroid Build Coastguard Worker     case DW_FORM_line_strp:
65*7304104dSAndroid Build Coastguard Worker     case DW_FORM_sec_offset:
66*7304104dSAndroid Build Coastguard Worker     case DW_FORM_GNU_ref_alt:
67*7304104dSAndroid Build Coastguard Worker     case DW_FORM_GNU_strp_alt:
68*7304104dSAndroid Build Coastguard Worker       result = cu->offset_size;
69*7304104dSAndroid Build Coastguard Worker       break;
70*7304104dSAndroid Build Coastguard Worker 
71*7304104dSAndroid Build Coastguard Worker     case DW_FORM_block1:
72*7304104dSAndroid Build Coastguard Worker       if (unlikely ((size_t) (endp - startp) < 1))
73*7304104dSAndroid Build Coastguard Worker 	goto invalid;
74*7304104dSAndroid Build Coastguard Worker       result = *valp + 1;
75*7304104dSAndroid Build Coastguard Worker       break;
76*7304104dSAndroid Build Coastguard Worker 
77*7304104dSAndroid Build Coastguard Worker     case DW_FORM_block2:
78*7304104dSAndroid Build Coastguard Worker       if (unlikely ((size_t) (endp - startp) < 2))
79*7304104dSAndroid Build Coastguard Worker 	goto invalid;
80*7304104dSAndroid Build Coastguard Worker       result = read_2ubyte_unaligned (cu->dbg, valp) + 2;
81*7304104dSAndroid Build Coastguard Worker       break;
82*7304104dSAndroid Build Coastguard Worker 
83*7304104dSAndroid Build Coastguard Worker     case DW_FORM_block4:
84*7304104dSAndroid Build Coastguard Worker       if (unlikely ((size_t) (endp - startp) < 4))
85*7304104dSAndroid Build Coastguard Worker 	goto invalid;
86*7304104dSAndroid Build Coastguard Worker       result = read_4ubyte_unaligned (cu->dbg, valp) + 4;
87*7304104dSAndroid Build Coastguard Worker       break;
88*7304104dSAndroid Build Coastguard Worker 
89*7304104dSAndroid Build Coastguard Worker     case DW_FORM_block:
90*7304104dSAndroid Build Coastguard Worker     case DW_FORM_exprloc:
91*7304104dSAndroid Build Coastguard Worker       if (valp >= endp)
92*7304104dSAndroid Build Coastguard Worker        goto invalid;
93*7304104dSAndroid Build Coastguard Worker       get_uleb128 (u128, valp, endp);
94*7304104dSAndroid Build Coastguard Worker       result = u128 + (valp - startp);
95*7304104dSAndroid Build Coastguard Worker       break;
96*7304104dSAndroid Build Coastguard Worker 
97*7304104dSAndroid Build Coastguard Worker     case DW_FORM_string:
98*7304104dSAndroid Build Coastguard Worker       {
99*7304104dSAndroid Build Coastguard Worker 	const unsigned char *endstrp = memchr (valp, '\0',
100*7304104dSAndroid Build Coastguard Worker 					       (size_t) (endp - startp));
101*7304104dSAndroid Build Coastguard Worker 	if (unlikely (endstrp == NULL))
102*7304104dSAndroid Build Coastguard Worker 	  goto invalid;
103*7304104dSAndroid Build Coastguard Worker 	result = (size_t) (endstrp - startp) + 1;
104*7304104dSAndroid Build Coastguard Worker 	break;
105*7304104dSAndroid Build Coastguard Worker       }
106*7304104dSAndroid Build Coastguard Worker 
107*7304104dSAndroid Build Coastguard Worker     case DW_FORM_sdata:
108*7304104dSAndroid Build Coastguard Worker     case DW_FORM_udata:
109*7304104dSAndroid Build Coastguard Worker     case DW_FORM_ref_udata:
110*7304104dSAndroid Build Coastguard Worker     case DW_FORM_addrx:
111*7304104dSAndroid Build Coastguard Worker     case DW_FORM_loclistx:
112*7304104dSAndroid Build Coastguard Worker     case DW_FORM_rnglistx:
113*7304104dSAndroid Build Coastguard Worker     case DW_FORM_strx:
114*7304104dSAndroid Build Coastguard Worker     case DW_FORM_GNU_addr_index:
115*7304104dSAndroid Build Coastguard Worker     case DW_FORM_GNU_str_index:
116*7304104dSAndroid Build Coastguard Worker       if (valp >= endp)
117*7304104dSAndroid Build Coastguard Worker        goto invalid;
118*7304104dSAndroid Build Coastguard Worker       get_uleb128 (u128, valp, endp);
119*7304104dSAndroid Build Coastguard Worker       result = valp - startp;
120*7304104dSAndroid Build Coastguard Worker       break;
121*7304104dSAndroid Build Coastguard Worker 
122*7304104dSAndroid Build Coastguard Worker     case DW_FORM_indirect:
123*7304104dSAndroid Build Coastguard Worker       /* The amount of data to skip in the DIE is the size of the actual
124*7304104dSAndroid Build Coastguard Worker 	 FORM data (which is __libdw_form_val_len) plus the size of the
125*7304104dSAndroid Build Coastguard Worker 	 uleb128 encoding that FORM (which is valp - startp).  */
126*7304104dSAndroid Build Coastguard Worker       if (valp >= endp)
127*7304104dSAndroid Build Coastguard Worker 	goto invalid;
128*7304104dSAndroid Build Coastguard Worker       get_uleb128 (u128, valp, endp);
129*7304104dSAndroid Build Coastguard Worker       if (*valp == DW_FORM_indirect || *valp == DW_FORM_implicit_const)
130*7304104dSAndroid Build Coastguard Worker 	return (size_t) -1;
131*7304104dSAndroid Build Coastguard Worker       result = __libdw_form_val_len (cu, u128, valp);
132*7304104dSAndroid Build Coastguard Worker       if (result != (size_t) -1)
133*7304104dSAndroid Build Coastguard Worker 	result += valp - startp;
134*7304104dSAndroid Build Coastguard Worker       else
135*7304104dSAndroid Build Coastguard Worker         return (size_t) -1;
136*7304104dSAndroid Build Coastguard Worker       break;
137*7304104dSAndroid Build Coastguard Worker 
138*7304104dSAndroid Build Coastguard Worker     default:
139*7304104dSAndroid Build Coastguard Worker       goto invalid;
140*7304104dSAndroid Build Coastguard Worker     }
141*7304104dSAndroid Build Coastguard Worker 
142*7304104dSAndroid Build Coastguard Worker   if (unlikely (result > (size_t) (endp - startp)))
143*7304104dSAndroid Build Coastguard Worker     {
144*7304104dSAndroid Build Coastguard Worker     invalid:
145*7304104dSAndroid Build Coastguard Worker       __libdw_seterrno (DWARF_E_INVALID_DWARF);
146*7304104dSAndroid Build Coastguard Worker       result = (size_t) -1;
147*7304104dSAndroid Build Coastguard Worker     }
148*7304104dSAndroid Build Coastguard Worker 
149*7304104dSAndroid Build Coastguard Worker   return result;
150*7304104dSAndroid Build Coastguard Worker }
151