| /* |
| * Copyright 2016 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 <cassert> |
| #include <cstdarg> |
| #include <cstdint> |
| #include <cstdio> |
| #include <cstdlib> |
| #include <string> |
| |
| #include "wabt/config.h" |
| |
| #include "wabt/binary-writer.h" |
| #include "wabt/common.h" |
| #include "wabt/error-formatter.h" |
| #include "wabt/feature.h" |
| #include "wabt/filenames.h" |
| #include "wabt/ir.h" |
| #include "wabt/option-parser.h" |
| #include "wabt/resolve-names.h" |
| #include "wabt/stream.h" |
| #include "wabt/validator.h" |
| #include "wabt/wast-parser.h" |
| |
| using namespace wabt; |
| |
| static const char* s_infile; |
| static std::string s_outfile; |
| static bool s_dump_module; |
| static int s_verbose; |
| static WriteBinaryOptions s_write_binary_options; |
| static bool s_validate = true; |
| static Features s_features; |
| |
| static std::unique_ptr<FileStream> s_log_stream; |
| |
| static const char s_description[] = |
| R"( read a file in the wasm text format, check it for errors, and |
| convert it to the wasm binary format. |
| |
| examples: |
| # parse test.wat and write to .wasm binary file with the same name |
| $ wat2wasm test.wat |
| |
| # parse test.wat and write to binary file test.wasm |
| $ wat2wasm test.wat -o test.wasm |
| |
| # parse spec-test.wast, and write verbose output to stdout (including |
| # the meaning of every byte) |
| $ wat2wasm spec-test.wast -v |
| )"; |
| |
| static void ParseOptions(int argc, char* argv[]) { |
| OptionParser parser("wat2wasm", s_description); |
| |
| parser.AddOption('v', "verbose", "Use multiple times for more info", []() { |
| s_verbose++; |
| s_log_stream = FileStream::CreateStderr(); |
| }); |
| parser.AddOption('d', "dump-module", |
| "Print a hexdump of the module to stdout", |
| []() { s_dump_module = true; }); |
| s_features.AddOptions(&parser); |
| parser.AddOption('o', "output", "FILE", |
| "Output wasm binary file. Use \"-\" to write to stdout.", |
| [](const char* argument) { s_outfile = argument; }); |
| parser.AddOption( |
| 'r', "relocatable", |
| "Create a relocatable wasm binary (suitable for linking with e.g. lld)", |
| []() { s_write_binary_options.relocatable = true; }); |
| parser.AddOption( |
| "no-canonicalize-leb128s", |
| "Write all LEB128 sizes as 5-bytes instead of their minimal size", |
| []() { s_write_binary_options.canonicalize_lebs = false; }); |
| parser.AddOption("debug-names", |
| "Write debug names to the generated binary file", |
| []() { s_write_binary_options.write_debug_names = true; }); |
| parser.AddOption("no-check", "Don't check for invalid modules", |
| []() { s_validate = false; }); |
| parser.AddArgument("filename", OptionParser::ArgumentCount::One, |
| [](const char* argument) { s_infile = argument; }); |
| |
| parser.Parse(argc, argv); |
| } |
| |
| static void WriteBufferToFile(std::string_view filename, |
| const OutputBuffer& buffer) { |
| if (s_dump_module) { |
| std::unique_ptr<FileStream> stream = FileStream::CreateStdout(); |
| if (s_verbose) { |
| stream->Writef(";; dump\n"); |
| } |
| if (!buffer.data.empty()) { |
| stream->WriteMemoryDump(buffer.data.data(), buffer.data.size()); |
| } |
| } |
| |
| if (filename == "-") { |
| buffer.WriteToStdout(); |
| } else { |
| buffer.WriteToFile(filename); |
| } |
| } |
| |
| static std::string DefaultOuputName(std::string_view input_name) { |
| // Strip existing extension and add .wasm |
| std::string result(StripExtension(GetBasename(input_name))); |
| result += kWasmExtension; |
| |
| return result; |
| } |
| |
| int ProgramMain(int argc, char** argv) { |
| InitStdio(); |
| |
| ParseOptions(argc, argv); |
| |
| std::vector<uint8_t> file_data; |
| Result result = ReadFile(s_infile, &file_data); |
| Errors errors; |
| std::unique_ptr<WastLexer> lexer = WastLexer::CreateBufferLexer( |
| s_infile, file_data.data(), file_data.size(), &errors); |
| if (Failed(result)) { |
| WABT_FATAL("unable to read file: %s\n", s_infile); |
| } |
| |
| std::unique_ptr<Module> module; |
| WastParseOptions parse_wast_options(s_features); |
| result = ParseWatModule(lexer.get(), &module, &errors, &parse_wast_options); |
| |
| if (Succeeded(result) && s_validate) { |
| ValidateOptions options(s_features); |
| result = ValidateModule(module.get(), &errors, options); |
| } |
| |
| if (Succeeded(result)) { |
| MemoryStream stream(s_log_stream.get()); |
| s_write_binary_options.features = s_features; |
| result = WriteBinaryModule(&stream, module.get(), s_write_binary_options); |
| |
| if (Succeeded(result)) { |
| if (s_outfile.empty()) { |
| s_outfile = DefaultOuputName(s_infile); |
| } |
| WriteBufferToFile(s_outfile.c_str(), stream.output_buffer()); |
| } |
| } |
| |
| auto line_finder = lexer->MakeLineFinder(); |
| FormatErrorsToFile(errors, Location::Type::Text, line_finder.get()); |
| |
| return result != Result::Ok; |
| } |
| |
| int main(int argc, char** argv) { |
| WABT_TRY |
| return ProgramMain(argc, argv); |
| WABT_CATCH_BAD_ALLOC_AND_EXIT |
| } |