|  | /* | 
|  | * Copyright (C) 2012-2018 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 "DFGArrayMode.h" | 
|  |  | 
|  | #if ENABLE(DFG_JIT) | 
|  |  | 
|  | #include "ArrayPrototype.h" | 
|  | #include "CacheableIdentifierInlines.h" | 
|  | #include "DFGAbstractValue.h" | 
|  | #include "DFGGraph.h" | 
|  | #include "JSCInlines.h" | 
|  |  | 
|  | namespace JSC { namespace DFG { | 
|  |  | 
|  | ArrayMode ArrayMode::fromObserved(const ConcurrentJSLocker& locker, ArrayProfile* profile, Array::Action action, bool makeSafe) | 
|  | { | 
|  | Array::Class nonArray; | 
|  | if (profile->usesOriginalArrayStructures(locker)) | 
|  | nonArray = Array::OriginalNonArray; | 
|  | else | 
|  | nonArray = Array::NonArray; | 
|  |  | 
|  | auto handleContiguousModes = [&] (Array::Type type, ArrayModes observed) { | 
|  | Array::Class isArray; | 
|  | Array::Conversion converts; | 
|  |  | 
|  | RELEASE_ASSERT((observed & (asArrayModesIgnoringTypedArrays(toIndexingShape(type)) | asArrayModesIgnoringTypedArrays(toIndexingShape(type) | ArrayClass) | asArrayModesIgnoringTypedArrays(toIndexingShape(type) | ArrayClass | CopyOnWrite))) == observed); | 
|  |  | 
|  | if (observed & asArrayModesIgnoringTypedArrays(toIndexingShape(type))) { | 
|  | if ((observed & asArrayModesIgnoringTypedArrays(toIndexingShape(type))) == observed) | 
|  | isArray = nonArray; | 
|  | else | 
|  | isArray = Array::PossiblyArray; | 
|  | } else | 
|  | isArray = Array::Array; | 
|  |  | 
|  | if (action == Array::Write && (observed & asArrayModesIgnoringTypedArrays(toIndexingShape(type) | ArrayClass | CopyOnWrite))) | 
|  | converts = Array::Convert; | 
|  | else | 
|  | converts = Array::AsIs; | 
|  |  | 
|  | return ArrayMode(type, isArray, converts, action).withProfile(locker, profile, makeSafe); | 
|  | }; | 
|  |  | 
|  | ArrayModes observed = profile->observedArrayModes(locker); | 
|  | switch (observed) { | 
|  | case 0: | 
|  | return ArrayMode(Array::Unprofiled); | 
|  | case asArrayModesIgnoringTypedArrays(NonArray): | 
|  | if (action == Array::Write && !profile->mayInterceptIndexedAccesses(locker)) | 
|  | return ArrayMode(Array::SelectUsingArguments, nonArray, Array::OutOfBounds, Array::Convert, action); | 
|  | return ArrayMode(Array::SelectUsingPredictions, nonArray, action).withSpeculationFromProfile(locker, profile, makeSafe); | 
|  |  | 
|  | case asArrayModesIgnoringTypedArrays(ArrayWithUndecided): | 
|  | if (action == Array::Write) | 
|  | return ArrayMode(Array::SelectUsingArguments, Array::Array, Array::OutOfBounds, Array::Convert, action); | 
|  | return ArrayMode(Array::Undecided, Array::Array, Array::OutOfBounds, Array::AsIs, action).withProfile(locker, profile, makeSafe); | 
|  |  | 
|  | case asArrayModesIgnoringTypedArrays(NonArray) | asArrayModesIgnoringTypedArrays(ArrayWithUndecided): | 
|  | if (action == Array::Write && !profile->mayInterceptIndexedAccesses(locker)) | 
|  | return ArrayMode(Array::SelectUsingArguments, Array::PossiblyArray, Array::OutOfBounds, Array::Convert, action); | 
|  | return ArrayMode(Array::SelectUsingPredictions, action).withSpeculationFromProfile(locker, profile, makeSafe); | 
|  |  | 
|  | case asArrayModesIgnoringTypedArrays(NonArrayWithInt32): | 
|  | case asArrayModesIgnoringTypedArrays(ArrayWithInt32): | 
|  | case asArrayModesIgnoringTypedArrays(CopyOnWriteArrayWithInt32): | 
|  | case asArrayModesIgnoringTypedArrays(NonArrayWithInt32) | asArrayModesIgnoringTypedArrays(ArrayWithInt32): | 
|  | case asArrayModesIgnoringTypedArrays(NonArrayWithInt32) | asArrayModesIgnoringTypedArrays(CopyOnWriteArrayWithInt32): | 
|  | case asArrayModesIgnoringTypedArrays(ArrayWithInt32) | asArrayModesIgnoringTypedArrays(CopyOnWriteArrayWithInt32): | 
|  | case asArrayModesIgnoringTypedArrays(NonArrayWithInt32) | asArrayModesIgnoringTypedArrays(ArrayWithInt32) | asArrayModesIgnoringTypedArrays(CopyOnWriteArrayWithInt32): | 
|  | return handleContiguousModes(Array::Int32, observed); | 
|  |  | 
|  | case asArrayModesIgnoringTypedArrays(NonArrayWithDouble): | 
|  | case asArrayModesIgnoringTypedArrays(ArrayWithDouble): | 
|  | case asArrayModesIgnoringTypedArrays(CopyOnWriteArrayWithDouble): | 
|  | case asArrayModesIgnoringTypedArrays(NonArrayWithDouble) | asArrayModesIgnoringTypedArrays(ArrayWithDouble): | 
|  | case asArrayModesIgnoringTypedArrays(NonArrayWithDouble) | asArrayModesIgnoringTypedArrays(CopyOnWriteArrayWithDouble): | 
|  | case asArrayModesIgnoringTypedArrays(ArrayWithDouble) | asArrayModesIgnoringTypedArrays(CopyOnWriteArrayWithDouble): | 
|  | case asArrayModesIgnoringTypedArrays(NonArrayWithDouble) | asArrayModesIgnoringTypedArrays(ArrayWithDouble) | asArrayModesIgnoringTypedArrays(CopyOnWriteArrayWithDouble): | 
|  | return handleContiguousModes(Array::Double, observed); | 
|  |  | 
|  | case asArrayModesIgnoringTypedArrays(NonArrayWithContiguous): | 
|  | case asArrayModesIgnoringTypedArrays(ArrayWithContiguous): | 
|  | case asArrayModesIgnoringTypedArrays(CopyOnWriteArrayWithContiguous): | 
|  | case asArrayModesIgnoringTypedArrays(NonArrayWithContiguous) | asArrayModesIgnoringTypedArrays(ArrayWithContiguous): | 
|  | case asArrayModesIgnoringTypedArrays(NonArrayWithContiguous) | asArrayModesIgnoringTypedArrays(CopyOnWriteArrayWithContiguous): | 
|  | case asArrayModesIgnoringTypedArrays(ArrayWithContiguous) | asArrayModesIgnoringTypedArrays(CopyOnWriteArrayWithContiguous): | 
|  | case asArrayModesIgnoringTypedArrays(NonArrayWithContiguous) | asArrayModesIgnoringTypedArrays(ArrayWithContiguous) | asArrayModesIgnoringTypedArrays(CopyOnWriteArrayWithContiguous): | 
|  | return handleContiguousModes(Array::Contiguous, observed); | 
|  |  | 
|  | case asArrayModesIgnoringTypedArrays(NonArrayWithArrayStorage): | 
|  | return ArrayMode(Array::ArrayStorage, nonArray, Array::AsIs, action).withProfile(locker, profile, makeSafe); | 
|  | case asArrayModesIgnoringTypedArrays(NonArrayWithSlowPutArrayStorage): | 
|  | case asArrayModesIgnoringTypedArrays(NonArrayWithArrayStorage) | asArrayModesIgnoringTypedArrays(NonArrayWithSlowPutArrayStorage): | 
|  | return ArrayMode(Array::SlowPutArrayStorage, nonArray, Array::AsIs, action).withProfile(locker, profile, makeSafe); | 
|  | case asArrayModesIgnoringTypedArrays(ArrayWithArrayStorage): | 
|  | return ArrayMode(Array::ArrayStorage, Array::Array, Array::AsIs, action).withProfile(locker, profile, makeSafe); | 
|  | case asArrayModesIgnoringTypedArrays(ArrayWithSlowPutArrayStorage): | 
|  | case asArrayModesIgnoringTypedArrays(ArrayWithArrayStorage) | asArrayModesIgnoringTypedArrays(ArrayWithSlowPutArrayStorage): | 
|  | return ArrayMode(Array::SlowPutArrayStorage, Array::Array, Array::AsIs, action).withProfile(locker, profile, makeSafe); | 
|  | case asArrayModesIgnoringTypedArrays(NonArrayWithArrayStorage) | asArrayModesIgnoringTypedArrays(ArrayWithArrayStorage): | 
|  | return ArrayMode(Array::ArrayStorage, Array::PossiblyArray, Array::AsIs, action).withProfile(locker, profile, makeSafe); | 
|  | case asArrayModesIgnoringTypedArrays(NonArrayWithSlowPutArrayStorage) | asArrayModesIgnoringTypedArrays(ArrayWithSlowPutArrayStorage): | 
|  | case asArrayModesIgnoringTypedArrays(NonArrayWithArrayStorage) | asArrayModesIgnoringTypedArrays(ArrayWithArrayStorage) | asArrayModesIgnoringTypedArrays(NonArrayWithSlowPutArrayStorage) | asArrayModesIgnoringTypedArrays(ArrayWithSlowPutArrayStorage): | 
|  | return ArrayMode(Array::SlowPutArrayStorage, Array::PossiblyArray, Array::AsIs, action).withProfile(locker, profile, makeSafe); | 
|  | case Int8ArrayMode: | 
|  | return ArrayMode(Array::Int8Array, nonArray, Array::AsIs, action).withProfile(locker, profile, makeSafe); | 
|  | case Int16ArrayMode: | 
|  | return ArrayMode(Array::Int16Array, nonArray, Array::AsIs, action).withProfile(locker, profile, makeSafe); | 
|  | case Int32ArrayMode: | 
|  | return ArrayMode(Array::Int32Array, nonArray, Array::AsIs, action).withProfile(locker, profile, makeSafe); | 
|  | case Uint8ArrayMode: | 
|  | return ArrayMode(Array::Uint8Array, nonArray, Array::AsIs, action).withProfile(locker, profile, makeSafe); | 
|  | case Uint8ClampedArrayMode: | 
|  | return ArrayMode(Array::Uint8ClampedArray, nonArray, Array::AsIs, action).withProfile(locker, profile, makeSafe); | 
|  | case Uint16ArrayMode: | 
|  | return ArrayMode(Array::Uint16Array, nonArray, Array::AsIs, action).withProfile(locker, profile, makeSafe); | 
|  | case Uint32ArrayMode: | 
|  | return ArrayMode(Array::Uint32Array, nonArray, Array::AsIs, action).withProfile(locker, profile, makeSafe); | 
|  | case Float32ArrayMode: | 
|  | return ArrayMode(Array::Float32Array, nonArray, Array::AsIs, action).withProfile(locker, profile, makeSafe); | 
|  | case Float64ArrayMode: | 
|  | return ArrayMode(Array::Float64Array, nonArray, Array::AsIs, action).withProfile(locker, profile, makeSafe); | 
|  |  | 
|  | default: | 
|  | // If we have seen multiple TypedArray types, or a TypedArray and non-typed array, it doesn't make sense to try to convert the object since you can't convert typed arrays. | 
|  | if (observed & ALL_TYPED_ARRAY_MODES) | 
|  | return ArrayMode(Array::Generic, nonArray, Array::AsIs, action).withProfile(locker, profile, makeSafe); | 
|  |  | 
|  | if ((observed & asArrayModesIgnoringTypedArrays(NonArray)) && profile->mayInterceptIndexedAccesses(locker)) | 
|  | return ArrayMode(Array::SelectUsingPredictions).withSpeculationFromProfile(locker, profile, makeSafe); | 
|  |  | 
|  | Array::Type type; | 
|  | Array::Class arrayClass; | 
|  |  | 
|  | if (shouldUseSlowPutArrayStorage(observed)) | 
|  | type = Array::SlowPutArrayStorage; | 
|  | else if (shouldUseFastArrayStorage(observed)) | 
|  | type = Array::ArrayStorage; | 
|  | else if (shouldUseContiguous(observed)) | 
|  | type = Array::Contiguous; | 
|  | else if (shouldUseDouble(observed)) | 
|  | type = Array::Double; | 
|  | else if (shouldUseInt32(observed)) | 
|  | type = Array::Int32; | 
|  | else | 
|  | type = Array::SelectUsingArguments; | 
|  |  | 
|  | if (hasSeenArray(observed) && hasSeenNonArray(observed)) | 
|  | arrayClass = Array::PossiblyArray; | 
|  | else if (hasSeenArray(observed)) | 
|  | arrayClass = Array::Array; | 
|  | else if (hasSeenNonArray(observed)) | 
|  | arrayClass = nonArray; | 
|  | else | 
|  | arrayClass = Array::PossiblyArray; | 
|  |  | 
|  | return ArrayMode(type, arrayClass, Array::Convert, action).withProfile(locker, profile, makeSafe); | 
|  | } | 
|  | } | 
|  |  | 
|  | static bool canBecomeGetArrayLength(Graph& graph, Node* node) | 
|  | { | 
|  | if (node->op() == GetArrayLength) | 
|  | return true; | 
|  | if (node->op() != GetById) | 
|  | return false; | 
|  | auto uid = node->cacheableIdentifier().uid(); | 
|  | return uid == graph.m_vm.propertyNames->length.impl(); | 
|  | } | 
|  |  | 
|  | ArrayMode ArrayMode::refine( | 
|  | Graph& graph, Node* node, | 
|  | SpeculatedType base, SpeculatedType index, SpeculatedType value) const | 
|  | { | 
|  | if (!base || !index) { | 
|  | // It can be that we had a legitimate arrayMode but no incoming predictions. That'll | 
|  | // happen if we inlined code based on, say, a global variable watchpoint, but later | 
|  | // realized that the callsite could not have possibly executed. It may be worthwhile | 
|  | // to fix that, but for now I'm leaving it as-is. | 
|  | return ArrayMode(Array::ForceExit, action()); | 
|  | } | 
|  |  | 
|  | if (!isInt32Speculation(index)) | 
|  | return ArrayMode(Array::Generic, action()); | 
|  |  | 
|  | // If we had exited because of an exotic object behavior, then don't try to specialize. | 
|  | if (graph.hasExitSite(node->origin.semantic, ExoticObjectMode)) | 
|  | return ArrayMode(Array::Generic, action()); | 
|  |  | 
|  | // Note: our profiling currently doesn't give us good information in case we have | 
|  | // an unlikely control flow path that sets the base to a non-cell value. Value | 
|  | // profiling and prediction propagation will probably tell us that the value is | 
|  | // either a cell or not, but that doesn't tell us which is more likely: that this | 
|  | // is an array access on a cell (what we want and can optimize) or that the user is | 
|  | // doing a crazy by-val access on a primitive (we can't easily optimize this and | 
|  | // don't want to). So, for now, we assume that if the base is not a cell according | 
|  | // to value profiling, but the array profile tells us something else, then we | 
|  | // should just trust the array profile. | 
|  |  | 
|  | auto typedArrayResult = [&] (ArrayMode result) -> ArrayMode { | 
|  | if (node->op() == PutByValDirect) { | 
|  | // This is semantically identical to defineOwnProperty({configurable: true, writable:true, enumerable:true}), | 
|  | // which we can't model as a simple store to the typed array since typed array indexed properties | 
|  | // are non-configurable. | 
|  | return ArrayMode(Array::Generic, action()); | 
|  | } | 
|  | return result; | 
|  | }; | 
|  |  | 
|  | switch (type()) { | 
|  | case Array::SelectUsingArguments: | 
|  | if (!value) | 
|  | return withType(Array::ForceExit); | 
|  | if (isInt32Speculation(value)) | 
|  | return withTypeAndConversion(Array::Int32, Array::Convert); | 
|  | if (isFullNumberSpeculation(value)) | 
|  | return withTypeAndConversion(Array::Double, Array::Convert); | 
|  | return withTypeAndConversion(Array::Contiguous, Array::Convert); | 
|  | case Array::Undecided: { | 
|  | // As long as we have a JSArray getting its length shouldn't require any sane chainness. | 
|  | if (canBecomeGetArrayLength(graph, node) && isJSArray()) | 
|  | return *this; | 
|  |  | 
|  | // If we have an OriginalArray and the JSArray prototype chain is sane, | 
|  | // any indexed access always return undefined. We have a fast path for that. | 
|  | JSGlobalObject* globalObject = graph.globalObjectFor(node->origin.semantic); | 
|  | Structure* arrayPrototypeStructure = globalObject->arrayPrototype()->structure(graph.m_vm); | 
|  | Structure* objectPrototypeStructure = globalObject->objectPrototype()->structure(graph.m_vm); | 
|  | if (node->op() == GetByVal | 
|  | && isJSArrayWithOriginalStructure() | 
|  | && !graph.hasExitSite(node->origin.semantic, OutOfBounds) | 
|  | && arrayPrototypeStructure->transitionWatchpointSetIsStillValid() | 
|  | && objectPrototypeStructure->transitionWatchpointSetIsStillValid() | 
|  | && globalObject->arrayPrototypeChainIsSane()) { | 
|  | graph.registerAndWatchStructureTransition(arrayPrototypeStructure); | 
|  | graph.registerAndWatchStructureTransition(objectPrototypeStructure); | 
|  | if (globalObject->arrayPrototypeChainIsSane()) | 
|  | return withSpeculation(Array::InBoundsSaneChain); | 
|  | } | 
|  | return ArrayMode(Array::Generic, action()); | 
|  | } | 
|  | case Array::Int32: | 
|  | if (!value || isInt32Speculation(value)) | 
|  | return *this; | 
|  | if (isFullNumberSpeculation(value)) | 
|  | return withTypeAndConversion(Array::Double, Array::Convert); | 
|  | return withTypeAndConversion(Array::Contiguous, Array::Convert); | 
|  |  | 
|  | case Array::Double: | 
|  | if (!value || isFullNumberSpeculation(value)) | 
|  | return *this; | 
|  | return withTypeAndConversion(Array::Contiguous, Array::Convert); | 
|  |  | 
|  | case Array::Contiguous: | 
|  | return *this; | 
|  |  | 
|  | case Array::Int8Array: | 
|  | case Array::Int16Array: | 
|  | case Array::Int32Array: | 
|  | case Array::Uint8Array: | 
|  | case Array::Uint8ClampedArray: | 
|  | case Array::Uint16Array: | 
|  | case Array::Uint32Array: | 
|  | case Array::Float32Array: | 
|  | case Array::Float64Array: | 
|  | if (node->op() == PutByVal) { | 
|  | if (graph.hasExitSite(node->origin.semantic, OutOfBounds) || !isInBounds()) | 
|  | return typedArrayResult(withSpeculation(Array::OutOfBounds)); | 
|  | } | 
|  | return typedArrayResult(withSpeculation(Array::InBounds)); | 
|  | case Array::Unprofiled: | 
|  | case Array::SelectUsingPredictions: { | 
|  | base &= ~SpecOther; | 
|  |  | 
|  | if (isStringSpeculation(base)) | 
|  | return withType(Array::String); | 
|  |  | 
|  | if (isDirectArgumentsSpeculation(base) || isScopedArgumentsSpeculation(base)) { | 
|  | // Handle out-of-bounds accesses as generic accesses. | 
|  | Array::Type type = isDirectArgumentsSpeculation(base) ? Array::DirectArguments : Array::ScopedArguments; | 
|  | if (graph.hasExitSite(node->origin.semantic, OutOfBounds) || !isInBounds()) { | 
|  | // FIXME: Support OOB access for ScopedArguments. | 
|  | // https://bugs.webkit.org/show_bug.cgi?id=179596 | 
|  | if (type == Array::DirectArguments) | 
|  | return ArrayMode(type, Array::NonArray, Array::OutOfBounds, Array::AsIs, action()); | 
|  | return ArrayMode(Array::Generic, action()); | 
|  | } | 
|  | return withType(type); | 
|  | } | 
|  |  | 
|  | ArrayMode result; | 
|  | switch (node->op()) { | 
|  | case PutByVal: | 
|  | if (graph.hasExitSite(node->origin.semantic, OutOfBounds) || !isInBounds()) | 
|  | result = withSpeculation(Array::OutOfBounds); | 
|  | else | 
|  | result = withSpeculation(Array::InBounds); | 
|  | break; | 
|  |  | 
|  | default: | 
|  | result = withSpeculation(Array::InBounds); | 
|  | break; | 
|  | } | 
|  |  | 
|  | if (isInt8ArraySpeculation(base)) | 
|  | return typedArrayResult(result.withType(Array::Int8Array)); | 
|  |  | 
|  | if (isInt16ArraySpeculation(base)) | 
|  | return typedArrayResult(result.withType(Array::Int16Array)); | 
|  |  | 
|  | if (isInt32ArraySpeculation(base)) | 
|  | return typedArrayResult(result.withType(Array::Int32Array)); | 
|  |  | 
|  | if (isUint8ArraySpeculation(base)) | 
|  | return typedArrayResult(result.withType(Array::Uint8Array)); | 
|  |  | 
|  | if (isUint8ClampedArraySpeculation(base)) | 
|  | return typedArrayResult(result.withType(Array::Uint8ClampedArray)); | 
|  |  | 
|  | if (isUint16ArraySpeculation(base)) | 
|  | return typedArrayResult(result.withType(Array::Uint16Array)); | 
|  |  | 
|  | if (isUint32ArraySpeculation(base)) | 
|  | return typedArrayResult(result.withType(Array::Uint32Array)); | 
|  |  | 
|  | if (isFloat32ArraySpeculation(base)) | 
|  | return typedArrayResult(result.withType(Array::Float32Array)); | 
|  |  | 
|  | if (isFloat64ArraySpeculation(base)) | 
|  | return typedArrayResult(result.withType(Array::Float64Array)); | 
|  |  | 
|  | if (type() == Array::Unprofiled) | 
|  | return ArrayMode(Array::ForceExit, action()); | 
|  | return ArrayMode(Array::Generic, action()); | 
|  | } | 
|  |  | 
|  | default: | 
|  | return *this; | 
|  | } | 
|  | } | 
|  |  | 
|  | Structure* ArrayMode::originalArrayStructure(Graph& graph, const CodeOrigin& codeOrigin) const | 
|  | { | 
|  | JSGlobalObject* globalObject = graph.globalObjectFor(codeOrigin); | 
|  |  | 
|  | switch (arrayClass()) { | 
|  | case Array::OriginalCopyOnWriteArray: { | 
|  | if (conversion() == Array::AsIs) { | 
|  | switch (type()) { | 
|  | case Array::Int32: | 
|  | return globalObject->originalArrayStructureForIndexingType(CopyOnWriteArrayWithInt32); | 
|  | case Array::Double: | 
|  | return globalObject->originalArrayStructureForIndexingType(CopyOnWriteArrayWithDouble); | 
|  | case Array::Contiguous: | 
|  | return globalObject->originalArrayStructureForIndexingType(CopyOnWriteArrayWithContiguous); | 
|  | default: | 
|  | CRASH(); | 
|  | return nullptr; | 
|  | } | 
|  | } | 
|  | FALLTHROUGH; | 
|  | } | 
|  |  | 
|  | case Array::OriginalArray: { | 
|  | switch (type()) { | 
|  | case Array::Int32: | 
|  | return globalObject->originalArrayStructureForIndexingType(ArrayWithInt32); | 
|  | case Array::Double: | 
|  | return globalObject->originalArrayStructureForIndexingType(ArrayWithDouble); | 
|  | case Array::Contiguous: | 
|  | return globalObject->originalArrayStructureForIndexingType(ArrayWithContiguous); | 
|  | case Array::Undecided: | 
|  | return globalObject->originalArrayStructureForIndexingType(ArrayWithUndecided); | 
|  | case Array::ArrayStorage: | 
|  | return globalObject->originalArrayStructureForIndexingType(ArrayWithArrayStorage); | 
|  | default: | 
|  | CRASH(); | 
|  | return nullptr; | 
|  | } | 
|  | } | 
|  |  | 
|  | case Array::OriginalNonArray: { | 
|  | TypedArrayType type = typedArrayType(); | 
|  | if (type == NotTypedArray) | 
|  | return nullptr; | 
|  |  | 
|  | return globalObject->typedArrayStructureConcurrently(type); | 
|  | } | 
|  |  | 
|  | default: | 
|  | return nullptr; | 
|  | } | 
|  | } | 
|  |  | 
|  | Structure* ArrayMode::originalArrayStructure(Graph& graph, Node* node) const | 
|  | { | 
|  | return originalArrayStructure(graph, node->origin.semantic); | 
|  | } | 
|  |  | 
|  | bool ArrayMode::alreadyChecked(Graph& graph, Node* node, const AbstractValue& value, IndexingType shape) const | 
|  | { | 
|  | ASSERT(isSpecific()); | 
|  |  | 
|  | IndexingType indexingModeMask = IsArray | IndexingShapeMask; | 
|  | if (action() == Array::Write) | 
|  | indexingModeMask |= CopyOnWrite; | 
|  |  | 
|  | switch (arrayClass()) { | 
|  | case Array::Array: { | 
|  | if (arrayModesAlreadyChecked(value.m_arrayModes, asArrayModesIgnoringTypedArrays(shape | IsArray))) | 
|  | return true; | 
|  | if (!value.m_structure.isFinite()) | 
|  | return false; | 
|  | for (unsigned i = value.m_structure.size(); i--;) { | 
|  | RegisteredStructure structure = value.m_structure[i]; | 
|  | if ((structure->indexingMode() & indexingModeMask) != (shape | IsArray)) | 
|  | return false; | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | // Array::OriginalNonArray can be shown when the value is a TypedArray with original structure. | 
|  | // But here, we already filtered TypedArrays. So, just handle it like a NonArray. | 
|  | case Array::OriginalNonArray: | 
|  | case Array::NonArray: { | 
|  | if (arrayModesAlreadyChecked(value.m_arrayModes, asArrayModesIgnoringTypedArrays(shape))) | 
|  | return true; | 
|  | if (!value.m_structure.isFinite()) | 
|  | return false; | 
|  | for (unsigned i = value.m_structure.size(); i--;) { | 
|  | RegisteredStructure structure = value.m_structure[i]; | 
|  | if ((structure->indexingMode() & indexingModeMask) != shape) | 
|  | return false; | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | case Array::PossiblyArray: { | 
|  | if (arrayModesAlreadyChecked(value.m_arrayModes, asArrayModesIgnoringTypedArrays(shape) | asArrayModesIgnoringTypedArrays(shape | IsArray))) | 
|  | return true; | 
|  | if (!value.m_structure.isFinite()) | 
|  | return false; | 
|  | for (unsigned i = value.m_structure.size(); i--;) { | 
|  | RegisteredStructure structure = value.m_structure[i]; | 
|  | if ((structure->indexingMode() & (indexingModeMask & ~IsArray)) != shape) | 
|  | return false; | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | // If ArrayMode is Array::OriginalCopyOnWriteArray or Array::OriginalArray, CheckArray is never emitted. Instead, we always emit CheckStructure. | 
|  | // So, we should perform the same check to the CheckStructure here. | 
|  | case Array::OriginalArray: | 
|  | case Array::OriginalCopyOnWriteArray: { | 
|  | if (!value.m_structure.isFinite()) | 
|  | return false; | 
|  | Structure* originalStructure = originalArrayStructure(graph, node); | 
|  | if (value.m_structure.size() != 1) | 
|  | return false; | 
|  | return value.m_structure.onlyStructure().get() == originalStructure; | 
|  | } | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | bool ArrayMode::alreadyChecked(Graph& graph, Node* node, const AbstractValue& value) const | 
|  | { | 
|  | switch (type()) { | 
|  | case Array::Generic: | 
|  | return true; | 
|  |  | 
|  | case Array::ForceExit: | 
|  | return false; | 
|  |  | 
|  | case Array::String: | 
|  | return speculationChecked(value.m_type, SpecString); | 
|  |  | 
|  | case Array::Int32: | 
|  | return alreadyChecked(graph, node, value, Int32Shape); | 
|  |  | 
|  | case Array::Double: | 
|  | return alreadyChecked(graph, node, value, DoubleShape); | 
|  |  | 
|  | case Array::Contiguous: | 
|  | return alreadyChecked(graph, node, value, ContiguousShape); | 
|  |  | 
|  | case Array::ArrayStorage: | 
|  | return alreadyChecked(graph, node, value, ArrayStorageShape); | 
|  |  | 
|  | case Array::Undecided: | 
|  | return alreadyChecked(graph, node, value, UndecidedShape); | 
|  |  | 
|  | case Array::SlowPutArrayStorage: | 
|  | switch (arrayClass()) { | 
|  | case Array::OriginalArray: | 
|  | case Array::OriginalCopyOnWriteArray: { | 
|  | CRASH(); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | case Array::Array: { | 
|  | if (arrayModesAlreadyChecked(value.m_arrayModes, asArrayModesIgnoringTypedArrays(ArrayWithArrayStorage) | asArrayModesIgnoringTypedArrays(ArrayWithSlowPutArrayStorage))) | 
|  | return true; | 
|  | if (value.m_structure.isTop()) | 
|  | return false; | 
|  | for (unsigned i = value.m_structure.size(); i--;) { | 
|  | RegisteredStructure structure = value.m_structure[i]; | 
|  | if (!hasAnyArrayStorage(structure->indexingType())) | 
|  | return false; | 
|  | if (!(structure->indexingType() & IsArray)) | 
|  | return false; | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | // Array::OriginalNonArray can be shown when the value is a TypedArray with original structure. | 
|  | // But here, we already filtered TypedArrays. So, just handle it like a NonArray. | 
|  | case Array::NonArray: | 
|  | case Array::OriginalNonArray: { | 
|  | if (arrayModesAlreadyChecked(value.m_arrayModes, asArrayModesIgnoringTypedArrays(NonArrayWithArrayStorage) | asArrayModesIgnoringTypedArrays(NonArrayWithSlowPutArrayStorage))) | 
|  | return true; | 
|  | if (value.m_structure.isTop()) | 
|  | return false; | 
|  | for (unsigned i = value.m_structure.size(); i--;) { | 
|  | RegisteredStructure structure = value.m_structure[i]; | 
|  | if (!hasAnyArrayStorage(structure->indexingType())) | 
|  | return false; | 
|  | if (structure->indexingType() & IsArray) | 
|  | return false; | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | case Array::PossiblyArray: { | 
|  | if (arrayModesAlreadyChecked(value.m_arrayModes, asArrayModesIgnoringTypedArrays(NonArrayWithArrayStorage) | asArrayModesIgnoringTypedArrays(ArrayWithArrayStorage) | asArrayModesIgnoringTypedArrays(NonArrayWithSlowPutArrayStorage) | asArrayModesIgnoringTypedArrays(ArrayWithSlowPutArrayStorage))) | 
|  | return true; | 
|  | if (value.m_structure.isTop()) | 
|  | return false; | 
|  | for (unsigned i = value.m_structure.size(); i--;) { | 
|  | RegisteredStructure structure = value.m_structure[i]; | 
|  | if (!hasAnyArrayStorage(structure->indexingType())) | 
|  | return false; | 
|  | } | 
|  | return true; | 
|  | } | 
|  | default: | 
|  | CRASH(); | 
|  | } | 
|  |  | 
|  | case Array::DirectArguments: | 
|  | return speculationChecked(value.m_type, SpecDirectArguments); | 
|  |  | 
|  | case Array::ScopedArguments: | 
|  | return speculationChecked(value.m_type, SpecScopedArguments); | 
|  |  | 
|  | case Array::Int8Array: | 
|  | return speculationChecked(value.m_type, SpecInt8Array); | 
|  |  | 
|  | case Array::Int16Array: | 
|  | return speculationChecked(value.m_type, SpecInt16Array); | 
|  |  | 
|  | case Array::Int32Array: | 
|  | return speculationChecked(value.m_type, SpecInt32Array); | 
|  |  | 
|  | case Array::Uint8Array: | 
|  | return speculationChecked(value.m_type, SpecUint8Array); | 
|  |  | 
|  | case Array::Uint8ClampedArray: | 
|  | return speculationChecked(value.m_type, SpecUint8ClampedArray); | 
|  |  | 
|  | case Array::Uint16Array: | 
|  | return speculationChecked(value.m_type, SpecUint16Array); | 
|  |  | 
|  | case Array::Uint32Array: | 
|  | return speculationChecked(value.m_type, SpecUint32Array); | 
|  |  | 
|  | case Array::Float32Array: | 
|  | return speculationChecked(value.m_type, SpecFloat32Array); | 
|  |  | 
|  | case Array::Float64Array: | 
|  | return speculationChecked(value.m_type, SpecFloat64Array); | 
|  |  | 
|  | case Array::AnyTypedArray: | 
|  | return speculationChecked(value.m_type, SpecTypedArrayView); | 
|  |  | 
|  | case Array::SelectUsingPredictions: | 
|  | case Array::Unprofiled: | 
|  | case Array::SelectUsingArguments: | 
|  | break; | 
|  | } | 
|  |  | 
|  | CRASH(); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | const char* arrayActionToString(Array::Action action) | 
|  | { | 
|  | switch (action) { | 
|  | case Array::Read: | 
|  | return "Read"; | 
|  | case Array::Write: | 
|  | return "Write"; | 
|  | default: | 
|  | return "Unknown!"; | 
|  | } | 
|  | } | 
|  |  | 
|  | const char* arrayTypeToString(Array::Type type) | 
|  | { | 
|  | switch (type) { | 
|  | case Array::SelectUsingPredictions: | 
|  | return "SelectUsingPredictions"; | 
|  | case Array::SelectUsingArguments: | 
|  | return "SelectUsingArguments"; | 
|  | case Array::Unprofiled: | 
|  | return "Unprofiled"; | 
|  | case Array::Generic: | 
|  | return "Generic"; | 
|  | case Array::ForceExit: | 
|  | return "ForceExit"; | 
|  | case Array::String: | 
|  | return "String"; | 
|  | case Array::Undecided: | 
|  | return "Undecided"; | 
|  | case Array::Int32: | 
|  | return "Int32"; | 
|  | case Array::Double: | 
|  | return "Double"; | 
|  | case Array::Contiguous: | 
|  | return "Contiguous"; | 
|  | case Array::ArrayStorage: | 
|  | return "ArrayStorage"; | 
|  | case Array::SlowPutArrayStorage: | 
|  | return "SlowPutArrayStorage"; | 
|  | case Array::DirectArguments: | 
|  | return "DirectArguments"; | 
|  | case Array::ScopedArguments: | 
|  | return "ScopedArguments"; | 
|  | case Array::Int8Array: | 
|  | return "Int8Array"; | 
|  | case Array::Int16Array: | 
|  | return "Int16Array"; | 
|  | case Array::Int32Array: | 
|  | return "Int32Array"; | 
|  | case Array::Uint8Array: | 
|  | return "Uint8Array"; | 
|  | case Array::Uint8ClampedArray: | 
|  | return "Uint8ClampedArray"; | 
|  | case Array::Uint16Array: | 
|  | return "Uint16Array"; | 
|  | case Array::Uint32Array: | 
|  | return "Uint32Array"; | 
|  | case Array::Float32Array: | 
|  | return "Float32Array"; | 
|  | case Array::Float64Array: | 
|  | return "Float64Array"; | 
|  | case Array::AnyTypedArray: | 
|  | return "AnyTypedArray"; | 
|  | default: | 
|  | // Better to return something then it is to crash. Remember, this method | 
|  | // is being called from our main diagnostic tool, the IR dumper. It's like | 
|  | // a stack trace. So if we get here then probably something has already | 
|  | // gone wrong. | 
|  | return "Unknown!"; | 
|  | } | 
|  | } | 
|  |  | 
|  | const char* arrayClassToString(Array::Class arrayClass) | 
|  | { | 
|  | switch (arrayClass) { | 
|  | case Array::Array: | 
|  | return "Array"; | 
|  | case Array::OriginalArray: | 
|  | return "OriginalArray"; | 
|  | case Array::OriginalCopyOnWriteArray: | 
|  | return "OriginalCopyOnWriteArray"; | 
|  | case Array::NonArray: | 
|  | return "NonArray"; | 
|  | case Array::OriginalNonArray: | 
|  | return "OriginalNonArray"; | 
|  | case Array::PossiblyArray: | 
|  | return "PossiblyArray"; | 
|  | default: | 
|  | return "Unknown!"; | 
|  | } | 
|  | } | 
|  |  | 
|  | const char* arraySpeculationToString(Array::Speculation speculation) | 
|  | { | 
|  | switch (speculation) { | 
|  | case Array::InBoundsSaneChain: | 
|  | return "InBoundsSaneChain"; | 
|  | case Array::InBounds: | 
|  | return "InBounds"; | 
|  | case Array::ToHole: | 
|  | return "ToHole"; | 
|  | case Array::OutOfBounds: | 
|  | return "OutOfBounds"; | 
|  | case Array::OutOfBoundsSaneChain: | 
|  | return "OutOfBoundsSaneChain"; | 
|  | default: | 
|  | return "Unknown!"; | 
|  | } | 
|  | } | 
|  |  | 
|  | const char* arrayConversionToString(Array::Conversion conversion) | 
|  | { | 
|  | switch (conversion) { | 
|  | case Array::AsIs: | 
|  | return "AsIs"; | 
|  | case Array::Convert: | 
|  | return "Convert"; | 
|  | default: | 
|  | return "Unknown!"; | 
|  | } | 
|  | } | 
|  |  | 
|  | IndexingType toIndexingShape(Array::Type type) | 
|  | { | 
|  | switch (type) { | 
|  | case Array::Int32: | 
|  | return Int32Shape; | 
|  | case Array::Double: | 
|  | return DoubleShape; | 
|  | case Array::Contiguous: | 
|  | return ContiguousShape; | 
|  | case Array::Undecided: | 
|  | return UndecidedShape; | 
|  | case Array::ArrayStorage: | 
|  | return ArrayStorageShape; | 
|  | case Array::SlowPutArrayStorage: | 
|  | return SlowPutArrayStorageShape; | 
|  | default: | 
|  | return NoIndexingShape; | 
|  | } | 
|  | } | 
|  |  | 
|  | TypedArrayType toTypedArrayType(Array::Type type) | 
|  | { | 
|  | switch (type) { | 
|  | case Array::Int8Array: | 
|  | return TypeInt8; | 
|  | case Array::Int16Array: | 
|  | return TypeInt16; | 
|  | case Array::Int32Array: | 
|  | return TypeInt32; | 
|  | case Array::Uint8Array: | 
|  | return TypeUint8; | 
|  | case Array::Uint8ClampedArray: | 
|  | return TypeUint8Clamped; | 
|  | case Array::Uint16Array: | 
|  | return TypeUint16; | 
|  | case Array::Uint32Array: | 
|  | return TypeUint32; | 
|  | case Array::Float32Array: | 
|  | return TypeFloat32; | 
|  | case Array::Float64Array: | 
|  | return TypeFloat64; | 
|  | case Array::AnyTypedArray: | 
|  | RELEASE_ASSERT_NOT_REACHED(); | 
|  | return NotTypedArray; | 
|  | default: | 
|  | return NotTypedArray; | 
|  | } | 
|  | } | 
|  |  | 
|  | Array::Type toArrayType(TypedArrayType type) | 
|  | { | 
|  | switch (type) { | 
|  | case TypeInt8: | 
|  | return Array::Int8Array; | 
|  | case TypeInt16: | 
|  | return Array::Int16Array; | 
|  | case TypeInt32: | 
|  | return Array::Int32Array; | 
|  | case TypeUint8: | 
|  | return Array::Uint8Array; | 
|  | case TypeUint8Clamped: | 
|  | return Array::Uint8ClampedArray; | 
|  | case TypeUint16: | 
|  | return Array::Uint16Array; | 
|  | case TypeUint32: | 
|  | return Array::Uint32Array; | 
|  | case TypeFloat32: | 
|  | return Array::Float32Array; | 
|  | case TypeFloat64: | 
|  | return Array::Float64Array; | 
|  | default: | 
|  | return Array::Generic; | 
|  | } | 
|  | } | 
|  |  | 
|  | Array::Type refineTypedArrayType(Array::Type oldType, TypedArrayType newType) | 
|  | { | 
|  | if (oldType == Array::Generic) | 
|  | return oldType; | 
|  | Array::Type newArrayType = toArrayType(newType); | 
|  | if (newArrayType == Array::Generic) | 
|  | return newArrayType; | 
|  |  | 
|  | if (oldType != newArrayType) | 
|  | return Array::AnyTypedArray; | 
|  | return oldType; | 
|  | } | 
|  |  | 
|  | bool permitsBoundsCheckLowering(Array::Type type) | 
|  | { | 
|  | switch (type) { | 
|  | case Array::Int32: | 
|  | case Array::Double: | 
|  | case Array::Contiguous: | 
|  | case Array::ArrayStorage: | 
|  | case Array::SlowPutArrayStorage: | 
|  | case Array::Int8Array: | 
|  | case Array::Int16Array: | 
|  | case Array::Int32Array: | 
|  | case Array::Uint8Array: | 
|  | case Array::Uint8ClampedArray: | 
|  | case Array::Uint16Array: | 
|  | case Array::Uint32Array: | 
|  | case Array::Float32Array: | 
|  | case Array::Float64Array: | 
|  | case Array::AnyTypedArray: | 
|  | return true; | 
|  | default: | 
|  | // These don't allow for bounds check lowering either because the bounds | 
|  | // check isn't a speculation (like String, sort of) or because the type implies an impure access. | 
|  | return false; | 
|  | } | 
|  | } | 
|  |  | 
|  | bool ArrayMode::permitsBoundsCheckLowering() const | 
|  | { | 
|  | return DFG::permitsBoundsCheckLowering(type()) && isInBounds(); | 
|  | } | 
|  |  | 
|  | void ArrayMode::dump(PrintStream& out) const | 
|  | { | 
|  | out.print(type(), "+", arrayClass(), "+", speculation(), "+", conversion(), "+", action()); | 
|  | } | 
|  |  | 
|  | } } // namespace JSC::DFG | 
|  |  | 
|  | namespace WTF { | 
|  |  | 
|  | void printInternal(PrintStream& out, JSC::DFG::Array::Action action) | 
|  | { | 
|  | out.print(JSC::DFG::arrayActionToString(action)); | 
|  | } | 
|  |  | 
|  | void printInternal(PrintStream& out, JSC::DFG::Array::Type type) | 
|  | { | 
|  | out.print(JSC::DFG::arrayTypeToString(type)); | 
|  | } | 
|  |  | 
|  | void printInternal(PrintStream& out, JSC::DFG::Array::Class arrayClass) | 
|  | { | 
|  | out.print(JSC::DFG::arrayClassToString(arrayClass)); | 
|  | } | 
|  |  | 
|  | void printInternal(PrintStream& out, JSC::DFG::Array::Speculation speculation) | 
|  | { | 
|  | out.print(JSC::DFG::arraySpeculationToString(speculation)); | 
|  | } | 
|  |  | 
|  | void printInternal(PrintStream& out, JSC::DFG::Array::Conversion conversion) | 
|  | { | 
|  | out.print(JSC::DFG::arrayConversionToString(conversion)); | 
|  | } | 
|  |  | 
|  | } // namespace WTF | 
|  |  | 
|  | #endif // ENABLE(DFG_JIT) | 
|  |  |