Coverage Report

Created: 2025-11-24 06:51

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/WasmEdge/lib/loader/ast/section.cpp
Line
Count
Source
1
// SPDX-License-Identifier: Apache-2.0
2
// SPDX-FileCopyrightText: 2019-2024 Second State INC
3
4
#include "aot/version.h"
5
#include "common/defines.h"
6
#include "loader/loader.h"
7
8
#include <tuple>
9
#include <utility>
10
11
using namespace std::literals;
12
13
namespace WasmEdge {
14
namespace Loader {
15
16
// Load content of custom section. See "include/loader/loader.h".
17
41.8k
Expect<void> Loader::loadSection(AST::CustomSection &Sec) {
18
41.8k
  return loadSectionContent(Sec, [this, &Sec]() -> Expect<void> {
19
41.8k
    auto ReportError = [this](auto E) {
20
105
      return logLoadError(E, FMgr.getLastOffset(), ASTNodeAttr::Sec_Custom);
21
105
    };
22
23
    // Read name.
24
41.8k
    auto StartOffset = FMgr.getOffset();
25
41.8k
    EXPECTED_TRY(std::string Name, FMgr.readName().map_error(ReportError));
26
41.7k
    Sec.setName(Name);
27
41.7k
    auto ReadSize = FMgr.getOffset() - StartOffset;
28
29
    // Read remain bytes. Check is overread or not first.
30
41.7k
    if (unlikely(Sec.getContentSize() < ReadSize)) {
31
25
      return logLoadError(ErrCode::Value::UnexpectedEnd, FMgr.getLastOffset(),
32
25
                          ASTNodeAttr::Sec_Custom);
33
25
    }
34
41.7k
    EXPECTED_TRY(
35
41.7k
        std::vector<uint8_t> Bytes,
36
41.7k
        FMgr.readBytes(Sec.getContentSize() - ReadSize).map_error(ReportError));
37
41.7k
    Sec.getContent().insert(Sec.getContent().end(), Bytes.begin(), Bytes.end());
38
41.7k
    return {};
39
41.7k
  });
40
41.8k
}
41
42
// Load vector of type section. See "include/loader/loader.h".
43
5.16k
Expect<void> Loader::loadSection(AST::TypeSection &Sec) {
44
5.16k
  return loadSectionContent(Sec, [this, &Sec]() -> Expect<void> {
45
5.14k
    auto ReportError = [this](auto E) {
46
27
      return logLoadError(E, FMgr.getLastOffset(), ASTNodeAttr::Sec_Type);
47
27
    };
48
49
    // Read the recursive type vector size.
50
5.14k
    EXPECTED_TRY(uint32_t VecCnt, loadVecCnt().map_error(ReportError));
51
    // Read the recursive types.
52
5.13k
    Sec.getContent().clear();
53
5.13k
    uint32_t SubTypeCnt = 0;
54
15.8k
    for (uint32_t I = 0; I < VecCnt; I++) {
55
11.0k
      EXPECTED_TRY(uint8_t CodeByte, FMgr.peekByte().map_error(ReportError));
56
57
11.0k
      TypeCode Code = static_cast<TypeCode>(CodeByte);
58
11.0k
      if (!Conf.hasProposal(Proposal::GC) && Code != TypeCode::Func) {
59
0
        return logNeedProposal(ErrCode::Value::IntegerTooLong, Proposal::GC,
60
0
                               FMgr.getOffset(), ASTNodeAttr::Sec_Type);
61
0
      }
62
11.0k
      if (Code == TypeCode::Rec) {
63
        // Case: 0x4E vec(subtype).
64
485
        FMgr.readByte();
65
485
        EXPECTED_TRY(uint32_t RecVecCnt, loadVecCnt().map_error(ReportError));
66
1.13k
        for (uint32_t J = 0; J < RecVecCnt; ++J) {
67
680
          Sec.getContent().emplace_back();
68
680
          EXPECTED_TRY(loadType(Sec.getContent().back()).map_error([](auto E) {
69
662
            spdlog::error(ErrInfo::InfoAST(ASTNodeAttr::Sec_Type));
70
662
            return E;
71
662
          }));
72
662
          Sec.getContent().back().setRecursiveInfo(J, RecVecCnt);
73
662
          Sec.getContent().back().setTypeIndex(SubTypeCnt);
74
662
          SubTypeCnt++;
75
662
        }
76
10.5k
      } else {
77
        // Case: subtype.
78
10.5k
        Sec.getContent().emplace_back();
79
10.5k
        Sec.getContent().back().setTypeIndex(SubTypeCnt);
80
10.5k
        SubTypeCnt++;
81
10.5k
        EXPECTED_TRY(loadType(Sec.getContent().back()).map_error([](auto E) {
82
10.5k
          spdlog::error(ErrInfo::InfoAST(ASTNodeAttr::Sec_Type));
83
10.5k
          return E;
84
10.5k
        }));
85
10.5k
      }
86
11.0k
    }
87
4.87k
    return {};
88
5.13k
  });
89
5.16k
}
90
91
// Load vector of import section. See "include/loader/loader.h".
92
286
Expect<void> Loader::loadSection(AST::ImportSection &Sec) {
93
286
  return loadSectionContent(Sec, [this, &Sec]() {
94
269
    return loadSectionContentVec(
95
11.7k
        Sec, [this](AST::ImportDesc &ImpDesc) { return loadDesc(ImpDesc); });
96
269
  });
97
286
}
98
99
// Load vector of function section. See "include/loader/loader.h".
100
4.73k
Expect<void> Loader::loadSection(AST::FunctionSection &Sec) {
101
4.73k
  return loadSectionContent(Sec, [this, &Sec]() {
102
4.72k
    return loadSectionContentVec(
103
31.6k
        Sec, [this](uint32_t &FuncIdx) -> Expect<void> {
104
31.6k
          EXPECTED_TRY(uint32_t Idx, FMgr.readU32().map_error([this](auto E) {
105
31.6k
            return logLoadError(E, FMgr.getLastOffset(),
106
31.6k
                                ASTNodeAttr::Sec_Function);
107
31.6k
          }));
108
31.6k
          FuncIdx = Idx;
109
31.6k
          return {};
110
31.6k
        });
111
4.72k
  });
112
4.73k
}
113
114
// Load vector of table section. See "include/loader/loader.h".
115
515
Expect<void> Loader::loadSection(AST::TableSection &Sec) {
116
515
  return loadSectionContent(Sec, [this, &Sec]() {
117
500
    return loadSectionContentVec(
118
1.06k
        Sec, [this](AST::TableSegment &TabSeg) { return loadSegment(TabSeg); });
119
500
  });
120
515
}
121
122
// Load vector of memory section. See "include/loader/loader.h".
123
1.46k
Expect<void> Loader::loadSection(AST::MemorySection &Sec) {
124
1.46k
  return loadSectionContent(Sec, [this, &Sec]() {
125
1.45k
    return loadSectionContentVec(
126
2.28k
        Sec, [this](AST::MemoryType &MemType) { return loadType(MemType); });
127
1.45k
  });
128
1.46k
}
129
130
// Load vector of global section. See "include/loader/loader.h".
131
602
Expect<void> Loader::loadSection(AST::GlobalSection &Sec) {
132
602
  return loadSectionContent(Sec, [this, &Sec]() {
133
913
    return loadSectionContentVec(Sec, [this](AST::GlobalSegment &GlobSeg) {
134
913
      return loadSegment(GlobSeg);
135
913
    });
136
588
  });
137
602
}
138
139
// Load vector of export section. See "include/loader/loader.h".
140
1.03k
Expect<void> Loader::loadSection(AST::ExportSection &Sec) {
141
1.03k
  return loadSectionContent(Sec, [this, &Sec]() {
142
1.02k
    return loadSectionContentVec(
143
18.2k
        Sec, [this](AST::ExportDesc &ExpDesc) { return loadDesc(ExpDesc); });
144
1.02k
  });
145
1.03k
}
146
147
// Load start function index. See "include/loader/loader.h".
148
32
Expect<void> Loader::loadSection(AST::StartSection &Sec) {
149
32
  return loadSectionContent(Sec, [this, &Sec]() -> Expect<void> {
150
    // Read u32 of start function index.
151
18
    EXPECTED_TRY(uint32_t Idx, FMgr.readU32().map_error([this](auto E) {
152
16
      return logLoadError(E, FMgr.getLastOffset(), ASTNodeAttr::Sec_Start);
153
16
    }));
154
16
    Sec.setContent(Idx);
155
16
    return {};
156
18
  });
157
32
}
158
159
// Load vector of element section. See "include/loader/loader.h".
160
1.31k
Expect<void> Loader::loadSection(AST::ElementSection &Sec) {
161
1.31k
  return loadSectionContent(Sec, [this, &Sec]() {
162
5.95k
    return loadSectionContentVec(Sec, [this](AST::ElementSegment &ElemSeg) {
163
5.95k
      return loadSegment(ElemSeg);
164
5.95k
    });
165
1.29k
  });
166
1.31k
}
167
168
// Load vector of code section. See "include/loader/loader.h".
169
4.92k
Expect<void> Loader::loadSection(AST::CodeSection &Sec) {
170
4.92k
  return loadSectionContent(Sec, [this, &Sec]() {
171
19.4k
    return loadSectionContentVec(Sec, [this](AST::CodeSegment &CodeSeg) {
172
19.4k
      return loadSegment(CodeSeg);
173
19.4k
    });
174
4.91k
  });
175
4.92k
}
176
177
// Load vector of data section. See "include/loader/loader.h".
178
1.52k
Expect<void> Loader::loadSection(AST::DataSection &Sec) {
179
1.52k
  return loadSectionContent(Sec, [this, &Sec]() {
180
3.81k
    return loadSectionContentVec(Sec, [this](AST::DataSegment &DataSeg) {
181
3.81k
      return loadSegment(DataSeg);
182
3.81k
    });
183
1.50k
  });
184
1.52k
}
185
186
// Load content of data count section. See "include/loader/loader.h".
187
93
Expect<void> Loader::loadSection(AST::DataCountSection &Sec) {
188
93
  return loadSectionContent(Sec, [this, &Sec]() -> Expect<void> {
189
    // Read u32 of data count.
190
80
    EXPECTED_TRY(uint32_t Cnt, FMgr.readU32().map_error([this](auto E) {
191
78
      return logLoadError(E, FMgr.getLastOffset(), ASTNodeAttr::Sec_DataCount);
192
78
    }));
193
78
    Sec.setContent(Cnt);
194
78
    return {};
195
80
  });
196
93
}
197
198
// Load content of tag section. See "include/loader/loader.h".
199
59
Expect<void> Loader::loadSection(AST::TagSection &Sec) {
200
59
  return loadSectionContent(Sec, [this, &Sec]() {
201
55
    return loadSectionContentVec(
202
1.65k
        Sec, [this](AST::TagType &TgType) { return loadType(TgType); });
203
55
  });
204
59
}
205
206
namespace {
207
208
0
inline constexpr uint32_t HostVersion() noexcept {
209
0
  return WasmEdge::AOT::kBinaryVersion;
210
0
}
211
212
0
inline constexpr uint8_t HostOSType() noexcept {
213
0
#if WASMEDGE_OS_LINUX
214
0
  return UINT8_C(1);
215
#elif WASMEDGE_OS_MACOS
216
  return UINT8_C(2);
217
#elif WASMEDGE_OS_WINDOWS
218
  return UINT8_C(3);
219
#else
220
  // Means WasmEdge is not yet supported on this OS.
221
  return UINT8_C(-1);
222
#endif
223
0
}
224
225
0
inline constexpr uint8_t HostArchType() noexcept {
226
0
#if defined(__x86_64__) || defined(_M_X64)
227
0
  return UINT8_C(1);
228
#elif defined(__aarch64__)
229
  return UINT8_C(2);
230
#elif defined(__riscv) && __riscv_xlen == 64
231
  return UINT8_C(3);
232
#elif defined(__arm__) && __ARM_ARCH == 7
233
  return UINT8_C(4);
234
#elif defined(__s390x__)
235
  return UINT8_C(5);
236
#else
237
  // Means universal wasm binary is not yet supported on this arch.
238
  return UINT8_C(-1);
239
#endif
240
0
}
241
242
} // namespace
243
244
// If there is any loader error occurs in the loadSection, then fallback
245
// to the interpreter mode with info level log.
246
0
Expect<void> Loader::loadSection(FileMgr &VecMgr, AST::AOTSection &Sec) {
247
0
  EXPECTED_TRY(auto Version, VecMgr.readU32().map_error([](auto E) {
248
0
    spdlog::error(E);
249
0
    spdlog::error("    AOT binary version read error:{}"sv, E);
250
0
    return E;
251
0
  }));
252
0
  Sec.setVersion(Version);
253
0
  if (unlikely(Sec.getVersion() != HostVersion())) {
254
0
    spdlog::error(ErrCode::Value::MalformedSection);
255
0
    spdlog::error("    AOT binary version unmatched."sv);
256
0
    return Unexpect(ErrCode::Value::MalformedSection);
257
0
  }
258
259
0
  EXPECTED_TRY(auto OSType, VecMgr.readByte().map_error([](auto E) {
260
0
    spdlog::error(E);
261
0
    spdlog::error("    AOT os type read error:{}"sv, E);
262
0
    return E;
263
0
  }));
264
0
  Sec.setOSType(OSType);
265
0
  if (unlikely(Sec.getOSType() != HostOSType())) {
266
0
    spdlog::error(ErrCode::Value::MalformedSection);
267
0
    spdlog::error("    AOT OS type unmatched."sv);
268
0
    return Unexpect(ErrCode::Value::MalformedSection);
269
0
  }
270
271
0
  EXPECTED_TRY(auto ArchType, VecMgr.readByte().map_error([](auto E) {
272
0
    spdlog::error(E);
273
0
    spdlog::error("    AOT arch type read error:{}"sv, E);
274
0
    return E;
275
0
  }));
276
0
  Sec.setArchType(ArchType);
277
0
  if (unlikely(Sec.getArchType() != HostArchType())) {
278
0
    spdlog::error(ErrCode::Value::MalformedSection);
279
0
    spdlog::error("    AOT arch type unmatched."sv);
280
0
    return Unexpect(ErrCode::Value::MalformedSection);
281
0
  }
282
283
0
  EXPECTED_TRY(auto VersionAddress, VecMgr.readU64().map_error([](auto E) {
284
0
    spdlog::error(E);
285
0
    spdlog::error("    AOT version address read error:{}"sv, E);
286
0
    return E;
287
0
  }));
288
0
  Sec.setVersionAddress(VersionAddress);
289
290
0
  EXPECTED_TRY(auto IntrinsicsAddress, VecMgr.readU64().map_error([](auto E) {
291
0
    spdlog::error(E);
292
0
    spdlog::error("    AOT intrinsics address read error:{}"sv, E);
293
0
    return E;
294
0
  }));
295
0
  Sec.setIntrinsicsAddress(IntrinsicsAddress);
296
297
0
  EXPECTED_TRY(auto TypesSize, VecMgr.readU64().map_error([](auto E) {
298
0
    spdlog::error(E);
299
0
    spdlog::error("    AOT types size read error:{}"sv, E);
300
0
    return E;
301
0
  }));
302
0
  if (TypesSize > VecMgr.getRemainSize()) {
303
0
    spdlog::error(ErrCode::Value::IntegerTooLong);
304
0
    spdlog::error("    AOT types size too large"sv);
305
0
    return Unexpect(ErrCode::Value::IntegerTooLong);
306
0
  }
307
0
  Sec.getTypesAddress().resize(TypesSize);
308
309
0
  for (size_t I = 0; I < Sec.getTypesAddress().size(); ++I) {
310
0
    EXPECTED_TRY(auto TypesAddress, VecMgr.readU64().map_error([](auto E) {
311
0
      spdlog::error(E);
312
0
      spdlog::error("    AOT type address read error:{}"sv, E);
313
0
      return E;
314
0
    }));
315
0
    Sec.getTypesAddress()[I] = TypesAddress;
316
0
  }
317
318
0
  EXPECTED_TRY(auto CodesSize, VecMgr.readU64().map_error([](auto E) {
319
0
    spdlog::error(E);
320
0
    spdlog::error("    AOT code size read error:{}"sv, E);
321
0
    return E;
322
0
  }));
323
0
  if (CodesSize > VecMgr.getRemainSize()) {
324
0
    spdlog::error(ErrCode::Value::IntegerTooLong);
325
0
    spdlog::error("    AOT code size too large"sv);
326
0
    return Unexpect(ErrCode::Value::IntegerTooLong);
327
0
  }
328
0
  Sec.getCodesAddress().resize(CodesSize);
329
330
0
  for (size_t I = 0; I < Sec.getCodesAddress().size(); ++I) {
331
0
    EXPECTED_TRY(auto CodesAddress, VecMgr.readU64().map_error([](auto E) {
332
0
      spdlog::error(E);
333
0
      spdlog::error("    AOT code address read error:{}"sv, E);
334
0
      return E;
335
0
    }));
336
0
    Sec.getCodesAddress()[I] = CodesAddress;
337
0
  }
338
339
0
  EXPECTED_TRY(auto SectionsSize, VecMgr.readU32().map_error([](auto E) {
340
0
    spdlog::error(E);
341
0
    spdlog::error("    AOT section count read error:{}"sv, E);
342
0
    return E;
343
0
  }));
344
0
  if (SectionsSize > VecMgr.getRemainSize()) {
345
0
    spdlog::error(ErrCode::Value::IntegerTooLong);
346
0
    spdlog::error("    AOT section count too large"sv);
347
0
    return Unexpect(ErrCode::Value::IntegerTooLong);
348
0
  }
349
0
  Sec.getSections().resize(SectionsSize);
350
351
0
  for (auto &Section : Sec.getSections()) {
352
0
    EXPECTED_TRY(std::get<0>(Section), VecMgr.readByte().map_error([](auto E) {
353
0
      spdlog::error(E);
354
0
      spdlog::error("    AOT section type read error:{}"sv, E);
355
0
      return E;
356
0
    }));
357
0
    EXPECTED_TRY(std::get<1>(Section), VecMgr.readU64().map_error([](auto E) {
358
0
      spdlog::error(E);
359
0
      spdlog::error("    AOT section offset read error:{}"sv, E);
360
0
      return E;
361
0
    }));
362
0
    EXPECTED_TRY(std::get<2>(Section), VecMgr.readU64().map_error([](auto E) {
363
0
      spdlog::error(E);
364
0
      spdlog::error("    AOT section size read error:{}"sv, E);
365
0
      return E;
366
0
    }));
367
0
    EXPECTED_TRY(uint32_t Size, VecMgr.readU32().map_error([](auto E) {
368
0
      spdlog::error(E);
369
0
      spdlog::error("    AOT section data size read error:{}"sv, E);
370
0
      return E;
371
0
    }));
372
0
    if (Size > VecMgr.getRemainSize()) {
373
0
      spdlog::error(ErrCode::Value::IntegerTooLong);
374
0
      spdlog::error("    AOT section data size is too large"sv);
375
0
      return Unexpect(ErrCode::Value::IntegerTooLong);
376
0
    }
377
0
    if (std::get<2>(Section) < Size) {
378
0
      spdlog::error(ErrCode::Value::IntegerTooLong);
379
0
      spdlog::error("    AOT section data size is larger then section size"sv);
380
0
      return Unexpect(ErrCode::Value::IntegerTooLong);
381
0
    }
382
0
    EXPECTED_TRY(auto Data, VecMgr.readBytes(Size).map_error([](auto E) {
383
0
      spdlog::error(E);
384
0
      spdlog::error("    AOT section data read error:{}"sv, E);
385
0
      return E;
386
0
    }));
387
0
    std::get<3>(Section) = std::move(Data);
388
0
  }
389
0
  return {};
390
0
}
391
392
} // namespace Loader
393
} // namespace WasmEdge