]>
git.saurik.com Git - apple/javascriptcore.git/blob - bytecode/PreciseJumpTargets.cpp
2 * Copyright (C) 2013 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 "PreciseJumpTargets.h"
31 static void addSimpleSwitchTargets(SimpleJumpTable
& jumpTable
, unsigned bytecodeOffset
, Vector
<unsigned, 32>& out
)
33 for (unsigned i
= jumpTable
.branchOffsets
.size(); i
--;)
34 out
.append(bytecodeOffset
+ jumpTable
.branchOffsets
[i
]);
37 void computePreciseJumpTargets(CodeBlock
* codeBlock
, Vector
<unsigned, 32>& out
)
39 ASSERT(out
.isEmpty());
41 // We will derive a superset of the jump targets that the code block thinks it has.
42 // So, if the code block claims there are none, then we are done.
43 if (!codeBlock
->numberOfJumpTargets())
46 for (unsigned i
= codeBlock
->numberOfExceptionHandlers(); i
--;)
47 out
.append(codeBlock
->exceptionHandler(i
).target
);
49 Interpreter
* interpreter
= codeBlock
->vm()->interpreter
;
50 Instruction
* instructionsBegin
= codeBlock
->instructions().begin();
51 unsigned instructionCount
= codeBlock
->instructions().size();
52 for (unsigned bytecodeOffset
= 0; bytecodeOffset
< instructionCount
;) {
53 OpcodeID opcodeID
= interpreter
->getOpcodeID(instructionsBegin
[bytecodeOffset
].u
.opcode
);
54 Instruction
* current
= instructionsBegin
+ bytecodeOffset
;
57 out
.append(bytecodeOffset
+ current
[1].u
.operand
);
63 out
.append(bytecodeOffset
+ current
[2].u
.operand
);
74 out
.append(bytecodeOffset
+ current
[3].u
.operand
);
77 addSimpleSwitchTargets(codeBlock
->immediateSwitchJumpTable(current
[1].u
.operand
), bytecodeOffset
, out
);
78 out
.append(bytecodeOffset
+ current
[2].u
.operand
);
81 addSimpleSwitchTargets(codeBlock
->characterSwitchJumpTable(current
[1].u
.operand
), bytecodeOffset
, out
);
82 out
.append(bytecodeOffset
+ current
[2].u
.operand
);
84 case op_switch_string
: {
85 StringJumpTable
& table
= codeBlock
->stringSwitchJumpTable(current
[1].u
.operand
);
86 StringJumpTable::StringOffsetTable::iterator iter
= table
.offsetTable
.begin();
87 StringJumpTable::StringOffsetTable::iterator end
= table
.offsetTable
.end();
88 for (; iter
!= end
; ++iter
)
89 out
.append(bytecodeOffset
+ iter
->value
.branchOffset
);
90 out
.append(bytecodeOffset
+ current
[2].u
.operand
);
94 out
.append(bytecodeOffset
+ current
[5].u
.operand
);
97 out
.append(bytecodeOffset
+ current
[6].u
.operand
);
99 case op_check_has_instance
:
100 out
.append(bytecodeOffset
+ current
[4].u
.operand
);
103 out
.append(bytecodeOffset
);
108 bytecodeOffset
+= opcodeLengths
[opcodeID
];
111 std::sort(out
.begin(), out
.end());
113 // We will have duplicates, and we must remove them.
114 unsigned toIndex
= 0;
115 unsigned fromIndex
= 0;
116 unsigned lastValue
= UINT_MAX
;
117 while (fromIndex
< out
.size()) {
118 unsigned value
= out
[fromIndex
++];
119 if (value
== lastValue
)
121 out
[toIndex
++] = value
;