diff options
author | Konstantin Tokarev <annulen@yandex.ru> | 2016-08-25 19:20:41 +0300 |
---|---|---|
committer | Konstantin Tokarev <annulen@yandex.ru> | 2017-02-02 12:30:55 +0000 |
commit | 6882a04fb36642862b11efe514251d32070c3d65 (patch) | |
tree | b7959826000b061fd5ccc7512035c7478742f7b0 /Source/JavaScriptCore/b3/B3MathExtras.cpp | |
parent | ab6df191029eeeb0b0f16f127d553265659f739e (diff) | |
download | qtwebkit-6882a04fb36642862b11efe514251d32070c3d65.tar.gz |
Imported QtWebKit TP3 (git b57bc6801f1876c3220d5a4bfea33d620d477443)
Change-Id: I3b1d8a2808782c9f34d50240000e20cb38d3680f
Reviewed-by: Konstantin Tokarev <annulen@yandex.ru>
Diffstat (limited to 'Source/JavaScriptCore/b3/B3MathExtras.cpp')
-rw-r--r-- | Source/JavaScriptCore/b3/B3MathExtras.cpp | 127 |
1 files changed, 127 insertions, 0 deletions
diff --git a/Source/JavaScriptCore/b3/B3MathExtras.cpp b/Source/JavaScriptCore/b3/B3MathExtras.cpp new file mode 100644 index 000000000..65e3276d0 --- /dev/null +++ b/Source/JavaScriptCore/b3/B3MathExtras.cpp @@ -0,0 +1,127 @@ +/* + * Copyright (C) 2015 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. ``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 + * 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 "B3MathExtras.h" + +#if ENABLE(B3_JIT) + +#include "B3BasicBlockInlines.h" +#include "B3CCallValue.h" +#include "B3Const32Value.h" +#include "B3ConstDoubleValue.h" +#include "B3ConstPtrValue.h" +#include "B3ControlValue.h" +#include "B3UpsilonValue.h" +#include "B3ValueInlines.h" + +namespace JSC { namespace B3 { + +std::pair<BasicBlock*, Value*> powDoubleInt32(Procedure& procedure, BasicBlock* start, Origin origin, Value* x, Value* y) +{ + BasicBlock* functionCallCase = procedure.addBlock(); + BasicBlock* loopPreHeaderCase = procedure.addBlock(); + BasicBlock* loopTestForEvenCase = procedure.addBlock(); + BasicBlock* loopOdd = procedure.addBlock(); + BasicBlock* loopEvenOdd = procedure.addBlock(); + BasicBlock* continuation = procedure.addBlock(); + + Value* shouldGoSlowPath = start->appendNew<Value>(procedure, Above, origin, + y, + start->appendNew<Const32Value>(procedure, origin, 1000)); + start->appendNew<ControlValue>( + procedure, Branch, origin, + shouldGoSlowPath, + FrequentedBlock(functionCallCase), FrequentedBlock(loopPreHeaderCase)); + + // Function call. + Value* yAsDouble = functionCallCase->appendNew<Value>(procedure, IToD, origin, y); + double (*powDouble)(double, double) = pow; + Value* powResult = functionCallCase->appendNew<CCallValue>( + procedure, Double, origin, + functionCallCase->appendNew<ConstPtrValue>(procedure, origin, bitwise_cast<void*>(powDouble)), + x, yAsDouble); + UpsilonValue* powResultUpsilon = functionCallCase->appendNew<UpsilonValue>(procedure, origin, powResult); + functionCallCase->appendNew<ControlValue>(procedure, Jump, origin, FrequentedBlock(continuation)); + + // Loop pre-header. + Value* initialResult = loopPreHeaderCase->appendNew<ConstDoubleValue>(procedure, origin, 1.); + UpsilonValue* initialLoopValue = loopPreHeaderCase->appendNew<UpsilonValue>(procedure, origin, initialResult); + UpsilonValue* initialResultValue = loopPreHeaderCase->appendNew<UpsilonValue>(procedure, origin, initialResult); + UpsilonValue* initialSquaredInput = loopPreHeaderCase->appendNew<UpsilonValue>(procedure, origin, x); + UpsilonValue* initialLoopCounter = loopPreHeaderCase->appendNew<UpsilonValue>(procedure, origin, y); + loopPreHeaderCase->appendNew<ControlValue>(procedure, Jump, origin, FrequentedBlock(loopTestForEvenCase)); + + // Test if what is left of the counter is even. + Value* inLoopCounter = loopTestForEvenCase->appendNew<Value>(procedure, Phi, Int32, origin); + Value* inLoopSquaredInput = loopTestForEvenCase->appendNew<Value>(procedure, Phi, Double, origin); + Value* lastCounterBit = loopTestForEvenCase->appendNew<Value>(procedure, BitAnd, origin, + inLoopCounter, + loopTestForEvenCase->appendNew<Const32Value>(procedure, origin, 1)); + loopTestForEvenCase->appendNew<ControlValue>( + procedure, Branch, origin, + lastCounterBit, + FrequentedBlock(loopOdd), FrequentedBlock(loopEvenOdd)); + + // Counter is odd. + Value* inLoopResult = loopOdd->appendNew<Value>(procedure, Phi, Double, origin); + Value* updatedResult = loopOdd->appendNew<Value>(procedure, Mul, origin, inLoopResult, inLoopSquaredInput); + UpsilonValue* updatedLoopResultUpsilon = loopOdd->appendNew<UpsilonValue>(procedure, origin, updatedResult); + initialLoopValue->setPhi(inLoopResult); + updatedLoopResultUpsilon->setPhi(inLoopResult); + UpsilonValue* updatedLoopResult = loopOdd->appendNew<UpsilonValue>(procedure, origin, updatedResult); + + loopOdd->appendNew<ControlValue>(procedure, Jump, origin, FrequentedBlock(loopEvenOdd)); + + // Even value and following the Odd. + Value* squaredInput = loopEvenOdd->appendNew<Value>(procedure, Mul, origin, inLoopSquaredInput, inLoopSquaredInput); + UpsilonValue* squaredInputUpsilon = loopEvenOdd->appendNew<UpsilonValue>(procedure, origin, squaredInput); + initialSquaredInput->setPhi(inLoopSquaredInput); + squaredInputUpsilon->setPhi(inLoopSquaredInput); + + Value* updatedCounter = loopEvenOdd->appendNew<Value>(procedure, ZShr, origin, + inLoopCounter, + loopEvenOdd->appendNew<Const32Value>(procedure, origin, 1)); + UpsilonValue* updatedCounterUpsilon = loopEvenOdd->appendNew<UpsilonValue>(procedure, origin, updatedCounter); + initialLoopCounter->setPhi(inLoopCounter); + updatedCounterUpsilon->setPhi(inLoopCounter); + + loopEvenOdd->appendNew<ControlValue>( + procedure, Branch, origin, + updatedCounter, + FrequentedBlock(loopTestForEvenCase), FrequentedBlock(continuation)); + + // Inline loop. + Value* finalResultPhi = continuation->appendNew<Value>(procedure, Phi, Double, origin); + powResultUpsilon->setPhi(finalResultPhi); + initialResultValue->setPhi(finalResultPhi); + updatedLoopResult->setPhi(finalResultPhi); + return std::make_pair(continuation, finalResultPhi); +} + +} } // namespace JSC::B3 + +#endif // ENABLE(B3_JIT) + |