Coverage Report

Created: 2025-08-25 06:58

/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
48.7k
Expect<void> Loader::loadSection(AST::CustomSection &Sec) {
18
48.7k
  return loadSectionContent(Sec, [this, &Sec]() -> Expect<void> {
19
48.6k
    auto ReportError = [this](auto E) {
20
101
      return logLoadError(E, FMgr.getLastOffset(), ASTNodeAttr::Sec_Custom);
21
101
    };
22
23
    // Read name.
24
48.6k
    auto StartOffset = FMgr.getOffset();
25
48.6k
    EXPECTED_TRY(std::string Name, FMgr.readName().map_error(ReportError));
26
48.5k
    Sec.setName(Name);
27
48.5k
    auto ReadSize = FMgr.getOffset() - StartOffset;
28
29
    // Read remain bytes. Check is overread or not first.
30
48.5k
    if (unlikely(Sec.getContentSize() < ReadSize)) {
31
27
      return logLoadError(ErrCode::Value::UnexpectedEnd, FMgr.getLastOffset(),
32
27
                          ASTNodeAttr::Sec_Custom);
33
27
    }
34
48.5k
    EXPECTED_TRY(
35
48.5k
        std::vector<uint8_t> Bytes,
36
48.5k
        FMgr.readBytes(Sec.getContentSize() - ReadSize).map_error(ReportError));
37
48.5k
    Sec.getContent().insert(Sec.getContent().end(), Bytes.begin(), Bytes.end());
38
48.5k
    return {};
39
48.5k
  });
40
48.7k
}
41
42
// Load vector of type section. See "include/loader/loader.h".
43
3.87k
Expect<void> Loader::loadSection(AST::TypeSection &Sec) {
44
3.87k
  return loadSectionContent(Sec, [this, &Sec]() -> Expect<void> {
45
3.85k
    auto ReportError = [this](auto E) {
46
18
      return logLoadError(E, FMgr.getLastOffset(), ASTNodeAttr::Sec_Type);
47
18
    };
48
49
    // Read the recursive type vector size.
50
3.85k
    EXPECTED_TRY(uint32_t VecCnt, loadVecCnt().map_error(ReportError));
51
    // Read the recursive types.
52
3.83k
    Sec.getContent().clear();
53
3.83k
    uint32_t SubTypeCnt = 0;
54
10.8k
    for (uint32_t I = 0; I < VecCnt; I++) {
55
7.04k
      EXPECTED_TRY(uint8_t CodeByte, FMgr.peekByte().map_error(ReportError));
56
57
7.04k
      TypeCode Code = static_cast<TypeCode>(CodeByte);
58
7.04k
      if (!Conf.hasProposal(Proposal::GC) && Code != TypeCode::Func) {
59
22
        return logNeedProposal(ErrCode::Value::IntegerTooLong, Proposal::GC,
60
22
                               FMgr.getOffset(), ASTNodeAttr::Sec_Type);
61
22
      }
62
7.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
7.02k
      } else {
77
        // Case: subtype.
78
7.02k
        Sec.getContent().emplace_back();
79
7.02k
        Sec.getContent().back().setTypeIndex(SubTypeCnt);
80
7.02k
        SubTypeCnt++;
81
7.02k
        EXPECTED_TRY(loadType(Sec.getContent().back()).map_error([](auto E) {
82
7.02k
          spdlog::error(ErrInfo::InfoAST(ASTNodeAttr::Sec_Type));
83
7.02k
          return E;
84
7.02k
        }));
85
7.02k
      }
86
7.02k
    }
87
3.75k
    return {};
88
3.83k
  });
89
3.87k
}
90
91
// Load vector of import section. See "include/loader/loader.h".
92
249
Expect<void> Loader::loadSection(AST::ImportSection &Sec) {
93
249
  return loadSectionContent(Sec, [this, &Sec]() {
94
231
    return loadSectionContentVec(
95
10.4k
        Sec, [this](AST::ImportDesc &ImpDesc) { return loadDesc(ImpDesc); });
96
231
  });
97
249
}
98
99
// Load vector of function section. See "include/loader/loader.h".
100
3.68k
Expect<void> Loader::loadSection(AST::FunctionSection &Sec) {
101
3.68k
  return loadSectionContent(Sec, [this, &Sec]() {
102
3.67k
    return loadSectionContentVec(
103
29.7k
        Sec, [this](uint32_t &FuncIdx) -> Expect<void> {
104
29.7k
          EXPECTED_TRY(uint32_t Idx, FMgr.readU32().map_error([this](auto E) {
105
29.7k
            return logLoadError(E, FMgr.getLastOffset(),
106
29.7k
                                ASTNodeAttr::Sec_Function);
107
29.7k
          }));
108
29.7k
          FuncIdx = Idx;
109
29.7k
          return {};
110
29.7k
        });
111
3.67k
  });
112
3.68k
}
113
114
// Load vector of table section. See "include/loader/loader.h".
115
360
Expect<void> Loader::loadSection(AST::TableSection &Sec) {
116
360
  return loadSectionContent(Sec, [this, &Sec]() {
117
345
    return loadSectionContentVec(
118
497
        Sec, [this](AST::TableSegment &TabSeg) { return loadSegment(TabSeg); });
119
345
  });
120
360
}
121
122
// Load vector of memory section. See "include/loader/loader.h".
123
1.38k
Expect<void> Loader::loadSection(AST::MemorySection &Sec) {
124
1.38k
  return loadSectionContent(Sec, [this, &Sec]() {
125
1.36k
    return loadSectionContentVec(
126
2.81k
        Sec, [this](AST::MemoryType &MemType) { return loadType(MemType); });
127
1.36k
  });
128
1.38k
}
129
130
// Load vector of global section. See "include/loader/loader.h".
131
475
Expect<void> Loader::loadSection(AST::GlobalSection &Sec) {
132
475
  return loadSectionContent(Sec, [this, &Sec]() {
133
741
    return loadSectionContentVec(Sec, [this](AST::GlobalSegment &GlobSeg) {
134
741
      return loadSegment(GlobSeg);
135
741
    });
136
462
  });
137
475
}
138
139
// Load vector of export section. See "include/loader/loader.h".
140
990
Expect<void> Loader::loadSection(AST::ExportSection &Sec) {
141
990
  return loadSectionContent(Sec, [this, &Sec]() {
142
975
    return loadSectionContentVec(
143
15.8k
        Sec, [this](AST::ExportDesc &ExpDesc) { return loadDesc(ExpDesc); });
144
975
  });
145
990
}
146
147
// Load start function index. See "include/loader/loader.h".
148
37
Expect<void> Loader::loadSection(AST::StartSection &Sec) {
149
37
  return loadSectionContent(Sec, [this, &Sec]() -> Expect<void> {
150
    // Read u32 of start function index.
151
23
    EXPECTED_TRY(uint32_t Idx, FMgr.readU32().map_error([this](auto E) {
152
21
      return logLoadError(E, FMgr.getLastOffset(), ASTNodeAttr::Sec_Start);
153
21
    }));
154
21
    Sec.setContent(Idx);
155
21
    return {};
156
23
  });
157
37
}
158
159
// Load vector of element section. See "include/loader/loader.h".
160
1.27k
Expect<void> Loader::loadSection(AST::ElementSection &Sec) {
161
1.27k
  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.26k
  });
166
1.27k
}
167
168
// Load vector of code section. See "include/loader/loader.h".
169
3.86k
Expect<void> Loader::loadSection(AST::CodeSection &Sec) {
170
3.86k
  return loadSectionContent(Sec, [this, &Sec]() {
171
17.0k
    return loadSectionContentVec(Sec, [this](AST::CodeSegment &CodeSeg) {
172
17.0k
      return loadSegment(CodeSeg);
173
17.0k
    });
174
3.86k
  });
175
3.86k
}
176
177
// Load vector of data section. See "include/loader/loader.h".
178
1.46k
Expect<void> Loader::loadSection(AST::DataSection &Sec) {
179
1.46k
  return loadSectionContent(Sec, [this, &Sec]() {
180
4.20k
    return loadSectionContentVec(Sec, [this](AST::DataSegment &DataSeg) {
181
4.20k
      return loadSegment(DataSeg);
182
4.20k
    });
183
1.44k
  });
184
1.46k
}
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