diff options
author | Thomas Lively <tlively@google.com> | 2022-11-15 09:38:44 -0800 |
---|---|---|
committer | GitHub <noreply@github.com> | 2022-11-15 09:38:44 -0800 |
commit | 236f12f7df64f10e1238bf2d66b1216f700b15df (patch) | |
tree | 5a380b173d8f186b3763e41345fede9bfa6af1eb /src | |
parent | df41c85856f2ffeea622e879840098f6200aa428 (diff) | |
download | binaryen-236f12f7df64f10e1238bf2d66b1216f700b15df.tar.gz binaryen-236f12f7df64f10e1238bf2d66b1216f700b15df.tar.bz2 binaryen-236f12f7df64f10e1238bf2d66b1216f700b15df.zip |
[Parser] Parse struct allocation and accessor instructions (#5255)
Including support for parsing field indices. Although only numeric field indices
are supported at the moment, set up the code to make it straightforward to
implement type-dependent symbolic field names in the future.
Diffstat (limited to 'src')
-rw-r--r-- | src/wasm/wat-parser.cpp | 135 |
1 files changed, 128 insertions, 7 deletions
diff --git a/src/wasm/wat-parser.cpp b/src/wasm/wat-parser.cpp index fe8b7a2cb..f88bc9872 100644 --- a/src/wasm/wat-parser.cpp +++ b/src/wasm/wat-parser.cpp @@ -659,6 +659,7 @@ struct NullInstrParserCtx { using InstrsT = Ok; using ExprT = Ok; + using FieldIdxT = Ok; using LocalT = Ok; using GlobalT = Ok; using MemoryT = Ok; @@ -671,6 +672,12 @@ struct NullInstrParserCtx { ExprT makeExpr(InstrsT) { return Ok{}; } + template<typename HeapTypeT> FieldIdxT getFieldFromIdx(HeapTypeT, uint32_t) { + return Ok{}; + } + template<typename HeapTypeT> FieldIdxT getFieldFromName(HeapTypeT, Name) { + return Ok{}; + } LocalT getLocalFromIdx(uint32_t) { return Ok{}; } LocalT getLocalFromName(Name) { return Ok{}; } GlobalT getGlobalFromIdx(uint32_t) { return Ok{}; } @@ -727,7 +734,7 @@ struct NullInstrParserCtx { InstrT makeReturn(Index) { return Ok{}; } template<typename HeapTypeT> InstrT makeRefNull(Index, HeapTypeT) { - return {}; + return Ok{}; } InstrT makeRefIs(Index, RefIsOp) { return Ok{}; } @@ -735,6 +742,21 @@ struct NullInstrParserCtx { InstrT makeI31New(Index) { return Ok{}; } InstrT makeI31Get(Index, bool) { return Ok{}; } + + template<typename HeapTypeT> InstrT makeStructNew(Index, HeapTypeT) { + return Ok{}; + } + template<typename HeapTypeT> InstrT makeStructNewDefault(Index, HeapTypeT) { + return Ok{}; + } + template<typename HeapTypeT> + InstrT makeStructGet(Index, HeapTypeT, FieldIdxT, bool) { + return Ok{}; + } + template<typename HeapTypeT> + InstrT makeStructSet(Index, HeapTypeT, FieldIdxT) { + return Ok{}; + } }; // Phase 1: Parse definition spans for top-level module elements and determine @@ -816,7 +838,7 @@ struct ParseDeclsCtx : NullTypeParserCtx, NullInstrParserCtx { std::optional<InstrsT>, Index pos) { if (import && hasNonImport) { - return in.err("import after non-import"); + return in.err(pos, "import after non-import"); } auto f = addFuncDecl(pos, name, import); CHECK_ERR(f); @@ -1090,8 +1112,11 @@ struct ParseModuleTypesCtx : TypeParserCtx<ParseModuleTypesCtx>, TypeUse type, std::optional<LocalsT> locals, std::optional<InstrsT>, - Index) { + Index pos) { auto& f = wasm.functions[index]; + if (!type.type.isSignature()) { + return in.err(pos, "expected signature type"); + } f->type = type.type; for (Index i = 0; i < type.names.size(); ++i) { if (type.names[i].is()) { @@ -1140,6 +1165,7 @@ struct ParseDefsCtx : TypeParserCtx<ParseDefsCtx> { using InstrsT = std::vector<Expression*>; using ExprT = Expression*; + using FieldIdxT = Index; using LocalT = Index; using GlobalT = Name; using MemoryT = Name; @@ -1296,6 +1322,21 @@ struct ParseDefsCtx : TypeParserCtx<ParseDefsCtx> { return types[idx]; } + Result<Index> getFieldFromIdx(HeapType type, uint32_t idx) { + if (!type.isStruct()) { + return in.err("expected struct type"); + } + if (idx >= type.getStruct().fields.size()) { + return in.err("struct index out of bounds"); + } + return idx; + } + + Result<Index> getFieldFromName(HeapType type, Name name) { + // TODO: Field names + return in.err("symbolic field names note yet supported"); + } + Result<Index> getLocalFromIdx(uint32_t idx) { if (!func) { return in.err("cannot access locals outside of a funcion"); @@ -1436,6 +1477,17 @@ struct ParseDefsCtx : TypeParserCtx<ParseDefsCtx> { Memarg getMemarg(uint64_t offset, uint32_t align) { return {offset, align}; } + Result<> validateTypeAnnotation(Index pos, HeapType type, Expression* child) { + if (child->type == Type::unreachable) { + return Ok{}; + } + if (!child->type.isRef() || + !HeapType::isSubType(child->type.getHeapType(), type)) { + return in.err(pos, "invalid reference type on stack"); + } + return Ok{}; + } + Result<Name> getMemory(Index pos, Name* mem) { if (mem) { return *mem; @@ -1796,6 +1848,47 @@ struct ParseDefsCtx : TypeParserCtx<ParseDefsCtx> { CHECK_ERR(val); return push(pos, builder.makeI31Get(*val, signed_)); } + + Result<> makeStructNew(Index pos, HeapType type) { + if (!type.isStruct()) { + return in.err(pos, "expected struct type annotation"); + } + size_t numOps = type.getStruct().fields.size(); + std::vector<Expression*> ops(numOps); + for (size_t i = 0; i < numOps; ++i) { + auto op = pop(pos); + CHECK_ERR(op); + ops[numOps - i - 1] = *op; + } + return push(pos, builder.makeStructNew(type, ops)); + } + + Result<> makeStructNewDefault(Index pos, HeapType type) { + return push(pos, builder.makeStructNew(type, std::array<Expression*, 0>{})); + } + + Result<> makeStructGet(Index pos, HeapType type, Index field, bool signed_) { + assert(type.isStruct()); + const auto& fields = type.getStruct().fields; + assert(fields.size() > field); + auto fieldType = fields[field].type; + auto ref = pop(pos); + CHECK_ERR(ref); + CHECK_ERR(validateTypeAnnotation(pos, type, *ref)); + return push(pos, builder.makeStructGet(field, *ref, fieldType, signed_)); + } + + Result<> makeStructSet(Index pos, HeapType type, Index field) { + assert(type.isStruct()); + const auto& fields = type.getStruct().fields; + assert(fields.size() > field); + auto val = pop(pos); + CHECK_ERR(val); + auto ref = pop(pos); + CHECK_ERR(ref); + CHECK_ERR(validateTypeAnnotation(pos, type, *ref)); + return push(pos, builder.makeStructSet(field, *ref, *val)); + } }; // ================ @@ -1980,6 +2073,8 @@ Result<typename Ctx::InstrT> makeStringSliceIter(Ctx&, Index); // Modules template<typename Ctx> MaybeResult<Index> maybeTypeidx(Ctx& ctx); template<typename Ctx> Result<typename Ctx::HeapTypeT> typeidx(Ctx&); +template<typename Ctx> +Result<typename Ctx::FieldIdxT> fieldidx(Ctx&, typename Ctx::HeapTypeT); template<typename Ctx> MaybeResult<typename Ctx::MemoryT> maybeMemidx(Ctx&); template<typename Ctx> Result<typename Ctx::MemoryT> memidx(Ctx&); template<typename Ctx> Result<typename Ctx::GlobalT> globalidx(Ctx&); @@ -2906,17 +3001,30 @@ Result<typename Ctx::InstrT> makeBrOnStatic(Ctx& ctx, Index pos, BrOnOp op) { template<typename Ctx> Result<typename Ctx::InstrT> makeStructNewStatic(Ctx& ctx, Index pos, bool default_) { - return ctx.in.err("unimplemented instruction"); + auto type = typeidx(ctx); + CHECK_ERR(type); + if (default_) { + return ctx.makeStructNewDefault(pos, *type); + } + return ctx.makeStructNew(pos, *type); } template<typename Ctx> Result<typename Ctx::InstrT> makeStructGet(Ctx& ctx, Index pos, bool signed_) { - return ctx.in.err("unimplemented instruction"); + auto type = typeidx(ctx); + CHECK_ERR(type); + auto field = fieldidx(ctx, *type); + CHECK_ERR(field); + return ctx.makeStructGet(pos, *type, *field, signed_); } template<typename Ctx> Result<typename Ctx::InstrT> makeStructSet(Ctx& ctx, Index pos) { - return ctx.in.err("unimplemented instruction"); + auto type = typeidx(ctx); + CHECK_ERR(type); + auto field = fieldidx(ctx, *type); + CHECK_ERR(field); + return ctx.makeStructSet(pos, *type, *field); } template<typename Ctx> @@ -3058,6 +3166,20 @@ template<typename Ctx> Result<typename Ctx::HeapTypeT> typeidx(Ctx& ctx) { return ctx.in.err("expected type index or identifier"); } +// fieldidx_t ::= x:u32 => x +// | v:id => x (if t.fields[x] = v) +template<typename Ctx> +Result<typename Ctx::FieldIdxT> fieldidx(Ctx& ctx, + typename Ctx::HeapTypeT type) { + if (auto x = ctx.in.takeU32()) { + return ctx.getFieldFromIdx(type, *x); + } + if (auto id = ctx.in.takeID()) { + return ctx.getFieldFromName(type, *id); + } + return ctx.in.err("expected field index or identifier"); +} + // memidx ::= x:u32 => x // | v:id => x (if memories[x] = v) template<typename Ctx> @@ -3327,7 +3449,6 @@ template<typename Ctx> MaybeResult<> func(Ctx& ctx) { return ctx.in.err("expected end of function"); } - // TODO: Use `pos` instead of `in` for error position. CHECK_ERR( ctx.addFunc(name, *exports, import.getPtr(), *type, localVars, insts, pos)); return Ok{}; |