summaryrefslogtreecommitdiff
path: root/src/interp/binary-reader-interp.cc
diff options
context:
space:
mode:
Diffstat (limited to 'src/interp/binary-reader-interp.cc')
-rw-r--r--src/interp/binary-reader-interp.cc88
1 files changed, 88 insertions, 0 deletions
diff --git a/src/interp/binary-reader-interp.cc b/src/interp/binary-reader-interp.cc
index 1d3daf8e..7e9b8d5e 100644
--- a/src/interp/binary-reader-interp.cc
+++ b/src/interp/binary-reader-interp.cc
@@ -241,7 +241,10 @@ class BinaryReaderInterp : public BinaryReaderNop {
Result OnTableInitExpr(Index segment_index, Index table_index) override;
Result OnTernaryExpr(Opcode opcode) override;
Result OnThrowExpr(Index tag_index) override;
+ Result OnThrowRefExpr() override;
Result OnTryExpr(Type sig_type) override;
+ Result OnTryTableExpr(Type sig_type,
+ const CatchClauseVector& catches) override;
Result OnUnreachableExpr() override;
Result EndFunctionBody(Index index) override;
Result OnSimdLaneOpExpr(Opcode opcode, uint64_t value) override;
@@ -1072,6 +1075,11 @@ Result BinaryReaderInterp::OnEndExpr() {
HandlerDesc& desc = func_->handlers[local_label->handler_desc_index];
desc.try_end_offset = istream_.end();
assert(desc.catches.size() == 0);
+ } else if (label_type == LabelType::TryTable) {
+ // TryTable blocks need a try_end_offset
+ Label* local_label = TopLabel();
+ HandlerDesc& desc = func_->handlers[local_label->handler_desc_index];
+ desc.try_end_offset = istream_.end();
} else if (label_type == LabelType::Catch) {
istream_.EmitCatchDrop(1);
}
@@ -1518,6 +1526,12 @@ Result BinaryReaderInterp::OnThrowExpr(Index tag_index) {
return Result::Ok;
}
+Result BinaryReaderInterp::OnThrowRefExpr() {
+ CHECK_RESULT(validator_.OnThrowRef(GetLocation()));
+ istream_.Emit(Opcode::ThrowRef);
+ return Result::Ok;
+}
+
Result BinaryReaderInterp::OnRethrowExpr(Index depth) {
Index catch_depth;
CHECK_RESULT(validator_.OnRethrow(GetLocation(), Var(depth, GetLocation())));
@@ -1549,6 +1563,80 @@ Result BinaryReaderInterp::OnTryExpr(Type sig_type) {
return Result::Ok;
}
+Result BinaryReaderInterp::OnTryTableExpr(Type sig_type,
+ const CatchClauseVector& catches) {
+ // we can just emit the catch handlers beforehand, so long as we skip over
+ // them when entering the try.
+ CHECK_RESULT(validator_.BeginTryTable(GetLocation(), sig_type));
+
+ u32 exn_stack_height;
+ CHECK_RESULT(
+ validator_.GetCatchCount(label_stack_.size() - 1, &exn_stack_height));
+ // NOTE: *NOT* GetLocalCount. we don't count the parameters, as they're not
+ // part of the frame.
+ u32 value_stack_height = validator_.type_stack_size() + local_count_;
+
+ HandlerDesc desc =
+ HandlerDesc{HandlerKind::Catch, Istream::kInvalidOffset,
+ Istream::kInvalidOffset, {},
+ {Istream::kInvalidOffset}, value_stack_height,
+ exn_stack_height};
+
+ istream_.Emit(Opcode::Br);
+ auto offset = istream_.EmitFixupU32();
+
+ bool has_catch_all = false;
+ for (const auto& raw_catch : catches) {
+ TableCatch catch_;
+ catch_.kind = raw_catch.kind;
+ catch_.tag = Var(raw_catch.tag, GetLocation());
+ catch_.target = Var(raw_catch.depth, GetLocation());
+ CHECK_RESULT(validator_.OnTryTableCatch(GetLocation(), catch_));
+ // stop emitting handlers after catch_all - but we must still validate the
+ // handlers we don't emit
+ if (has_catch_all) {
+ continue;
+ }
+ if (catch_.IsCatchAll()) {
+ has_catch_all = true;
+ desc.catch_all_ref = catch_.IsRef();
+ desc.catch_all_offset = istream_.end();
+ } else {
+ desc.catches.push_back(
+ CatchDesc{raw_catch.tag, istream_.end(), catch_.IsRef()});
+ }
+ // we can't use GetBrDropKeepCount because we're not in a real block.
+ SharedValidator::Label* vlabel;
+ CHECK_RESULT(validator_.GetLabel(raw_catch.depth, &vlabel));
+ // we keep the exception's results.
+ // (this has already been validated, above)
+ Index keep_count = vlabel->br_types().size();
+ // we drop everything between the current block and the br target.
+ // (we have already taken the TryTable block parameters into account, in
+ // BeginTryTable)
+ Index drop_count = validator_.type_stack_size() - vlabel->type_stack_limit;
+ Index catch_drop_count;
+ // we use the regular catch count
+ CHECK_RESULT(validator_.GetCatchCount(raw_catch.depth, &catch_drop_count));
+ // but increment, as we are semantically in a catch
+ catch_drop_count++;
+ EmitBr(raw_catch.depth, drop_count, keep_count, catch_drop_count);
+ }
+
+ CHECK_RESULT(validator_.EndTryTable(GetLocation(), sig_type));
+
+ desc.try_start_offset = istream_.end();
+
+ // as usual, the label is pushed after the catch handlers
+ PushLabel(LabelKind::Try, Istream::kInvalidOffset, Istream::kInvalidOffset,
+ func_->handlers.size());
+ func_->handlers.push_back(std::move(desc));
+
+ istream_.ResolveFixupU32(offset);
+
+ return Result::Ok;
+}
+
Result BinaryReaderInterp::OnCatchExpr(Index tag_index) {
CHECK_RESULT(
validator_.OnCatch(GetLocation(), Var(tag_index, GetLocation()), false));