summaryrefslogtreecommitdiff
path: root/src/wasm-interpreter.h
diff options
context:
space:
mode:
Diffstat (limited to 'src/wasm-interpreter.h')
-rw-r--r--src/wasm-interpreter.h47
1 files changed, 36 insertions, 11 deletions
diff --git a/src/wasm-interpreter.h b/src/wasm-interpreter.h
index 571f0d1a5..f37a6edd6 100644
--- a/src/wasm-interpreter.h
+++ b/src/wasm-interpreter.h
@@ -143,13 +143,13 @@ public:
if (!ret.breaking() &&
(curr->type.isConcrete() || ret.value.type.isConcrete())) {
#if 1 // def WASM_INTERPRETER_DEBUG
- if (ret.value.type != curr->type) {
+ if (!Type::isSubType(ret.value.type, curr->type)) {
std::cerr << "expected " << curr->type << ", seeing " << ret.value.type
<< " from\n"
<< curr << '\n';
}
#endif
- assert(ret.value.type == curr->type);
+ assert(Type::isSubType(ret.value.type, curr->type));
}
depth--;
return ret;
@@ -1095,7 +1095,7 @@ public:
return Literal(uint64_t(val));
}
}
- Flow visitAtomicFence(AtomicFence*) {
+ Flow visitAtomicFence(AtomicFence* curr) {
// Wasm currently supports only sequentially consistent atomics, in which
// case atomic_fence can be lowered to nothing.
NOTE_ENTER("AtomicFence");
@@ -1123,6 +1123,26 @@ public:
Flow visitSIMDLoadExtend(SIMDLoad*) { WASM_UNREACHABLE("unimp"); }
Flow visitPush(Push*) { WASM_UNREACHABLE("unimp"); }
Flow visitPop(Pop*) { WASM_UNREACHABLE("unimp"); }
+ Flow visitRefNull(RefNull* curr) {
+ NOTE_ENTER("RefNull");
+ return Literal::makeNullref();
+ }
+ Flow visitRefIsNull(RefIsNull* curr) {
+ NOTE_ENTER("RefIsNull");
+ Flow flow = visit(curr->value);
+ if (flow.breaking()) {
+ return flow;
+ }
+ Literal value = flow.value;
+ NOTE_EVAL1(value);
+ return Literal(value.type == nullref);
+ }
+ Flow visitRefFunc(RefFunc* curr) {
+ NOTE_ENTER("RefFunc");
+ NOTE_NAME(curr->func);
+ return Literal::makeFuncref(curr->func);
+ }
+ // TODO Implement EH instructions
Flow visitTry(Try*) { WASM_UNREACHABLE("unimp"); }
Flow visitThrow(Throw*) { WASM_UNREACHABLE("unimp"); }
Flow visitRethrow(Rethrow*) { WASM_UNREACHABLE("unimp"); }
@@ -1217,8 +1237,10 @@ public:
return Literal(load64u(addr)).castToF64();
case v128:
return Literal(load128(addr).data());
- case anyref: // anyref cannot be loaded from memory
- case exnref: // exnref cannot be loaded from memory
+ case funcref:
+ case anyref:
+ case nullref:
+ case exnref:
case none:
case unreachable:
WASM_UNREACHABLE("unexpected type");
@@ -1272,8 +1294,10 @@ public:
case v128:
store128(addr, value.getv128());
break;
- case anyref: // anyref cannot be stored from memory
- case exnref: // exnref cannot be stored in memory
+ case funcref:
+ case anyref:
+ case nullref:
+ case exnref:
case none:
case unreachable:
WASM_UNREACHABLE("unexpected type");
@@ -1464,7 +1488,7 @@ private:
for (size_t i = 0; i < function->getNumLocals(); i++) {
if (i < arguments.size()) {
assert(i < params.size());
- if (params[i] != arguments[i].type) {
+ if (!Type::isSubType(arguments[i].type, params[i])) {
std::cerr << "Function `" << function->name << "` expects type "
<< params[i] << " for parameter " << i << ", got "
<< arguments[i].type << "." << std::endl;
@@ -1473,7 +1497,7 @@ private:
locals[i] = arguments[i];
} else {
assert(function->isVar(i));
- locals[i].type = function->getLocalType(i);
+ locals[i] = Literal::makeZero(function->getLocalType(i));
}
}
}
@@ -1580,7 +1604,8 @@ private:
}
NOTE_EVAL1(index);
NOTE_EVAL1(flow.value);
- assert(curr->isTee() ? flow.value.type == curr->type : true);
+ assert(curr->isTee() ? Type::isSubType(flow.value.type, curr->type)
+ : true);
scope.locals[index] = flow.value;
return curr->isTee() ? flow : Flow();
}
@@ -2067,7 +2092,7 @@ public:
// cannot still be breaking, it means we missed our stop
assert(!flow.breaking() || flow.breakTo == RETURN_FLOW);
Literal ret = flow.value;
- if (function->sig.results != ret.type) {
+ if (!Type::isSubType(ret.type, function->sig.results)) {
std::cerr << "calling " << function->name << " resulted in " << ret
<< " but the function type is " << function->sig.results
<< '\n';