/* * Copyright (C) 2012, 2013 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 "ArrayProfile.h" #include "CodeBlock.h" #include "JSCInlines.h" #include #include #include namespace JSC { void dumpArrayModes(PrintStream& out, ArrayModes arrayModes) { if (!arrayModes) { out.print(""); return; } if (arrayModes == ALL_ARRAY_MODES) { out.print("TOP"); return; } CommaPrinter comma("|"); if (arrayModes & asArrayModes(NonArray)) out.print(comma, "NonArray"); if (arrayModes & asArrayModes(NonArrayWithInt32)) out.print(comma, "NonArrayWithInt32"); if (arrayModes & asArrayModes(NonArrayWithDouble)) out.print(comma, "NonArrayWithDouble"); if (arrayModes & asArrayModes(NonArrayWithContiguous)) out.print(comma, "NonArrayWithContiguous"); if (arrayModes & asArrayModes(NonArrayWithArrayStorage)) out.print(comma, "NonArrayWithArrayStorage"); if (arrayModes & asArrayModes(NonArrayWithSlowPutArrayStorage)) out.print(comma, "NonArrayWithSlowPutArrayStorage"); if (arrayModes & asArrayModes(ArrayClass)) out.print(comma, "ArrayClass"); if (arrayModes & asArrayModes(ArrayWithUndecided)) out.print(comma, "ArrayWithUndecided"); if (arrayModes & asArrayModes(ArrayWithInt32)) out.print(comma, "ArrayWithInt32"); if (arrayModes & asArrayModes(ArrayWithDouble)) out.print(comma, "ArrayWithDouble"); if (arrayModes & asArrayModes(ArrayWithContiguous)) out.print(comma, "ArrayWithContiguous"); if (arrayModes & asArrayModes(ArrayWithArrayStorage)) out.print(comma, "ArrayWithArrayStorage"); if (arrayModes & asArrayModes(ArrayWithSlowPutArrayStorage)) out.print(comma, "ArrayWithSlowPutArrayStorage"); if (arrayModes & Int8ArrayMode) out.print(comma, "Int8ArrayMode"); if (arrayModes & Int16ArrayMode) out.print(comma, "Int16ArrayMode"); if (arrayModes & Int32ArrayMode) out.print(comma, "Int32ArrayMode"); if (arrayModes & Uint8ArrayMode) out.print(comma, "Uint8ArrayMode"); if (arrayModes & Uint8ClampedArrayMode) out.print(comma, "Uint8ClampedArrayMode"); if (arrayModes & Uint16ArrayMode) out.print(comma, "Uint16ArrayMode"); if (arrayModes & Uint32ArrayMode) out.print(comma, "Uint32ArrayMode"); if (arrayModes & Float32ArrayMode) out.print(comma, "Float32ArrayMode"); if (arrayModes & Float64ArrayMode) out.print(comma, "Float64ArrayMode"); } void ArrayProfile::computeUpdatedPrediction(const ConcurrentJITLocker& locker, CodeBlock* codeBlock) { if (!m_lastSeenStructureID) return; Structure* lastSeenStructure = codeBlock->heap()->structureIDTable().get(m_lastSeenStructureID); computeUpdatedPrediction(locker, codeBlock, lastSeenStructure); m_lastSeenStructureID = 0; } void ArrayProfile::computeUpdatedPrediction(const ConcurrentJITLocker&, CodeBlock* codeBlock, Structure* lastSeenStructure) { m_observedArrayModes |= arrayModeFromStructure(lastSeenStructure); if (!m_didPerformFirstRunPruning && hasTwoOrMoreBitsSet(m_observedArrayModes)) { m_observedArrayModes = arrayModeFromStructure(lastSeenStructure); m_didPerformFirstRunPruning = true; } m_mayInterceptIndexedAccesses |= lastSeenStructure->typeInfo().interceptsGetOwnPropertySlotByIndexEvenWhenLengthIsNotZero(); JSGlobalObject* globalObject = codeBlock->globalObject(); if (!globalObject->isOriginalArrayStructure(lastSeenStructure) && !globalObject->isOriginalTypedArrayStructure(lastSeenStructure)) m_usesOriginalArrayStructures = false; } CString ArrayProfile::briefDescription(const ConcurrentJITLocker& locker, CodeBlock* codeBlock) { computeUpdatedPrediction(locker, codeBlock); return briefDescriptionWithoutUpdating(locker); } CString ArrayProfile::briefDescriptionWithoutUpdating(const ConcurrentJITLocker&) { StringPrintStream out; bool hasPrinted = false; if (m_observedArrayModes) { if (hasPrinted) out.print(", "); out.print(ArrayModesDump(m_observedArrayModes)); hasPrinted = true; } if (m_mayStoreToHole) { if (hasPrinted) out.print(", "); out.print("Hole"); hasPrinted = true; } if (m_outOfBounds) { if (hasPrinted) out.print(", "); out.print("OutOfBounds"); hasPrinted = true; } if (m_mayInterceptIndexedAccesses) { if (hasPrinted) out.print(", "); out.print("Intercept"); hasPrinted = true; } if (m_usesOriginalArrayStructures) { if (hasPrinted) out.print(", "); out.print("Original"); hasPrinted = true; } UNUSED_PARAM(hasPrinted); return out.toCString(); } } // namespace JSC