/* * 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. */ #ifndef wasm_wasm_builder_h #define wasm_wasm_builder_h #include "ir/manipulation.h" #include "parsing.h" #include "wasm.h" #include namespace wasm { // Useful data structures struct NameType { Name name; Type type; NameType() : name(nullptr), type(Type::none) {} NameType(Name name, Type type) : name(name), type(type) {} }; // General AST node builder class Builder { Module& wasm; public: Builder(Module& wasm) : wasm(wasm) {} // make* functions create an expression instance. static std::unique_ptr makeFunction(Name name, HeapType type, std::vector&& vars, Expression* body = nullptr) { assert(type.isSignature()); auto func = std::make_unique(); func->name = name; func->type = type; func->body = body; func->vars.swap(vars); return func; } static std::unique_ptr makeFunction(Name name, std::vector&& params, HeapType type, std::vector&& vars, Expression* body = nullptr) { assert(type.isSignature()); auto func = std::make_unique(); func->name = name; func->type = type; func->body = body; for (size_t i = 0; i < params.size(); ++i) { NameType& param = params[i]; assert(func->getParams()[i] == param.type); Index index = func->localNames.size(); func->localIndices[param.name] = index; func->localNames[index] = param.name; } for (auto& var : vars) { func->vars.push_back(var.type); Index index = func->localNames.size(); func->localIndices[var.name] = index; func->localNames[index] = var.name; } return func; } static std::unique_ptr makeTable(Name name, Type type = Type(HeapType::func, Nullable), Address initial = 0, Address max = Table::kMaxSize, Type addressType = Type::i32) { auto table = std::make_unique
(); table->name = name; table->type = type; table->addressType = addressType; table->initial = initial; table->max = max; return table; } static std::unique_ptr makeElementSegment(Name name, Name table, Expression* offset = nullptr, Type type = Type(HeapType::func, Nullable)) { auto seg = std::make_unique(); seg->name = name; seg->table = table; seg->offset = offset; seg->type = type; return seg; } static std::unique_ptr makeMemory(Name name, Address initial = 0, Address max = Memory::kMaxSize32, bool shared = false, Type addressType = Type::i32) { auto memory = std::make_unique(); memory->name = name; memory->initial = initial; memory->max = max; memory->shared = shared; memory->addressType = addressType; return memory; } static std::unique_ptr makeDataSegment(Name name = "", Name memory = "", bool isPassive = false, Expression* offset = nullptr, const char* init = "", Address size = 0) { auto seg = std::make_unique(); seg->name = name; seg->memory = memory; seg->isPassive = isPassive; seg->offset = offset; seg->data.resize(size); std::copy_n(init, size, seg->data.begin()); return seg; } static std::unique_ptr makeExport(Name name, Name value, ExternalKind kind) { auto export_ = std::make_unique(); export_->name = name; export_->value = value; export_->kind = kind; return export_; } enum Mutability { Mutable, Immutable }; static std::unique_ptr makeGlobal(Name name, Type type, Expression* init, Mutability mutable_) { auto glob = std::make_unique(); glob->name = name; glob->type = type; glob->init = init; glob->mutable_ = mutable_ == Mutable; return glob; } static std::unique_ptr makeTag(Name name, Signature sig) { auto tag = std::make_unique(); tag->name = name; tag->sig = sig; return tag; } // IR nodes Nop* makeNop() { return wasm.allocator.alloc(); } Block* makeBlock(Expression* first = nullptr) { auto* ret = wasm.allocator.alloc(); if (first) { ret->list.push_back(first); ret->finalize(); } return ret; } Block* makeBlock(Name name, Expression* first = nullptr) { auto* ret = makeBlock(first); ret->name = name; ret->finalize(); return ret; } template using bool_if_not_expr_t = std::enable_if_t>, bool>; template = true> Block* makeBlock(const T& items, std::optional type = std::nullopt) { auto* ret = wasm.allocator.alloc(); ret->list.set(items); ret->finalize(type); return ret; } template = true> Block* makeBlock(Name name, const T& items, std::optional type = std::nullopt) { auto* ret = wasm.allocator.alloc(); ret->name = name; ret->list.set(items); ret->finalize(type); return ret; } Block* makeBlock(std::initializer_list&& items, std::optional type = std::nullopt) { return makeBlock(items, type); } Block* makeBlock(Name name, std::initializer_list&& items, std::optional type = std::nullopt) { return makeBlock(name, items, type); } If* makeIf(Expression* condition, Expression* ifTrue, Expression* ifFalse = nullptr, std::optional type = std::nullopt) { auto* ret = wasm.allocator.alloc(); ret->condition = condition; ret->ifTrue = ifTrue; ret->ifFalse = ifFalse; ret->finalize(type); return ret; } Loop* makeLoop(Name name, Expression* body, std::optional type = std::nullopt) { auto* ret = wasm.allocator.alloc(); ret->name = name; ret->body = body; ret->finalize(type); return ret; } Break* makeBreak(Name name, Expression* value = nullptr, Expression* condition = nullptr) { auto* ret = wasm.allocator.alloc(); ret->name = name; ret->value = value; ret->condition = condition; ret->finalize(); return ret; } template Switch* makeSwitch(T& list, Name default_, Expression* condition, Expression* value = nullptr) { auto* ret = wasm.allocator.alloc(); ret->targets.set(list); ret->default_ = default_; ret->value = value; ret->condition = condition; return ret; } Call* makeCall(Name target, const std::vector& args, Type type, bool isReturn = false) { auto* call = wasm.allocator.alloc(); // not all functions may exist yet, so type must be provided call->type = type; call->target = target; call->operands.set(args); call->isReturn = isReturn; call->finalize(); return call; } template Call* makeCall(Name target, const T& args, Type type, bool isReturn = false) { auto* call = wasm.allocator.alloc(); // not all functions may exist yet, so type must be provided call->type = type; call->target = target; call->operands.set(args); call->isReturn = isReturn; call->finalize(); return call; } template CallIndirect* makeCallIndirect(const Name table, Expression* target, const T& args, HeapType heapType, bool isReturn = false) { assert(heapType.isSignature()); auto* call = wasm.allocator.alloc(); call->table = table; call->heapType = heapType; call->type = heapType.getSignature().results; call->target = target; call->operands.set(args); call->isReturn = isReturn; call->finalize(); return call; } template CallRef* makeCallRef(Expression* target, const T& args, Type type, bool isReturn = false) { auto* call = wasm.allocator.alloc(); call->type = type; call->target = target; call->operands.set(args); call->isReturn = isReturn; call->finalize(); return call; } LocalGet* makeLocalGet(Index index, Type type) { auto* ret = wasm.allocator.alloc(); ret->index = index; ret->type = type; return ret; } LocalSet* makeLocalSet(Index index, Expression* value) { auto* ret = wasm.allocator.alloc(); ret->index = index; ret->value = value; ret->makeSet(); ret->finalize(); return ret; } LocalSet* makeLocalTee(Index index, Expression* value, Type type) { auto* ret = wasm.allocator.alloc(); ret->index = index; ret->value = value; ret->makeTee(type); return ret; } GlobalGet* makeGlobalGet(Name name, Type type) { auto* ret = wasm.allocator.alloc(); ret->name = name; ret->type = type; return ret; } GlobalSet* makeGlobalSet(Name name, Expression* value) { auto* ret = wasm.allocator.alloc(); ret->name = name; ret->value = value; ret->finalize(); return ret; } Load* makeLoad(unsigned bytes, bool signed_, Address offset, unsigned align, Expression* ptr, Type type, Name memory) { auto* ret = wasm.allocator.alloc(); ret->isAtomic = false; ret->bytes = bytes; ret->signed_ = signed_; ret->offset = offset; ret->align = align; ret->ptr = ptr; ret->type = type; ret->memory = memory; ret->finalize(); return ret; } Load* makeAtomicLoad( unsigned bytes, Address offset, Expression* ptr, Type type, Name memory) { Load* load = makeLoad(bytes, false, offset, bytes, ptr, type, memory); load->isAtomic = true; return load; } AtomicWait* makeAtomicWait(Expression* ptr, Expression* expected, Expression* timeout, Type expectedType, Address offset, Name memory) { auto* wait = wasm.allocator.alloc(); wait->offset = offset; wait->ptr = ptr; wait->expected = expected; wait->timeout = timeout; wait->expectedType = expectedType; wait->finalize(); wait->memory = memory; return wait; } AtomicNotify* makeAtomicNotify(Expression* ptr, Expression* notifyCount, Address offset, Name memory) { auto* notify = wasm.allocator.alloc(); notify->offset = offset; notify->ptr = ptr; notify->notifyCount = notifyCount; notify->finalize(); notify->memory = memory; return notify; } AtomicFence* makeAtomicFence() { return wasm.allocator.alloc(); } Store* makeStore(unsigned bytes, Address offset, unsigned align, Expression* ptr, Expression* value, Type type, Name memory) { auto* ret = wasm.allocator.alloc(); ret->isAtomic = false; ret->bytes = bytes; ret->offset = offset; ret->align = align; ret->ptr = ptr; ret->value = value; ret->valueType = type; ret->memory = memory; ret->finalize(); return ret; } Store* makeAtomicStore(unsigned bytes, Address offset, Expression* ptr, Expression* value, Type type, Name memory) { Store* store = makeStore(bytes, offset, bytes, ptr, value, type, memory); store->isAtomic = true; return store; } AtomicRMW* makeAtomicRMW(AtomicRMWOp op, unsigned bytes, Address offset, Expression* ptr, Expression* value, Type type, Name memory) { auto* ret = wasm.allocator.alloc(); ret->op = op; ret->bytes = bytes; ret->offset = offset; ret->ptr = ptr; ret->value = value; ret->type = type; ret->finalize(); ret->memory = memory; return ret; } AtomicCmpxchg* makeAtomicCmpxchg(unsigned bytes, Address offset, Expression* ptr, Expression* expected, Expression* replacement, Type type, Name memory) { auto* ret = wasm.allocator.alloc(); ret->bytes = bytes; ret->offset = offset; ret->ptr = ptr; ret->expected = expected; ret->replacement = replacement; ret->type = type; ret->finalize(); ret->memory = memory; return ret; } SIMDExtract* makeSIMDExtract(SIMDExtractOp op, Expression* vec, uint8_t index) { auto* ret = wasm.allocator.alloc(); ret->op = op; ret->vec = vec; ret->index = index; ret->finalize(); return ret; } SIMDReplace* makeSIMDReplace(SIMDReplaceOp op, Expression* vec, uint8_t index, Expression* value) { auto* ret = wasm.allocator.alloc(); ret->op = op; ret->vec = vec; ret->index = index; ret->value = value; ret->finalize(); return ret; } SIMDShuffle* makeSIMDShuffle(Expression* left, Expression* right, const std::array& mask) { auto* ret = wasm.allocator.alloc(); ret->left = left; ret->right = right; ret->mask = mask; ret->finalize(); return ret; } SIMDTernary* makeSIMDTernary(SIMDTernaryOp op, Expression* a, Expression* b, Expression* c) { auto* ret = wasm.allocator.alloc(); ret->op = op; ret->a = a; ret->b = b; ret->c = c; ret->finalize(); return ret; } SIMDShift* makeSIMDShift(SIMDShiftOp op, Expression* vec, Expression* shift) { auto* ret = wasm.allocator.alloc(); ret->op = op; ret->vec = vec; ret->shift = shift; ret->finalize(); return ret; } SIMDLoad* makeSIMDLoad(SIMDLoadOp op, Address offset, Address align, Expression* ptr, Name memory) { auto* ret = wasm.allocator.alloc(); ret->op = op; ret->offset = offset; ret->align = align; ret->ptr = ptr; ret->memory = memory; ret->finalize(); return ret; } SIMDLoadStoreLane* makeSIMDLoadStoreLane(SIMDLoadStoreLaneOp op, Address offset, Address align, uint8_t index, Expression* ptr, Expression* vec, Name memory) { auto* ret = wasm.allocator.alloc(); ret->op = op; ret->offset = offset; ret->align = align; ret->index = index; ret->ptr = ptr; ret->vec = vec; ret->finalize(); ret->memory = memory; return ret; } MemoryInit* makeMemoryInit(Name segment, Expression* dest, Expression* offset, Expression* size, Name memory) { auto* ret = wasm.allocator.alloc(); ret->segment = segment; ret->dest = dest; ret->offset = offset; ret->size = size; ret->memory = memory; ret->finalize(); return ret; } DataDrop* makeDataDrop(Name segment) { auto* ret = wasm.allocator.alloc(); ret->segment = segment; ret->finalize(); return ret; } MemoryCopy* makeMemoryCopy(Expression* dest, Expression* source, Expression* size, Name destMemory, Name sourceMemory) { auto* ret = wasm.allocator.alloc(); ret->dest = dest; ret->source = source; ret->size = size; ret->destMemory = destMemory; ret->sourceMemory = sourceMemory; ret->finalize(); return ret; } MemoryFill* makeMemoryFill(Expression* dest, Expression* value, Expression* size, Name memory) { auto* ret = wasm.allocator.alloc(); ret->dest = dest; ret->value = value; ret->size = size; ret->memory = memory; ret->finalize(); return ret; } Const* makeConst(Literal value) { assert(value.type.isNumber()); auto* ret = wasm.allocator.alloc(); ret->value = value; ret->type = value.type; return ret; } template Const* makeConst(T x) { return makeConst(Literal(x)); } Unary* makeUnary(UnaryOp op, Expression* value) { auto* ret = wasm.allocator.alloc(); ret->op = op; ret->value = value; ret->finalize(); return ret; } Const* makeConstPtr(uint64_t val, Type addressType) { return makeConst(Literal::makeFromInt64(val, addressType)); } Binary* makeBinary(BinaryOp op, Expression* left, Expression* right) { auto* ret = wasm.allocator.alloc(); ret->op = op; ret->left = left; ret->right = right; ret->finalize(); return ret; } Select* makeSelect(Expression* condition, Expression* ifTrue, Expression* ifFalse) { auto* ret = wasm.allocator.alloc