2012-02-09 09:43:37 +00:00
|
|
|
// Copyright 2012 the V8 project authors. All rights reserved.
|
2008-07-03 15:10:15 +00:00
|
|
|
// Redistribution and use in source and binary forms, with or without
|
|
|
|
// modification, are permitted provided that the following conditions are
|
|
|
|
// met:
|
|
|
|
//
|
|
|
|
// * Redistributions of source code must retain the above copyright
|
|
|
|
// notice, this list of conditions and the following disclaimer.
|
|
|
|
// * Redistributions in binary form must reproduce the above
|
|
|
|
// copyright notice, this list of conditions and the following
|
|
|
|
// disclaimer in the documentation and/or other materials provided
|
|
|
|
// with the distribution.
|
|
|
|
// * Neither the name of Google Inc. nor the names of its
|
|
|
|
// contributors may be used to endorse or promote products derived
|
|
|
|
// from this software without specific prior written permission.
|
|
|
|
//
|
|
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
|
|
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
|
|
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
|
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
|
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
|
|
|
#ifndef V8_FRAMES_H_
|
|
|
|
#define V8_FRAMES_H_
|
|
|
|
|
2011-05-06 06:50:20 +00:00
|
|
|
#include "allocation.h"
|
2011-03-18 20:35:07 +00:00
|
|
|
#include "handles.h"
|
2011-01-12 14:14:14 +00:00
|
|
|
#include "safepoint-table.h"
|
|
|
|
|
2009-05-25 10:05:56 +00:00
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
typedef uint32_t RegList;
|
|
|
|
|
|
|
|
// 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
|
|
|
};
|
|
|
|
|
|
|
|
|
2013-04-08 11:17:32 +00:00
|
|
|
class StackHandlerConstants : public AllStatic {
|
|
|
|
public:
|
|
|
|
static const int kNextOffset = 0 * kPointerSize;
|
|
|
|
static const int kCodeOffset = 1 * kPointerSize;
|
|
|
|
static const int kStateOffset = 2 * kPointerSize;
|
|
|
|
static const int kContextOffset = 3 * kPointerSize;
|
|
|
|
static const int kFPOffset = 4 * kPointerSize;
|
|
|
|
|
2013-07-23 13:46:10 +00:00
|
|
|
static const int kSize = kFPOffset + kFPOnStackSize;
|
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:
|
2011-11-11 13:48:14 +00:00
|
|
|
enum Kind {
|
2012-02-09 09:43:37 +00:00
|
|
|
JS_ENTRY,
|
|
|
|
CATCH,
|
|
|
|
FINALLY,
|
|
|
|
LAST_KIND = FINALLY
|
2008-07-03 15:10:15 +00:00
|
|
|
};
|
|
|
|
|
2011-11-11 13:48:14 +00:00
|
|
|
static const int kKindWidth = 2;
|
2012-02-09 09:43:37 +00:00
|
|
|
STATIC_ASSERT(LAST_KIND < (1 << kKindWidth));
|
|
|
|
static const int kIndexWidth = 32 - kKindWidth;
|
2011-11-11 13:48:14 +00:00
|
|
|
class KindField: public BitField<StackHandler::Kind, 0, kKindWidth> {};
|
2012-02-09 09:43:37 +00:00
|
|
|
class IndexField: public BitField<unsigned, kKindWidth, kIndexWidth> {};
|
2011-11-11 13:48:14 +00:00
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
// Get the address of this stack handler.
|
|
|
|
inline Address address() const;
|
|
|
|
|
|
|
|
// Get the next stack handler in the chain.
|
|
|
|
inline StackHandler* next() const;
|
|
|
|
|
|
|
|
// Tells whether the given address is inside this handler.
|
|
|
|
inline bool includes(Address address) const;
|
|
|
|
|
|
|
|
// Garbage collection support.
|
2010-08-30 08:54:43 +00:00
|
|
|
inline void Iterate(ObjectVisitor* v, Code* holder) const;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
|
|
|
// Conversion support.
|
|
|
|
static inline StackHandler* FromAddress(Address address);
|
|
|
|
|
|
|
|
// Testers
|
2012-02-09 09:43:37 +00:00
|
|
|
inline bool is_js_entry() const;
|
|
|
|
inline bool is_catch() const;
|
|
|
|
inline bool is_finally() const;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2013-05-08 08:08:23 +00:00
|
|
|
// Generator support to preserve stack handlers.
|
|
|
|
void Unwind(Isolate* isolate, FixedArray* array, int offset,
|
|
|
|
int previous_handler_offset) const;
|
|
|
|
int Rewind(Isolate* isolate, FixedArray* array, int offset, Address fp);
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
private:
|
|
|
|
// Accessors.
|
2011-11-11 13:48:14 +00:00
|
|
|
inline Kind kind() const;
|
2013-05-08 08:08:23 +00:00
|
|
|
inline unsigned index() const;
|
2008-07-03 15:10:15 +00:00
|
|
|
|
2011-08-12 10:52:49 +00:00
|
|
|
inline Object** context_address() const;
|
2011-11-11 13:48:14 +00:00
|
|
|
inline Object** code_address() const;
|
2013-07-30 23:59:55 +00:00
|
|
|
inline void SetFp(Address slot, Address fp);
|
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,
|
|
|
|
// context and function.
|
|
|
|
// StandardFrame::IterateExpressions assumes that kContextOffset is the last
|
|
|
|
// object pointer.
|
2013-07-23 13:46:10 +00:00
|
|
|
static const int kFixedFrameSize = kPCOnStackSize + kFPOnStackSize +
|
|
|
|
2 * kPointerSize;
|
2013-02-05 08:09:32 +00:00
|
|
|
static const int kExpressionsOffset = -3 * kPointerSize;
|
|
|
|
static const int kMarkerOffset = -2 * kPointerSize;
|
|
|
|
static const int kContextOffset = -1 * kPointerSize;
|
|
|
|
static const int kCallerFPOffset = 0 * kPointerSize;
|
2013-07-23 13:46:10 +00:00
|
|
|
static const int kCallerPCOffset = +1 * kFPOnStackSize;
|
2013-07-30 23:59:55 +00:00
|
|
|
static const int kCallerSPOffset = kCallerPCOffset + 1 * kPCOnStackSize;
|
2013-02-05 08:09:32 +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 {
|
|
|
|
State() : sp(NULL), fp(NULL), pc_address(NULL) { }
|
|
|
|
Address sp;
|
|
|
|
Address fp;
|
|
|
|
Address* pc_address;
|
|
|
|
};
|
|
|
|
|
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; }
|
|
|
|
|
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; }
|
|
|
|
|
|
|
|
// 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
|
|
|
|
|
|
|
// Checks if this frame includes any stack handlers.
|
|
|
|
bool HasHandler() const;
|
|
|
|
|
|
|
|
// 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;
|
|
|
|
static void IteratePc(ObjectVisitor* v, Address* pc_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);
|
|
|
|
|
|
|
|
// Get the top handler from the current stack iterator.
|
|
|
|
inline StackHandler* top_handler() const;
|
|
|
|
|
|
|
|
// 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) {
|
|
|
|
ASSERT(frame->is_entry());
|
|
|
|
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) {
|
|
|
|
ASSERT(frame->is_entry_construct());
|
|
|
|
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) {
|
|
|
|
ASSERT(frame->is_exit());
|
|
|
|
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) {
|
|
|
|
ASSERT(frame->is_standard());
|
|
|
|
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);
|
|
|
|
|
|
|
|
// 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 n'th expression stack element is in a stack
|
|
|
|
// handler or not. Requires traversing all handlers in this frame.
|
|
|
|
bool IsExpressionInsideHandler(int n) const;
|
|
|
|
|
|
|
|
// 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:
|
|
|
|
FrameSummary(Object* receiver,
|
|
|
|
JSFunction* function,
|
|
|
|
Code* code,
|
|
|
|
int offset,
|
|
|
|
bool is_constructor)
|
2013-02-25 14:46:09 +00:00
|
|
|
: receiver_(receiver, function->GetIsolate()),
|
2010-12-07 11:31:57 +00:00
|
|
|
function_(function),
|
|
|
|
code_(code),
|
|
|
|
offset_(offset),
|
|
|
|
is_constructor_(is_constructor) { }
|
|
|
|
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;
|
|
|
|
|
2013-05-08 08:08:23 +00:00
|
|
|
// Generator support to preserve operand stack and stack handlers.
|
|
|
|
void SaveOperandStack(FixedArray* store, int* stack_handler_index) const;
|
|
|
|
void RestoreOperandStack(FixedArray* store, int stack_handler_index);
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
// 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
|
|
|
|
2011-06-29 13:02:00 +00:00
|
|
|
// Returns the levels of inlining for this frame.
|
|
|
|
virtual int GetInlineCount() { return 1; }
|
|
|
|
|
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);
|
|
|
|
|
2013-05-17 08:27:56 +00:00
|
|
|
// Architecture-specific register description.
|
|
|
|
static Register fp_register();
|
|
|
|
static Register context_register();
|
|
|
|
|
2008-07-03 15:10:15 +00:00
|
|
|
static JavaScriptFrame* cast(StackFrame* frame) {
|
|
|
|
ASSERT(frame->is_java_script());
|
|
|
|
return static_cast<JavaScriptFrame*>(frame);
|
|
|
|
}
|
|
|
|
|
2013-02-15 09:27:10 +00:00
|
|
|
static void PrintTop(Isolate* isolate,
|
|
|
|
FILE* file,
|
|
|
|
bool print_args,
|
|
|
|
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;
|
|
|
|
|
2011-06-29 13:02:00 +00:00
|
|
|
virtual int GetInlineCount();
|
|
|
|
|
2010-12-07 11:31:57 +00:00
|
|
|
// 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);
|
|
|
|
|
|
|
|
DeoptimizationInputData* GetDeoptimizationData(int* deopt_index);
|
|
|
|
|
|
|
|
protected:
|
2013-06-27 09:34:31 +00:00
|
|
|
inline explicit OptimizedFrame(StackFrameIteratorBase* iterator);
|
2010-12-07 11:31:57 +00:00
|
|
|
|
|
|
|
private:
|
2012-06-14 14:06:22 +00:00
|
|
|
JSFunction* LiteralAt(FixedArray* literal_array, int literal_id);
|
|
|
|
|
2013-06-27 09:34:31 +00:00
|
|
|
friend class StackFrameIteratorBase;
|
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) {
|
|
|
|
ASSERT(frame->is_arguments_adaptor());
|
|
|
|
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) {
|
|
|
|
ASSERT(frame->is_internal());
|
|
|
|
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-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) {
|
|
|
|
ASSERT(frame->is_construct());
|
|
|
|
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 {
|
|
|
|
ASSERT(!done());
|
|
|
|
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 {
|
|
|
|
ASSERT(!done());
|
|
|
|
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_
|