Coverage Report

Created: 2026-03-10 08:46

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/binutils-gdb/bfd/wasm-module.c
Line
Count
Source
1
/* BFD back-end for WebAssembly modules.
2
   Copyright (C) 2017-2026 Free Software Foundation, Inc.
3
4
   Based on srec.c, mmo.c, and binary.c
5
6
   This file is part of BFD, the Binary File Descriptor library.
7
8
   This program is free software; you can redistribute it and/or modify
9
   it under the terms of the GNU General Public License as published by
10
   the Free Software Foundation; either version 3 of the License, or
11
   (at your option) any later version.
12
13
   This program is distributed in the hope that it will be useful,
14
   but WITHOUT ANY WARRANTY; without even the implied warranty of
15
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16
   GNU General Public License for more details.
17
18
   You should have received a copy of the GNU General Public License
19
   along with this program; if not, write to the Free Software
20
   Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston,
21
   MA 02110-1301, USA.  */
22
23
/* The WebAssembly module format is a simple object file format
24
   including up to 11 numbered sections, plus any number of named
25
   "custom" sections. It is described at:
26
   https://github.com/WebAssembly/design/blob/master/BinaryEncoding.md. */
27
28
#include "sysdep.h"
29
#include "bfd.h"
30
#include "libiberty.h"
31
#include "libbfd.h"
32
#include "wasm-module.h"
33
34
#include <limits.h>
35
#ifndef CHAR_BIT
36
#define CHAR_BIT 8
37
#endif
38
39
typedef struct
40
{
41
  asymbol *      symbols;
42
  bfd_size_type  symcount;
43
} tdata_type;
44
45
static const char * const wasm_numbered_sections[] =
46
{
47
  NULL, /* Custom section, different layout.  */
48
  WASM_SECTION ( 1, "type"),
49
  WASM_SECTION ( 2, "import"),
50
  WASM_SECTION ( 3, "function"),
51
  WASM_SECTION ( 4, "table"),
52
  WASM_SECTION ( 5, "memory"),
53
  WASM_SECTION ( 6, "global"),
54
  WASM_SECTION ( 7, "export"),
55
  WASM_SECTION ( 8, "start"),
56
  WASM_SECTION ( 9, "element"),
57
  WASM_SECTION (10, "code"),
58
  WASM_SECTION (11, "data"),
59
};
60
61
4.59k
#define WASM_NUMBERED_SECTIONS ARRAY_SIZE (wasm_numbered_sections)
62
63
/* Resolve SECTION_CODE to a section name if there is one, NULL
64
   otherwise.  */
65
66
static const char *
67
wasm_section_code_to_name (bfd_byte section_code)
68
2.51k
{
69
2.51k
  if (section_code < WASM_NUMBERED_SECTIONS)
70
2.50k
    return wasm_numbered_sections[section_code];
71
72
10
  return NULL;
73
2.51k
}
74
75
/* Translate section name NAME to a section code, or 0 if it's a
76
   custom name.  */
77
78
static unsigned int
79
wasm_section_name_to_code (const char *name)
80
378
{
81
378
  unsigned i;
82
83
1.35k
  for (i = 1; i < WASM_NUMBERED_SECTIONS; i++)
84
1.29k
    if (strcmp (name, wasm_numbered_sections[i]) == 0)
85
320
      return i;
86
87
58
  return 0;
88
378
}
89
90
/* WebAssembly LEB128 integers are sufficiently like DWARF LEB128
91
   integers that we use _bfd_safe_read_leb128, but there are two
92
   points of difference:
93
94
   - WebAssembly requires a 32-bit value to be encoded in at most 5
95
     bytes, etc.
96
   - _bfd_safe_read_leb128 accepts incomplete LEB128 encodings at the
97
     end of the buffer, while these are invalid in WebAssembly.
98
99
   Those differences mean that we will accept some files that are
100
   invalid WebAssembly.  */
101
102
/* Read an LEB128-encoded integer from ABFD's I/O stream, reading one
103
   byte at a time.  Set ERROR_RETURN if no complete integer could be
104
   read, LENGTH_RETURN to the number of bytes read (including bytes in
105
   incomplete numbers).  SIGN means interpret the number as SLEB128. */
106
107
static bfd_vma
108
wasm_read_leb128 (bfd *abfd,
109
      bool *error_return,
110
      unsigned int *length_return,
111
      bool sign)
112
5.61k
{
113
5.61k
  bfd_vma result = 0;
114
5.61k
  unsigned int num_read = 0;
115
5.61k
  unsigned int shift = 0;
116
5.61k
  unsigned char byte = 0;
117
5.61k
  unsigned char lost, mask;
118
5.61k
  int status = 1;
119
120
8.25k
  while (bfd_read (&byte, 1, abfd) == 1)
121
8.20k
    {
122
8.20k
      num_read++;
123
124
8.20k
      if (shift < CHAR_BIT * sizeof (result))
125
7.20k
  {
126
7.20k
    result |= ((bfd_vma) (byte & 0x7f)) << shift;
127
    /* These bits overflowed.  */
128
7.20k
    lost = byte ^ (result >> shift);
129
    /* And this is the mask of possible overflow bits.  */
130
7.20k
    mask = 0x7f ^ ((bfd_vma) 0x7f << shift >> shift);
131
7.20k
    shift += 7;
132
7.20k
  }
133
1.00k
      else
134
1.00k
  {
135
1.00k
    lost = byte;
136
1.00k
    mask = 0x7f;
137
1.00k
  }
138
8.20k
      if ((lost & mask) != (sign && (bfd_signed_vma) result < 0 ? mask : 0))
139
967
  status |= 2;
140
141
8.20k
      if ((byte & 0x80) == 0)
142
5.57k
  {
143
5.57k
    status &= ~1;
144
5.57k
    if (sign && shift < CHAR_BIT * sizeof (result) && (byte & 0x40))
145
0
      result |= -((bfd_vma) 1 << shift);
146
5.57k
    break;
147
5.57k
  }
148
8.20k
    }
149
150
5.61k
  if (length_return != NULL)
151
5.61k
    *length_return = num_read;
152
5.61k
  if (error_return != NULL)
153
5.61k
    *error_return = status != 0;
154
155
5.61k
  return result;
156
5.61k
}
157
158
/* Encode an integer V as LEB128 and write it to ABFD, return TRUE on
159
   success.  */
160
161
static bool
162
wasm_write_uleb128 (bfd *abfd, bfd_vma v)
163
123
{
164
123
  do
165
123
    {
166
123
      bfd_byte c = v & 0x7f;
167
123
      v >>= 7;
168
169
123
      if (v)
170
0
  c |= 0x80;
171
172
123
      if (bfd_write (&c, 1, abfd) != 1)
173
0
  return false;
174
123
    }
175
123
  while (v);
176
177
123
  return true;
178
123
}
179
180
/* Read the LEB128 integer at P, saving it to X; at end of buffer,
181
   jump to error_return.  */
182
#define READ_LEB128(x, p, end)            \
183
332
  do                  \
184
332
    {                 \
185
332
      if ((p) >= (end))             \
186
332
  goto error_return;           \
187
332
      (x) = _bfd_safe_read_leb128 (abfd, &(p), false, (end));   \
188
329
    }                  \
189
332
  while (0)
190
191
/* Verify the magic number at the beginning of a WebAssembly module
192
   ABFD, setting ERRORPTR if there's a mismatch.  */
193
194
static bool
195
wasm_read_magic (bfd *abfd, bool *errorptr)
196
97.1k
{
197
97.1k
  bfd_byte magic_const[SIZEOF_WASM_MAGIC] = WASM_MAGIC;
198
97.1k
  bfd_byte magic[SIZEOF_WASM_MAGIC];
199
200
97.1k
  if (bfd_read (magic, sizeof (magic), abfd) == sizeof (magic)
201
96.9k
      && memcmp (magic, magic_const, sizeof (magic)) == 0)
202
778
    return true;
203
204
96.4k
  *errorptr = true;
205
96.4k
  return false;
206
97.1k
}
207
208
/* Read the version number from ABFD, returning TRUE if it's a supported
209
   version. Set ERRORPTR otherwise.  */
210
211
static bool
212
wasm_read_version (bfd *abfd, bool *errorptr)
213
778
{
214
778
  bfd_byte vers_const[SIZEOF_WASM_VERSION] = WASM_VERSION;
215
778
  bfd_byte vers[SIZEOF_WASM_VERSION];
216
217
778
  if (bfd_read (vers, sizeof (vers), abfd) == sizeof (vers)
218
      /* Don't attempt to parse newer versions, which are likely to
219
   require code changes.  */
220
776
      && memcmp (vers, vers_const, sizeof (vers)) == 0)
221
766
    return true;
222
223
12
  *errorptr = true;
224
12
  return false;
225
778
}
226
227
/* Read the WebAssembly header (magic number plus version number) from
228
   ABFD, setting ERRORPTR to TRUE if there is a mismatch.  */
229
230
static bool
231
wasm_read_header (bfd *abfd, bool *errorptr)
232
97.1k
{
233
97.1k
  if (! wasm_read_magic (abfd, errorptr))
234
96.4k
    return false;
235
236
778
  if (! wasm_read_version (abfd, errorptr))
237
12
    return false;
238
239
766
  return true;
240
778
}
241
242
/* Scan the "function" subsection of the "name" section ASECT in the
243
   wasm module ABFD. Create symbols. Return TRUE on success.  */
244
245
static bool
246
wasm_scan_name_function_section (bfd *abfd, sec_ptr asect)
247
53
{
248
53
  bfd_byte *p;
249
53
  bfd_byte *end;
250
53
  bfd_vma payload_size;
251
53
  bfd_vma symcount = 0;
252
53
  tdata_type *tdata = abfd->tdata.any;
253
53
  asymbol *symbols = NULL;
254
53
  sec_ptr space_function_index;
255
53
  size_t amt;
256
257
53
  p = asect->contents;
258
53
  end = asect->contents + asect->size;
259
260
53
  if (!p)
261
0
    return false;
262
263
173
  while (p < end)
264
171
    {
265
171
      bfd_byte subsection_code = *p++;
266
171
      if (subsection_code == WASM_FUNCTION_SUBSECTION)
267
22
  break;
268
269
      /* subsection_code is documented to be a varuint7, meaning that
270
   it has to be a single byte in the 0 - 127 range.  If it isn't,
271
   the spec must have changed underneath us, so give up.  */
272
149
      if (subsection_code & 0x80)
273
2
  return false;
274
275
147
      READ_LEB128 (payload_size, p, end);
276
277
146
      if (payload_size > (size_t) (end - p))
278
26
  return false;
279
280
120
      p += payload_size;
281
120
    }
282
283
24
  if (p >= end)
284
2
    return false;
285
286
22
  READ_LEB128 (payload_size, p, end);
287
288
22
  if (payload_size > (size_t) (end - p))
289
7
    return false;
290
291
15
  end = p + payload_size;
292
293
15
  READ_LEB128 (symcount, p, end);
294
295
  /* Sanity check: each symbol has at least two bytes.  */
296
14
  if (symcount > payload_size / 2)
297
3
    return false;
298
299
11
  tdata->symcount = symcount;
300
301
11
  space_function_index
302
11
    = bfd_make_section_with_flags (abfd, WASM_SECTION_FUNCTION_INDEX,
303
11
           SEC_READONLY | SEC_CODE);
304
305
11
  if (!space_function_index)
306
0
    space_function_index
307
0
      = bfd_get_section_by_name (abfd, WASM_SECTION_FUNCTION_INDEX);
308
309
11
  if (!space_function_index)
310
0
    return false;
311
312
11
  if (_bfd_mul_overflow (tdata->symcount, sizeof (asymbol), &amt))
313
0
    {
314
0
      bfd_set_error (bfd_error_file_too_big);
315
0
      return false;
316
0
    }
317
11
  symbols = bfd_alloc (abfd, amt);
318
11
  if (!symbols)
319
0
    return false;
320
321
82
  for (symcount = 0; p < end && symcount < tdata->symcount; symcount++)
322
74
    {
323
74
      bfd_vma idx;
324
74
      bfd_vma len;
325
74
      char *name;
326
74
      asymbol *sym;
327
328
74
      READ_LEB128 (idx, p, end);
329
74
      READ_LEB128 (len, p, end);
330
331
73
      if (len > (size_t) (end - p))
332
2
  goto error_return;
333
334
71
      name = bfd_alloc (abfd, len + 1);
335
71
      if (!name)
336
0
  goto error_return;
337
338
71
      memcpy (name, p, len);
339
71
      name[len] = 0;
340
71
      p += len;
341
342
71
      sym = &symbols[symcount];
343
71
      sym->the_bfd = abfd;
344
71
      sym->name = name;
345
71
      sym->value = idx;
346
71
      sym->flags = BSF_GLOBAL | BSF_FUNCTION;
347
71
      sym->section = space_function_index;
348
71
      sym->udata.p = NULL;
349
71
    }
350
351
8
  if (symcount < tdata->symcount)
352
2
    goto error_return;
353
354
6
  tdata->symbols = symbols;
355
6
  abfd->symcount = symcount;
356
357
6
  return true;
358
359
7
 error_return:
360
7
  if (symbols)
361
5
    bfd_release (abfd, symbols);
362
7
  tdata->symcount = 0;
363
7
  return false;
364
8
}
365
366
/* Read a byte from ABFD and return it, or EOF for EOF or error.
367
   Set ERRORPTR on non-EOF error.  */
368
369
static int
370
wasm_read_byte (bfd *abfd, bool *errorptr)
371
4.15k
{
372
4.15k
  bfd_byte byte;
373
374
4.15k
  if (bfd_read (&byte, 1, abfd) != 1)
375
79
    {
376
79
      if (bfd_get_error () != bfd_error_file_truncated)
377
0
  *errorptr = true;
378
79
      return EOF;
379
79
    }
380
381
4.08k
  return byte;
382
4.15k
}
383
384
/* Scan the wasm module ABFD, creating sections and symbols.
385
   Return TRUE on success.  */
386
387
static bool
388
wasm_scan (bfd *abfd)
389
383
{
390
383
  bool error = false;
391
  /* Fake VMAs for now. Choose 0x80000000 as base to avoid clashes
392
     with actual data addresses.  */
393
383
  bfd_vma vma = 0x80000000;
394
383
  int section_code;
395
383
  unsigned int bytes_read;
396
383
  asection *bfdsec;
397
398
383
  if (bfd_seek (abfd, 0, SEEK_SET) != 0)
399
0
    goto error_return;
400
401
383
  if (!wasm_read_header (abfd, &error))
402
0
    goto error_return;
403
404
4.15k
  while ((section_code = wasm_read_byte (abfd, &error)) != EOF)
405
4.08k
    {
406
4.08k
      if (section_code != 0)
407
2.51k
  {
408
2.51k
    const char *sname = wasm_section_code_to_name (section_code);
409
410
2.51k
    if (!sname)
411
10
      goto error_return;
412
413
2.50k
    bfdsec = bfd_make_section_anyway_with_flags (abfd, sname,
414
2.50k
                   SEC_HAS_CONTENTS);
415
2.50k
    if (bfdsec == NULL)
416
0
      goto error_return;
417
418
2.50k
    bfdsec->size = wasm_read_leb128 (abfd, &error, &bytes_read, false);
419
2.50k
    if (error)
420
25
      goto error_return;
421
2.50k
  }
422
1.56k
      else
423
1.56k
  {
424
1.56k
    bfd_vma payload_len;
425
1.56k
    bfd_vma namelen;
426
1.56k
    char *name;
427
1.56k
    char *prefix = WASM_SECTION_PREFIX;
428
1.56k
    size_t prefixlen = strlen (prefix);
429
1.56k
    ufile_ptr filesize;
430
431
1.56k
    payload_len = wasm_read_leb128 (abfd, &error, &bytes_read, false);
432
1.56k
    if (error)
433
18
      goto error_return;
434
1.54k
    namelen = wasm_read_leb128 (abfd, &error, &bytes_read, false);
435
1.54k
    if (error || bytes_read > payload_len
436
1.52k
        || namelen > payload_len - bytes_read)
437
81
      goto error_return;
438
1.46k
    payload_len -= namelen + bytes_read;
439
1.46k
    filesize = bfd_get_file_size (abfd);
440
1.46k
    if (filesize != 0 && namelen > filesize)
441
45
      {
442
45
        bfd_set_error (bfd_error_file_truncated);
443
45
        return false;
444
45
      }
445
1.42k
    name = bfd_alloc (abfd, namelen + prefixlen + 1);
446
1.42k
    if (!name)
447
0
      goto error_return;
448
1.42k
    memcpy (name, prefix, prefixlen);
449
1.42k
    if (bfd_read (name + prefixlen, namelen, abfd) != namelen)
450
9
      goto error_return;
451
1.41k
    name[prefixlen + namelen] = 0;
452
453
1.41k
    bfdsec = bfd_make_section_anyway_with_flags (abfd, name,
454
1.41k
                   SEC_HAS_CONTENTS);
455
1.41k
    if (bfdsec == NULL)
456
0
      goto error_return;
457
458
1.41k
    bfdsec->size = payload_len;
459
1.41k
  }
460
461
3.89k
      bfdsec->vma = vma;
462
3.89k
      bfdsec->lma = vma;
463
3.89k
      bfdsec->alignment_power = 0;
464
3.89k
      bfdsec->filepos = bfd_tell (abfd);
465
3.89k
      if (bfdsec->size != 0)
466
1.57k
  {
467
1.57k
    bfdsec->contents = _bfd_alloc_and_read (abfd, bfdsec->size,
468
1.57k
              bfdsec->size);
469
1.57k
    if (!bfdsec->contents)
470
116
      goto error_return;
471
1.45k
    bfdsec->alloced = 1;
472
1.45k
  }
473
474
3.77k
      vma += bfdsec->size;
475
3.77k
    }
476
477
  /* Make sure we're at actual EOF.  There's no indication in the
478
     WebAssembly format of how long the file is supposed to be.  */
479
79
  if (error)
480
0
    goto error_return;
481
482
79
  return true;
483
484
259
 error_return:
485
259
  return false;
486
79
}
487
488
/* Put a numbered section ASECT of ABFD into the table of numbered
489
   sections pointed to by FSARG.  */
490
491
static void
492
wasm_register_section (bfd *abfd ATTRIBUTE_UNUSED,
493
           asection *asect,
494
           void *fsarg)
495
189
{
496
189
  sec_ptr *numbered_sections = fsarg;
497
189
  int idx = wasm_section_name_to_code (asect->name);
498
499
189
  if (idx == 0)
500
29
    return;
501
502
160
  numbered_sections[idx] = asect;
503
160
}
504
505
struct compute_section_arg
506
{
507
  bfd_vma pos;
508
  bool failed;
509
};
510
511
/* Compute the file position of ABFD's section ASECT.  FSARG is a
512
   pointer to the current file position.
513
514
   We allow section names of the form .wasm.id to encode the numbered
515
   section with ID id, if it exists; otherwise, a custom section with
516
   ID "id" is produced.  Arbitrary section names are for sections that
517
   are assumed already to contain a section header; those are appended
518
   to the WebAssembly module verbatim.  */
519
520
static void
521
wasm_compute_custom_section_file_position (bfd *abfd,
522
             sec_ptr asect,
523
             void *fsarg)
524
189
{
525
189
  struct compute_section_arg *fs = fsarg;
526
189
  int idx;
527
528
189
  if (fs->failed)
529
0
    return;
530
531
189
  idx = wasm_section_name_to_code (asect->name);
532
533
189
  if (idx != 0)
534
160
    return;
535
536
29
  if (startswith (asect->name, WASM_SECTION_PREFIX))
537
27
    {
538
27
      const char *name = asect->name + strlen (WASM_SECTION_PREFIX);
539
27
      bfd_size_type payload_len = asect->size;
540
27
      bfd_size_type name_len = strlen (name);
541
27
      bfd_size_type nl = name_len;
542
543
27
      payload_len += name_len;
544
545
27
      do
546
27
  {
547
27
    payload_len++;
548
27
    nl >>= 7;
549
27
  }
550
27
      while (nl);
551
552
27
      if (bfd_seek (abfd, fs->pos, SEEK_SET) != 0
553
27
    || ! wasm_write_uleb128 (abfd, 0)
554
27
    || ! wasm_write_uleb128 (abfd, payload_len)
555
27
    || ! wasm_write_uleb128 (abfd, name_len)
556
27
    || bfd_write (name, name_len, abfd) != name_len)
557
0
  goto error_return;
558
27
      fs->pos = asect->filepos = bfd_tell (abfd);
559
27
    }
560
2
  else
561
2
    {
562
2
      asect->filepos = fs->pos;
563
2
    }
564
565
566
29
  fs->pos += asect->size;
567
29
  return;
568
569
0
 error_return:
570
0
  fs->failed = true;
571
0
}
572
573
/* Compute the file positions for the sections of ABFD.  Currently,
574
   this writes all numbered sections first, in order, then all custom
575
   sections, in section order.
576
577
   The spec says that the numbered sections must appear in order of
578
   their ids, but custom sections can appear in any position and any
579
   order, and more than once. FIXME: support that.  */
580
581
static bool
582
wasm_compute_section_file_positions (bfd *abfd)
583
28
{
584
28
  bfd_byte magic[SIZEOF_WASM_MAGIC] = WASM_MAGIC;
585
28
  bfd_byte vers[SIZEOF_WASM_VERSION] = WASM_VERSION;
586
28
  sec_ptr numbered_sections[WASM_NUMBERED_SECTIONS];
587
28
  struct compute_section_arg fs;
588
28
  unsigned int i;
589
590
28
  if (bfd_seek (abfd, (bfd_vma) 0, SEEK_SET) != 0
591
28
      || bfd_write (magic, sizeof (magic), abfd) != (sizeof magic)
592
28
      || bfd_write (vers, sizeof (vers), abfd) != sizeof (vers))
593
0
    return false;
594
595
364
  for (i = 0; i < WASM_NUMBERED_SECTIONS; i++)
596
336
    numbered_sections[i] = NULL;
597
598
28
  bfd_map_over_sections (abfd, wasm_register_section, numbered_sections);
599
600
28
  fs.pos = bfd_tell (abfd);
601
364
  for (i = 0; i < WASM_NUMBERED_SECTIONS; i++)
602
336
    {
603
336
      sec_ptr sec = numbered_sections[i];
604
336
      bfd_size_type size;
605
606
336
      if (! sec)
607
315
  continue;
608
21
      size = sec->size;
609
21
      if (bfd_seek (abfd, fs.pos, SEEK_SET) != 0)
610
0
  return false;
611
21
      if (! wasm_write_uleb128 (abfd, i)
612
21
    || ! wasm_write_uleb128 (abfd, size))
613
0
  return false;
614
21
      fs.pos = sec->filepos = bfd_tell (abfd);
615
21
      fs.pos += size;
616
21
    }
617
618
28
  fs.failed = false;
619
620
28
  bfd_map_over_sections (abfd, wasm_compute_custom_section_file_position, &fs);
621
622
28
  if (fs.failed)
623
0
    return false;
624
625
28
  abfd->output_has_begun = true;
626
627
28
  return true;
628
28
}
629
630
static bool
631
wasm_set_section_contents (bfd *abfd,
632
         sec_ptr section,
633
         const void *location,
634
         file_ptr offset,
635
         bfd_size_type count)
636
52
{
637
52
  if (count == 0)
638
0
    return true;
639
640
52
  if (! abfd->output_has_begun
641
28
      && ! wasm_compute_section_file_positions (abfd))
642
0
    return false;
643
644
52
  if (bfd_seek (abfd, section->filepos + offset, SEEK_SET) != 0
645
52
      || bfd_write (location, count, abfd) != count)
646
0
    return false;
647
648
52
  return true;
649
52
}
650
651
static bool
652
wasm_write_object_contents (bfd* abfd)
653
74
{
654
74
  bfd_byte magic[] = WASM_MAGIC;
655
74
  bfd_byte vers[] = WASM_VERSION;
656
657
74
  if (bfd_seek (abfd, 0, SEEK_SET) != 0)
658
0
    return false;
659
660
74
  if (bfd_write (magic, sizeof (magic), abfd) != sizeof (magic)
661
74
      || bfd_write (vers, sizeof (vers), abfd) != sizeof (vers))
662
0
    return false;
663
664
74
  return true;
665
74
}
666
667
static bool
668
wasm_mkobject (bfd *abfd)
669
458
{
670
458
  tdata_type *tdata = (tdata_type *) bfd_alloc (abfd, sizeof (tdata_type));
671
672
458
  if (! tdata)
673
0
    return false;
674
675
458
  tdata->symbols = NULL;
676
458
  tdata->symcount = 0;
677
678
458
  abfd->tdata.any = tdata;
679
680
458
  return true;
681
458
}
682
683
static long
684
wasm_get_symtab_upper_bound (bfd *abfd)
685
74
{
686
74
  tdata_type *tdata = abfd->tdata.any;
687
688
74
  return (tdata->symcount + 1) * (sizeof (asymbol *));
689
74
}
690
691
static long
692
wasm_canonicalize_symtab (bfd *abfd, asymbol **alocation)
693
74
{
694
74
  tdata_type *tdata = abfd->tdata.any;
695
74
  size_t i;
696
697
100
  for (i = 0; i < tdata->symcount; i++)
698
26
    alocation[i] = &tdata->symbols[i];
699
74
  alocation[i] = NULL;
700
701
74
  return tdata->symcount;
702
74
}
703
704
static asymbol *
705
wasm_make_empty_symbol (bfd *abfd)
706
4.13k
{
707
4.13k
  size_t amt = sizeof (asymbol);
708
4.13k
  asymbol *new_symbol = (asymbol *) bfd_zalloc (abfd, amt);
709
710
4.13k
  if (! new_symbol)
711
0
    return NULL;
712
4.13k
  new_symbol->the_bfd = abfd;
713
4.13k
  return new_symbol;
714
4.13k
}
715
716
static void
717
wasm_print_symbol (bfd *abfd,
718
       void * filep,
719
       asymbol *symbol,
720
       bfd_print_symbol_type how)
721
0
{
722
0
  FILE *file = (FILE *) filep;
723
724
0
  switch (how)
725
0
    {
726
0
    case bfd_print_symbol_name:
727
0
      fprintf (file, "%s", symbol->name);
728
0
      break;
729
730
0
    default:
731
0
      bfd_print_symbol_vandf (abfd, filep, symbol);
732
0
      fprintf (file, " %-5s %s", symbol->section->name, symbol->name);
733
0
    }
734
0
}
735
736
static void
737
wasm_get_symbol_info (bfd *abfd ATTRIBUTE_UNUSED,
738
          asymbol *symbol,
739
          symbol_info *ret)
740
0
{
741
0
  bfd_symbol_info (symbol, ret);
742
0
}
743
744
/* Check whether ABFD is a WebAssembly module; if so, scan it.  */
745
746
static bfd_cleanup
747
wasm_object_p (bfd *abfd)
748
96.8k
{
749
96.8k
  bool error;
750
96.8k
  asection *s;
751
752
96.8k
  if (bfd_seek (abfd, 0, SEEK_SET) != 0)
753
0
    return NULL;
754
755
96.8k
  if (!wasm_read_header (abfd, &error))
756
96.4k
    {
757
96.4k
      bfd_set_error (bfd_error_wrong_format);
758
96.4k
      return NULL;
759
96.4k
    }
760
761
383
  if (!wasm_mkobject (abfd))
762
0
    return NULL;
763
764
383
  if (!wasm_scan (abfd)
765
79
      || !bfd_default_set_arch_mach (abfd, bfd_arch_wasm32, 0))
766
304
    {
767
304
      bfd_release (abfd, abfd->tdata.any);
768
304
      abfd->tdata.any = NULL;
769
304
      return NULL;
770
304
    }
771
772
79
  s = bfd_get_section_by_name (abfd, WASM_NAME_SECTION);
773
79
  if (s != NULL && wasm_scan_name_function_section (abfd, s))
774
6
    abfd->flags |= HAS_SYMS;
775
776
79
  return _bfd_no_cleanup;
777
383
}
778
779
/* BFD_JUMP_TABLE_WRITE */
780
#define wasm_set_arch_mach      _bfd_generic_set_arch_mach
781
782
/* BFD_JUMP_TABLE_SYMBOLS */
783
#define wasm_get_symbol_version_string    _bfd_nosymbols_get_symbol_version_string
784
#define wasm_bfd_is_local_label_name     bfd_generic_is_local_label_name
785
#define wasm_bfd_is_target_special_symbol _bfd_bool_bfd_asymbol_false
786
#define wasm_get_lineno       _bfd_nosymbols_get_lineno
787
#define wasm_find_nearest_line      _bfd_nosymbols_find_nearest_line
788
#define wasm_find_nearest_line_with_alt   _bfd_nosymbols_find_nearest_line_with_alt
789
#define wasm_find_line        _bfd_nosymbols_find_line
790
#define wasm_find_inliner_info      _bfd_nosymbols_find_inliner_info
791
#define wasm_bfd_make_debug_symbol    _bfd_nosymbols_bfd_make_debug_symbol
792
#define wasm_read_minisymbols     _bfd_generic_read_minisymbols
793
#define wasm_minisymbol_to_symbol   _bfd_generic_minisymbol_to_symbol
794
795
const bfd_target wasm_vec =
796
{
797
  "wasm",     /* Name.  */
798
  bfd_target_unknown_flavour,
799
  BFD_ENDIAN_LITTLE,
800
  BFD_ENDIAN_LITTLE,
801
  (HAS_SYMS | WP_TEXT),   /* Object flags.  */
802
  (SEC_CODE | SEC_DATA | SEC_HAS_CONTENTS), /* Section flags.  */
803
  0,        /* Leading underscore.  */
804
  ' ',        /* AR_pad_char.  */
805
  255,        /* AR_max_namelen.  */
806
  0,        /* Match priority.  */
807
  TARGET_KEEP_UNUSED_SECTION_SYMBOLS, /* keep unused section symbols.  */
808
  false,      /* merge sections */
809
  /* Routines to byte-swap various sized integers from the data sections.  */
810
  bfd_getl64, bfd_getl_signed_64, bfd_putl64,
811
  bfd_getl32, bfd_getl_signed_32, bfd_putl32,
812
  bfd_getl16, bfd_getl_signed_16, bfd_putl16,
813
814
  /* Routines to byte-swap various sized integers from the file headers.  */
815
  bfd_getl64, bfd_getl_signed_64, bfd_putl64,
816
  bfd_getl32, bfd_getl_signed_32, bfd_putl32,
817
  bfd_getl16, bfd_getl_signed_16, bfd_putl16,
818
819
  {
820
    _bfd_dummy_target,
821
    wasm_object_p,    /* bfd_check_format.  */
822
    _bfd_dummy_target,
823
    _bfd_dummy_target,
824
  },
825
  {
826
    _bfd_bool_bfd_false_error,
827
    wasm_mkobject,
828
    _bfd_bool_bfd_false_error,
829
    _bfd_bool_bfd_false_error,
830
  },
831
  {       /* bfd_write_contents.  */
832
    _bfd_bool_bfd_false_error,
833
    wasm_write_object_contents,
834
    _bfd_bool_bfd_false_error,
835
    _bfd_bool_bfd_false_error,
836
  },
837
838
  BFD_JUMP_TABLE_GENERIC (_bfd_generic),
839
  BFD_JUMP_TABLE_COPY (_bfd_generic),
840
  BFD_JUMP_TABLE_CORE (_bfd_nocore),
841
  BFD_JUMP_TABLE_ARCHIVE (_bfd_noarchive),
842
  BFD_JUMP_TABLE_SYMBOLS (wasm),
843
  BFD_JUMP_TABLE_RELOCS (_bfd_norelocs),
844
  BFD_JUMP_TABLE_WRITE (wasm),
845
  BFD_JUMP_TABLE_LINK (_bfd_nolink),
846
  BFD_JUMP_TABLE_DYNAMIC (_bfd_nodynamic),
847
848
  NULL,
849
850
  NULL,
851
};