/* * 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 "wabt/interp/binary-reader-interp.h" #include #include #include "wabt/binary-reader-nop.h" #include "wabt/feature.h" #include "wabt/interp/interp.h" #include "wabt/shared-validator.h" #include "wabt/stream.h" namespace wabt { namespace interp { namespace { ValueTypes ToInterp(Index count, Type* types) { return ValueTypes(&types[0], &types[count]); } Mutability ToMutability(bool mut) { return mut ? Mutability::Var : Mutability::Const; } SegmentMode ToSegmentMode(uint8_t flags) { if ((flags & SegDeclared) == SegDeclared) { return SegmentMode::Declared; } else if ((flags & SegPassive) == SegPassive) { return SegmentMode::Passive; } else { return SegmentMode::Active; } } // This is only used to distinguish try blocks and all other blocks, // so there are only two kinds. enum class LabelKind { Block, Try }; struct Label { LabelKind kind; Istream::Offset offset; Istream::Offset fixup_offset; // Only needs to be set for try blocks. u32 handler_desc_index; }; struct FixupMap { using Offset = Istream::Offset; using Fixups = std::vector; void Clear(); void Append(Index, Offset); void Resolve(Istream&, Index); std::map map; }; class BinaryReaderInterp : public BinaryReaderNop { public: BinaryReaderInterp(ModuleDesc* module, std::string_view filename, Errors* errors, const Features& features); // Implement BinaryReader. bool OnError(const Error&) override; Result EndModule() override; Result OnTypeCount(Index count) override; Result OnFuncType(Index index, Index param_count, Type* param_types, Index result_count, Type* result_types) override; Result OnImportFunc(Index import_index, std::string_view module_name, std::string_view field_name, Index func_index, Index sig_index) override; Result OnImportTable(Index import_index, std::string_view module_name, std::string_view field_name, Index table_index, Type elem_type, const Limits* elem_limits) override; Result OnImportMemory(Index import_index, std::string_view module_name, std::string_view field_name, Index memory_index, const Limits* page_limits) override; Result OnImportGlobal(Index import_index, std::string_view module_name, std::string_view field_name, Index global_index, Type type, bool mutable_) override; Result OnImportTag(Index import_index, std::string_view module_name, std::string_view field_name, Index tag_index, Index sig_index) override; Result OnFunctionCount(Index count) override; Result OnFunction(Index index, Index sig_index) override; Result OnTableCount(Index count) override; Result OnTable(Index index, Type elem_type, const Limits* elem_limits) override; Result OnMemoryCount(Index count) override; Result OnMemory(Index index, const Limits* limits) override; Result OnGlobalCount(Index count) override; Result BeginGlobal(Index index, Type type, bool mutable_) override; Result BeginGlobalInitExpr(Index index) override; Result EndGlobalInitExpr(Index index) override; Result OnTagCount(Index count) override; Result OnTagType(Index index, Index sig_index) override; Result OnExport(Index index, ExternalKind kind, Index item_index, std::string_view name) override; Result OnStartFunction(Index func_index) override; Result BeginFunctionBody(Index index, Offset size) override; Result OnLocalDeclCount(Index count) override; Result OnLocalDecl(Index decl_index, Index count, Type type) override; Result EndLocalDecls() override; Result OnOpcode(Opcode Opcode) override; Result OnAtomicLoadExpr(Opcode opcode, Index memidx, Address alignment_log2, Address offset) override; Result OnAtomicStoreExpr(Opcode opcode, Index memidx, Address alignment_log2, Address offset) override; Result OnAtomicRmwExpr(Opcode opcode, Index memidx, Address alignment_log2, Address offset) override; Result OnAtomicRmwCmpxchgExpr(Opcode opcode, Index memidx, Address alignment_log2, Address offset) override; Result OnAtomicWaitExpr(Opcode opcode, Index memidx, Address alignment_log2, Address offset) override; Result OnAtomicFenceExpr(uint32_t consistency_model) override; Result OnAtomicNotifyExpr(Opcode opcode, Index memidx, Address alignment_log2, Address offset) override; Result OnBinaryExpr(Opcode opcode) override; Result OnBlockExpr(Type sig_type) override; Result OnBrExpr(Index depth) override; Result OnBrIfExpr(Index depth) override; Result OnBrTableExpr(Index num_targets, Index* target_depths, Index default_target_depth) override; Result OnCallExpr(Index func_index) override; Result OnCallIndirectExpr(Index sig_index, Index table_index) override; Result OnCatchExpr(Index tag_index) override; Result OnCatchAllExpr() override; Result OnDelegateExpr(Index depth) override; Result OnReturnCallExpr(Index func_index) override; Result OnReturnCallIndirectExpr(Index sig_index, Index table_index) override; Result OnCompareExpr(Opcode opcode) override; Result OnConvertExpr(Opcode opcode) override; Result OnDropExpr() override; Result OnElseExpr() override; Result OnEndExpr() override; Result OnF32ConstExpr(uint32_t value_bits) override; Result OnF64ConstExpr(uint64_t value_bits) override; Result OnV128ConstExpr(v128 value_bits) override; Result OnGlobalGetExpr(Index global_index) override; Result OnGlobalSetExpr(Index global_index) override; Result OnI32ConstExpr(uint32_t value) override; Result OnI64ConstExpr(uint64_t value) override; Result OnIfExpr(Type sig_type) override; Result OnLoadExpr(Opcode opcode, Index memidx, Address alignment_log2, Address offset) override; Result OnLocalGetExpr(Index local_index) override; Result OnLocalSetExpr(Index local_index) override; Result OnLocalTeeExpr(Index local_index) override; Result OnLoopExpr(Type sig_type) override; Result OnMemoryCopyExpr(Index destmemidx, Index srcmemidx) override; Result OnDataDropExpr(Index segment_index) override; Result OnMemoryGrowExpr(Index memidx) override; Result OnMemoryFillExpr(Index memidx) override; Result OnMemoryInitExpr(Index segment_index, Index memidx) override; Result OnMemorySizeExpr(Index memidx) override; Result OnRefFuncExpr(Index func_index) override; Result OnRefNullExpr(Type type) override; Result OnRefIsNullExpr() override; Result OnNopExpr() override; Result OnRethrowExpr(Index depth) override; Result OnReturnExpr() override; Result OnSelectExpr(Index result_count, Type* result_types) override; Result OnStoreExpr(Opcode opcode, Index memidx, Address alignment_log2, Address offset) override; Result OnUnaryExpr(Opcode opcode) override; Result OnTableCopyExpr(Index dst_index, Index src_index) override; Result OnTableGetExpr(Index table_index) override; Result OnTableSetExpr(Index table_index) override; Result OnTableGrowExpr(Index table_index) override; Result OnTableSizeExpr(Index table_index) override; Result OnTableFillExpr(Index table_index) override; Result OnElemDropExpr(Index segment_index) override; Result OnTableInitExpr(Index segment_index, Index table_index) override; Result OnTernaryExpr(Opcode opcode) override; Result OnThrowExpr(Index tag_index) override; Result OnTryExpr(Type sig_type) override; Result OnUnreachableExpr() override; Result EndFunctionBody(Index index) override; Result OnSimdLaneOpExpr(Opcode opcode, uint64_t value) override; Result OnSimdLoadLaneExpr(Opcode opcode, Index memidx, Address alignment_log2, Address offset, uint64_t value) override; Result OnSimdStoreLaneExpr(Opcode opcode, Index memidx, Address alignment_log2, Address offset, uint64_t value) override; Result OnSimdShuffleOpExpr(Opcode opcode, v128 value) override; Result OnLoadSplatExpr(Opcode opcode, Index memidx, Address alignment_log2, Address offset) override; Result OnLoadZeroExpr(Opcode opcode, Index memidx, Address alignment_log2, Address offset) override; Result OnElemSegmentCount(Index count) override; Result BeginElemSegment(Index index, Index table_index, uint8_t flags) override; Result BeginElemSegmentInitExpr(Index index) override; Result EndElemSegmentInitExpr(Index index) override; Result OnElemSegmentElemType(Index index, Type elem_type) override; Result OnElemSegmentElemExprCount(Index index, Index count) override; Result BeginElemExpr(Index elem_index, Index expr_index) override; Result EndElemExpr(Index elem_index, Index expr_index) override; Result OnDataCount(Index count) override; Result BeginDataSegmentInitExpr(Index index) override; Result EndDataSegmentInitExpr(Index index) override; Result BeginDataSegment(Index index, Index memory_index, uint8_t flags) override; Result OnDataSegmentData(Index index, const void* data, Address size) override; private: Location GetLocation() const; Label* GetLabel(Index depth); Label* GetNearestTryLabel(Index depth); Label* TopLabel(); void PushLabel(LabelKind label = LabelKind::Block, Istream::Offset offset = Istream::kInvalidOffset, Istream::Offset fixup_offset = Istream::kInvalidOffset, u32 handler_desc_index = kInvalidIndex); void PopLabel(); void PrintError(const char* format, ...); Result GetDropCount(Index keep_count, size_t type_stack_limit, Index* out_drop_count); Result GetBrDropKeepCount(Index depth, Index* out_drop_count, Index* out_keep_count); Result GetReturnDropKeepCount(Index* out_drop_count, Index* out_keep_count); Result GetReturnCallDropKeepCount(const FuncType&, Index keep_extra, Index* out_drop_count, Index* out_keep_count); Result BeginInitExpr(FuncDesc* init_func); Result EndInitExpr(); void EmitBr(Index depth, Index drop_count, Index keep_count, Index catch_drop_count); void FixupTopLabel(); u32 GetFuncOffset(Index func_index); Index TranslateLocalIndex(Index local_index); Index num_func_imports() const; Errors* errors_ = nullptr; ModuleDesc& module_; Istream& istream_; SharedValidator validator_; FuncDesc* func_; std::vector