Commit 8a6718b1 authored by Clemens Hammacher's avatar Clemens Hammacher Committed by Commit Bot

[wasm] [interpreter] Refactor and extend InterpretedFrame

Similar to WasmInterpreter::Thread, we now also use the pimpl idiom for
InterpretedFrame, hiding the implementation completely in the .cc file.
This allows us to store just two things per InterpretedFrameImpl: The
corresponding thread, and the frame index.
The external interface changes to always return a std::unique_ptr,
because the object layout is not known via the public interface, hence
objects cannot be stack allocated. They also cannot be copied or passed
by value.

The frame inspection interface will be tested after another fix in
https://chromium-review.googlesource.com/474749.

R=ahaas@chromium.org
BUG=v8:5822

Change-Id: I7b109da73df745fac97ec72cb0cf4f0ad71e5da9
Reviewed-on: https://chromium-review.googlesource.com/472887Reviewed-by: 's avatarAndreas Haas <ahaas@chromium.org>
Commit-Queue: Clemens Hammacher <clemensh@chromium.org>
Cr-Commit-Position: refs/heads/master@{#44600}
parent 46355724
...@@ -303,10 +303,9 @@ class InterpreterHandle { ...@@ -303,10 +303,9 @@ class InterpreterHandle {
WasmInterpreter::Thread* thread = interpreter()->GetThread(0); WasmInterpreter::Thread* thread = interpreter()->GetThread(0);
DCHECK_LT(0, thread->GetFrameCount()); DCHECK_LT(0, thread->GetFrameCount());
wasm::InterpretedFrame frame = auto frame = thread->GetFrame(thread->GetFrameCount() - 1);
thread->GetFrame(thread->GetFrameCount() - 1); return compiled_module->GetFunctionOffset(frame->function()->func_index) +
return compiled_module->GetFunctionOffset(frame.function()->func_index) + frame->pc();
frame.pc();
} }
std::vector<std::pair<uint32_t, int>> GetInterpretedStack( std::vector<std::pair<uint32_t, int>> GetInterpretedStack(
...@@ -320,8 +319,8 @@ class InterpreterHandle { ...@@ -320,8 +319,8 @@ class InterpreterHandle {
std::vector<std::pair<uint32_t, int>> stack; std::vector<std::pair<uint32_t, int>> stack;
stack.reserve(frame_range.second - frame_range.first); stack.reserve(frame_range.second - frame_range.first);
for (uint32_t fp = frame_range.first; fp < frame_range.second; ++fp) { for (uint32_t fp = frame_range.first; fp < frame_range.second; ++fp) {
wasm::InterpretedFrame frame = thread->GetFrame(fp); auto frame = thread->GetFrame(fp);
stack.emplace_back(frame.function()->func_index, frame.pc()); stack.emplace_back(frame->function()->func_index, frame->pc());
} }
return stack; return stack;
} }
...@@ -336,8 +335,7 @@ class InterpreterHandle { ...@@ -336,8 +335,7 @@ class InterpreterHandle {
DCHECK_LE(0, idx); DCHECK_LE(0, idx);
DCHECK_GT(frame_range.second - frame_range.first, idx); DCHECK_GT(frame_range.second - frame_range.first, idx);
return std::unique_ptr<wasm::InterpretedFrame>(new wasm::InterpretedFrame( return thread->GetFrame(frame_range.first + idx);
thread->GetMutableFrame(frame_range.first + idx)));
} }
void Unwind(Address frame_pointer) { void Unwind(Address frame_pointer) {
......
...@@ -1176,19 +1176,6 @@ class ThreadImpl { ...@@ -1176,19 +1176,6 @@ class ThreadImpl {
return static_cast<int>(frames_.size()); return static_cast<int>(frames_.size());
} }
template <typename FrameCons>
InterpretedFrame GetMutableFrame(int index, FrameCons frame_cons) {
DCHECK_LE(0, index);
DCHECK_GT(frames_.size(), index);
Frame* frame = &frames_[index];
DCHECK_GE(kMaxInt, frame->pc);
DCHECK_GE(kMaxInt, frame->sp);
DCHECK_GE(kMaxInt, frame->llimit());
return frame_cons(frame->code->function, static_cast<int>(frame->pc),
static_cast<int>(frame->sp),
static_cast<int>(frame->llimit()));
}
WasmVal GetReturnValue(uint32_t index) { WasmVal GetReturnValue(uint32_t index) {
if (state_ == WasmInterpreter::TRAPPED) return WasmVal(0xdeadbeef); if (state_ == WasmInterpreter::TRAPPED) return WasmVal(0xdeadbeef);
DCHECK_EQ(WasmInterpreter::FINISHED, state_); DCHECK_EQ(WasmInterpreter::FINISHED, state_);
...@@ -1199,6 +1186,11 @@ class ThreadImpl { ...@@ -1199,6 +1186,11 @@ class ThreadImpl {
return stack_[act.sp + index]; return stack_[act.sp + index];
} }
WasmVal GetStackValue(uint32_t index) {
DCHECK_GT(stack_.size(), index);
return stack_[index];
}
TrapReason GetTrapReason() { return trap_reason_; } TrapReason GetTrapReason() { return trap_reason_; }
pc_t GetBreakpointPc() { return break_pc_; } pc_t GetBreakpointPc() { return break_pc_; }
...@@ -1282,6 +1274,8 @@ class ThreadImpl { ...@@ -1282,6 +1274,8 @@ class ThreadImpl {
unsigned arity; unsigned arity;
}; };
friend class InterpretedFrameImpl;
CodeMap* codemap_; CodeMap* codemap_;
WasmInstance* instance_; WasmInstance* instance_;
ZoneVector<WasmVal> stack_; ZoneVector<WasmVal> stack_;
...@@ -2234,6 +2228,66 @@ class ThreadImpl { ...@@ -2234,6 +2228,66 @@ class ThreadImpl {
} }
}; };
class InterpretedFrameImpl {
public:
InterpretedFrameImpl(ThreadImpl* thread, int index)
: thread_(thread), index_(index) {
DCHECK_LE(0, index);
}
const WasmFunction* function() const { return frame()->code->function; }
int pc() const {
DCHECK_LE(0, frame()->pc);
DCHECK_GE(kMaxInt, frame()->pc);
return static_cast<int>(frame()->pc);
}
int GetParameterCount() const {
DCHECK_GE(kMaxInt, function()->sig->parameter_count());
return static_cast<int>(function()->sig->parameter_count());
}
int GetLocalCount() const {
size_t num_locals = function()->sig->parameter_count() +
frame()->code->locals.type_list.size();
DCHECK_GE(kMaxInt, num_locals);
return static_cast<int>(num_locals);
}
int GetStackHeight() const {
bool is_top_frame =
static_cast<size_t>(index_) + 1 == thread_->frames_.size();
size_t stack_limit =
is_top_frame ? thread_->stack_.size() : thread_->frames_[index_ + 1].sp;
DCHECK_LE(GetLocalCount(), stack_limit);
return static_cast<int>(stack_limit) - GetLocalCount();
}
WasmVal GetLocalValue(int index) const {
DCHECK_LE(0, index);
DCHECK_GT(GetLocalCount(), index);
return thread_->GetStackValue(static_cast<int>(frame()->sp) + index);
}
WasmVal GetStackValue(int index) const {
DCHECK_LE(0, index);
// Index must be within the number of stack values of this frame.
DCHECK_GT(GetStackHeight(), index);
return thread_->GetStackValue(static_cast<int>(frame()->sp) +
GetLocalCount() + index);
}
private:
ThreadImpl* thread_;
int index_;
ThreadImpl::Frame* frame() const {
DCHECK_GT(thread_->frames_.size(), index_);
return &thread_->frames_[index_];
}
};
// Converters between WasmInterpreter::Thread and WasmInterpreter::ThreadImpl. // Converters between WasmInterpreter::Thread and WasmInterpreter::ThreadImpl.
// Thread* is the public interface, without knowledge of the object layout. // Thread* is the public interface, without knowledge of the object layout.
// This cast is potentially risky, but as long as we always cast it back before // This cast is potentially risky, but as long as we always cast it back before
...@@ -2245,6 +2299,14 @@ ThreadImpl* ToImpl(WasmInterpreter::Thread* thread) { ...@@ -2245,6 +2299,14 @@ ThreadImpl* ToImpl(WasmInterpreter::Thread* thread) {
return reinterpret_cast<ThreadImpl*>(thread); return reinterpret_cast<ThreadImpl*>(thread);
} }
// Same conversion for InterpretedFrame and InterpretedFrameImpl.
InterpretedFrame* ToFrame(InterpretedFrameImpl* impl) {
return reinterpret_cast<InterpretedFrame*>(impl);
}
const InterpretedFrameImpl* ToImpl(const InterpretedFrame* frame) {
return reinterpret_cast<const InterpretedFrameImpl*>(frame);
}
} // namespace } // namespace
//============================================================================ //============================================================================
...@@ -2275,16 +2337,11 @@ pc_t WasmInterpreter::Thread::GetBreakpointPc() { ...@@ -2275,16 +2337,11 @@ pc_t WasmInterpreter::Thread::GetBreakpointPc() {
int WasmInterpreter::Thread::GetFrameCount() { int WasmInterpreter::Thread::GetFrameCount() {
return ToImpl(this)->GetFrameCount(); return ToImpl(this)->GetFrameCount();
} }
const InterpretedFrame WasmInterpreter::Thread::GetFrame(int index) { std::unique_ptr<InterpretedFrame> WasmInterpreter::Thread::GetFrame(int index) {
return GetMutableFrame(index); DCHECK_LE(0, index);
} DCHECK_GT(GetFrameCount(), index);
InterpretedFrame WasmInterpreter::Thread::GetMutableFrame(int index) { return std::unique_ptr<InterpretedFrame>(
// We have access to the constructor of InterpretedFrame, but ThreadImpl has ToFrame(new InterpretedFrameImpl(ToImpl(this), index)));
// not. So pass it as a lambda (should all get inlined).
auto frame_cons = [](const WasmFunction* function, int pc, int fp, int sp) {
return InterpretedFrame(function, pc, fp, sp);
};
return ToImpl(this)->GetMutableFrame(index, frame_cons);
} }
WasmVal WasmInterpreter::Thread::GetReturnValue(int index) { WasmVal WasmInterpreter::Thread::GetReturnValue(int index) {
return ToImpl(this)->GetReturnValue(index); return ToImpl(this)->GetReturnValue(index);
...@@ -2435,31 +2492,25 @@ ControlTransferMap WasmInterpreter::ComputeControlTransfersForTesting( ...@@ -2435,31 +2492,25 @@ ControlTransferMap WasmInterpreter::ComputeControlTransfersForTesting(
//============================================================================ //============================================================================
// Implementation of the frame inspection interface. // Implementation of the frame inspection interface.
//============================================================================ //============================================================================
const WasmFunction* InterpretedFrame::function() const {
return ToImpl(this)->function();
}
int InterpretedFrame::pc() const { return ToImpl(this)->pc(); }
int InterpretedFrame::GetParameterCount() const { int InterpretedFrame::GetParameterCount() const {
USE(fp_); return ToImpl(this)->GetParameterCount();
USE(sp_);
// TODO(clemensh): Return the correct number of parameters.
return 0;
} }
int InterpretedFrame::GetLocalCount() const {
WasmVal InterpretedFrame::GetLocalVal(int index) const { return ToImpl(this)->GetLocalCount();
CHECK_GE(index, 0);
UNIMPLEMENTED();
WasmVal none;
none.type = kWasmStmt;
return none;
} }
int InterpretedFrame::GetStackHeight() const {
WasmVal InterpretedFrame::GetExprVal(int pc) const { return ToImpl(this)->GetStackHeight();
UNIMPLEMENTED(); }
WasmVal none; WasmVal InterpretedFrame::GetLocalValue(int index) const {
none.type = kWasmStmt; return ToImpl(this)->GetLocalValue(index);
return none; }
WasmVal InterpretedFrame::GetStackValue(int index) const {
return ToImpl(this)->GetStackValue(index);
} }
void InterpretedFrame::SetLocalVal(int index, WasmVal val) { UNIMPLEMENTED(); }
void InterpretedFrame::SetExprVal(int pc, WasmVal val) { UNIMPLEMENTED(); }
} // namespace wasm } // namespace wasm
} // namespace internal } // namespace internal
......
...@@ -82,30 +82,37 @@ FOREACH_UNION_MEMBER(DECLARE_CAST) ...@@ -82,30 +82,37 @@ FOREACH_UNION_MEMBER(DECLARE_CAST)
#undef DECLARE_CAST #undef DECLARE_CAST
// Representation of frames within the interpreter. // Representation of frames within the interpreter.
//
// Layout of a frame:
// -----------------
// stack slot #N ‾\.
// ... | stack entries: GetStackHeight(); GetStackValue()
// stack slot #0 _/·
// local #L ‾\.
// ... | locals: GetLocalCount(); GetLocalValue()
// local #P+1 |
// param #P | ‾\.
// ... | | parameters: GetParameterCount(); GetLocalValue()
// param #0 _/· _/·
// -----------------
//
class InterpretedFrame { class InterpretedFrame {
public: public:
const WasmFunction* function() const { return function_; } const WasmFunction* function() const;
int pc() const { return pc_; } int pc() const;
//==========================================================================
// Stack frame inspection.
//==========================================================================
int GetParameterCount() const; int GetParameterCount() const;
WasmVal GetLocalVal(int index) const; int GetLocalCount() const;
WasmVal GetExprVal(int pc) const; int GetStackHeight() const;
void SetLocalVal(int index, WasmVal val); WasmVal GetLocalValue(int index) const;
void SetExprVal(int pc, WasmVal val); WasmVal GetStackValue(int index) const;
private: private:
friend class WasmInterpreter; friend class WasmInterpreter;
// Don't instante InterpretedFrames; they will be allocated as
InterpretedFrame(const WasmFunction* function, int pc, int fp, int sp) // InterpretedFrameImpl in the interpreter implementation.
: function_(function), pc_(pc), fp_(fp), sp_(sp) {} InterpretedFrame() = delete;
DISALLOW_COPY_AND_ASSIGN(InterpretedFrame);
const WasmFunction* function_;
int pc_;
int fp_;
int sp_;
}; };
// An interpreter capable of executing WASM. // An interpreter capable of executing WASM.
...@@ -154,8 +161,8 @@ class V8_EXPORT_PRIVATE WasmInterpreter { ...@@ -154,8 +161,8 @@ class V8_EXPORT_PRIVATE WasmInterpreter {
pc_t GetBreakpointPc(); pc_t GetBreakpointPc();
// TODO(clemensh): Make this uint32_t. // TODO(clemensh): Make this uint32_t.
int GetFrameCount(); int GetFrameCount();
const InterpretedFrame GetFrame(int index); // The InterpretedFrame is only valid as long as the Thread is paused.
InterpretedFrame GetMutableFrame(int index); std::unique_ptr<InterpretedFrame> GetFrame(int index);
WasmVal GetReturnValue(int index = 0); WasmVal GetReturnValue(int index = 0);
TrapReason GetTrapReason(); TrapReason GetTrapReason();
......
...@@ -579,7 +579,7 @@ class WasmDebugInfo : public FixedArray { ...@@ -579,7 +579,7 @@ class WasmDebugInfo : public FixedArray {
Address frame_pointer); Address frame_pointer);
std::unique_ptr<wasm::InterpretedFrame> GetInterpretedFrame( std::unique_ptr<wasm::InterpretedFrame> GetInterpretedFrame(
Address frame_pointer, int idx); Address frame_pointer, int frame_index);
// Unwind the interpreted stack belonging to the passed interpreter entry // Unwind the interpreted stack belonging to the passed interpreter entry
// frame. // frame.
......
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