2012-02-09 09:43:37 +00:00
|
|
|
// Copyright 2012 the V8 project authors. All rights reserved.
|
2014-04-29 06:42:26 +00:00
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
|
// found in the LICENSE file.
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
#ifndef V8_FRAMES_H_
|
|
|
|
#define V8_FRAMES_H_
|
|
|
|
|
2014-06-03 08:12:43 +00:00
|
|
|
#include "src/allocation.h"
|
|
|
|
#include "src/handles.h"
|
|
|
|
#include "src/safepoint-table.h"
|
2011-01-12 14:14:14 +00:00
|
|
|
|
2009-05-25 10:05:56 +00:00
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2014-03-21 09:28:26 +00:00
|
|
|
#if V8_TARGET_ARCH_ARM64
|
2014-02-12 09:19:30 +00:00
|
|
|
typedef uint64_t RegList;
|
|
|
|
#else
|
2008-07-03 15:10:15 +00:00
|
|
|
typedef uint32_t RegList;
|
2014-02-12 09:19:30 +00:00
|
|
|
#endif
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// Get the number of registers in a given register list.
|
|
|
|
int NumRegs(RegList list);
|
|
|
|
|
2012-04-05 14:10:39 +00:00
|
|
|
void SetUpJSCallerSavedCodeData();
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Return the code of the n-th saved register available to JavaScript.
|
|
|
|
int JSCallerSavedCode(int n);
|
|
|
|
|
|
|
|
|
|
|
|
// Forward declarations.
|
2013-07-23 15:01:38 +00:00
|
|
|
class ExternalCallbackScope;
|
2013-06-27 09:34:31 +00:00
|
|
|
class StackFrameIteratorBase;
|
2008-07-03 15:10:15 +00:00
|
|
|
class ThreadLocalTop;
|
2011-03-18 20:35:07 +00:00
|
|
|
class Isolate;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2011-09-20 10:08:39 +00:00
|
|
|
class InnerPointerToCodeCache {
|
2010-08-30 08:54:43 +00:00
|
|
|
public:
|
2011-09-20 10:08:39 +00:00
|
|
|
struct InnerPointerToCodeCacheEntry {
|
|
|
|
Address inner_pointer;
|
2010-08-30 08:54:43 +00:00
|
|
|
Code* code;
|
2011-01-12 14:14:14 +00:00
|
|
|
SafepointEntry safepoint_entry;
|
2010-08-30 08:54:43 +00:00
|
|
|
};
|
|
|
|
|
2011-09-20 10:08:39 +00:00
|
|
|
explicit InnerPointerToCodeCache(Isolate* isolate) : isolate_(isolate) {
|
2011-03-18 20:35:07 +00:00
|
|
|
Flush();
|
2010-08-30 08:54:43 +00:00
|
|
|
}
|
|
|
|
|
2011-09-20 10:08:39 +00:00
|
|
|
Code* GcSafeFindCodeForInnerPointer(Address inner_pointer);
|
|
|
|
Code* GcSafeCastToCode(HeapObject* object, Address inner_pointer);
|
2010-08-30 08:54:43 +00:00
|
|
|
|
2011-03-18 20:35:07 +00:00
|
|
|
void Flush() {
|
2010-08-30 08:54:43 +00:00
|
|
|
memset(&cache_[0], 0, sizeof(cache_));
|
|
|
|
}
|
|
|
|
|
2011-09-20 10:08:39 +00:00
|
|
|
InnerPointerToCodeCacheEntry* GetCacheEntry(Address inner_pointer);
|
2010-08-30 08:54:43 +00:00
|
|
|
|
|
|
|
private:
|
2011-09-20 10:08:39 +00:00
|
|
|
InnerPointerToCodeCacheEntry* cache(int index) { return &cache_[index]; }
|
2011-03-18 20:35:07 +00:00
|
|
|
|
|
|
|
Isolate* isolate_;
|
|
|
|
|
2011-09-20 10:08:39 +00:00
|
|
|
static const int kInnerPointerToCodeCacheSize = 1024;
|
|
|
|
InnerPointerToCodeCacheEntry cache_[kInnerPointerToCodeCacheSize];
|
2011-03-18 20:35:07 +00:00
|
|
|
|
2011-09-20 10:08:39 +00:00
|
|
|
DISALLOW_COPY_AND_ASSIGN(InnerPointerToCodeCache);
|
2010-08-30 08:54:43 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2015-03-25 13:13:51 +00:00
|
|
|
// Every try-block pushes the context register.
|
|
|
|
class TryBlockConstant : public AllStatic {
|
|
|
|
public:
|
|
|
|
static const int kElementCount = 1;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2013-04-08 11:17:32 +00:00
|
|
|
class StackHandlerConstants : public AllStatic {
|
|
|
|
public:
|
2015-03-25 13:13:51 +00:00
|
|
|
static const int kNextOffset = 0 * kPointerSize;
|
2013-04-08 11:17:32 +00:00
|
|
|
|
2015-03-25 13:13:51 +00:00
|
|
|
static const int kSize = kNextOffset + kPointerSize;
|
2013-05-08 08:08:23 +00:00
|
|
|
static const int kSlotCount = kSize >> kPointerSizeLog2;
|
2013-04-08 11:17:32 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
class StackHandler BASE_EMBEDDED {
|
|
|
|
public:
|
|
|
|
// Get the address of this stack handler.
|
|
|
|
inline Address address() const;
|
|
|
|
|
|
|
|
// Get the next stack handler in the chain.
|
|
|
|
inline StackHandler* next() const;
|
|
|
|
|
|
|
|
// Conversion support.
|
|
|
|
static inline StackHandler* FromAddress(Address address);
|
|
|
|
|
|
|
|
private:
|
2008-08-13 09:32:07 +00:00
|
|
|
DISALLOW_IMPLICIT_CONSTRUCTORS(StackHandler);
|
2008-07-03 15:10:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2013-01-29 09:12:20 +00:00
|
|
|
#define STACK_FRAME_TYPE_LIST(V) \
|
|
|
|
V(ENTRY, EntryFrame) \
|
|
|
|
V(ENTRY_CONSTRUCT, EntryConstructFrame) \
|
|
|
|
V(EXIT, ExitFrame) \
|
|
|
|
V(JAVA_SCRIPT, JavaScriptFrame) \
|
|
|
|
V(OPTIMIZED, OptimizedFrame) \
|
|
|
|
V(STUB, StubFrame) \
|
|
|
|
V(STUB_FAILURE_TRAMPOLINE, StubFailureTrampolineFrame) \
|
|
|
|
V(INTERNAL, InternalFrame) \
|
|
|
|
V(CONSTRUCT, ConstructFrame) \
|
|
|
|
V(ARGUMENTS_ADAPTOR, ArgumentsAdaptorFrame)
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
|
2013-02-05 08:09:32 +00:00
|
|
|
class StandardFrameConstants : public AllStatic {
|
|
|
|
public:
|
|
|
|
// Fixed part of the frame consists of return address, caller fp,
|
2015-06-04 14:44:00 +00:00
|
|
|
// constant pool (if FLAG_enable_embedded_constant_pool), context, and
|
|
|
|
// function. StandardFrame::IterateExpressions assumes that kLastObjectOffset
|
|
|
|
// is the last object pointer.
|
2013-12-30 11:23:59 +00:00
|
|
|
static const int kCPSlotSize =
|
2015-06-04 14:44:00 +00:00
|
|
|
FLAG_enable_embedded_constant_pool ? kPointerSize : 0;
|
2013-12-30 11:23:59 +00:00
|
|
|
static const int kFixedFrameSizeFromFp = 2 * kPointerSize + kCPSlotSize;
|
[turbofan] Unify referencing of stack slots
Previously, it was not possible to specify StackSlotOperands for all
slots in both the caller and callee stacks. Specifically, the region
of the callee's stack including the saved return address, frame
pointer, function pointer and context pointer could not be addressed
by the register allocator/gap resolver.
In preparation for better tail call support, which will use the gap
resolver to reconcile outgoing parameters, this change makes it
possible to address all slots on the stack, because slots in the
previously inaccessible dead zone may become parameter slots for
outgoing tail calls. All caller stack slots are accessible as they
were before, with slot -1 corresponding to the last stack
parameter. Stack slot indices >= 0 access the callee stack, with slot
0 corresponding to the callee's saved return address, 1 corresponding
to the saved frame pointer, 2 corresponding to the current function
context, 3 corresponding to the frame marker/JSFunction, and slots 4
and above corresponding to spill slots.
The following changes were specifically needed:
* Frame has been changed to explicitly manage three areas of the
callee frame, the fixed header, the spill slot area, and the
callee-saved register area.
* Conversions from stack slot indices to fp offsets all now go through
a common bottleneck: OptimizedFrame::StackSlotOffsetRelativeToFp
* The generation of deoptimization translation tables has been changed
to support the new stack slot indexing scheme. Crankshaft, which
doesn't support the new slot numbering in its register allocator,
must adapt the indexes when creating translation tables.
* Callee-saved parameters are now kept below spill slots, not above,
to support saving only the optimal set of used registers, which is
only known after register allocation is finished and spill slots
have been allocated.
Review URL: https://codereview.chromium.org/1261923007
Cr-Commit-Position: refs/heads/master@{#30224}
2015-08-18 14:47:56 +00:00
|
|
|
static const int kFixedFrameSizeAboveFp = kPCOnStackSize + kFPOnStackSize;
|
2015-06-04 14:44:00 +00:00
|
|
|
static const int kFixedFrameSize =
|
[turbofan] Unify referencing of stack slots
Previously, it was not possible to specify StackSlotOperands for all
slots in both the caller and callee stacks. Specifically, the region
of the callee's stack including the saved return address, frame
pointer, function pointer and context pointer could not be addressed
by the register allocator/gap resolver.
In preparation for better tail call support, which will use the gap
resolver to reconcile outgoing parameters, this change makes it
possible to address all slots on the stack, because slots in the
previously inaccessible dead zone may become parameter slots for
outgoing tail calls. All caller stack slots are accessible as they
were before, with slot -1 corresponding to the last stack
parameter. Stack slot indices >= 0 access the callee stack, with slot
0 corresponding to the callee's saved return address, 1 corresponding
to the saved frame pointer, 2 corresponding to the current function
context, 3 corresponding to the frame marker/JSFunction, and slots 4
and above corresponding to spill slots.
The following changes were specifically needed:
* Frame has been changed to explicitly manage three areas of the
callee frame, the fixed header, the spill slot area, and the
callee-saved register area.
* Conversions from stack slot indices to fp offsets all now go through
a common bottleneck: OptimizedFrame::StackSlotOffsetRelativeToFp
* The generation of deoptimization translation tables has been changed
to support the new stack slot indexing scheme. Crankshaft, which
doesn't support the new slot numbering in its register allocator,
must adapt the indexes when creating translation tables.
* Callee-saved parameters are now kept below spill slots, not above,
to support saving only the optimal set of used registers, which is
only known after register allocation is finished and spill slots
have been allocated.
Review URL: https://codereview.chromium.org/1261923007
Cr-Commit-Position: refs/heads/master@{#30224}
2015-08-18 14:47:56 +00:00
|
|
|
kFixedFrameSizeAboveFp + kFixedFrameSizeFromFp;
|
|
|
|
static const int kFixedSlotCountAboveFp =
|
|
|
|
kFixedFrameSizeAboveFp / kPointerSize;
|
|
|
|
static const int kFixedSlotCount = kFixedFrameSize / kPointerSize;
|
2015-08-28 06:34:55 +00:00
|
|
|
static const int kCPSlotCount = kCPSlotSize / kPointerSize;
|
2015-06-04 14:44:00 +00:00
|
|
|
static const int kExpressionsOffset = -3 * kPointerSize - kCPSlotSize;
|
|
|
|
static const int kMarkerOffset = -2 * kPointerSize - kCPSlotSize;
|
|
|
|
static const int kContextOffset = -1 * kPointerSize - kCPSlotSize;
|
|
|
|
static const int kConstantPoolOffset = kCPSlotSize ? -1 * kPointerSize : 0;
|
|
|
|
static const int kCallerFPOffset = 0 * kPointerSize;
|
|
|
|
static const int kCallerPCOffset = +1 * kFPOnStackSize;
|
|
|
|
static const int kCallerSPOffset = kCallerPCOffset + 1 * kPCOnStackSize;
|
|
|
|
|
|
|
|
static const int kLastObjectOffset = kContextOffset;
|
2013-02-05 08:09:32 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2015-07-07 08:14:20 +00:00
|
|
|
class ArgumentsAdaptorFrameConstants : public AllStatic {
|
|
|
|
public:
|
|
|
|
// FP-relative.
|
|
|
|
static const int kLengthOffset = StandardFrameConstants::kExpressionsOffset;
|
|
|
|
|
|
|
|
static const int kFrameSize =
|
|
|
|
StandardFrameConstants::kFixedFrameSize + kPointerSize;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class InternalFrameConstants : public AllStatic {
|
|
|
|
public:
|
|
|
|
// FP-relative.
|
|
|
|
static const int kCodeOffset = StandardFrameConstants::kExpressionsOffset;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class ConstructFrameConstants : public AllStatic {
|
|
|
|
public:
|
|
|
|
// FP-relative.
|
|
|
|
static const int kImplicitReceiverOffset =
|
2015-07-17 08:51:24 +00:00
|
|
|
StandardFrameConstants::kExpressionsOffset - 4 * kPointerSize;
|
2015-07-07 08:14:20 +00:00
|
|
|
static const int kOriginalConstructorOffset =
|
2015-07-17 08:51:24 +00:00
|
|
|
StandardFrameConstants::kExpressionsOffset - 3 * kPointerSize;
|
2015-07-07 08:14:20 +00:00
|
|
|
static const int kLengthOffset =
|
2015-07-17 08:51:24 +00:00
|
|
|
StandardFrameConstants::kExpressionsOffset - 2 * kPointerSize;
|
|
|
|
static const int kAllocationSiteOffset =
|
2015-07-07 08:14:20 +00:00
|
|
|
StandardFrameConstants::kExpressionsOffset - 1 * kPointerSize;
|
|
|
|
static const int kCodeOffset =
|
|
|
|
StandardFrameConstants::kExpressionsOffset - 0 * kPointerSize;
|
|
|
|
|
|
|
|
static const int kFrameSize =
|
2015-07-17 08:51:24 +00:00
|
|
|
StandardFrameConstants::kFixedFrameSize + 5 * kPointerSize;
|
2015-07-07 08:14:20 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2015-08-27 10:32:26 +00:00
|
|
|
class InterpreterFrameConstants : public AllStatic {
|
|
|
|
public:
|
|
|
|
// Register file pointer relative.
|
|
|
|
static const int kLastParamFromRegisterPointer =
|
|
|
|
StandardFrameConstants::kFixedFrameSize + kPointerSize;
|
2015-09-02 13:03:06 +00:00
|
|
|
static const int kFunctionFromRegisterPointer = kPointerSize;
|
2015-08-27 10:32:26 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Abstract base class for all stack frames.
|
|
|
|
class StackFrame BASE_EMBEDDED {
|
|
|
|
public:
|
|
|
|
#define DECLARE_TYPE(type, ignore) type,
|
|
|
|
enum Type {
|
|
|
|
NONE = 0,
|
|
|
|
STACK_FRAME_TYPE_LIST(DECLARE_TYPE)
|
2011-09-15 11:30:45 +00:00
|
|
|
NUMBER_OF_TYPES,
|
|
|
|
// Used by FrameScope to indicate that the stack frame is constructed
|
|
|
|
// manually and the FrameScope does not need to emit code.
|
|
|
|
MANUAL
|
2008-07-03 15:10:15 +00:00
|
|
|
};
|
|
|
|
#undef DECLARE_TYPE
|
|
|
|
|
|
|
|
// Opaque data type for identifying stack frames. Used extensively
|
|
|
|
// by the debugger.
|
2010-08-16 17:14:34 +00:00
|
|
|
// ID_MIN_VALUE and ID_MAX_VALUE are specified to ensure that enumeration type
|
|
|
|
// has correct value range (see Issue 830 for more details).
|
|
|
|
enum Id {
|
|
|
|
ID_MIN_VALUE = kMinInt,
|
|
|
|
ID_MAX_VALUE = kMaxInt,
|
|
|
|
NO_ID = 0
|
|
|
|
};
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2011-05-17 11:11:12 +00:00
|
|
|
// Used to mark the outermost JS entry frame.
|
|
|
|
enum JsFrameMarker {
|
|
|
|
INNER_JSENTRY_FRAME = 0,
|
|
|
|
OUTERMOST_JSENTRY_FRAME = 1
|
|
|
|
};
|
|
|
|
|
2010-09-16 08:51:13 +00:00
|
|
|
struct State {
|
2014-03-14 15:11:58 +00:00
|
|
|
State() : sp(NULL), fp(NULL), pc_address(NULL),
|
|
|
|
constant_pool_address(NULL) { }
|
2010-09-16 08:51:13 +00:00
|
|
|
Address sp;
|
|
|
|
Address fp;
|
|
|
|
Address* pc_address;
|
2014-03-14 15:11:58 +00:00
|
|
|
Address* constant_pool_address;
|
2010-09-16 08:51:13 +00:00
|
|
|
};
|
|
|
|
|
Simplify isolates access during stack iteration (WAS: Move SafeStackFrameIterator::active_count_...)
While trying to fix Mac and Windows versions for this change:
http://codereview.chromium.org/6771047/, I figured out, that we
already store an isolate in StackFrameIterator, so we can use it in
frame objects, instead of requiring it from caller.
I've changed iterators usage to the following scheme: whenever a
caller maintains an isolate pointer, it just passes it to stack
iterator, and no more worries about passing it to frame content
accessors. If a caller uses current isolate, it can omit passing it
to iterator, in this case, an iterator will use the current isolate,
too.
There was a special case with LiveEdit, which creates
detached copies of frame objects.
R=vitalyr@chromium.org
BUG=none
TEST=none
Review URL: http://codereview.chromium.org/6794019
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@7499 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2011-04-05 09:01:47 +00:00
|
|
|
// Copy constructor; it breaks the connection to host iterator
|
|
|
|
// (as an iterator usually lives on stack).
|
2010-04-06 17:58:28 +00:00
|
|
|
StackFrame(const StackFrame& original) {
|
|
|
|
this->state_ = original.state_;
|
|
|
|
this->iterator_ = NULL;
|
Simplify isolates access during stack iteration (WAS: Move SafeStackFrameIterator::active_count_...)
While trying to fix Mac and Windows versions for this change:
http://codereview.chromium.org/6771047/, I figured out, that we
already store an isolate in StackFrameIterator, so we can use it in
frame objects, instead of requiring it from caller.
I've changed iterators usage to the following scheme: whenever a
caller maintains an isolate pointer, it just passes it to stack
iterator, and no more worries about passing it to frame content
accessors. If a caller uses current isolate, it can omit passing it
to iterator, in this case, an iterator will use the current isolate,
too.
There was a special case with LiveEdit, which creates
detached copies of frame objects.
R=vitalyr@chromium.org
BUG=none
TEST=none
Review URL: http://codereview.chromium.org/6794019
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@7499 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2011-04-05 09:01:47 +00:00
|
|
|
this->isolate_ = original.isolate_;
|
2010-04-06 17:58:28 +00:00
|
|
|
}
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Type testers.
|
|
|
|
bool is_entry() const { return type() == ENTRY; }
|
|
|
|
bool is_entry_construct() const { return type() == ENTRY_CONSTRUCT; }
|
|
|
|
bool is_exit() const { return type() == EXIT; }
|
2010-12-07 11:31:57 +00:00
|
|
|
bool is_optimized() const { return type() == OPTIMIZED; }
|
2008-07-03 15:10:15 +00:00
|
|
|
bool is_arguments_adaptor() const { return type() == ARGUMENTS_ADAPTOR; }
|
|
|
|
bool is_internal() const { return type() == INTERNAL; }
|
2013-01-29 09:12:20 +00:00
|
|
|
bool is_stub_failure_trampoline() const {
|
|
|
|
return type() == STUB_FAILURE_TRAMPOLINE;
|
|
|
|
}
|
2008-10-10 09:09:38 +00:00
|
|
|
bool is_construct() const { return type() == CONSTRUCT; }
|
2008-07-03 15:10:15 +00:00
|
|
|
virtual bool is_standard() const { return false; }
|
|
|
|
|
2010-12-07 11:31:57 +00:00
|
|
|
bool is_java_script() const {
|
|
|
|
Type type = this->type();
|
|
|
|
return (type == JAVA_SCRIPT) || (type == OPTIMIZED);
|
|
|
|
}
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Accessors.
|
|
|
|
Address sp() const { return state_.sp; }
|
|
|
|
Address fp() const { return state_.fp; }
|
2009-06-10 15:08:25 +00:00
|
|
|
Address caller_sp() const { return GetCallerStackPointer(); }
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2012-06-12 10:22:33 +00:00
|
|
|
// If this frame is optimized and was dynamically aligned return its old
|
|
|
|
// unaligned frame pointer. When the frame is deoptimized its FP will shift
|
|
|
|
// up one word and become unaligned.
|
|
|
|
Address UnpaddedFP() const;
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Address pc() const { return *pc_address(); }
|
|
|
|
void set_pc(Address pc) { *pc_address() = pc; }
|
|
|
|
|
2014-03-14 15:11:58 +00:00
|
|
|
Address constant_pool() const { return *constant_pool_address(); }
|
2015-06-04 14:44:00 +00:00
|
|
|
void set_constant_pool(Address constant_pool) {
|
|
|
|
*constant_pool_address() = constant_pool;
|
2014-03-14 15:11:58 +00:00
|
|
|
}
|
|
|
|
|
2010-04-06 17:58:28 +00:00
|
|
|
virtual void SetCallerFp(Address caller_fp) = 0;
|
|
|
|
|
2012-05-03 17:31:34 +00:00
|
|
|
// Manually changes value of fp in this object.
|
|
|
|
void UpdateFp(Address fp) { state_.fp = fp; }
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
Address* pc_address() const { return state_.pc_address; }
|
|
|
|
|
2014-03-14 15:11:58 +00:00
|
|
|
Address* constant_pool_address() const {
|
|
|
|
return state_.constant_pool_address;
|
|
|
|
}
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Get the id of this stack frame.
|
2009-06-10 15:08:25 +00:00
|
|
|
Id id() const { return static_cast<Id>(OffsetFrom(caller_sp())); }
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2015-03-05 13:02:34 +00:00
|
|
|
// Get the top handler from the current stack iterator.
|
|
|
|
inline StackHandler* top_handler() const;
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Get the type of this frame.
|
|
|
|
virtual Type type() const = 0;
|
|
|
|
|
|
|
|
// Get the code associated with this frame.
|
2010-08-17 11:44:01 +00:00
|
|
|
// This method could be called during marking phase of GC.
|
|
|
|
virtual Code* unchecked_code() const = 0;
|
|
|
|
|
|
|
|
// Get the code associated with this frame.
|
2011-10-03 11:13:20 +00:00
|
|
|
inline Code* LookupCode() const;
|
2010-08-30 08:54:43 +00:00
|
|
|
|
|
|
|
// Get the code object that contains the given pc.
|
2011-03-18 20:35:07 +00:00
|
|
|
static inline Code* GetContainingCode(Isolate* isolate, Address pc);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2010-12-07 11:31:57 +00:00
|
|
|
// Get the code object containing the given pc and fill in the
|
|
|
|
// safepoint entry and the number of stack slots. The pc must be at
|
|
|
|
// a safepoint.
|
Simplify isolates access during stack iteration (WAS: Move SafeStackFrameIterator::active_count_...)
While trying to fix Mac and Windows versions for this change:
http://codereview.chromium.org/6771047/, I figured out, that we
already store an isolate in StackFrameIterator, so we can use it in
frame objects, instead of requiring it from caller.
I've changed iterators usage to the following scheme: whenever a
caller maintains an isolate pointer, it just passes it to stack
iterator, and no more worries about passing it to frame content
accessors. If a caller uses current isolate, it can omit passing it
to iterator, in this case, an iterator will use the current isolate,
too.
There was a special case with LiveEdit, which creates
detached copies of frame objects.
R=vitalyr@chromium.org
BUG=none
TEST=none
Review URL: http://codereview.chromium.org/6794019
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@7499 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2011-04-05 09:01:47 +00:00
|
|
|
static Code* GetSafepointData(Isolate* isolate,
|
|
|
|
Address pc,
|
2011-01-12 14:14:14 +00:00
|
|
|
SafepointEntry* safepoint_entry,
|
2010-12-07 11:31:57 +00:00
|
|
|
unsigned* stack_slots);
|
|
|
|
|
2010-08-30 08:54:43 +00:00
|
|
|
virtual void Iterate(ObjectVisitor* v) const = 0;
|
2015-06-04 14:44:00 +00:00
|
|
|
static void IteratePc(ObjectVisitor* v, Address* pc_address,
|
|
|
|
Address* constant_pool_address, Code* holder);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2012-02-27 15:15:53 +00:00
|
|
|
// Sets a callback function for return-address rewriting profilers
|
|
|
|
// to resolve the location of a return address to the location of the
|
|
|
|
// profiler's stashed return address.
|
|
|
|
static void SetReturnAddressLocationResolver(
|
|
|
|
ReturnAddressLocationResolver resolver);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2013-06-28 13:40:41 +00:00
|
|
|
// Resolves pc_address through the resolution address function if one is set.
|
|
|
|
static inline Address* ResolveReturnAddressLocation(Address* pc_address);
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Printing support.
|
|
|
|
enum PrintMode { OVERVIEW, DETAILS };
|
|
|
|
virtual void Print(StringStream* accumulator,
|
|
|
|
PrintMode mode,
|
|
|
|
int index) const { }
|
|
|
|
|
2012-12-18 16:25:45 +00:00
|
|
|
Isolate* isolate() const { return isolate_; }
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
protected:
|
2013-06-27 09:34:31 +00:00
|
|
|
inline explicit StackFrame(StackFrameIteratorBase* iterator);
|
2008-07-03 15:10:15 +00:00
|
|
|
virtual ~StackFrame() { }
|
|
|
|
|
|
|
|
// Compute the stack pointer for the calling frame.
|
|
|
|
virtual Address GetCallerStackPointer() const = 0;
|
|
|
|
|
|
|
|
// Printing support.
|
|
|
|
static void PrintIndex(StringStream* accumulator,
|
|
|
|
PrintMode mode,
|
|
|
|
int index);
|
|
|
|
|
|
|
|
// Compute the stack frame type for the given state.
|
2013-06-27 09:34:31 +00:00
|
|
|
static Type ComputeType(const StackFrameIteratorBase* iterator, State* state);
|
2013-06-25 10:09:19 +00:00
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
bool can_access_heap_objects() const;
|
|
|
|
#endif
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
private:
|
2013-06-27 09:34:31 +00:00
|
|
|
const StackFrameIteratorBase* iterator_;
|
Simplify isolates access during stack iteration (WAS: Move SafeStackFrameIterator::active_count_...)
While trying to fix Mac and Windows versions for this change:
http://codereview.chromium.org/6771047/, I figured out, that we
already store an isolate in StackFrameIterator, so we can use it in
frame objects, instead of requiring it from caller.
I've changed iterators usage to the following scheme: whenever a
caller maintains an isolate pointer, it just passes it to stack
iterator, and no more worries about passing it to frame content
accessors. If a caller uses current isolate, it can omit passing it
to iterator, in this case, an iterator will use the current isolate,
too.
There was a special case with LiveEdit, which creates
detached copies of frame objects.
R=vitalyr@chromium.org
BUG=none
TEST=none
Review URL: http://codereview.chromium.org/6794019
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@7499 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2011-04-05 09:01:47 +00:00
|
|
|
Isolate* isolate_;
|
2008-09-12 03:29:06 +00:00
|
|
|
State state_;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2013-06-28 13:40:41 +00:00
|
|
|
static ReturnAddressLocationResolver return_address_location_resolver_;
|
|
|
|
|
2009-03-20 14:49:12 +00:00
|
|
|
// Fill in the state of the calling frame.
|
|
|
|
virtual void ComputeCallerState(State* state) const = 0;
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Get the type and the state of the calling frame.
|
2009-03-20 14:49:12 +00:00
|
|
|
virtual Type GetCallerState(State* state) const;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
Simplify isolates access during stack iteration (WAS: Move SafeStackFrameIterator::active_count_...)
While trying to fix Mac and Windows versions for this change:
http://codereview.chromium.org/6771047/, I figured out, that we
already store an isolate in StackFrameIterator, so we can use it in
frame objects, instead of requiring it from caller.
I've changed iterators usage to the following scheme: whenever a
caller maintains an isolate pointer, it just passes it to stack
iterator, and no more worries about passing it to frame content
accessors. If a caller uses current isolate, it can omit passing it
to iterator, in this case, an iterator will use the current isolate,
too.
There was a special case with LiveEdit, which creates
detached copies of frame objects.
R=vitalyr@chromium.org
BUG=none
TEST=none
Review URL: http://codereview.chromium.org/6794019
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@7499 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2011-04-05 09:01:47 +00:00
|
|
|
static const intptr_t kIsolateTag = 1;
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
friend class StackFrameIterator;
|
2013-06-27 09:34:31 +00:00
|
|
|
friend class StackFrameIteratorBase;
|
2008-07-03 15:10:15 +00:00
|
|
|
friend class StackHandlerIterator;
|
2009-03-20 14:49:12 +00:00
|
|
|
friend class SafeStackFrameIterator;
|
2010-09-16 08:51:13 +00:00
|
|
|
|
2010-04-06 17:58:28 +00:00
|
|
|
private:
|
|
|
|
void operator=(const StackFrame& original);
|
2008-07-03 15:10:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// Entry frames are used to enter JavaScript execution from C.
|
|
|
|
class EntryFrame: public StackFrame {
|
|
|
|
public:
|
|
|
|
virtual Type type() const { return ENTRY; }
|
|
|
|
|
2010-08-17 11:44:01 +00:00
|
|
|
virtual Code* unchecked_code() const;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// Garbage collection support.
|
|
|
|
virtual void Iterate(ObjectVisitor* v) const;
|
|
|
|
|
|
|
|
static EntryFrame* cast(StackFrame* frame) {
|
2014-08-04 11:34:54 +00:00
|
|
|
DCHECK(frame->is_entry());
|
2008-07-03 15:10:15 +00:00
|
|
|
return static_cast<EntryFrame*>(frame);
|
|
|
|
}
|
2010-04-06 17:58:28 +00:00
|
|
|
virtual void SetCallerFp(Address caller_fp);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
protected:
|
2013-06-27 09:34:31 +00:00
|
|
|
inline explicit EntryFrame(StackFrameIteratorBase* iterator);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// The caller stack pointer for entry frames is always zero. The
|
|
|
|
// real information about the caller frame is available through the
|
|
|
|
// link to the top exit frame.
|
|
|
|
virtual Address GetCallerStackPointer() const { return 0; }
|
|
|
|
|
|
|
|
private:
|
2009-03-20 14:49:12 +00:00
|
|
|
virtual void ComputeCallerState(State* state) const;
|
2008-07-03 15:10:15 +00:00
|
|
|
virtual Type GetCallerState(State* state) const;
|
|
|
|
|
2013-06-27 09:34:31 +00:00
|
|
|
friend class StackFrameIteratorBase;
|
2008-07-03 15:10:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class EntryConstructFrame: public EntryFrame {
|
|
|
|
public:
|
|
|
|
virtual Type type() const { return ENTRY_CONSTRUCT; }
|
|
|
|
|
2010-08-17 11:44:01 +00:00
|
|
|
virtual Code* unchecked_code() const;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
static EntryConstructFrame* cast(StackFrame* frame) {
|
2014-08-04 11:34:54 +00:00
|
|
|
DCHECK(frame->is_entry_construct());
|
2008-07-03 15:10:15 +00:00
|
|
|
return static_cast<EntryConstructFrame*>(frame);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
2013-06-27 09:34:31 +00:00
|
|
|
inline explicit EntryConstructFrame(StackFrameIteratorBase* iterator);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
private:
|
2013-06-27 09:34:31 +00:00
|
|
|
friend class StackFrameIteratorBase;
|
2008-07-03 15:10:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// Exit frames are used to exit JavaScript execution and go to C.
|
|
|
|
class ExitFrame: public StackFrame {
|
|
|
|
public:
|
|
|
|
virtual Type type() const { return EXIT; }
|
|
|
|
|
2010-08-17 11:44:01 +00:00
|
|
|
virtual Code* unchecked_code() const;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2009-11-04 08:51:48 +00:00
|
|
|
Object*& code_slot() const;
|
|
|
|
|
2009-01-15 19:08:34 +00:00
|
|
|
// Garbage collection support.
|
2008-07-03 15:10:15 +00:00
|
|
|
virtual void Iterate(ObjectVisitor* v) const;
|
|
|
|
|
2010-04-06 17:58:28 +00:00
|
|
|
virtual void SetCallerFp(Address caller_fp);
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
static ExitFrame* cast(StackFrame* frame) {
|
2014-08-04 11:34:54 +00:00
|
|
|
DCHECK(frame->is_exit());
|
2008-07-03 15:10:15 +00:00
|
|
|
return static_cast<ExitFrame*>(frame);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Compute the state and type of an exit frame given a frame
|
|
|
|
// pointer. Used when constructing the first stack frame seen by an
|
|
|
|
// iterator and the frames following entry frames.
|
|
|
|
static Type GetStateForFramePointer(Address fp, State* state);
|
2010-09-16 08:23:34 +00:00
|
|
|
static Address ComputeStackPointer(Address fp);
|
|
|
|
static void FillState(Address fp, Address sp, State* state);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
protected:
|
2013-06-27 09:34:31 +00:00
|
|
|
inline explicit ExitFrame(StackFrameIteratorBase* iterator);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
virtual Address GetCallerStackPointer() const;
|
|
|
|
|
|
|
|
private:
|
2009-03-20 14:49:12 +00:00
|
|
|
virtual void ComputeCallerState(State* state) const;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2013-06-27 09:34:31 +00:00
|
|
|
friend class StackFrameIteratorBase;
|
2008-07-03 15:10:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class StandardFrame: public StackFrame {
|
|
|
|
public:
|
|
|
|
// Testers.
|
|
|
|
virtual bool is_standard() const { return true; }
|
|
|
|
|
|
|
|
// Accessors.
|
|
|
|
inline Object* context() const;
|
|
|
|
|
|
|
|
// Access the expressions in the stack frame including locals.
|
|
|
|
inline Object* GetExpression(int index) const;
|
|
|
|
inline void SetExpression(int index, Object* value);
|
|
|
|
int ComputeExpressionsCount() const;
|
2011-06-29 13:02:00 +00:00
|
|
|
static Object* GetExpression(Address fp, int index);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2010-04-06 17:58:28 +00:00
|
|
|
virtual void SetCallerFp(Address caller_fp);
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
static StandardFrame* cast(StackFrame* frame) {
|
2014-08-04 11:34:54 +00:00
|
|
|
DCHECK(frame->is_standard());
|
2008-07-03 15:10:15 +00:00
|
|
|
return static_cast<StandardFrame*>(frame);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
2013-06-27 09:34:31 +00:00
|
|
|
inline explicit StandardFrame(StackFrameIteratorBase* iterator);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2009-03-20 14:49:12 +00:00
|
|
|
virtual void ComputeCallerState(State* state) const;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// Accessors.
|
|
|
|
inline Address caller_fp() const;
|
|
|
|
inline Address caller_pc() const;
|
|
|
|
|
|
|
|
// Computes the address of the PC field in the standard frame given
|
|
|
|
// by the provided frame pointer.
|
|
|
|
static inline Address ComputePCAddress(Address fp);
|
|
|
|
|
2014-03-14 15:11:58 +00:00
|
|
|
// Computes the address of the constant pool field in the standard
|
|
|
|
// frame given by the provided frame pointer.
|
|
|
|
static inline Address ComputeConstantPoolAddress(Address fp);
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Iterate over expression stack including stack handlers, locals,
|
|
|
|
// and parts of the fixed part including context and code fields.
|
|
|
|
void IterateExpressions(ObjectVisitor* v) const;
|
|
|
|
|
|
|
|
// Returns the address of the n'th expression stack element.
|
|
|
|
Address GetExpressionAddress(int n) const;
|
2011-06-29 13:02:00 +00:00
|
|
|
static Address GetExpressionAddress(Address fp, int n);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// Determines if the standard frame for the given frame pointer is
|
|
|
|
// an arguments adaptor frame.
|
|
|
|
static inline bool IsArgumentsAdaptorFrame(Address fp);
|
|
|
|
|
2008-10-10 09:09:38 +00:00
|
|
|
// Determines if the standard frame for the given frame pointer is a
|
|
|
|
// construct frame.
|
|
|
|
static inline bool IsConstructFrame(Address fp);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2012-12-18 16:25:45 +00:00
|
|
|
// Used by OptimizedFrames and StubFrames.
|
|
|
|
void IterateCompiledFrame(ObjectVisitor* v) const;
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
private:
|
|
|
|
friend class StackFrame;
|
2013-06-27 09:34:31 +00:00
|
|
|
friend class SafeStackFrameIterator;
|
2008-07-03 15:10:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2010-12-07 11:31:57 +00:00
|
|
|
class FrameSummary BASE_EMBEDDED {
|
|
|
|
public:
|
2015-08-31 15:04:22 +00:00
|
|
|
FrameSummary(Object* receiver, JSFunction* function, Code* code, int offset,
|
|
|
|
bool is_constructor);
|
|
|
|
|
2010-12-07 11:31:57 +00:00
|
|
|
Handle<Object> receiver() { return receiver_; }
|
|
|
|
Handle<JSFunction> function() { return function_; }
|
|
|
|
Handle<Code> code() { return code_; }
|
2011-04-01 10:30:09 +00:00
|
|
|
Address pc() { return code_->address() + offset_; }
|
2010-12-07 11:31:57 +00:00
|
|
|
int offset() { return offset_; }
|
|
|
|
bool is_constructor() { return is_constructor_; }
|
|
|
|
|
|
|
|
void Print();
|
|
|
|
|
|
|
|
private:
|
|
|
|
Handle<Object> receiver_;
|
|
|
|
Handle<JSFunction> function_;
|
|
|
|
Handle<Code> code_;
|
|
|
|
int offset_;
|
|
|
|
bool is_constructor_;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
class JavaScriptFrame: public StandardFrame {
|
|
|
|
public:
|
|
|
|
virtual Type type() const { return JAVA_SCRIPT; }
|
|
|
|
|
|
|
|
// Accessors.
|
2013-07-11 16:45:58 +00:00
|
|
|
inline JSFunction* function() const;
|
2008-07-03 15:10:15 +00:00
|
|
|
inline Object* receiver() const;
|
|
|
|
inline void set_receiver(Object* value);
|
|
|
|
|
|
|
|
// Access the parameters.
|
2011-04-06 14:23:27 +00:00
|
|
|
inline Address GetParameterSlot(int index) const;
|
|
|
|
inline Object* GetParameter(int index) const;
|
|
|
|
inline int ComputeParametersCount() const {
|
|
|
|
return GetNumberOfIncomingArguments();
|
|
|
|
}
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2013-04-26 12:09:32 +00:00
|
|
|
// Access the operand stack.
|
|
|
|
inline Address GetOperandSlot(int index) const;
|
|
|
|
inline Object* GetOperand(int index) const;
|
|
|
|
inline int ComputeOperandsCount() const;
|
|
|
|
|
2015-03-25 13:13:51 +00:00
|
|
|
// Generator support to preserve operand stack.
|
|
|
|
void SaveOperandStack(FixedArray* store) const;
|
|
|
|
void RestoreOperandStack(FixedArray* store);
|
2013-05-08 08:08:23 +00:00
|
|
|
|
2012-12-11 23:27:38 +00:00
|
|
|
// Debugger access.
|
|
|
|
void SetParameterValue(int index, Object* value) const;
|
|
|
|
|
2008-10-10 09:09:38 +00:00
|
|
|
// Check if this frame is a constructor frame invoked through 'new'.
|
2008-07-03 15:10:15 +00:00
|
|
|
bool IsConstructor() const;
|
|
|
|
|
2015-07-06 18:02:41 +00:00
|
|
|
// Returns the original constructor function that was used in the constructor
|
|
|
|
// call to this frame. Note that this is only valid on constructor frames.
|
|
|
|
Object* GetOriginalConstructor() const;
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Check if this frame has "adapted" arguments in the sense that the
|
|
|
|
// actual passed arguments are available in an arguments adaptor
|
|
|
|
// frame below it on the stack.
|
|
|
|
inline bool has_adapted_arguments() const;
|
2011-06-29 13:02:00 +00:00
|
|
|
int GetArgumentsLength() const;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2009-01-15 19:08:34 +00:00
|
|
|
// Garbage collection support.
|
2008-07-03 15:10:15 +00:00
|
|
|
virtual void Iterate(ObjectVisitor* v) const;
|
|
|
|
|
|
|
|
// Printing support.
|
|
|
|
virtual void Print(StringStream* accumulator,
|
|
|
|
PrintMode mode,
|
|
|
|
int index) const;
|
|
|
|
|
|
|
|
// Determine the code for the frame.
|
2010-08-17 11:44:01 +00:00
|
|
|
virtual Code* unchecked_code() const;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2010-12-07 11:31:57 +00:00
|
|
|
// Return a list with JSFunctions of this frame.
|
|
|
|
virtual void GetFunctions(List<JSFunction*>* functions);
|
|
|
|
|
|
|
|
// Build a list with summaries for this frame including all inlined frames.
|
|
|
|
virtual void Summarize(List<FrameSummary>* frames);
|
|
|
|
|
2015-03-25 13:13:51 +00:00
|
|
|
// Lookup exception handler for current {pc}, returns -1 if none found. Also
|
|
|
|
// returns the expected number of stack slots at the handler site.
|
2015-05-29 10:05:22 +00:00
|
|
|
virtual int LookupExceptionHandlerInTable(
|
|
|
|
int* stack_slots, HandlerTable::CatchPrediction* prediction);
|
2015-03-25 13:13:51 +00:00
|
|
|
|
2013-05-17 08:27:56 +00:00
|
|
|
// Architecture-specific register description.
|
|
|
|
static Register fp_register();
|
|
|
|
static Register context_register();
|
2013-12-30 11:23:59 +00:00
|
|
|
static Register constant_pool_pointer_register();
|
2013-05-17 08:27:56 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
static JavaScriptFrame* cast(StackFrame* frame) {
|
2014-08-04 11:34:54 +00:00
|
|
|
DCHECK(frame->is_java_script());
|
2008-07-03 15:10:15 +00:00
|
|
|
return static_cast<JavaScriptFrame*>(frame);
|
|
|
|
}
|
|
|
|
|
2014-07-08 08:28:08 +00:00
|
|
|
static void PrintFunctionAndOffset(JSFunction* function, Code* code,
|
|
|
|
Address pc, FILE* file,
|
|
|
|
bool print_line_number);
|
|
|
|
|
|
|
|
static void PrintTop(Isolate* isolate, FILE* file, bool print_args,
|
2013-02-15 09:27:10 +00:00
|
|
|
bool print_line_number);
|
2011-10-20 09:38:24 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
protected:
|
2013-06-27 09:34:31 +00:00
|
|
|
inline explicit JavaScriptFrame(StackFrameIteratorBase* iterator);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
virtual Address GetCallerStackPointer() const;
|
|
|
|
|
2011-04-06 14:23:27 +00:00
|
|
|
virtual int GetNumberOfIncomingArguments() const;
|
|
|
|
|
2010-12-07 11:31:57 +00:00
|
|
|
// Garbage collection support. Iterates over incoming arguments,
|
|
|
|
// receiver, and any callee-saved registers.
|
|
|
|
void IterateArguments(ObjectVisitor* v) const;
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
private:
|
2009-03-20 14:49:12 +00:00
|
|
|
inline Object* function_slot_object() const;
|
|
|
|
|
2013-06-27 09:34:31 +00:00
|
|
|
friend class StackFrameIteratorBase;
|
2008-07-03 15:10:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2012-12-18 16:25:45 +00:00
|
|
|
class StubFrame : public StandardFrame {
|
|
|
|
public:
|
|
|
|
virtual Type type() const { return STUB; }
|
|
|
|
|
|
|
|
// GC support.
|
|
|
|
virtual void Iterate(ObjectVisitor* v) const;
|
|
|
|
|
|
|
|
// Determine the code for the frame.
|
|
|
|
virtual Code* unchecked_code() const;
|
|
|
|
|
|
|
|
protected:
|
2013-06-27 09:34:31 +00:00
|
|
|
inline explicit StubFrame(StackFrameIteratorBase* iterator);
|
2012-12-18 16:25:45 +00:00
|
|
|
|
|
|
|
virtual Address GetCallerStackPointer() const;
|
|
|
|
|
|
|
|
virtual int GetNumberOfIncomingArguments() const;
|
|
|
|
|
2013-06-27 09:34:31 +00:00
|
|
|
friend class StackFrameIteratorBase;
|
2012-12-18 16:25:45 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2012-12-10 11:09:12 +00:00
|
|
|
class OptimizedFrame : public JavaScriptFrame {
|
2010-12-07 11:31:57 +00:00
|
|
|
public:
|
|
|
|
virtual Type type() const { return OPTIMIZED; }
|
|
|
|
|
|
|
|
// GC support.
|
|
|
|
virtual void Iterate(ObjectVisitor* v) const;
|
|
|
|
|
|
|
|
// Return a list with JSFunctions of this frame.
|
|
|
|
// The functions are ordered bottom-to-top (i.e. functions.last()
|
|
|
|
// is the top-most activation)
|
|
|
|
virtual void GetFunctions(List<JSFunction*>* functions);
|
|
|
|
|
|
|
|
virtual void Summarize(List<FrameSummary>* frames);
|
|
|
|
|
2015-03-25 13:13:51 +00:00
|
|
|
// Lookup exception handler for current {pc}, returns -1 if none found. Also
|
|
|
|
// returns the expected number of stack slots at the handler site.
|
2015-05-29 10:05:22 +00:00
|
|
|
virtual int LookupExceptionHandlerInTable(
|
|
|
|
int* stack_slots, HandlerTable::CatchPrediction* prediction);
|
2015-03-25 13:13:51 +00:00
|
|
|
|
2010-12-07 11:31:57 +00:00
|
|
|
DeoptimizationInputData* GetDeoptimizationData(int* deopt_index);
|
|
|
|
|
[turbofan] Unify referencing of stack slots
Previously, it was not possible to specify StackSlotOperands for all
slots in both the caller and callee stacks. Specifically, the region
of the callee's stack including the saved return address, frame
pointer, function pointer and context pointer could not be addressed
by the register allocator/gap resolver.
In preparation for better tail call support, which will use the gap
resolver to reconcile outgoing parameters, this change makes it
possible to address all slots on the stack, because slots in the
previously inaccessible dead zone may become parameter slots for
outgoing tail calls. All caller stack slots are accessible as they
were before, with slot -1 corresponding to the last stack
parameter. Stack slot indices >= 0 access the callee stack, with slot
0 corresponding to the callee's saved return address, 1 corresponding
to the saved frame pointer, 2 corresponding to the current function
context, 3 corresponding to the frame marker/JSFunction, and slots 4
and above corresponding to spill slots.
The following changes were specifically needed:
* Frame has been changed to explicitly manage three areas of the
callee frame, the fixed header, the spill slot area, and the
callee-saved register area.
* Conversions from stack slot indices to fp offsets all now go through
a common bottleneck: OptimizedFrame::StackSlotOffsetRelativeToFp
* The generation of deoptimization translation tables has been changed
to support the new stack slot indexing scheme. Crankshaft, which
doesn't support the new slot numbering in its register allocator,
must adapt the indexes when creating translation tables.
* Callee-saved parameters are now kept below spill slots, not above,
to support saving only the optimal set of used registers, which is
only known after register allocation is finished and spill slots
have been allocated.
Review URL: https://codereview.chromium.org/1261923007
Cr-Commit-Position: refs/heads/master@{#30224}
2015-08-18 14:47:56 +00:00
|
|
|
static int StackSlotOffsetRelativeToFp(int slot_index);
|
|
|
|
|
2010-12-07 11:31:57 +00:00
|
|
|
protected:
|
2013-06-27 09:34:31 +00:00
|
|
|
inline explicit OptimizedFrame(StackFrameIteratorBase* iterator);
|
2010-12-07 11:31:57 +00:00
|
|
|
|
|
|
|
private:
|
2013-06-27 09:34:31 +00:00
|
|
|
friend class StackFrameIteratorBase;
|
2015-06-16 06:04:42 +00:00
|
|
|
|
|
|
|
Object* StackSlotAt(int index) const;
|
2010-12-07 11:31:57 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Arguments adaptor frames are automatically inserted below
|
|
|
|
// JavaScript frames when the actual number of parameters does not
|
|
|
|
// match the formal number of parameters.
|
|
|
|
class ArgumentsAdaptorFrame: public JavaScriptFrame {
|
|
|
|
public:
|
|
|
|
virtual Type type() const { return ARGUMENTS_ADAPTOR; }
|
|
|
|
|
|
|
|
// Determine the code for the frame.
|
2010-08-17 11:44:01 +00:00
|
|
|
virtual Code* unchecked_code() const;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
static ArgumentsAdaptorFrame* cast(StackFrame* frame) {
|
2014-08-04 11:34:54 +00:00
|
|
|
DCHECK(frame->is_arguments_adaptor());
|
2008-07-03 15:10:15 +00:00
|
|
|
return static_cast<ArgumentsAdaptorFrame*>(frame);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Printing support.
|
|
|
|
virtual void Print(StringStream* accumulator,
|
|
|
|
PrintMode mode,
|
|
|
|
int index) const;
|
2011-09-08 19:57:14 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
protected:
|
2013-06-27 09:34:31 +00:00
|
|
|
inline explicit ArgumentsAdaptorFrame(StackFrameIteratorBase* iterator);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2011-10-03 11:13:20 +00:00
|
|
|
virtual int GetNumberOfIncomingArguments() const;
|
2011-04-06 14:23:27 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
virtual Address GetCallerStackPointer() const;
|
|
|
|
|
|
|
|
private:
|
2013-06-27 09:34:31 +00:00
|
|
|
friend class StackFrameIteratorBase;
|
2008-07-03 15:10:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class InternalFrame: public StandardFrame {
|
|
|
|
public:
|
|
|
|
virtual Type type() const { return INTERNAL; }
|
|
|
|
|
2009-01-15 19:08:34 +00:00
|
|
|
// Garbage collection support.
|
2008-07-03 15:10:15 +00:00
|
|
|
virtual void Iterate(ObjectVisitor* v) const;
|
|
|
|
|
|
|
|
// Determine the code for the frame.
|
2010-08-17 11:44:01 +00:00
|
|
|
virtual Code* unchecked_code() const;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
static InternalFrame* cast(StackFrame* frame) {
|
2014-08-04 11:34:54 +00:00
|
|
|
DCHECK(frame->is_internal());
|
2008-07-03 15:10:15 +00:00
|
|
|
return static_cast<InternalFrame*>(frame);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
2013-06-27 09:34:31 +00:00
|
|
|
inline explicit InternalFrame(StackFrameIteratorBase* iterator);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
virtual Address GetCallerStackPointer() const;
|
|
|
|
|
|
|
|
private:
|
2013-06-27 09:34:31 +00:00
|
|
|
friend class StackFrameIteratorBase;
|
2008-07-03 15:10:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2013-02-05 08:09:32 +00:00
|
|
|
class StubFailureTrampolineFrame: public StandardFrame {
|
2013-01-29 09:12:20 +00:00
|
|
|
public:
|
2013-02-05 08:09:32 +00:00
|
|
|
// sizeof(Arguments) - sizeof(Arguments*) is 3 * kPointerSize), but the
|
|
|
|
// presubmit script complains about using sizeof() on a type.
|
|
|
|
static const int kFirstRegisterParameterFrameOffset =
|
|
|
|
StandardFrameConstants::kMarkerOffset - 3 * kPointerSize;
|
|
|
|
|
|
|
|
static const int kCallerStackParameterCountFrameOffset =
|
|
|
|
StandardFrameConstants::kMarkerOffset - 2 * kPointerSize;
|
|
|
|
|
2013-01-29 09:12:20 +00:00
|
|
|
virtual Type type() const { return STUB_FAILURE_TRAMPOLINE; }
|
|
|
|
|
2013-02-05 08:09:32 +00:00
|
|
|
// Get the code associated with this frame.
|
|
|
|
// This method could be called during marking phase of GC.
|
|
|
|
virtual Code* unchecked_code() const;
|
|
|
|
|
2013-01-29 09:12:20 +00:00
|
|
|
virtual void Iterate(ObjectVisitor* v) const;
|
|
|
|
|
2013-03-08 16:18:50 +00:00
|
|
|
// Architecture-specific register description.
|
|
|
|
static Register fp_register();
|
|
|
|
static Register context_register();
|
2013-12-30 11:23:59 +00:00
|
|
|
static Register constant_pool_pointer_register();
|
2013-03-08 16:18:50 +00:00
|
|
|
|
2013-01-29 09:12:20 +00:00
|
|
|
protected:
|
|
|
|
inline explicit StubFailureTrampolineFrame(
|
2013-06-27 09:34:31 +00:00
|
|
|
StackFrameIteratorBase* iterator);
|
2013-01-29 09:12:20 +00:00
|
|
|
|
2013-02-05 08:09:32 +00:00
|
|
|
virtual Address GetCallerStackPointer() const;
|
|
|
|
|
2013-01-29 09:12:20 +00:00
|
|
|
private:
|
2013-06-27 09:34:31 +00:00
|
|
|
friend class StackFrameIteratorBase;
|
2013-01-29 09:12:20 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2008-10-10 09:09:38 +00:00
|
|
|
// Construct frames are special trampoline frames introduced to handle
|
|
|
|
// function invocations through 'new'.
|
|
|
|
class ConstructFrame: public InternalFrame {
|
|
|
|
public:
|
|
|
|
virtual Type type() const { return CONSTRUCT; }
|
|
|
|
|
|
|
|
static ConstructFrame* cast(StackFrame* frame) {
|
2014-08-04 11:34:54 +00:00
|
|
|
DCHECK(frame->is_construct());
|
2008-10-10 09:09:38 +00:00
|
|
|
return static_cast<ConstructFrame*>(frame);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
2013-06-27 09:34:31 +00:00
|
|
|
inline explicit ConstructFrame(StackFrameIteratorBase* iterator);
|
2008-10-10 09:09:38 +00:00
|
|
|
|
|
|
|
private:
|
2013-06-27 09:34:31 +00:00
|
|
|
friend class StackFrameIteratorBase;
|
2008-10-10 09:09:38 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2013-06-27 09:34:31 +00:00
|
|
|
class StackFrameIteratorBase BASE_EMBEDDED {
|
2008-07-03 15:10:15 +00:00
|
|
|
public:
|
Simplify isolates access during stack iteration (WAS: Move SafeStackFrameIterator::active_count_...)
While trying to fix Mac and Windows versions for this change:
http://codereview.chromium.org/6771047/, I figured out, that we
already store an isolate in StackFrameIterator, so we can use it in
frame objects, instead of requiring it from caller.
I've changed iterators usage to the following scheme: whenever a
caller maintains an isolate pointer, it just passes it to stack
iterator, and no more worries about passing it to frame content
accessors. If a caller uses current isolate, it can omit passing it
to iterator, in this case, an iterator will use the current isolate,
too.
There was a special case with LiveEdit, which creates
detached copies of frame objects.
R=vitalyr@chromium.org
BUG=none
TEST=none
Review URL: http://codereview.chromium.org/6794019
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@7499 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2011-04-05 09:01:47 +00:00
|
|
|
Isolate* isolate() const { return isolate_; }
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
bool done() const { return frame_ == NULL; }
|
2013-06-25 10:09:19 +00:00
|
|
|
|
2013-06-27 09:34:31 +00:00
|
|
|
protected:
|
|
|
|
// An iterator that iterates over a given thread's stack.
|
|
|
|
StackFrameIteratorBase(Isolate* isolate, bool can_access_heap_objects);
|
2008-07-03 15:10:15 +00:00
|
|
|
|
Simplify isolates access during stack iteration (WAS: Move SafeStackFrameIterator::active_count_...)
While trying to fix Mac and Windows versions for this change:
http://codereview.chromium.org/6771047/, I figured out, that we
already store an isolate in StackFrameIterator, so we can use it in
frame objects, instead of requiring it from caller.
I've changed iterators usage to the following scheme: whenever a
caller maintains an isolate pointer, it just passes it to stack
iterator, and no more worries about passing it to frame content
accessors. If a caller uses current isolate, it can omit passing it
to iterator, in this case, an iterator will use the current isolate,
too.
There was a special case with LiveEdit, which creates
detached copies of frame objects.
R=vitalyr@chromium.org
BUG=none
TEST=none
Review URL: http://codereview.chromium.org/6794019
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@7499 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2011-04-05 09:01:47 +00:00
|
|
|
Isolate* isolate_;
|
2008-07-03 15:10:15 +00:00
|
|
|
#define DECLARE_SINGLETON(ignore, type) type type##_;
|
|
|
|
STACK_FRAME_TYPE_LIST(DECLARE_SINGLETON)
|
|
|
|
#undef DECLARE_SINGLETON
|
|
|
|
StackFrame* frame_;
|
|
|
|
StackHandler* handler_;
|
2013-06-25 10:09:19 +00:00
|
|
|
const bool can_access_heap_objects_;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
StackHandler* handler() const {
|
2014-08-04 11:34:54 +00:00
|
|
|
DCHECK(!done());
|
2008-07-03 15:10:15 +00:00
|
|
|
return handler_;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the type-specific frame singleton in a given state.
|
|
|
|
StackFrame* SingletonFor(StackFrame::Type type, StackFrame::State* state);
|
2009-03-20 14:49:12 +00:00
|
|
|
// A helper function, can return a NULL pointer.
|
|
|
|
StackFrame* SingletonFor(StackFrame::Type type);
|
|
|
|
|
2013-06-27 09:34:31 +00:00
|
|
|
private:
|
2008-07-03 15:10:15 +00:00
|
|
|
friend class StackFrame;
|
2013-06-27 09:34:31 +00:00
|
|
|
DISALLOW_COPY_AND_ASSIGN(StackFrameIteratorBase);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class StackFrameIterator: public StackFrameIteratorBase {
|
|
|
|
public:
|
|
|
|
// An iterator that iterates over the isolate's current thread's stack,
|
|
|
|
explicit StackFrameIterator(Isolate* isolate);
|
|
|
|
// An iterator that iterates over a given thread's stack.
|
|
|
|
StackFrameIterator(Isolate* isolate, ThreadLocalTop* t);
|
|
|
|
|
|
|
|
StackFrame* frame() const {
|
2014-08-04 11:34:54 +00:00
|
|
|
DCHECK(!done());
|
2013-06-27 09:34:31 +00:00
|
|
|
return frame_;
|
|
|
|
}
|
|
|
|
void Advance();
|
|
|
|
|
|
|
|
private:
|
|
|
|
// Go back to the first frame.
|
|
|
|
void Reset(ThreadLocalTop* top);
|
|
|
|
|
2008-08-28 09:55:41 +00:00
|
|
|
DISALLOW_COPY_AND_ASSIGN(StackFrameIterator);
|
2008-07-03 15:10:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// Iterator that supports iterating through all JavaScript frames.
|
2013-06-24 08:38:37 +00:00
|
|
|
class JavaScriptFrameIterator BASE_EMBEDDED {
|
2008-07-03 15:10:15 +00:00
|
|
|
public:
|
2013-06-24 08:38:37 +00:00
|
|
|
inline explicit JavaScriptFrameIterator(Isolate* isolate);
|
|
|
|
inline JavaScriptFrameIterator(Isolate* isolate, ThreadLocalTop* top);
|
2008-07-03 15:10:15 +00:00
|
|
|
// Skip frames until the frame with the given id is reached.
|
2013-06-24 08:38:37 +00:00
|
|
|
JavaScriptFrameIterator(Isolate* isolate, StackFrame::Id id);
|
2011-03-18 20:35:07 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
inline JavaScriptFrame* frame() const;
|
|
|
|
|
|
|
|
bool done() const { return iterator_.done(); }
|
|
|
|
void Advance();
|
|
|
|
|
|
|
|
// Advance to the frame holding the arguments for the current
|
|
|
|
// frame. This only affects the current frame if it has adapted
|
|
|
|
// arguments.
|
|
|
|
void AdvanceToArgumentsFrame();
|
|
|
|
|
|
|
|
private:
|
2013-06-24 08:38:37 +00:00
|
|
|
StackFrameIterator iterator_;
|
2009-03-03 11:56:44 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// NOTE: The stack trace frame iterator is an iterator that only
|
|
|
|
// traverse proper JavaScript frames; that is JavaScript frames that
|
|
|
|
// have proper JavaScript functions. This excludes the problematic
|
|
|
|
// functions in runtime.js.
|
|
|
|
class StackTraceFrameIterator: public JavaScriptFrameIterator {
|
|
|
|
public:
|
Simplify isolates access during stack iteration (WAS: Move SafeStackFrameIterator::active_count_...)
While trying to fix Mac and Windows versions for this change:
http://codereview.chromium.org/6771047/, I figured out, that we
already store an isolate in StackFrameIterator, so we can use it in
frame objects, instead of requiring it from caller.
I've changed iterators usage to the following scheme: whenever a
caller maintains an isolate pointer, it just passes it to stack
iterator, and no more worries about passing it to frame content
accessors. If a caller uses current isolate, it can omit passing it
to iterator, in this case, an iterator will use the current isolate,
too.
There was a special case with LiveEdit, which creates
detached copies of frame objects.
R=vitalyr@chromium.org
BUG=none
TEST=none
Review URL: http://codereview.chromium.org/6794019
git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@7499 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
2011-04-05 09:01:47 +00:00
|
|
|
explicit StackTraceFrameIterator(Isolate* isolate);
|
2009-03-03 11:56:44 +00:00
|
|
|
void Advance();
|
2010-02-01 10:34:57 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
bool IsValidFrame();
|
2009-03-03 11:56:44 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2013-06-27 09:34:31 +00:00
|
|
|
class SafeStackFrameIterator: public StackFrameIteratorBase {
|
2009-03-03 11:56:44 +00:00
|
|
|
public:
|
2011-03-18 20:35:07 +00:00
|
|
|
SafeStackFrameIterator(Isolate* isolate,
|
|
|
|
Address fp, Address sp,
|
2013-07-03 16:20:59 +00:00
|
|
|
Address js_entry_sp);
|
2009-03-03 11:56:44 +00:00
|
|
|
|
2013-07-23 15:01:38 +00:00
|
|
|
inline StackFrame* frame() const;
|
2009-03-03 11:56:44 +00:00
|
|
|
void Advance();
|
|
|
|
|
2013-07-03 14:04:37 +00:00
|
|
|
StackFrame::Type top_frame_type() const { return top_frame_type_; }
|
|
|
|
|
2013-06-24 13:27:48 +00:00
|
|
|
private:
|
2013-06-25 07:14:06 +00:00
|
|
|
void AdvanceOneFrame();
|
|
|
|
|
2009-03-20 14:49:12 +00:00
|
|
|
bool IsValidStackAddress(Address addr) const {
|
2013-06-27 09:34:31 +00:00
|
|
|
return low_bound_ <= addr && addr <= high_bound_;
|
2009-03-03 11:56:44 +00:00
|
|
|
}
|
2009-03-20 14:49:12 +00:00
|
|
|
bool IsValidFrame(StackFrame* frame) const;
|
|
|
|
bool IsValidCaller(StackFrame* frame);
|
2013-06-27 09:34:31 +00:00
|
|
|
bool IsValidExitFrame(Address fp) const;
|
|
|
|
bool IsValidTop(ThreadLocalTop* top) const;
|
2009-03-03 11:56:44 +00:00
|
|
|
|
2013-06-27 09:34:31 +00:00
|
|
|
const Address low_bound_;
|
|
|
|
const Address high_bound_;
|
2013-07-03 14:04:37 +00:00
|
|
|
StackFrame::Type top_frame_type_;
|
2013-07-23 15:01:38 +00:00
|
|
|
ExternalCallbackScope* external_callback_scope_;
|
2008-07-03 15:10:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
class StackFrameLocator BASE_EMBEDDED {
|
|
|
|
public:
|
2013-02-15 09:27:10 +00:00
|
|
|
explicit StackFrameLocator(Isolate* isolate) : iterator_(isolate) {}
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Find the nth JavaScript frame on the stack. The caller must
|
|
|
|
// guarantee that such a frame exists.
|
|
|
|
JavaScriptFrame* FindJavaScriptFrame(int n);
|
|
|
|
|
|
|
|
private:
|
|
|
|
StackFrameIterator iterator_;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
2010-04-06 17:58:28 +00:00
|
|
|
// Reads all frames on the current stack and copies them into the current
|
|
|
|
// zone memory.
|
2013-02-15 09:27:10 +00:00
|
|
|
Vector<StackFrame*> CreateStackMap(Isolate* isolate, Zone* zone);
|
2010-04-06 17:58:28 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
} } // namespace v8::internal
|
|
|
|
|
|
|
|
#endif // V8_FRAMES_H_
|