blob: 6f501991490d4c09145ae81484719a6a7b000b00 [file] [edit]
/*
* Copyright 2017 WebAssembly Community Group participants
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "gtest/gtest.h"
#include <memory>
#include "wabt/cast.h"
#include "wabt/wast-lexer.h"
#include "wabt/wast-parser.h"
using namespace wabt;
namespace {
std::string repeat(std::string s, size_t count) {
std::string result;
for (size_t i = 0; i < count; ++i) {
result += s;
}
return result;
}
Errors ParseInvalidModule(std::string text) {
Errors errors;
auto lexer =
WastLexer::CreateBufferLexer("test", text.c_str(), text.size(), &errors);
std::unique_ptr<Module> module;
Features features;
WastParseOptions options(features);
Result result = ParseWatModule(lexer.get(), &module, &errors, &options);
EXPECT_EQ(Result::Error, result);
return errors;
}
} // end of anonymous namespace
TEST(WastParser, LongToken) {
std::string text;
text = "(module (memory ";
text += repeat("a", 0x5000);
text += "))";
Errors errors = ParseInvalidModule(text);
ASSERT_EQ(1u, errors.size());
ASSERT_EQ(ErrorLevel::Error, errors[0].error_level);
ASSERT_EQ(1u, errors[0].loc.line);
ASSERT_EQ(17u, errors[0].loc.first_column);
ASSERT_STREQ(
R"(unexpected token "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa...", expected a natural number (e.g. 123).)",
errors[0].message.c_str());
}
TEST(WastParser, LongTokenSpace) {
std::string text;
text = "notparen";
text += repeat(" ", 0x10000);
text += "notmodule";
Errors errors = ParseInvalidModule(text);
ASSERT_EQ(2u, errors.size());
ASSERT_EQ(ErrorLevel::Error, errors[0].error_level);
ASSERT_EQ(1u, errors[0].loc.line);
ASSERT_EQ(1u, errors[0].loc.first_column);
ASSERT_STREQ(
R"(unexpected token "notparen", expected a module field or a module.)",
errors[0].message.c_str());
ASSERT_EQ(1u, errors[1].loc.line);
ASSERT_EQ(65545u, errors[1].loc.first_column);
ASSERT_STREQ(R"(unexpected token notmodule, expected EOF.)",
errors[1].message.c_str());
}
TEST(WastParser, Memory64Alignment) {
// from align.wast, the large align param should be OK with memory64 enabled
// TODO: this test can be deleted once the spec test suite is updated
// to include
// https://github.com/WebAssembly/testsuite/commit/4b24564c844e3d34bf46dfcb3c774ee5163e31cc
std::string text = R"(
(module
(memory 1)
(func
i32.const 0
i32.load offset=0xFFFF_FFFF_FFFF_FFFF align=0x8000_0000_0000_0000
drop
)
))";
Features features;
features.enable_memory64();
WastParseOptions options(features);
Errors errors;
auto lexer =
WastLexer::CreateBufferLexer("test", text.c_str(), text.size(), &errors);
std::unique_ptr<Script> script;
Result result = ParseWastScript(lexer.get(), &script, &errors, &options);
EXPECT_EQ(Result::Ok, result);
}
TEST(WastParser, InvalidBinaryModule) {
std::string text = R"((module binary "\00asm\bc\0a\00\00"))";
std::vector<uint8_t> expected_data = {
0, 'a', 's', 'm', static_cast<uint8_t>('\xbc'), '\x0a', 0, 0};
Errors errors;
auto lexer =
WastLexer::CreateBufferLexer("test", text.c_str(), text.size(), &errors);
std::unique_ptr<Script> script;
Features features;
WastParseOptions options(features);
options.parse_binary_modules = false;
Result result = ParseWastScript(lexer.get(), &script, &errors, &options);
EXPECT_EQ(result, Result::Ok);
EXPECT_EQ(0u, errors.size());
EXPECT_EQ(1u, script->commands.size());
Command* cmd = script->commands[0].get();
EXPECT_TRUE(isa<ScriptModuleCommand>(cmd));
auto* mod_cmd = cast<ScriptModuleCommand>(cmd);
ScriptModule* script_mod = mod_cmd->script_module.get();
EXPECT_NE(script_mod, nullptr);
EXPECT_TRUE(isa<BinaryScriptModule>(script_mod));
auto* binary_mod = cast<BinaryScriptModule>(script_mod);
EXPECT_EQ(expected_data, binary_mod->data);
}