Coverage Report

Created: 2026-03-10 08:46

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/binutils-gdb/bfd/pef.c
Line
Count
Source
1
/* PEF support for BFD.
2
   Copyright (C) 1999-2026 Free Software Foundation, Inc.
3
4
   This file is part of BFD, the Binary File Descriptor library.
5
6
   This program is free software; you can redistribute it and/or modify
7
   it under the terms of the GNU General Public License as published by
8
   the Free Software Foundation; either version 3 of the License, or
9
   (at your option) any later version.
10
11
   This program is distributed in the hope that it will be useful,
12
   but WITHOUT ANY WARRANTY; without even the implied warranty of
13
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14
   GNU General Public License for more details.
15
16
   You should have received a copy of the GNU General Public License
17
   along with this program; if not, write to the Free Software
18
   Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston,
19
   MA 02110-1301, USA.  */
20
21
/* PEF (Preferred Executable Format) is the binary file format for late
22
   classic Mac OS versions (before Darwin).  It is supported by both m68k
23
   and PowerPc.  It is also called CFM (Code Fragment Manager).  */
24
25
#include "sysdep.h"
26
#include "safe-ctype.h"
27
#include "pef.h"
28
#include "pef-traceback.h"
29
#include "bfd.h"
30
#include "libbfd.h"
31
#include "libiberty.h"
32
33
#ifndef BFD_IO_FUNCS
34
809
#define BFD_IO_FUNCS 0
35
#endif
36
37
#define bfd_pef_close_and_cleanup       _bfd_generic_close_and_cleanup
38
#define bfd_pef_bfd_free_cached_info        _bfd_generic_bfd_free_cached_info
39
#define bfd_pef_new_section_hook        _bfd_generic_new_section_hook
40
#define bfd_pef_bfd_is_local_label_name       bfd_generic_is_local_label_name
41
#define bfd_pef_bfd_is_target_special_symbol        _bfd_bool_bfd_asymbol_false
42
#define bfd_pef_get_lineno          _bfd_nosymbols_get_lineno
43
#define bfd_pef_find_nearest_line       _bfd_nosymbols_find_nearest_line
44
#define bfd_pef_find_nearest_line_with_alt      _bfd_nosymbols_find_nearest_line_with_alt
45
#define bfd_pef_find_line         _bfd_nosymbols_find_line
46
#define bfd_pef_find_inliner_info       _bfd_nosymbols_find_inliner_info
47
#define bfd_pef_get_symbol_version_string     _bfd_nosymbols_get_symbol_version_string
48
#define bfd_pef_bfd_make_debug_symbol       _bfd_nosymbols_bfd_make_debug_symbol
49
#define bfd_pef_read_minisymbols        _bfd_generic_read_minisymbols
50
#define bfd_pef_minisymbol_to_symbol        _bfd_generic_minisymbol_to_symbol
51
#define bfd_pef_set_arch_mach         _bfd_generic_set_arch_mach
52
#define bfd_pef_get_section_contents        _bfd_generic_get_section_contents
53
#define bfd_pef_set_section_contents        _bfd_generic_set_section_contents
54
#define bfd_pef_bfd_get_relocated_section_contents  bfd_generic_get_relocated_section_contents
55
#define bfd_pef_bfd_relax_section       bfd_generic_relax_section
56
#define bfd_pef_bfd_gc_sections         bfd_generic_gc_sections
57
#define bfd_pef_bfd_lookup_section_flags      bfd_generic_lookup_section_flags
58
#define bfd_pef_bfd_is_group_section        bfd_generic_is_group_section
59
#define bfd_pef_bfd_group_name          bfd_generic_group_name
60
#define bfd_pef_bfd_discard_group       bfd_generic_discard_group
61
#define bfd_pef_section_already_linked        _bfd_generic_section_already_linked
62
#define bfd_pef_bfd_define_common_symbol      bfd_generic_define_common_symbol
63
#define bfd_pef_bfd_link_hide_symbol        _bfd_generic_link_hide_symbol
64
#define bfd_pef_bfd_define_start_stop       bfd_generic_define_start_stop
65
#define bfd_pef_bfd_link_hash_table_create      _bfd_generic_link_hash_table_create
66
#define bfd_pef_bfd_link_add_symbols        _bfd_generic_link_add_symbols
67
#define bfd_pef_bfd_link_just_syms        _bfd_generic_link_just_syms
68
#define bfd_pef_bfd_copy_link_hash_symbol_type \
69
  _bfd_generic_copy_link_hash_symbol_type
70
#define bfd_pef_bfd_final_link          _bfd_generic_final_link
71
#define bfd_pef_bfd_link_split_section        _bfd_generic_link_split_section
72
#define bfd_pef_bfd_link_check_relocs       _bfd_generic_link_check_relocs
73
74
static int
75
bfd_pef_parse_traceback_table (bfd *abfd,
76
             asection *section,
77
             unsigned char *buf,
78
             size_t len,
79
             size_t pos,
80
             asymbol *sym,
81
             FILE *file)
82
182k
{
83
182k
  struct traceback_table table;
84
182k
  size_t offset;
85
182k
  const char *s;
86
182k
  asymbol tmpsymbol;
87
88
182k
  if (sym == NULL)
89
0
    sym = & tmpsymbol;
90
91
182k
  sym->name = NULL;
92
182k
  sym->value = 0;
93
182k
  sym->the_bfd = abfd;
94
182k
  sym->section = section;
95
182k
  sym->flags = 0;
96
182k
  sym->udata.i = 0;
97
98
  /* memcpy is fine since all fields are unsigned char.  */
99
182k
  if ((pos + 8) > len)
100
376
    return -1;
101
182k
  memcpy (&table, buf + pos, 8);
102
103
  /* Calling code relies on returned symbols having a name and
104
     correct offset.  */
105
182k
  if ((table.lang != TB_C) && (table.lang != TB_CPLUSPLUS))
106
33.9k
    return -1;
107
108
148k
  if (! (table.flags2 & TB_NAME_PRESENT))
109
133k
    return -1;
110
111
14.8k
  if (! (table.flags1 & TB_HAS_TBOFF))
112
6.54k
    return -1;
113
114
8.29k
  offset = 8;
115
116
8.29k
  if ((table.flags5 & TB_FLOATPARAMS) || (table.fixedparams))
117
4.90k
    offset += 4;
118
119
8.29k
  if (table.flags1 & TB_HAS_TBOFF)
120
8.29k
    {
121
8.29k
      struct traceback_table_tboff off;
122
123
8.29k
      if ((pos + offset + 4) > len)
124
16
  return -1;
125
8.27k
      off.tb_offset = bfd_getb32 (buf + pos + offset);
126
8.27k
      offset += 4;
127
128
      /* Need to subtract 4 because the offset includes the 0x0L
129
   preceding the table.  */
130
8.27k
      if (file != NULL)
131
0
  fprintf (file, " [offset = 0x%lx]", off.tb_offset);
132
133
8.27k
      if ((file == NULL) && ((off.tb_offset + 4) > (pos + offset)))
134
2.62k
  return -1;
135
136
5.65k
      sym->value = pos - off.tb_offset - 4;
137
5.65k
    }
138
139
5.65k
  if (table.flags2 & TB_INT_HNDL)
140
2.35k
    offset += 4;
141
142
5.65k
  if (table.flags1 & TB_HAS_CTL)
143
2.94k
    {
144
2.94k
      struct traceback_table_anchors anchors;
145
146
2.94k
      if ((pos + offset + 4) > len)
147
10
  return -1;
148
2.93k
      anchors.ctl_info = bfd_getb32 (buf + pos + offset);
149
2.93k
      offset += 4;
150
151
2.93k
      if (anchors.ctl_info > 1024)
152
922
  return -1;
153
154
2.01k
      offset += anchors.ctl_info * 4;
155
2.01k
    }
156
157
4.72k
  if (table.flags2 & TB_NAME_PRESENT)
158
4.72k
    {
159
4.72k
      struct traceback_table_routine name;
160
4.72k
      char *namebuf;
161
162
4.72k
      if ((pos + offset + 2) > len)
163
263
  return -1;
164
4.46k
      name.name_len = bfd_getb16 (buf + pos + offset);
165
4.46k
      offset += 2;
166
167
4.46k
      if (name.name_len > 4096)
168
715
  return -1;
169
170
3.74k
      if ((pos + offset + name.name_len) > len)
171
306
  return -1;
172
173
3.44k
      namebuf = bfd_alloc (abfd, name.name_len + 1);
174
3.44k
      if (namebuf == NULL)
175
0
  return -1;
176
177
3.44k
      memcpy (namebuf, buf + pos + offset, name.name_len);
178
3.44k
      namebuf[name.name_len] = '\0';
179
180
      /* Strip leading period inserted by compiler.  */
181
3.44k
      if (namebuf[0] == '.')
182
179
  memmove (namebuf, namebuf + 1, name.name_len);
183
184
3.44k
      sym->name = namebuf;
185
186
5.70k
      for (s = sym->name; (*s != '\0'); s++)
187
2.82k
  if (! ISPRINT (*s))
188
566
    return -1;
189
190
2.87k
      offset += name.name_len;
191
2.87k
    }
192
193
2.87k
  if (table.flags2 & TB_USES_ALLOCA)
194
1.96k
    offset += 4;
195
196
2.87k
  if (table.flags4 & TB_HAS_VEC_INFO)
197
579
    offset += 4;
198
199
2.87k
  if (file != NULL)
200
0
    fprintf (file, " [length = 0x%lx]", (unsigned long) offset);
201
202
2.87k
  return offset;
203
4.72k
}
204
205
static void
206
bfd_pef_print_symbol (bfd *abfd,
207
          void * afile,
208
          asymbol *symbol,
209
          bfd_print_symbol_type how)
210
0
{
211
0
  FILE *file = (FILE *) afile;
212
0
  const char *symname = (symbol->name != bfd_symbol_error_name
213
0
       ? symbol->name : _("<corrupt>"));
214
215
0
  switch (how)
216
0
    {
217
0
    case bfd_print_symbol_name:
218
0
      fprintf (file, "%s", symname);
219
0
      break;
220
0
    default:
221
0
      bfd_print_symbol_vandf (abfd, (void *) file, symbol);
222
0
      fprintf (file, " %-5s %s", symbol->section->name, symname);
223
0
      if (startswith (symname, "__traceback_"))
224
0
  {
225
0
    unsigned char *buf;
226
0
    size_t offset = symbol->value + 4;
227
0
    size_t len = symbol->udata.i;
228
229
0
    buf = bfd_malloc (len);
230
0
    if (buf == NULL
231
0
        || !bfd_get_section_contents (abfd, symbol->section, buf,
232
0
              offset, len)
233
0
        || bfd_pef_parse_traceback_table (abfd, symbol->section, buf,
234
0
            len, 0, NULL, file) < 0)
235
0
      fprintf (file, " [ERROR]");
236
0
    free (buf);
237
0
  }
238
0
    }
239
0
}
240
241
static void
242
bfd_pef_convert_architecture (unsigned long architecture,
243
            enum bfd_architecture *type,
244
            unsigned long *subtype)
245
906
{
246
906
  const unsigned long ARCH_POWERPC = 0x70777063; /* 'pwpc'.  */
247
906
  const unsigned long ARCH_M68K = 0x6d36386b; /* 'm68k'.  */
248
249
906
  *subtype = bfd_arch_unknown;
250
906
  *type = bfd_arch_unknown;
251
252
906
  if (architecture == ARCH_POWERPC)
253
94
    *type = bfd_arch_powerpc;
254
812
  else if (architecture == ARCH_M68K)
255
715
    *type = bfd_arch_m68k;
256
906
}
257
258
static bool
259
bfd_pef_mkobject (bfd *abfd ATTRIBUTE_UNUSED)
260
131
{
261
131
  return true;
262
131
}
263
264
static const char *bfd_pef_section_name (bfd_pef_section *section)
265
35.1k
{
266
35.1k
  switch (section->section_kind)
267
35.1k
    {
268
14.4k
    case BFD_PEF_SECTION_CODE: return "code";
269
1.00k
    case BFD_PEF_SECTION_UNPACKED_DATA: return "unpacked-data";
270
567
    case BFD_PEF_SECTION_PACKED_DATA: return "packed-data";
271
268
    case BFD_PEF_SECTION_CONSTANT: return "constant";
272
584
    case BFD_PEF_SECTION_LOADER: return "loader";
273
252
    case BFD_PEF_SECTION_DEBUG: return "debug";
274
122
    case BFD_PEF_SECTION_EXEC_DATA: return "exec-data";
275
140
    case BFD_PEF_SECTION_EXCEPTION: return "exception";
276
239
    case BFD_PEF_SECTION_TRACEBACK: return "traceback";
277
17.5k
    default: return "unknown";
278
35.1k
    }
279
35.1k
}
280
281
static unsigned long bfd_pef_section_flags (bfd_pef_section *section)
282
35.1k
{
283
35.1k
  switch (section->section_kind)
284
35.1k
    {
285
14.4k
    case BFD_PEF_SECTION_CODE:
286
14.4k
      return SEC_HAS_CONTENTS | SEC_LOAD | SEC_ALLOC | SEC_CODE;
287
1.00k
    case BFD_PEF_SECTION_UNPACKED_DATA:
288
1.57k
    case BFD_PEF_SECTION_PACKED_DATA:
289
1.84k
    case BFD_PEF_SECTION_CONSTANT:
290
2.42k
    case BFD_PEF_SECTION_LOADER:
291
2.67k
    case BFD_PEF_SECTION_DEBUG:
292
2.79k
    case BFD_PEF_SECTION_EXEC_DATA:
293
2.93k
    case BFD_PEF_SECTION_EXCEPTION:
294
3.17k
    case BFD_PEF_SECTION_TRACEBACK:
295
20.7k
    default:
296
20.7k
      return SEC_HAS_CONTENTS | SEC_LOAD | SEC_ALLOC;
297
35.1k
    }
298
35.1k
}
299
300
static asection *
301
bfd_pef_make_bfd_section (bfd *abfd, bfd_pef_section *section)
302
35.1k
{
303
35.1k
  asection *bfdsec;
304
35.1k
  const char *name = bfd_pef_section_name (section);
305
306
35.1k
  bfdsec = bfd_make_section_anyway (abfd, name);
307
35.1k
  if (bfdsec == NULL)
308
0
    return NULL;
309
310
35.1k
  bfdsec->vma = section->default_address + section->container_offset;
311
35.1k
  bfdsec->lma = section->default_address + section->container_offset;
312
35.1k
  bfdsec->size = section->container_length;
313
35.1k
  bfdsec->filepos = section->container_offset;
314
35.1k
  bfdsec->alignment_power = section->alignment;
315
316
35.1k
  bfdsec->flags = bfd_pef_section_flags (section);
317
318
35.1k
  return bfdsec;
319
35.1k
}
320
321
int
322
bfd_pef_parse_loader_header (bfd *abfd ATTRIBUTE_UNUSED,
323
           unsigned char *buf,
324
           size_t len,
325
           bfd_pef_loader_header *header)
326
254
{
327
254
  BFD_ASSERT (len == 56);
328
329
254
  header->main_section = bfd_getb32 (buf);
330
254
  header->main_offset = bfd_getb32 (buf + 4);
331
254
  header->init_section = bfd_getb32 (buf + 8);
332
254
  header->init_offset = bfd_getb32 (buf + 12);
333
254
  header->term_section = bfd_getb32 (buf + 16);
334
254
  header->term_offset = bfd_getb32 (buf + 20);
335
254
  header->imported_library_count = bfd_getb32 (buf + 24);
336
254
  header->total_imported_symbol_count = bfd_getb32 (buf + 28);
337
254
  header->reloc_section_count = bfd_getb32 (buf + 32);
338
254
  header->reloc_instr_offset = bfd_getb32 (buf + 36);
339
254
  header->loader_strings_offset = bfd_getb32 (buf + 40);
340
254
  header->export_hash_offset = bfd_getb32 (buf + 44);
341
254
  header->export_hash_table_power = bfd_getb32 (buf + 48);
342
254
  header->exported_symbol_count = bfd_getb32 (buf + 52);
343
344
254
  return 0;
345
254
}
346
347
int
348
bfd_pef_parse_imported_library (bfd *abfd ATTRIBUTE_UNUSED,
349
        unsigned char *buf,
350
        size_t len,
351
        bfd_pef_imported_library *header)
352
537
{
353
537
  BFD_ASSERT (len == 24);
354
355
537
  header->name_offset = bfd_getb32 (buf);
356
537
  header->old_implementation_version = bfd_getb32 (buf + 4);
357
537
  header->current_version = bfd_getb32 (buf + 8);
358
537
  header->imported_symbol_count = bfd_getb32 (buf + 12);
359
537
  header->first_imported_symbol = bfd_getb32 (buf + 16);
360
537
  header->options = buf[20];
361
537
  header->reserved_a = buf[21];
362
537
  header->reserved_b = bfd_getb16 (buf + 22);
363
364
537
  return 0;
365
537
}
366
367
int
368
bfd_pef_parse_imported_symbol (bfd *abfd ATTRIBUTE_UNUSED,
369
             unsigned char *buf,
370
             size_t len,
371
             bfd_pef_imported_symbol *symbol)
372
2.85k
{
373
2.85k
  unsigned long value;
374
375
2.85k
  BFD_ASSERT (len == 4);
376
377
2.85k
  value = bfd_getb32 (buf);
378
2.85k
  symbol->symbol_class = value >> 24;
379
2.85k
  symbol->name = value & 0x00ffffff;
380
381
2.85k
  return 0;
382
2.85k
}
383
384
int
385
bfd_pef_scan_section (bfd *abfd, bfd_pef_section *section)
386
35.3k
{
387
35.3k
  unsigned char buf[28];
388
389
35.3k
  if (bfd_seek (abfd, section->header_offset, SEEK_SET) != 0
390
35.3k
      || bfd_read (buf, 28, abfd) != 28)
391
219
    return -1;
392
393
35.1k
  section->name_offset = bfd_h_get_32 (abfd, buf);
394
35.1k
  section->default_address = bfd_h_get_32 (abfd, buf + 4);
395
35.1k
  section->total_length = bfd_h_get_32 (abfd, buf + 8);
396
35.1k
  section->unpacked_length = bfd_h_get_32 (abfd, buf + 12);
397
35.1k
  section->container_length = bfd_h_get_32 (abfd, buf + 16);
398
35.1k
  section->container_offset = bfd_h_get_32 (abfd, buf + 20);
399
35.1k
  section->section_kind = buf[24];
400
35.1k
  section->share_kind = buf[25];
401
35.1k
  section->alignment = buf[26];
402
35.1k
  section->reserved = buf[27];
403
404
35.1k
  section->bfd_section = bfd_pef_make_bfd_section (abfd, section);
405
35.1k
  if (section->bfd_section == NULL)
406
0
    return -1;
407
408
35.1k
  return 0;
409
35.1k
}
410
411
void
412
bfd_pef_print_loader_header (bfd *abfd ATTRIBUTE_UNUSED,
413
           bfd_pef_loader_header *header,
414
           FILE *file)
415
0
{
416
0
  fprintf (file, "main_section: %ld\n", header->main_section);
417
0
  fprintf (file, "main_offset: %lu\n", header->main_offset);
418
0
  fprintf (file, "init_section: %ld\n", header->init_section);
419
0
  fprintf (file, "init_offset: %lu\n", header->init_offset);
420
0
  fprintf (file, "term_section: %ld\n", header->term_section);
421
0
  fprintf (file, "term_offset: %lu\n", header->term_offset);
422
0
  fprintf (file, "imported_library_count: %lu\n",
423
0
     header->imported_library_count);
424
0
  fprintf (file, "total_imported_symbol_count: %lu\n",
425
0
     header->total_imported_symbol_count);
426
0
  fprintf (file, "reloc_section_count: %lu\n", header->reloc_section_count);
427
0
  fprintf (file, "reloc_instr_offset: %lu\n", header->reloc_instr_offset);
428
0
  fprintf (file, "loader_strings_offset: %lu\n",
429
0
     header->loader_strings_offset);
430
0
  fprintf (file, "export_hash_offset: %lu\n", header->export_hash_offset);
431
0
  fprintf (file, "export_hash_table_power: %lu\n",
432
0
     header->export_hash_table_power);
433
0
  fprintf (file, "exported_symbol_count: %lu\n",
434
0
     header->exported_symbol_count);
435
0
}
436
437
int
438
bfd_pef_print_loader_section (bfd *abfd, FILE *file)
439
0
{
440
0
  bfd_pef_loader_header header;
441
0
  asection *loadersec = NULL;
442
0
  unsigned char *loaderbuf = NULL;
443
0
  size_t loaderlen = 0;
444
445
0
  loadersec = bfd_get_section_by_name (abfd, "loader");
446
0
  if (loadersec == NULL)
447
0
    return -1;
448
449
0
  loaderlen = loadersec->size;
450
0
  if (loaderlen < 56)
451
0
    return -1;
452
0
  if (bfd_seek (abfd, loadersec->filepos, SEEK_SET) != 0)
453
0
    return -1;
454
0
  loaderbuf = _bfd_malloc_and_read (abfd, loaderlen, loaderlen);
455
0
  if (loaderbuf == NULL)
456
0
    return -1;
457
458
0
  if (bfd_pef_parse_loader_header (abfd, loaderbuf, 56, &header) < 0)
459
0
    {
460
0
      free (loaderbuf);
461
0
      return -1;
462
0
    }
463
464
0
  bfd_pef_print_loader_header (abfd, &header, file);
465
0
  return 0;
466
0
}
467
468
int
469
bfd_pef_scan_start_address (bfd *abfd)
470
590
{
471
590
  bfd_pef_loader_header header;
472
590
  asection *section;
473
474
590
  asection *loadersec = NULL;
475
590
  unsigned char *loaderbuf = NULL;
476
590
  size_t loaderlen = 0;
477
590
  int ret;
478
479
590
  loadersec = bfd_get_section_by_name (abfd, "loader");
480
590
  if (loadersec == NULL)
481
306
    goto end;
482
483
284
  loaderlen = loadersec->size;
484
284
  if (loaderlen < 56)
485
11
    goto wrong;
486
273
  if (bfd_seek (abfd, loadersec->filepos, SEEK_SET) != 0)
487
0
    goto error;
488
273
  loaderbuf = _bfd_malloc_and_read (abfd, loaderlen, loaderlen);
489
273
  if (loaderbuf == NULL)
490
127
    goto error;
491
492
146
  ret = bfd_pef_parse_loader_header (abfd, loaderbuf, 56, &header);
493
146
  if (ret < 0)
494
0
    goto wrong;
495
496
146
  if (header.main_section < 0)
497
0
    goto end;
498
499
4.89k
  for (section = abfd->sections; section != NULL; section = section->next)
500
4.79k
    if ((long) (section->index + 1) == header.main_section)
501
46
      break;
502
503
146
  if (section == NULL)
504
100
    goto wrong;
505
506
46
  abfd->start_address = section->vma + header.main_offset;
507
508
352
 end:
509
352
  free (loaderbuf);
510
352
  return 0;
511
512
111
 wrong:
513
111
  bfd_set_error (bfd_error_wrong_format);
514
238
 error:
515
238
  free (loaderbuf);
516
238
  return -1;
517
111
}
518
519
int
520
bfd_pef_scan (bfd *abfd,
521
        bfd_pef_header *header,
522
        bfd_pef_data_struct *mdata)
523
906
{
524
906
  unsigned int i;
525
906
  enum bfd_architecture cputype;
526
906
  unsigned long cpusubtype;
527
528
906
  mdata->header = *header;
529
530
906
  bfd_pef_convert_architecture (header->architecture, &cputype, &cpusubtype);
531
906
  if (cputype == bfd_arch_unknown)
532
97
    {
533
97
      _bfd_error_handler (_("bfd_pef_scan: unknown architecture 0x%lx"),
534
97
        header->architecture);
535
97
      return -1;
536
97
    }
537
809
  bfd_set_arch_mach (abfd, cputype, cpusubtype);
538
539
809
  mdata->header = *header;
540
541
809
  abfd->flags = (abfd->xvec->object_flags
542
809
     | (abfd->flags & (BFD_IN_MEMORY | BFD_IO_FUNCS)));
543
544
809
  if (header->section_count != 0)
545
805
    {
546
805
      mdata->sections = bfd_alloc (abfd, header->section_count * sizeof (bfd_pef_section));
547
548
805
      if (mdata->sections == NULL)
549
0
  return -1;
550
551
35.9k
      for (i = 0; i < header->section_count; i++)
552
35.3k
  {
553
35.3k
    bfd_pef_section *cur = &mdata->sections[i];
554
35.3k
    cur->header_offset = 40 + (i * 28);
555
35.3k
    if (bfd_pef_scan_section (abfd, cur) < 0)
556
219
      return -1;
557
35.3k
  }
558
805
    }
559
560
590
  if (bfd_pef_scan_start_address (abfd) < 0)
561
238
    return -1;
562
563
352
  return 0;
564
590
}
565
566
static int
567
bfd_pef_read_header (bfd *abfd, bfd_pef_header *header)
568
97.3k
{
569
97.3k
  unsigned char buf[40];
570
571
97.3k
  if (bfd_seek (abfd, 0, SEEK_SET) != 0
572
97.3k
      || bfd_read (buf, 40, abfd) != 40)
573
4.57k
    return -1;
574
575
92.8k
  header->tag1 = bfd_getb32 (buf);
576
92.8k
  header->tag2 = bfd_getb32 (buf + 4);
577
92.8k
  header->architecture = bfd_getb32 (buf + 8);
578
92.8k
  header->format_version = bfd_getb32 (buf + 12);
579
92.8k
  header->timestamp = bfd_getb32 (buf + 16);
580
92.8k
  header->old_definition_version = bfd_getb32 (buf + 20);
581
92.8k
  header->old_implementation_version = bfd_getb32 (buf + 24);
582
92.8k
  header->current_version = bfd_getb32 (buf + 28);
583
92.8k
  header->section_count = bfd_getb32 (buf + 32) + 1;
584
92.8k
  header->instantiated_section_count = bfd_getb32 (buf + 34);
585
92.8k
  header->reserved = bfd_getb32 (buf + 36);
586
587
92.8k
  return 0;
588
97.3k
}
589
590
static bfd_cleanup
591
bfd_pef_object_p (bfd *abfd)
592
97.3k
{
593
97.3k
  bfd_pef_header header;
594
97.3k
  bfd_pef_data_struct *mdata;
595
596
97.3k
  if (bfd_pef_read_header (abfd, &header) != 0)
597
4.57k
    return NULL;
598
599
92.8k
  if (header.tag1 != BFD_PEF_TAG1 || header.tag2 != BFD_PEF_TAG2)
600
91.8k
    {
601
91.8k
      bfd_set_error (bfd_error_wrong_format);
602
91.8k
      return NULL;
603
91.8k
    }
604
605
906
  mdata = (bfd_pef_data_struct *) bfd_zalloc (abfd, sizeof (*mdata));
606
906
  if (mdata == NULL)
607
0
    return NULL;
608
609
906
  if (bfd_pef_scan (abfd, &header, mdata))
610
554
    {
611
554
      bfd_release (abfd, mdata);
612
554
      return NULL;
613
554
    }
614
615
352
  abfd->tdata.pef_data = mdata;
616
352
  return _bfd_no_cleanup;
617
906
}
618
619
static int
620
bfd_pef_parse_traceback_tables (bfd *abfd,
621
        asection *sec,
622
        unsigned char *buf,
623
        size_t len,
624
        long *nsym,
625
        asymbol **csym)
626
969
{
627
969
  char *name;
628
629
969
  asymbol function;
630
969
  asymbol traceback;
631
632
969
  const char *const tbprefix = "__traceback_";
633
969
  size_t tbnamelen;
634
635
969
  size_t pos = 0;
636
969
  unsigned long count = 0;
637
969
  int ret;
638
639
969
  for (;;)
640
183k
    {
641
      /* We're reading symbols two at a time.  */
642
183k
      if (csym && ((csym[count] == NULL) || (csym[count + 1] == NULL)))
643
323
  break;
644
645
183k
      pos += 3;
646
183k
      pos -= (pos % 4);
647
648
597k
      while ((pos + 4) <= len)
649
596k
  {
650
596k
    if (bfd_getb32 (buf + pos) == 0)
651
182k
      break;
652
413k
    pos += 4;
653
413k
  }
654
655
183k
      if ((pos + 4) > len)
656
646
  break;
657
658
182k
      ret = bfd_pef_parse_traceback_table (abfd, sec, buf, len, pos + 4,
659
182k
             &function, 0);
660
182k
      if (ret < 0)
661
179k
  {
662
    /* Skip over 0x0L to advance to next possible traceback table.  */
663
179k
    pos += 4;
664
179k
    continue;
665
179k
  }
666
667
2.87k
      BFD_ASSERT (function.name != NULL);
668
669
      /* Don't bother to compute the name if we are just
670
   counting symbols.  */
671
2.87k
      if (csym)
672
958
  {
673
958
    tbnamelen = strlen (tbprefix) + strlen (function.name);
674
958
    name = bfd_alloc (abfd, tbnamelen + 1);
675
958
    if (name == NULL)
676
0
      {
677
0
        bfd_release (abfd, (void *) function.name);
678
0
        function.name = NULL;
679
0
        break;
680
0
      }
681
958
    snprintf (name, tbnamelen + 1, "%s%s", tbprefix, function.name);
682
958
    traceback.name = name;
683
958
    traceback.value = pos;
684
958
    traceback.the_bfd = abfd;
685
958
    traceback.section = sec;
686
958
    traceback.flags = 0;
687
958
    traceback.udata.i = ret;
688
689
958
    *(csym[count]) = function;
690
958
    *(csym[count + 1]) = traceback;
691
958
  }
692
693
2.87k
      pos += ret;
694
2.87k
      count += 2;
695
2.87k
    }
696
697
969
  *nsym = count;
698
969
  return 0;
699
969
}
700
701
static int
702
bfd_pef_parse_function_stub (bfd *abfd ATTRIBUTE_UNUSED,
703
           unsigned char *buf,
704
           size_t len,
705
           unsigned long *offset)
706
0
{
707
0
  BFD_ASSERT (len == 24);
708
709
0
  if ((bfd_getb32 (buf) & 0xffff0000) != 0x81820000)
710
0
    return -1;
711
0
  if (bfd_getb32 (buf + 4) != 0x90410014)
712
0
    return -1;
713
0
  if (bfd_getb32 (buf + 8) != 0x800c0000)
714
0
    return -1;
715
0
  if (bfd_getb32 (buf + 12) != 0x804c0004)
716
0
    return -1;
717
0
  if (bfd_getb32 (buf + 16) != 0x7c0903a6)
718
0
    return -1;
719
0
  if (bfd_getb32 (buf + 20) != 0x4e800420)
720
0
    return -1;
721
722
0
  if (offset != NULL)
723
0
    *offset = (bfd_getb32 (buf) & 0x0000ffff) / 4;
724
725
0
  return 0;
726
0
}
727
728
static int
729
bfd_pef_parse_function_stubs (bfd *abfd,
730
            asection *codesec,
731
            unsigned char *codebuf,
732
            size_t codelen,
733
            unsigned char *loaderbuf,
734
            size_t loaderlen,
735
            unsigned long *nsym,
736
            asymbol **csym)
737
108
{
738
108
  const char *const sprefix = "__stub_";
739
108
  size_t codepos = 0;
740
108
  unsigned long count = 0;
741
108
  bfd_pef_loader_header header;
742
108
  bfd_pef_imported_library *libraries = NULL;
743
108
  bfd_pef_imported_symbol *imports = NULL;
744
108
  unsigned long i;
745
108
  int ret;
746
747
108
  if (loaderlen < 56)
748
0
    goto error;
749
750
108
  ret = bfd_pef_parse_loader_header (abfd, loaderbuf, 56, &header);
751
108
  if (ret < 0)
752
0
    goto error;
753
754
108
  if ((loaderlen - 56) / 24 < header.imported_library_count)
755
18
    goto error;
756
757
90
  if ((loaderlen - 56 - header.imported_library_count * 24) / 4
758
90
      < header.total_imported_symbol_count)
759
15
    goto error;
760
761
75
  libraries = bfd_malloc
762
75
    (header.imported_library_count * sizeof (bfd_pef_imported_library));
763
75
  imports = bfd_malloc
764
75
    (header.total_imported_symbol_count * sizeof (bfd_pef_imported_symbol));
765
75
  if (libraries == NULL || imports == NULL)
766
0
    goto error;
767
768
612
  for (i = 0; i < header.imported_library_count; i++)
769
537
    {
770
537
      ret = bfd_pef_parse_imported_library
771
537
  (abfd, loaderbuf + 56 + (i * 24), 24, &libraries[i]);
772
537
      if (ret < 0)
773
0
  goto error;
774
537
    }
775
776
2.93k
  for (i = 0; i < header.total_imported_symbol_count; i++)
777
2.85k
    {
778
2.85k
      ret = (bfd_pef_parse_imported_symbol
779
2.85k
       (abfd,
780
2.85k
        loaderbuf + 56 + (header.imported_library_count * 24) + (i * 4),
781
2.85k
        4, &imports[i]));
782
2.85k
      if (ret < 0)
783
0
  goto error;
784
2.85k
    }
785
786
75
  codepos = 0;
787
788
75
  for (;;)
789
75
    {
790
75
      asymbol sym;
791
75
      const char *symname;
792
75
      char *name;
793
75
      unsigned long sym_index;
794
795
75
      if (csym && (csym[count] == NULL))
796
25
  break;
797
798
50
      codepos += 3;
799
50
      codepos -= (codepos % 4);
800
801
4.68k
      while ((codepos + 4) <= codelen)
802
4.63k
  {
803
4.63k
    if ((bfd_getb32 (codebuf + codepos) & 0xffff0000) == 0x81820000)
804
0
      break;
805
4.63k
    codepos += 4;
806
4.63k
  }
807
808
50
      if ((codepos + 24) > codelen)
809
50
  break;
810
811
0
      ret = bfd_pef_parse_function_stub (abfd, codebuf + codepos, 24, &sym_index);
812
0
      if (ret < 0)
813
0
  {
814
0
    codepos += 24;
815
0
    continue;
816
0
  }
817
818
0
      if (sym_index >= header.total_imported_symbol_count)
819
0
  {
820
0
    codepos += 24;
821
0
    continue;
822
0
  }
823
824
0
      {
825
0
  size_t max, namelen;
826
0
  const char *s;
827
828
0
  if (loaderlen < (header.loader_strings_offset + imports[sym_index].name))
829
0
    goto error;
830
831
0
  max = loaderlen - (header.loader_strings_offset + imports[sym_index].name);
832
0
  symname = (char *) loaderbuf;
833
0
  symname += header.loader_strings_offset + imports[sym_index].name;
834
0
  namelen = 0;
835
0
  for (s = symname; s < (symname + max); s++)
836
0
    {
837
0
      if (*s == '\0')
838
0
        break;
839
0
      if (! ISPRINT (*s))
840
0
        goto error;
841
0
      namelen++;
842
0
    }
843
0
  if (*s != '\0')
844
0
    goto error;
845
846
0
  name = bfd_alloc (abfd, strlen (sprefix) + namelen + 1);
847
0
  if (name == NULL)
848
0
    break;
849
850
0
  snprintf (name, strlen (sprefix) + namelen + 1, "%s%s",
851
0
      sprefix, symname);
852
0
  sym.name = name;
853
0
      }
854
855
0
      sym.value = codepos;
856
0
      sym.the_bfd = abfd;
857
0
      sym.section = codesec;
858
0
      sym.flags = 0;
859
0
      sym.udata.i = 0;
860
861
0
      codepos += 24;
862
863
0
      if (csym != NULL)
864
0
  *(csym[count]) = sym;
865
866
0
      count++;
867
0
    }
868
869
75
  goto end;
870
871
75
 end:
872
75
  free (libraries);
873
75
  free (imports);
874
75
  *nsym = count;
875
75
  return 0;
876
877
33
 error:
878
33
  free (libraries);
879
33
  free (imports);
880
33
  *nsym = count;
881
33
  return -1;
882
75
}
883
884
static long
885
bfd_pef_parse_symbols (bfd *abfd, asymbol **csym)
886
1.08k
{
887
1.08k
  unsigned long count = 0;
888
889
1.08k
  asection *codesec = NULL;
890
1.08k
  unsigned char *codebuf = NULL;
891
1.08k
  size_t codelen = 0;
892
893
1.08k
  asection *loadersec = NULL;
894
1.08k
  unsigned char *loaderbuf = NULL;
895
1.08k
  size_t loaderlen = 0;
896
897
1.08k
  codesec = bfd_get_section_by_name (abfd, "code");
898
1.08k
  if (codesec != NULL)
899
1.07k
    {
900
1.07k
      codelen = codesec->size;
901
1.07k
      if (bfd_seek (abfd, codesec->filepos, SEEK_SET) != 0)
902
0
  goto end;
903
1.07k
      codebuf = _bfd_malloc_and_read (abfd, codelen, codelen);
904
1.07k
      if (codebuf == NULL)
905
108
  goto end;
906
1.07k
    }
907
908
972
  loadersec = bfd_get_section_by_name (abfd, "loader");
909
972
  if (loadersec != NULL)
910
108
    {
911
108
      loaderlen = loadersec->size;
912
108
      if (bfd_seek (abfd, loadersec->filepos, SEEK_SET) != 0)
913
0
  goto end;
914
108
      loaderbuf = _bfd_malloc_and_read (abfd, loaderlen, loaderlen);
915
108
      if (loaderbuf == NULL)
916
0
  goto end;
917
108
    }
918
919
972
  count = 0;
920
972
  if (codesec != NULL)
921
969
    {
922
969
      long ncount = 0;
923
969
      bfd_pef_parse_traceback_tables (abfd, codesec, codebuf, codelen,
924
969
              &ncount, csym);
925
969
      count += ncount;
926
969
    }
927
928
972
  if ((codesec != NULL) && (loadersec != NULL))
929
108
    {
930
108
      unsigned long ncount = 0;
931
108
      bfd_pef_parse_function_stubs
932
108
  (abfd, codesec, codebuf, codelen, loaderbuf, loaderlen, &ncount,
933
108
   (csym != NULL) ? (csym + count) : NULL);
934
108
      count += ncount;
935
108
    }
936
937
972
  if (csym != NULL)
938
324
    csym[count] = NULL;
939
940
1.08k
 end:
941
1.08k
  free (codebuf);
942
1.08k
  free (loaderbuf);
943
1.08k
  return count;
944
972
}
945
946
static long
947
bfd_pef_count_symbols (bfd *abfd)
948
720
{
949
720
  return bfd_pef_parse_symbols (abfd, NULL);
950
720
}
951
952
static long
953
bfd_pef_get_symtab_upper_bound (bfd *abfd)
954
360
{
955
360
  long nsyms = bfd_pef_count_symbols (abfd);
956
957
360
  if (nsyms < 0)
958
0
    return nsyms;
959
360
  return ((nsyms + 1) * sizeof (asymbol *));
960
360
}
961
962
static long
963
bfd_pef_canonicalize_symtab (bfd *abfd, asymbol **alocation)
964
360
{
965
360
  long i;
966
360
  asymbol *syms;
967
360
  long ret;
968
360
  long nsyms = bfd_pef_count_symbols (abfd);
969
970
360
  if (nsyms < 0)
971
0
    return nsyms;
972
973
360
  syms = bfd_alloc (abfd, nsyms * sizeof (asymbol));
974
360
  if (syms == NULL)
975
0
    return -1;
976
977
2.27k
  for (i = 0; i < nsyms; i++)
978
1.91k
    alocation[i] = &syms[i];
979
980
360
  alocation[nsyms] = NULL;
981
982
360
  ret = bfd_pef_parse_symbols (abfd, alocation);
983
360
  if (ret != nsyms)
984
0
    return 0;
985
986
360
  return ret;
987
360
}
988
989
#define bfd_pef_make_empty_symbol _bfd_generic_make_empty_symbol
990
991
static void
992
bfd_pef_get_symbol_info (bfd *abfd ATTRIBUTE_UNUSED,
993
       asymbol *symbol,
994
       symbol_info *ret)
995
232
{
996
232
  bfd_symbol_info (symbol, ret);
997
232
}
998
999
static int
1000
bfd_pef_sizeof_headers (bfd *abfd ATTRIBUTE_UNUSED,
1001
      struct bfd_link_info *info ATTRIBUTE_UNUSED)
1002
0
{
1003
0
  return 0;
1004
0
}
1005
1006
const bfd_target pef_vec =
1007
{
1008
  "pef",      /* Name.  */
1009
  bfd_target_pef_flavour, /* Flavour.  */
1010
  BFD_ENDIAN_BIG,   /* Byteorder.  */
1011
  BFD_ENDIAN_BIG,   /* Header_byteorder.  */
1012
  (HAS_RELOC | EXEC_P |   /* Object flags.  */
1013
   HAS_LINENO | HAS_DEBUG |
1014
   HAS_SYMS | HAS_LOCALS | DYNAMIC | WP_TEXT | D_PAGED),
1015
  (SEC_ALLOC | SEC_LOAD | SEC_READONLY | SEC_CODE | SEC_DATA
1016
   | SEC_ROM | SEC_HAS_CONTENTS), /* Section_flags.  */
1017
  0,        /* Symbol_leading_char.  */
1018
  ' ',        /* AR_pad_char.  */
1019
  16,       /* AR_max_namelen.  */
1020
  0,        /* match priority.  */
1021
  TARGET_KEEP_UNUSED_SECTION_SYMBOLS, /* keep unused section symbols.  */
1022
  false,      /* merge sections */
1023
  bfd_getb64, bfd_getb_signed_64, bfd_putb64,
1024
  bfd_getb32, bfd_getb_signed_32, bfd_putb32,
1025
  bfd_getb16, bfd_getb_signed_16, bfd_putb16, /* Data.  */
1026
  bfd_getb64, bfd_getb_signed_64, bfd_putb64,
1027
  bfd_getb32, bfd_getb_signed_32, bfd_putb32,
1028
  bfd_getb16, bfd_getb_signed_16, bfd_putb16, /* Headers.  */
1029
  {       /* bfd_check_format.  */
1030
    _bfd_dummy_target,
1031
    bfd_pef_object_p,   /* bfd_check_format.  */
1032
    _bfd_dummy_target,
1033
    _bfd_dummy_target,
1034
  },
1035
  {       /* bfd_set_format.  */
1036
    _bfd_bool_bfd_false_error,
1037
    bfd_pef_mkobject,
1038
    _bfd_bool_bfd_false_error,
1039
    _bfd_bool_bfd_false_error,
1040
  },
1041
  {       /* bfd_write_contents.  */
1042
    _bfd_bool_bfd_false_error,
1043
    _bfd_bool_bfd_true,
1044
    _bfd_bool_bfd_false_error,
1045
    _bfd_bool_bfd_false_error,
1046
  },
1047
1048
  BFD_JUMP_TABLE_GENERIC (bfd_pef),
1049
  BFD_JUMP_TABLE_COPY (_bfd_generic),
1050
  BFD_JUMP_TABLE_CORE (_bfd_nocore),
1051
  BFD_JUMP_TABLE_ARCHIVE (_bfd_noarchive),
1052
  BFD_JUMP_TABLE_SYMBOLS (bfd_pef),
1053
  BFD_JUMP_TABLE_RELOCS (_bfd_norelocs),
1054
  BFD_JUMP_TABLE_WRITE (bfd_pef),
1055
  BFD_JUMP_TABLE_LINK (bfd_pef),
1056
  BFD_JUMP_TABLE_DYNAMIC (_bfd_nodynamic),
1057
1058
  NULL,
1059
1060
  NULL
1061
};
1062
1063
#define bfd_pef_xlib_close_and_cleanup        _bfd_generic_close_and_cleanup
1064
#define bfd_pef_xlib_bfd_free_cached_info     _bfd_generic_bfd_free_cached_info
1065
#define bfd_pef_xlib_new_section_hook       _bfd_generic_new_section_hook
1066
#define bfd_pef_xlib_get_section_contents     _bfd_generic_get_section_contents
1067
#define bfd_pef_xlib_set_section_contents     _bfd_generic_set_section_contents
1068
#define bfd_pef_xlib_set_section_contents_in_window _bfd_generic_set_section_contents_in_window
1069
1070
static int
1071
bfd_pef_xlib_read_header (bfd *abfd, bfd_pef_xlib_header *header)
1072
96.8k
{
1073
96.8k
  unsigned char buf[80];
1074
1075
96.8k
  if (bfd_seek (abfd, 0, SEEK_SET) != 0
1076
96.8k
      || bfd_read (buf, sizeof buf, abfd) != sizeof buf)
1077
15.5k
    return -1;
1078
1079
81.2k
  header->tag1 = bfd_getb32 (buf);
1080
81.2k
  header->tag2 = bfd_getb32 (buf + 4);
1081
81.2k
  header->current_format = bfd_getb32 (buf + 8);
1082
81.2k
  header->container_strings_offset = bfd_getb32 (buf + 12);
1083
81.2k
  header->export_hash_offset = bfd_getb32 (buf + 16);
1084
81.2k
  header->export_key_offset = bfd_getb32 (buf + 20);
1085
81.2k
  header->export_symbol_offset = bfd_getb32 (buf + 24);
1086
81.2k
  header->export_names_offset = bfd_getb32 (buf + 28);
1087
81.2k
  header->export_hash_table_power = bfd_getb32 (buf + 32);
1088
81.2k
  header->exported_symbol_count = bfd_getb32 (buf + 36);
1089
81.2k
  header->frag_name_offset = bfd_getb32 (buf + 40);
1090
81.2k
  header->frag_name_length = bfd_getb32 (buf + 44);
1091
81.2k
  header->dylib_path_offset = bfd_getb32 (buf + 48);
1092
81.2k
  header->dylib_path_length = bfd_getb32 (buf + 52);
1093
81.2k
  header->cpu_family = bfd_getb32 (buf + 56);
1094
81.2k
  header->cpu_model = bfd_getb32 (buf + 60);
1095
81.2k
  header->date_time_stamp = bfd_getb32 (buf + 64);
1096
81.2k
  header->current_version = bfd_getb32 (buf + 68);
1097
81.2k
  header->old_definition_version = bfd_getb32 (buf + 72);
1098
81.2k
  header->old_implementation_version = bfd_getb32 (buf + 76);
1099
1100
81.2k
  return 0;
1101
96.8k
}
1102
1103
static int
1104
bfd_pef_xlib_scan (bfd *abfd, bfd_pef_xlib_header *header)
1105
0
{
1106
0
  bfd_pef_xlib_data_struct *mdata = NULL;
1107
1108
0
  mdata = bfd_alloc (abfd, sizeof (* mdata));
1109
0
  if (mdata == NULL)
1110
0
    return -1;
1111
1112
0
  mdata->header = *header;
1113
1114
0
  abfd->flags = (abfd->xvec->object_flags
1115
0
     | (abfd->flags & (BFD_IN_MEMORY | BFD_IO_FUNCS)));
1116
1117
0
  abfd->tdata.pef_xlib_data = mdata;
1118
1119
0
  return 0;
1120
0
}
1121
1122
static bfd_cleanup
1123
bfd_pef_xlib_object_p (bfd *abfd)
1124
96.8k
{
1125
96.8k
  bfd_pef_xlib_header header;
1126
1127
96.8k
  if (bfd_pef_xlib_read_header (abfd, &header) != 0
1128
81.2k
      || header.tag1 != BFD_PEF_XLIB_TAG1
1129
2
      || (header.tag2 != BFD_PEF_VLIB_TAG2
1130
2
    && header.tag2 != BFD_PEF_BLIB_TAG2))
1131
96.8k
    {
1132
96.8k
      if (bfd_get_error () != bfd_error_system_call)
1133
96.8k
  bfd_set_error (bfd_error_wrong_format);
1134
96.8k
      return NULL;
1135
96.8k
    }
1136
1137
0
  if (bfd_pef_xlib_scan (abfd, &header) != 0)
1138
0
    return NULL;
1139
1140
0
  return _bfd_no_cleanup;
1141
0
}
1142
1143
const bfd_target pef_xlib_vec =
1144
{
1145
  "pef-xlib",     /* Name.  */
1146
  bfd_target_pef_xlib_flavour,  /* Flavour.  */
1147
  BFD_ENDIAN_BIG,   /* Byteorder */
1148
  BFD_ENDIAN_BIG,   /* Header_byteorder.  */
1149
  (HAS_RELOC | EXEC_P |   /* Object flags.  */
1150
   HAS_LINENO | HAS_DEBUG |
1151
   HAS_SYMS | HAS_LOCALS | DYNAMIC | WP_TEXT | D_PAGED),
1152
  (SEC_ALLOC | SEC_LOAD | SEC_READONLY | SEC_CODE | SEC_DATA
1153
   | SEC_ROM | SEC_HAS_CONTENTS),/* Section_flags.  */
1154
  0,        /* Symbol_leading_char.  */
1155
  ' ',        /* AR_pad_char.  */
1156
  16,       /* AR_max_namelen.  */
1157
  0,        /* match priority.  */
1158
  TARGET_KEEP_UNUSED_SECTION_SYMBOLS, /* keep unused section symbols.  */
1159
  TARGET_MERGE_SECTIONS,
1160
  bfd_getb64, bfd_getb_signed_64, bfd_putb64,
1161
  bfd_getb32, bfd_getb_signed_32, bfd_putb32,
1162
  bfd_getb16, bfd_getb_signed_16, bfd_putb16, /* Data.  */
1163
  bfd_getb64, bfd_getb_signed_64, bfd_putb64,
1164
  bfd_getb32, bfd_getb_signed_32, bfd_putb32,
1165
  bfd_getb16, bfd_getb_signed_16, bfd_putb16, /* Headers.  */
1166
  {       /* bfd_check_format.  */
1167
    _bfd_dummy_target,
1168
    bfd_pef_xlib_object_p,  /* bfd_check_format.  */
1169
    _bfd_dummy_target,
1170
    _bfd_dummy_target,
1171
  },
1172
  {       /* bfd_set_format.  */
1173
    _bfd_bool_bfd_false_error,
1174
    bfd_pef_mkobject,
1175
    _bfd_bool_bfd_false_error,
1176
    _bfd_bool_bfd_false_error,
1177
  },
1178
  {       /* bfd_write_contents.  */
1179
    _bfd_bool_bfd_false_error,
1180
    _bfd_bool_bfd_true,
1181
    _bfd_bool_bfd_false_error,
1182
    _bfd_bool_bfd_false_error,
1183
  },
1184
1185
  BFD_JUMP_TABLE_GENERIC (bfd_pef_xlib),
1186
  BFD_JUMP_TABLE_COPY (_bfd_generic),
1187
  BFD_JUMP_TABLE_CORE (_bfd_nocore),
1188
  BFD_JUMP_TABLE_ARCHIVE (_bfd_noarchive),
1189
  BFD_JUMP_TABLE_SYMBOLS (_bfd_nosymbols),
1190
  BFD_JUMP_TABLE_RELOCS (_bfd_norelocs),
1191
  BFD_JUMP_TABLE_WRITE (_bfd_nowrite),
1192
  BFD_JUMP_TABLE_LINK (_bfd_nolink),
1193
  BFD_JUMP_TABLE_DYNAMIC (_bfd_nodynamic),
1194
1195
  NULL,
1196
1197
  NULL
1198
};