OpenCores
URL https://opencores.org/ocsvn/openrisc/openrisc/trunk

Subversion Repositories openrisc

[/] [openrisc/] [trunk/] [gnu-dev/] [or1k-gcc/] [gcc/] [ada/] [vxaddr2line.adb] - Blame information for rev 774

Go to most recent revision | Details | Compare with Previous | View Log

Line No. Rev Author Line
1 706 jeremybenn
------------------------------------------------------------------------------
2
--                                                                          --
3
--                         GNAT COMPILER COMPONENTS                         --
4
--                                                                          --
5
--                           V X A D D R 2 L I N E                          --
6
--                                                                          --
7
--                                 B o d y                                  --
8
--                                                                          --
9
--                     Copyright (C) 2002-2011, AdaCore                     --
10
--                                                                          --
11
-- GNAT is free software;  you can  redistribute it  and/or modify it under --
12
-- terms of the  GNU General Public License as published  by the Free Soft- --
13
-- ware  Foundation;  either version 3,  or (at your option) any later ver- --
14
-- sion.  GNAT is distributed in the hope that it will be useful, but WITH- --
15
-- OUT ANY WARRANTY;  without even the  implied warranty of MERCHANTABILITY --
16
-- or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License --
17
-- for  more details.  You should have  received  a copy of the GNU General --
18
-- Public License  distributed with GNAT; see file COPYING3.  If not, go to --
19
-- http://www.gnu.org/licenses for a complete copy of the license.          --
20
--                                                                          --
21
-- GNAT was originally developed  by the GNAT team at  New York University. --
22
-- Extensive contributions were provided by Ada Core Technologies Inc.      --
23
--                                                                          --
24
------------------------------------------------------------------------------
25
 
26
--  This program is meant to be used with vxworks to compute symbolic
27
--  backtraces on the host from non-symbolic backtraces obtained on the target.
28
 
29
--  The basic idea is to automate the computation of the necessary address
30
--  adjustments prior to calling addr2line when the application has only been
31
--  partially linked on the host.
32
 
33
--  Variants for various targets are supported, and the command line should
34
--  be like :
35
 
36
--  <target>-addr2line [-a <target_arch>] <exe_file> <ref_address>
37
--                     <backtrace addresses>
38
 
39
--  Where:
40
--  <target_arch> :
41
--    selects the target architecture. In the absence of this parameter the
42
--    default variant is chosen based on the Detect_Arch result. Generally,
43
--    this parameter will only be used if vxaddr2line is recompiled manually.
44
--    Otherwise, the command name will always be of the form
45
--    <target>-vxaddr2line where there is no ambiguity on the target's
46
--    architecture.
47
 
48
--  <exe_file> :
49
--    The name of the partially linked binary file for the application.
50
 
51
--  <ref_address> :
52
--    Runtime address (on the target) of a reference symbol you choose,
53
--    which name shall match the value of the Ref_Symbol variable declared
54
--    below. A symbol with a small offset from the beginning of the text
55
--    segment is better, so "adainit" is a good choice.
56
 
57
--  <backtrace addresses> :
58
--    The call chain addresses you obtained at run time on the target and
59
--    for which you want a symbolic association.
60
 
61
--  TO ADD A NEW ARCHITECTURE add an appropriate value to Architecture type
62
--  (in a format <host>_<target>), and then an appropriate value to Config_List
63
--  array
64
 
65
with Ada.Text_IO;       use Ada.Text_IO;
66
with Ada.Command_Line;  use Ada.Command_Line;
67
with Ada.Strings.Fixed; use Ada.Strings.Fixed;
68
with Interfaces;        use Interfaces;
69
 
70
with GNAT.OS_Lib;               use GNAT.OS_Lib;
71
with GNAT.Directory_Operations; use GNAT.Directory_Operations;
72
with GNAT.Expect;               use GNAT.Expect;
73
with GNAT.Regpat;               use GNAT.Regpat;
74
 
75
procedure VxAddr2Line is
76
 
77
   package Unsigned_32_IO is new Modular_IO (Unsigned_32);
78
   --  Instantiate Modular_IO to have Put
79
 
80
   Ref_Symbol : constant String := "adainit";
81
   --  This is the name of the reference symbol which runtime address shall
82
   --  be provided as the <ref_address> argument.
83
 
84
   --  All supported architectures
85
   type Architecture is
86
     (DEC_ALPHA,
87
      LINUX_E500V2,
88
      LINUX_I586,
89
      LINUX_POWERPC,
90
      WINDOWS_E500V2,
91
      WINDOWS_I586,
92
      WINDOWS_M68K,
93
      WINDOWS_POWERPC,
94
      SOLARIS_E500V2,
95
      SOLARIS_I586,
96
      SOLARIS_POWERPC);
97
 
98
   type Arch_Record is record
99
      Addr2line_Binary : String_Access;
100
      --  Name of the addr2line utility to use
101
 
102
      Nm_Binary : String_Access;
103
      --  Name of the host nm utility, which will be used to find out the
104
      --  offset of the reference symbol in the text segment of the partially
105
      --  linked executable.
106
 
107
      Addr_Digits_To_Skip : Integer;
108
      --  When addresses such as 0xfffffc0001dfed50 are provided, for instance
109
      --  on ALPHA, indicate the number of leading digits that can be ignored,
110
      --  which will avoid computational overflows. Typically only useful when
111
      --  64bit addresses are provided.
112
 
113
      Bt_Offset_From_Call : Unsigned_32;
114
      --  Offset from a backtrace address to the address of the corresponding
115
      --  call instruction. This should always be 0, except on platforms where
116
      --  the backtrace addresses actually correspond to return and not call
117
      --  points. In such cases, a negative value is most likely.
118
   end record;
119
 
120
   --  Configuration for each of the architectures
121
   Arch_List : array (Architecture'Range) of Arch_Record :=
122
     (DEC_ALPHA =>
123
        (Addr2line_Binary    => null,
124
         Nm_Binary           => null,
125
         Addr_Digits_To_Skip => 8,
126
         Bt_Offset_From_Call => 0),
127
      LINUX_E500V2 =>
128
        (Addr2line_Binary    => null,
129
         Nm_Binary           => null,
130
         Addr_Digits_To_Skip => 0,
131
         Bt_Offset_From_Call => -4),
132
      LINUX_I586 =>
133
        (Addr2line_Binary    => null,
134
         Nm_Binary           => null,
135
         Addr_Digits_To_Skip => 0,
136
         Bt_Offset_From_Call => -2),
137
      LINUX_POWERPC =>
138
        (Addr2line_Binary    => null,
139
         Nm_Binary           => null,
140
         Addr_Digits_To_Skip => 0,
141
         Bt_Offset_From_Call => -4),
142
      SOLARIS_E500V2 =>
143
        (Addr2line_Binary    => null,
144
         Nm_Binary           => null,
145
         Addr_Digits_To_Skip => 0,
146
         Bt_Offset_From_Call => -4),
147
      SOLARIS_I586 =>
148
        (Addr2line_Binary    => null,
149
         Nm_Binary           => null,
150
         Addr_Digits_To_Skip => 0,
151
         Bt_Offset_From_Call => -2),
152
      SOLARIS_POWERPC =>
153
        (Addr2line_Binary    => null,
154
         Nm_Binary           => null,
155
         Addr_Digits_To_Skip => 0,
156
         Bt_Offset_From_Call => -4),
157
      WINDOWS_E500V2 =>
158
        (Addr2line_Binary    => null,
159
         Nm_Binary           => null,
160
         Addr_Digits_To_Skip => 0,
161
         Bt_Offset_From_Call => -4),
162
      WINDOWS_I586 =>
163
        (Addr2line_Binary    => null,
164
         Nm_Binary           => null,
165
         Addr_Digits_To_Skip => 0,
166
         Bt_Offset_From_Call => -2),
167
      WINDOWS_M68K =>
168
        (Addr2line_Binary    => null,
169
         Nm_Binary           => null,
170
         Addr_Digits_To_Skip => 0,
171
         Bt_Offset_From_Call => -4),
172
      WINDOWS_POWERPC =>
173
        (Addr2line_Binary    => null,
174
         Nm_Binary           => null,
175
         Addr_Digits_To_Skip => 0,
176
         Bt_Offset_From_Call => -4)
177
     );
178
 
179
   --  Current architecture
180
   Cur_Arch : Architecture;
181
 
182
   --  State of architecture detection
183
   Detect_Success : Boolean := False;
184
 
185
   -----------------------
186
   -- Local subprograms --
187
   -----------------------
188
 
189
   procedure Error (Msg : String);
190
   pragma No_Return (Error);
191
   --  Prints the message and then terminates the program
192
 
193
   procedure Usage;
194
   --  Displays the short help message and then terminates the program
195
 
196
   function Get_Reference_Offset return Unsigned_32;
197
   --  Computes the static offset of the reference symbol by calling nm
198
 
199
   function Get_Value_From_Hex_Arg (Arg : Natural) return Unsigned_32;
200
   --  Threats the argument number Arg as a C-style hexadecimal literal
201
   --  and returns its integer value
202
 
203
   function Hex_Image (Value : Unsigned_32) return String_Access;
204
   --  Returns access to a string that contains hexadecimal image of Value
205
 
206
   --  Separate functions that provide build-time customization:
207
 
208
   procedure Detect_Arch;
209
   --  Saves in Cur_Arch the current architecture, based on the name of
210
   --  vxaddr2line instance and properties of the host. Detect_Success is False
211
   --  if detection fails
212
 
213
   -----------------
214
   -- Detect_Arch --
215
   -----------------
216
 
217
   procedure Detect_Arch is
218
      Name   : constant String := Base_Name (Command_Name);
219
      Proc   : constant String :=
220
                 Name (Name'First .. Index (Name, "-") - 1);
221
      Target : constant String :=
222
                 Name (Name'First .. Index (Name, "vxaddr2line") - 1);
223
 
224
   begin
225
      Detect_Success := False;
226
 
227
      if Proc = "" then
228
         return;
229
      end if;
230
 
231
      if Proc = "alpha" then
232
         Cur_Arch := DEC_ALPHA;
233
      else
234
         --  Let's detect the host.
235
         --  ??? A naive implementation that can't distinguish between Unixes
236
         if Directory_Separator = '/' then
237
            Cur_Arch := Architecture'Value ("solaris_" & Proc);
238
         else
239
            Cur_Arch := Architecture'Value ("windows_" & Proc);
240
         end if;
241
      end if;
242
 
243
      if Arch_List (Cur_Arch).Addr2line_Binary = null then
244
         Arch_List (Cur_Arch).Addr2line_Binary := new String'
245
           (Target & "addr2line");
246
      end if;
247
      if Arch_List (Cur_Arch).Nm_Binary = null then
248
         Arch_List (Cur_Arch).Nm_Binary := new String'
249
           (Target & "nm");
250
      end if;
251
 
252
      Detect_Success := True;
253
 
254
   exception
255
      when others =>
256
         return;
257
   end Detect_Arch;
258
 
259
   -----------
260
   -- Error --
261
   -----------
262
 
263
   procedure Error (Msg : String) is
264
   begin
265
      Put_Line (Msg);
266
      OS_Exit (1);
267
      raise Program_Error;
268
   end Error;
269
 
270
   --------------------------
271
   -- Get_Reference_Offset --
272
   --------------------------
273
 
274
   function Get_Reference_Offset return Unsigned_32 is
275
      Nm_Cmd  : constant String_Access :=
276
                  Locate_Exec_On_Path (Arch_List (Cur_Arch).Nm_Binary.all);
277
 
278
      Nm_Args : constant Argument_List :=
279
                  (new String'("-P"),
280
                   new String'(Argument (1)));
281
 
282
      Forever   : aliased String := "^@@@@";
283
      Reference : aliased String := Ref_Symbol & "\s+\S\s+([\da-fA-F]+)";
284
 
285
      Pd     : Process_Descriptor;
286
      Result : Expect_Match;
287
 
288
   begin
289
      --  If Nm is not found, abort
290
 
291
      if Nm_Cmd = null then
292
         Error ("Couldn't find " & Arch_List (Cur_Arch).Nm_Binary.all);
293
      end if;
294
 
295
      Non_Blocking_Spawn
296
        (Pd, Nm_Cmd.all, Nm_Args, Buffer_Size => 0, Err_To_Out => True);
297
 
298
      --  Expect a string containing the reference symbol
299
 
300
      Expect (Pd, Result,
301
              Regexp_Array'(1 => Reference'Unchecked_Access),
302
              Timeout => -1);
303
 
304
      --  If we are here, the pattern was matched successfully
305
 
306
      declare
307
         Match_String : constant String := Expect_Out_Match (Pd);
308
         Matches      : Match_Array (0 .. 1);
309
         Value        : Unsigned_32;
310
 
311
      begin
312
         Match (Reference, Match_String, Matches);
313
         Value := Unsigned_32'Value
314
           ("16#"
315
            & Match_String (Matches (1).First .. Matches (1).Last) & "#");
316
 
317
         --  Expect a string that will never be emitted, so that the
318
         --  process can be correctly terminated (with Process_Died)
319
 
320
         Expect (Pd, Result,
321
                 Regexp_Array'(1 => Forever'Unchecked_Access),
322
                 Timeout => -1);
323
 
324
      exception
325
         when Process_Died =>
326
            return Value;
327
      end;
328
 
329
      --  We cannot get here
330
 
331
      raise Program_Error;
332
 
333
   exception
334
      when Invalid_Process =>
335
         Error ("Could not spawn a process " & Nm_Cmd.all);
336
 
337
      when others    =>
338
 
339
         --  The process died without matching the reference symbol or the
340
         --  format wasn't recognized.
341
 
342
         Error ("Unexpected output from " & Nm_Cmd.all);
343
   end Get_Reference_Offset;
344
 
345
   ----------------------------
346
   -- Get_Value_From_Hex_Arg --
347
   ----------------------------
348
 
349
   function Get_Value_From_Hex_Arg (Arg : Natural) return Unsigned_32 is
350
      Cur_Arg : constant String := Argument (Arg);
351
      Offset  : Natural;
352
 
353
   begin
354
      --  Skip "0x" prefix if present
355
 
356
      if Cur_Arg'Length > 2 and then Cur_Arg (1 .. 2) = "0x" then
357
         Offset := 3;
358
      else
359
         Offset := 1;
360
      end if;
361
 
362
      --  Add architecture-specific offset
363
 
364
      Offset := Offset + Arch_List (Cur_Arch).Addr_Digits_To_Skip;
365
 
366
      --  Convert to value
367
 
368
      return Unsigned_32'Value
369
        ("16#" & Cur_Arg (Offset .. Cur_Arg'Last) & "#");
370
 
371
   exception
372
      when Constraint_Error =>
373
 
374
         Error ("Can't parse backtrace address '" & Cur_Arg & "'");
375
         raise;
376
   end Get_Value_From_Hex_Arg;
377
 
378
   ---------------
379
   -- Hex_Image --
380
   ---------------
381
 
382
   function Hex_Image (Value : Unsigned_32) return String_Access is
383
      Result    : String (1 .. 20);
384
      Start_Pos : Natural;
385
 
386
   begin
387
      Unsigned_32_IO.Put (Result, Value, 16);
388
      Start_Pos := Index (Result, "16#") + 3;
389
      return new String'(Result (Start_Pos .. Result'Last - 1));
390
   end Hex_Image;
391
 
392
   -----------
393
   -- Usage --
394
   -----------
395
 
396
   procedure Usage is
397
   begin
398
      Put_Line ("Usage : " & Base_Name (Command_Name)
399
                & " <executable> <"
400
                & Ref_Symbol & " offset on target> <addr1> ...");
401
 
402
      OS_Exit (1);
403
   end Usage;
404
 
405
   Ref_Static_Offset, Ref_Runtime_Address, Bt_Address : Unsigned_32;
406
 
407
   Addr2line_Cmd : String_Access;
408
 
409
   Addr2line_Args : Argument_List (1 .. 501);
410
   --  We expect that there won't be more than 500 backtrace frames
411
 
412
   Addr2line_Args_Count : Natural;
413
 
414
   Success : Boolean;
415
 
416
--  Start of processing for VxAddr2Line
417
 
418
begin
419
 
420
   Detect_Arch;
421
 
422
   --  There should be at least two arguments
423
 
424
   if Argument_Count < 2 then
425
      Usage;
426
   end if;
427
 
428
   --  ??? HARD LIMIT! There should be at most 501 arguments
429
 
430
   if Argument_Count > 501 then
431
      Error ("Too many backtrace frames");
432
   end if;
433
 
434
   --  Do we have a valid architecture?
435
 
436
   if not Detect_Success then
437
      Put_Line ("Couldn't detect the architecture");
438
      return;
439
   end if;
440
 
441
   Addr2line_Cmd :=
442
     Locate_Exec_On_Path (Arch_List (Cur_Arch).Addr2line_Binary.all);
443
 
444
   --  If Addr2line is not found, abort
445
 
446
   if Addr2line_Cmd = null then
447
      Error ("Couldn't find " & Arch_List (Cur_Arch).Addr2line_Binary.all);
448
   end if;
449
 
450
   --  The first argument specifies the image file. Check if it exists
451
 
452
   if not Is_Regular_File (Argument (1)) then
453
      Error ("Couldn't find the executable " & Argument (1));
454
   end if;
455
 
456
   --  The second argument specifies the reference symbol runtime address.
457
   --  Let's parse and store it
458
 
459
   Ref_Runtime_Address := Get_Value_From_Hex_Arg (2);
460
 
461
   --  Run nm command to get the reference symbol static offset
462
 
463
   Ref_Static_Offset := Get_Reference_Offset;
464
 
465
   --  Build addr2line parameters. First, the standard part
466
 
467
   Addr2line_Args (1) := new String'("--exe=" & Argument (1));
468
   Addr2line_Args_Count := 1;
469
 
470
   --  Now, append to this the adjusted backtraces in arguments 4 and further
471
 
472
   for J in 3 .. Argument_Count loop
473
 
474
      --  Basically, for each address in the runtime backtrace ...
475
 
476
      --  o We compute its offset relatively to the runtime address of the
477
      --    reference symbol,
478
 
479
      --  and then ...
480
 
481
      --  o We add this offset to the static one for the reference symbol in
482
      --    the executable to find the executable offset corresponding to the
483
      --    backtrace address.
484
 
485
      Bt_Address := Get_Value_From_Hex_Arg (J);
486
 
487
      Bt_Address :=
488
        Bt_Address - Ref_Runtime_Address
489
                   + Ref_Static_Offset
490
                   + Arch_List (Cur_Arch).Bt_Offset_From_Call;
491
 
492
      Addr2line_Args_Count := Addr2line_Args_Count + 1;
493
      Addr2line_Args (Addr2line_Args_Count) := Hex_Image (Bt_Address);
494
   end loop;
495
 
496
   --  Run the resulting command
497
 
498
   Spawn (Addr2line_Cmd.all,
499
          Addr2line_Args (1 .. Addr2line_Args_Count), Success);
500
 
501
   if not Success then
502
      Error ("Couldn't spawn " & Addr2line_Cmd.all);
503
   end if;
504
 
505
exception
506
   when others =>
507
 
508
      --  Mask all exceptions
509
 
510
      return;
511
end VxAddr2Line;

powered by: WebSVN 2.1.0

© copyright 1999-2025 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.