2 * Copyright (C) 2008 Apple Inc. All rights reserved.
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
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.
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.
29 #ifndef SamplingTool_h
30 #define SamplingTool_h
32 #include <wtf/Assertions.h>
33 #include <wtf/HashMap.h>
34 #include <wtf/Threading.h>
47 struct ScopeSampleRecord
{
48 ScopeSampleRecord(ScopeNode
* scope
)
52 , m_opcodeSampleCount(0)
64 void sample(CodeBlock
*, Instruction
*);
66 RefPtr
<ScopeNode
> m_scope
;
67 CodeBlock
* m_codeBlock
;
69 int m_opcodeSampleCount
;
74 typedef WTF::HashMap
<ScopeNode
*, ScopeSampleRecord
*> ScopeSampleRecordMap
;
78 friend class CallRecord
;
79 friend class HostCallRecord
;
81 #if ENABLE(OPCODE_SAMPLING)
82 class CallRecord
: Noncopyable
{
84 CallRecord(SamplingTool
* samplingTool
)
85 : m_samplingTool(samplingTool
)
86 , m_savedSample(samplingTool
->m_sample
)
87 , m_savedCodeBlock(samplingTool
->m_codeBlock
)
93 m_samplingTool
->m_sample
= m_savedSample
;
94 m_samplingTool
->m_codeBlock
= m_savedCodeBlock
;
98 SamplingTool
* m_samplingTool
;
99 intptr_t m_savedSample
;
100 CodeBlock
* m_savedCodeBlock
;
103 class HostCallRecord
: public CallRecord
{
105 HostCallRecord(SamplingTool
* samplingTool
)
106 : CallRecord(samplingTool
)
108 samplingTool
->m_sample
|= 0x1;
112 class CallRecord
: Noncopyable
{
114 CallRecord(SamplingTool
*)
119 class HostCallRecord
: public CallRecord
{
121 HostCallRecord(SamplingTool
* samplingTool
)
122 : CallRecord(samplingTool
)
128 SamplingTool(Interpreter
* interpreter
)
129 : m_interpreter(interpreter
)
134 , m_opcodeSampleCount(0)
135 , m_scopeSampleMap(new ScopeSampleRecordMap())
137 memset(m_opcodeSamples
, 0, sizeof(m_opcodeSamples
));
138 memset(m_opcodeSamplesInCTIFunctions
, 0, sizeof(m_opcodeSamplesInCTIFunctions
));
143 deleteAllValues(*m_scopeSampleMap
);
146 void start(unsigned hertz
=10000);
148 void dump(ExecState
*);
150 void notifyOfScope(ScopeNode
* scope
);
152 void sample(CodeBlock
* codeBlock
, Instruction
* vPC
)
154 ASSERT(!(reinterpret_cast<intptr_t>(vPC
) & 0x3));
155 m_codeBlock
= codeBlock
;
156 m_sample
= reinterpret_cast<intptr_t>(vPC
);
159 CodeBlock
** codeBlockSlot() { return &m_codeBlock
; }
160 intptr_t* sampleSlot() { return &m_sample
; }
162 void* encodeSample(Instruction
* vPC
, bool inCTIFunction
= false, bool inHostFunction
= false)
164 ASSERT(!(reinterpret_cast<intptr_t>(vPC
) & 0x3));
165 return reinterpret_cast<void*>(reinterpret_cast<intptr_t>(vPC
) | (static_cast<intptr_t>(inCTIFunction
) << 1) | static_cast<intptr_t>(inHostFunction
));
171 Sample(volatile intptr_t sample
, CodeBlock
* volatile codeBlock
)
173 , m_codeBlock(codeBlock
)
177 bool isNull() { return !m_sample
|| !m_codeBlock
; }
178 CodeBlock
* codeBlock() { return m_codeBlock
; }
179 Instruction
* vPC() { return reinterpret_cast<Instruction
*>(m_sample
& ~0x3); }
180 bool inHostFunction() { return m_sample
& 0x1; }
181 bool inCTIFunction() { return m_sample
& 0x2; }
185 CodeBlock
* m_codeBlock
;
188 static void* threadStartFunc(void*);
191 Interpreter
* m_interpreter
;
193 // Sampling thread state.
196 ThreadIdentifier m_samplingThread
;
198 // State tracked by the main thread, used by the sampling thread.
199 CodeBlock
* m_codeBlock
;
202 // Gathered sample data.
203 long long m_sampleCount
;
204 long long m_opcodeSampleCount
;
205 unsigned m_opcodeSamples
[numOpcodeIDs
];
206 unsigned m_opcodeSamplesInCTIFunctions
[numOpcodeIDs
];
208 Mutex m_scopeSampleMapMutex
;
209 OwnPtr
<ScopeSampleRecordMap
> m_scopeSampleMap
;
214 #endif // SamplingTool_h