Coverage Report

Created: 2025-07-11 06:21

/src/WasmEdge/lib/loader/ast/section.cpp
Line
Count
Source (jump to first uncovered line)
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
47.1k
Expect<void> Loader::loadSection(AST::CustomSection &Sec) {
18
47.1k
  return loadSectionContent(Sec, [this, &Sec]() -> Expect<void> {
19
47.1k
    auto ReportError = [this](auto E) {
20
119
      return logLoadError(E, FMgr.getLastOffset(), ASTNodeAttr::Sec_Custom);
21
119
    };
22
23
    // Read name.
24
47.1k
    auto StartOffset = FMgr.getOffset();
25
47.1k
    EXPECTED_TRY(std::string Name, FMgr.readName().map_error(ReportError));
26
47.0k
    Sec.setName(Name);
27
47.0k
    auto ReadSize = FMgr.getOffset() - StartOffset;
28
29
    // Read remain bytes. Check is overread or not first.
30
47.0k
    if (unlikely(Sec.getContentSize() < ReadSize)) {
31
33
      return logLoadError(ErrCode::Value::UnexpectedEnd, FMgr.getLastOffset(),
32
33
                          ASTNodeAttr::Sec_Custom);
33
33
    }
34
46.9k
    EXPECTED_TRY(
35
46.9k
        std::vector<uint8_t> Bytes,
36
46.9k
        FMgr.readBytes(Sec.getContentSize() - ReadSize).map_error(ReportError));
37
46.9k
    Sec.getContent().insert(Sec.getContent().end(), Bytes.begin(), Bytes.end());
38
46.9k
    return {};
39
46.9k
  });
40
47.1k
}
41
42
// Load vector of type section. See "include/loader/loader.h".
43
4.58k
Expect<void> Loader::loadSection(AST::TypeSection &Sec) {
44
4.58k
  return loadSectionContent(Sec, [this, &Sec]() -> Expect<void> {
45
4.56k
    auto ReportError = [this](auto E) {
46
25
      return logLoadError(E, FMgr.getLastOffset(), ASTNodeAttr::Sec_Type);
47
25
    };
48
49
    // Read the recursive type vector size.
50
4.56k
    EXPECTED_TRY(uint32_t VecCnt, loadVecCnt().map_error(ReportError));
51
    // Read the recursive types.
52
4.54k
    Sec.getContent().clear();
53
4.54k
    uint32_t SubTypeCnt = 0;
54
12.5k
    for (uint32_t I = 0; I < VecCnt; I++) {
55
8.05k
      EXPECTED_TRY(uint8_t CodeByte, FMgr.peekByte().map_error(ReportError));
56
57
8.04k
      TypeCode Code = static_cast<TypeCode>(CodeByte);
58
8.04k
      if (!Conf.hasProposal(Proposal::GC) && Code != TypeCode::Func) {
59
23
        return logNeedProposal(ErrCode::Value::IntegerTooLong, Proposal::GC,
60
23
                               FMgr.getOffset(), ASTNodeAttr::Sec_Type);
61
23
      }
62
8.02k
      if (Code == TypeCode::Rec) {
63
        // Case: 0x4E vec(subtype).
64
0
        FMgr.readByte();
65
0
        EXPECTED_TRY(uint32_t RecVecCnt, loadVecCnt().map_error(ReportError));
66
0
        for (uint32_t J = 0; J < RecVecCnt; ++J) {
67
0
          Sec.getContent().emplace_back();
68
0
          EXPECTED_TRY(loadType(Sec.getContent().back()).map_error([](auto E) {
69
0
            spdlog::error(ErrInfo::InfoAST(ASTNodeAttr::Sec_Type));
70
0
            return E;
71
0
          }));
72
0
          Sec.getContent().back().setRecursiveInfo(J, RecVecCnt);
73
0
          Sec.getContent().back().setTypeIndex(SubTypeCnt);
74
0
          SubTypeCnt++;
75
0
        }
76
8.02k
      } else {
77
        // Case: subtype.
78
8.02k
        Sec.getContent().emplace_back();
79
8.02k
        Sec.getContent().back().setTypeIndex(SubTypeCnt);
80
8.02k
        SubTypeCnt++;
81
8.02k
        EXPECTED_TRY(loadType(Sec.getContent().back()).map_error([](auto E) {
82
8.02k
          spdlog::error(ErrInfo::InfoAST(ASTNodeAttr::Sec_Type));
83
8.02k
          return E;
84
8.02k
        }));
85
8.02k
      }
86
8.02k
    }
87
4.45k
    return {};
88
4.54k
  });
89
4.58k
}
90
91
// Load vector of import section. See "include/loader/loader.h".
92
288
Expect<void> Loader::loadSection(AST::ImportSection &Sec) {
93
288
  return loadSectionContent(Sec, [this, &Sec]() {
94
269
    return loadSectionContentVec(
95
17.2k
        Sec, [this](AST::ImportDesc &ImpDesc) { return loadDesc(ImpDesc); });
96
269
  });
97
288
}
98
99
// Load vector of function section. See "include/loader/loader.h".
100
4.38k
Expect<void> Loader::loadSection(AST::FunctionSection &Sec) {
101
4.38k
  return loadSectionContent(Sec, [this, &Sec]() {
102
4.37k
    return loadSectionContentVec(
103
35.4k
        Sec, [this](uint32_t &FuncIdx) -> Expect<void> {
104
35.4k
          EXPECTED_TRY(uint32_t Idx, FMgr.readU32().map_error([this](auto E) {
105
35.4k
            return logLoadError(E, FMgr.getLastOffset(),
106
35.4k
                                ASTNodeAttr::Sec_Function);
107
35.4k
          }));
108
35.4k
          FuncIdx = Idx;
109
35.4k
          return {};
110
35.4k
        });
111
4.37k
  });
112
4.38k
}
113
114
// Load vector of table section. See "include/loader/loader.h".
115
450
Expect<void> Loader::loadSection(AST::TableSection &Sec) {
116
450
  return loadSectionContent(Sec, [this, &Sec]() {
117
434
    return loadSectionContentVec(
118
586
        Sec, [this](AST::TableSegment &TabSeg) { return loadSegment(TabSeg); });
119
434
  });
120
450
}
121
122
// Load vector of memory section. See "include/loader/loader.h".
123
1.52k
Expect<void> Loader::loadSection(AST::MemorySection &Sec) {
124
1.52k
  return loadSectionContent(Sec, [this, &Sec]() {
125
1.50k
    return loadSectionContentVec(
126
3.00k
        Sec, [this](AST::MemoryType &MemType) { return loadType(MemType); });
127
1.50k
  });
128
1.52k
}
129
130
// Load vector of global section. See "include/loader/loader.h".
131
679
Expect<void> Loader::loadSection(AST::GlobalSection &Sec) {
132
679
  return loadSectionContent(Sec, [this, &Sec]() {
133
930
    return loadSectionContentVec(Sec, [this](AST::GlobalSegment &GlobSeg) {
134
930
      return loadSegment(GlobSeg);
135
930
    });
136
662
  });
137
679
}
138
139
// Load vector of export section. See "include/loader/loader.h".
140
1.22k
Expect<void> Loader::loadSection(AST::ExportSection &Sec) {
141
1.22k
  return loadSectionContent(Sec, [this, &Sec]() {
142
1.20k
    return loadSectionContentVec(
143
20.0k
        Sec, [this](AST::ExportDesc &ExpDesc) { return loadDesc(ExpDesc); });
144
1.20k
  });
145
1.22k
}
146
147
// Load start function index. See "include/loader/loader.h".
148
45
Expect<void> Loader::loadSection(AST::StartSection &Sec) {
149
45
  return loadSectionContent(Sec, [this, &Sec]() -> Expect<void> {
150
    // Read u32 of start function index.
151
27
    EXPECTED_TRY(uint32_t Idx, FMgr.readU32().map_error([this](auto E) {
152
24
      return logLoadError(E, FMgr.getLastOffset(), ASTNodeAttr::Sec_Start);
153
24
    }));
154
24
    Sec.setContent(Idx);
155
24
    return {};
156
27
  });
157
45
}
158
159
// Load vector of element section. See "include/loader/loader.h".
160
1.41k
Expect<void> Loader::loadSection(AST::ElementSection &Sec) {
161
1.41k
  return loadSectionContent(Sec, [this, &Sec]() {
162
6.15k
    return loadSectionContentVec(Sec, [this](AST::ElementSegment &ElemSeg) {
163
6.15k
      return loadSegment(ElemSeg);
164
6.15k
    });
165
1.39k
  });
166
1.41k
}
167
168
// Load vector of code section. See "include/loader/loader.h".
169
4.60k
Expect<void> Loader::loadSection(AST::CodeSection &Sec) {
170
4.60k
  return loadSectionContent(Sec, [this, &Sec]() {
171
20.4k
    return loadSectionContentVec(Sec, [this](AST::CodeSegment &CodeSeg) {
172
20.4k
      return loadSegment(CodeSeg);
173
20.4k
    });
174
4.59k
  });
175
4.60k
}
176
177
// Load vector of data section. See "include/loader/loader.h".
178
1.58k
Expect<void> Loader::loadSection(AST::DataSection &Sec) {
179
1.58k
  return loadSectionContent(Sec, [this, &Sec]() {
180
5.21k
    return loadSectionContentVec(Sec, [this](AST::DataSegment &DataSeg) {
181
5.21k
      return loadSegment(DataSeg);
182
5.21k
    });
183
1.55k
  });
184
1.58k
}
185
186
// Load content of data count section. See "include/loader/loader.h".
187
104
Expect<void> Loader::loadSection(AST::DataCountSection &Sec) {
188
104
  return loadSectionContent(Sec, [this, &Sec]() -> Expect<void> {
189
    // Read u32 of data count.
190
90
    EXPECTED_TRY(uint32_t Cnt, FMgr.readU32().map_error([this](auto E) {
191
88
      return logLoadError(E, FMgr.getLastOffset(), ASTNodeAttr::Sec_DataCount);
192
88
    }));
193
88
    Sec.setContent(Cnt);
194
88
    return {};
195
90
  });
196
104
}
197
198
// Load content of tag section. See "include/loader/loader.h".
199
0
Expect<void> Loader::loadSection(AST::TagSection &Sec) {
200
0
  return loadSectionContent(Sec, [this, &Sec]() {
201
0
    return loadSectionContentVec(
202
0
        Sec, [this](AST::TagType &TgType) { return loadType(TgType); });
203
0
  });
204
0
}
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
#else
235
  // Means universal wasm binary is not yet supported on this arch.
236
  return UINT8_C(-1);
237
#endif
238
0
}
239
240
} // namespace
241
242
// If there is any loader error occurs in the loadSection, then fallback
243
// to the interpreter mode with info level log.
244
0
Expect<void> Loader::loadSection(FileMgr &VecMgr, AST::AOTSection &Sec) {
245
0
  EXPECTED_TRY(auto Version, VecMgr.readU32().map_error([](auto E) {
246
0
    spdlog::error(E);
247
0
    spdlog::error("    AOT binary version read error:{}"sv, E);
248
0
    return E;
249
0
  }));
250
0
  Sec.setVersion(Version);
251
0
  if (unlikely(Sec.getVersion() != HostVersion())) {
252
0
    spdlog::error(ErrCode::Value::MalformedSection);
253
0
    spdlog::error("    AOT binary version unmatched."sv);
254
0
    return Unexpect(ErrCode::Value::MalformedSection);
255
0
  }
256
257
0
  EXPECTED_TRY(auto OSType, VecMgr.readByte().map_error([](auto E) {
258
0
    spdlog::error(E);
259
0
    spdlog::error("    AOT os type read error:{}"sv, E);
260
0
    return E;
261
0
  }));
262
0
  Sec.setOSType(OSType);
263
0
  if (unlikely(Sec.getOSType() != HostOSType())) {
264
0
    spdlog::error(ErrCode::Value::MalformedSection);
265
0
    spdlog::error("    AOT OS type unmatched."sv);
266
0
    return Unexpect(ErrCode::Value::MalformedSection);
267
0
  }
268
269
0
  EXPECTED_TRY(auto ArchType, VecMgr.readByte().map_error([](auto E) {
270
0
    spdlog::error(E);
271
0
    spdlog::error("    AOT arch type read error:{}"sv, E);
272
0
    return E;
273
0
  }));
274
0
  Sec.setArchType(ArchType);
275
0
  if (unlikely(Sec.getArchType() != HostArchType())) {
276
0
    spdlog::error(ErrCode::Value::MalformedSection);
277
0
    spdlog::error("    AOT arch type unmatched."sv);
278
0
    return Unexpect(ErrCode::Value::MalformedSection);
279
0
  }
280
281
0
  EXPECTED_TRY(auto VersionAddress, VecMgr.readU64().map_error([](auto E) {
282
0
    spdlog::error(E);
283
0
    spdlog::error("    AOT version address read error:{}"sv, E);
284
0
    return E;
285
0
  }));
286
0
  Sec.setVersionAddress(VersionAddress);
287
288
0
  EXPECTED_TRY(auto IntrinsicsAddress, VecMgr.readU64().map_error([](auto E) {
289
0
    spdlog::error(E);
290
0
    spdlog::error("    AOT intrinsics address read error:{}"sv, E);
291
0
    return E;
292
0
  }));
293
0
  Sec.setIntrinsicsAddress(IntrinsicsAddress);
294
295
0
  EXPECTED_TRY(auto TypesSize, VecMgr.readU64().map_error([](auto E) {
296
0
    spdlog::error(E);
297
0
    spdlog::error("    AOT types size read error:{}"sv, E);
298
0
    return E;
299
0
  }));
300
0
  if (TypesSize > VecMgr.getRemainSize()) {
301
0
    spdlog::error(ErrCode::Value::IntegerTooLong);
302
0
    spdlog::error("    AOT types size too large"sv);
303
0
    return Unexpect(ErrCode::Value::IntegerTooLong);
304
0
  }
305
0
  Sec.getTypesAddress().resize(TypesSize);
306
307
0
  for (size_t I = 0; I < Sec.getTypesAddress().size(); ++I) {
308
0
    EXPECTED_TRY(auto TypesAddress, VecMgr.readU64().map_error([](auto E) {
309
0
      spdlog::error(E);
310
0
      spdlog::error("    AOT type address read error:{}"sv, E);
311
0
      return E;
312
0
    }));
313
0
    Sec.getTypesAddress()[I] = TypesAddress;
314
0
  }
315
316
0
  EXPECTED_TRY(auto CodesSize, VecMgr.readU64().map_error([](auto E) {
317
0
    spdlog::error(E);
318
0
    spdlog::error("    AOT code size read error:{}"sv, E);
319
0
    return E;
320
0
  }));
321
0
  if (CodesSize > VecMgr.getRemainSize()) {
322
0
    spdlog::error(ErrCode::Value::IntegerTooLong);
323
0
    spdlog::error("    AOT code size too large"sv);
324
0
    return Unexpect(ErrCode::Value::IntegerTooLong);
325
0
  }
326
0
  Sec.getCodesAddress().resize(CodesSize);
327
328
0
  for (size_t I = 0; I < Sec.getCodesAddress().size(); ++I) {
329
0
    EXPECTED_TRY(auto CodesAddress, VecMgr.readU64().map_error([](auto E) {
330
0
      spdlog::error(E);
331
0
      spdlog::error("    AOT code address read error:{}"sv, E);
332
0
      return E;
333
0
    }));
334
0
    Sec.getCodesAddress()[I] = CodesAddress;
335
0
  }
336
337
0
  EXPECTED_TRY(auto SectionsSize, VecMgr.readU32().map_error([](auto E) {
338
0
    spdlog::error(E);
339
0
    spdlog::error("    AOT section count read error:{}"sv, E);
340
0
    return E;
341
0
  }));
342
0
  if (SectionsSize > VecMgr.getRemainSize()) {
343
0
    spdlog::error(ErrCode::Value::IntegerTooLong);
344
0
    spdlog::error("    AOT section count too large"sv);
345
0
    return Unexpect(ErrCode::Value::IntegerTooLong);
346
0
  }
347
0
  Sec.getSections().resize(SectionsSize);
348
349
0
  for (auto &Section : Sec.getSections()) {
350
0
    EXPECTED_TRY(std::get<0>(Section), VecMgr.readByte().map_error([](auto E) {
351
0
      spdlog::error(E);
352
0
      spdlog::error("    AOT section type read error:{}"sv, E);
353
0
      return E;
354
0
    }));
355
0
    EXPECTED_TRY(std::get<1>(Section), VecMgr.readU64().map_error([](auto E) {
356
0
      spdlog::error(E);
357
0
      spdlog::error("    AOT section offset read error:{}"sv, E);
358
0
      return E;
359
0
    }));
360
0
    EXPECTED_TRY(std::get<2>(Section), VecMgr.readU64().map_error([](auto E) {
361
0
      spdlog::error(E);
362
0
      spdlog::error("    AOT section size read error:{}"sv, E);
363
0
      return E;
364
0
    }));
365
0
    EXPECTED_TRY(uint32_t Size, VecMgr.readU32().map_error([](auto E) {
366
0
      spdlog::error(E);
367
0
      spdlog::error("    AOT section data size read error:{}"sv, E);
368
0
      return E;
369
0
    }));
370
0
    if (Size > VecMgr.getRemainSize()) {
371
0
      spdlog::error(ErrCode::Value::IntegerTooLong);
372
0
      spdlog::error("    AOT section data size is too large"sv);
373
0
      return Unexpect(ErrCode::Value::IntegerTooLong);
374
0
    }
375
0
    if (std::get<2>(Section) < Size) {
376
0
      spdlog::error(ErrCode::Value::IntegerTooLong);
377
0
      spdlog::error("    AOT section data size is larger then section size"sv);
378
0
      return Unexpect(ErrCode::Value::IntegerTooLong);
379
0
    }
380
0
    EXPECTED_TRY(auto Data, VecMgr.readBytes(Size).map_error([](auto E) {
381
0
      spdlog::error(E);
382
0
      spdlog::error("    AOT section data read error:{}"sv, E);
383
0
      return E;
384
0
    }));
385
0
    std::get<3>(Section) = std::move(Data);
386
0
  }
387
0
  return {};
388
0
}
389
390
} // namespace Loader
391
} // namespace WasmEdge