diff options
author | Lorry Tar Creator <lorry-tar-importer@lorry> | 2017-06-27 06:07:23 +0000 |
---|---|---|
committer | Lorry Tar Creator <lorry-tar-importer@lorry> | 2017-06-27 06:07:23 +0000 |
commit | 1bf1084f2b10c3b47fd1a588d85d21ed0eb41d0c (patch) | |
tree | 46dcd36c86e7fbc6e5df36deb463b33e9967a6f7 /Source/JavaScriptCore/bytecode/BytecodeRewriter.cpp | |
parent | 32761a6cee1d0dee366b885b7b9c777e67885688 (diff) | |
download | WebKitGtk-tarball-master.tar.gz |
webkitgtk-2.16.5HEADwebkitgtk-2.16.5master
Diffstat (limited to 'Source/JavaScriptCore/bytecode/BytecodeRewriter.cpp')
-rw-r--r-- | Source/JavaScriptCore/bytecode/BytecodeRewriter.cpp | 116 |
1 files changed, 116 insertions, 0 deletions
diff --git a/Source/JavaScriptCore/bytecode/BytecodeRewriter.cpp b/Source/JavaScriptCore/bytecode/BytecodeRewriter.cpp new file mode 100644 index 000000000..6dadb6e74 --- /dev/null +++ b/Source/JavaScriptCore/bytecode/BytecodeRewriter.cpp @@ -0,0 +1,116 @@ +/* + * Copyright (C) 2016 Yusuke Suzuki <utatane.tea@gmail.com> + * Copyright (C) 2016 Apple Inc. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * 1. Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * 2. 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. + * + * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS 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 APPLE INC. OR ITS 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. + */ + +#include "config.h" +#include "BytecodeRewriter.h" + +#include "HeapInlines.h" +#include "PreciseJumpTargetsInlines.h" +#include <wtf/BubbleSort.h> + +namespace JSC { + +void BytecodeRewriter::applyModification() +{ + for (size_t insertionIndex = m_insertions.size(); insertionIndex--;) { + Insertion& insertion = m_insertions[insertionIndex]; + if (insertion.type == Insertion::Type::Remove) + m_graph.instructions().remove(insertion.index.bytecodeOffset, insertion.length()); + else { + if (insertion.includeBranch == IncludeBranch::Yes) { + int finalOffset = insertion.index.bytecodeOffset + calculateDifference(m_insertions.begin(), m_insertions.begin() + insertionIndex); + adjustJumpTargetsInFragment(finalOffset, insertion); + } + m_graph.instructions().insertVector(insertion.index.bytecodeOffset, insertion.instructions); + } + } + m_insertions.clear(); +} + +void BytecodeRewriter::execute() +{ + WTF::bubbleSort(m_insertions.begin(), m_insertions.end(), [] (const Insertion& lhs, const Insertion& rhs) { + return lhs.index < rhs.index; + }); + + UnlinkedCodeBlock* codeBlock = m_graph.codeBlock(); + codeBlock->applyModification(*this); +} + +void BytecodeRewriter::adjustJumpTargetsInFragment(unsigned finalOffset, Insertion& insertion) +{ + auto& fragment = insertion.instructions; + UnlinkedInstruction* instructionsBegin = fragment.data(); + for (unsigned fragmentOffset = 0, fragmentCount = fragment.size(); fragmentOffset < fragmentCount;) { + UnlinkedInstruction& instruction = fragment[fragmentOffset]; + OpcodeID opcodeID = instruction.u.opcode; + if (isBranch(opcodeID)) { + unsigned bytecodeOffset = finalOffset + fragmentOffset; + UnlinkedCodeBlock* codeBlock = m_graph.codeBlock(); + extractStoredJumpTargetsForBytecodeOffset(codeBlock, codeBlock->vm()->interpreter, instructionsBegin, fragmentOffset, [&](int32_t& label) { + int absoluteOffset = adjustAbsoluteOffset(label); + label = absoluteOffset - static_cast<int>(bytecodeOffset); + }); + } + fragmentOffset += opcodeLength(opcodeID); + } +} + +void BytecodeRewriter::insertImpl(InsertionPoint insertionPoint, IncludeBranch includeBranch, Vector<UnlinkedInstruction>&& fragment) +{ + ASSERT(insertionPoint.position == Position::Before || insertionPoint.position == Position::After); + m_insertions.append(Insertion { + insertionPoint, + Insertion::Type::Insert, + includeBranch, + 0, + WTFMove(fragment) + }); +} + +int BytecodeRewriter::adjustJumpTarget(InsertionPoint startPoint, InsertionPoint jumpTargetPoint) +{ + if (startPoint < jumpTargetPoint) { + int jumpTarget = jumpTargetPoint.bytecodeOffset; + auto start = std::lower_bound(m_insertions.begin(), m_insertions.end(), startPoint, [&] (const Insertion& insertion, InsertionPoint startPoint) { + return insertion.index < startPoint; + }); + if (start != m_insertions.end()) { + auto end = std::lower_bound(m_insertions.begin(), m_insertions.end(), jumpTargetPoint, [&] (const Insertion& insertion, InsertionPoint jumpTargetPoint) { + return insertion.index < jumpTargetPoint; + }); + jumpTarget += calculateDifference(start, end); + } + return jumpTarget - startPoint.bytecodeOffset; + } + + if (startPoint == jumpTargetPoint) + return 0; + + return -adjustJumpTarget(jumpTargetPoint, startPoint); +} + +} // namespace JSC |