xref: /dflybsd-src/contrib/gcc-8.0/libbacktrace/fileline.c (revision 38fd149817dfbff97799f62fcb70be98c4e32523)
1*38fd1498Szrj /* fileline.c -- Get file and line number information in a backtrace.
2*38fd1498Szrj    Copyright (C) 2012-2018 Free Software Foundation, Inc.
3*38fd1498Szrj    Written by Ian Lance Taylor, Google.
4*38fd1498Szrj 
5*38fd1498Szrj Redistribution and use in source and binary forms, with or without
6*38fd1498Szrj modification, are permitted provided that the following conditions are
7*38fd1498Szrj met:
8*38fd1498Szrj 
9*38fd1498Szrj     (1) Redistributions of source code must retain the above copyright
10*38fd1498Szrj     notice, this list of conditions and the following disclaimer.
11*38fd1498Szrj 
12*38fd1498Szrj     (2) Redistributions in binary form must reproduce the above copyright
13*38fd1498Szrj     notice, this list of conditions and the following disclaimer in
14*38fd1498Szrj     the documentation and/or other materials provided with the
15*38fd1498Szrj     distribution.
16*38fd1498Szrj 
17*38fd1498Szrj     (3) The name of the author may not be used to
18*38fd1498Szrj     endorse or promote products derived from this software without
19*38fd1498Szrj     specific prior written permission.
20*38fd1498Szrj 
21*38fd1498Szrj THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
22*38fd1498Szrj IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
23*38fd1498Szrj WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
24*38fd1498Szrj DISCLAIMED. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT,
25*38fd1498Szrj INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
26*38fd1498Szrj (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
27*38fd1498Szrj SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
28*38fd1498Szrj HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
29*38fd1498Szrj STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
30*38fd1498Szrj IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
31*38fd1498Szrj POSSIBILITY OF SUCH DAMAGE.  */
32*38fd1498Szrj 
33*38fd1498Szrj #include "config.h"
34*38fd1498Szrj 
35*38fd1498Szrj #include <sys/types.h>
36*38fd1498Szrj #include <sys/stat.h>
37*38fd1498Szrj #include <errno.h>
38*38fd1498Szrj #include <fcntl.h>
39*38fd1498Szrj #include <stdlib.h>
40*38fd1498Szrj #include <unistd.h>
41*38fd1498Szrj 
42*38fd1498Szrj #include "backtrace.h"
43*38fd1498Szrj #include "internal.h"
44*38fd1498Szrj 
45*38fd1498Szrj #ifndef HAVE_GETEXECNAME
46*38fd1498Szrj #define getexecname() NULL
47*38fd1498Szrj #endif
48*38fd1498Szrj 
49*38fd1498Szrj /* Initialize the fileline information from the executable.  Returns 1
50*38fd1498Szrj    on success, 0 on failure.  */
51*38fd1498Szrj 
52*38fd1498Szrj static int
fileline_initialize(struct backtrace_state * state,backtrace_error_callback error_callback,void * data)53*38fd1498Szrj fileline_initialize (struct backtrace_state *state,
54*38fd1498Szrj 		     backtrace_error_callback error_callback, void *data)
55*38fd1498Szrj {
56*38fd1498Szrj   int failed;
57*38fd1498Szrj   fileline fileline_fn;
58*38fd1498Szrj   int pass;
59*38fd1498Szrj   int called_error_callback;
60*38fd1498Szrj   int descriptor;
61*38fd1498Szrj   const char *filename;
62*38fd1498Szrj   char buf[64];
63*38fd1498Szrj 
64*38fd1498Szrj   if (!state->threaded)
65*38fd1498Szrj     failed = state->fileline_initialization_failed;
66*38fd1498Szrj   else
67*38fd1498Szrj     failed = backtrace_atomic_load_int (&state->fileline_initialization_failed);
68*38fd1498Szrj 
69*38fd1498Szrj   if (failed)
70*38fd1498Szrj     {
71*38fd1498Szrj       error_callback (data, "failed to read executable information", -1);
72*38fd1498Szrj       return 0;
73*38fd1498Szrj     }
74*38fd1498Szrj 
75*38fd1498Szrj   if (!state->threaded)
76*38fd1498Szrj     fileline_fn = state->fileline_fn;
77*38fd1498Szrj   else
78*38fd1498Szrj     fileline_fn = backtrace_atomic_load_pointer (&state->fileline_fn);
79*38fd1498Szrj   if (fileline_fn != NULL)
80*38fd1498Szrj     return 1;
81*38fd1498Szrj 
82*38fd1498Szrj   /* We have not initialized the information.  Do it now.  */
83*38fd1498Szrj 
84*38fd1498Szrj   descriptor = -1;
85*38fd1498Szrj   called_error_callback = 0;
86*38fd1498Szrj   for (pass = 0; pass < 5; ++pass)
87*38fd1498Szrj     {
88*38fd1498Szrj       int does_not_exist;
89*38fd1498Szrj 
90*38fd1498Szrj       switch (pass)
91*38fd1498Szrj 	{
92*38fd1498Szrj 	case 0:
93*38fd1498Szrj 	  filename = state->filename;
94*38fd1498Szrj 	  break;
95*38fd1498Szrj 	case 1:
96*38fd1498Szrj 	  filename = getexecname ();
97*38fd1498Szrj 	  break;
98*38fd1498Szrj 	case 2:
99*38fd1498Szrj 	  filename = "/proc/self/exe";
100*38fd1498Szrj 	  break;
101*38fd1498Szrj 	case 3:
102*38fd1498Szrj 	  filename = "/proc/curproc/file";
103*38fd1498Szrj 	  break;
104*38fd1498Szrj 	case 4:
105*38fd1498Szrj 	  snprintf (buf, sizeof (buf), "/proc/%ld/object/a.out",
106*38fd1498Szrj 		    (long) getpid ());
107*38fd1498Szrj 	  filename = buf;
108*38fd1498Szrj 	  break;
109*38fd1498Szrj 	default:
110*38fd1498Szrj 	  abort ();
111*38fd1498Szrj 	}
112*38fd1498Szrj 
113*38fd1498Szrj       if (filename == NULL)
114*38fd1498Szrj 	continue;
115*38fd1498Szrj 
116*38fd1498Szrj       descriptor = backtrace_open (filename, error_callback, data,
117*38fd1498Szrj 				   &does_not_exist);
118*38fd1498Szrj       if (descriptor < 0 && !does_not_exist)
119*38fd1498Szrj 	{
120*38fd1498Szrj 	  called_error_callback = 1;
121*38fd1498Szrj 	  break;
122*38fd1498Szrj 	}
123*38fd1498Szrj       if (descriptor >= 0)
124*38fd1498Szrj 	break;
125*38fd1498Szrj     }
126*38fd1498Szrj 
127*38fd1498Szrj   if (descriptor < 0)
128*38fd1498Szrj     {
129*38fd1498Szrj       if (!called_error_callback)
130*38fd1498Szrj 	{
131*38fd1498Szrj 	  if (state->filename != NULL)
132*38fd1498Szrj 	    error_callback (data, state->filename, ENOENT);
133*38fd1498Szrj 	  else
134*38fd1498Szrj 	    error_callback (data,
135*38fd1498Szrj 			    "libbacktrace could not find executable to open",
136*38fd1498Szrj 			    0);
137*38fd1498Szrj 	}
138*38fd1498Szrj       failed = 1;
139*38fd1498Szrj     }
140*38fd1498Szrj 
141*38fd1498Szrj   if (!failed)
142*38fd1498Szrj     {
143*38fd1498Szrj       if (!backtrace_initialize (state, filename, descriptor, error_callback,
144*38fd1498Szrj 				 data, &fileline_fn))
145*38fd1498Szrj 	failed = 1;
146*38fd1498Szrj     }
147*38fd1498Szrj 
148*38fd1498Szrj   if (failed)
149*38fd1498Szrj     {
150*38fd1498Szrj       if (!state->threaded)
151*38fd1498Szrj 	state->fileline_initialization_failed = 1;
152*38fd1498Szrj       else
153*38fd1498Szrj 	backtrace_atomic_store_int (&state->fileline_initialization_failed, 1);
154*38fd1498Szrj       return 0;
155*38fd1498Szrj     }
156*38fd1498Szrj 
157*38fd1498Szrj   if (!state->threaded)
158*38fd1498Szrj     state->fileline_fn = fileline_fn;
159*38fd1498Szrj   else
160*38fd1498Szrj     {
161*38fd1498Szrj       backtrace_atomic_store_pointer (&state->fileline_fn, fileline_fn);
162*38fd1498Szrj 
163*38fd1498Szrj       /* Note that if two threads initialize at once, one of the data
164*38fd1498Szrj 	 sets may be leaked.  */
165*38fd1498Szrj     }
166*38fd1498Szrj 
167*38fd1498Szrj   return 1;
168*38fd1498Szrj }
169*38fd1498Szrj 
170*38fd1498Szrj /* Given a PC, find the file name, line number, and function name.  */
171*38fd1498Szrj 
172*38fd1498Szrj int
backtrace_pcinfo(struct backtrace_state * state,uintptr_t pc,backtrace_full_callback callback,backtrace_error_callback error_callback,void * data)173*38fd1498Szrj backtrace_pcinfo (struct backtrace_state *state, uintptr_t pc,
174*38fd1498Szrj 		  backtrace_full_callback callback,
175*38fd1498Szrj 		  backtrace_error_callback error_callback, void *data)
176*38fd1498Szrj {
177*38fd1498Szrj   if (!fileline_initialize (state, error_callback, data))
178*38fd1498Szrj     return 0;
179*38fd1498Szrj 
180*38fd1498Szrj   if (state->fileline_initialization_failed)
181*38fd1498Szrj     return 0;
182*38fd1498Szrj 
183*38fd1498Szrj   return state->fileline_fn (state, pc, callback, error_callback, data);
184*38fd1498Szrj }
185*38fd1498Szrj 
186*38fd1498Szrj /* Given a PC, find the symbol for it, and its value.  */
187*38fd1498Szrj 
188*38fd1498Szrj int
backtrace_syminfo(struct backtrace_state * state,uintptr_t pc,backtrace_syminfo_callback callback,backtrace_error_callback error_callback,void * data)189*38fd1498Szrj backtrace_syminfo (struct backtrace_state *state, uintptr_t pc,
190*38fd1498Szrj 		   backtrace_syminfo_callback callback,
191*38fd1498Szrj 		   backtrace_error_callback error_callback, void *data)
192*38fd1498Szrj {
193*38fd1498Szrj   if (!fileline_initialize (state, error_callback, data))
194*38fd1498Szrj     return 0;
195*38fd1498Szrj 
196*38fd1498Szrj   if (state->fileline_initialization_failed)
197*38fd1498Szrj     return 0;
198*38fd1498Szrj 
199*38fd1498Szrj   state->syminfo_fn (state, pc, callback, error_callback, data);
200*38fd1498Szrj   return 1;
201*38fd1498Szrj }
202