1*7304104dSAndroid Build Coastguard Worker /* Retrieves the DWARF descriptor for debugaltlink data.
2*7304104dSAndroid Build Coastguard Worker Copyright (C) 2014, 2018 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 "libdwP.h"
34*7304104dSAndroid Build Coastguard Worker #include "libelfP.h"
35*7304104dSAndroid Build Coastguard Worker #include "libdwelfP.h"
36*7304104dSAndroid Build Coastguard Worker #include "system.h"
37*7304104dSAndroid Build Coastguard Worker
38*7304104dSAndroid Build Coastguard Worker #include <inttypes.h>
39*7304104dSAndroid Build Coastguard Worker #include <fcntl.h>
40*7304104dSAndroid Build Coastguard Worker #include <limits.h>
41*7304104dSAndroid Build Coastguard Worker #include <stdlib.h>
42*7304104dSAndroid Build Coastguard Worker #include <stdio.h>
43*7304104dSAndroid Build Coastguard Worker #include <string.h>
44*7304104dSAndroid Build Coastguard Worker #include <sys/types.h>
45*7304104dSAndroid Build Coastguard Worker #include <sys/stat.h>
46*7304104dSAndroid Build Coastguard Worker
47*7304104dSAndroid Build Coastguard Worker
48*7304104dSAndroid Build Coastguard Worker char *
49*7304104dSAndroid Build Coastguard Worker internal_function
__libdw_filepath(const char * debugdir,const char * dir,const char * file)50*7304104dSAndroid Build Coastguard Worker __libdw_filepath (const char *debugdir, const char *dir, const char *file)
51*7304104dSAndroid Build Coastguard Worker {
52*7304104dSAndroid Build Coastguard Worker if (file == NULL)
53*7304104dSAndroid Build Coastguard Worker return NULL;
54*7304104dSAndroid Build Coastguard Worker
55*7304104dSAndroid Build Coastguard Worker if (file[0] == '/')
56*7304104dSAndroid Build Coastguard Worker return strdup (file);
57*7304104dSAndroid Build Coastguard Worker
58*7304104dSAndroid Build Coastguard Worker if (dir != NULL && dir[0] == '/')
59*7304104dSAndroid Build Coastguard Worker {
60*7304104dSAndroid Build Coastguard Worker size_t dirlen = strlen (dir);
61*7304104dSAndroid Build Coastguard Worker size_t filelen = strlen (file);
62*7304104dSAndroid Build Coastguard Worker size_t len = dirlen + 1 + filelen + 1;
63*7304104dSAndroid Build Coastguard Worker char *path = malloc (len);
64*7304104dSAndroid Build Coastguard Worker if (path != NULL)
65*7304104dSAndroid Build Coastguard Worker {
66*7304104dSAndroid Build Coastguard Worker char *c = mempcpy (path, dir, dirlen);
67*7304104dSAndroid Build Coastguard Worker if (dir[dirlen - 1] != '/')
68*7304104dSAndroid Build Coastguard Worker *c++ = '/';
69*7304104dSAndroid Build Coastguard Worker mempcpy (c, file, filelen + 1);
70*7304104dSAndroid Build Coastguard Worker }
71*7304104dSAndroid Build Coastguard Worker return path;
72*7304104dSAndroid Build Coastguard Worker }
73*7304104dSAndroid Build Coastguard Worker
74*7304104dSAndroid Build Coastguard Worker if (debugdir != NULL)
75*7304104dSAndroid Build Coastguard Worker {
76*7304104dSAndroid Build Coastguard Worker size_t debugdirlen = strlen (debugdir);
77*7304104dSAndroid Build Coastguard Worker size_t dirlen = dir != NULL ? strlen (dir) : 0;
78*7304104dSAndroid Build Coastguard Worker size_t filelen = strlen (file);
79*7304104dSAndroid Build Coastguard Worker size_t len = debugdirlen + 1 + dirlen + 1 + filelen + 1;
80*7304104dSAndroid Build Coastguard Worker char *path = malloc (len);
81*7304104dSAndroid Build Coastguard Worker if (path != NULL)
82*7304104dSAndroid Build Coastguard Worker {
83*7304104dSAndroid Build Coastguard Worker char *c = mempcpy (path, debugdir, debugdirlen);
84*7304104dSAndroid Build Coastguard Worker if (dirlen > 0)
85*7304104dSAndroid Build Coastguard Worker {
86*7304104dSAndroid Build Coastguard Worker c = mempcpy (c, dir, dirlen);
87*7304104dSAndroid Build Coastguard Worker if (dir[dirlen - 1] != '/')
88*7304104dSAndroid Build Coastguard Worker *c++ = '/';
89*7304104dSAndroid Build Coastguard Worker }
90*7304104dSAndroid Build Coastguard Worker mempcpy (c, file, filelen + 1);
91*7304104dSAndroid Build Coastguard Worker return path;
92*7304104dSAndroid Build Coastguard Worker }
93*7304104dSAndroid Build Coastguard Worker }
94*7304104dSAndroid Build Coastguard Worker
95*7304104dSAndroid Build Coastguard Worker return NULL;
96*7304104dSAndroid Build Coastguard Worker }
97*7304104dSAndroid Build Coastguard Worker
98*7304104dSAndroid Build Coastguard Worker static void
find_debug_altlink(Dwarf * dbg)99*7304104dSAndroid Build Coastguard Worker find_debug_altlink (Dwarf *dbg)
100*7304104dSAndroid Build Coastguard Worker {
101*7304104dSAndroid Build Coastguard Worker const char *altname;
102*7304104dSAndroid Build Coastguard Worker const void *build_id;
103*7304104dSAndroid Build Coastguard Worker ssize_t build_id_len = INTUSE(dwelf_dwarf_gnu_debugaltlink) (dbg,
104*7304104dSAndroid Build Coastguard Worker &altname,
105*7304104dSAndroid Build Coastguard Worker &build_id);
106*7304104dSAndroid Build Coastguard Worker
107*7304104dSAndroid Build Coastguard Worker /* Couldn't even get the debugaltlink. It probably doesn't exist. */
108*7304104dSAndroid Build Coastguard Worker if (build_id_len <= 0)
109*7304104dSAndroid Build Coastguard Worker return;
110*7304104dSAndroid Build Coastguard Worker
111*7304104dSAndroid Build Coastguard Worker const uint8_t *id = (const uint8_t *) build_id;
112*7304104dSAndroid Build Coastguard Worker size_t id_len = build_id_len;
113*7304104dSAndroid Build Coastguard Worker int fd = -1;
114*7304104dSAndroid Build Coastguard Worker
115*7304104dSAndroid Build Coastguard Worker /* We only look in the standard path. And relative to the dbg file. */
116*7304104dSAndroid Build Coastguard Worker #define DEBUGINFO_PATH "/usr/lib/debug"
117*7304104dSAndroid Build Coastguard Worker
118*7304104dSAndroid Build Coastguard Worker /* We don't handle very short or really large build-ids. We need at
119*7304104dSAndroid Build Coastguard Worker at least 3 and allow for up to 64 (normally ids are 20 long). */
120*7304104dSAndroid Build Coastguard Worker #define MIN_BUILD_ID_BYTES 3
121*7304104dSAndroid Build Coastguard Worker #define MAX_BUILD_ID_BYTES 64
122*7304104dSAndroid Build Coastguard Worker if (id_len >= MIN_BUILD_ID_BYTES && id_len <= MAX_BUILD_ID_BYTES)
123*7304104dSAndroid Build Coastguard Worker {
124*7304104dSAndroid Build Coastguard Worker /* Note sizeof a string literal includes the trailing zero. */
125*7304104dSAndroid Build Coastguard Worker char id_path[sizeof DEBUGINFO_PATH - 1 + sizeof "/.build-id/" - 1
126*7304104dSAndroid Build Coastguard Worker + 2 + 1 + (MAX_BUILD_ID_BYTES - 1) * 2 + sizeof ".debug"];
127*7304104dSAndroid Build Coastguard Worker sprintf (&id_path[0], "%s%s", DEBUGINFO_PATH, "/.build-id/");
128*7304104dSAndroid Build Coastguard Worker sprintf (&id_path[sizeof DEBUGINFO_PATH - 1 + sizeof "/.build-id/" - 1],
129*7304104dSAndroid Build Coastguard Worker "%02" PRIx8 "/", (uint8_t) id[0]);
130*7304104dSAndroid Build Coastguard Worker for (size_t i = 1; i < id_len; ++i)
131*7304104dSAndroid Build Coastguard Worker sprintf (&id_path[sizeof DEBUGINFO_PATH - 1 + sizeof "/.build-id/" - 1
132*7304104dSAndroid Build Coastguard Worker + 3 + (i - 1) * 2], "%02" PRIx8, (uint8_t) id[i]);
133*7304104dSAndroid Build Coastguard Worker strcpy (&id_path[sizeof DEBUGINFO_PATH - 1 + sizeof "/.build-id/" - 1
134*7304104dSAndroid Build Coastguard Worker + 3 + (id_len - 1) * 2], ".debug");
135*7304104dSAndroid Build Coastguard Worker
136*7304104dSAndroid Build Coastguard Worker fd = TEMP_FAILURE_RETRY (open (id_path, O_RDONLY));
137*7304104dSAndroid Build Coastguard Worker }
138*7304104dSAndroid Build Coastguard Worker
139*7304104dSAndroid Build Coastguard Worker /* Fall back on (possible relative) alt file path. */
140*7304104dSAndroid Build Coastguard Worker if (fd < 0)
141*7304104dSAndroid Build Coastguard Worker {
142*7304104dSAndroid Build Coastguard Worker char *altpath = __libdw_filepath (dbg->debugdir, NULL, altname);
143*7304104dSAndroid Build Coastguard Worker if (altpath != NULL)
144*7304104dSAndroid Build Coastguard Worker {
145*7304104dSAndroid Build Coastguard Worker fd = TEMP_FAILURE_RETRY (open (altpath, O_RDONLY));
146*7304104dSAndroid Build Coastguard Worker free (altpath);
147*7304104dSAndroid Build Coastguard Worker }
148*7304104dSAndroid Build Coastguard Worker }
149*7304104dSAndroid Build Coastguard Worker
150*7304104dSAndroid Build Coastguard Worker if (fd >= 0)
151*7304104dSAndroid Build Coastguard Worker {
152*7304104dSAndroid Build Coastguard Worker Dwarf *alt = dwarf_begin (fd, O_RDONLY);
153*7304104dSAndroid Build Coastguard Worker if (alt != NULL)
154*7304104dSAndroid Build Coastguard Worker {
155*7304104dSAndroid Build Coastguard Worker dbg->alt_dwarf = alt;
156*7304104dSAndroid Build Coastguard Worker dbg->alt_fd = fd;
157*7304104dSAndroid Build Coastguard Worker }
158*7304104dSAndroid Build Coastguard Worker else
159*7304104dSAndroid Build Coastguard Worker close (fd);
160*7304104dSAndroid Build Coastguard Worker }
161*7304104dSAndroid Build Coastguard Worker }
162*7304104dSAndroid Build Coastguard Worker
163*7304104dSAndroid Build Coastguard Worker Dwarf *
dwarf_getalt(Dwarf * main)164*7304104dSAndroid Build Coastguard Worker dwarf_getalt (Dwarf *main)
165*7304104dSAndroid Build Coastguard Worker {
166*7304104dSAndroid Build Coastguard Worker /* Only try once. */
167*7304104dSAndroid Build Coastguard Worker if (main == NULL || main->alt_dwarf == (void *) -1)
168*7304104dSAndroid Build Coastguard Worker return NULL;
169*7304104dSAndroid Build Coastguard Worker
170*7304104dSAndroid Build Coastguard Worker if (main->alt_dwarf != NULL)
171*7304104dSAndroid Build Coastguard Worker return main->alt_dwarf;
172*7304104dSAndroid Build Coastguard Worker
173*7304104dSAndroid Build Coastguard Worker find_debug_altlink (main);
174*7304104dSAndroid Build Coastguard Worker
175*7304104dSAndroid Build Coastguard Worker /* If we found nothing, make sure we don't try again. */
176*7304104dSAndroid Build Coastguard Worker if (main->alt_dwarf == NULL)
177*7304104dSAndroid Build Coastguard Worker {
178*7304104dSAndroid Build Coastguard Worker main->alt_dwarf = (void *) -1;
179*7304104dSAndroid Build Coastguard Worker return NULL;
180*7304104dSAndroid Build Coastguard Worker }
181*7304104dSAndroid Build Coastguard Worker
182*7304104dSAndroid Build Coastguard Worker return main->alt_dwarf;
183*7304104dSAndroid Build Coastguard Worker }
184*7304104dSAndroid Build Coastguard Worker INTDEF (dwarf_getalt)
185