Imported Upstream version 0.165
[platform/upstream/elfutils.git] / libebl / eblobjnote.c
1 /* Print contents of object file note.
2    Copyright (C) 2002, 2007, 2009, 2011, 2015, 2016 Red Hat, Inc.
3    This file is part of elfutils.
4    Written by Ulrich Drepper <drepper@redhat.com>, 2002.
5
6    This file is free software; you can redistribute it and/or modify
7    it under the terms of either
8
9      * the GNU Lesser General Public License as published by the Free
10        Software Foundation; either version 3 of the License, or (at
11        your option) any later version
12
13    or
14
15      * the GNU General Public License as published by the Free
16        Software Foundation; either version 2 of the License, or (at
17        your option) any later version
18
19    or both in parallel, as here.
20
21    elfutils is distributed in the hope that it will be useful, but
22    WITHOUT ANY WARRANTY; without even the implied warranty of
23    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
24    General Public License for more details.
25
26    You should have received copies of the GNU General Public License and
27    the GNU Lesser General Public License along with this program.  If
28    not, see <http://www.gnu.org/licenses/>.  */
29
30 #ifdef HAVE_CONFIG_H
31 # include <config.h>
32 #endif
33
34 #include <inttypes.h>
35 #include <stdio.h>
36 #include <stdlib.h>
37 #include <string.h>
38 #include <libeblP.h>
39
40
41 void
42 ebl_object_note (Ebl *ebl, const char *name, uint32_t type,
43                  uint32_t descsz, const char *desc)
44 {
45   if (! ebl->object_note (name, type, descsz, desc))
46     {
47       /* The machine specific function did not know this type.  */
48
49       if (strcmp ("stapsdt", name) == 0)
50         {
51           if (type != 3)
52             {
53               printf (gettext ("unknown SDT version %u\n"), type);
54               return;
55             }
56
57           /* Descriptor starts with three addresses, pc, base ref and
58              semaphore.  Then three zero terminated strings provider,
59              name and arguments.  */
60
61           union
62           {
63             Elf64_Addr a64[3];
64             Elf32_Addr a32[3];
65           } addrs;
66
67           size_t addrs_size = gelf_fsize (ebl->elf, ELF_T_ADDR, 3, EV_CURRENT);
68           if (descsz < addrs_size + 3)
69             {
70             invalid_sdt:
71               printf (gettext ("invalid SDT probe descriptor\n"));
72               return;
73             }
74
75           Elf_Data src =
76             {
77               .d_type = ELF_T_ADDR, .d_version = EV_CURRENT,
78               .d_buf = (void *) desc, .d_size = addrs_size
79             };
80
81           Elf_Data dst =
82             {
83               .d_type = ELF_T_ADDR, .d_version = EV_CURRENT,
84               .d_buf = &addrs, .d_size = addrs_size
85             };
86
87           if (gelf_xlatetom (ebl->elf, &dst, &src,
88                              elf_getident (ebl->elf, NULL)[EI_DATA]) == NULL)
89             {
90               printf ("%s\n", elf_errmsg (-1));
91               return;
92             }
93
94           const char *provider = desc + addrs_size;
95           const char *pname = memchr (provider, '\0', desc + descsz - provider);
96           if (pname == NULL)
97             goto invalid_sdt;
98
99           ++pname;
100           const char *args = memchr (pname, '\0', desc + descsz - pname);
101           if (args == NULL ||
102               memchr (++args, '\0', desc + descsz - pname) != desc + descsz - 1)
103             goto invalid_sdt;
104
105           GElf_Addr pc;
106           GElf_Addr base;
107           GElf_Addr sem;
108           if (gelf_getclass (ebl->elf) == ELFCLASS32)
109             {
110               pc = addrs.a32[0];
111               base = addrs.a32[1];
112               sem = addrs.a32[2];
113             }
114           else
115             {
116               pc = addrs.a64[0];
117               base = addrs.a64[1];
118               sem = addrs.a64[2];
119             }
120
121           printf (gettext ("    PC: "));
122           printf ("%#" PRIx64 ",", pc);
123           printf (gettext (" Base: "));
124           printf ("%#" PRIx64 ",", base);
125           printf (gettext (" Semaphore: "));
126           printf ("%#" PRIx64 "\n", sem);
127           printf (gettext ("    Provider: "));
128           printf ("%s,", provider);
129           printf (gettext (" Name: "));
130           printf ("%s,", pname);
131           printf (gettext (" Args: "));
132           printf ("'%s'\n", args);
133           return;
134         }
135
136       switch (type)
137         {
138         case NT_GNU_BUILD_ID:
139           if (strcmp (name, "GNU") == 0 && descsz > 0)
140             {
141               printf (gettext ("    Build ID: "));
142               uint_fast32_t i;
143               for (i = 0; i < descsz - 1; ++i)
144                 printf ("%02" PRIx8, (uint8_t) desc[i]);
145               printf ("%02" PRIx8 "\n", (uint8_t) desc[i]);
146             }
147           break;
148
149         case NT_GNU_GOLD_VERSION:
150           if (strcmp (name, "GNU") == 0 && descsz > 0)
151             /* A non-null terminated version string.  */
152             printf (gettext ("    Linker version: %.*s\n"),
153                     (int) descsz, desc);
154           break;
155
156         case NT_GNU_ABI_TAG:
157           if (strcmp (name, "GNU") == 0 && descsz >= 8 && descsz % 4 == 0)
158             {
159               Elf_Data in =
160                 {
161                   .d_version = EV_CURRENT,
162                   .d_type = ELF_T_WORD,
163                   .d_size = descsz,
164                   .d_buf = (void *) desc
165                 };
166               /* Normally NT_GNU_ABI_TAG is just 4 words (16 bytes).  If it
167                  is much (4*) larger dynamically allocate memory to convert.  */
168 #define FIXED_TAG_BYTES 16
169               uint32_t sbuf[FIXED_TAG_BYTES];
170               uint32_t *buf;
171               if (unlikely (descsz / 4 > FIXED_TAG_BYTES))
172                 {
173                   buf = malloc (descsz);
174                   if (unlikely (buf == NULL))
175                     return;
176                 }
177               else
178                 buf = sbuf;
179               Elf_Data out =
180                 {
181                   .d_version = EV_CURRENT,
182                   .d_type = ELF_T_WORD,
183                   .d_size = descsz,
184                   .d_buf = buf
185                 };
186
187               if (elf32_xlatetom (&out, &in, ebl->data) != NULL)
188                 {
189                   const char *os;
190                   switch (buf[0])
191                     {
192                     case ELF_NOTE_OS_LINUX:
193                       os = "Linux";
194                       break;
195
196                     case ELF_NOTE_OS_GNU:
197                       os = "GNU";
198                       break;
199
200                     case ELF_NOTE_OS_SOLARIS2:
201                       os = "Solaris";
202                       break;
203
204                     case ELF_NOTE_OS_FREEBSD:
205                       os = "FreeBSD";
206                       break;
207
208                     default:
209                       os = "???";
210                       break;
211                     }
212
213                   printf (gettext ("    OS: %s, ABI: "), os);
214                   for (size_t cnt = 1; cnt < descsz / 4; ++cnt)
215                     {
216                       if (cnt > 1)
217                         putchar_unlocked ('.');
218                       printf ("%" PRIu32, buf[cnt]);
219                     }
220                   putchar_unlocked ('\n');
221                 }
222               if (descsz / 4 > FIXED_TAG_BYTES)
223                 free (buf);
224               break;
225             }
226           /* FALLTHROUGH */
227
228         default:
229           /* Unknown type.  */
230           break;
231         }
232     }
233 }