Revert "[runtime] Refactor TransitionsAccessor"

This reverts commit c927ada76c.

Reason for revert: GC stress failures: https://ci.chromium.org/ui/p/v8/builders/ci/V8%20Linux%20-%20gc%20stress/37276/overview

Original change's description:
> [runtime] Refactor TransitionsAccessor
>
> Problems:
> - The class uses a bare Map field, but some methods can trigger GC
> causing it to have a potential dangling pointer in case of map
> compaction.
> - Some methods invalidate the object state and should not be used again.
> - Complicate logic with a no_gc and a gc aware constructors. Some
> methods can only be called if the object is constructed with a
> particular constructor (e.g, Insert and PutPrototypeTransition).
>
> Note: Most usages of this class is done by constructing an object and
> calling a single method:
> `TransitionAccessor(...).Method(...)`
> So we can easily change them to a static method.
>
> This CL:
> 1. Adds DISALLOW_GARBAGE_COLLECTION to the class.
> 2. Makes methods that can trigger GC static.
> 3. Creates static helper functions that wrap the class in a different
> scope, since TransitionsAccessor now forces the scope to disallow gc.
> 4. Removes now unnecessary "Reload" logic.
>
> Bug: chromium:1295133, v8:12578
> Change-Id: I85484e7235fbd5e69894e26f5e1c491c6f69635e
> Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3450416
> Reviewed-by: Dominik Inführ <dinfuehr@chromium.org>
> Reviewed-by: Toon Verwaest <verwaest@chromium.org>
> Commit-Queue: Victor Gomes <victorgomes@chromium.org>
> Cr-Commit-Position: refs/heads/main@{#79051}

Bug: chromium:1295133, v8:12578
Change-Id: Ia567cdcae73bc7fdfaf08b62eeeb899d6a933e21
No-Presubmit: true
No-Tree-Checks: true
No-Try: true
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3456682
Auto-Submit: Deepti Gandluri <gdeepti@chromium.org>
Commit-Queue: Rubber Stamper <rubber-stamper@appspot.gserviceaccount.com>
Bot-Commit: Rubber Stamper <rubber-stamper@appspot.gserviceaccount.com>
Owners-Override: Deepti Gandluri <gdeepti@chromium.org>
Cr-Commit-Position: refs/heads/main@{#79052}
This commit is contained in:
Deepti Gandluri 2022-02-11 16:41:49 +00:00 committed by V8 LUCI CQ
parent c927ada76c
commit 7c60201194
21 changed files with 341 additions and 403 deletions

View File

@ -1119,9 +1119,10 @@ PropertyAccessInfo AccessInfoFactory::LookupTransition(
PropertyAttributes attrs) const {
// Check if the {map} has a data transition with the given {name}.
Map transition =
TransitionsAccessor(isolate(), *map.object(), true)
TransitionsAccessor(isolate(), map.object(), true)
.SearchTransition(*name.object(), PropertyKind::kData, attrs);
if (transition.is_null()) return Invalid();
base::Optional<MapRef> maybe_transition_map =
TryMakeRef(broker(), transition);
if (!maybe_transition_map.has_value()) return Invalid();

View File

@ -563,7 +563,8 @@ int InstanceSizeWithMinSlack(JSHeapBroker* broker, MapRef map) {
DCHECK(map.object()->GetBackPointer().IsUndefined(broker->isolate()));
static constexpr bool kConcurrentAccess = true;
TransitionsAccessor(broker->isolate(), *map.object(), kConcurrentAccess)
TransitionsAccessor(broker->isolate(), *map.object(), &no_gc,
kConcurrentAccess)
.TraverseTransitionTree([&](Map m) {
maps.push_back(broker->CanonicalPersistentHandle(m));
});

View File

@ -520,9 +520,11 @@ void Map::MapVerify(Isolate* isolate) {
}
}
SLOW_DCHECK(instance_descriptors(isolate).IsSortedNoDuplicates());
SLOW_DCHECK(TransitionsAccessor(isolate, *this).IsSortedNoDuplicates());
DisallowGarbageCollection no_gc;
SLOW_DCHECK(
TransitionsAccessor(isolate, *this).IsConsistentWithBackPointers());
TransitionsAccessor(isolate, *this, &no_gc).IsSortedNoDuplicates());
SLOW_DCHECK(TransitionsAccessor(isolate, *this, &no_gc)
.IsConsistentWithBackPointers());
// Only JSFunction maps have has_prototype_slot() bit set and constructible
// JSFunction objects must have prototype slot.
CHECK_IMPLIES(has_prototype_slot(),

View File

@ -2592,7 +2592,8 @@ void Map::MapPrint(std::ostream& os) {
// the isolate to iterate over the transitions.
if (!IsReadOnlyHeapObject(*this)) {
Isolate* isolate = GetIsolateFromWritableObject(*this);
TransitionsAccessor transitions(isolate, *this);
DisallowGarbageCollection no_gc;
TransitionsAccessor transitions(isolate, *this, &no_gc);
int nof_transitions = transitions.NumberOfTransitions();
if (nof_transitions > 0) {
os << "\n - transitions #" << nof_transitions << ": ";
@ -2779,13 +2780,14 @@ void TransitionsAccessor::PrintTransitionTree(
descriptors.PrintDescriptorDetails(os, descriptor,
PropertyDetails::kForTransitions);
}
TransitionsAccessor transitions(isolate_, target);
TransitionsAccessor transitions(isolate_, target, no_gc);
transitions.PrintTransitionTree(os, level + 1, no_gc);
}
}
void JSObject::PrintTransitions(std::ostream& os) {
TransitionsAccessor ta(GetIsolate(), map());
DisallowGarbageCollection no_gc;
TransitionsAccessor ta(GetIsolate(), map(), &no_gc);
if (ta.NumberOfTransitions() == 0) return;
os << "\n - transitions";
ta.PrintTransitions(os);
@ -2889,8 +2891,9 @@ V8_EXPORT_PRIVATE extern void _v8_internal_Print_TransitionTree(void* object) {
printf("Please provide a valid Map\n");
} else {
#if defined(DEBUG) || defined(OBJECT_PRINT)
i::DisallowGarbageCollection no_gc;
i::Map map = i::Map::unchecked_cast(o);
i::TransitionsAccessor transitions(i::Isolate::Current(), map);
i::TransitionsAccessor transitions(i::Isolate::Current(), map, &no_gc);
transitions.PrintTransitionTree();
#endif
}

View File

@ -2535,7 +2535,7 @@ void MarkCompactCollector::ClearPotentialSimpleMapTransition(Map dead_target) {
Map parent = Map::cast(potential_parent);
DisallowGarbageCollection no_gc_obviously;
if (non_atomic_marking_state()->IsBlackOrGrey(parent) &&
TransitionsAccessor(isolate(), parent)
TransitionsAccessor(isolate(), parent, &no_gc_obviously)
.HasSimpleTransitionTo(dead_target)) {
ClearPotentialSimpleMapTransition(parent, dead_target);
}

View File

@ -486,13 +486,14 @@ Handle<Object> JsonParser<Char>::BuildJsonObject(
descriptor_index)),
isolate_);
} else {
TransitionsAccessor transitions(isolate(), *map);
DisallowGarbageCollection no_gc;
TransitionsAccessor transitions(isolate(), *map, &no_gc);
expected = transitions.ExpectedTransitionKey();
if (!expected.is_null()) {
// Directly read out the target while reading out the key, otherwise it
// might die while building the string below.
target =
TransitionsAccessor(isolate(), *map).ExpectedTransitionTarget();
target = TransitionsAccessor(isolate(), *map, &no_gc)
.ExpectedTransitionTarget();
}
}
@ -504,7 +505,7 @@ Handle<Object> JsonParser<Char>::BuildJsonObject(
map = ParentOfDescriptorOwner(isolate_, map, feedback, descriptor);
feedback_descriptors = 0;
}
if (!TransitionsAccessor(isolate(), *map)
if (!TransitionsAccessor(isolate(), map)
.FindTransitionToField(key)
.ToHandle(&target)) {
break;

View File

@ -3230,8 +3230,7 @@ void JSObject::MigrateToMap(Isolate* isolate, Handle<JSObject> object,
old_map->set_owns_descriptors(false);
DCHECK(old_map->is_abandoned_prototype_map());
// Ensure that no transition was inserted for prototype migrations.
DCHECK_EQ(0,
TransitionsAccessor(isolate, *old_map).NumberOfTransitions());
DCHECK_EQ(0, TransitionsAccessor(isolate, old_map).NumberOfTransitions());
DCHECK(new_map->GetBackPointer(isolate).IsUndefined(isolate));
DCHECK(object->map(isolate) != *old_map);
}
@ -4203,10 +4202,10 @@ Maybe<bool> JSObject::PreventExtensionsWithTransition(
Handle<Map> old_map(object->map(), isolate);
old_map = Map::Update(isolate, old_map);
Handle<Map> transition_map;
MaybeHandle<Map> maybe_transition_map =
TransitionsAccessor::SearchSpecial(isolate, old_map, *transition_marker);
if (maybe_transition_map.ToHandle(&transition_map)) {
TransitionsAccessor transitions(isolate, old_map);
Map transition = transitions.SearchSpecial(*transition_marker);
if (!transition.is_null()) {
Handle<Map> transition_map(transition, isolate);
DCHECK(transition_map->has_dictionary_elements() ||
transition_map->has_typed_array_elements() ||
transition_map->elements_kind() == SLOW_STRING_WRAPPER_ELEMENTS ||
@ -4216,7 +4215,7 @@ Maybe<bool> JSObject::PreventExtensionsWithTransition(
new_element_dictionary = CreateElementDictionary(isolate, object);
}
JSObject::MigrateToMap(isolate, object, transition_map);
} else if (TransitionsAccessor::CanHaveMoreTransitions(isolate, old_map)) {
} else if (transitions.CanHaveMoreTransitions()) {
// Create a new descriptor array with the appropriate property attributes
Handle<Map> new_map = Map::CopyForPreventExtensions(
isolate, old_map, attrs, transition_marker, "CopyForPreventExtensions");

View File

@ -768,7 +768,8 @@ void Map::SetBackPointer(HeapObject value, WriteBarrierMode mode) {
// static
Map Map::ElementsTransitionMap(Isolate* isolate, ConcurrencyMode cmode) {
return TransitionsAccessor(isolate, *this,
DisallowGarbageCollection no_gc;
return TransitionsAccessor(isolate, *this, &no_gc,
cmode == ConcurrencyMode::kConcurrent)
.SearchSpecial(ReadOnlyRoots(isolate).elements_transition_symbol());
}

View File

@ -277,7 +277,7 @@ Handle<Map> MapUpdater::UpdateImpl() {
}
DCHECK_EQ(kEnd, state_);
if (FLAG_fast_map_update) {
TransitionsAccessor::SetMigrationTarget(isolate_, old_map_, *result_map_);
TransitionsAccessor(isolate_, old_map_).SetMigrationTarget(*result_map_);
}
return result_map_;
}
@ -304,7 +304,7 @@ IntegrityLevelTransitionInfo DetectIntegrityLevelTransitions(
// be the last one in the transition tree).
DCHECK(!map.is_extensible());
Map previous = Map::cast(map.GetBackPointer(isolate));
TransitionsAccessor last_transitions(isolate, previous, is_concurrent);
TransitionsAccessor last_transitions(isolate, previous, no_gc, is_concurrent);
if (!last_transitions.HasIntegrityLevelTransitionTo(
map, &info.integrity_level_symbol, &info.integrity_level)) {
// The last transition was not integrity level transition - just bail out.
@ -322,7 +322,7 @@ IntegrityLevelTransitionInfo DetectIntegrityLevelTransitions(
// with integrity level transitions, just bail out.
while (!source_map.is_extensible()) {
previous = Map::cast(source_map.GetBackPointer(isolate));
TransitionsAccessor transitions(isolate, previous, is_concurrent);
TransitionsAccessor transitions(isolate, previous, no_gc, is_concurrent);
if (!transitions.HasIntegrityLevelTransitionTo(source_map)) {
return info;
}
@ -390,7 +390,7 @@ base::Optional<Map> MapUpdater::TryUpdateNoLock(Isolate* isolate, Map old_map,
if (info.has_integrity_level_transition) {
// Now replay the integrity level transition.
result = TransitionsAccessor(isolate, result,
result = TransitionsAccessor(isolate, result, &no_gc,
cmode == ConcurrencyMode::kConcurrent)
.SearchSpecial(info.integrity_level_symbol);
}
@ -423,13 +423,14 @@ MapUpdater::State MapUpdater::Normalize(const char* reason) {
// static
void MapUpdater::CompleteInobjectSlackTracking(Isolate* isolate,
Map initial_map) {
DisallowGarbageCollection no_gc;
// Has to be an initial map.
DCHECK(initial_map.GetBackPointer().IsUndefined(isolate));
const int slack = initial_map.ComputeMinObjectSlack(isolate);
DCHECK_GE(slack, 0);
TransitionsAccessor transitions(isolate, initial_map);
TransitionsAccessor transitions(isolate, initial_map, &no_gc);
TransitionsAccessor::TraverseCallback callback;
if (slack != 0) {
// Resize the initial map and all maps in its transition tree.
@ -517,7 +518,7 @@ bool MapUpdater::TrySaveIntegrityLevelTransitions() {
Handle<Map> previous =
handle(Map::cast(old_map_->GetBackPointer()), isolate_);
Symbol integrity_level_symbol;
TransitionsAccessor last_transitions(isolate_, *previous);
TransitionsAccessor last_transitions(isolate_, previous);
if (!last_transitions.HasIntegrityLevelTransitionTo(
*old_map_, &integrity_level_symbol, &integrity_level_)) {
// The last transition was not integrity level transition - just bail out.
@ -537,7 +538,7 @@ bool MapUpdater::TrySaveIntegrityLevelTransitions() {
while (!integrity_source_map_->is_extensible()) {
previous =
handle(Map::cast(integrity_source_map_->GetBackPointer()), isolate_);
TransitionsAccessor transitions(isolate_, *previous);
TransitionsAccessor transitions(isolate_, previous);
if (!transitions.HasIntegrityLevelTransitionTo(*integrity_source_map_)) {
return false;
}
@ -640,11 +641,12 @@ MapUpdater::State MapUpdater::FindTargetMap() {
int root_nof = root_map_->NumberOfOwnDescriptors();
for (InternalIndex i : InternalIndex::Range(root_nof, old_nof_)) {
PropertyDetails old_details = GetDetails(i);
Handle<Map> tmp_map;
MaybeHandle<Map> maybe_tmp_map = TransitionsAccessor::SearchTransition(
isolate_, target_map_, GetKey(i), old_details.kind(),
old_details.attributes());
if (!maybe_tmp_map.ToHandle(&tmp_map)) break;
Map transition = TransitionsAccessor(isolate_, target_map_)
.SearchTransition(GetKey(i), old_details.kind(),
old_details.attributes());
if (transition.is_null()) break;
Handle<Map> tmp_map(transition, isolate_);
Handle<DescriptorArray> tmp_descriptors(
tmp_map->instance_descriptors(isolate_), isolate_);
@ -725,9 +727,10 @@ MapUpdater::State MapUpdater::FindTargetMap() {
}
// We try to replay the integrity level transition here.
MaybeHandle<Map> maybe_transition = TransitionsAccessor::SearchSpecial(
isolate_, target_map_, *integrity_level_symbol_);
if (maybe_transition.ToHandle(&result_map_)) {
Map transition = TransitionsAccessor(isolate_, target_map_)
.SearchSpecial(*integrity_level_symbol_);
if (!transition.is_null()) {
result_map_ = handle(transition, isolate_);
state_ = kEnd;
return state_; // Done.
}
@ -736,11 +739,11 @@ MapUpdater::State MapUpdater::FindTargetMap() {
// Find the last compatible target map in the transition tree.
for (InternalIndex i : InternalIndex::Range(target_nof, old_nof_)) {
PropertyDetails old_details = GetDetails(i);
Handle<Map> tmp_map;
MaybeHandle<Map> maybe_tmp_map = TransitionsAccessor::SearchTransition(
isolate_, target_map_, GetKey(i), old_details.kind(),
old_details.attributes());
if (!maybe_tmp_map.ToHandle(&tmp_map)) break;
Map transition = TransitionsAccessor(isolate_, target_map_)
.SearchTransition(GetKey(i), old_details.kind(),
old_details.attributes());
if (transition.is_null()) break;
Handle<Map> tmp_map(transition, isolate_);
Handle<DescriptorArray> tmp_descriptors(
tmp_map->instance_descriptors(isolate_), isolate_);
#ifdef DEBUG
@ -930,13 +933,15 @@ Handle<DescriptorArray> MapUpdater::BuildDescriptorArray() {
}
Handle<Map> MapUpdater::FindSplitMap(Handle<DescriptorArray> descriptors) {
DisallowGarbageCollection no_gc;
int root_nof = root_map_->NumberOfOwnDescriptors();
Map current = *root_map_;
for (InternalIndex i : InternalIndex::Range(root_nof, old_nof_)) {
Name name = descriptors->GetKey(i);
PropertyDetails details = descriptors->GetDetails(i);
Map next =
TransitionsAccessor(isolate_, current)
TransitionsAccessor(isolate_, current, &no_gc)
.SearchTransition(name, details.kind(), details.attributes());
if (next.is_null()) break;
DescriptorArray next_descriptors = next.instance_descriptors(isolate_);
@ -976,20 +981,21 @@ MapUpdater::State MapUpdater::ConstructNewMap() {
}
InternalIndex split_index(split_nof);
PropertyDetails split_details = GetDetails(split_index);
TransitionsAccessor transitions(isolate_, split_map);
// Invalidate a transition target at |key|.
MaybeHandle<Map> maybe_transition = TransitionsAccessor::SearchTransition(
isolate_, split_map, GetKey(split_index), split_details.kind(),
split_details.attributes());
if (!maybe_transition.is_null()) {
maybe_transition.ToHandleChecked()->DeprecateTransitionTree(isolate_);
Handle<Map> maybe_transition(
transitions.SearchTransition(GetKey(split_index), split_details.kind(),
split_details.attributes()),
isolate_);
if (!maybe_transition->is_null()) {
maybe_transition->DeprecateTransitionTree(isolate_);
}
// If |maybe_transition| is not nullptr then the transition array already
// contains entry for given descriptor. This means that the transition
// could be inserted regardless of whether transitions array is full or not.
if (maybe_transition.is_null() &&
!TransitionsAccessor::CanHaveMoreTransitions(isolate_, split_map)) {
if (maybe_transition->is_null() && !transitions.CanHaveMoreTransitions()) {
return Normalize("Normalize_CantHaveMoreTransitions");
}
@ -1050,7 +1056,8 @@ MapUpdater::State MapUpdater::ConstructNewMap() {
MapUpdater::State MapUpdater::ConstructNewMapWithIntegrityLevelTransition() {
DCHECK_EQ(kAtIntegrityLevelSource, state_);
if (!TransitionsAccessor::CanHaveMoreTransitions(isolate_, target_map_)) {
TransitionsAccessor transitions(isolate_, target_map_);
if (!transitions.CanHaveMoreTransitions()) {
return Normalize("Normalize_CantHaveMoreTransitions");
}
@ -1120,6 +1127,7 @@ void MapUpdater::UpdateFieldType(Isolate* isolate, Handle<Map> map,
const MaybeObjectHandle& new_wrapped_type) {
DCHECK(new_wrapped_type->IsSmi() || new_wrapped_type->IsWeak());
// We store raw pointers in the queue, so no allocations are allowed.
DisallowGarbageCollection no_gc;
PropertyDetails details =
map->instance_descriptors(isolate).GetDetails(descriptor);
if (details.location() != PropertyLocation::kField) return;
@ -1136,7 +1144,7 @@ void MapUpdater::UpdateFieldType(Isolate* isolate, Handle<Map> map,
Map current = backlog.front();
backlog.pop();
TransitionsAccessor transitions(isolate, current);
TransitionsAccessor transitions(isolate, current, &no_gc);
int num_transitions = transitions.NumberOfTransitions();
for (int i = 0; i < num_transitions; ++i) {
Map target = transitions.GetTarget(i);

View File

@ -557,7 +557,8 @@ bool Map::HasOutOfObjectProperties() const {
void Map::DeprecateTransitionTree(Isolate* isolate) {
if (is_deprecated()) return;
TransitionsAccessor transitions(isolate, *this);
DisallowGarbageCollection no_gc;
TransitionsAccessor transitions(isolate, *this, &no_gc);
int num_transitions = transitions.NumberOfTransitions();
for (int i = 0; i < num_transitions; ++i) {
transitions.GetTarget(i).DeprecateTransitionTree(isolate);
@ -644,7 +645,7 @@ Map SearchMigrationTarget(Isolate* isolate, Map old_map) {
Map target = old_map;
do {
target = TransitionsAccessor(isolate, target).GetMigrationTarget();
target = TransitionsAccessor(isolate, target, &no_gc).GetMigrationTarget();
} while (!target.is_null() && target.is_deprecated());
if (target.is_null()) return Map();
@ -692,7 +693,8 @@ MaybeHandle<Map> Map::TryUpdate(Isolate* isolate, Handle<Map> old_map) {
isolate, *old_map, ConcurrencyMode::kNotConcurrent);
if (!new_map.has_value()) return MaybeHandle<Map>();
if (FLAG_fast_map_update) {
TransitionsAccessor::SetMigrationTarget(isolate, old_map, new_map.value());
TransitionsAccessor(isolate, *old_map, &no_gc)
.SetMigrationTarget(new_map.value());
}
return handle(new_map.value(), isolate);
}
@ -714,7 +716,7 @@ Map Map::TryReplayPropertyTransitions(Isolate* isolate, Map old_map,
for (InternalIndex i : InternalIndex::Range(root_nof, old_nof)) {
PropertyDetails old_details = old_descriptors.GetDetails(i);
Map transition =
TransitionsAccessor(isolate, new_map, is_concurrent)
TransitionsAccessor(isolate, new_map, &no_gc, is_concurrent)
.SearchTransition(old_descriptors.GetKey(i), old_details.kind(),
old_details.attributes());
if (transition.is_null()) return Map();
@ -1402,7 +1404,7 @@ void Map::ConnectTransition(Isolate* isolate, Handle<Map> parent,
LOG(isolate, MapEvent("Transition", parent, child, "prototype", name));
}
} else {
TransitionsAccessor::Insert(isolate, parent, name, child, flag);
TransitionsAccessor(isolate, parent).Insert(name, child, flag);
if (FLAG_log_maps) {
LOG(isolate, MapEvent("Transition", parent, child, "", name));
}
@ -1430,7 +1432,7 @@ Handle<Map> Map::CopyReplaceDescriptors(Isolate* isolate, Handle<Map> map,
result->InitializeDescriptors(isolate, *descriptors);
} else {
if (flag == INSERT_TRANSITION &&
TransitionsAccessor::CanHaveMoreTransitions(isolate, map)) {
TransitionsAccessor(isolate, map).CanHaveMoreTransitions()) {
result->InitializeDescriptors(isolate, *descriptors);
DCHECK(!maybe_name.is_null());
@ -1542,7 +1544,7 @@ Handle<Map> Map::CopyAsElementsKind(Isolate* isolate, Handle<Map> map,
bool insert_transition =
flag == INSERT_TRANSITION &&
TransitionsAccessor::CanHaveMoreTransitions(isolate, map) &&
TransitionsAccessor(isolate, map).CanHaveMoreTransitions() &&
maybe_elements_transition_map.is_null();
if (insert_transition) {
@ -1576,10 +1578,10 @@ Handle<Map> Map::AsLanguageMode(Isolate* isolate, Handle<Map> initial_map,
DCHECK_EQ(LanguageMode::kStrict, shared_info->language_mode());
Handle<Symbol> transition_symbol =
isolate->factory()->strict_function_transition_symbol();
MaybeHandle<Map> maybe_transition = TransitionsAccessor::SearchSpecial(
isolate, initial_map, *transition_symbol);
Map maybe_transition = TransitionsAccessor(isolate, initial_map)
.SearchSpecial(*transition_symbol);
if (!maybe_transition.is_null()) {
return maybe_transition.ToHandleChecked();
return handle(maybe_transition, isolate);
}
initial_map->NotifyLeafMapLayoutChange(isolate);
@ -1593,7 +1595,7 @@ Handle<Map> Map::AsLanguageMode(Isolate* isolate, Handle<Map> initial_map,
map->set_prototype(initial_map->prototype());
map->set_construction_counter(initial_map->construction_counter());
if (TransitionsAccessor::CanHaveMoreTransitions(isolate, initial_map)) {
if (TransitionsAccessor(isolate, initial_map).CanHaveMoreTransitions()) {
Map::ConnectTransition(isolate, initial_map, map, transition_symbol,
SPECIAL_TRANSITION);
}
@ -1805,10 +1807,11 @@ Handle<Map> Map::TransitionToDataProperty(Isolate* isolate, Handle<Map> map,
// Migrate to the newest map before storing the property.
map = Update(isolate, map);
MaybeHandle<Map> maybe_transition = TransitionsAccessor::SearchTransition(
isolate, map, *name, PropertyKind::kData, attributes);
Handle<Map> transition;
if (maybe_transition.ToHandle(&transition)) {
Map maybe_transition =
TransitionsAccessor(isolate, map)
.SearchTransition(*name, PropertyKind::kData, attributes);
if (!maybe_transition.is_null()) {
Handle<Map> transition(maybe_transition, isolate);
InternalIndex descriptor = transition->LastAdded();
DCHECK_EQ(attributes, transition->instance_descriptors(isolate)
@ -1900,10 +1903,11 @@ Handle<Map> Map::TransitionToAccessorProperty(Isolate* isolate, Handle<Map> map,
? KEEP_INOBJECT_PROPERTIES
: CLEAR_INOBJECT_PROPERTIES;
MaybeHandle<Map> maybe_transition = TransitionsAccessor::SearchTransition(
isolate, map, *name, PropertyKind::kAccessor, attributes);
Handle<Map> transition;
if (maybe_transition.ToHandle(&transition)) {
Map maybe_transition =
TransitionsAccessor(isolate, map)
.SearchTransition(*name, PropertyKind::kAccessor, attributes);
if (!maybe_transition.is_null()) {
Handle<Map> transition(maybe_transition, isolate);
DescriptorArray descriptors = transition->instance_descriptors(isolate);
InternalIndex last_descriptor = transition->LastAdded();
DCHECK(descriptors.GetKey(last_descriptor).Equals(*name));
@ -1996,7 +2000,7 @@ Handle<Map> Map::CopyAddDescriptor(Isolate* isolate, Handle<Map> map,
// Share descriptors only if map owns descriptors and it not an initial map.
if (flag == INSERT_TRANSITION && map->owns_descriptors() &&
!map->GetBackPointer().IsUndefined(isolate) &&
TransitionsAccessor::CanHaveMoreTransitions(isolate, map)) {
TransitionsAccessor(isolate, map).CanHaveMoreTransitions()) {
return ShareDescriptor(isolate, map, descriptors, descriptor);
}
@ -2148,11 +2152,12 @@ bool Map::EquivalentToForNormalization(const Map other,
}
int Map::ComputeMinObjectSlack(Isolate* isolate) {
DisallowGarbageCollection no_gc;
// Has to be an initial map.
DCHECK(GetBackPointer().IsUndefined(isolate));
int slack = UnusedPropertyFields();
TransitionsAccessor transitions(isolate, *this);
TransitionsAccessor transitions(isolate, *this, &no_gc);
TransitionsAccessor::TraverseCallback callback = [&](Map map) {
slack = std::min(slack, map.UnusedPropertyFields());
};
@ -2278,11 +2283,11 @@ void Map::StartInobjectSlackTracking() {
Handle<Map> Map::TransitionToPrototype(Isolate* isolate, Handle<Map> map,
Handle<HeapObject> prototype) {
Handle<Map> new_map =
TransitionsAccessor::GetPrototypeTransition(isolate, map, prototype);
TransitionsAccessor(isolate, map).GetPrototypeTransition(prototype);
if (new_map.is_null()) {
new_map = Copy(isolate, map, "TransitionToPrototype");
TransitionsAccessor::PutPrototypeTransition(isolate, map, prototype,
new_map);
TransitionsAccessor(isolate, map)
.PutPrototypeTransition(prototype, new_map);
Map::SetPrototype(isolate, new_map, prototype);
}
return new_map;

View File

@ -17,23 +17,9 @@
namespace v8 {
namespace internal {
// static
TransitionArray TransitionsAccessor::GetTransitionArray(
Isolate* isolate, MaybeObject raw_transitions) {
DCHECK_EQ(kFullTransitionArray, GetEncoding(isolate, raw_transitions));
USE(isolate);
return TransitionArray::cast(raw_transitions.GetHeapObjectAssumeStrong());
}
// static
TransitionArray TransitionsAccessor::GetTransitionArray(Isolate* isolate,
Handle<Map> map) {
MaybeObject raw_transitions = map->raw_transitions(isolate, kAcquireLoad);
return GetTransitionArray(isolate, raw_transitions);
}
TransitionArray TransitionsAccessor::transitions() {
return GetTransitionArray(isolate_, raw_transitions_);
DCHECK_EQ(kFullTransitionArray, encoding());
return TransitionArray::cast(raw_transitions_->GetHeapObjectAssumeStrong());
}
OBJECT_CONSTRUCTORS_IMPL(TransitionArray, WeakFixedArray)
@ -207,13 +193,26 @@ int TransitionArray::SearchName(Name name, bool concurrent_search,
}
TransitionsAccessor::TransitionsAccessor(Isolate* isolate, Map map,
DisallowGarbageCollection* no_gc,
bool concurrent_access)
: isolate_(isolate), map_(map), concurrent_access_(concurrent_access) {
Initialize();
USE(no_gc);
}
TransitionsAccessor::TransitionsAccessor(Isolate* isolate, Handle<Map> map,
bool concurrent_access)
: isolate_(isolate),
map_(map),
raw_transitions_(map.raw_transitions(isolate_, kAcquireLoad)),
encoding_(GetEncoding(isolate_, raw_transitions_)),
map_handle_(map),
map_(*map),
concurrent_access_(concurrent_access) {
DCHECK_IMPLIES(encoding_ == kMigrationTarget, map_.is_deprecated());
Initialize();
}
void TransitionsAccessor::Reload() {
DCHECK(!map_handle_.is_null());
map_ = *map_handle_;
Initialize();
}
int TransitionsAccessor::Capacity() { return transitions().Capacity(); }
@ -240,36 +239,13 @@ TransitionsAccessor::Encoding TransitionsAccessor::GetEncoding(
}
}
// static
TransitionsAccessor::Encoding TransitionsAccessor::GetEncoding(
Isolate* isolate, TransitionArray array) {
return GetEncoding(isolate, MaybeObject::FromObject(array));
}
// static
TransitionsAccessor::Encoding TransitionsAccessor::GetEncoding(
Isolate* isolate, Handle<Map> map) {
MaybeObject raw_transitions = map->raw_transitions(isolate, kAcquireLoad);
return GetEncoding(isolate, raw_transitions);
}
// static
MaybeHandle<Map> TransitionsAccessor::SearchTransition(
Isolate* isolate, Handle<Map> map, Name name, PropertyKind kind,
PropertyAttributes attributes) {
Map result = TransitionsAccessor(isolate, *map)
.SearchTransition(name, kind, attributes);
if (result.is_null()) return MaybeHandle<Map>();
return MaybeHandle<Map>(result, isolate);
}
// static
MaybeHandle<Map> TransitionsAccessor::SearchSpecial(Isolate* isolate,
Handle<Map> map,
Symbol name) {
Map result = TransitionsAccessor(isolate, *map).SearchSpecial(name);
if (result.is_null()) return MaybeHandle<Map>();
return MaybeHandle<Map>(result, isolate);
void TransitionsAccessor::Initialize() {
raw_transitions_ = map_.raw_transitions(isolate_, kAcquireLoad);
encoding_ = GetEncoding(isolate_, raw_transitions_);
DCHECK_IMPLIES(encoding_ == kMigrationTarget, map_.is_deprecated());
#if DEBUG
needs_reload_ = false;
#endif
}
int TransitionArray::number_of_transitions() const {

View File

@ -12,13 +12,10 @@
namespace v8 {
namespace internal {
// static
Map TransitionsAccessor::GetSimpleTransition(Isolate* isolate,
Handle<Map> map) {
MaybeObject raw_transitions = map->raw_transitions(isolate, kAcquireLoad);
switch (GetEncoding(isolate, raw_transitions)) {
Map TransitionsAccessor::GetSimpleTransition() {
switch (encoding()) {
case kWeakRef:
return Map::cast(raw_transitions->GetHeapObjectAssumeWeak());
return Map::cast(raw_transitions_->GetHeapObjectAssumeWeak());
default:
return Map();
}
@ -37,56 +34,56 @@ bool TransitionsAccessor::HasSimpleTransitionTo(Map map) {
UNREACHABLE();
}
// static
void TransitionsAccessor::Insert(Isolate* isolate, Handle<Map> map,
Handle<Name> name, Handle<Map> target,
void TransitionsAccessor::Insert(Handle<Name> name, Handle<Map> target,
SimpleTransitionFlag flag) {
Encoding encoding = GetEncoding(isolate, map);
DCHECK_NE(kPrototypeInfo, encoding);
target->SetBackPointer(*map);
DCHECK(!concurrent_access_);
DCHECK(!map_handle_.is_null());
DCHECK_NE(kPrototypeInfo, encoding());
target->SetBackPointer(map_);
// If the map doesn't have any transitions at all yet, install the new one.
if (encoding == kUninitialized || encoding == kMigrationTarget) {
if (encoding() == kUninitialized || encoding() == kMigrationTarget) {
if (flag == SIMPLE_PROPERTY_TRANSITION) {
ReplaceTransitions(isolate, map, HeapObjectReference::Weak(*target));
ReplaceTransitions(HeapObjectReference::Weak(*target));
return;
}
// If the flag requires a full TransitionArray, allocate one.
Handle<TransitionArray> result =
isolate->factory()->NewTransitionArray(1, 0);
isolate_->factory()->NewTransitionArray(1, 0);
result->Set(0, *name, HeapObjectReference::Weak(*target));
ReplaceTransitions(isolate, map, result);
DCHECK_EQ(kFullTransitionArray, GetEncoding(isolate, *result));
ReplaceTransitions(MaybeObject::FromObject(*result));
Reload();
DCHECK_EQ(kFullTransitionArray, encoding());
return;
}
if (encoding == kWeakRef) {
Map simple_transition = GetSimpleTransition(isolate, map);
if (encoding() == kWeakRef) {
Map simple_transition = GetSimpleTransition();
DCHECK(!simple_transition.is_null());
if (flag == SIMPLE_PROPERTY_TRANSITION) {
Name key = GetSimpleTransitionKey(simple_transition);
PropertyDetails old_details =
simple_transition.GetLastDescriptorDetails(isolate);
PropertyDetails old_details = GetSimpleTargetDetails(simple_transition);
PropertyDetails new_details = GetTargetDetails(*name, *target);
if (key.Equals(*name) && old_details.kind() == new_details.kind() &&
old_details.attributes() == new_details.attributes()) {
ReplaceTransitions(isolate, map, HeapObjectReference::Weak(*target));
ReplaceTransitions(HeapObjectReference::Weak(*target));
return;
}
}
// Otherwise allocate a full TransitionArray with slack for a new entry.
Handle<Map> map(simple_transition, isolate_);
Handle<TransitionArray> result =
isolate->factory()->NewTransitionArray(1, 1);
// Reload `simple_transition`. Allocations might have caused it to be
// cleared.
simple_transition = GetSimpleTransition(isolate, map);
isolate_->factory()->NewTransitionArray(1, 1);
// Reload state; allocations might have caused it to be cleared.
Reload();
simple_transition = GetSimpleTransition();
if (simple_transition.is_null()) {
result->Set(0, *name, HeapObjectReference::Weak(*target));
ReplaceTransitions(isolate, map, result);
DCHECK_EQ(kFullTransitionArray, GetEncoding(isolate, *result));
ReplaceTransitions(MaybeObject::FromObject(*result));
Reload();
DCHECK_EQ(kFullTransitionArray, encoding());
return;
}
@ -119,27 +116,28 @@ void TransitionsAccessor::Insert(Isolate* isolate, Handle<Map> map,
result->SetRawTarget(insertion_index, HeapObjectReference::Weak(*target));
SLOW_DCHECK(result->IsSortedNoDuplicates());
ReplaceTransitions(isolate, map, result);
DCHECK_EQ(kFullTransitionArray, GetEncoding(isolate, *result));
ReplaceTransitions(MaybeObject::FromObject(*result));
Reload();
DCHECK_EQ(kFullTransitionArray, encoding());
return;
}
// At this point, we know that the map has a full TransitionArray.
DCHECK_EQ(kFullTransitionArray, encoding);
DCHECK_EQ(kFullTransitionArray, encoding());
int number_of_transitions = 0;
int new_nof = 0;
int insertion_index = kNotFound;
const bool is_special_transition = flag == SPECIAL_TRANSITION;
DCHECK_EQ(is_special_transition,
IsSpecialTransition(ReadOnlyRoots(isolate), *name));
IsSpecialTransition(ReadOnlyRoots(isolate_), *name));
PropertyDetails details = is_special_transition
? PropertyDetails::Empty()
: GetTargetDetails(*name, *target);
{
DisallowGarbageCollection no_gc;
TransitionArray array = GetTransitionArray(isolate, map);
TransitionArray array = transitions();
number_of_transitions = array.number_of_transitions();
int index =
@ -150,7 +148,7 @@ void TransitionsAccessor::Insert(Isolate* isolate, Handle<Map> map,
// If an existing entry was found, overwrite it and return.
if (index != kNotFound) {
base::SharedMutexGuard<base::kExclusive> shared_mutex_guard(
isolate->full_transition_array_access());
isolate_->full_transition_array_access());
array.SetRawTarget(index, HeapObjectReference::Weak(*target));
return;
}
@ -163,7 +161,7 @@ void TransitionsAccessor::Insert(Isolate* isolate, Handle<Map> map,
// If there is enough capacity, insert new entry into the existing array.
if (new_nof <= array.Capacity()) {
base::SharedMutexGuard<base::kExclusive> shared_mutex_guard(
isolate->full_transition_array_access());
isolate_->full_transition_array_access());
array.SetNumberOfTransitions(new_nof);
for (int i = number_of_transitions; i > insertion_index; --i) {
array.SetKey(i, array.GetKey(i - 1));
@ -177,15 +175,16 @@ void TransitionsAccessor::Insert(Isolate* isolate, Handle<Map> map,
}
// We're gonna need a bigger TransitionArray.
Handle<TransitionArray> result = isolate->factory()->NewTransitionArray(
Handle<TransitionArray> result = isolate_->factory()->NewTransitionArray(
new_nof,
Map::SlackForArraySize(number_of_transitions, kMaxNumberOfTransitions));
// The map's transition array may have shrunk during the allocation above as
// it was weakly traversed, though it is guaranteed not to disappear. Trim the
// result copy if needed, and recompute variables.
Reload();
DisallowGarbageCollection no_gc;
TransitionArray array = GetTransitionArray(isolate, map);
TransitionArray array = transitions();
if (array.number_of_transitions() != number_of_transitions) {
DCHECK_LT(array.number_of_transitions(), number_of_transitions);
@ -218,7 +217,7 @@ void TransitionsAccessor::Insert(Isolate* isolate, Handle<Map> map,
}
SLOW_DCHECK(result->IsSortedNoDuplicates());
ReplaceTransitions(isolate, map, result);
ReplaceTransitions(MaybeObject::FromObject(*result));
}
Map TransitionsAccessor::SearchTransition(Name name, PropertyKind kind,
@ -306,14 +305,10 @@ void TransitionsAccessor::ForEachTransitionTo(
UNREACHABLE();
}
// static
bool TransitionsAccessor::CanHaveMoreTransitions(Isolate* isolate,
Handle<Map> map) {
if (map->is_dictionary_map()) return false;
MaybeObject raw_transitions = map->raw_transitions(isolate, kAcquireLoad);
if (GetEncoding(isolate, raw_transitions) == kFullTransitionArray) {
return GetTransitionArray(isolate, raw_transitions)
.number_of_transitions() < kMaxNumberOfTransitions;
bool TransitionsAccessor::CanHaveMoreTransitions() {
if (map_.is_dictionary_map()) return false;
if (encoding() == kFullTransitionArray) {
return transitions().number_of_transitions() < kMaxNumberOfTransitions;
}
return true;
}
@ -380,33 +375,31 @@ Handle<WeakFixedArray> TransitionArray::GrowPrototypeTransitionArray(
return array;
}
// static
void TransitionsAccessor::PutPrototypeTransition(Isolate* isolate,
Handle<Map> map,
Handle<Object> prototype,
void TransitionsAccessor::PutPrototypeTransition(Handle<Object> prototype,
Handle<Map> target_map) {
DCHECK(HeapObject::cast(*prototype).map().IsMap());
// Don't cache prototype transition if this map is either shared, or a map of
// a prototype.
if (map->is_prototype_map()) return;
if (map->is_dictionary_map() || !FLAG_cache_prototype_transitions) return;
if (map_.is_prototype_map()) return;
if (map_.is_dictionary_map() || !FLAG_cache_prototype_transitions) return;
const int header = TransitionArray::kProtoTransitionHeaderSize;
Handle<WeakFixedArray> cache(GetPrototypeTransitions(isolate, map), isolate);
Handle<WeakFixedArray> cache(GetPrototypeTransitions(), isolate_);
int capacity = cache->length() - header;
int transitions = TransitionArray::NumberOfPrototypeTransitions(*cache) + 1;
base::SharedMutexGuard<base::kExclusive> scope(
isolate->full_transition_array_access());
isolate_->full_transition_array_access());
if (transitions > capacity) {
// Grow the array if compacting it doesn't free space.
if (!TransitionArray::CompactPrototypeTransitionArray(isolate, *cache)) {
if (!TransitionArray::CompactPrototypeTransitionArray(isolate_, *cache)) {
if (capacity == TransitionArray::kMaxCachedPrototypeTransitions) return;
cache = TransitionArray::GrowPrototypeTransitionArray(
cache, 2 * transitions, isolate);
SetPrototypeTransitions(isolate, map, cache);
cache, 2 * transitions, isolate_);
Reload();
SetPrototypeTransitions(cache);
}
}
@ -418,11 +411,10 @@ void TransitionsAccessor::PutPrototypeTransition(Isolate* isolate,
TransitionArray::SetNumberOfPrototypeTransitions(*cache, last + 1);
}
// static
Handle<Map> TransitionsAccessor::GetPrototypeTransition(
Isolate* isolate, Handle<Map> map, Handle<Object> prototype) {
Handle<Object> prototype) {
DisallowGarbageCollection no_gc;
WeakFixedArray cache = GetPrototypeTransitions(isolate, map);
WeakFixedArray cache = GetPrototypeTransitions();
int length = TransitionArray::NumberOfPrototypeTransitions(cache);
for (int i = 0; i < length; i++) {
MaybeObject target =
@ -430,28 +422,21 @@ Handle<Map> TransitionsAccessor::GetPrototypeTransition(
DCHECK(target->IsWeakOrCleared());
HeapObject heap_object;
if (target->GetHeapObjectIfWeak(&heap_object)) {
Map target_map = Map::cast(heap_object);
if (target_map.prototype() == *prototype) {
return handle(target_map, isolate);
Map map = Map::cast(heap_object);
if (map.prototype() == *prototype) {
return handle(map, isolate_);
}
}
}
return Handle<Map>();
}
// static
WeakFixedArray TransitionsAccessor::GetPrototypeTransitions(Isolate* isolate,
Handle<Map> map) {
MaybeObject raw_transitions = map->raw_transitions(isolate, kAcquireLoad);
if (GetEncoding(isolate, raw_transitions) != kFullTransitionArray) {
return ReadOnlyRoots(isolate).empty_weak_fixed_array();
WeakFixedArray TransitionsAccessor::GetPrototypeTransitions() {
if (encoding() != kFullTransitionArray ||
!transitions().HasPrototypeTransitions()) {
return ReadOnlyRoots(isolate_).empty_weak_fixed_array();
}
TransitionArray transition_array =
GetTransitionArray(isolate, raw_transitions);
if (!transition_array.HasPrototypeTransitions()) {
return ReadOnlyRoots(isolate).empty_weak_fixed_array();
}
return transition_array.GetPrototypeTransitions();
return transitions().GetPrototypeTransitions();
}
// static
@ -476,15 +461,14 @@ int TransitionsAccessor::NumberOfTransitions() {
UNREACHABLE();
}
// static
void TransitionsAccessor::SetMigrationTarget(Isolate* isolate, Handle<Map> map,
Map migration_target) {
void TransitionsAccessor::SetMigrationTarget(Map migration_target) {
// We only cache the migration target for maps with empty transitions for GC's
// sake.
if (GetEncoding(isolate, map) != kUninitialized) return;
DCHECK(map->is_deprecated());
map->set_raw_transitions(MaybeObject::FromObject(migration_target),
if (encoding() != kUninitialized) return;
DCHECK(map_.is_deprecated());
map_.set_raw_transitions(MaybeObject::FromObject(migration_target),
kReleaseStore);
MarkNeedsReload();
}
Map TransitionsAccessor::GetMigrationTarget() {
@ -494,58 +478,44 @@ Map TransitionsAccessor::GetMigrationTarget() {
return Map();
}
// static
void TransitionsAccessor::ReplaceTransitions(Isolate* isolate, Handle<Map> map,
MaybeObject new_transitions) {
void TransitionsAccessor::ReplaceTransitions(MaybeObject new_transitions) {
if (encoding() == kFullTransitionArray) {
#if DEBUG
if (GetEncoding(isolate, map) == kFullTransitionArray) {
TransitionArray old_transitions = transitions();
CheckNewTransitionsAreConsistent(
isolate, map, new_transitions->GetHeapObjectAssumeStrong());
DCHECK_NE(GetTransitionArray(isolate, map),
new_transitions->GetHeapObjectAssumeStrong());
}
old_transitions, new_transitions->GetHeapObjectAssumeStrong());
DCHECK(old_transitions != new_transitions->GetHeapObjectAssumeStrong());
#endif
map->set_raw_transitions(new_transitions, kReleaseStore);
USE(isolate);
}
map_.set_raw_transitions(new_transitions, kReleaseStore);
MarkNeedsReload();
}
// static
void TransitionsAccessor::ReplaceTransitions(
Isolate* isolate, Handle<Map> map,
Handle<TransitionArray> new_transitions) {
ReplaceTransitions(isolate, map, MaybeObject::FromObject(*new_transitions));
}
// static
void TransitionsAccessor::SetPrototypeTransitions(
Isolate* isolate, Handle<Map> map,
Handle<WeakFixedArray> proto_transitions) {
EnsureHasFullTransitionArray(isolate, map);
GetTransitionArray(isolate, map->raw_transitions(isolate, kAcquireLoad))
.SetPrototypeTransitions(*proto_transitions);
EnsureHasFullTransitionArray();
transitions().SetPrototypeTransitions(*proto_transitions);
}
// static
void TransitionsAccessor::EnsureHasFullTransitionArray(Isolate* isolate,
Handle<Map> map) {
Encoding encoding =
GetEncoding(isolate, map->raw_transitions(isolate, kAcquireLoad));
if (encoding == kFullTransitionArray) return;
void TransitionsAccessor::EnsureHasFullTransitionArray() {
if (encoding() == kFullTransitionArray) return;
int nof =
(encoding == kUninitialized || encoding == kMigrationTarget) ? 0 : 1;
Handle<TransitionArray> result = isolate->factory()->NewTransitionArray(nof);
(encoding() == kUninitialized || encoding() == kMigrationTarget) ? 0 : 1;
Handle<TransitionArray> result = isolate_->factory()->NewTransitionArray(nof);
Reload(); // Reload after possible GC.
if (nof == 1) {
if (encoding == kUninitialized) {
if (encoding() == kUninitialized) {
// If allocation caused GC and cleared the target, trim the new array.
result->SetNumberOfTransitions(0);
} else {
// Otherwise populate the new array.
Map target = GetSimpleTransition(isolate, map);
Name key = GetSimpleTransitionKey(target);
result->Set(0, key, HeapObjectReference::Weak(target));
Handle<Map> target(GetSimpleTransition(), isolate_);
Name key = GetSimpleTransitionKey(*target);
result->Set(0, key, HeapObjectReference::Weak(*target));
}
}
ReplaceTransitions(isolate, map, result);
ReplaceTransitions(MaybeObject::FromObject(*result));
Reload(); // Reload after replacing transitions.
}
void TransitionsAccessor::TraverseTransitionTreeInternal(
@ -605,21 +575,18 @@ void TransitionsAccessor::TraverseTransitionTreeInternal(
}
#ifdef DEBUG
// static
void TransitionsAccessor::CheckNewTransitionsAreConsistent(Isolate* isolate,
Handle<Map> map,
Object transitions) {
void TransitionsAccessor::CheckNewTransitionsAreConsistent(
TransitionArray old_transitions, Object transitions) {
// This function only handles full transition arrays.
TransitionArray old_transitions = GetTransitionArray(isolate, map);
DCHECK_EQ(kFullTransitionArray, GetEncoding(isolate, old_transitions));
DCHECK_EQ(kFullTransitionArray, encoding());
TransitionArray new_transitions = TransitionArray::cast(transitions);
for (int i = 0; i < old_transitions.number_of_transitions(); i++) {
Map target = old_transitions.GetTarget(i);
if (target.instance_descriptors(isolate) ==
map->instance_descriptors(isolate)) {
if (target.instance_descriptors(isolate_) ==
map_.instance_descriptors(isolate_)) {
Name key = old_transitions.GetKey(i);
int new_target_index;
if (IsSpecialTransition(ReadOnlyRoots(isolate), key)) {
if (IsSpecialTransition(ReadOnlyRoots(isolate_), key)) {
new_target_index = new_transitions.SearchSpecial(Symbol::cast(key));
} else {
PropertyDetails details = GetTargetDetails(key, target);

View File

@ -50,23 +50,20 @@ class V8_EXPORT_PRIVATE TransitionsAccessor {
// in background threads. It acquires a reader lock for critical paths, as
// well as blocking the accessor from modifying the TransitionsArray.
inline TransitionsAccessor(Isolate* isolate, Map map,
DisallowGarbageCollection* no_gc,
bool concurrent_access = false);
inline TransitionsAccessor(Isolate* isolate, Handle<Map> map,
bool concurrent_access = false);
// Insert a new transition into |map|'s transition array, extending it
// as necessary. This can trigger GC.
static void Insert(Isolate* isolate, Handle<Map> map, Handle<Name> name,
Handle<Map> target, SimpleTransitionFlag flag);
// as necessary.
// Requires the constructor that takes a Handle<Map> to have been used.
// This TransitionsAccessor instance is unusable after this operation.
void Insert(Handle<Name> name, Handle<Map> target, SimpleTransitionFlag flag);
Map SearchTransition(Name name, PropertyKind kind,
PropertyAttributes attributes);
static inline MaybeHandle<Map> SearchTransition(
Isolate* isolate, Handle<Map> map, Name name, PropertyKind kind,
PropertyAttributes attributes);
Map SearchSpecial(Symbol name);
static inline MaybeHandle<Map> SearchSpecial(Isolate* isolate,
Handle<Map> map, Symbol name);
// Returns true for non-property transitions like elements kind, or
// or frozen/sealed transitions.
static bool IsSpecialTransition(ReadOnlyRoots roots, Name name);
@ -95,12 +92,11 @@ class V8_EXPORT_PRIVATE TransitionsAccessor {
// object space. Otherwise ClearNonLiveReferences would leak memory while
// applying in-place right trimming.
static const int kMaxNumberOfTransitions = 1024 + 512;
bool CanHaveMoreTransitions();
inline Name GetKey(int transition_number);
inline Map GetTarget(int transition_number);
static inline PropertyDetails GetTargetDetails(Name name, Map target);
static bool CanHaveMoreTransitions(Isolate* isolate, Handle<Map> map);
static bool IsMatchingMap(Map target, Name name, PropertyKind kind,
PropertyAttributes attributes);
@ -128,20 +124,15 @@ class V8_EXPORT_PRIVATE TransitionsAccessor {
// prototype is set, rather than creating a new map every time. The
// transitions are in the form of a map where the keys are prototype objects
// and the values are the maps they transition to.
// PutPrototypeTransition can trigger GC.
static void PutPrototypeTransition(Isolate* isolate, Handle<Map>,
Handle<Object> prototype,
Handle<Map> target_map);
static Handle<Map> GetPrototypeTransition(Isolate* isolate, Handle<Map> map,
Handle<Object> prototype);
void PutPrototypeTransition(Handle<Object> prototype, Handle<Map> target_map);
Handle<Map> GetPrototypeTransition(Handle<Object> prototype);
// During the first-time Map::Update and Map::TryUpdate, the migration target
// map could be cached in the raw_transitions slot of the old map that is
// deprecated from the map transition tree. The next time old map is updated,
// we will check this cache slot as a shortcut to get the migration target
// map.
static void SetMigrationTarget(Isolate* isolate, Handle<Map> map,
Map migration_target);
void SetMigrationTarget(Map migration_target);
Map GetMigrationTarget();
#if DEBUG || OBJECT_PRINT
@ -152,9 +143,8 @@ class V8_EXPORT_PRIVATE TransitionsAccessor {
DisallowGarbageCollection* no_gc);
#endif
#if DEBUG
static void CheckNewTransitionsAreConsistent(Isolate* isolate,
Handle<Map> map,
Object transitions);
void CheckNewTransitionsAreConsistent(TransitionArray old_transitions,
Object transitions);
bool IsConsistentWithBackPointers();
bool IsSortedNoDuplicates();
#endif
@ -169,14 +159,17 @@ class V8_EXPORT_PRIVATE TransitionsAccessor {
kFullTransitionArray,
};
inline Encoding encoding() { return encoding_; }
inline void Reload();
inline Encoding encoding() {
DCHECK(!needs_reload_);
return encoding_;
}
inline int Capacity();
inline TransitionArray transitions();
DISALLOW_GARBAGE_COLLECTION(no_gc_)
private:
friend class MarkCompactCollector; // For HasSimpleTransitionTo.
friend class third_party_heap::Impl;
@ -184,44 +177,44 @@ class V8_EXPORT_PRIVATE TransitionsAccessor {
static inline Encoding GetEncoding(Isolate* isolate,
MaybeObject raw_transitions);
static inline Encoding GetEncoding(Isolate* isolate, TransitionArray array);
static inline Encoding GetEncoding(Isolate* isolate, Handle<Map> map);
static inline TransitionArray GetTransitionArray(Isolate* isolate,
MaybeObject raw_transitions);
static inline TransitionArray GetTransitionArray(Isolate* isolate,
Handle<Map> map);
static inline Map GetSimpleTransition(Isolate* isolate, Handle<Map> map);
static inline Name GetSimpleTransitionKey(Map transition);
inline PropertyDetails GetSimpleTargetDetails(Map transition);
static inline Name GetSimpleTransitionKey(Map transition);
static inline Map GetTargetFromRaw(MaybeObject raw);
static void EnsureHasFullTransitionArray(Isolate* isolate, Handle<Map> map);
static void SetPrototypeTransitions(Isolate* isolate, Handle<Map> map,
Handle<WeakFixedArray> proto_transitions);
static WeakFixedArray GetPrototypeTransitions(Isolate* isolate,
Handle<Map> map);
void MarkNeedsReload() {
#if DEBUG
needs_reload_ = true;
#endif
}
static inline void ReplaceTransitions(Isolate* isolate, Handle<Map> map,
MaybeObject new_transitions);
static inline void ReplaceTransitions(
Isolate* isolate, Handle<Map> map,
Handle<TransitionArray> new_transitions);
inline void Initialize();
inline Map GetSimpleTransition();
bool HasSimpleTransitionTo(Map map);
void ReplaceTransitions(MaybeObject new_transitions);
inline Map GetTargetMapFromWeakRef();
void EnsureHasFullTransitionArray();
void SetPrototypeTransitions(Handle<WeakFixedArray> proto_transitions);
WeakFixedArray GetPrototypeTransitions();
void TraverseTransitionTreeInternal(const TraverseCallback& callback,
DisallowGarbageCollection* no_gc);
Isolate* isolate_;
Handle<Map> map_handle_;
Map map_;
MaybeObject raw_transitions_;
Encoding encoding_;
bool concurrent_access_;
#if DEBUG
bool needs_reload_;
#endif
DISALLOW_IMPLICIT_CONSTRUCTORS(TransitionsAccessor);
};

View File

@ -2218,16 +2218,11 @@ Maybe<uint32_t> ValueDeserializer::ReadJSObjectProperties(
// transition was found.
Handle<Object> key;
Handle<Map> target;
Handle<String> expected_key;
{
TransitionsAccessor transitions(isolate_, *map);
expected_key = transitions.ExpectedTransitionKey();
if (!expected_key.is_null()) {
target = transitions.ExpectedTransitionTarget();
}
}
TransitionsAccessor transitions(isolate_, map);
Handle<String> expected_key = transitions.ExpectedTransitionKey();
if (!expected_key.is_null() && ReadExpectedString(expected_key)) {
key = expected_key;
target = transitions.ExpectedTransitionTarget();
} else {
if (!ReadObject().ToHandle(&key) || !IsValidObjectKey(*key, isolate_)) {
return Nothing<uint32_t>();
@ -2236,7 +2231,7 @@ Maybe<uint32_t> ValueDeserializer::ReadJSObjectProperties(
key =
isolate_->factory()->InternalizeString(Handle<String>::cast(key));
// Don't reuse |transitions| because it could be stale.
transitioning = TransitionsAccessor(isolate_, *map)
transitioning = TransitionsAccessor(isolate_, map)
.FindTransitionToField(Handle<String>::cast(key))
.ToHandle(&target);
} else {

View File

@ -117,7 +117,7 @@ void GeneralizeAllTransitionsToFieldAsMutable(Isolate* isolate, Handle<Map> map,
// Collect all outgoing field transitions.
{
DisallowGarbageCollection no_gc;
TransitionsAccessor transitions(isolate, *map);
TransitionsAccessor transitions(isolate, *map, &no_gc);
transitions.ForEachTransitionTo(
*name,
[&](Map target) {

View File

@ -2880,7 +2880,8 @@ TEST(OptimizedAllocationArrayLiterals) {
}
static int CountMapTransitions(i::Isolate* isolate, Map map) {
return TransitionsAccessor(isolate, map).NumberOfTransitions();
DisallowGarbageCollection no_gc;
return TransitionsAccessor(isolate, map, &no_gc).NumberOfTransitions();
}

View File

@ -1122,7 +1122,7 @@ TEST(TransitionLookup) {
// Ensure we didn't overflow transition array and therefore all the
// combinations of cases are covered.
CHECK(TransitionsAccessor::CanHaveMoreTransitions(isolate, root_map));
CHECK(TransitionsAccessor(isolate, root_map).CanHaveMoreTransitions());
// Now try querying keys.
bool positive_lookup_tested = false;

View File

@ -40,7 +40,7 @@ class ConcurrentSearchThread : public v8::base::Thread {
background_thread_started_->Signal();
CHECK_EQ(TransitionsAccessor(CcTest::i_isolate(), *map_, true)
CHECK_EQ(TransitionsAccessor(CcTest::i_isolate(), map_, true)
.SearchTransition(*name_, PropertyKind::kData, NONE),
result_map_ ? **result_map_ : Map());
}
@ -74,11 +74,11 @@ class ConcurrentSearchOnOutdatedAccessorThread final
LocalHeap local_heap(heap_, ThreadKind::kBackground, std::move(ph_));
UnparkedScope scope(&local_heap);
TransitionsAccessor accessor(CcTest::i_isolate(), map_, true);
background_thread_started_->Signal();
main_thread_finished_->Wait();
CHECK_EQ(TransitionsAccessor(CcTest::i_isolate(), *map_, true)
.SearchTransition(*name_, PropertyKind::kData, NONE),
CHECK_EQ(accessor.SearchTransition(*name_, PropertyKind::kData, NONE),
result_map_ ? **result_map_ : Map());
}
@ -102,7 +102,7 @@ TEST(FullFieldTransitions_OnlySearch) {
attributes, PropertyConstness::kMutable,
Representation::Tagged(), OMIT_TRANSITION)
.ToHandleChecked();
TransitionsAccessor::Insert(isolate, map0, name, map1, PROPERTY_TRANSITION);
TransitionsAccessor(isolate, map0).Insert(name, map1, PROPERTY_TRANSITION);
{
TestTransitionsAccessor transitions(isolate, map0);
CHECK(transitions.IsFullTransitionArrayEncoding());
@ -124,9 +124,8 @@ TEST(FullFieldTransitions_OnlySearch) {
background_thread_started.Wait();
CHECK_EQ(*map1, *TransitionsAccessor::SearchTransition(isolate, map0, *name,
kind, attributes)
.ToHandleChecked());
CHECK_EQ(*map1, TransitionsAccessor(isolate, map0)
.SearchTransition(*name, kind, attributes));
thread->Join();
}
@ -155,7 +154,7 @@ TEST(FullFieldTransitions) {
attributes, PropertyConstness::kMutable,
Representation::Tagged(), OMIT_TRANSITION)
.ToHandleChecked();
TransitionsAccessor::Insert(isolate, map0, name1, map1, PROPERTY_TRANSITION);
TransitionsAccessor(isolate, map0).Insert(name1, map1, PROPERTY_TRANSITION);
{
TestTransitionsAccessor transitions(isolate, map0);
CHECK(transitions.IsFullTransitionArrayEncoding());
@ -177,13 +176,11 @@ TEST(FullFieldTransitions) {
background_thread_started.Wait();
CHECK_EQ(*map1, *TransitionsAccessor::SearchTransition(isolate, map0, *name1,
kind, attributes)
.ToHandleChecked());
TransitionsAccessor::Insert(isolate, map0, name2, map2, PROPERTY_TRANSITION);
CHECK_EQ(*map2, *TransitionsAccessor::SearchTransition(isolate, map0, *name2,
kind, attributes)
.ToHandleChecked());
CHECK_EQ(*map1, TransitionsAccessor(isolate, map0)
.SearchTransition(*name1, kind, attributes));
TransitionsAccessor(isolate, map0).Insert(name2, map2, PROPERTY_TRANSITION);
CHECK_EQ(*map2, TransitionsAccessor(isolate, map0)
.SearchTransition(*name2, kind, attributes));
thread->Join();
}
@ -213,8 +210,8 @@ TEST(WeakRefToFullFieldTransitions) {
attributes, PropertyConstness::kMutable,
Representation::Tagged(), OMIT_TRANSITION)
.ToHandleChecked();
TransitionsAccessor::Insert(isolate, map0, name1, map1,
SIMPLE_PROPERTY_TRANSITION);
TransitionsAccessor(isolate, map0)
.Insert(name1, map1, SIMPLE_PROPERTY_TRANSITION);
{
TestTransitionsAccessor transitions(isolate, map0);
CHECK(transitions.IsWeakRefEncoding());
@ -236,18 +233,16 @@ TEST(WeakRefToFullFieldTransitions) {
background_thread_started.Wait();
CHECK_EQ(*map1, *TransitionsAccessor::SearchTransition(isolate, map0, *name1,
kind, attributes)
.ToHandleChecked());
TransitionsAccessor::Insert(isolate, map0, name2, map2,
SIMPLE_PROPERTY_TRANSITION);
CHECK_EQ(*map1, TransitionsAccessor(isolate, map0)
.SearchTransition(*name1, kind, attributes));
TransitionsAccessor(isolate, map0)
.Insert(name2, map2, SIMPLE_PROPERTY_TRANSITION);
{
TestTransitionsAccessor transitions(isolate, map0);
CHECK(transitions.IsFullTransitionArrayEncoding());
}
CHECK_EQ(*map2, *TransitionsAccessor::SearchTransition(isolate, map0, *name2,
kind, attributes)
.ToHandleChecked());
CHECK_EQ(*map2, TransitionsAccessor(isolate, map0)
.SearchTransition(*name2, kind, attributes));
thread->Join();
}
@ -283,8 +278,8 @@ TEST(FullFieldTransitions_withSlack) {
attributes, PropertyConstness::kMutable,
Representation::Tagged(), OMIT_TRANSITION)
.ToHandleChecked();
TransitionsAccessor::Insert(isolate, map0, name1, map1, PROPERTY_TRANSITION);
TransitionsAccessor::Insert(isolate, map0, name2, map2, PROPERTY_TRANSITION);
TransitionsAccessor(isolate, map0).Insert(name1, map1, PROPERTY_TRANSITION);
TransitionsAccessor(isolate, map0).Insert(name2, map2, PROPERTY_TRANSITION);
{
TestTransitionsAccessor transitions(isolate, map0);
CHECK(transitions.IsFullTransitionArrayEncoding());
@ -306,22 +301,19 @@ TEST(FullFieldTransitions_withSlack) {
background_thread_started.Wait();
CHECK_EQ(*map1, *TransitionsAccessor::SearchTransition(isolate, map0, *name1,
kind, attributes)
.ToHandleChecked());
CHECK_EQ(*map2, *TransitionsAccessor::SearchTransition(isolate, map0, *name2,
kind, attributes)
.ToHandleChecked());
CHECK_EQ(*map1, TransitionsAccessor(isolate, map0)
.SearchTransition(*name1, kind, attributes));
CHECK_EQ(*map2, TransitionsAccessor(isolate, map0)
.SearchTransition(*name2, kind, attributes));
{
// Check that we have enough slack for the 3rd insertion into the
// TransitionArray.
TestTransitionsAccessor transitions(isolate, map0);
CHECK_GE(transitions.Capacity(), 3);
}
TransitionsAccessor::Insert(isolate, map0, name3, map3, PROPERTY_TRANSITION);
CHECK_EQ(*map3, *TransitionsAccessor::SearchTransition(isolate, map0, *name3,
kind, attributes)
.ToHandleChecked());
TransitionsAccessor(isolate, map0).Insert(name3, map3, PROPERTY_TRANSITION);
CHECK_EQ(*map3, TransitionsAccessor(isolate, map0)
.SearchTransition(*name3, kind, attributes));
thread->Join();
}
@ -367,10 +359,9 @@ TEST(UninitializedToFullFieldTransitions) {
background_thread_started.Wait();
TransitionsAccessor::Insert(isolate, map0, name1, map1, PROPERTY_TRANSITION);
CHECK_EQ(*map1, *TransitionsAccessor::SearchTransition(isolate, map0, *name1,
kind, attributes)
.ToHandleChecked());
TransitionsAccessor(isolate, map0).Insert(name1, map1, PROPERTY_TRANSITION);
CHECK_EQ(*map1, TransitionsAccessor(isolate, map0)
.SearchTransition(*name1, kind, attributes));
{
TestTransitionsAccessor transitions(isolate, map0);
CHECK(transitions.IsFullTransitionArrayEncoding());
@ -404,7 +395,7 @@ TEST(FullFieldTransitions_BackgroundSearchOldPointer) {
attributes, PropertyConstness::kMutable,
Representation::Tagged(), OMIT_TRANSITION)
.ToHandleChecked();
TransitionsAccessor::Insert(isolate, map0, name1, map1, PROPERTY_TRANSITION);
TransitionsAccessor(isolate, map0).Insert(name1, map1, PROPERTY_TRANSITION);
{
TestTransitionsAccessor transitions(isolate, map0);
CHECK(transitions.IsFullTransitionArrayEncoding());
@ -429,19 +420,17 @@ TEST(FullFieldTransitions_BackgroundSearchOldPointer) {
background_thread_started.Wait();
CHECK_EQ(*map1, *TransitionsAccessor::SearchTransition(isolate, map0, *name1,
kind, attributes)
.ToHandleChecked());
CHECK_EQ(*map1, TransitionsAccessor(isolate, map0)
.SearchTransition(*name1, kind, attributes));
{
// Check that we do not have enough slack for the 2nd insertion into the
// TransitionArray.
TestTransitionsAccessor transitions(isolate, map0);
CHECK_EQ(transitions.Capacity(), 1);
}
TransitionsAccessor::Insert(isolate, map0, name2, map2, PROPERTY_TRANSITION);
CHECK_EQ(*map2, *TransitionsAccessor::SearchTransition(isolate, map0, *name2,
kind, attributes)
.ToHandleChecked());
TransitionsAccessor(isolate, map0).Insert(name2, map2, PROPERTY_TRANSITION);
CHECK_EQ(*map2, TransitionsAccessor(isolate, map0)
.SearchTransition(*name2, kind, attributes));
main_thread_finished.Signal();
thread->Join();

View File

@ -66,7 +66,8 @@ static Handle<AccessorPair> CreateAccessorPair(bool with_getter,
// Check cached migration target map after Map::Update() and Map::TryUpdate()
static void CheckMigrationTarget(Isolate* isolate, Map old_map, Map new_map) {
Map target = TransitionsAccessor(isolate, old_map).GetMigrationTarget();
Map target = TransitionsAccessor(isolate, handle(old_map, isolate))
.GetMigrationTarget();
if (target.is_null()) return;
CHECK_EQ(new_map, target);
CHECK_EQ(MapUpdater::TryUpdateNoLock(isolate, old_map,
@ -390,10 +391,10 @@ class Expectations {
heap_type);
Handle<String> name = CcTest::MakeName("prop", property_index);
MaybeHandle<Map> target = TransitionsAccessor::SearchTransition(
isolate_, map, *name, PropertyKind::kData, attributes);
Map target = TransitionsAccessor(isolate_, map)
.SearchTransition(*name, PropertyKind::kData, attributes);
CHECK(!target.is_null());
return target.ToHandleChecked();
return handle(target, isolate_);
}
Handle<Map> AddAccessorConstant(Handle<Map> map,
@ -2064,10 +2065,10 @@ TEST(ReconfigurePropertySplitMapTransitionsOverflow) {
}
Handle<String> name = CcTest::MakeName("prop", i);
MaybeHandle<Map> target = TransitionsAccessor::SearchTransition(
isolate, map2, *name, PropertyKind::kData, NONE);
Map target = TransitionsAccessor(isolate, map2)
.SearchTransition(*name, PropertyKind::kData, NONE);
CHECK(!target.is_null());
map2 = target.ToHandleChecked();
map2 = handle(target, isolate);
}
map2 = ReconfigureProperty(isolate, map2, InternalIndex(kSplitProp),
@ -2089,14 +2090,14 @@ TEST(ReconfigurePropertySplitMapTransitionsOverflow) {
// Fill in transition tree of |map2| so that it can't have more transitions.
for (int i = 0; i < TransitionsAccessor::kMaxNumberOfTransitions; i++) {
CHECK(TransitionsAccessor::CanHaveMoreTransitions(isolate, map2));
CHECK(TransitionsAccessor(isolate, map2).CanHaveMoreTransitions());
Handle<String> name = CcTest::MakeName("foo", i);
Map::CopyWithField(isolate, map2, name, any_type, NONE,
PropertyConstness::kMutable, Representation::Smi(),
INSERT_TRANSITION)
.ToHandleChecked();
}
CHECK(!TransitionsAccessor::CanHaveMoreTransitions(isolate, map2));
CHECK(!TransitionsAccessor(isolate, map2).CanHaveMoreTransitions());
// Try to update |map|, since there is no place for propX transition at |map2|
// |map| should become normalized.
@ -3093,7 +3094,7 @@ TEST(DeletePropertyGeneralizesConstness) {
// |new_parent_map| must have exactly one outgoing transition to |new_map|.
{
TransitionsAccessor ta(isolate, *new_parent_map);
TransitionsAccessor ta(isolate, new_parent_map);
CHECK_EQ(ta.NumberOfTransitions(), 1);
CHECK_EQ(ta.GetTarget(0), *new_map);
}

View File

@ -45,22 +45,19 @@ TEST(TransitionArray_SimpleFieldTransitions) {
CHECK(map0->raw_transitions()->IsSmi());
{
TransitionsAccessor::Insert(isolate, map0, name1, map1,
SIMPLE_PROPERTY_TRANSITION);
TestTransitionsAccessor transitions(isolate, map0);
transitions.Insert(name1, map1, SIMPLE_PROPERTY_TRANSITION);
}
{
{
TestTransitionsAccessor transitions(isolate, map0);
CHECK(transitions.IsWeakRefEncoding());
CHECK_EQ(*map1, transitions.SearchTransition(*name1, PropertyKind::kData,
attributes));
CHECK_EQ(1, transitions.NumberOfTransitions());
CHECK_EQ(*name1, transitions.GetKey(0));
CHECK_EQ(*map1, transitions.GetTarget(0));
}
TestTransitionsAccessor transitions(isolate, map0);
CHECK(transitions.IsWeakRefEncoding());
CHECK_EQ(*map1, transitions.SearchTransition(*name1, PropertyKind::kData,
attributes));
CHECK_EQ(1, transitions.NumberOfTransitions());
CHECK_EQ(*name1, transitions.GetKey(0));
CHECK_EQ(*map1, transitions.GetTarget(0));
TransitionsAccessor::Insert(isolate, map0, name2, map2,
SIMPLE_PROPERTY_TRANSITION);
transitions.Insert(name2, map2, SIMPLE_PROPERTY_TRANSITION);
}
{
TestTransitionsAccessor transitions(isolate, map0);
@ -108,22 +105,19 @@ TEST(TransitionArray_FullFieldTransitions) {
CHECK(map0->raw_transitions()->IsSmi());
{
TransitionsAccessor::Insert(isolate, map0, name1, map1,
PROPERTY_TRANSITION);
TestTransitionsAccessor transitions(isolate, map0);
transitions.Insert(name1, map1, PROPERTY_TRANSITION);
}
{
{
TestTransitionsAccessor transitions(isolate, map0);
CHECK(transitions.IsFullTransitionArrayEncoding());
CHECK_EQ(*map1, transitions.SearchTransition(*name1, PropertyKind::kData,
attributes));
CHECK_EQ(1, transitions.NumberOfTransitions());
CHECK_EQ(*name1, transitions.GetKey(0));
CHECK_EQ(*map1, transitions.GetTarget(0));
}
TestTransitionsAccessor transitions(isolate, map0);
CHECK(transitions.IsFullTransitionArrayEncoding());
CHECK_EQ(*map1, transitions.SearchTransition(*name1, PropertyKind::kData,
attributes));
CHECK_EQ(1, transitions.NumberOfTransitions());
CHECK_EQ(*name1, transitions.GetKey(0));
CHECK_EQ(*map1, transitions.GetTarget(0));
TransitionsAccessor::Insert(isolate, map0, name2, map2,
PROPERTY_TRANSITION);
transitions.Insert(name2, map2, PROPERTY_TRANSITION);
}
{
TestTransitionsAccessor transitions(isolate, map0);
@ -172,10 +166,10 @@ TEST(TransitionArray_DifferentFieldNames) {
names[i] = name;
maps[i] = map;
TransitionsAccessor::Insert(isolate, map0, name, map, PROPERTY_TRANSITION);
TransitionsAccessor(isolate, map0).Insert(name, map, PROPERTY_TRANSITION);
}
TransitionsAccessor transitions(isolate, *map0);
TransitionsAccessor transitions(isolate, map0);
for (int i = 0; i < PROPS_COUNT; i++) {
CHECK_EQ(*maps[i], transitions.SearchTransition(
*names[i], PropertyKind::kData, attributes));
@ -220,11 +214,11 @@ TEST(TransitionArray_SameFieldNamesDifferentAttributesSimple) {
.ToHandleChecked();
attr_maps[i] = map;
TransitionsAccessor::Insert(isolate, map0, name, map, PROPERTY_TRANSITION);
TransitionsAccessor(isolate, map0).Insert(name, map, PROPERTY_TRANSITION);
}
// Ensure that transitions for |name| field are valid.
TransitionsAccessor transitions(isolate, *map0);
TransitionsAccessor transitions(isolate, map0);
for (int i = 0; i < ATTRS_COUNT; i++) {
PropertyAttributes attributes = static_cast<PropertyAttributes>(i);
CHECK_EQ(*attr_maps[i], transitions.SearchTransition(
@ -264,7 +258,7 @@ TEST(TransitionArray_SameFieldNamesDifferentAttributes) {
names[i] = name;
maps[i] = map;
TransitionsAccessor::Insert(isolate, map0, name, map, PROPERTY_TRANSITION);
TransitionsAccessor(isolate, map0).Insert(name, map, PROPERTY_TRANSITION);
}
const int ATTRS_COUNT = (READ_ONLY | DONT_ENUM | DONT_DELETE) + 1;
@ -283,11 +277,11 @@ TEST(TransitionArray_SameFieldNamesDifferentAttributes) {
.ToHandleChecked();
attr_maps[i] = map;
TransitionsAccessor::Insert(isolate, map0, name, map, PROPERTY_TRANSITION);
TransitionsAccessor(isolate, map0).Insert(name, map, PROPERTY_TRANSITION);
}
// Ensure that transitions for |name| field are valid.
TransitionsAccessor transitions(isolate, *map0);
TransitionsAccessor transitions(isolate, map0);
for (int i = 0; i < ATTRS_COUNT; i++) {
PropertyAttributes attr = static_cast<PropertyAttributes>(i);
CHECK_EQ(*attr_maps[i],

View File

@ -12,10 +12,11 @@ namespace internal {
class TestTransitionsAccessor : public TransitionsAccessor {
public:
TestTransitionsAccessor(Isolate* isolate, Map map)
: TransitionsAccessor(isolate, map) {}
TestTransitionsAccessor(Isolate* isolate, Map map,
DisallowGarbageCollection* no_gc)
: TransitionsAccessor(isolate, map, no_gc) {}
TestTransitionsAccessor(Isolate* isolate, Handle<Map> map)
: TransitionsAccessor(isolate, *map) {}
: TransitionsAccessor(isolate, map) {}
// Expose internals for tests.
bool IsUninitializedEncoding() { return encoding() == kUninitialized; }