/* * 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 "src/wat-writer.h" #include #include #include #include #include #include #include #include #include #include #include "src/cast.h" #include "src/common.h" #include "src/expr-visitor.h" #include "src/ir.h" #include "src/literal.h" #include "src/stream.h" #define WABT_TRACING 0 #include "src/tracing.h" #define INDENT_SIZE 2 #define NO_FORCE_NEWLINE 0 #define FORCE_NEWLINE 1 namespace wabt { namespace { static const uint8_t s_is_char_escaped[] = { 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1}; // This table matches the characters allowed by wast-lexer.cc for `symbol`. // The disallowed printable characters are: "(),;[]{} and . static const uint8_t s_valid_name_chars[256] = { // 0 1 2 3 4 5 6 7 8 9 a b c d e f /* 0x00 */ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, /* 0x10 */ 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, /* 0x20 */ 0, 1, 0, 1, 1, 1, 1, 1, 0, 0, 1, 1, 0, 1, 1, 1, /* 0x30 */ 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, /* 0x40 */ 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, /* 0x50 */ 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 0, 1, 1, /* 0x60 */ 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, /* 0x70 */ 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0, 1, 0, 1, 0, }; enum class NextChar { None, Space, Newline, ForceNewline, }; struct ExprTree { explicit ExprTree(const Expr* expr, Index result_count) : expr(expr), result_count(result_count) {} const Expr* expr; std::vector children; Index result_count; }; struct Label { Label(LabelType label_type, const std::string& name, const TypeVector& param_types, const TypeVector& result_types) : name(name), label_type(label_type), param_types(param_types), result_types(result_types) {} std::string name; LabelType label_type; TypeVector param_types; TypeVector result_types; }; class WatWriter { public: WatWriter(Stream* stream, const WriteWatOptions& options) : options_(options), stream_(stream) {} Result WriteModule(const Module& module); private: void Indent(); void Dedent(); void WriteIndent(); void WriteNextChar(); void WriteDataWithNextChar(const void* src, size_t size); void Writef(const char* format, ...); void WritePutc(char c); void WritePuts(const char* s, NextChar next_char); void WritePutsSpace(const char* s); void WritePutsNewline(const char* s); void WriteNewline(bool force); void WriteOpen(const char* name, NextChar next_char); void WriteOpenNewline(const char* name); void WriteOpenSpace(const char* name); void WriteClose(NextChar next_char); void WriteCloseNewline(); void WriteCloseSpace(); void WriteString(const std::string& str, NextChar next_char); void WriteName(string_view str, NextChar next_char); void WriteNameOrIndex(string_view str, Index index, NextChar next_char); void WriteQuotedData(const void* data, size_t length); void WriteQuotedString(string_view str, NextChar next_char); void WriteVar(const Var& var, NextChar next_char); void WriteBrVar(const Var& var, NextChar next_char); void WriteType(Type type, NextChar next_char); void WriteTypes(const TypeVector& types, const char* name); void WriteFuncSigSpace(const FuncSignature& func_sig); void WriteBeginBlock(LabelType label_type, const Block& block, const char* text); void WriteEndBlock(); void WriteConst(const Const& const_); void WriteExpr(const Expr* expr); template void WriteLoadStoreExpr(const Expr* expr); void WriteExprList(const ExprList& exprs); void WriteInitExpr(const ExprList& expr); template void WriteTypeBindings(const char* prefix, const T& types, const std::vector& index_to_name, Index binding_index_offset = 0); void WriteBeginFunc(const Func& func); void WriteFunc(const Func& func); void WriteBeginGlobal(const Global& global); void WriteGlobal(const Global& global); void WriteEvent(const Event& event); void WriteLimits(const Limits& limits); void WriteTable(const Table& table); void WriteElemSegment(const ElemSegment& segment); void WriteMemory(const Memory& memory); void WriteDataSegment(const DataSegment& segment); void WriteImport(const Import& import); void WriteExport(const Export& export_); void WriteFuncType(const FuncType& func_type); void WriteStartFunction(const Var& start); class ExprVisitorDelegate; Index GetLabelStackSize() { return label_stack_.size(); } Label* GetLabel(const Var& var); Index GetLabelArity(const Var& var); Index GetFuncParamCount(const Var& var); Index GetFuncResultCount(const Var& var); void PushExpr(const Expr* expr, Index operand_count, Index result_count); void FlushExprTree(const ExprTree& expr_tree); void FlushExprTreeVector(const std::vector&); void FlushExprTreeStack(); void WriteFoldedExpr(const Expr*); void WriteFoldedExprList(const ExprList&); void BuildInlineExportMap(); void WriteInlineExports(ExternalKind, Index); bool IsInlineExport(const Export& export_); void BuildInlineImportMap(); void WriteInlineImport(ExternalKind, Index); const WriteWatOptions& options_; const Module* module_ = nullptr; const Func* current_func_ = nullptr; Stream* stream_ = nullptr; Result result_ = Result::Ok; int indent_ = 0; NextChar next_char_ = NextChar::None; std::vector