blob: eb0e4f142b763fcec208347f8f4c79cc95fedaa5 [file] [log] [blame]
/*
* Copyright (C) 2021 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 "JITPlan.h"
#if ENABLE(JIT)
#include "AbstractSlotVisitor.h"
#include "CodeBlock.h"
#include "HeapInlines.h"
#include "JSCellInlines.h"
#include "VMInlines.h"
#include <wtf/CompilationThread.h>
namespace JSC {
extern Seconds totalBaselineCompileTime;
extern Seconds totalDFGCompileTime;
extern Seconds totalFTLCompileTime;
extern Seconds totalFTLDFGCompileTime;
extern Seconds totalFTLB3CompileTime;
JITPlan::JITPlan(JITCompilationMode mode, CodeBlock* codeBlock)
: m_mode(mode)
, m_vm(&codeBlock->vm())
, m_codeBlock(codeBlock)
{
}
void JITPlan::cancel()
{
RELEASE_ASSERT(m_stage != JITPlanStage::Canceled);
ASSERT(m_vm);
m_stage = JITPlanStage::Canceled;
m_vm = nullptr;
m_codeBlock = nullptr;
}
void JITPlan::notifyCompiling()
{
m_stage = JITPlanStage::Compiling;
}
void JITPlan::notifyReady()
{
m_stage = JITPlanStage::Ready;
}
auto JITPlan::tier() const -> Tier
{
switch (m_mode) {
case JITCompilationMode::InvalidCompilation:
RELEASE_ASSERT_NOT_REACHED();
return Tier::Baseline;
case JITCompilationMode::Baseline:
return Tier::Baseline;
case JITCompilationMode::DFG:
case JITCompilationMode::UnlinkedDFG:
return Tier::DFG;
case JITCompilationMode::FTL:
case JITCompilationMode::FTLForOSREntry:
return Tier::FTL;
}
RELEASE_ASSERT_NOT_REACHED();
}
JITCompilationKey JITPlan::key()
{
JSCell* codeBlock;
if (m_mode == JITCompilationMode::Baseline)
codeBlock = m_codeBlock->unlinkedCodeBlock();
else
codeBlock = m_codeBlock->baselineAlternative();
return JITCompilationKey(codeBlock, m_mode);
}
bool JITPlan::isKnownToBeLiveAfterGC()
{
if (m_stage == JITPlanStage::Canceled)
return false;
if (!m_vm->heap.isMarked(m_codeBlock->ownerExecutable()))
return false;
return true;
}
bool JITPlan::isKnownToBeLiveDuringGC(AbstractSlotVisitor& visitor)
{
if (m_stage == JITPlanStage::Canceled)
return false;
if (!visitor.isMarked(m_codeBlock->ownerExecutable()))
return false;
return true;
}
bool JITPlan::iterateCodeBlocksForGC(AbstractSlotVisitor& visitor, const Function<void(CodeBlock*)>& func)
{
if (!isKnownToBeLiveDuringGC(visitor))
return false;
// Compilation writes lots of values to a CodeBlock without performing
// an explicit barrier. So, we need to be pessimistic and assume that
// all our CodeBlocks must be visited during GC.
func(m_codeBlock);
return true;
}
bool JITPlan::checkLivenessAndVisitChildren(AbstractSlotVisitor& visitor)
{
if (!isKnownToBeLiveDuringGC(visitor))
return false;
visitor.appendUnbarriered(m_codeBlock);
return true;
}
bool JITPlan::computeCompileTimes() const
{
return reportCompileTimes()
|| Options::reportTotalCompileTimes()
|| (m_vm && m_vm->m_perBytecodeProfiler);
}
bool JITPlan::reportCompileTimes() const
{
return Options::reportCompileTimes()
|| (Options::reportBaselineCompileTimes() && m_mode == JITCompilationMode::Baseline)
|| (Options::reportDFGCompileTimes() && isDFG())
|| (Options::reportFTLCompileTimes() && isFTL());
}
void JITPlan::compileInThread(JITWorklistThread* thread)
{
m_thread = thread;
MonotonicTime before;
CString codeBlockName;
if (UNLIKELY(computeCompileTimes()))
before = MonotonicTime::now();
if (UNLIKELY(reportCompileTimes()))
codeBlockName = toCString(*m_codeBlock);
CompilationScope compilationScope;
#if ENABLE(DFG_JIT)
if (DFG::logCompilationChanges(m_mode) || Options::logPhaseTimes())
dataLog("DFG(Plan) compiling ", *m_codeBlock, " with ", m_mode, ", instructions size = ", m_codeBlock->instructionsSize(), "\n");
#endif // ENABLE(DFG_JIT)
CompilationPath path = compileInThreadImpl();
RELEASE_ASSERT((path == CancelPath) == (m_stage == JITPlanStage::Canceled));
MonotonicTime after;
if (UNLIKELY(computeCompileTimes())) {
after = MonotonicTime::now();
if (Options::reportTotalCompileTimes()) {
if (isFTL()) {
totalFTLCompileTime += after - before;
totalFTLDFGCompileTime += m_timeBeforeFTL - before;
totalFTLB3CompileTime += after - m_timeBeforeFTL;
} else if (mode() == JITCompilationMode::Baseline)
totalBaselineCompileTime += after - before;
else
totalDFGCompileTime += after - before;
}
}
const char* pathName = nullptr;
switch (path) {
case FailPath:
pathName = "N/A (fail)";
break;
case BaselinePath:
pathName = "Baseline";
break;
case DFGPath:
pathName = "DFG";
break;
case FTLPath:
pathName = "FTL";
break;
case CancelPath:
pathName = "Canceled";
break;
default:
RELEASE_ASSERT_NOT_REACHED();
break;
}
if (m_codeBlock) { // m_codeBlock will be null if the compilation was cancelled.
if (path == FTLPath)
CODEBLOCK_LOG_EVENT(m_codeBlock, "ftlCompile", ("took ", (after - before).milliseconds(), " ms (DFG: ", (m_timeBeforeFTL - before).milliseconds(), ", B3: ", (after - m_timeBeforeFTL).milliseconds(), ") with ", pathName));
else
CODEBLOCK_LOG_EVENT(m_codeBlock, "dfgCompile", ("took ", (after - before).milliseconds(), " ms with ", pathName));
}
if (UNLIKELY(reportCompileTimes())) {
dataLog("Optimized ", codeBlockName, " using ", m_mode, " with ", pathName, " into ", codeSize(), " bytes in ", (after - before).milliseconds(), " ms");
if (path == FTLPath)
dataLog(" (DFG: ", (m_timeBeforeFTL - before).milliseconds(), ", B3: ", (after - m_timeBeforeFTL).milliseconds(), ")");
dataLog(".\n");
}
}
} // namespace JSC
#endif // ENABLE(DFG_JIT)