summaryrefslogtreecommitdiff
path: root/src/binary-reader-ir.cc
diff options
context:
space:
mode:
authorSam Clegg <sbc@chromium.org>2021-12-02 17:30:09 -0800
committerGitHub <noreply@github.com>2021-12-02 17:30:09 -0800
commit0dfa83c1bc645eaa65d8e6ee51f42f8578931ecc (patch)
tree976045d56346480584f4c91eea47e7bd808db2a1 /src/binary-reader-ir.cc
parent952560950f176913c1f54525f250e30d26c2b49a (diff)
downloadwabt-0dfa83c1bc645eaa65d8e6ee51f42f8578931ecc.tar.gz
wabt-0dfa83c1bc645eaa65d8e6ee51f42f8578931ecc.tar.bz2
wabt-0dfa83c1bc645eaa65d8e6ee51f42f8578931ecc.zip
Perform init expression validation outside of the binary reader. NFC (#1770)
Rather than spocial casing them in the reader we now use the same instruction callbacks for instruction that appear in init expressions as instructions that appear in normal functions. The result of this change is the validation of init expressions is pushed further up the stack. For example, objdump will now quite happily dump modules that use arbitrary instructions in thier init expressions even though they are not valid. To me, this makes sense since objdump does not do instruction validation elsewhere. The change is pre-cursor to allowing a wider variety of instruction to be present in init expressions. See https://github.com/WebAssembly/extended-const
Diffstat (limited to 'src/binary-reader-ir.cc')
-rw-r--r--src/binary-reader-ir.cc104
1 files changed, 31 insertions, 73 deletions
diff --git a/src/binary-reader-ir.cc b/src/binary-reader-ir.cc
index 6d3478c6..74d047af 100644
--- a/src/binary-reader-ir.cc
+++ b/src/binary-reader-ir.cc
@@ -259,15 +259,6 @@ class BinaryReaderIR : public BinaryReaderNop {
Result OnTagType(Index index, Index sig_index) override;
Result EndTagSection() override { return Result::Ok; }
- Result OnInitExprF32ConstExpr(Index index, uint32_t value) override;
- Result OnInitExprF64ConstExpr(Index index, uint64_t value) override;
- Result OnInitExprV128ConstExpr(Index index, v128 value) override;
- Result OnInitExprGlobalGetExpr(Index index, Index global_index) override;
- Result OnInitExprI32ConstExpr(Index index, uint32_t value) override;
- Result OnInitExprI64ConstExpr(Index index, uint64_t value) override;
- Result OnInitExprRefNull(Index index, Type type) override;
- Result OnInitExprRefFunc(Index index, Index func_index) override;
-
Result OnDataSymbol(Index index, uint32_t flags, string_view name,
Index segment, uint32_t offset, uint32_t size) override;
Result OnFunctionSymbol(Index index, uint32_t flags, string_view name,
@@ -289,6 +280,7 @@ class BinaryReaderIR : public BinaryReaderNop {
void PushLabel(LabelType label_type,
ExprList* first,
Expr* context = nullptr);
+ Result EndInitExpr();
Result PopLabel();
Result GetLabelAt(LabelNode** label, Index depth);
Result TopLabel(LabelNode** label);
@@ -313,6 +305,7 @@ class BinaryReaderIR : public BinaryReaderNop {
Func* current_func_ = nullptr;
std::vector<LabelNode> label_stack_;
ExprList* current_init_expr_ = nullptr;
+ bool current_init_expr_ended_ = false;
const char* filename_;
};
@@ -375,9 +368,13 @@ Result BinaryReaderIR::TopLabelExpr(LabelNode** label, Expr** expr) {
Result BinaryReaderIR::AppendExpr(std::unique_ptr<Expr> expr) {
expr->loc = GetLocation();
- LabelNode* label;
- CHECK_RESULT(TopLabel(&label));
- label->exprs->push_back(std::move(expr));
+ if (current_init_expr_) {
+ current_init_expr_->push_back(std::move(expr));
+ } else {
+ LabelNode* label;
+ CHECK_RESULT(TopLabel(&label));
+ label->exprs->push_back(std::move(expr));
+ }
return Result::Ok;
}
@@ -611,12 +608,12 @@ Result BinaryReaderIR::BeginGlobalInitExpr(Index index) {
assert(index == module_->globals.size() - 1);
Global* global = module_->globals[index];
current_init_expr_ = &global->init_expr;
+ current_init_expr_ended_ = false;
return Result::Ok;
}
Result BinaryReaderIR::EndGlobalInitExpr(Index index) {
- current_init_expr_ = nullptr;
- return Result::Ok;
+ return EndInitExpr();
}
Result BinaryReaderIR::OnExportCount(Index count) {
@@ -803,6 +800,11 @@ Result BinaryReaderIR::OnElseExpr() {
}
Result BinaryReaderIR::OnEndExpr() {
+ if (current_init_expr_) {
+ assert(!current_init_expr_ended_);
+ current_init_expr_ended_ = true;
+ return Result::Ok;
+ }
if (label_stack_.size() > 1) {
LabelNode* label;
Expr* expr;
@@ -1157,14 +1159,26 @@ Result BinaryReaderIR::BeginElemSegmentInitExpr(Index index) {
assert(index == module_->elem_segments.size() - 1);
ElemSegment* segment = module_->elem_segments[index];
current_init_expr_ = &segment->offset;
+ current_init_expr_ended_ = false;
return Result::Ok;
}
-Result BinaryReaderIR::EndElemSegmentInitExpr(Index index) {
+Result BinaryReaderIR::EndInitExpr() {
+ // This logic is currently duplicated in binary-reader-interp.cpp.
+ // TODO(sbc): Find a way to do this validation in SharedValidator instead.
+ if (!current_init_expr_ended_) {
+ PrintError("expected END opcode after initializer expression");
+ return Result::Error;
+ }
current_init_expr_ = nullptr;
+ current_init_expr_ended_ = false;
return Result::Ok;
}
+Result BinaryReaderIR::EndElemSegmentInitExpr(Index index) {
+ return EndInitExpr();
+}
+
Result BinaryReaderIR::OnElemSegmentElemType(Index index, Type elem_type) {
assert(index == module_->elem_segments.size() - 1);
ElemSegment* segment = module_->elem_segments[index];
@@ -1229,12 +1243,12 @@ Result BinaryReaderIR::BeginDataSegmentInitExpr(Index index) {
assert(index == module_->data_segments.size() - 1);
DataSegment* segment = module_->data_segments[index];
current_init_expr_ = &segment->offset;
+ current_init_expr_ended_ = false;
return Result::Ok;
}
Result BinaryReaderIR::EndDataSegmentInitExpr(Index index) {
- current_init_expr_ = nullptr;
- return Result::Ok;
+ return EndInitExpr();
}
Result BinaryReaderIR::OnDataSegmentData(Index index,
@@ -1416,62 +1430,6 @@ Result BinaryReaderIR::OnLocalNameLocalCount(Index index, Index count) {
return Result::Ok;
}
-Result BinaryReaderIR::OnInitExprF32ConstExpr(Index index, uint32_t value) {
- Location loc = GetLocation();
- current_init_expr_->push_back(
- MakeUnique<ConstExpr>(Const::F32(value, loc), loc));
- return Result::Ok;
-}
-
-Result BinaryReaderIR::OnInitExprF64ConstExpr(Index index, uint64_t value) {
- Location loc = GetLocation();
- current_init_expr_->push_back(
- MakeUnique<ConstExpr>(Const::F64(value, loc), loc));
- return Result::Ok;
-}
-
-Result BinaryReaderIR::OnInitExprV128ConstExpr(Index index, v128 value) {
- Location loc = GetLocation();
- current_init_expr_->push_back(
- MakeUnique<ConstExpr>(Const::V128(value, loc), loc));
- return Result::Ok;
-}
-
-Result BinaryReaderIR::OnInitExprGlobalGetExpr(Index index,
- Index global_index) {
- Location loc = GetLocation();
- current_init_expr_->push_back(
- MakeUnique<GlobalGetExpr>(Var(global_index, loc), loc));
- return Result::Ok;
-}
-
-Result BinaryReaderIR::OnInitExprI32ConstExpr(Index index, uint32_t value) {
- Location loc = GetLocation();
- current_init_expr_->push_back(
- MakeUnique<ConstExpr>(Const::I32(value, loc), loc));
- return Result::Ok;
-}
-
-Result BinaryReaderIR::OnInitExprI64ConstExpr(Index index, uint64_t value) {
- Location loc = GetLocation();
- current_init_expr_->push_back(
- MakeUnique<ConstExpr>(Const::I64(value, loc), loc));
- return Result::Ok;
-}
-
-Result BinaryReaderIR::OnInitExprRefNull(Index index, Type type) {
- Location loc = GetLocation();
- current_init_expr_->push_back(MakeUnique<RefNullExpr>(type, loc));
- return Result::Ok;
-}
-
-Result BinaryReaderIR::OnInitExprRefFunc(Index index, Index func_index) {
- Location loc = GetLocation();
- current_init_expr_->push_back(
- MakeUnique<RefFuncExpr>(Var(func_index, loc), loc));
- return Result::Ok;
-}
-
Result BinaryReaderIR::OnLocalName(Index func_index,
Index local_index,
string_view name) {