2019-05-02 17:41:08 +00:00
|
|
|
// Copyright 2019 the V8 project authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
|
|
|
#include "src/compiler/map-inference.h"
|
|
|
|
|
|
|
|
#include "src/compiler/compilation-dependencies.h"
|
2019-08-26 07:28:22 +00:00
|
|
|
#include "src/compiler/feedback-source.h"
|
2019-05-02 17:41:08 +00:00
|
|
|
#include "src/compiler/js-graph.h"
|
|
|
|
#include "src/compiler/simplified-operator.h"
|
|
|
|
#include "src/objects/map-inl.h"
|
|
|
|
#include "src/zone/zone-handle-set.h"
|
|
|
|
|
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
|
|
|
namespace compiler {
|
|
|
|
|
|
|
|
MapInference::MapInference(JSHeapBroker* broker, Node* object, Node* effect)
|
|
|
|
: broker_(broker), object_(object) {
|
|
|
|
ZoneHandleSet<Map> maps;
|
|
|
|
auto result =
|
2019-07-01 17:16:26 +00:00
|
|
|
NodeProperties::InferReceiverMapsUnsafe(broker_, object_, effect, &maps);
|
2019-05-02 17:41:08 +00:00
|
|
|
maps_.insert(maps_.end(), maps.begin(), maps.end());
|
|
|
|
maps_state_ = (result == NodeProperties::kUnreliableReceiverMaps)
|
|
|
|
? kUnreliableDontNeedGuard
|
|
|
|
: kReliableOrGuarded;
|
|
|
|
DCHECK_EQ(maps_.empty(), result == NodeProperties::kNoReceiverMaps);
|
|
|
|
}
|
|
|
|
|
|
|
|
MapInference::~MapInference() { CHECK(Safe()); }
|
|
|
|
|
|
|
|
bool MapInference::Safe() const { return maps_state_ != kUnreliableNeedGuard; }
|
|
|
|
|
|
|
|
void MapInference::SetNeedGuardIfUnreliable() {
|
|
|
|
CHECK(HaveMaps());
|
|
|
|
if (maps_state_ == kUnreliableDontNeedGuard) {
|
|
|
|
maps_state_ = kUnreliableNeedGuard;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void MapInference::SetGuarded() { maps_state_ = kReliableOrGuarded; }
|
|
|
|
|
|
|
|
bool MapInference::HaveMaps() const { return !maps_.empty(); }
|
|
|
|
|
|
|
|
bool MapInference::AllOfInstanceTypesAreJSReceiver() const {
|
|
|
|
return AllOfInstanceTypesUnsafe(InstanceTypeChecker::IsJSReceiver);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MapInference::AllOfInstanceTypesAre(InstanceType type) const {
|
|
|
|
CHECK(!InstanceTypeChecker::IsString(type));
|
|
|
|
return AllOfInstanceTypesUnsafe(
|
|
|
|
[type](InstanceType other) { return type == other; });
|
|
|
|
}
|
|
|
|
|
2019-05-03 12:52:42 +00:00
|
|
|
bool MapInference::AnyOfInstanceTypesAre(InstanceType type) const {
|
|
|
|
CHECK(!InstanceTypeChecker::IsString(type));
|
|
|
|
return AnyOfInstanceTypesUnsafe(
|
|
|
|
[type](InstanceType other) { return type == other; });
|
|
|
|
}
|
|
|
|
|
2019-05-02 17:41:08 +00:00
|
|
|
bool MapInference::AllOfInstanceTypes(std::function<bool(InstanceType)> f) {
|
|
|
|
SetNeedGuardIfUnreliable();
|
|
|
|
return AllOfInstanceTypesUnsafe(f);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MapInference::AllOfInstanceTypesUnsafe(
|
|
|
|
std::function<bool(InstanceType)> f) const {
|
|
|
|
CHECK(HaveMaps());
|
2019-05-09 14:58:44 +00:00
|
|
|
|
2019-06-13 12:16:53 +00:00
|
|
|
auto instance_type = [this, f](Handle<Map> map) {
|
|
|
|
MapRef map_ref(broker_, map);
|
|
|
|
return f(map_ref.instance_type());
|
|
|
|
};
|
|
|
|
return std::all_of(maps_.begin(), maps_.end(), instance_type);
|
2019-05-02 17:41:08 +00:00
|
|
|
}
|
|
|
|
|
2019-05-03 12:52:42 +00:00
|
|
|
bool MapInference::AnyOfInstanceTypesUnsafe(
|
|
|
|
std::function<bool(InstanceType)> f) const {
|
|
|
|
CHECK(HaveMaps());
|
2019-05-09 14:58:44 +00:00
|
|
|
|
2019-06-13 12:16:53 +00:00
|
|
|
auto instance_type = [this, f](Handle<Map> map) {
|
|
|
|
MapRef map_ref(broker_, map);
|
|
|
|
return f(map_ref.instance_type());
|
|
|
|
};
|
|
|
|
|
|
|
|
return std::any_of(maps_.begin(), maps_.end(), instance_type);
|
2019-05-03 12:52:42 +00:00
|
|
|
}
|
|
|
|
|
2019-05-02 17:41:08 +00:00
|
|
|
MapHandles const& MapInference::GetMaps() {
|
|
|
|
SetNeedGuardIfUnreliable();
|
|
|
|
return maps_;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MapInference::InsertMapChecks(JSGraph* jsgraph, Node** effect,
|
|
|
|
Node* control,
|
2019-08-26 07:28:22 +00:00
|
|
|
const FeedbackSource& feedback) {
|
2019-05-02 17:41:08 +00:00
|
|
|
CHECK(HaveMaps());
|
|
|
|
CHECK(feedback.IsValid());
|
|
|
|
ZoneHandleSet<Map> maps;
|
|
|
|
for (Handle<Map> map : maps_) maps.insert(map, jsgraph->graph()->zone());
|
|
|
|
*effect = jsgraph->graph()->NewNode(
|
|
|
|
jsgraph->simplified()->CheckMaps(CheckMapsFlag::kNone, maps, feedback),
|
|
|
|
object_, *effect, control);
|
|
|
|
SetGuarded();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MapInference::RelyOnMapsViaStability(
|
|
|
|
CompilationDependencies* dependencies) {
|
|
|
|
CHECK(HaveMaps());
|
|
|
|
return RelyOnMapsHelper(dependencies, nullptr, nullptr, nullptr, {});
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MapInference::RelyOnMapsPreferStability(
|
|
|
|
CompilationDependencies* dependencies, JSGraph* jsgraph, Node** effect,
|
2019-08-26 07:28:22 +00:00
|
|
|
Node* control, const FeedbackSource& feedback) {
|
2019-05-02 17:41:08 +00:00
|
|
|
CHECK(HaveMaps());
|
|
|
|
if (Safe()) return false;
|
|
|
|
if (RelyOnMapsViaStability(dependencies)) return true;
|
|
|
|
CHECK(RelyOnMapsHelper(nullptr, jsgraph, effect, control, feedback));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool MapInference::RelyOnMapsHelper(CompilationDependencies* dependencies,
|
|
|
|
JSGraph* jsgraph, Node** effect,
|
|
|
|
Node* control,
|
2019-08-26 07:28:22 +00:00
|
|
|
const FeedbackSource& feedback) {
|
2019-05-02 17:41:08 +00:00
|
|
|
if (Safe()) return true;
|
|
|
|
|
2019-06-12 14:58:02 +00:00
|
|
|
auto is_stable = [this](Handle<Map> map) {
|
|
|
|
MapRef map_ref(broker_, map);
|
|
|
|
return map_ref.is_stable();
|
|
|
|
};
|
2019-05-02 17:41:08 +00:00
|
|
|
if (dependencies != nullptr &&
|
|
|
|
std::all_of(maps_.cbegin(), maps_.cend(), is_stable)) {
|
|
|
|
for (Handle<Map> map : maps_) {
|
|
|
|
dependencies->DependOnStableMap(MapRef(broker_, map));
|
|
|
|
}
|
|
|
|
SetGuarded();
|
|
|
|
return true;
|
|
|
|
} else if (feedback.IsValid()) {
|
|
|
|
InsertMapChecks(jsgraph, effect, control, feedback);
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Reduction MapInference::NoChange() {
|
|
|
|
SetGuarded();
|
|
|
|
maps_.clear(); // Just to make some CHECKs fail if {this} gets used after.
|
|
|
|
return Reducer::NoChange();
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace compiler
|
|
|
|
} // namespace internal
|
|
|
|
} // namespace v8
|