]> git.saurik.com Git - apple/javascriptcore.git/blame - interpreter/JSStack.h
JavaScriptCore-1218.35.tar.gz
[apple/javascriptcore.git] / interpreter / JSStack.h
CommitLineData
9dae56ea 1/*
f9bf01c6 2 * Copyright (C) 2008, 2009 Apple Inc. All rights reserved.
9dae56ea
A
3 *
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
6 * are met:
7 *
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
13 * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
14 * its contributors may be used to endorse or promote products derived
15 * from this software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
18 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
19 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
20 * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
21 * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
22 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
23 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
24 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 */
28
93a37866
A
29#ifndef JSStack_h
30#define JSStack_h
9dae56ea 31
ba379fdc
A
32#include "ExecutableAllocator.h"
33#include "Register.h"
9dae56ea 34#include <wtf/Noncopyable.h>
14957cd0 35#include <wtf/PageReservation.h>
ba379fdc 36#include <wtf/VMTags.h>
9dae56ea 37
93a37866
A
38#if !defined(NDEBUG) && !defined(ENABLE_DEBUG_JSSTACK)
39#define ENABLE_DEBUG_JSSTACK 1
40#endif
41
9dae56ea
A
42namespace JSC {
43
6fe7ccc8
A
44 class ConservativeRoots;
45 class DFGCodeBlocks;
93a37866
A
46 class ExecState;
47 class JITStubRoutineSet;
48 class VM;
6fe7ccc8 49 class LLIntOffsetsExtractor;
9dae56ea 50
93a37866
A
51 class JSStack {
52 WTF_MAKE_NONCOPYABLE(JSStack);
9dae56ea
A
53 public:
54 enum CallFrameHeaderEntry {
14957cd0
A
55 CallFrameHeaderSize = 6,
56
57 ArgumentCount = -6,
58 CallerFrame = -5,
59 Callee = -4,
60 ScopeChain = -3,
61 ReturnPC = -2, // This is either an Instruction* or a pointer into JIT generated code stored as an Instruction*.
62 CodeBlock = -1,
9dae56ea
A
63 };
64
14957cd0 65 static const size_t defaultCapacity = 512 * 1024;
14957cd0 66 static const size_t commitSize = 16 * 1024;
93a37866 67 // Allow 8k of excess registers before we start trying to reap the stack
ba379fdc 68 static const ptrdiff_t maxExcessCapacity = 8 * 1024;
9dae56ea 69
93a37866
A
70 JSStack(VM&, size_t capacity = defaultCapacity);
71 ~JSStack();
14957cd0
A
72
73 void gatherConservativeRoots(ConservativeRoots&);
93a37866 74 void gatherConservativeRoots(ConservativeRoots&, JITStubRoutineSet&, DFGCodeBlocks&);
9dae56ea 75
6fe7ccc8 76 Register* begin() const { return static_cast<Register*>(m_reservation.base()); }
9dae56ea 77 Register* end() const { return m_end; }
6fe7ccc8 78 size_t size() const { return end() - begin(); }
9dae56ea 79
6fe7ccc8 80 bool grow(Register*);
9dae56ea 81
14957cd0
A
82 static size_t committedByteCount();
83 static void initializeThreading();
84
85 Register* const * addressOfEnd() const
86 {
87 return &m_end;
88 }
9dae56ea 89
93a37866
A
90 Register* getTopOfFrame(CallFrame*);
91 Register* getStartOfFrame(CallFrame*);
92 Register* getTopOfStack();
93
94 CallFrame* pushFrame(CallFrame* callerFrame, class CodeBlock*,
95 JSScope*, int argsCount, JSObject* callee);
96
97 void popFrame(CallFrame*);
98
99 void enableErrorStackReserve();
100 void disableErrorStackReserve();
101
102#if ENABLE(DEBUG_JSSTACK)
103 void installFence(CallFrame*, const char *function = "", int lineNo = 0);
104 void validateFence(CallFrame*, const char *function = "", int lineNo = 0);
105 static const int FenceSize = 4;
106#else // !ENABLE(DEBUG_JSSTACK)
107 void installFence(CallFrame*, const char* = "", int = 0) { }
108 void validateFence(CallFrame*, const char* = "", int = 0) { }
109#endif // !ENABLE(DEBUG_JSSTACK)
110
9dae56ea 111 private:
93a37866
A
112 Register* reservationEnd() const
113 {
114 char* base = static_cast<char*>(m_reservation.base());
115 char* reservationEnd = base + m_reservation.size();
116 return reinterpret_cast_ptr<Register*>(reservationEnd);
117 }
118
119#if ENABLE(DEBUG_JSSTACK)
120 static JSValue generateFenceValue(size_t argIndex);
121 void installTrapsAfterFrame(CallFrame*);
122#else
123 void installTrapsAfterFrame(CallFrame*) { }
124#endif
125
6fe7ccc8 126 bool growSlowCase(Register*);
93a37866 127 void shrink(Register*);
ba379fdc 128 void releaseExcessCapacity();
14957cd0 129 void addToCommittedByteCount(long);
93a37866 130
9dae56ea 131 Register* m_end;
ba379fdc 132 Register* m_commitEnd;
93a37866 133 Register* m_useableEnd;
14957cd0 134 PageReservation m_reservation;
93a37866 135 CallFrame*& m_topCallFrame;
9dae56ea 136
93a37866
A
137 friend class LLIntOffsetsExtractor;
138 };
ba379fdc 139
93a37866 140 inline void JSStack::shrink(Register* newEnd)
ba379fdc
A
141 {
142 if (newEnd >= m_end)
143 return;
144 m_end = newEnd;
6fe7ccc8 145 if (m_end == m_reservation.base() && (m_commitEnd - begin()) >= maxExcessCapacity)
ba379fdc
A
146 releaseExcessCapacity();
147 }
148
93a37866 149 inline bool JSStack::grow(Register* newEnd)
ba379fdc 150 {
6fe7ccc8 151 if (newEnd <= m_end)
ba379fdc 152 return true;
6fe7ccc8 153 return growSlowCase(newEnd);
ba379fdc
A
154 }
155
9dae56ea
A
156} // namespace JSC
157
93a37866 158#endif // JSStack_h