summaryrefslogtreecommitdiff
path: root/src/wast-parser.cc
diff options
context:
space:
mode:
authorDiego Frias <styx5242@gmail.com>2023-09-15 13:49:31 -0700
committerGitHub <noreply@github.com>2023-09-15 13:49:31 -0700
commitb22be5a08611c53e8a893bbab049e6f18da6c55a (patch)
tree9b998aee94bada6db5b5715afe99d806167c8d52 /src/wast-parser.cc
parent0869da3eb84616b0dabefc34ea0d5326c1f43888 (diff)
downloadwabt-b22be5a08611c53e8a893bbab049e6f18da6c55a.tar.gz
wabt-b22be5a08611c53e8a893bbab049e6f18da6c55a.tar.bz2
wabt-b22be5a08611c53e8a893bbab049e6f18da6c55a.zip
Implement custom section reading/writing (#2284)
Diffstat (limited to 'src/wast-parser.cc')
-rw-r--r--src/wast-parser.cc76
1 files changed, 69 insertions, 7 deletions
diff --git a/src/wast-parser.cc b/src/wast-parser.cc
index 5115650f..fb7bd48e 100644
--- a/src/wast-parser.cc
+++ b/src/wast-parser.cc
@@ -599,14 +599,16 @@ TokenType WastParser::Peek(size_t n) {
tokens_.push_back(cur);
} else {
// Custom annotation. For now, discard until matching Rpar, unless it is
- // a code metadata annotation. In that case, we know how to parse it.
+ // a code metadata annotation or custom section. In those cases, we know
+ // how to parse it.
if (!options_->features.annotations_enabled()) {
Error(cur.loc, "annotations not enabled: %s", cur.to_string().c_str());
tokens_.push_back(Token(cur.loc, TokenType::Invalid));
continue;
}
- if (options_->features.code_metadata_enabled() &&
- cur.text().find("metadata.code.") == 0) {
+ if ((options_->features.code_metadata_enabled() &&
+ cur.text().find("metadata.code.") == 0) ||
+ cur.text() == "custom") {
tokens_.push_back(cur);
continue;
}
@@ -1172,7 +1174,7 @@ Result WastParser::ParseModule(std::unique_ptr<Module>* out_module) {
auto module_command = cast<ScriptModuleCommand>(std::move(command));
*module = std::move(module_command->module);
}
- } else if (IsModuleField(PeekPair())) {
+ } else if (IsModuleField(PeekPair()) || PeekIsCustom()) {
// Parse an inline module (i.e. one with no surrounding (module)).
CHECK_RESULT(ParseModuleFieldList(module.get()));
} else if (PeekMatch(TokenType::Eof)) {
@@ -1198,7 +1200,7 @@ Result WastParser::ParseScript(std::unique_ptr<Script>* out_script) {
// Don't consume the Lpar yet, even though it is required. This way the
// sub-parser functions (e.g. ParseFuncModuleField) can consume it and keep
// the parsing structure more regular.
- if (IsModuleField(PeekPair())) {
+ if (IsModuleField(PeekPair()) || PeekIsCustom()) {
// Parse an inline module (i.e. one with no surrounding (module)).
auto command = std::make_unique<ModuleCommand>();
command->module.loc = GetLocation();
@@ -1222,9 +1224,69 @@ Result WastParser::ParseScript(std::unique_ptr<Script>* out_script) {
}
}
+Result WastParser::ParseCustomSectionAnnotation(Module* module) {
+ WABT_TRACE(ParseCustomSectionAnnotation);
+ Location loc = GetLocation();
+ Token token = Consume();
+ if (token.text() != "custom") {
+ assert(
+ !"ParseCustomSectionAnnotation should only be called if PeekIsCustom() is true");
+ return Result::Error;
+ }
+ std::string section_name;
+ CHECK_RESULT(ParseQuotedText(&section_name));
+ if (Match(TokenType::Lpar)) {
+ if (!PeekMatch(TokenType::After) && !PeekMatch(TokenType::Before)) {
+ return ErrorExpected({"before", "after"});
+ }
+ Consume();
+ switch (Peek()) {
+ case TokenType::Function:
+ case TokenType::Type:
+ case TokenType::Import:
+ case TokenType::Export:
+ case TokenType::Table:
+ case TokenType::Global:
+ case TokenType::Elem:
+ case TokenType::Data:
+ case TokenType::Memory:
+ case TokenType::Code:
+ case TokenType::Start: {
+ Consume();
+ break;
+ }
+ default: {
+ return ErrorExpected({"type", "import", "function", "table", "memory",
+ "global", "export", "start", "elem", "code",
+ "data"});
+ }
+ }
+ EXPECT(Rpar);
+ }
+ std::vector<uint8_t> data;
+ CHECK_RESULT(ParseTextList(&data));
+ EXPECT(Rpar);
+
+ Custom custom = Custom(loc, section_name, data);
+ module->customs.push_back(custom);
+
+ return Result::Ok;
+}
+
+bool WastParser::PeekIsCustom() {
+ // If IsLparAnn succeeds, tokens_.front() must have text, as it is an LparAnn
+ // token.
+ return options_->features.annotations_enabled() && IsLparAnn(PeekPair()) &&
+ tokens_.front().text() == "custom";
+}
+
Result WastParser::ParseModuleFieldList(Module* module) {
WABT_TRACE(ParseModuleFieldList);
- while (IsModuleField(PeekPair())) {
+ while (IsModuleField(PeekPair()) || PeekIsCustom()) {
+ if (PeekIsCustom()) {
+ CHECK_RESULT(ParseCustomSectionAnnotation(module));
+ continue;
+ }
if (Failed(ParseModuleField(module))) {
CHECK_RESULT(Synchronize(IsModuleField));
}
@@ -3527,7 +3589,7 @@ Result WastParser::ParseScriptModule(
auto tsm = std::make_unique<TextScriptModule>();
tsm->module.name = name;
tsm->module.loc = loc;
- if (IsModuleField(PeekPair())) {
+ if (IsModuleField(PeekPair()) || PeekIsCustom()) {
CHECK_RESULT(ParseModuleFieldList(&tsm->module));
} else if (!PeekMatch(TokenType::Rpar)) {
ConsumeIfLpar();