-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathpargs_tmpl.c
619 lines (571 loc) · 20.9 KB
/
pargs_tmpl.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
// Word size (32/64) dependent ELF core file read functions.
// To be included from pargs.c
//
// 2017, Georg Sauthoff <[email protected]>
//
// SPDX-License-Identifier: GPL-3.0-or-later
#ifndef PARGS_ELF_CLASS
#error "PARGS_ELF_CLASS must be set to 32 or 64 before including this file"
#endif
#define WIDTH PARGS_ELF_CLASS
#ifdef CAT
#error "symbol CAT already defined ..."
#endif
#define CAT_P(A, B) A ## B
// one redirection to expand A and B
#define CAT(A, B) CAT_P(A, B)
#define CAT3_P(A, B, C) A ## B ## C
#define CAT3(A, B, C) CAT3_P(A, B, C)
#include <assert.h> // static_assert()
#include <stddef.h> // offsetof()
#include <sys/procfs.h> // prpsinfo_t
// necessary e.g. when compiling with clang 3.4 under CentOS/RHEL 6
#ifndef static_assert
#define static_assert _Static_assert
#endif
#define DECL_U_CP(W, NAME, SRC) \
CAT3(uint, W, _t) NAME; \
memcpy(&NAME, (SRC), sizeof NAME); \
if (lm->need_to_swap) \
NAME = CAT(__builtin_bswap, W) (NAME)
// expands to parse_auxv_32() or parse_auxv_64()
//
// We look for the AT_EXECFN value because it's an address that points
// to the exec filename which is located on top of the envp/argv
// strings. Thus, we can use that address to identify the right
// section and search for the vectors.
static int CAT(parse_auxv_, WIDTH) (const char *filename, Landmarks *lm,
const unsigned char *begin, const unsigned char *end)
{
debug("aux note size: %zd", end-begin);
lm->auxv_note.begin = begin;
lm->auxv_note.end = end;
lm->execfn_addr = 0;
for (const unsigned char *p = begin; p < end; p += (WIDTH/8) * 2) {
DECL_U_CP(WIDTH, key, p);
if (!key || key == AT_EXECFN) {
DECL_U_CP(WIDTH, value, p + WIDTH/8);
switch (key) {
case 0:
if (value) {
fprintf(stderr, "Unexpected auxv value for sentinel key.\n");
return -1;
}
if (!lm->execfn_addr) {
fprintf(stderr, "Didn't see AT_EXECFN in %s.\n", filename);
return -1;
}
return 0;
case AT_EXECFN:
debug("Found AT_EXECFN value: 0x%" CAT(PRIx, WIDTH), value);
lm->execfn_addr = value;
break;
}
}
}
fprintf(stderr, "Sentinel missing in auxv note in %s\n", filename);
return -1;
}
// expands to parse_prpsinfo_32() or parse_prpsinfo_64()
static int CAT(parse_prpsinfo_, WIDTH) (const char *filename, Landmarks *lm,
const unsigned char *begin, const unsigned char *end)
{
debug("prpsinfo note size: %zu", (size_t)(end - begin));
uint32_t pid;
ssize_t extra_off = 0;
// __WORDSIZE is used in sys/procfs.h's definition of prpsinfo
static_assert(__WORDSIZE == 32 || __WORDSIZE == 64,
"unexpected __WORDSIZE");
// why 12 bytes? On 64 bit (LP64),in contrast to 32 bit (ILP32):
// there are 4 bytes padding after pr_nice
// pr_flag is 8 bytes instead of 4
// pr_uid + pr_gid use 8 bytes instead of 4
if (__WORDSIZE < WIDTH)
extra_off = 12;
else if (__WORDSIZE > WIDTH) {
extra_off = -12;
}
// true e.g when reading a Debian 8 ppc64 -m32 core on Fedora 26 x86-64
// (both with pargs and pargs32)
// because pr_uid/pr_gid are unconditionally 4 bytes big on Debian 8 ppc64
if (sizeof(prpsinfo_t) + extra_off + 4 == (size_t)(end - begin))
extra_off += 4;
if (sizeof(prpsinfo_t) + extra_off > (size_t)(end - begin)) {
fprintf(stderr, "prpsinfo section overflows in %s\n", filename);
return -1;
}
// according to sys/procfs.h, pid should be 4 byte everwhere
static_assert(sizeof(pid_t) == 4, "pid_t is not 4 byte large ...");
memcpy(&pid, begin + offsetof(prpsinfo_t, pr_pid) + extra_off, sizeof pid);
if (lm->need_to_swap)
pid = __builtin_bswap32(pid);
debug("found PID: %ju", (intmax_t)pid);
lm->pid = pid;
// other possibly interesting fields:
// pr_fname - just 16 bytes, though.
// Thus, may truncate (without zero-terminating!).
// pr_psargs - also relatively small, args are concatenated,
// expected arguments aren't necessarily included
return 0;
}
// expands to parse_note_32() or parse_note_64()
static int CAT(parse_note_, WIDTH) (const char *filename,
Landmarks *lm, uint32_t note_type,
const unsigned char *name_begin, const unsigned char *name_end,
const unsigned char *desc_begin, const unsigned char *desc_end)
{
(void)name_begin;
(void)name_end;
switch (note_type) {
case NT_AUXV:
debug("Reading auxv note");
return CAT(parse_auxv_, WIDTH) (filename, lm, desc_begin, desc_end);
break;
case NT_PRPSINFO:
debug("Reading prpsinfo note");
return CAT(parse_prpsinfo_, WIDTH) (filename, lm, desc_begin, desc_end);
break;
default:
debug("Skipping other note");
}
return 0;
}
// expands to parse_notes_32() or parse_notes_64()
static int CAT(parse_notes_, WIDTH) (
const Range *range, const char *filename, Landmarks *lm,
const Range *section_range)
{
(void)range;
const unsigned char *x = section_range->begin;
const unsigned char *y = section_range->end;
size_t i = 0;
for (const unsigned char *p = x; p < y; ++i) {
DECL_U_CP(32, name_size, p);
uint32_t aligned_name_size = align32_up_4(name_size);
DECL_U_CP(32, desc_size, p + offsetof(CAT3(Elf, WIDTH,_Nhdr), n_descsz));
uint32_t aligned_desc_size = align32_up_4(desc_size);
static_assert(sizeof(CAT3(Elf, WIDTH, _Nhdr)) == 3*4,
"Note header isn't 3*4");
if (p + sizeof(CAT3(Elf, WIDTH, _Nhdr))
+ aligned_name_size + desc_size > y) {
fprintf(stderr, "%zuth note overflows NOTE section in %s.\n",
i, filename);
return -1;
}
DECL_U_CP(32, note_type, p + offsetof(CAT3(Elf, WIDTH, _Nhdr), n_type));
p += sizeof(CAT3(Elf, WIDTH, _Nhdr));
int r = CAT(parse_note_, WIDTH) (filename, lm, note_type,
p, p + name_size,
p + aligned_name_size, p + aligned_name_size + desc_size);
p += aligned_name_size + aligned_desc_size;
if (r)
return r;
}
return 0;
}
/*
The interetesting vectors like argv, envp, pieces like exec_fn
and the stack are located in the same memory region that is mapped
at the top of the address space. Basically it looks like this:
highest address := end of mapping
| execfn[0] .. execfn[ex_l-1] \0 <-------------------+
| +-> envp[m-1][0] .. envp[m-1][envp_l[m-1]-1] \0 |
| | .. |
| | envp[0][0] .. envp[0][envp_l[0]-1] \0 |
| | argv[n-1][0] .. argv[n-1][argv_l[n-1]-1] \0 <--------+
| | .. | |
| +---> argv[0][0] .. argv[0][argv_l[0]-1] \0 <----------------+
| | | .. | | |
| | | platform[0] .. platform[p_l-1] \0 <-------------+ | | |
| | | random[0] .. random[r_l-1] <- 16 bytes <------+ | | | |
| | | .. | | | | |
| | | \0 .. \0 := 2 * $word_size/8 bytes, end-of-auxv | | | | |
| | | .. | | | | |
| | | AT_RANDOM addr_p -----------------------------+ | | | |
| | | .. | | | |
| | | AT_PLATFORM addr_r -------------------------------+ | | |
| | | .. | | |
| | | AT_EXECFN addr_e ---------------------------------+ | |
| | | .. | |
| | | .. <- first auxv pair (2 *$word_size/8 bytes) | |
| | | .. | |
| | | envp[m] = \0 .. \0 := $word_size/8 bytes | |
| | +-- envp[m-1] := $word_size/8 bytes | |
| | .. | |
| +---- envp[0] | |
| argv[n] = \0 .. \0 := $word_size/8 bytes -----------+ |
| .. |
| argv[n-1] := $word_size/8 bytes -------------+
| ..
| argv[0]
| argc := $word_size/8 bytes
| ..
| stack_frame[2] start
| ..
| stack_frame[2] end
| ..
| .. := more stack frames ...
| ..
| .. := stack grows downwards
|
|
\./
lowest address
Note that the auxillary vector (auvx) is also copied in the NOTE
section of core files - as separate NOTE. Thus, we don't have to
search for it here and can just use the copy. Addresses still have to
be resolved in this stack section, though.
See also:
How programs get run: ELF binaries (2015)
https://lwn.net/Articles/631631/
Readelf dump of AUXV values of an example core file (elf-utils):
eu-readelf -a core
Readelf hexdump output of an example core file:
readelf -x $sectionnr core
*/
static int CAT(find_vectors_, WIDTH) (const char *filename,
Landmarks *lm, const Range *section_range,
CAT3(uint, WIDTH, _t) section_virt_addr)
{
lm->vector_section = *section_range;
lm->vector_base_addr = section_virt_addr;
const unsigned char *b = section_range->begin;
const unsigned char *e = section_range->end;
if (lm->execfn_addr < section_virt_addr) {
fprintf(stderr, "Execfn address underflows section.\n");
return -1;
}
CAT3(uint, WIDTH, _t) execfn_off = lm->execfn_addr - section_virt_addr;
const unsigned char *p = memchr(b + execfn_off, 0, e - (b + execfn_off));
if (!p) {
fprintf(stderr, "exec filename is not null-terminated in %s.\n",
filename);
return -1;
}
p = b + execfn_off;
debug("exec filename: %s", p);
unsigned char envp_tail[4 * (WIDTH/8)] = {0};
{
unsigned j = sizeof envp_tail - (WIDTH/8);
for (unsigned i = 0; i < sizeof envp_tail / (WIDTH/8);
++i, j -= (WIDTH/8)) {
if (p-b <= 2) {
fprintf(stderr, "Underflow in envp tail search.\n");
return -1;
}
p = memrchr(b, 0, (p-b)-2);
if (!p)
return -1;
++p;
CAT3(uint, WIDTH, _t) addr = section_virt_addr + (p - b);
debug("Trailing environment string `%s' at offset %zd"
" (virt address %" CAT(PRIx, WIDTH) ")", p, (p - b), addr);
if (lm->need_to_swap)
addr = CAT(__builtin_bswap, WIDTH) (addr);
memcpy(envp_tail + j, &addr, WIDTH/8);
}
}
p = aligned_naive_memmemr(b, execfn_off / (WIDTH/8) * (WIDTH/8),
envp_tail, sizeof envp_tail, WIDTH/8);
if (!p) {
fprintf(stderr, "Can't find envp tail in %s.\n", filename);
return -1;
}
debug("Found envp tail at offset %zd (virt address %" CAT(PRIx, WIDTH) ")",
(p - b), section_virt_addr + (CAT3(uint, WIDTH, _t)) (p - b));
lm->envp.end = p + sizeof envp_tail;
CAT3(uint, WIDTH, _t) virt_begin =
section_virt_addr + ((p + sizeof envp_tail) - b);
CAT3(uint, WIDTH, _t) virt_end = section_virt_addr + (e - b);
unsigned char argv_tail[WIDTH/8] = {0};
p = aligned_naive_memmemr(b, p - b, argv_tail, sizeof argv_tail, WIDTH/8);
if (!p) {
fprintf(stderr, "Can't find argv tail in %s.\n", filename);
return -1;
}
debug("Found argv tail at offset %zd (virt address %" CAT(PRIx, WIDTH) ")",
(p - b), section_virt_addr + (CAT3(uint, WIDTH, _t)) (p - b));
lm->envp.begin = p + sizeof argv_tail;
lm->argv.end = p;
p -= WIDTH/8;
lm->argc = 0;
for (; p > b; p -= WIDTH/8) {
// amd64: argc is int, which is 4 byte, but argv[0] is 8 byte aligned
// thus, we can read argc as uint64_t where the 4 most significant bytes
// are zero
CAT3(uint, WIDTH, _t) t;
memcpy(&t, p, WIDTH/8);
if (lm->need_to_swap)
t = CAT(__builtin_bswap, WIDTH) (t);
if (t < virt_begin || t >= virt_end) {
lm->argc = t;
debug("Found argc = %d at offset %zd (virt address %"
CAT(PRIx, WIDTH) ")", lm->argc,
(p - b), section_virt_addr + (CAT3(uint, WIDTH, _t)) (p - b));
lm->argv.begin = p + WIDTH/8;
break;
}
}
if (!lm->argc) {
fprintf(stderr, "Didn't find argc in %s.\n", filename);
return -1;
}
return 0;
}
// expands to parse_segment_32() or parse_segment_64()
static int CAT(parse_segment_, PARGS_ELF_CLASS) (
const Range *range, const char *filename, Landmarks *lm,
const Range *segment_range, uint32_t segment_type,
CAT3(uint, WIDTH, _t) segment_virt_addr)
{
CAT3(uint, WIDTH, _t) segment_size =
segment_range->end - segment_range->begin;
switch (segment_type) {
case PT_NOTE:
debug("Reading NOTE segment");
return CAT(parse_notes_, WIDTH) (range, filename, lm, segment_range);
case PT_LOAD:
if (lm->execfn_addr > segment_virt_addr
&& lm->execfn_addr < segment_virt_addr + segment_size) {
debug("Found segment that includes the execfn address");
return CAT(find_vectors_, WIDTH) (filename, lm, segment_range,
segment_virt_addr);
} else {
debug("Skipping PROGBITS segment");
}
break;
default: debug("Skipping other segment");
}
return 0;
}
/*
Note that the ELF header points to two tables:
- the program header table
- the section header table
The dumped memory content we are interested in is reachable via
both tables (cf. Wikipedia's [two-views figure][elfviews]). Core files
generated by `gcore` contain a section header table, while core
files generated by the kernel don't. Thus, we have to traverse
the program headers instead of the section headers.
The first traversal stop is the notes segement. It's
corresponding program header has type PT_NOTE where as the
referencing section has type SHT_NOTE. The vectors and other
interesting bits are in a segment referenced by a header of type
PT_LOAD or SHT_PROGBITS, respectively.
[elfviews]: https://commons.wikimedia.org/wiki/File:Elf-layout--en.svg
*/
// expands to parse_landmarks_32() or parse_landmarks_64()
static int CAT(parse_landmarks_, PARGS_ELF_CLASS) (
const Range *range, const char *filename, Landmarks *lm)
{
const unsigned char *b = range->begin;
const unsigned char *e = range->end;
if ((size_t)(e - b) < sizeof(CAT3(Elf, WIDTH, _Ehdr))) {
fprintf(stderr, "file too small for ELF%d header\n", WIDTH);
return -1;
}
// under strict-aliasing rules a Elf(32|64_Ehdr) pointer can't alias
// a unsigned char pointer
// the compiler should be able to optimize the memcpy() away
// we could also copy the complete struct, but we just need a few fields
// also, we need to byte-swap some integer fields when core file was
// generated a machine with a different byte order ...
DECL_U_CP(16, type, b + offsetof(CAT3(Elf, WIDTH, _Ehdr), e_type));
if (type != ET_CORE) {
fprintf(stderr, "%s is not a core file.\n", filename);
return -1;
}
DECL_U_CP(16, first_segment_count,
b + offsetof(CAT3(Elf, WIDTH, _Ehdr), e_phnum));
debug("%" PRIu16 " segments", first_segment_count);
if (!first_segment_count) {
fprintf(stderr, "File %s has no segments.\n", filename);
return -1;
}
DECL_U_CP(WIDTH, program_header_off,
b + offsetof(CAT3(Elf, WIDTH, _Ehdr), e_phoff));
DECL_U_CP(16, program_header_size,
b + offsetof(CAT3(Elf, WIDTH, _Ehdr), e_phentsize));
if (program_header_off + program_header_size > (size_t)(e - b)) {
fprintf(stderr, "%s: program header table overflows\n", filename);
return -1;
}
CAT3(uint, WIDTH, _t) segment_count = first_segment_count;
// cf. the ELF standard, e.g. elf(5) and search for PN_XNUM
if (segment_count >= PN_XNUM) {
debug("file has more than 2**16-1 segments");
const unsigned char *p = b + program_header_off;
DECL_U_CP(WIDTH, new_segment_count,
p + offsetof(CAT3(Elf, WIDTH, _Shdr), sh_size));
segment_count = new_segment_count;
debug("new segment count: %" CAT(PRIu, WIDTH), segment_count);
}
if (program_header_off + segment_count * program_header_size > (size_t)(e - b)) {
fprintf(stderr, "%s: program header table overflows\n", filename);
return -1;
}
CAT3(uint, WIDTH, _t) i = 0;
for (const unsigned char *p = b + program_header_off
+ (segment_count != first_segment_count) * program_header_size;
p < b + program_header_off + segment_count * program_header_size;
p += program_header_size, ++i) {
debug("Reading %" CAT(PRIu, WIDTH) "th segment", i+1);
DECL_U_CP(WIDTH, segment_off,
p + offsetof(CAT3(Elf, WIDTH, _Phdr), p_offset));
DECL_U_CP(WIDTH, segment_size,
p + offsetof(CAT3(Elf, WIDTH, _Phdr), p_filesz));
if (segment_off + segment_size > (size_t)(e - b)) {
fprintf(stderr, "segment %" CAT(PRIu, WIDTH) " overflows %s.\n",
i, filename);
return -1;
}
DECL_U_CP(32, segment_type, p + offsetof(CAT3(Elf, WIDTH, _Phdr), p_type));
DECL_U_CP(WIDTH, segment_virt_addr,
p + offsetof(CAT3(Elf, WIDTH, _Phdr), p_vaddr));
Range segment_range = { b + segment_off, b + segment_off + segment_size };
int r = CAT(parse_segment_, WIDTH) (range, filename, lm,
&segment_range, segment_type, segment_virt_addr);
if (r)
return r;
}
if (!lm->execfn_addr) {
fprintf(stderr, "Couldn't find any executable filename in %s.\n",
filename);
return -1;
}
return 0;
}
static int CAT(fput_core_vector_, WIDTH) (const Landmarks *lm,
const Range *vec,
const char *prefix, const char *delim, FILE *o)
{
const unsigned char *b = lm->vector_section.begin;
const unsigned char *e = lm->vector_section.end;
size_t i = 0;
for (const unsigned char *p = vec->begin; p < vec->end; p += WIDTH/8, ++i) {
DECL_U_CP(WIDTH, addr, p);
if (addr < lm->vector_base_addr) {
fprintf(stderr, "Pointer underflows section.\n");
return -1;
}
CAT3(uint, WIDTH, _t) off = addr - lm->vector_base_addr;
if (off > (size_t)(e - b)) {
fprintf(stderr, "Pointer points outside section.\n");
return -1;
}
const char *s = (const char*) (b + off);
const char *t = memchr(s, 0, e - (b + off));
if (!t) {
fprintf(stderr, "Pointer is not null-terminated.\n");
return -1;
}
if (i)
fputs(delim, o);
if (prefix)
fprintf(o, "%s[%zu]: ", prefix, i);
fputs(s, o);
}
return 0;
}
static const char *CAT(get_core_str_, WIDTH) (CAT3(uint, WIDTH, _t) addr,
const Landmarks *lm)
{
if (addr < lm->vector_base_addr) {
fprintf(stderr, "Start of string underflows.\n");
return 0;
}
CAT3(uint, WIDTH, _t) off = addr - lm->vector_base_addr;
const unsigned char *b = lm->vector_section.begin;
const unsigned char *e = lm->vector_section.end;
if (off > (size_t)(e - b)) {
fprintf(stderr, "Start of string overflows.\n");
return 0;
}
const char *t = memchr(b + off, 0, e - (b + off));
if (!t) {
fprintf(stderr, "String isn't null-terminated.");
return 0;
}
return (const char*) (b + off);
}
static const unsigned char *CAT(
get_core_mem_, WIDTH) (CAT3(uint, WIDTH, _t) addr, size_t n,
const Landmarks *lm)
{
if (addr < lm->vector_base_addr) {
fprintf(stderr, "Start of string underflows.\n");
return 0;
}
CAT3(uint, WIDTH, _t) off = addr - lm->vector_base_addr;
const unsigned char *b = lm->vector_section.begin;
const unsigned char *e = lm->vector_section.end;
if (off > (size_t)(e - b)) {
fprintf(stderr, "Start of string overflows.\n");
return 0;
}
if (off + n > (size_t)(e - b)) {
fprintf(stderr, "End of string overflows.\n");
return 0;
}
return b + off;
}
static int CAT(pp_core_aux_ref_, WIDTH) (CAT3(uint, WIDTH, _t) key,
CAT3(uint, WIDTH, _t) val, const Landmarks *lm, FILE *o)
{
switch (key) {
case AT_PLATFORM:
case AT_EXECFN: {
const char *s = CAT(get_core_str_, WIDTH) (val, lm);
fprintf(o, " %s", s);
break;
}
case AT_RANDOM:
{
static const size_t n = 16;
const unsigned char *s = CAT(get_core_mem_, WIDTH) (val, n, lm);
for (size_t i = 0; i < n; ++i)
fprintf(o, " %.2x", (unsigned) s[i]);
break;
}
default: break;
}
return 0;
}
static int CAT(fput_core_auxv_, WIDTH) (const Landmarks *lm, FILE *o,
const Args *args)
{
if (!lm->auxv_note.begin) {
fprintf(stderr, "auxv not found.\n");
return -1;
}
size_t i = 0;
for (const unsigned char *p = lm->auxv_note.begin; ;
p += 2 * (WIDTH/8), ++i) {
DECL_U_CP(WIDTH, key, p);
DECL_U_CP(WIDTH, value, p + WIDTH/8);
if (!key)
return 0;
if (i)
fputc('\n', o);
if (key < sizeof auxv_type_map / sizeof auxv_type_map[0])
fprintf(o, "%-16s", auxv_type_map[key].key);
else
fprintf(o, "unk_%" CAT(PRIu, WIDTH), key);
fprintf(o, " 0x%.16" CAT(PRIx, WIDTH), value);
int r = pp_aux(key, value, o);
if (r)
return r;
r = CAT(pp_core_aux_ref_, WIDTH) (key, value, lm, o);
if (r)
return r;
r = pp_aux_v(key, o, args);
if (r)
return r;
}
return 0;
}
#undef WIDTH
#undef CAT
#undef CAT3
#undef CAT_P
#undef CAT3_P