1 //===- WasmYAML.cpp - Wasm YAMLIO implementation --------------------------===//
3 // The LLVM Compiler Infrastructure
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
8 //===----------------------------------------------------------------------===//
10 // This file defines classes for handling the YAML representation of wasm.
12 //===----------------------------------------------------------------------===//
14 #include "llvm/ObjectYAML/WasmYAML.h"
15 #include "llvm/ADT/StringRef.h"
16 #include "llvm/Support/Casting.h"
17 #include "llvm/Support/ErrorHandling.h"
18 #include "llvm/Support/YAMLTraits.h"
24 // Declared here rather than in the header to comply with:
25 // http://llvm.org/docs/CodingStandards.html#provide-a-virtual-method-anchor-for-classes-in-headers
26 Section::~Section() = default;
28 } // end namespace WasmYAML
32 void MappingTraits<WasmYAML::FileHeader>::mapping(
33 IO &IO, WasmYAML::FileHeader &FileHdr) {
34 IO.mapRequired("Version", FileHdr.Version);
37 void MappingTraits<WasmYAML::Object>::mapping(IO &IO,
38 WasmYAML::Object &Object) {
39 IO.setContext(&Object);
40 IO.mapTag("!WASM", true);
41 IO.mapRequired("FileHeader", Object.Header);
42 IO.mapOptional("Sections", Object.Sections);
43 IO.setContext(nullptr);
46 static void commonSectionMapping(IO &IO, WasmYAML::Section &Section) {
47 IO.mapRequired("Type", Section.Type);
48 IO.mapOptional("Relocations", Section.Relocations);
51 static void sectionMapping(IO &IO, WasmYAML::NameSection &Section) {
52 commonSectionMapping(IO, Section);
53 IO.mapRequired("Name", Section.Name);
54 IO.mapOptional("FunctionNames", Section.FunctionNames);
57 static void sectionMapping(IO &IO, WasmYAML::LinkingSection &Section) {
58 commonSectionMapping(IO, Section);
59 IO.mapRequired("Name", Section.Name);
60 IO.mapRequired("DataSize", Section.DataSize);
61 IO.mapRequired("DataAlignment", Section.DataAlignment);
62 IO.mapRequired("SymbolInfo", Section.SymbolInfos);
65 static void sectionMapping(IO &IO, WasmYAML::CustomSection &Section) {
66 commonSectionMapping(IO, Section);
67 IO.mapRequired("Name", Section.Name);
68 IO.mapRequired("Payload", Section.Payload);
71 static void sectionMapping(IO &IO, WasmYAML::TypeSection &Section) {
72 commonSectionMapping(IO, Section);
73 IO.mapOptional("Signatures", Section.Signatures);
76 static void sectionMapping(IO &IO, WasmYAML::ImportSection &Section) {
77 commonSectionMapping(IO, Section);
78 IO.mapOptional("Imports", Section.Imports);
81 static void sectionMapping(IO &IO, WasmYAML::FunctionSection &Section) {
82 commonSectionMapping(IO, Section);
83 IO.mapOptional("FunctionTypes", Section.FunctionTypes);
86 static void sectionMapping(IO &IO, WasmYAML::TableSection &Section) {
87 commonSectionMapping(IO, Section);
88 IO.mapOptional("Tables", Section.Tables);
91 static void sectionMapping(IO &IO, WasmYAML::MemorySection &Section) {
92 commonSectionMapping(IO, Section);
93 IO.mapOptional("Memories", Section.Memories);
96 static void sectionMapping(IO &IO, WasmYAML::GlobalSection &Section) {
97 commonSectionMapping(IO, Section);
98 IO.mapOptional("Globals", Section.Globals);
101 static void sectionMapping(IO &IO, WasmYAML::ExportSection &Section) {
102 commonSectionMapping(IO, Section);
103 IO.mapOptional("Exports", Section.Exports);
106 static void sectionMapping(IO &IO, WasmYAML::StartSection &Section) {
107 commonSectionMapping(IO, Section);
108 IO.mapOptional("StartFunction", Section.StartFunction);
111 static void sectionMapping(IO &IO, WasmYAML::ElemSection &Section) {
112 commonSectionMapping(IO, Section);
113 IO.mapOptional("Segments", Section.Segments);
116 static void sectionMapping(IO &IO, WasmYAML::CodeSection &Section) {
117 commonSectionMapping(IO, Section);
118 IO.mapRequired("Functions", Section.Functions);
121 static void sectionMapping(IO &IO, WasmYAML::DataSection &Section) {
122 commonSectionMapping(IO, Section);
123 IO.mapRequired("Segments", Section.Segments);
126 void MappingTraits<std::unique_ptr<WasmYAML::Section>>::mapping(
127 IO &IO, std::unique_ptr<WasmYAML::Section> &Section) {
128 WasmYAML::SectionType SectionType;
130 SectionType = Section->Type;
132 IO.mapRequired("Type", SectionType);
134 switch (SectionType) {
135 case wasm::WASM_SEC_CUSTOM: {
136 StringRef SectionName;
137 if (IO.outputting()) {
138 auto CustomSection = cast<WasmYAML::CustomSection>(Section.get());
139 SectionName = CustomSection->Name;
141 IO.mapRequired("Name", SectionName);
143 if (SectionName == "linking") {
144 if (!IO.outputting())
145 Section.reset(new WasmYAML::LinkingSection());
146 sectionMapping(IO, *cast<WasmYAML::LinkingSection>(Section.get()));
147 } else if (SectionName == "name") {
148 if (!IO.outputting())
149 Section.reset(new WasmYAML::NameSection());
150 sectionMapping(IO, *cast<WasmYAML::NameSection>(Section.get()));
152 if (!IO.outputting())
153 Section.reset(new WasmYAML::CustomSection(SectionName));
154 sectionMapping(IO, *cast<WasmYAML::CustomSection>(Section.get()));
158 case wasm::WASM_SEC_TYPE:
159 if (!IO.outputting())
160 Section.reset(new WasmYAML::TypeSection());
161 sectionMapping(IO, *cast<WasmYAML::TypeSection>(Section.get()));
163 case wasm::WASM_SEC_IMPORT:
164 if (!IO.outputting())
165 Section.reset(new WasmYAML::ImportSection());
166 sectionMapping(IO, *cast<WasmYAML::ImportSection>(Section.get()));
168 case wasm::WASM_SEC_FUNCTION:
169 if (!IO.outputting())
170 Section.reset(new WasmYAML::FunctionSection());
171 sectionMapping(IO, *cast<WasmYAML::FunctionSection>(Section.get()));
173 case wasm::WASM_SEC_TABLE:
174 if (!IO.outputting())
175 Section.reset(new WasmYAML::TableSection());
176 sectionMapping(IO, *cast<WasmYAML::TableSection>(Section.get()));
178 case wasm::WASM_SEC_MEMORY:
179 if (!IO.outputting())
180 Section.reset(new WasmYAML::MemorySection());
181 sectionMapping(IO, *cast<WasmYAML::MemorySection>(Section.get()));
183 case wasm::WASM_SEC_GLOBAL:
184 if (!IO.outputting())
185 Section.reset(new WasmYAML::GlobalSection());
186 sectionMapping(IO, *cast<WasmYAML::GlobalSection>(Section.get()));
188 case wasm::WASM_SEC_EXPORT:
189 if (!IO.outputting())
190 Section.reset(new WasmYAML::ExportSection());
191 sectionMapping(IO, *cast<WasmYAML::ExportSection>(Section.get()));
193 case wasm::WASM_SEC_START:
194 if (!IO.outputting())
195 Section.reset(new WasmYAML::StartSection());
196 sectionMapping(IO, *cast<WasmYAML::StartSection>(Section.get()));
198 case wasm::WASM_SEC_ELEM:
199 if (!IO.outputting())
200 Section.reset(new WasmYAML::ElemSection());
201 sectionMapping(IO, *cast<WasmYAML::ElemSection>(Section.get()));
203 case wasm::WASM_SEC_CODE:
204 if (!IO.outputting())
205 Section.reset(new WasmYAML::CodeSection());
206 sectionMapping(IO, *cast<WasmYAML::CodeSection>(Section.get()));
208 case wasm::WASM_SEC_DATA:
209 if (!IO.outputting())
210 Section.reset(new WasmYAML::DataSection());
211 sectionMapping(IO, *cast<WasmYAML::DataSection>(Section.get()));
214 llvm_unreachable("Unknown section type");
218 void ScalarEnumerationTraits<WasmYAML::SectionType>::enumeration(
219 IO &IO, WasmYAML::SectionType &Type) {
220 #define ECase(X) IO.enumCase(Type, #X, wasm::WASM_SEC_##X);
236 void MappingTraits<WasmYAML::Signature>::mapping(
237 IO &IO, WasmYAML::Signature &Signature) {
238 IO.mapOptional("Index", Signature.Index);
239 IO.mapRequired("ReturnType", Signature.ReturnType);
240 IO.mapRequired("ParamTypes", Signature.ParamTypes);
243 void MappingTraits<WasmYAML::Table>::mapping(IO &IO, WasmYAML::Table &Table) {
244 IO.mapRequired("ElemType", Table.ElemType);
245 IO.mapRequired("Limits", Table.TableLimits);
248 void MappingTraits<WasmYAML::Function>::mapping(IO &IO,
249 WasmYAML::Function &Function) {
250 IO.mapRequired("Locals", Function.Locals);
251 IO.mapRequired("Body", Function.Body);
254 void MappingTraits<WasmYAML::Relocation>::mapping(
255 IO &IO, WasmYAML::Relocation &Relocation) {
256 IO.mapRequired("Type", Relocation.Type);
257 IO.mapRequired("Index", Relocation.Index);
258 IO.mapRequired("Offset", Relocation.Offset);
259 IO.mapOptional("Addend", Relocation.Addend, 0);
262 void MappingTraits<WasmYAML::NameEntry>::mapping(
263 IO &IO, WasmYAML::NameEntry &NameEntry) {
264 IO.mapRequired("Index", NameEntry.Index);
265 IO.mapRequired("Name", NameEntry.Name);
268 void MappingTraits<WasmYAML::LocalDecl>::mapping(
269 IO &IO, WasmYAML::LocalDecl &LocalDecl) {
270 IO.mapRequired("Type", LocalDecl.Type);
271 IO.mapRequired("Count", LocalDecl.Count);
274 void MappingTraits<WasmYAML::Limits>::mapping(IO &IO,
275 WasmYAML::Limits &Limits) {
276 if (!IO.outputting() || Limits.Flags)
277 IO.mapOptional("Flags", Limits.Flags);
278 IO.mapRequired("Initial", Limits.Initial);
279 if (!IO.outputting() || Limits.Flags & wasm::WASM_LIMITS_FLAG_HAS_MAX)
280 IO.mapOptional("Maximum", Limits.Maximum);
283 void MappingTraits<WasmYAML::ElemSegment>::mapping(
284 IO &IO, WasmYAML::ElemSegment &Segment) {
285 IO.mapRequired("Offset", Segment.Offset);
286 IO.mapRequired("Functions", Segment.Functions);
289 void MappingTraits<WasmYAML::Import>::mapping(IO &IO,
290 WasmYAML::Import &Import) {
291 IO.mapRequired("Module", Import.Module);
292 IO.mapRequired("Field", Import.Field);
293 IO.mapRequired("Kind", Import.Kind);
294 if (Import.Kind == wasm::WASM_EXTERNAL_FUNCTION) {
295 IO.mapRequired("SigIndex", Import.SigIndex);
296 } else if (Import.Kind == wasm::WASM_EXTERNAL_GLOBAL) {
297 IO.mapRequired("GlobalType", Import.GlobalImport.Type);
298 IO.mapRequired("GlobalMutable", Import.GlobalImport.Mutable);
299 } else if (Import.Kind == wasm::WASM_EXTERNAL_TABLE) {
300 IO.mapRequired("Table", Import.TableImport);
301 } else if (Import.Kind == wasm::WASM_EXTERNAL_MEMORY ) {
302 IO.mapRequired("Memory", Import.Memory);
304 llvm_unreachable("unhandled import type");
308 void MappingTraits<WasmYAML::Export>::mapping(IO &IO,
309 WasmYAML::Export &Export) {
310 IO.mapRequired("Name", Export.Name);
311 IO.mapRequired("Kind", Export.Kind);
312 IO.mapRequired("Index", Export.Index);
315 void MappingTraits<WasmYAML::Global>::mapping(IO &IO,
316 WasmYAML::Global &Global) {
317 IO.mapRequired("Type", Global.Type);
318 IO.mapRequired("Mutable", Global.Mutable);
319 IO.mapRequired("InitExpr", Global.InitExpr);
322 void MappingTraits<wasm::WasmInitExpr>::mapping(IO &IO,
323 wasm::WasmInitExpr &Expr) {
324 WasmYAML::Opcode Op = Expr.Opcode;
325 IO.mapRequired("Opcode", Op);
327 switch (Expr.Opcode) {
328 case wasm::WASM_OPCODE_I32_CONST:
329 IO.mapRequired("Value", Expr.Value.Int32);
331 case wasm::WASM_OPCODE_I64_CONST:
332 IO.mapRequired("Value", Expr.Value.Int64);
334 case wasm::WASM_OPCODE_F32_CONST:
335 IO.mapRequired("Value", Expr.Value.Float32);
337 case wasm::WASM_OPCODE_F64_CONST:
338 IO.mapRequired("Value", Expr.Value.Float64);
340 case wasm::WASM_OPCODE_GET_GLOBAL:
341 IO.mapRequired("Index", Expr.Value.Global);
346 void MappingTraits<WasmYAML::DataSegment>::mapping(
347 IO &IO, WasmYAML::DataSegment &Segment) {
348 IO.mapRequired("Index", Segment.Index);
349 IO.mapRequired("Offset", Segment.Offset);
350 IO.mapRequired("Content", Segment.Content);
353 void MappingTraits<WasmYAML::SymbolInfo>::mapping(IO &IO,
354 WasmYAML::SymbolInfo &Info) {
355 IO.mapRequired("Name", Info.Name);
356 IO.mapRequired("Flags", Info.Flags);
359 void ScalarEnumerationTraits<WasmYAML::ValueType>::enumeration(
360 IO &IO, WasmYAML::ValueType &Type) {
361 #define ECase(X) IO.enumCase(Type, #X, wasm::WASM_TYPE_##X);
372 void ScalarEnumerationTraits<WasmYAML::ExportKind>::enumeration(
373 IO &IO, WasmYAML::ExportKind &Kind) {
374 #define ECase(X) IO.enumCase(Kind, #X, wasm::WASM_EXTERNAL_##X);
382 void ScalarEnumerationTraits<WasmYAML::Opcode>::enumeration(
383 IO &IO, WasmYAML::Opcode &Code) {
384 #define ECase(X) IO.enumCase(Code, #X, wasm::WASM_OPCODE_##X);
394 void ScalarEnumerationTraits<WasmYAML::TableType>::enumeration(
395 IO &IO, WasmYAML::TableType &Type) {
396 #define ECase(X) IO.enumCase(Type, #X, wasm::WASM_TYPE_##X);
401 void ScalarEnumerationTraits<WasmYAML::RelocType>::enumeration(
402 IO &IO, WasmYAML::RelocType &Type) {
403 #define WASM_RELOC(name, value) IO.enumCase(Type, #name, wasm::name);
404 #include "llvm/BinaryFormat/WasmRelocs/WebAssembly.def"
408 } // end namespace yaml
410 } // end namespace llvm