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