Coverage Report

Created: 2026-04-04 08:16

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/binutils-gdb/bfd/pdb.c
Line
Count
Source
1
/* BFD back-end for PDB Multi-Stream Format archives.
2
   Copyright (C) 2022-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
/* This describes the MSF file archive format, which is used for the
22
   PDB debug info generated by MSVC. See https://llvm.org/docs/PDB/MsfFile.html
23
   for a full description of the format.  */
24
25
#include "sysdep.h"
26
#include "bfd.h"
27
#include "libbfd.h"
28
29
/* "Microsoft C/C++ MSF 7.00\r\n\x1a\x44\x53\0\0\0" */
30
static const uint8_t pdb_magic[] =
31
{ 0x4d, 0x69, 0x63, 0x72, 0x6f, 0x73, 0x6f, 0x66,
32
  0x74, 0x20, 0x43, 0x2f, 0x43, 0x2b, 0x2b, 0x20,
33
  0x4d, 0x53, 0x46, 0x20, 0x37, 0x2e, 0x30, 0x30,
34
  0x0d, 0x0a, 0x1a, 0x44, 0x53, 0x00, 0x00, 0x00 };
35
36
12.2k
#define arch_eltdata(bfd) ((struct areltdata *) ((bfd)->arelt_data))
37
38
static bfd_cleanup
39
pdb_archive_p (bfd *abfd)
40
84.8k
{
41
84.8k
  int ret;
42
84.8k
  char magic[sizeof (pdb_magic)];
43
44
84.8k
  ret = bfd_read (magic, sizeof (magic), abfd);
45
84.8k
  if (ret != sizeof (magic))
46
1.03k
    {
47
1.03k
      bfd_set_error (bfd_error_wrong_format);
48
1.03k
      return NULL;
49
1.03k
    }
50
51
83.7k
  if (memcmp (magic, pdb_magic, sizeof (magic)))
52
83.6k
    {
53
83.6k
      bfd_set_error (bfd_error_wrong_format);
54
83.6k
      return NULL;
55
83.6k
    }
56
57
99
  void *tdata = bfd_zalloc (abfd, sizeof (struct artdata));
58
99
  if (tdata == NULL)
59
0
    return NULL;
60
99
  bfd_ardata (abfd) = tdata;
61
62
99
  return _bfd_no_cleanup;
63
99
}
64
65
static bfd *
66
pdb_get_elt_at_index (bfd *abfd, symindex sym_index)
67
3.68k
{
68
3.68k
  char int_buf[sizeof (uint32_t)];
69
3.68k
  uint32_t block_size, block_map_addr, block, num_files;
70
3.68k
  uint32_t first_dir_block, dir_offset, file_size, block_off, left;
71
3.68k
  char name[10];
72
3.68k
  bfd *file;
73
3.68k
  char *buf;
74
75
  /* Get block_size.  */
76
77
3.68k
  if (bfd_seek (abfd, sizeof (pdb_magic), SEEK_SET))
78
0
    return NULL;
79
80
3.68k
  if (bfd_read (int_buf, sizeof (uint32_t), abfd) != sizeof (uint32_t))
81
0
    {
82
0
      bfd_set_error (bfd_error_malformed_archive);
83
0
      return NULL;
84
0
    }
85
86
3.68k
  block_size = bfd_getl32 (int_buf);
87
3.68k
  if ((block_size & -block_size) != block_size
88
3.68k
      || block_size < 512
89
3.68k
      || block_size > 4096)
90
1
    {
91
1
      bfd_set_error (bfd_error_malformed_archive);
92
1
      return NULL;
93
1
    }
94
95
  /* Get block_map_addr.  */
96
97
3.68k
  if (bfd_seek (abfd, 4 * sizeof (uint32_t), SEEK_CUR))
98
0
    return NULL;
99
100
3.68k
  if (bfd_read (int_buf, sizeof (uint32_t), abfd) != sizeof (uint32_t))
101
0
    {
102
0
      bfd_set_error (bfd_error_malformed_archive);
103
0
      return NULL;
104
0
    }
105
106
3.68k
  block_map_addr = bfd_getl32 (int_buf);
107
108
  /* Get num_files.  */
109
110
3.68k
  if (bfd_seek (abfd, block_map_addr * block_size, SEEK_SET))
111
0
    return NULL;
112
113
3.68k
  if (bfd_read (int_buf, sizeof (uint32_t), abfd) != sizeof (uint32_t))
114
0
    {
115
0
      bfd_set_error (bfd_error_malformed_archive);
116
0
      return NULL;
117
0
    }
118
119
3.68k
  first_dir_block = bfd_getl32 (int_buf);
120
121
3.68k
  if (bfd_seek (abfd, first_dir_block * block_size, SEEK_SET))
122
0
    return NULL;
123
124
3.68k
  if (bfd_read (int_buf, sizeof (uint32_t), abfd) != sizeof (uint32_t))
125
1
    {
126
1
      bfd_set_error (bfd_error_malformed_archive);
127
1
      return NULL;
128
1
    }
129
130
3.68k
  num_files = bfd_getl32 (int_buf);
131
132
3.68k
  if (sym_index >= num_files)
133
4
    {
134
4
      bfd_set_error (bfd_error_no_more_archived_files);
135
4
      return NULL;
136
4
    }
137
138
  /* Read file size.  */
139
140
3.67k
  dir_offset = sizeof (uint32_t) * (sym_index + 1);
141
142
3.67k
  if (dir_offset >= block_size)
143
406
    {
144
406
      uint32_t block_map_addr_off;
145
146
406
      block_map_addr_off = ((dir_offset / block_size) * sizeof (uint32_t));
147
148
406
      if (bfd_seek (abfd, (block_map_addr * block_size) + block_map_addr_off,
149
406
        SEEK_SET))
150
0
  return NULL;
151
152
406
      if (bfd_read (int_buf, sizeof (uint32_t), abfd) != sizeof (uint32_t))
153
0
  {
154
0
    bfd_set_error (bfd_error_malformed_archive);
155
0
    return NULL;
156
0
  }
157
158
406
      block = bfd_getl32 (int_buf);
159
406
    }
160
3.27k
  else
161
3.27k
    {
162
3.27k
      block = first_dir_block;
163
3.27k
    }
164
165
3.67k
  if (bfd_seek (abfd, (block * block_size) + (dir_offset % block_size),
166
3.67k
    SEEK_SET))
167
0
    return NULL;
168
169
3.67k
  if (bfd_read (int_buf, sizeof (uint32_t), abfd) != sizeof (uint32_t))
170
3
    {
171
3
      bfd_set_error (bfd_error_malformed_archive);
172
3
      return NULL;
173
3
    }
174
175
3.67k
  file_size = bfd_getl32 (int_buf);
176
177
  /* Undocumented? Seen on PDBs created by MSVC 2022.  */
178
3.67k
  if (file_size == 0xffffffff)
179
3.42k
    file_size = 0;
180
181
  /* Create BFD. */
182
183
  /* Four hex digits is enough - even though MSF allows for 32 bits, the
184
     PDB format itself only uses 16 bits for stream numbers.  */
185
3.67k
  sprintf (name, "%04lx", sym_index);
186
187
3.67k
  file = bfd_create (name, abfd);
188
189
3.67k
  if (!file)
190
0
    return NULL;
191
192
3.67k
  if (!bfd_make_writable (file))
193
0
    goto fail;
194
195
3.67k
  file->arelt_data =
196
3.67k
    (struct areltdata *) bfd_zmalloc (sizeof (struct areltdata));
197
198
3.67k
  if (!file->arelt_data)
199
0
    goto fail;
200
201
3.67k
  arch_eltdata (file)->parsed_size = file_size;
202
3.67k
  arch_eltdata (file)->key = sym_index;
203
204
3.67k
  if (file_size == 0)
205
3.55k
    return file;
206
207
120
  block_off = 0;
208
209
  /* Sum number of blocks in previous files.  */
210
211
120
  if (sym_index != 0)
212
99
    {
213
99
      dir_offset = sizeof (uint32_t);
214
215
99
      if (bfd_seek (abfd, (first_dir_block * block_size) + sizeof (uint32_t),
216
99
        SEEK_SET))
217
0
  goto fail;
218
219
3.60k
      for (symindex i = 0; i < sym_index; i++)
220
3.51k
  {
221
3.51k
    uint32_t size, num_blocks;
222
223
3.51k
    if ((dir_offset % block_size) == 0)
224
15
      {
225
15
        uint32_t block_map_addr_off;
226
227
15
        block_map_addr_off =
228
15
    ((dir_offset / block_size) * sizeof (uint32_t));
229
230
15
        if (bfd_seek
231
15
      (abfd, (block_map_addr * block_size) + block_map_addr_off,
232
15
       SEEK_SET))
233
0
    goto fail;
234
235
15
        if (bfd_read (int_buf, sizeof (uint32_t), abfd) !=
236
15
      sizeof (uint32_t))
237
0
    {
238
0
      bfd_set_error (bfd_error_malformed_archive);
239
0
      goto fail;
240
0
    }
241
242
15
        block = bfd_getl32 (int_buf);
243
244
15
        if (bfd_seek (abfd, block * block_size, SEEK_SET))
245
0
    goto fail;
246
15
      }
247
248
3.51k
    if (bfd_read (int_buf, sizeof (uint32_t), abfd) !=
249
3.51k
        sizeof (uint32_t))
250
0
      {
251
0
        bfd_set_error (bfd_error_malformed_archive);
252
0
        goto fail;
253
0
      }
254
255
3.51k
    size = bfd_getl32 (int_buf);
256
257
3.51k
    if (size == 0xffffffff)
258
3.27k
      size = 0;
259
260
3.51k
    num_blocks = (size + block_size - 1) / block_size;
261
3.51k
    block_off += num_blocks;
262
263
3.51k
    dir_offset += sizeof (uint32_t);
264
3.51k
  }
265
99
    }
266
267
  /* Read blocks, and write into new BFD.  */
268
269
120
  dir_offset = sizeof (uint32_t) * (num_files + block_off + 1);
270
271
120
  if (dir_offset >= block_size)
272
44
    {
273
44
      uint32_t block_map_addr_off;
274
275
44
      block_map_addr_off = ((dir_offset / block_size) * sizeof (uint32_t));
276
277
44
      if (bfd_seek (abfd, (block_map_addr * block_size) + block_map_addr_off,
278
44
        SEEK_SET))
279
0
  goto fail;
280
281
44
      if (bfd_read (int_buf, sizeof (uint32_t), abfd) != sizeof (uint32_t))
282
14
  {
283
14
    bfd_set_error (bfd_error_malformed_archive);
284
14
    goto fail;
285
14
  }
286
287
30
      block = bfd_getl32 (int_buf);
288
30
    }
289
76
  else
290
76
    {
291
76
      block = first_dir_block;
292
76
    }
293
294
106
  buf = bfd_malloc (block_size);
295
106
  if (!buf)
296
0
    goto fail;
297
298
106
  left = file_size;
299
106
  do
300
284
    {
301
284
      uint32_t file_block, to_read;
302
303
284
      if ((dir_offset % block_size) == 0 && left != file_size)
304
2
  {
305
2
    uint32_t block_map_addr_off;
306
307
2
    block_map_addr_off =
308
2
      ((dir_offset / block_size) * sizeof (uint32_t));
309
310
2
    if (bfd_seek
311
2
        (abfd, (block_map_addr * block_size) + block_map_addr_off,
312
2
         SEEK_SET))
313
0
      goto fail2;
314
315
2
    if (bfd_read (int_buf, sizeof (uint32_t), abfd) !=
316
2
        sizeof (uint32_t))
317
0
      {
318
0
        bfd_set_error (bfd_error_malformed_archive);
319
0
        goto fail2;
320
0
      }
321
322
2
    block = bfd_getl32 (int_buf);
323
2
  }
324
325
284
      if (bfd_seek (abfd, (block * block_size) + (dir_offset % block_size),
326
284
        SEEK_SET))
327
0
  goto fail2;
328
329
284
      if (bfd_read (int_buf, sizeof (uint32_t), abfd) != sizeof (uint32_t))
330
9
  {
331
9
    bfd_set_error (bfd_error_malformed_archive);
332
9
    goto fail2;
333
9
  }
334
335
275
      file_block = bfd_getl32 (int_buf);
336
337
275
      if (bfd_seek (abfd, file_block * block_size, SEEK_SET))
338
0
  goto fail2;
339
340
275
      to_read = left > block_size ? block_size : left;
341
342
275
      if (bfd_read (buf, to_read, abfd) != to_read)
343
65
  {
344
65
    bfd_set_error (bfd_error_malformed_archive);
345
65
    goto fail2;
346
65
  }
347
348
210
      if (bfd_write (buf, to_read, file) != to_read)
349
0
  goto fail2;
350
351
210
      if (left > block_size)
352
178
  left -= block_size;
353
32
      else
354
32
  break;
355
356
178
      dir_offset += sizeof (uint32_t);
357
178
    }
358
178
  while (left > 0);
359
360
32
  free (buf);
361
362
32
  return file;
363
364
74
fail2:
365
74
  free (buf);
366
367
88
fail:
368
88
  bfd_close (file);
369
88
  return NULL;
370
74
}
371
372
static bfd *
373
pdb_openr_next_archived_file (bfd *archive, bfd *last_file)
374
3.68k
{
375
3.68k
  if (!last_file)
376
97
    return pdb_get_elt_at_index (archive, 0);
377
3.58k
  else
378
3.58k
    return pdb_get_elt_at_index (archive, arch_eltdata (last_file)->key + 1);
379
3.68k
}
380
381
static int
382
pdb_generic_stat_arch_elt (bfd *abfd, struct stat *buf)
383
1.28k
{
384
1.28k
  buf->st_mtime = 0;
385
1.28k
  buf->st_uid = 0;
386
1.28k
  buf->st_gid = 0;
387
1.28k
  buf->st_mode = 0644;
388
1.28k
  buf->st_size = arch_eltdata (abfd)->parsed_size;
389
390
1.28k
  return 0;
391
1.28k
}
392
393
static uint32_t
394
pdb_allocate_block (uint32_t *num_blocks, uint32_t block_size)
395
65
{
396
65
  uint32_t block;
397
398
65
  block = *num_blocks;
399
400
65
  (*num_blocks)++;
401
402
  /* If new interval, skip two blocks for free space map.  */
403
404
65
  if ((block % block_size) == 1)
405
0
    {
406
0
      block += 2;
407
0
      (*num_blocks) += 2;
408
0
    }
409
410
65
  return block;
411
65
}
412
413
static bool
414
pdb_write_directory (bfd *abfd, uint32_t block_size, uint32_t num_files,
415
         uint32_t block_map_addr, uint32_t * num_blocks,
416
         uint32_t *stream0_start)
417
30
{
418
30
  char tmp[sizeof (uint32_t)];
419
30
  uint32_t block, left, block_map_off;
420
30
  bfd *arelt;
421
30
  char *buf;
422
423
  /* Allocate first block for directory.  */
424
425
30
  block = pdb_allocate_block (num_blocks, block_size);
426
30
  left = block_size;
427
428
  /* Write allocated block no. at beginning of block map.  */
429
430
30
  if (bfd_seek (abfd, block_map_addr * block_size, SEEK_SET))
431
0
    return false;
432
433
30
  bfd_putl32 (block, tmp);
434
435
30
  if (bfd_write (tmp, sizeof (uint32_t), abfd) != sizeof (uint32_t))
436
0
    return false;
437
438
30
  block_map_off = sizeof (uint32_t);
439
440
  /* Write num_files at beginning of directory.  */
441
442
30
  if (bfd_seek (abfd, block * block_size, SEEK_SET))
443
0
    return false;
444
445
30
  bfd_putl32 (num_files, tmp);
446
447
30
  if (bfd_write (tmp, sizeof (uint32_t), abfd) != sizeof (uint32_t))
448
0
    return false;
449
450
30
  left -= sizeof (uint32_t);
451
452
  /* Write file sizes.  */
453
454
30
  arelt = abfd->archive_head;
455
1.31k
  while (arelt)
456
1.28k
    {
457
1.28k
      if (left == 0)
458
0
  {
459
0
    if (block_map_off == block_size) /* Too many blocks.  */
460
0
      {
461
0
        bfd_set_error (bfd_error_invalid_operation);
462
0
        return false;
463
0
      }
464
465
0
    block = pdb_allocate_block (num_blocks, block_size);
466
0
    left = block_size;
467
468
0
    if (bfd_seek
469
0
        (abfd, (block_map_addr * block_size) + block_map_off, SEEK_SET))
470
0
      return false;
471
472
0
    bfd_putl32 (block, tmp);
473
474
0
    if (bfd_write (tmp, sizeof (uint32_t), abfd) != sizeof (uint32_t))
475
0
      return false;
476
477
0
    block_map_off += sizeof (uint32_t);
478
479
0
    if (bfd_seek (abfd, block * block_size, SEEK_SET))
480
0
      return false;
481
0
  }
482
483
1.28k
      bfd_putl32 (bfd_get_size (arelt), tmp);
484
485
1.28k
      if (bfd_write (tmp, sizeof (uint32_t), abfd) != sizeof (uint32_t))
486
0
  return false;
487
488
1.28k
      left -= sizeof (uint32_t);
489
490
1.28k
      arelt = arelt->archive_next;
491
1.28k
    }
492
493
  /* Write blocks.  */
494
495
30
  buf = bfd_malloc (block_size);
496
30
  if (!buf)
497
0
    return false;
498
499
30
  arelt = abfd->archive_head;
500
1.31k
  while (arelt)
501
1.28k
    {
502
1.28k
      ufile_ptr size = bfd_get_size (arelt);
503
1.28k
      uint32_t req_blocks = (size + block_size - 1) / block_size;
504
505
1.28k
      if (bfd_seek (arelt, 0, SEEK_SET))
506
0
  {
507
0
    free (buf);
508
0
    return false;
509
0
  }
510
511
1.29k
      for (uint32_t i = 0; i < req_blocks; i++)
512
5
  {
513
5
    uint32_t file_block, to_read;
514
515
5
    if (left == 0)
516
0
      {
517
0
        if (block_map_off == block_size) /* Too many blocks.  */
518
0
    {
519
0
      bfd_set_error (bfd_error_invalid_operation);
520
0
      free (buf);
521
0
      return false;
522
0
    }
523
524
0
        block = pdb_allocate_block (num_blocks, block_size);
525
0
        left = block_size;
526
527
0
        if (bfd_seek
528
0
      (abfd, (block_map_addr * block_size) + block_map_off,
529
0
       SEEK_SET))
530
0
    {
531
0
      free (buf);
532
0
      return false;
533
0
    }
534
535
0
        bfd_putl32 (block, tmp);
536
537
0
        if (bfd_write (tmp, sizeof (uint32_t), abfd) !=
538
0
      sizeof (uint32_t))
539
0
    {
540
0
      free (buf);
541
0
      return false;
542
0
    }
543
544
0
        block_map_off += sizeof (uint32_t);
545
546
0
        if (bfd_seek (abfd, block * block_size, SEEK_SET))
547
0
    {
548
0
      free (buf);
549
0
      return false;
550
0
    }
551
0
      }
552
553
    /* Allocate block and write number into directory.  */
554
555
5
    file_block = pdb_allocate_block (num_blocks, block_size);
556
557
5
    bfd_putl32 (file_block, tmp);
558
559
5
    if (bfd_write (tmp, sizeof (uint32_t), abfd) != sizeof (uint32_t))
560
0
      {
561
0
        free (buf);
562
0
        return false;
563
0
      }
564
565
5
    if (arelt == abfd->archive_head && i == 0)
566
3
      *stream0_start = file_block;
567
568
5
    left -= sizeof (uint32_t);
569
570
    /* Read file contents into buffer.  */
571
572
5
    to_read = size > block_size ? block_size : size;
573
574
5
    if (bfd_read (buf, to_read, arelt) != to_read)
575
0
      {
576
0
        free (buf);
577
0
        return false;
578
0
      }
579
580
5
    size -= to_read;
581
582
5
    if (to_read < block_size)
583
5
      memset (buf + to_read, 0, block_size - to_read);
584
585
5
    if (bfd_seek (abfd, file_block * block_size, SEEK_SET))
586
0
      {
587
0
        free (buf);
588
0
        return false;
589
0
      }
590
591
    /* Write file contents into allocated block.  */
592
593
5
    if (bfd_write (buf, block_size, abfd) != block_size)
594
0
      {
595
0
        free (buf);
596
0
        return false;
597
0
      }
598
599
5
    if (bfd_seek
600
5
        (abfd, (block * block_size) + block_size - left, SEEK_SET))
601
0
      {
602
0
        free (buf);
603
0
        return false;
604
0
      }
605
5
  }
606
607
1.28k
      arelt = arelt->archive_next;
608
1.28k
    }
609
610
30
  memset (buf, 0, left);
611
612
30
  if (bfd_write (buf, left, abfd) != left)
613
0
    {
614
0
      free (buf);
615
0
      return false;
616
0
    }
617
618
30
  free (buf);
619
620
30
  return true;
621
30
}
622
623
static bool
624
pdb_write_bitmap (bfd *abfd, uint32_t block_size, uint32_t num_blocks,
625
      uint32_t stream0_start)
626
30
{
627
30
  char *buf;
628
30
  uint32_t num_intervals = (num_blocks + block_size - 1) / block_size;
629
630
30
  buf = bfd_malloc (block_size);
631
30
  if (!buf)
632
0
    return false;
633
634
60
  for (uint32_t i = 0; i < num_intervals; i++)
635
30
    {
636
30
      if (bfd_seek (abfd, ((i * block_size) + 1) * block_size, SEEK_SET))
637
0
  {
638
0
    free (buf);
639
0
    return false;
640
0
  }
641
642
      /* All of our blocks are contiguous, making our free block map
643
   relatively simple.  0 = used, 1 = free.  */
644
645
30
      if (num_blocks >= 8)
646
0
  memset (buf, 0,
647
0
    (num_blocks / 8) >
648
0
    block_size ? block_size : (num_blocks / 8));
649
650
30
      if (num_blocks < block_size * 8)
651
30
  {
652
30
    unsigned int off = num_blocks / 8;
653
654
30
    if (num_blocks % 8)
655
30
      {
656
30
        buf[off] = 256 - (1 << (num_blocks % 8));
657
30
        off++;
658
30
      }
659
660
30
    if (off < block_size)
661
30
      memset (buf + off, 0xff, block_size - off);
662
30
  }
663
664
      /* Mark the blocks allocated to stream 0 as free.  This is because stream
665
   0 is intended to be used for the previous MSF directory, to allow
666
   atomic updates.  This doesn't apply to us, as we rewrite the whole
667
   file whenever any change is made.  */
668
669
30
      if (i == 0 && abfd->archive_head)
670
26
  {
671
26
    bfd *arelt = abfd->archive_head;
672
26
    uint32_t stream0_blocks =
673
26
      (bfd_get_size (arelt) + block_size - 1) / block_size;
674
675
26
    if (stream0_start % 8)
676
3
      {
677
3
        unsigned int high_bit;
678
679
3
        high_bit = (stream0_start % 8) + stream0_blocks;
680
3
        if (high_bit > 8)
681
0
    high_bit = 8;
682
683
3
        buf[stream0_start / 8] |=
684
3
    (1 << high_bit) - (1 << (stream0_start % 8));
685
686
3
        stream0_blocks -= high_bit - (stream0_start % 8);
687
3
        stream0_start += high_bit - (stream0_start % 8);
688
3
      }
689
690
26
    memset (buf + (stream0_start / 8), 0xff, stream0_blocks / 8);
691
26
    stream0_start += stream0_blocks / 8;
692
26
    stream0_blocks %= 8;
693
694
26
    if (stream0_blocks > 0)
695
0
      buf[stream0_start / 8] |= (1 << stream0_blocks) - 1;
696
26
  }
697
698
30
      if (num_blocks < block_size * 8)
699
30
  num_blocks = 0;
700
0
      else
701
0
  num_blocks -= block_size * 8;
702
703
30
      if (bfd_write (buf, block_size, abfd) != block_size)
704
0
  return false;
705
30
    }
706
707
30
  free (buf);
708
709
30
  return true;
710
30
}
711
712
static bool
713
pdb_write_contents (bfd *abfd)
714
30
{
715
30
  char tmp[sizeof (uint32_t)];
716
30
  const uint32_t block_size = 0x400;
717
30
  uint32_t block_map_addr;
718
30
  uint32_t num_blocks;
719
30
  uint32_t num_files = 0;
720
30
  uint32_t num_directory_bytes = sizeof (uint32_t);
721
30
  uint32_t stream0_start = 0;
722
30
  bfd *arelt;
723
724
30
  if (bfd_write (pdb_magic, sizeof (pdb_magic), abfd) != sizeof (pdb_magic))
725
0
    return false;
726
727
30
  bfd_putl32 (block_size, tmp);
728
729
30
  if (bfd_write (tmp, sizeof (uint32_t), abfd) != sizeof (uint32_t))
730
0
    return false;
731
732
30
  bfd_putl32 (1, tmp); /* Free block map block (always either 1 or 2).  */
733
734
30
  if (bfd_write (tmp, sizeof (uint32_t), abfd) != sizeof (uint32_t))
735
0
    return false;
736
737
30
  arelt = abfd->archive_head;
738
739
1.31k
  while (arelt)
740
1.28k
    {
741
1.28k
      uint32_t blocks_required =
742
1.28k
  (bfd_get_size (arelt) + block_size - 1) / block_size;
743
744
1.28k
      num_directory_bytes += sizeof (uint32_t); /* Size.  */
745
1.28k
      num_directory_bytes += blocks_required * sizeof (uint32_t); /* Blocks.  */
746
747
1.28k
      num_files++;
748
749
1.28k
      arelt = arelt->archive_next;
750
1.28k
    }
751
752
  /* Superblock plus two bitmap blocks.  */
753
30
  num_blocks = 3;
754
755
  /* Skip num_blocks for now.  */
756
30
  if (bfd_seek (abfd, sizeof (uint32_t), SEEK_CUR))
757
0
    return false;
758
759
30
  bfd_putl32 (num_directory_bytes, tmp);
760
761
30
  if (bfd_write (tmp, sizeof (uint32_t), abfd) != sizeof (uint32_t))
762
0
    return false;
763
764
  /* Skip unknown uint32_t (always 0?).  */
765
30
  if (bfd_seek (abfd, sizeof (uint32_t), SEEK_CUR))
766
0
    return false;
767
768
30
  block_map_addr = pdb_allocate_block (&num_blocks, block_size);
769
770
30
  bfd_putl32 (block_map_addr, tmp);
771
772
30
  if (bfd_write (tmp, sizeof (uint32_t), abfd) != sizeof (uint32_t))
773
0
    return false;
774
775
30
  if (!pdb_write_directory
776
30
      (abfd, block_size, num_files, block_map_addr, &num_blocks,
777
30
       &stream0_start))
778
0
    return false;
779
780
30
  if (!pdb_write_bitmap (abfd, block_size, num_blocks, stream0_start))
781
0
    return false;
782
783
  /* Write num_blocks now we know it.  */
784
785
30
  if (bfd_seek
786
30
      (abfd, sizeof (pdb_magic) + sizeof (uint32_t) + sizeof (uint32_t),
787
30
       SEEK_SET))
788
0
    return false;
789
790
30
  bfd_putl32 (num_blocks, tmp);
791
792
30
  if (bfd_write (tmp, sizeof (uint32_t), abfd) != sizeof (uint32_t))
793
0
    return false;
794
795
30
  return true;
796
30
}
797
798
#define pdb_bfd_free_cached_info _bfd_generic_bfd_free_cached_info
799
#define pdb_new_section_hook _bfd_generic_new_section_hook
800
#define pdb_get_section_contents _bfd_generic_get_section_contents
801
#define pdb_close_and_cleanup _bfd_generic_close_and_cleanup
802
803
#define pdb_slurp_armap _bfd_noarchive_slurp_armap
804
#define pdb_slurp_extended_name_table _bfd_noarchive_slurp_extended_name_table
805
#define pdb_construct_extended_name_table _bfd_noarchive_construct_extended_name_table
806
#define pdb_truncate_arname _bfd_noarchive_truncate_arname
807
#define pdb_write_armap _bfd_noarchive_write_armap
808
#define pdb_read_ar_hdr _bfd_noarchive_read_ar_hdr
809
#define pdb_write_ar_hdr _bfd_noarchive_write_ar_hdr
810
#define pdb_update_armap_timestamp _bfd_noarchive_update_armap_timestamp
811
812
const bfd_target pdb_vec =
813
{
814
  "pdb",
815
  bfd_target_unknown_flavour,
816
  BFD_ENDIAN_LITTLE,    /* target byte order */
817
  BFD_ENDIAN_LITTLE,    /* target headers byte order */
818
  0,        /* object flags */
819
  0,        /* section flags */
820
  0,        /* leading underscore */
821
  ' ',        /* ar_pad_char */
822
  16,       /* ar_max_namelen */
823
  0,        /* match priority.  */
824
  TARGET_KEEP_UNUSED_SECTION_SYMBOLS, /* keep unused section symbols.  */
825
  false,      /* merge sections */
826
  bfd_getl64, bfd_getl_signed_64, bfd_putl64,
827
  bfd_getl32, bfd_getl_signed_32, bfd_putl32,
828
  bfd_getl16, bfd_getl_signed_16, bfd_putl16, /* Data.  */
829
  bfd_getl64, bfd_getl_signed_64, bfd_putl64,
830
  bfd_getl32, bfd_getl_signed_32, bfd_putl32,
831
  bfd_getl16, bfd_getl_signed_16, bfd_putl16, /* Hdrs.  */
832
833
  {       /* bfd_check_format */
834
    _bfd_dummy_target,
835
    _bfd_dummy_target,
836
    pdb_archive_p,
837
    _bfd_dummy_target
838
  },
839
  {       /* bfd_set_format */
840
    _bfd_bool_bfd_false_error,
841
    _bfd_bool_bfd_false_error,
842
    _bfd_bool_bfd_true,
843
    _bfd_bool_bfd_false_error
844
  },
845
  {       /* bfd_write_contents */
846
    _bfd_bool_bfd_true,
847
    _bfd_bool_bfd_false_error,
848
    pdb_write_contents,
849
    _bfd_bool_bfd_false_error
850
  },
851
852
  BFD_JUMP_TABLE_GENERIC (pdb),
853
  BFD_JUMP_TABLE_COPY (_bfd_generic),
854
  BFD_JUMP_TABLE_CORE (_bfd_nocore),
855
  BFD_JUMP_TABLE_ARCHIVE (pdb),
856
  BFD_JUMP_TABLE_SYMBOLS (_bfd_nosymbols),
857
  BFD_JUMP_TABLE_RELOCS (_bfd_norelocs),
858
  BFD_JUMP_TABLE_WRITE (_bfd_generic),
859
  BFD_JUMP_TABLE_LINK (_bfd_nolink),
860
  BFD_JUMP_TABLE_DYNAMIC (_bfd_nodynamic),
861
862
  NULL,
863
864
  NULL
865
};