]>
git.saurik.com Git - apple/javascriptcore.git/blob - dfg/DFGCapabilities.cpp
2 * Copyright (C) 2011 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
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
13 * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
14 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
17 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 #include "DFGCapabilities.h"
29 #include "CodeBlock.h"
30 #include "DFGCommon.h"
31 #include "Interpreter.h"
33 namespace JSC
{ namespace DFG
{
36 bool mightCompileEval(CodeBlock
* codeBlock
)
38 return codeBlock
->instructionCount() <= Options::maximumOptimizationCandidateInstructionCount();
40 bool mightCompileProgram(CodeBlock
* codeBlock
)
42 return codeBlock
->instructionCount() <= Options::maximumOptimizationCandidateInstructionCount();
44 bool mightCompileFunctionForCall(CodeBlock
* codeBlock
)
46 return codeBlock
->instructionCount() <= Options::maximumOptimizationCandidateInstructionCount();
48 bool mightCompileFunctionForConstruct(CodeBlock
* codeBlock
)
50 return codeBlock
->instructionCount() <= Options::maximumOptimizationCandidateInstructionCount();
53 bool mightInlineFunctionForCall(CodeBlock
* codeBlock
)
55 return codeBlock
->instructionCount() <= Options::maximumFunctionForCallInlineCandidateInstructionCount()
56 && !codeBlock
->ownerExecutable()->needsActivation();
58 bool mightInlineFunctionForClosureCall(CodeBlock
* codeBlock
)
60 return codeBlock
->instructionCount() <= Options::maximumFunctionForClosureCallInlineCandidateInstructionCount()
61 && !codeBlock
->ownerExecutable()->needsActivation();
63 bool mightInlineFunctionForConstruct(CodeBlock
* codeBlock
)
65 return codeBlock
->instructionCount() <= Options::maximumFunctionForConstructInlineCandidateInstructionCount()
66 && !codeBlock
->ownerExecutable()->needsActivation();
69 static inline void debugFail(CodeBlock
* codeBlock
, OpcodeID opcodeID
, bool result
)
71 ASSERT_UNUSED(result
, !result
);
72 #if DFG_ENABLE(DEBUG_VERBOSE)
73 dataLogF("Cannot handle code block %p because of opcode %s.\n", codeBlock
, opcodeNames
[opcodeID
]);
75 UNUSED_PARAM(codeBlock
);
76 UNUSED_PARAM(opcodeID
);
81 static inline void debugFail(CodeBlock
* codeBlock
, OpcodeID opcodeID
, CapabilityLevel result
)
83 ASSERT(result
!= CanCompile
);
84 #if DFG_ENABLE(DEBUG_VERBOSE)
85 if (result
== CannotCompile
)
86 dataLogF("Cannot handle code block %p because of opcode %s.\n", codeBlock
, opcodeNames
[opcodeID
]);
88 dataLogF("Cannot compile code block %p because of opcode %s, but inlining might be possible.\n", codeBlock
, opcodeNames
[opcodeID
]);
91 UNUSED_PARAM(codeBlock
);
92 UNUSED_PARAM(opcodeID
);
97 template<typename ReturnType
, ReturnType (*canHandleOpcode
)(OpcodeID
, CodeBlock
*, Instruction
*)>
98 ReturnType
canHandleOpcodes(CodeBlock
* codeBlock
, ReturnType initialValue
)
100 Interpreter
* interpreter
= codeBlock
->vm()->interpreter
;
101 Instruction
* instructionsBegin
= codeBlock
->instructions().begin();
102 unsigned instructionCount
= codeBlock
->instructions().size();
103 ReturnType result
= initialValue
;
105 for (unsigned bytecodeOffset
= 0; bytecodeOffset
< instructionCount
; ) {
106 switch (interpreter
->getOpcodeID(instructionsBegin
[bytecodeOffset
].u
.opcode
)) {
107 #define DEFINE_OP(opcode, length) \
109 ReturnType current = canHandleOpcode( \
110 opcode, codeBlock, instructionsBegin + bytecodeOffset); \
111 if (current < result) { \
113 debugFail(codeBlock, opcode, current); \
115 bytecodeOffset += length; \
118 FOR_EACH_OPCODE_ID(DEFINE_OP
)
121 RELEASE_ASSERT_NOT_REACHED();
129 CapabilityLevel
canCompileOpcodes(CodeBlock
* codeBlock
)
131 if (!MacroAssembler::supportsFloatingPoint())
132 return CannotCompile
;
133 return canHandleOpcodes
<CapabilityLevel
, canCompileOpcode
>(codeBlock
, CanCompile
);
136 bool canInlineOpcodes(CodeBlock
* codeBlock
)
138 return canHandleOpcodes
<bool, canInlineOpcode
>(codeBlock
, true);
143 } } // namespace JSC::DFG