Coverage Report

Created: 2025-07-23 06:30

/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
44.0k
Expect<void> Loader::loadSection(AST::CustomSection &Sec) {
18
44.0k
  return loadSectionContent(Sec, [this, &Sec]() -> Expect<void> {
19
43.9k
    auto ReportError = [this](auto E) {
20
98
      return logLoadError(E, FMgr.getLastOffset(), ASTNodeAttr::Sec_Custom);
21
98
    };
22
23
    // Read name.
24
43.9k
    auto StartOffset = FMgr.getOffset();
25
43.9k
    EXPECTED_TRY(std::string Name, FMgr.readName().map_error(ReportError));
26
43.8k
    Sec.setName(Name);
27
43.8k
    auto ReadSize = FMgr.getOffset() - StartOffset;
28
29
    // Read remain bytes. Check is overread or not first.
30
43.8k
    if (unlikely(Sec.getContentSize() < ReadSize)) {
31
27
      return logLoadError(ErrCode::Value::UnexpectedEnd, FMgr.getLastOffset(),
32
27
                          ASTNodeAttr::Sec_Custom);
33
27
    }
34
43.8k
    EXPECTED_TRY(
35
43.8k
        std::vector<uint8_t> Bytes,
36
43.8k
        FMgr.readBytes(Sec.getContentSize() - ReadSize).map_error(ReportError));
37
43.8k
    Sec.getContent().insert(Sec.getContent().end(), Bytes.begin(), Bytes.end());
38
43.8k
    return {};
39
43.8k
  });
40
44.0k
}
41
42
// Load vector of type section. See "include/loader/loader.h".
43
3.88k
Expect<void> Loader::loadSection(AST::TypeSection &Sec) {
44
3.88k
  return loadSectionContent(Sec, [this, &Sec]() -> Expect<void> {
45
3.86k
    auto ReportError = [this](auto E) {
46
19
      return logLoadError(E, FMgr.getLastOffset(), ASTNodeAttr::Sec_Type);
47
19
    };
48
49
    // Read the recursive type vector size.
50
3.86k
    EXPECTED_TRY(uint32_t VecCnt, loadVecCnt().map_error(ReportError));
51
    // Read the recursive types.
52
3.85k
    Sec.getContent().clear();
53
3.85k
    uint32_t SubTypeCnt = 0;
54
11.2k
    for (uint32_t I = 0; I < VecCnt; I++) {
55
7.46k
      EXPECTED_TRY(uint8_t CodeByte, FMgr.peekByte().map_error(ReportError));
56
57
7.46k
      TypeCode Code = static_cast<TypeCode>(CodeByte);
58
7.46k
      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
7.44k
      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
7.44k
      } else {
77
        // Case: subtype.
78
7.44k
        Sec.getContent().emplace_back();
79
7.44k
        Sec.getContent().back().setTypeIndex(SubTypeCnt);
80
7.44k
        SubTypeCnt++;
81
7.44k
        EXPECTED_TRY(loadType(Sec.getContent().back()).map_error([](auto E) {
82
7.44k
          spdlog::error(ErrInfo::InfoAST(ASTNodeAttr::Sec_Type));
83
7.44k
          return E;
84
7.44k
        }));
85
7.44k
      }
86
7.44k
    }
87
3.76k
    return {};
88
3.85k
  });
89
3.88k
}
90
91
// Load vector of import section. See "include/loader/loader.h".
92
267
Expect<void> Loader::loadSection(AST::ImportSection &Sec) {
93
267
  return loadSectionContent(Sec, [this, &Sec]() {
94
249
    return loadSectionContentVec(
95
11.8k
        Sec, [this](AST::ImportDesc &ImpDesc) { return loadDesc(ImpDesc); });
96
249
  });
97
267
}
98
99
// Load vector of function section. See "include/loader/loader.h".
100
3.69k
Expect<void> Loader::loadSection(AST::FunctionSection &Sec) {
101
3.69k
  return loadSectionContent(Sec, [this, &Sec]() {
102
3.67k
    return loadSectionContentVec(
103
30.5k
        Sec, [this](uint32_t &FuncIdx) -> Expect<void> {
104
30.5k
          EXPECTED_TRY(uint32_t Idx, FMgr.readU32().map_error([this](auto E) {
105
30.5k
            return logLoadError(E, FMgr.getLastOffset(),
106
30.5k
                                ASTNodeAttr::Sec_Function);
107
30.5k
          }));
108
30.5k
          FuncIdx = Idx;
109
30.5k
          return {};
110
30.5k
        });
111
3.67k
  });
112
3.69k
}
113
114
// Load vector of table section. See "include/loader/loader.h".
115
367
Expect<void> Loader::loadSection(AST::TableSection &Sec) {
116
367
  return loadSectionContent(Sec, [this, &Sec]() {
117
352
    return loadSectionContentVec(
118
504
        Sec, [this](AST::TableSegment &TabSeg) { return loadSegment(TabSeg); });
119
352
  });
120
367
}
121
122
// Load vector of memory section. See "include/loader/loader.h".
123
1.39k
Expect<void> Loader::loadSection(AST::MemorySection &Sec) {
124
1.39k
  return loadSectionContent(Sec, [this, &Sec]() {
125
1.37k
    return loadSectionContentVec(
126
2.89k
        Sec, [this](AST::MemoryType &MemType) { return loadType(MemType); });
127
1.37k
  });
128
1.39k
}
129
130
// Load vector of global section. See "include/loader/loader.h".
131
492
Expect<void> Loader::loadSection(AST::GlobalSection &Sec) {
132
492
  return loadSectionContent(Sec, [this, &Sec]() {
133
741
    return loadSectionContentVec(Sec, [this](AST::GlobalSegment &GlobSeg) {
134
741
      return loadSegment(GlobSeg);
135
741
    });
136
477
  });
137
492
}
138
139
// Load vector of export section. See "include/loader/loader.h".
140
1.00k
Expect<void> Loader::loadSection(AST::ExportSection &Sec) {
141
1.00k
  return loadSectionContent(Sec, [this, &Sec]() {
142
988
    return loadSectionContentVec(
143
16.4k
        Sec, [this](AST::ExportDesc &ExpDesc) { return loadDesc(ExpDesc); });
144
988
  });
145
1.00k
}
146
147
// Load start function index. See "include/loader/loader.h".
148
35
Expect<void> Loader::loadSection(AST::StartSection &Sec) {
149
35
  return loadSectionContent(Sec, [this, &Sec]() -> Expect<void> {
150
    // Read u32 of start function index.
151
21
    EXPECTED_TRY(uint32_t Idx, FMgr.readU32().map_error([this](auto E) {
152
19
      return logLoadError(E, FMgr.getLastOffset(), ASTNodeAttr::Sec_Start);
153
19
    }));
154
19
    Sec.setContent(Idx);
155
19
    return {};
156
21
  });
157
35
}
158
159
// Load vector of element section. See "include/loader/loader.h".
160
1.29k
Expect<void> Loader::loadSection(AST::ElementSection &Sec) {
161
1.29k
  return loadSectionContent(Sec, [this, &Sec]() {
162
5.78k
    return loadSectionContentVec(Sec, [this](AST::ElementSegment &ElemSeg) {
163
5.78k
      return loadSegment(ElemSeg);
164
5.78k
    });
165
1.28k
  });
166
1.29k
}
167
168
// Load vector of code section. See "include/loader/loader.h".
169
3.87k
Expect<void> Loader::loadSection(AST::CodeSection &Sec) {
170
3.87k
  return loadSectionContent(Sec, [this, &Sec]() {
171
18.2k
    return loadSectionContentVec(Sec, [this](AST::CodeSegment &CodeSeg) {
172
18.2k
      return loadSegment(CodeSeg);
173
18.2k
    });
174
3.87k
  });
175
3.87k
}
176
177
// Load vector of data section. See "include/loader/loader.h".
178
1.47k
Expect<void> Loader::loadSection(AST::DataSection &Sec) {
179
1.47k
  return loadSectionContent(Sec, [this, &Sec]() {
180
4.15k
    return loadSectionContentVec(Sec, [this](AST::DataSegment &DataSeg) {
181
4.15k
      return loadSegment(DataSeg);
182
4.15k
    });
183
1.45k
  });
184
1.47k
}
185
186
// Load content of data count section. See "include/loader/loader.h".
187
89
Expect<void> Loader::loadSection(AST::DataCountSection &Sec) {
188
89
  return loadSectionContent(Sec, [this, &Sec]() -> Expect<void> {
189
    // Read u32 of data count.
190
76
    EXPECTED_TRY(uint32_t Cnt, FMgr.readU32().map_error([this](auto E) {
191
74
      return logLoadError(E, FMgr.getLastOffset(), ASTNodeAttr::Sec_DataCount);
192
74
    }));
193
74
    Sec.setContent(Cnt);
194
74
    return {};
195
76
  });
196
89
}
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