Commit 1737b2c7 authored by bmeurer's avatar bmeurer Committed by Commit bot

[turbofan] Introduce LoadFunctionPrototype simplified operator.

Add a LoadFunctionPrototype simplified operator, similar to what
Crankshaft has, that loads the prototype property of a constructor
function.

R=jarin@chromium.org
BUG=v8:5267

Review-Url: https://codereview.chromium.org/2517913002
Cr-Commit-Position: refs/heads/master@{#41127}
parent f31e8054
...@@ -94,6 +94,13 @@ PropertyAccessInfo PropertyAccessInfo::AccessorConstant( ...@@ -94,6 +94,13 @@ PropertyAccessInfo PropertyAccessInfo::AccessorConstant(
return PropertyAccessInfo(kAccessorConstant, holder, constant, receiver_maps); return PropertyAccessInfo(kAccessorConstant, holder, constant, receiver_maps);
} }
// static
PropertyAccessInfo PropertyAccessInfo::FunctionPrototype(
MapList const& receiver_maps) {
return PropertyAccessInfo(kFunctionPrototype, MaybeHandle<JSObject>(),
Handle<Object>(), receiver_maps);
}
// static // static
PropertyAccessInfo PropertyAccessInfo::Generic(MapList const& receiver_maps) { PropertyAccessInfo PropertyAccessInfo::Generic(MapList const& receiver_maps) {
return PropertyAccessInfo(kGeneric, MaybeHandle<JSObject>(), Handle<Object>(), return PropertyAccessInfo(kGeneric, MaybeHandle<JSObject>(), Handle<Object>(),
...@@ -144,9 +151,6 @@ bool PropertyAccessInfo::Merge(PropertyAccessInfo const* that) { ...@@ -144,9 +151,6 @@ bool PropertyAccessInfo::Merge(PropertyAccessInfo const* that) {
case kInvalid: case kInvalid:
break; break;
case kNotFound:
return true;
case kDataField: { case kDataField: {
// Check if we actually access the same field. // Check if we actually access the same field.
if (this->transition_map_.address() == that->transition_map_.address() && if (this->transition_map_.address() == that->transition_map_.address() &&
...@@ -173,6 +177,9 @@ bool PropertyAccessInfo::Merge(PropertyAccessInfo const* that) { ...@@ -173,6 +177,9 @@ bool PropertyAccessInfo::Merge(PropertyAccessInfo const* that) {
} }
return false; return false;
} }
case kNotFound:
case kFunctionPrototype:
case kGeneric: { case kGeneric: {
this->receiver_maps_.insert(this->receiver_maps_.end(), this->receiver_maps_.insert(this->receiver_maps_.end(),
that->receiver_maps_.begin(), that->receiver_maps_.begin(),
...@@ -449,6 +456,13 @@ bool AccessInfoFactory::ComputePropertyAccessInfos( ...@@ -449,6 +456,13 @@ bool AccessInfoFactory::ComputePropertyAccessInfos(
bool AccessInfoFactory::LookupSpecialFieldAccessor( bool AccessInfoFactory::LookupSpecialFieldAccessor(
Handle<Map> map, Handle<Name> name, PropertyAccessInfo* access_info) { Handle<Map> map, Handle<Name> name, PropertyAccessInfo* access_info) {
// Check for Function::prototype accessor.
if (map->IsJSFunctionMap() && map->is_constructor() &&
name.is_identical_to(factory()->prototype_string())) {
DCHECK(!map->has_non_instance_prototype());
*access_info = PropertyAccessInfo::FunctionPrototype(MapList{map});
return true;
}
// Check for special JSObject field accessors. // Check for special JSObject field accessors.
int offset; int offset;
if (Accessors::IsJSObjectFieldAccessor(map, name, &offset)) { if (Accessors::IsJSObjectFieldAccessor(map, name, &offset)) {
......
...@@ -62,6 +62,7 @@ class PropertyAccessInfo final { ...@@ -62,6 +62,7 @@ class PropertyAccessInfo final {
kDataConstant, kDataConstant,
kDataField, kDataField,
kAccessorConstant, kAccessorConstant,
kFunctionPrototype,
kGeneric kGeneric
}; };
...@@ -79,6 +80,7 @@ class PropertyAccessInfo final { ...@@ -79,6 +80,7 @@ class PropertyAccessInfo final {
static PropertyAccessInfo AccessorConstant(MapList const& receiver_maps, static PropertyAccessInfo AccessorConstant(MapList const& receiver_maps,
Handle<Object> constant, Handle<Object> constant,
MaybeHandle<JSObject> holder); MaybeHandle<JSObject> holder);
static PropertyAccessInfo FunctionPrototype(MapList const& receiver_maps);
static PropertyAccessInfo Generic(MapList const& receiver_maps); static PropertyAccessInfo Generic(MapList const& receiver_maps);
PropertyAccessInfo(); PropertyAccessInfo();
...@@ -89,6 +91,7 @@ class PropertyAccessInfo final { ...@@ -89,6 +91,7 @@ class PropertyAccessInfo final {
bool IsDataConstant() const { return kind() == kDataConstant; } bool IsDataConstant() const { return kind() == kDataConstant; }
bool IsDataField() const { return kind() == kDataField; } bool IsDataField() const { return kind() == kDataField; }
bool IsAccessorConstant() const { return kind() == kAccessorConstant; } bool IsAccessorConstant() const { return kind() == kAccessorConstant; }
bool IsFunctionPrototype() const { return kind() == kFunctionPrototype; }
bool IsGeneric() const { return kind() == kGeneric; } bool IsGeneric() const { return kind() == kGeneric; }
bool HasTransitionMap() const { return !transition_map().is_null(); } bool HasTransitionMap() const { return !transition_map().is_null(); }
......
...@@ -784,6 +784,9 @@ bool EffectControlLinearizer::TryWireInStateEffect(Node* node, ...@@ -784,6 +784,9 @@ bool EffectControlLinearizer::TryWireInStateEffect(Node* node,
case IrOpcode::kStoreTypedElement: case IrOpcode::kStoreTypedElement:
state = LowerStoreTypedElement(node, *effect, *control); state = LowerStoreTypedElement(node, *effect, *control);
break; break;
case IrOpcode::kLoadFunctionPrototype:
state = LowerLoadFunctionPrototype(node, *effect, *control);
break;
case IrOpcode::kFloat64RoundUp: case IrOpcode::kFloat64RoundUp:
state = LowerFloat64RoundUp(node, *effect, *control); state = LowerFloat64RoundUp(node, *effect, *control);
break; break;
...@@ -3273,6 +3276,50 @@ EffectControlLinearizer::LowerStoreTypedElement(Node* node, Node* effect, ...@@ -3273,6 +3276,50 @@ EffectControlLinearizer::LowerStoreTypedElement(Node* node, Node* effect,
return ValueEffectControl(nullptr, effect, control); return ValueEffectControl(nullptr, effect, control);
} }
EffectControlLinearizer::ValueEffectControl
EffectControlLinearizer::LowerLoadFunctionPrototype(Node* node, Node* effect,
Node* control) {
Node* function = node->InputAt(0);
// Load the {JSFunction::prototype-or-initial-map} field.
Node* function_prototype_or_initial_map = effect =
graph()->NewNode(simplified()->LoadField(
AccessBuilder::ForJSFunctionPrototypeOrInitialMap()),
function, effect, control);
Node* function_prototype_or_initial_map_map = effect =
graph()->NewNode(simplified()->LoadField(AccessBuilder::ForMap()),
function, effect, control);
// Check if the {function} has an initial map.
Node* check0 = graph()->NewNode(
machine()->WordEqual(), function_prototype_or_initial_map_map,
jsgraph()->HeapConstant(factory()->meta_map()));
Node* branch0 =
graph()->NewNode(common()->Branch(BranchHint::kTrue), check0, control);
Node* if_true0 = graph()->NewNode(common()->IfTrue(), branch0);
Node* etrue0 = effect;
Node* vtrue0;
{
// Load the "prototype" from the initial map.
vtrue0 = etrue0 = graph()->NewNode(
simplified()->LoadField(AccessBuilder::ForMapPrototype()),
function_prototype_or_initial_map, etrue0, if_true0);
}
Node* if_false0 = graph()->NewNode(common()->IfFalse(), branch0);
Node* efalse0 = effect;
Node* vfalse0 = function_prototype_or_initial_map;
control = graph()->NewNode(common()->Merge(2), if_true0, if_false0);
effect = graph()->NewNode(common()->EffectPhi(2), etrue0, efalse0, control);
Node* value =
graph()->NewNode(common()->Phi(MachineRepresentation::kTaggedPointer, 2),
vtrue0, vfalse0, control);
return ValueEffectControl(value, effect, control);
}
EffectControlLinearizer::ValueEffectControl EffectControlLinearizer::ValueEffectControl
EffectControlLinearizer::LowerFloat64RoundUp(Node* node, Node* effect, EffectControlLinearizer::LowerFloat64RoundUp(Node* node, Node* effect,
Node* control) { Node* control) {
......
...@@ -178,6 +178,8 @@ class V8_EXPORT_PRIVATE EffectControlLinearizer { ...@@ -178,6 +178,8 @@ class V8_EXPORT_PRIVATE EffectControlLinearizer {
Node* control); Node* control);
ValueEffectControl LowerStoreTypedElement(Node* node, Node* effect, ValueEffectControl LowerStoreTypedElement(Node* node, Node* effect,
Node* control); Node* control);
ValueEffectControl LowerLoadFunctionPrototype(Node* node, Node* effect,
Node* control);
// Lowering of optional operators. // Lowering of optional operators.
ValueEffectControl LowerFloat64RoundUp(Node* node, Node* effect, ValueEffectControl LowerFloat64RoundUp(Node* node, Node* effect,
......
...@@ -1194,6 +1194,10 @@ JSNativeContextSpecialization::BuildPropertyAccess( ...@@ -1194,6 +1194,10 @@ JSNativeContextSpecialization::BuildPropertyAccess(
jsgraph()->UndefinedConstant(), effect); jsgraph()->UndefinedConstant(), effect);
} }
} }
} else if (access_info.IsFunctionPrototype()) {
DCHECK_EQ(AccessMode::kLoad, access_mode);
value = effect = graph()->NewNode(simplified()->LoadFunctionPrototype(),
receiver, effect, control);
} else { } else {
DCHECK(access_info.IsGeneric()); DCHECK(access_info.IsGeneric());
DCHECK_EQ(AccessMode::kStore, access_mode); DCHECK_EQ(AccessMode::kStore, access_mode);
......
...@@ -317,6 +317,7 @@ ...@@ -317,6 +317,7 @@
V(StoreBuffer) \ V(StoreBuffer) \
V(StoreElement) \ V(StoreElement) \
V(StoreTypedElement) \ V(StoreTypedElement) \
V(LoadFunctionPrototype) \
V(ObjectIsCallable) \ V(ObjectIsCallable) \
V(ObjectIsNumber) \ V(ObjectIsNumber) \
V(ObjectIsReceiver) \ V(ObjectIsReceiver) \
......
...@@ -2385,6 +2385,12 @@ class RepresentationSelector { ...@@ -2385,6 +2385,12 @@ class RepresentationSelector {
SetOutput(node, MachineRepresentation::kNone); SetOutput(node, MachineRepresentation::kNone);
return; return;
} }
case IrOpcode::kLoadFunctionPrototype: {
if (truncation.IsUnused()) return VisitUnused(node);
VisitUnop(node, UseInfo::AnyTagged(),
MachineRepresentation::kTaggedPointer);
return;
}
case IrOpcode::kPlainPrimitiveToNumber: { case IrOpcode::kPlainPrimitiveToNumber: {
if (InputIs(node, Type::Boolean())) { if (InputIs(node, Type::Boolean())) {
VisitUnop(node, UseInfo::Bool(), MachineRepresentation::kWord32); VisitUnop(node, UseInfo::Bool(), MachineRepresentation::kWord32);
......
...@@ -495,6 +495,13 @@ struct SimplifiedOperatorGlobalCache final { ...@@ -495,6 +495,13 @@ struct SimplifiedOperatorGlobalCache final {
}; };
ArrayBufferWasNeuteredOperator kArrayBufferWasNeutered; ArrayBufferWasNeuteredOperator kArrayBufferWasNeutered;
struct LoadFunctionPrototypeOperator final : public Operator {
LoadFunctionPrototypeOperator()
: Operator(IrOpcode::kLoadFunctionPrototype, Operator::kEliminatable,
"LoadFunctionPrototype", 1, 1, 1, 1, 1, 0) {}
};
LoadFunctionPrototypeOperator kLoadFunctionPrototype;
template <CheckForMinusZeroMode kMode> template <CheckForMinusZeroMode kMode>
struct CheckedInt32MulOperator final struct CheckedInt32MulOperator final
: public Operator1<CheckForMinusZeroMode> { : public Operator1<CheckForMinusZeroMode> {
...@@ -639,6 +646,7 @@ SimplifiedOperatorBuilder::SimplifiedOperatorBuilder(Zone* zone) ...@@ -639,6 +646,7 @@ SimplifiedOperatorBuilder::SimplifiedOperatorBuilder(Zone* zone)
PURE_OP_LIST(GET_FROM_CACHE) PURE_OP_LIST(GET_FROM_CACHE)
CHECKED_OP_LIST(GET_FROM_CACHE) CHECKED_OP_LIST(GET_FROM_CACHE)
GET_FROM_CACHE(ArrayBufferWasNeutered) GET_FROM_CACHE(ArrayBufferWasNeutered)
GET_FROM_CACHE(LoadFunctionPrototype)
#undef GET_FROM_CACHE #undef GET_FROM_CACHE
const Operator* SimplifiedOperatorBuilder::CheckedInt32Mul( const Operator* SimplifiedOperatorBuilder::CheckedInt32Mul(
......
...@@ -390,6 +390,9 @@ class V8_EXPORT_PRIVATE SimplifiedOperatorBuilder final ...@@ -390,6 +390,9 @@ class V8_EXPORT_PRIVATE SimplifiedOperatorBuilder final
// store-typed-element buffer, [base + external + index], value // store-typed-element buffer, [base + external + index], value
const Operator* StoreTypedElement(ExternalArrayType const&); const Operator* StoreTypedElement(ExternalArrayType const&);
// load-function-prototype function
const Operator* LoadFunctionPrototype();
private: private:
Zone* zone() const { return zone_; } Zone* zone() const { return zone_; }
......
...@@ -1730,6 +1730,10 @@ Type* Typer::Visitor::TypeStoreTypedElement(Node* node) { ...@@ -1730,6 +1730,10 @@ Type* Typer::Visitor::TypeStoreTypedElement(Node* node) {
return nullptr; return nullptr;
} }
Type* Typer::Visitor::TypeLoadFunctionPrototype(Node* node) {
return Type::NonInternal();
}
Type* Typer::Visitor::TypeObjectIsCallable(Node* node) { Type* Typer::Visitor::TypeObjectIsCallable(Node* node) {
return TypeUnaryOp(node, ObjectIsCallable); return TypeUnaryOp(node, ObjectIsCallable);
} }
......
...@@ -1129,6 +1129,10 @@ void Verifier::Visitor::Check(Node* node) { ...@@ -1129,6 +1129,10 @@ void Verifier::Visitor::Check(Node* node) {
case IrOpcode::kStoreTypedElement: case IrOpcode::kStoreTypedElement:
CheckNotTyped(node); CheckNotTyped(node);
break; break;
case IrOpcode::kLoadFunctionPrototype:
CheckValueInputIs(node, 0, Type::Any());
CheckTypeIs(node, Type::NonInternal());
break;
case IrOpcode::kNumberSilenceNaN: case IrOpcode::kNumberSilenceNaN:
CheckValueInputIs(node, 0, Type::Number()); CheckValueInputIs(node, 0, Type::Number());
CheckTypeIs(node, Type::Number()); CheckTypeIs(node, Type::Number());
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment