Commit 9b62b332 authored by Jakob Kummerow's avatar Jakob Kummerow Committed by V8 LUCI CQ

[cleanup][wasm] Simplify DecodeLocals and PushControl

Some follow-up after getting rid of `let`.

Change-Id: I073372f4edd0847c4ffa428595a6f74158c87a98
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3773515Reviewed-by: 's avatarManos Koukoutos <manoskouk@chromium.org>
Auto-Submit: Jakob Kummerow <jkummerow@chromium.org>
Commit-Queue: Jakob Kummerow <jkummerow@chromium.org>
Cr-Commit-Position: refs/heads/main@{#81844}
parent ef593479
...@@ -937,9 +937,8 @@ struct ControlBase : public PcForErrors<validate> { ...@@ -937,9 +937,8 @@ struct ControlBase : public PcForErrors<validate> {
MOVE_ONLY_NO_DEFAULT_CONSTRUCTOR(ControlBase); MOVE_ONLY_NO_DEFAULT_CONSTRUCTOR(ControlBase);
ControlBase(ControlKind kind, uint32_t locals_count, uint32_t stack_depth, ControlBase(ControlKind kind, uint32_t stack_depth, uint32_t init_stack_depth,
uint32_t init_stack_depth, const uint8_t* pc, const uint8_t* pc, Reachability reachability)
Reachability reachability)
: PcForErrors<validate>(pc), : PcForErrors<validate>(pc),
kind(kind), kind(kind),
stack_depth(stack_depth), stack_depth(stack_depth),
...@@ -1259,66 +1258,47 @@ class WasmDecoder : public Decoder { ...@@ -1259,66 +1258,47 @@ class WasmDecoder : public Decoder {
} }
// Decodes local definitions in the current decoder. // Decodes local definitions in the current decoder.
// Returns the number of newly defined locals, or -1 if decoding failed.
// Writes the total length of decoded locals in {total_length}. // Writes the total length of decoded locals in {total_length}.
// If {insert_position} is defined, the decoded locals will be inserted into // The decoded locals will be appended to {this->local_types_}.
// the {this->local_types_}. The decoder's pc is not advanced. // The decoder's pc is not advanced.
int DecodeLocals(const byte* pc, uint32_t* total_length, void DecodeLocals(const byte* pc, uint32_t* total_length) {
const base::Optional<uint32_t> insert_position) {
uint32_t length; uint32_t length;
*total_length = 0; *total_length = 0;
int total_count = 0;
// The 'else' value is useless, we pass it for convenience.
auto insert_iterator = insert_position.has_value()
? local_types_.begin() + insert_position.value()
: local_types_.begin();
// Decode local declarations, if any. // Decode local declarations, if any.
uint32_t entries = read_u32v<validate>(pc, &length, "local decls count"); uint32_t entries = read_u32v<validate>(pc, &length, "local decls count");
if (!VALIDATE(ok())) { if (!VALIDATE(ok())) {
DecodeError(pc + *total_length, "invalid local decls count"); return DecodeError(pc + *total_length, "invalid local decls count");
return -1;
} }
*total_length += length; *total_length += length;
TRACE("local decls count: %u\n", entries); TRACE("local decls count: %u\n", entries);
while (entries-- > 0) { while (entries-- > 0) {
if (!VALIDATE(more())) { if (!VALIDATE(more())) {
DecodeError(end(), return DecodeError(
"expected more local decls but reached end of input"); end(), "expected more local decls but reached end of input");
return -1;
} }
uint32_t count = uint32_t count =
read_u32v<validate>(pc + *total_length, &length, "local count"); read_u32v<validate>(pc + *total_length, &length, "local count");
if (!VALIDATE(ok())) { if (!VALIDATE(ok())) {
DecodeError(pc + *total_length, "invalid local count"); return DecodeError(pc + *total_length, "invalid local count");
return -1;
} }
DCHECK_LE(local_types_.size(), kV8MaxWasmFunctionLocals); DCHECK_LE(local_types_.size(), kV8MaxWasmFunctionLocals);
if (!VALIDATE(count <= kV8MaxWasmFunctionLocals - local_types_.size())) { if (!VALIDATE(count <= kV8MaxWasmFunctionLocals - local_types_.size())) {
DecodeError(pc + *total_length, "local count too large"); return DecodeError(pc + *total_length, "local count too large");
return -1;
} }
*total_length += length; *total_length += length;
ValueType type = value_type_reader::read_value_type<validate>( ValueType type = value_type_reader::read_value_type<validate>(
this, pc + *total_length, &length, this->module_, enabled_); this, pc + *total_length, &length, this->module_, enabled_);
if (!VALIDATE(type != kWasmBottom)) return -1; if (!VALIDATE(type != kWasmBottom)) return;
*total_length += length; *total_length += length;
total_count += count;
if (insert_position.has_value()) { local_types_.insert(local_types_.end(), count, type);
// Move the insertion iterator to the end of the newly inserted locals. num_locals_ += count;
insert_iterator =
local_types_.insert(insert_iterator, count, type) + count;
num_locals_ += count;
}
} }
DCHECK(ok()); DCHECK(ok());
return total_count;
} }
// Shorthand that forwards to the {DecodeError} functions above, passing our // Shorthand that forwards to the {DecodeError} functions above, passing our
...@@ -2578,9 +2558,9 @@ class WasmFullDecoder : public WasmDecoder<validate, decoding_mode> { ...@@ -2578,9 +2558,9 @@ class WasmFullDecoder : public WasmDecoder<validate, decoding_mode> {
locals_offset_ = this->pc_offset(); locals_offset_ = this->pc_offset();
this->InitializeLocalsFromSig(); this->InitializeLocalsFromSig();
uint32_t params_count = static_cast<uint32_t>(this->num_locals()); uint32_t params_count = this->num_locals();
uint32_t locals_length; uint32_t locals_length;
this->DecodeLocals(this->pc(), &locals_length, params_count); this->DecodeLocals(this->pc(), &locals_length);
if (this->failed()) return TraceFailed(); if (this->failed()) return TraceFailed();
this->consume_bytes(locals_length); this->consume_bytes(locals_length);
int non_defaultable = 0; int non_defaultable = 0;
...@@ -2694,11 +2674,10 @@ class WasmFullDecoder : public WasmDecoder<validate, decoding_mode> { ...@@ -2694,11 +2674,10 @@ class WasmFullDecoder : public WasmDecoder<validate, decoding_mode> {
// Set up initial function block. // Set up initial function block.
{ {
DCHECK(control_.empty()); DCHECK(control_.empty());
constexpr uint32_t kLocalsCount = 0;
constexpr uint32_t kStackDepth = 0; constexpr uint32_t kStackDepth = 0;
constexpr uint32_t kInitStackDepth = 0; constexpr uint32_t kInitStackDepth = 0;
control_.emplace_back(kControlBlock, kLocalsCount, kStackDepth, control_.emplace_back(kControlBlock, kStackDepth, kInitStackDepth,
kInitStackDepth, this->pc_, kReachable); this->pc_, kReachable);
Control* c = &control_.back(); Control* c = &control_.back();
if (decoding_mode == kFunctionBody) { if (decoding_mode == kFunctionBody) {
InitMerge(&c->start_merge, 0, [](uint32_t) -> Value { UNREACHABLE(); }); InitMerge(&c->start_merge, 0, [](uint32_t) -> Value { UNREACHABLE(); });
...@@ -2949,7 +2928,7 @@ class WasmFullDecoder : public WasmDecoder<validate, decoding_mode> { ...@@ -2949,7 +2928,7 @@ class WasmFullDecoder : public WasmDecoder<validate, decoding_mode> {
this->module_); this->module_);
if (!this->Validate(this->pc_ + 1, imm)) return 0; if (!this->Validate(this->pc_ + 1, imm)) return 0;
ArgVector args = PeekArgs(imm.sig); ArgVector args = PeekArgs(imm.sig);
Control* block = PushControl(kControlBlock, 0, args.length()); Control* block = PushControl(kControlBlock, args.length());
SetBlockType(block, imm, args.begin()); SetBlockType(block, imm, args.begin());
CALL_INTERFACE_IF_OK_AND_REACHABLE(Block, block); CALL_INTERFACE_IF_OK_AND_REACHABLE(Block, block);
DropArgs(imm.sig); DropArgs(imm.sig);
...@@ -2988,7 +2967,7 @@ class WasmFullDecoder : public WasmDecoder<validate, decoding_mode> { ...@@ -2988,7 +2967,7 @@ class WasmFullDecoder : public WasmDecoder<validate, decoding_mode> {
this->module_); this->module_);
if (!this->Validate(this->pc_ + 1, imm)) return 0; if (!this->Validate(this->pc_ + 1, imm)) return 0;
ArgVector args = PeekArgs(imm.sig); ArgVector args = PeekArgs(imm.sig);
Control* try_block = PushControl(kControlTry, 0, args.length()); Control* try_block = PushControl(kControlTry, args.length());
SetBlockType(try_block, imm, args.begin()); SetBlockType(try_block, imm, args.begin());
try_block->previous_catch = current_catch_; try_block->previous_catch = current_catch_;
current_catch_ = static_cast<int>(control_depth() - 1); current_catch_ = static_cast<int>(control_depth() - 1);
...@@ -3166,7 +3145,7 @@ class WasmFullDecoder : public WasmDecoder<validate, decoding_mode> { ...@@ -3166,7 +3145,7 @@ class WasmFullDecoder : public WasmDecoder<validate, decoding_mode> {
this->module_); this->module_);
if (!this->Validate(this->pc_ + 1, imm)) return 0; if (!this->Validate(this->pc_ + 1, imm)) return 0;
ArgVector args = PeekArgs(imm.sig); ArgVector args = PeekArgs(imm.sig);
Control* block = PushControl(kControlLoop, 0, args.length()); Control* block = PushControl(kControlLoop, args.length());
SetBlockType(&control_.back(), imm, args.begin()); SetBlockType(&control_.back(), imm, args.begin());
CALL_INTERFACE_IF_OK_AND_REACHABLE(Loop, block); CALL_INTERFACE_IF_OK_AND_REACHABLE(Loop, block);
DropArgs(imm.sig); DropArgs(imm.sig);
...@@ -3181,7 +3160,7 @@ class WasmFullDecoder : public WasmDecoder<validate, decoding_mode> { ...@@ -3181,7 +3160,7 @@ class WasmFullDecoder : public WasmDecoder<validate, decoding_mode> {
Value cond = Peek(0, 0, kWasmI32); Value cond = Peek(0, 0, kWasmI32);
ArgVector args = PeekArgs(imm.sig, 1); ArgVector args = PeekArgs(imm.sig, 1);
if (!VALIDATE(this->ok())) return 0; if (!VALIDATE(this->ok())) return 0;
Control* if_block = PushControl(kControlIf, 0, 1 + args.length()); Control* if_block = PushControl(kControlIf, 1 + args.length());
SetBlockType(if_block, imm, args.begin()); SetBlockType(if_block, imm, args.begin());
CALL_INTERFACE_IF_OK_AND_REACHABLE(If, cond, if_block); CALL_INTERFACE_IF_OK_AND_REACHABLE(If, cond, if_block);
Drop(cond); Drop(cond);
...@@ -4023,8 +4002,7 @@ class WasmFullDecoder : public WasmDecoder<validate, decoding_mode> { ...@@ -4023,8 +4002,7 @@ class WasmFullDecoder : public WasmDecoder<validate, decoding_mode> {
// TODO(jkummerow): Consider refactoring control stack management so // TODO(jkummerow): Consider refactoring control stack management so
// that {drop_values} is never needed. That would require decoupling // that {drop_values} is never needed. That would require decoupling
// creation of the Control object from setting of its stack depth. // creation of the Control object from setting of its stack depth.
Control* PushControl(ControlKind kind, uint32_t locals_count = 0, Control* PushControl(ControlKind kind, uint32_t drop_values) {
uint32_t drop_values = 0) {
DCHECK(!control_.empty()); DCHECK(!control_.empty());
Reachability reachability = control_.back().innerReachability(); Reachability reachability = control_.back().innerReachability();
// In unreachable code, we may run out of stack. // In unreachable code, we may run out of stack.
...@@ -4032,8 +4010,8 @@ class WasmFullDecoder : public WasmDecoder<validate, decoding_mode> { ...@@ -4032,8 +4010,8 @@ class WasmFullDecoder : public WasmDecoder<validate, decoding_mode> {
stack_size() >= drop_values ? stack_size() - drop_values : 0; stack_size() >= drop_values ? stack_size() - drop_values : 0;
stack_depth = std::max(stack_depth, control_.back().stack_depth); stack_depth = std::max(stack_depth, control_.back().stack_depth);
uint32_t init_stack_depth = this->locals_initialization_stack_depth(); uint32_t init_stack_depth = this->locals_initialization_stack_depth();
control_.emplace_back(kind, locals_count, stack_depth, init_stack_depth, control_.emplace_back(kind, stack_depth, init_stack_depth, this->pc_,
this->pc_, reachability); reachability);
current_code_reachable_and_ok_ = this->ok() && reachability == kReachable; current_code_reachable_and_ok_ = this->ok() && reachability == kReachable;
return &control_.back(); return &control_.back();
} }
......
...@@ -24,7 +24,8 @@ bool DecodeLocalDecls(const WasmFeatures& enabled, BodyLocalDecls* decls, ...@@ -24,7 +24,8 @@ bool DecodeLocalDecls(const WasmFeatures& enabled, BodyLocalDecls* decls,
WasmDecoder<Decoder::kFullValidation> decoder( WasmDecoder<Decoder::kFullValidation> decoder(
zone, module, enabled, &no_features, nullptr, start, end, 0); zone, module, enabled, &no_features, nullptr, start, end, 0);
uint32_t length; uint32_t length;
if (decoder.DecodeLocals(decoder.pc(), &length, 0) < 0) { decoder.DecodeLocals(decoder.pc(), &length);
if (decoder.failed()) {
decls->encoded_size = 0; decls->encoded_size = 0;
return false; return false;
} }
......
...@@ -167,7 +167,7 @@ void FunctionBodyDisassembler::DecodeAsWat(MultiLineStringBuilder& out, ...@@ -167,7 +167,7 @@ void FunctionBodyDisassembler::DecodeAsWat(MultiLineStringBuilder& out,
// Decode and print locals. // Decode and print locals.
uint32_t locals_length; uint32_t locals_length;
InitializeLocalsFromSig(); InitializeLocalsFromSig();
DecodeLocals(pc_, &locals_length, static_cast<uint32_t>(num_locals())); DecodeLocals(pc_, &locals_length);
if (failed()) { if (failed()) {
// TODO(jkummerow): Improve error handling. // TODO(jkummerow): Improve error handling.
out << "Failed to decode locals\n"; out << "Failed to decode locals\n";
...@@ -190,7 +190,6 @@ void FunctionBodyDisassembler::DecodeAsWat(MultiLineStringBuilder& out, ...@@ -190,7 +190,6 @@ void FunctionBodyDisassembler::DecodeAsWat(MultiLineStringBuilder& out,
WasmOpcode opcode = GetOpcode(); WasmOpcode opcode = GetOpcode();
current_opcode_ = opcode; // Some immediates need to know this. current_opcode_ = opcode; // Some immediates need to know this.
uint32_t length;
// Deal with indentation. // Deal with indentation.
if (opcode == kExprEnd || opcode == kExprElse || opcode == kExprCatch || if (opcode == kExprEnd || opcode == kExprElse || opcode == kExprCatch ||
opcode == kExprCatchAll || opcode == kExprDelegate) { opcode == kExprCatchAll || opcode == kExprDelegate) {
...@@ -224,7 +223,7 @@ void FunctionBodyDisassembler::DecodeAsWat(MultiLineStringBuilder& out, ...@@ -224,7 +223,7 @@ void FunctionBodyDisassembler::DecodeAsWat(MultiLineStringBuilder& out,
label_stack_.emplace_back(out.line_number(), out.length(), label_stack_.emplace_back(out.line_number(), out.length(),
label_occurrence_index_++); label_occurrence_index_++);
} }
length = PrintImmediatesAndGetLength(out); uint32_t length = PrintImmediatesAndGetLength(out);
pc_ += length; pc_ += length;
out.NextLine(pc_offset()); out.NextLine(pc_offset());
......
...@@ -98,7 +98,7 @@ class ExtendedFunctionDis : public FunctionBodyDisassembler { ...@@ -98,7 +98,7 @@ class ExtendedFunctionDis : public FunctionBodyDisassembler {
// Decode and print locals. // Decode and print locals.
uint32_t locals_length; uint32_t locals_length;
InitializeLocalsFromSig(); InitializeLocalsFromSig();
DecodeLocals(pc_, &locals_length, 0); DecodeLocals(pc_, &locals_length);
if (failed()) { if (failed()) {
// TODO(jkummerow): Better error handling. // TODO(jkummerow): Better error handling.
out << "Failed to decode locals"; out << "Failed to decode locals";
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment