Structure* Structure::addPropertyTransition(JSGlobalData& globalData, Structure* structure, PropertyName propertyName, unsigned attributes, JSCell* specificValue, PropertyOffset& offset) { // If we have a specific function, we may have got to this point if there is // already a transition with the correct property name and attributes, but // specialized to a different function. In this case we just want to give up // and despecialize the transition. // In this case we clear the value of specificFunction which will result // in us adding a non-specific transition, and any subsequent lookup in // Structure::addPropertyTransitionToExistingStructure will just use that. if (specificValue && structure->m_transitionTable.contains(propertyName.uid(), attributes)) specificValue = 0; ASSERT(!structure->isDictionary()); ASSERT(structure->isObject()); ASSERT(!Structure::addPropertyTransitionToExistingStructure(structure, propertyName, attributes, specificValue, offset)); if (structure->m_specificFunctionThrashCount == maxSpecificFunctionThrashCount) specificValue = 0; if (structure->transitionCount() > s_maxTransitionLength) { Structure* transition = toCacheableDictionaryTransition(globalData, structure); ASSERT(structure != transition); offset = transition->putSpecificValue(globalData, propertyName, attributes, specificValue); if (transition->outOfLineSize() > transition->outOfLineCapacity()) transition->growOutOfLineCapacity(); return transition; } Structure* transition = create(globalData, structure); transition->m_cachedPrototypeChain.setMayBeNull(globalData, transition, structure->m_cachedPrototypeChain.get()); transition->m_previous.set(globalData, transition, structure); transition->m_nameInPrevious = propertyName.uid(); transition->m_attributesInPrevious = attributes; transition->m_specificValueInPrevious.setMayBeNull(globalData, transition, specificValue); if (structure->m_propertyTable) { if (structure->m_isPinnedPropertyTable) transition->m_propertyTable = structure->m_propertyTable->copy(globalData, transition, structure->m_propertyTable->size() + 1); else transition->m_propertyTable = structure->m_propertyTable.release(); } else { if (structure->m_previous) transition->materializePropertyMap(globalData); else transition->createPropertyMap(); } offset = transition->putSpecificValue(globalData, propertyName, attributes, specificValue); if (transition->outOfLineSize() > transition->outOfLineCapacity()) transition->growOutOfLineCapacity(); transition->m_offset = offset; checkOffset(transition->m_offset, transition->inlineCapacity()); structure->m_transitionTable.add(globalData, transition); return transition; }
void ObjectInitializationScope::verifyPropertiesAreInitialized(JSObject* object) { Butterfly* butterfly = object->butterfly(); Structure* structure = object->structure(m_vm); IndexingType indexingType = structure->indexingType(); unsigned vectorLength = butterfly->vectorLength(); if (UNLIKELY(hasUndecided(indexingType)) || !hasIndexedProperties(indexingType)) { // Nothing to verify. } else if (LIKELY(!hasAnyArrayStorage(indexingType))) { auto data = butterfly->contiguous().data(); for (unsigned i = 0; i < vectorLength; ++i) { if (isScribbledValue(data[i].get())) { dataLogLn("Found scribbled value at i = ", i); ASSERT_NOT_REACHED(); } } } else { ArrayStorage* storage = butterfly->arrayStorage(); for (unsigned i = 0; i < vectorLength; ++i) { if (isScribbledValue(storage->m_vector[i].get())) { dataLogLn("Found scribbled value at i = ", i); ASSERT_NOT_REACHED(); } } } auto isSafeEmptyValueForGCScanning = [] (JSValue value) { #if USE(JSVALUE64) return !value; #else return !value || !JSValue::encode(value); #endif }; for (int64_t i = 0; i < static_cast<int64_t>(structure->outOfLineCapacity()); i++) { // We rely on properties past the last offset be zero for concurrent GC. if (i + firstOutOfLineOffset > structure->lastOffset()) ASSERT(isSafeEmptyValueForGCScanning(butterfly->propertyStorage()[-i - 1].get())); else if (isScribbledValue(butterfly->propertyStorage()[-i - 1].get())) { dataLogLn("Found scribbled property at i = ", -i - 1); ASSERT_NOT_REACHED(); } } }
static void getButterflyDetails(JSObject* obj, void*& butterflyBase, size_t& butterflyCapacityInBytes, CopiedBlock*& butterflyBlock) { Structure* structure = obj->structure(); Butterfly* butterfly = obj->butterfly(); butterflyBase = butterfly->base(structure); butterflyBlock = CopiedSpace::blockFor(butterflyBase); size_t propertyCapacity = structure->outOfLineCapacity(); size_t preCapacity; size_t indexingPayloadSizeInBytes; bool hasIndexingHeader = obj->hasIndexingHeader(); if (UNLIKELY(hasIndexingHeader)) { preCapacity = butterfly->indexingHeader()->preCapacity(structure); indexingPayloadSizeInBytes = butterfly->indexingHeader()->indexingPayloadSizeInBytes(structure); } else { preCapacity = 0; indexingPayloadSizeInBytes = 0; } butterflyCapacityInBytes = Butterfly::totalSize(preCapacity, propertyCapacity, hasIndexingHeader, indexingPayloadSizeInBytes); }