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