PPC: Use Cells to check prototype chain validity (disabled by default).
Port 0179ec5797
Original commit message:
The cells are stored on prototypes (in their map's PrototypeInfo). When a prototype object changes its map, then both its own validity cell and those of all "downstream" prototypes are invalidated; handlers for a given receiver embed the currently valid cell for that receiver's prototype during their compilation and check it on execution.
R=michael_dawson@ca.ibm.com, dstence@us.ibm.com
BUG=
Review URL: https://codereview.chromium.org/1091563002
Cr-Commit-Position: refs/heads/master@{#27849}
This commit is contained in:
parent
e02807ee8a
commit
0dc5fd7080
@ -410,8 +410,8 @@ void NamedStoreHandlerCompiler::GenerateFieldTypeChecks(HeapType* field_type,
|
||||
|
||||
Register PropertyHandlerCompiler::CheckPrototypes(
|
||||
Register object_reg, Register holder_reg, Register scratch1,
|
||||
Register scratch2, Handle<Name> name, Label* miss,
|
||||
PrototypeCheckType check) {
|
||||
Register scratch2, Handle<Name> name, Label* miss, PrototypeCheckType check,
|
||||
ReturnHolder return_what) {
|
||||
Handle<Map> receiver_map = map();
|
||||
|
||||
// Make sure there's no overlap between holder and object registers.
|
||||
@ -419,6 +419,30 @@ Register PropertyHandlerCompiler::CheckPrototypes(
|
||||
DCHECK(!scratch2.is(object_reg) && !scratch2.is(holder_reg) &&
|
||||
!scratch2.is(scratch1));
|
||||
|
||||
if (FLAG_eliminate_prototype_chain_checks) {
|
||||
Handle<Cell> validity_cell =
|
||||
Map::GetOrCreatePrototypeChainValidityCell(receiver_map, isolate());
|
||||
if (!validity_cell.is_null()) {
|
||||
DCHECK_EQ(Smi::FromInt(Map::kPrototypeChainValid),
|
||||
validity_cell->value());
|
||||
__ mov(scratch1, Operand(validity_cell));
|
||||
__ LoadP(scratch1, FieldMemOperand(scratch1, Cell::kValueOffset));
|
||||
__ CmpSmiLiteral(scratch1, Smi::FromInt(Map::kPrototypeChainValid), r0);
|
||||
__ bne(miss);
|
||||
}
|
||||
|
||||
// The prototype chain of primitives (and their JSValue wrappers) depends
|
||||
// on the native context, which can't be guarded by validity cells.
|
||||
// |object_reg| holds the native context specific prototype in this case;
|
||||
// we need to check its map.
|
||||
if (check == CHECK_ALL_MAPS) {
|
||||
__ LoadP(scratch1, FieldMemOperand(object_reg, HeapObject::kMapOffset));
|
||||
Handle<WeakCell> cell = Map::WeakCellForMap(receiver_map);
|
||||
__ CmpWeakValue(scratch1, cell, scratch2);
|
||||
__ b(ne, miss);
|
||||
}
|
||||
}
|
||||
|
||||
// Keep track of the current object in register reg.
|
||||
Register reg = object_reg;
|
||||
int depth = 0;
|
||||
@ -462,39 +486,48 @@ Register PropertyHandlerCompiler::CheckPrototypes(
|
||||
current->property_dictionary()->FindEntry(name) ==
|
||||
NameDictionary::kNotFound);
|
||||
|
||||
if (FLAG_eliminate_prototype_chain_checks && depth > 1) {
|
||||
// TODO(jkummerow): Cache and re-use weak cell.
|
||||
__ LoadWeakValue(reg, isolate()->factory()->NewWeakCell(current), miss);
|
||||
}
|
||||
GenerateDictionaryNegativeLookup(masm(), miss, reg, name, scratch1,
|
||||
scratch2);
|
||||
|
||||
__ LoadP(scratch1, FieldMemOperand(reg, HeapObject::kMapOffset));
|
||||
reg = holder_reg; // From now on the object will be in holder_reg.
|
||||
__ LoadP(reg, FieldMemOperand(scratch1, Map::kPrototypeOffset));
|
||||
if (!FLAG_eliminate_prototype_chain_checks) {
|
||||
__ LoadP(scratch1, FieldMemOperand(reg, HeapObject::kMapOffset));
|
||||
__ LoadP(holder_reg, FieldMemOperand(scratch1, Map::kPrototypeOffset));
|
||||
}
|
||||
} else {
|
||||
Register map_reg = scratch1;
|
||||
__ LoadP(map_reg, FieldMemOperand(reg, HeapObject::kMapOffset));
|
||||
|
||||
if (!FLAG_eliminate_prototype_chain_checks) {
|
||||
__ LoadP(map_reg, FieldMemOperand(reg, HeapObject::kMapOffset));
|
||||
}
|
||||
if (current_map->IsJSGlobalObjectMap()) {
|
||||
GenerateCheckPropertyCell(masm(), Handle<JSGlobalObject>::cast(current),
|
||||
name, scratch2, miss);
|
||||
} else if (depth != 1 || check == CHECK_ALL_MAPS) {
|
||||
} else if (!FLAG_eliminate_prototype_chain_checks &&
|
||||
(depth != 1 || check == CHECK_ALL_MAPS)) {
|
||||
Handle<WeakCell> cell = Map::WeakCellForMap(current_map);
|
||||
__ CmpWeakValue(map_reg, cell, scratch2);
|
||||
__ bne(miss);
|
||||
}
|
||||
|
||||
reg = holder_reg; // From now on the object will be in holder_reg.
|
||||
|
||||
__ LoadP(reg, FieldMemOperand(map_reg, Map::kPrototypeOffset));
|
||||
if (!FLAG_eliminate_prototype_chain_checks) {
|
||||
__ LoadP(holder_reg, FieldMemOperand(map_reg, Map::kPrototypeOffset));
|
||||
}
|
||||
}
|
||||
|
||||
reg = holder_reg; // From now on the object will be in holder_reg.
|
||||
// Go to the next object in the prototype chain.
|
||||
current = prototype;
|
||||
current_map = handle(current->map());
|
||||
}
|
||||
|
||||
DCHECK(!current_map->IsJSGlobalProxyMap());
|
||||
|
||||
// Log the check depth.
|
||||
LOG(isolate(), IntEvent("check-maps-depth", depth + 1));
|
||||
|
||||
if (depth != 0 || check == CHECK_ALL_MAPS) {
|
||||
if (!FLAG_eliminate_prototype_chain_checks &&
|
||||
(depth != 0 || check == CHECK_ALL_MAPS)) {
|
||||
// Check the holder map.
|
||||
__ LoadP(scratch1, FieldMemOperand(reg, HeapObject::kMapOffset));
|
||||
Handle<WeakCell> cell = Map::WeakCellForMap(current_map);
|
||||
@ -502,8 +535,13 @@ Register PropertyHandlerCompiler::CheckPrototypes(
|
||||
__ bne(miss);
|
||||
}
|
||||
|
||||
bool return_holder = return_what == RETURN_HOLDER;
|
||||
if (FLAG_eliminate_prototype_chain_checks && return_holder && depth != 0) {
|
||||
__ LoadWeakValue(reg, isolate()->factory()->NewWeakCell(current), miss);
|
||||
}
|
||||
|
||||
// Return the register containing the holder.
|
||||
return reg;
|
||||
return return_holder ? reg : no_reg;
|
||||
}
|
||||
|
||||
|
||||
@ -716,7 +754,7 @@ Handle<Code> NamedLoadHandlerCompiler::CompileLoadGlobal(
|
||||
if (IC::ICUseVector(kind())) {
|
||||
PushVectorAndSlot();
|
||||
}
|
||||
FrontendHeader(receiver(), name, &miss);
|
||||
FrontendHeader(receiver(), name, &miss, DONT_RETURN_ANYTHING);
|
||||
|
||||
// Get the value from the cell.
|
||||
Register result = StoreDescriptor::ValueRegister();
|
||||
|
Loading…
Reference in New Issue
Block a user