[TurboFan] Prefer ProcessedFeedback over raw serialized Vector reads
Almost everywhere, we use the ProcessedFeedback abstraction to query feedback in the compiler. The remaining exception is in handling RegExp, Object and Array literals. By bringing this in line with other feedback queries, we no longer need to serialize all feedback vector slots (possibly wasteful of memory), and offer a uniform way to pick up feedback everywhere. Bug: v8:7790, v8:9396 Change-Id: Ice42587595fe30bebfbd7835d2b2e9e49601c92e Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/1807358 Commit-Queue: Michael Stanton <mvstanton@chromium.org> Reviewed-by: Georg Neis <neis@chromium.org> Reviewed-by: Maya Lekova <mslekova@chromium.org> Cr-Commit-Position: refs/heads/master@{#64016}
This commit is contained in:
parent
f8a08223d6
commit
942193f08e
@ -24,7 +24,7 @@
|
||||
#include "src/objects/literal-objects-inl.h"
|
||||
#include "src/objects/objects-inl.h"
|
||||
#include "src/objects/smi.h"
|
||||
#include "src/objects/template-objects-inl.h"
|
||||
#include "src/objects/template-objects.h"
|
||||
|
||||
namespace v8 {
|
||||
namespace internal {
|
||||
@ -2090,12 +2090,13 @@ void BytecodeGraphBuilder::VisitCloneObject() {
|
||||
|
||||
void BytecodeGraphBuilder::VisitGetTemplateObject() {
|
||||
DisallowHeapAccessIf no_heap_access(FLAG_concurrent_inlining);
|
||||
FeedbackSlot slot = bytecode_iterator().GetSlotOperand(1);
|
||||
ObjectRef description(
|
||||
FeedbackSource source =
|
||||
CreateFeedbackSource(bytecode_iterator().GetIndexOperand(1));
|
||||
TemplateObjectDescriptionRef description(
|
||||
broker(), bytecode_iterator().GetConstantForIndexOperand(0, isolate()));
|
||||
JSArrayRef template_object =
|
||||
shared_info().GetTemplateObject(description, feedback_vector(), slot);
|
||||
environment()->BindAccumulator(jsgraph()->Constant(template_object));
|
||||
Node* template_object = NewNode(javascript()->GetTemplateObject(
|
||||
description.object(), shared_info().object(), source));
|
||||
environment()->BindAccumulator(template_object);
|
||||
}
|
||||
|
||||
Node* const* BytecodeGraphBuilder::GetCallArgumentsFromRegisters(
|
||||
|
@ -29,7 +29,6 @@ class NativeContext;
|
||||
class ScriptContextTable;
|
||||
|
||||
namespace compiler {
|
||||
|
||||
// Whether we are loading a property or storing to a property.
|
||||
// For a store during literal creation, do not walk up the prototype chain.
|
||||
enum class AccessMode { kLoad, kStore, kStoreInLiteral, kHas };
|
||||
@ -95,10 +94,12 @@ enum class OddballType : uint8_t {
|
||||
V(PropertyCell) \
|
||||
V(SharedFunctionInfo) \
|
||||
V(SourceTextModule) \
|
||||
V(TemplateObjectDescription) \
|
||||
/* Subtypes of Object */ \
|
||||
V(HeapObject)
|
||||
|
||||
class CompilationDependencies;
|
||||
struct FeedbackSource;
|
||||
class JSHeapBroker;
|
||||
class ObjectData;
|
||||
class PerIsolateCompilerCache;
|
||||
@ -342,6 +343,8 @@ class JSRegExpRef : public JSObjectRef {
|
||||
ObjectRef source() const;
|
||||
ObjectRef flags() const;
|
||||
ObjectRef last_index() const;
|
||||
|
||||
void SerializeAsRegExpBoilerplate();
|
||||
};
|
||||
|
||||
class HeapNumberRef : public HeapObjectRef {
|
||||
@ -494,7 +497,6 @@ class FeedbackVectorRef : public HeapObjectRef {
|
||||
double invocation_count() const;
|
||||
|
||||
void Serialize();
|
||||
ObjectRef get(FeedbackSlot slot) const;
|
||||
FeedbackCellRef GetClosureFeedbackCell(int index) const;
|
||||
};
|
||||
|
||||
@ -533,6 +535,9 @@ class AllocationSiteRef : public HeapObjectRef {
|
||||
//
|
||||
// If PointsToLiteral() is false, then IsFastLiteral() is also false.
|
||||
bool IsFastLiteral() const;
|
||||
|
||||
void SerializeBoilerplate();
|
||||
|
||||
// We only serialize boilerplate if IsFastLiteral is true.
|
||||
base::Optional<JSObjectRef> boilerplate() const;
|
||||
|
||||
@ -789,7 +794,7 @@ class V8_EXPORT_PRIVATE SharedFunctionInfoRef : public HeapObjectRef {
|
||||
// wraps the retrieval of the template object and creates it if
|
||||
// necessary.
|
||||
JSArrayRef GetTemplateObject(
|
||||
ObjectRef description, FeedbackVectorRef vector, FeedbackSlot slot,
|
||||
TemplateObjectDescriptionRef description, FeedbackSource const& source,
|
||||
SerializationPolicy policy = SerializationPolicy::kAssumeSerialized);
|
||||
|
||||
void SerializeFunctionTemplateInfo();
|
||||
@ -843,6 +848,13 @@ class SourceTextModuleRef : public HeapObjectRef {
|
||||
base::Optional<CellRef> GetCell(int cell_index) const;
|
||||
};
|
||||
|
||||
class TemplateObjectDescriptionRef : public HeapObjectRef {
|
||||
public:
|
||||
DEFINE_REF_CONSTRUCTOR(TemplateObjectDescription, HeapObjectRef)
|
||||
|
||||
Handle<TemplateObjectDescription> object() const;
|
||||
};
|
||||
|
||||
class CellRef : public HeapObjectRef {
|
||||
public:
|
||||
DEFINE_REF_CONSTRUCTOR(Cell, HeapObjectRef)
|
||||
|
@ -26,6 +26,7 @@
|
||||
#include "src/objects/js-promise.h"
|
||||
#include "src/objects/js-regexp-inl.h"
|
||||
#include "src/objects/objects-inl.h"
|
||||
#include "src/objects/template-objects.h"
|
||||
|
||||
namespace v8 {
|
||||
namespace internal {
|
||||
@ -84,6 +85,8 @@ Reduction JSCreateLowering::Reduce(Node* node) {
|
||||
return ReduceJSCreateLiteralArrayOrObject(node);
|
||||
case IrOpcode::kJSCreateLiteralRegExp:
|
||||
return ReduceJSCreateLiteralRegExp(node);
|
||||
case IrOpcode::kJSGetTemplateObject:
|
||||
return ReduceJSGetTemplateObject(node);
|
||||
case IrOpcode::kJSCreateEmptyLiteralArray:
|
||||
return ReduceJSCreateEmptyLiteralArray(node);
|
||||
case IrOpcode::kJSCreateEmptyLiteralObject:
|
||||
@ -1073,15 +1076,10 @@ Reduction JSCreateLowering::ReduceJSCreateLiteralArrayOrObject(Node* node) {
|
||||
CreateLiteralParameters const& p = CreateLiteralParametersOf(node->op());
|
||||
Node* effect = NodeProperties::GetEffectInput(node);
|
||||
Node* control = NodeProperties::GetControlInput(node);
|
||||
|
||||
FeedbackVectorRef feedback_vector(broker(), p.feedback().vector);
|
||||
ObjectRef feedback = feedback_vector.get(p.feedback().slot);
|
||||
// TODO(turbofan): we should consider creating a ProcessedFeedback for
|
||||
// allocation sites/boiler plates so that we use GetFeedback here. Then
|
||||
// we can eventually get rid of the additional copy of feedback slots that
|
||||
// we currently have in FeedbackVectorData.
|
||||
if (feedback.IsAllocationSite()) {
|
||||
AllocationSiteRef site = feedback.AsAllocationSite();
|
||||
ProcessedFeedback const& feedback =
|
||||
broker()->GetFeedbackForArrayOrObjectLiteral(p.feedback());
|
||||
if (!feedback.IsInsufficient()) {
|
||||
AllocationSiteRef site = feedback.AsLiteral().value();
|
||||
if (site.IsFastLiteral()) {
|
||||
AllocationType allocation = AllocationType::kYoung;
|
||||
if (FLAG_allocation_site_pretenuring) {
|
||||
@ -1095,20 +1093,17 @@ Reduction JSCreateLowering::ReduceJSCreateLiteralArrayOrObject(Node* node) {
|
||||
return Replace(value);
|
||||
}
|
||||
}
|
||||
|
||||
return NoChange();
|
||||
}
|
||||
|
||||
Reduction JSCreateLowering::ReduceJSCreateEmptyLiteralArray(Node* node) {
|
||||
DCHECK_EQ(IrOpcode::kJSCreateEmptyLiteralArray, node->opcode());
|
||||
FeedbackParameter const& p = FeedbackParameterOf(node->op());
|
||||
FeedbackVectorRef fv(broker(), p.feedback().vector);
|
||||
ObjectRef feedback = fv.get(p.feedback().slot);
|
||||
// TODO(turbofan): we should consider creating a ProcessedFeedback for
|
||||
// allocation sites/boiler plates so that we use GetFeedback here. Then
|
||||
// we can eventually get rid of the additional copy of feedback slots that
|
||||
// we currently have in FeedbackVectorData.
|
||||
if (feedback.IsAllocationSite()) {
|
||||
AllocationSiteRef site = feedback.AsAllocationSite();
|
||||
ProcessedFeedback const& feedback =
|
||||
broker()->GetFeedbackForArrayOrObjectLiteral(p.feedback());
|
||||
if (!feedback.IsInsufficient()) {
|
||||
AllocationSiteRef site = feedback.AsLiteral().value();
|
||||
DCHECK(!site.PointsToLiteral());
|
||||
MapRef initial_map =
|
||||
native_context().GetInitialJSArrayMap(site.GetElementsKind());
|
||||
@ -1162,22 +1157,30 @@ Reduction JSCreateLowering::ReduceJSCreateLiteralRegExp(Node* node) {
|
||||
CreateLiteralParameters const& p = CreateLiteralParametersOf(node->op());
|
||||
Node* effect = NodeProperties::GetEffectInput(node);
|
||||
Node* control = NodeProperties::GetControlInput(node);
|
||||
|
||||
FeedbackVectorRef feedback_vector(broker(), p.feedback().vector);
|
||||
ObjectRef feedback = feedback_vector.get(p.feedback().slot);
|
||||
// TODO(turbofan): we should consider creating a ProcessedFeedback for
|
||||
// allocation sites/boiler plates so that we use GetFeedback here. Then
|
||||
// we can eventually get rid of the additional copy of feedback slots that
|
||||
// we currently have in FeedbackVectorData.
|
||||
if (feedback.IsJSRegExp()) {
|
||||
JSRegExpRef boilerplate = feedback.AsJSRegExp();
|
||||
Node* value = effect = AllocateLiteralRegExp(effect, control, boilerplate);
|
||||
ProcessedFeedback const& feedback =
|
||||
broker()->GetFeedbackForRegExpLiteral(p.feedback());
|
||||
if (!feedback.IsInsufficient()) {
|
||||
JSRegExpRef literal = feedback.AsRegExpLiteral().value();
|
||||
Node* value = effect = AllocateLiteralRegExp(effect, control, literal);
|
||||
ReplaceWithValue(node, value, effect, control);
|
||||
return Replace(value);
|
||||
}
|
||||
return NoChange();
|
||||
}
|
||||
|
||||
Reduction JSCreateLowering::ReduceJSGetTemplateObject(Node* node) {
|
||||
DCHECK_EQ(IrOpcode::kJSGetTemplateObject, node->opcode());
|
||||
GetTemplateObjectParameters const& parameters =
|
||||
GetTemplateObjectParametersOf(node->op());
|
||||
SharedFunctionInfoRef shared(broker(), parameters.shared());
|
||||
JSArrayRef template_object = shared.GetTemplateObject(
|
||||
TemplateObjectDescriptionRef(broker(), parameters.description()),
|
||||
parameters.feedback());
|
||||
Node* value = jsgraph()->Constant(template_object);
|
||||
ReplaceWithValue(node, value);
|
||||
return Replace(value);
|
||||
}
|
||||
|
||||
Reduction JSCreateLowering::ReduceJSCreateFunctionContext(Node* node) {
|
||||
DCHECK_EQ(IrOpcode::kJSCreateFunctionContext, node->opcode());
|
||||
const CreateFunctionContextParameters& parameters =
|
||||
|
@ -67,6 +67,7 @@ class V8_EXPORT_PRIVATE JSCreateLowering final
|
||||
Reduction ReduceJSCreateCatchContext(Node* node);
|
||||
Reduction ReduceJSCreateBlockContext(Node* node);
|
||||
Reduction ReduceJSCreateGeneratorObject(Node* node);
|
||||
Reduction ReduceJSGetTemplateObject(Node* node);
|
||||
Reduction ReduceNewArray(
|
||||
Node* node, Node* length, MapRef initial_map, ElementsKind elements_kind,
|
||||
AllocationType allocation,
|
||||
|
@ -562,6 +562,10 @@ void JSGenericLowering::LowerJSCreateLiteralArray(Node* node) {
|
||||
}
|
||||
}
|
||||
|
||||
void JSGenericLowering::LowerJSGetTemplateObject(Node* node) {
|
||||
UNREACHABLE(); // Eliminated in native context specialization.
|
||||
}
|
||||
|
||||
void JSGenericLowering::LowerJSCreateEmptyLiteralArray(Node* node) {
|
||||
CallDescriptor::Flags flags = FrameStateFlagForCall(node);
|
||||
FeedbackParameter const& p = FeedbackParameterOf(node->op());
|
||||
|
@ -1269,7 +1269,6 @@ class FeedbackVectorData : public HeapObjectData {
|
||||
double invocation_count() const { return invocation_count_; }
|
||||
|
||||
void Serialize(JSHeapBroker* broker);
|
||||
const ZoneVector<ObjectData*>& feedback() { return feedback_; }
|
||||
FeedbackCellData* GetClosureFeedbackCell(JSHeapBroker* broker,
|
||||
int index) const;
|
||||
|
||||
@ -1277,7 +1276,6 @@ class FeedbackVectorData : public HeapObjectData {
|
||||
double const invocation_count_;
|
||||
|
||||
bool serialized_ = false;
|
||||
ZoneVector<ObjectData*> feedback_;
|
||||
ZoneVector<ObjectData*> closure_feedback_cell_array_;
|
||||
};
|
||||
|
||||
@ -1286,7 +1284,6 @@ FeedbackVectorData::FeedbackVectorData(JSHeapBroker* broker,
|
||||
Handle<FeedbackVector> object)
|
||||
: HeapObjectData(broker, storage, object),
|
||||
invocation_count_(object->invocation_count()),
|
||||
feedback_(broker->zone()),
|
||||
closure_feedback_cell_array_(broker->zone()) {}
|
||||
|
||||
FeedbackCellData* FeedbackVectorData::GetClosureFeedbackCell(
|
||||
@ -1310,26 +1307,6 @@ void FeedbackVectorData::Serialize(JSHeapBroker* broker) {
|
||||
|
||||
TraceScope tracer(broker, this, "FeedbackVectorData::Serialize");
|
||||
Handle<FeedbackVector> vector = Handle<FeedbackVector>::cast(object());
|
||||
DCHECK(feedback_.empty());
|
||||
feedback_.reserve(vector->length());
|
||||
for (int i = 0; i < vector->length(); ++i) {
|
||||
MaybeObject value = vector->get(i);
|
||||
ObjectData* slot_value =
|
||||
value->IsObject() ? broker->GetOrCreateData(value->cast<Object>())
|
||||
: nullptr;
|
||||
feedback_.push_back(slot_value);
|
||||
if (slot_value == nullptr) continue;
|
||||
|
||||
if (slot_value->IsAllocationSite() &&
|
||||
slot_value->AsAllocationSite()->IsFastLiteral()) {
|
||||
slot_value->AsAllocationSite()->SerializeBoilerplate(broker);
|
||||
} else if (slot_value->IsJSRegExp()) {
|
||||
slot_value->AsJSRegExp()->SerializeAsRegExpBoilerplate(broker);
|
||||
}
|
||||
}
|
||||
DCHECK_EQ(vector->length(), feedback_.size());
|
||||
TRACE(broker, "Copied " << feedback_.size() << " slots");
|
||||
|
||||
DCHECK(closure_feedback_cell_array_.empty());
|
||||
int length = vector->closure_feedback_cell_array().length();
|
||||
closure_feedback_cell_array_.reserve(length);
|
||||
@ -1908,6 +1885,13 @@ PropertyCellData* JSGlobalProxyData::GetPropertyCell(
|
||||
return result;
|
||||
}
|
||||
|
||||
class TemplateObjectDescriptionData : public HeapObjectData {
|
||||
public:
|
||||
TemplateObjectDescriptionData(JSHeapBroker* broker, ObjectData** storage,
|
||||
Handle<TemplateObjectDescription> object)
|
||||
: HeapObjectData(broker, storage, object) {}
|
||||
};
|
||||
|
||||
class CodeData : public HeapObjectData {
|
||||
public:
|
||||
CodeData(JSHeapBroker* broker, ObjectData** storage, Handle<Code> object)
|
||||
@ -2791,18 +2775,6 @@ OddballType MapRef::oddball_type() const {
|
||||
return OddballType::kOther;
|
||||
}
|
||||
|
||||
ObjectRef FeedbackVectorRef::get(FeedbackSlot slot) const {
|
||||
if (broker()->mode() == JSHeapBroker::kDisabled) {
|
||||
AllowHandleAllocation handle_allocation;
|
||||
AllowHandleDereference handle_dereference;
|
||||
Handle<Object> value(object()->Get(slot)->cast<Object>(),
|
||||
broker()->isolate());
|
||||
return ObjectRef(broker(), value);
|
||||
}
|
||||
int i = FeedbackVector::GetIndex(slot);
|
||||
return ObjectRef(broker(), data()->AsFeedbackVector()->feedback().at(i));
|
||||
}
|
||||
|
||||
FeedbackCellRef FeedbackVectorRef::GetClosureFeedbackCell(int index) const {
|
||||
if (broker()->mode() == JSHeapBroker::kDisabled) {
|
||||
AllowHandleAllocation handle_allocation;
|
||||
@ -2860,6 +2832,11 @@ bool AllocationSiteRef::IsFastLiteral() const {
|
||||
return data()->AsAllocationSite()->IsFastLiteral();
|
||||
}
|
||||
|
||||
void AllocationSiteRef::SerializeBoilerplate() {
|
||||
CHECK_EQ(broker()->mode(), JSHeapBroker::kSerializing);
|
||||
data()->AsAllocationSite()->SerializeBoilerplate(broker());
|
||||
}
|
||||
|
||||
void JSObjectRef::SerializeElements() {
|
||||
CHECK_EQ(broker()->mode(), JSHeapBroker::kSerializing);
|
||||
data()->AsJSObject()->SerializeElements(broker());
|
||||
@ -3775,6 +3752,11 @@ ObjectRef JSRegExpRef::source() const {
|
||||
return ObjectRef(broker(), ObjectRef::data()->AsJSRegExp()->source());
|
||||
}
|
||||
|
||||
void JSRegExpRef::SerializeAsRegExpBoilerplate() {
|
||||
CHECK_EQ(broker()->mode(), JSHeapBroker::kSerializing);
|
||||
JSObjectRef::data()->AsJSRegExp()->SerializeAsRegExpBoilerplate(broker());
|
||||
}
|
||||
|
||||
Handle<Object> ObjectRef::object() const { return data_->object(); }
|
||||
|
||||
#define DEF_OBJECT_GETTER(T) \
|
||||
@ -3859,41 +3841,42 @@ bool JSFunctionRef::serialized() const {
|
||||
}
|
||||
|
||||
JSArrayRef SharedFunctionInfoRef::GetTemplateObject(
|
||||
ObjectRef description, FeedbackVectorRef vector, FeedbackSlot slot,
|
||||
TemplateObjectDescriptionRef description, FeedbackSource const& source,
|
||||
SerializationPolicy policy) {
|
||||
// Look in the feedback vector for the array. A Smi indicates that it's
|
||||
// not yet cached here.
|
||||
ObjectRef candidate = vector.get(slot);
|
||||
if (!candidate.IsSmi()) {
|
||||
return candidate.AsJSArray();
|
||||
// First, see if we have processed feedback from the vector, respecting
|
||||
// the serialization policy.
|
||||
ProcessedFeedback const& feedback =
|
||||
policy == SerializationPolicy::kSerializeIfNeeded
|
||||
? broker()->ProcessFeedbackForTemplateObject(source)
|
||||
: broker()->GetFeedbackForTemplateObject(source);
|
||||
|
||||
if (!feedback.IsInsufficient()) {
|
||||
return feedback.AsTemplateObject().value();
|
||||
}
|
||||
|
||||
if (broker()->mode() == JSHeapBroker::kDisabled) {
|
||||
AllowHandleAllocation handle_allocation;
|
||||
AllowHandleDereference allow_handle_dereference;
|
||||
Handle<TemplateObjectDescription> tod =
|
||||
Handle<TemplateObjectDescription>::cast(description.object());
|
||||
Handle<JSArray> template_object =
|
||||
TemplateObjectDescription::GetTemplateObject(
|
||||
broker()->isolate(), broker()->target_native_context().object(),
|
||||
tod, object(), slot.ToInt());
|
||||
isolate(), broker()->target_native_context().object(),
|
||||
description.object(), object(), source.slot.ToInt());
|
||||
return JSArrayRef(broker(), template_object);
|
||||
}
|
||||
|
||||
JSArrayData* array = data()->AsSharedFunctionInfo()->GetTemplateObject(slot);
|
||||
JSArrayData* array =
|
||||
data()->AsSharedFunctionInfo()->GetTemplateObject(source.slot);
|
||||
if (array != nullptr) return JSArrayRef(broker(), array);
|
||||
|
||||
CHECK_EQ(policy, SerializationPolicy::kSerializeIfNeeded);
|
||||
CHECK(broker()->SerializingAllowed());
|
||||
|
||||
Handle<TemplateObjectDescription> tod =
|
||||
Handle<TemplateObjectDescription>::cast(description.object());
|
||||
Handle<JSArray> template_object =
|
||||
TemplateObjectDescription::GetTemplateObject(
|
||||
broker()->isolate(), broker()->target_native_context().object(), tod,
|
||||
object(), slot.ToInt());
|
||||
broker()->isolate(), broker()->target_native_context().object(),
|
||||
description.object(), object(), source.slot.ToInt());
|
||||
array = broker()->GetOrCreateData(template_object)->AsJSArray();
|
||||
data()->AsSharedFunctionInfo()->SetTemplateObject(slot, array);
|
||||
data()->AsSharedFunctionInfo()->SetTemplateObject(source.slot, array);
|
||||
return JSArrayRef(broker(), array);
|
||||
}
|
||||
|
||||
@ -4254,6 +4237,7 @@ void JSHeapBroker::SetFeedback(FeedbackSource const& source,
|
||||
}
|
||||
|
||||
bool JSHeapBroker::HasFeedback(FeedbackSource const& source) const {
|
||||
DCHECK(source.IsValid());
|
||||
return feedback_.find(source) != feedback_.end();
|
||||
}
|
||||
|
||||
@ -4413,6 +4397,47 @@ ProcessedFeedback const& JSHeapBroker::ReadFeedbackForInstanceOf(
|
||||
return *new (zone()) InstanceOfFeedback(optional_constructor, nexus.kind());
|
||||
}
|
||||
|
||||
ProcessedFeedback const& JSHeapBroker::ReadFeedbackForArrayOrObjectLiteral(
|
||||
FeedbackSource const& source) {
|
||||
FeedbackNexus nexus(source.vector, source.slot);
|
||||
HeapObject object;
|
||||
if (nexus.IsUninitialized() || !nexus.GetFeedback()->GetHeapObject(&object)) {
|
||||
return *new (zone()) InsufficientFeedback(nexus.kind());
|
||||
}
|
||||
|
||||
AllocationSiteRef site(this, handle(object, isolate()));
|
||||
if (site.IsFastLiteral()) {
|
||||
site.SerializeBoilerplate();
|
||||
}
|
||||
|
||||
return *new (zone()) LiteralFeedback(site, nexus.kind());
|
||||
}
|
||||
|
||||
ProcessedFeedback const& JSHeapBroker::ReadFeedbackForRegExpLiteral(
|
||||
FeedbackSource const& source) {
|
||||
FeedbackNexus nexus(source.vector, source.slot);
|
||||
HeapObject object;
|
||||
if (nexus.IsUninitialized() || !nexus.GetFeedback()->GetHeapObject(&object)) {
|
||||
return *new (zone()) InsufficientFeedback(nexus.kind());
|
||||
}
|
||||
|
||||
JSRegExpRef regexp(this, handle(object, isolate()));
|
||||
regexp.SerializeAsRegExpBoilerplate();
|
||||
return *new (zone()) RegExpLiteralFeedback(regexp, nexus.kind());
|
||||
}
|
||||
|
||||
ProcessedFeedback const& JSHeapBroker::ReadFeedbackForTemplateObject(
|
||||
FeedbackSource const& source) {
|
||||
FeedbackNexus nexus(source.vector, source.slot);
|
||||
HeapObject object;
|
||||
if (nexus.IsUninitialized() || !nexus.GetFeedback()->GetHeapObject(&object)) {
|
||||
return *new (zone()) InsufficientFeedback(nexus.kind());
|
||||
}
|
||||
|
||||
JSArrayRef array(this, handle(object, isolate()));
|
||||
return *new (zone()) TemplateObjectFeedback(array, nexus.kind());
|
||||
}
|
||||
|
||||
ProcessedFeedback const& JSHeapBroker::ReadFeedbackForCall(
|
||||
FeedbackSource const& source) {
|
||||
FeedbackNexus nexus(source.vector, source.slot);
|
||||
@ -4484,6 +4509,50 @@ ProcessedFeedback const& JSHeapBroker::GetFeedbackForGlobalAccess(
|
||||
: ProcessFeedbackForGlobalAccess(source);
|
||||
}
|
||||
|
||||
ProcessedFeedback const& JSHeapBroker::GetFeedbackForArrayOrObjectLiteral(
|
||||
FeedbackSource const& source) {
|
||||
return FLAG_concurrent_inlining
|
||||
? GetFeedback(source)
|
||||
: ProcessFeedbackForArrayOrObjectLiteral(source);
|
||||
}
|
||||
|
||||
ProcessedFeedback const& JSHeapBroker::GetFeedbackForRegExpLiteral(
|
||||
FeedbackSource const& source) {
|
||||
return FLAG_concurrent_inlining ? GetFeedback(source)
|
||||
: ProcessFeedbackForRegExpLiteral(source);
|
||||
}
|
||||
|
||||
ProcessedFeedback const& JSHeapBroker::GetFeedbackForTemplateObject(
|
||||
FeedbackSource const& source) {
|
||||
return FLAG_concurrent_inlining ? GetFeedback(source)
|
||||
: ProcessFeedbackForTemplateObject(source);
|
||||
}
|
||||
|
||||
ProcessedFeedback const& JSHeapBroker::ProcessFeedbackForArrayOrObjectLiteral(
|
||||
FeedbackSource const& source) {
|
||||
if (HasFeedback(source)) return GetFeedback(source);
|
||||
ProcessedFeedback const& feedback =
|
||||
ReadFeedbackForArrayOrObjectLiteral(source);
|
||||
SetFeedback(source, &feedback);
|
||||
return feedback;
|
||||
}
|
||||
|
||||
ProcessedFeedback const& JSHeapBroker::ProcessFeedbackForRegExpLiteral(
|
||||
FeedbackSource const& source) {
|
||||
if (HasFeedback(source)) return GetFeedback(source);
|
||||
ProcessedFeedback const& feedback = ReadFeedbackForRegExpLiteral(source);
|
||||
SetFeedback(source, &feedback);
|
||||
return feedback;
|
||||
}
|
||||
|
||||
ProcessedFeedback const& JSHeapBroker::ProcessFeedbackForTemplateObject(
|
||||
FeedbackSource const& source) {
|
||||
if (HasFeedback(source)) return GetFeedback(source);
|
||||
ProcessedFeedback const& feedback = ReadFeedbackForTemplateObject(source);
|
||||
SetFeedback(source, &feedback);
|
||||
return feedback;
|
||||
}
|
||||
|
||||
ProcessedFeedback const& JSHeapBroker::ProcessFeedbackForBinaryOperation(
|
||||
FeedbackSource const& source) {
|
||||
if (HasFeedback(source)) return GetFeedback(source);
|
||||
@ -4723,6 +4792,21 @@ NamedAccessFeedback const& ProcessedFeedback::AsNamedAccess() const {
|
||||
return *static_cast<NamedAccessFeedback const*>(this);
|
||||
}
|
||||
|
||||
LiteralFeedback const& ProcessedFeedback::AsLiteral() const {
|
||||
CHECK_EQ(kLiteral, kind());
|
||||
return *static_cast<LiteralFeedback const*>(this);
|
||||
}
|
||||
|
||||
RegExpLiteralFeedback const& ProcessedFeedback::AsRegExpLiteral() const {
|
||||
CHECK_EQ(kRegExpLiteral, kind());
|
||||
return *static_cast<RegExpLiteralFeedback const*>(this);
|
||||
}
|
||||
|
||||
TemplateObjectFeedback const& ProcessedFeedback::AsTemplateObject() const {
|
||||
CHECK_EQ(kTemplateObject, kind());
|
||||
return *static_cast<TemplateObjectFeedback const*>(this);
|
||||
}
|
||||
|
||||
BytecodeAnalysis const& JSHeapBroker::GetBytecodeAnalysis(
|
||||
Handle<BytecodeArray> bytecode_array, BailoutId osr_bailout_id,
|
||||
bool analyze_liveness, SerializationPolicy policy) {
|
||||
|
@ -125,6 +125,12 @@ class V8_EXPORT_PRIVATE JSHeapBroker {
|
||||
FeedbackSource const& source);
|
||||
ProcessedFeedback const& GetFeedbackForInstanceOf(
|
||||
FeedbackSource const& source);
|
||||
ProcessedFeedback const& GetFeedbackForArrayOrObjectLiteral(
|
||||
FeedbackSource const& source);
|
||||
ProcessedFeedback const& GetFeedbackForRegExpLiteral(
|
||||
FeedbackSource const& source);
|
||||
ProcessedFeedback const& GetFeedbackForTemplateObject(
|
||||
FeedbackSource const& source);
|
||||
ProcessedFeedback const& GetFeedbackForPropertyAccess(
|
||||
FeedbackSource const& source, AccessMode mode,
|
||||
base::Optional<NameRef> static_name);
|
||||
@ -143,6 +149,12 @@ class V8_EXPORT_PRIVATE JSHeapBroker {
|
||||
ProcessedFeedback const& ProcessFeedbackForPropertyAccess(
|
||||
FeedbackSource const& source, AccessMode mode,
|
||||
base::Optional<NameRef> static_name);
|
||||
ProcessedFeedback const& ProcessFeedbackForArrayOrObjectLiteral(
|
||||
FeedbackSource const& source);
|
||||
ProcessedFeedback const& ProcessFeedbackForRegExpLiteral(
|
||||
FeedbackSource const& source);
|
||||
ProcessedFeedback const& ProcessFeedbackForTemplateObject(
|
||||
FeedbackSource const& source);
|
||||
|
||||
bool FeedbackIsInsufficient(FeedbackSource const& source) const;
|
||||
|
||||
@ -182,6 +194,12 @@ class V8_EXPORT_PRIVATE JSHeapBroker {
|
||||
ProcessedFeedback const& ReadFeedbackForPropertyAccess(
|
||||
FeedbackSource const& source, AccessMode mode,
|
||||
base::Optional<NameRef> static_name);
|
||||
ProcessedFeedback const& ReadFeedbackForArrayOrObjectLiteral(
|
||||
FeedbackSource const& source);
|
||||
ProcessedFeedback const& ReadFeedbackForRegExpLiteral(
|
||||
FeedbackSource const& source);
|
||||
ProcessedFeedback const& ReadFeedbackForTemplateObject(
|
||||
FeedbackSource const& source);
|
||||
|
||||
void InitializeRefsMap();
|
||||
void CollectArrayAndObjectPrototypes();
|
||||
|
@ -12,6 +12,7 @@
|
||||
#include "src/heap/factory-inl.h"
|
||||
#include "src/objects/map.h"
|
||||
#include "src/objects/scope-info.h"
|
||||
#include "src/objects/template-objects.h"
|
||||
|
||||
namespace v8 {
|
||||
namespace internal {
|
||||
@ -73,7 +74,9 @@ Reduction JSHeapCopyReducer::Reduce(Node* node) {
|
||||
}
|
||||
case IrOpcode::kJSCreateEmptyLiteralArray: {
|
||||
FeedbackParameter const& p = FeedbackParameterOf(node->op());
|
||||
FeedbackVectorRef(broker(), p.feedback().vector).Serialize();
|
||||
if (p.feedback().IsValid()) {
|
||||
broker()->ProcessFeedbackForArrayOrObjectLiteral(p.feedback());
|
||||
}
|
||||
break;
|
||||
}
|
||||
case IrOpcode::kJSCreateFunctionContext: {
|
||||
@ -85,12 +88,25 @@ Reduction JSHeapCopyReducer::Reduce(Node* node) {
|
||||
case IrOpcode::kJSCreateLiteralArray:
|
||||
case IrOpcode::kJSCreateLiteralObject: {
|
||||
CreateLiteralParameters const& p = CreateLiteralParametersOf(node->op());
|
||||
FeedbackVectorRef(broker(), p.feedback().vector).Serialize();
|
||||
if (p.feedback().IsValid()) {
|
||||
broker()->ProcessFeedbackForArrayOrObjectLiteral(p.feedback());
|
||||
}
|
||||
break;
|
||||
}
|
||||
case IrOpcode::kJSCreateLiteralRegExp: {
|
||||
CreateLiteralParameters const& p = CreateLiteralParametersOf(node->op());
|
||||
FeedbackVectorRef(broker(), p.feedback().vector).Serialize();
|
||||
if (p.feedback().IsValid()) {
|
||||
broker()->ProcessFeedbackForRegExpLiteral(p.feedback());
|
||||
}
|
||||
break;
|
||||
}
|
||||
case IrOpcode::kJSGetTemplateObject: {
|
||||
GetTemplateObjectParameters const& p =
|
||||
GetTemplateObjectParametersOf(node->op());
|
||||
SharedFunctionInfoRef shared(broker(), p.shared());
|
||||
TemplateObjectDescriptionRef description(broker(), p.description());
|
||||
shared.GetTemplateObject(description, p.feedback(),
|
||||
SerializationPolicy::kSerializeIfNeeded);
|
||||
break;
|
||||
}
|
||||
case IrOpcode::kJSCreateWithContext: {
|
||||
|
@ -11,6 +11,7 @@
|
||||
#include "src/compiler/operator.h"
|
||||
#include "src/handles/handles-inl.h"
|
||||
#include "src/objects/objects-inl.h"
|
||||
#include "src/objects/template-objects.h"
|
||||
|
||||
namespace v8 {
|
||||
namespace internal {
|
||||
@ -472,6 +473,34 @@ const CreateBoundFunctionParameters& CreateBoundFunctionParametersOf(
|
||||
return OpParameter<CreateBoundFunctionParameters>(op);
|
||||
}
|
||||
|
||||
bool operator==(GetTemplateObjectParameters const& lhs,
|
||||
GetTemplateObjectParameters const& rhs) {
|
||||
return lhs.description().location() == rhs.description().location() &&
|
||||
lhs.shared().location() == rhs.shared().location() &&
|
||||
lhs.feedback() == rhs.feedback();
|
||||
}
|
||||
|
||||
bool operator!=(GetTemplateObjectParameters const& lhs,
|
||||
GetTemplateObjectParameters const& rhs) {
|
||||
return !(lhs == rhs);
|
||||
}
|
||||
|
||||
size_t hash_value(GetTemplateObjectParameters const& p) {
|
||||
return base::hash_combine(p.description().location(), p.shared().location(),
|
||||
FeedbackSource::Hash()(p.feedback()));
|
||||
}
|
||||
|
||||
std::ostream& operator<<(std::ostream& os,
|
||||
GetTemplateObjectParameters const& p) {
|
||||
return os << Brief(*p.description()) << ", " << Brief(*p.shared());
|
||||
}
|
||||
|
||||
const GetTemplateObjectParameters& GetTemplateObjectParametersOf(
|
||||
const Operator* op) {
|
||||
DCHECK(op->opcode() == IrOpcode::kJSGetTemplateObject);
|
||||
return OpParameter<GetTemplateObjectParameters>(op);
|
||||
}
|
||||
|
||||
bool operator==(CreateClosureParameters const& lhs,
|
||||
CreateClosureParameters const& rhs) {
|
||||
return lhs.allocation() == rhs.allocation() &&
|
||||
@ -1282,6 +1311,18 @@ const Operator* JSOperatorBuilder::CreateLiteralObject(
|
||||
parameters); // parameter
|
||||
}
|
||||
|
||||
const Operator* JSOperatorBuilder::GetTemplateObject(
|
||||
Handle<TemplateObjectDescription> description,
|
||||
Handle<SharedFunctionInfo> shared, FeedbackSource const& feedback) {
|
||||
GetTemplateObjectParameters parameters(description, shared, feedback);
|
||||
return new (zone()) Operator1<GetTemplateObjectParameters>( // --
|
||||
IrOpcode::kJSGetTemplateObject, // opcode
|
||||
Operator::kEliminatable, // properties
|
||||
"JSGetTemplateObject", // name
|
||||
0, 1, 1, 1, 1, 0, // counts
|
||||
parameters); // parameter
|
||||
}
|
||||
|
||||
const Operator* JSOperatorBuilder::CloneObject(FeedbackSource const& feedback,
|
||||
int literal_flags) {
|
||||
CloneObjectParameters parameters(feedback, literal_flags);
|
||||
|
@ -598,6 +598,35 @@ std::ostream& operator<<(std::ostream&, CreateClosureParameters const&);
|
||||
|
||||
const CreateClosureParameters& CreateClosureParametersOf(const Operator* op);
|
||||
|
||||
class GetTemplateObjectParameters final {
|
||||
public:
|
||||
GetTemplateObjectParameters(Handle<TemplateObjectDescription> description,
|
||||
Handle<SharedFunctionInfo> shared,
|
||||
FeedbackSource const& feedback)
|
||||
: description_(description), shared_(shared), feedback_(feedback) {}
|
||||
|
||||
Handle<TemplateObjectDescription> description() const { return description_; }
|
||||
Handle<SharedFunctionInfo> shared() const { return shared_; }
|
||||
FeedbackSource const& feedback() const { return feedback_; }
|
||||
|
||||
private:
|
||||
Handle<TemplateObjectDescription> const description_;
|
||||
Handle<SharedFunctionInfo> const shared_;
|
||||
FeedbackSource const feedback_;
|
||||
};
|
||||
|
||||
bool operator==(GetTemplateObjectParameters const&,
|
||||
GetTemplateObjectParameters const&);
|
||||
bool operator!=(GetTemplateObjectParameters const&,
|
||||
GetTemplateObjectParameters const&);
|
||||
|
||||
size_t hash_value(GetTemplateObjectParameters const&);
|
||||
|
||||
std::ostream& operator<<(std::ostream&, GetTemplateObjectParameters const&);
|
||||
|
||||
const GetTemplateObjectParameters& GetTemplateObjectParametersOf(
|
||||
const Operator* op);
|
||||
|
||||
// Defines shared information for the literal that should be created. This is
|
||||
// used as parameter by JSCreateLiteralArray, JSCreateLiteralObject and
|
||||
// JSCreateLiteralRegExp operators.
|
||||
@ -767,7 +796,6 @@ class V8_EXPORT_PRIVATE JSOperatorBuilder final
|
||||
const Operator* CreateEmptyLiteralArray(FeedbackSource const& feedback);
|
||||
const Operator* CreateArrayFromIterable();
|
||||
const Operator* CreateEmptyLiteralObject();
|
||||
|
||||
const Operator* CreateLiteralObject(
|
||||
Handle<ObjectBoilerplateDescription> constant,
|
||||
FeedbackSource const& feedback, int literal_flags,
|
||||
@ -778,6 +806,10 @@ class V8_EXPORT_PRIVATE JSOperatorBuilder final
|
||||
FeedbackSource const& feedback,
|
||||
int literal_flags);
|
||||
|
||||
const Operator* GetTemplateObject(
|
||||
Handle<TemplateObjectDescription> description,
|
||||
Handle<SharedFunctionInfo> shared, FeedbackSource const& feedback);
|
||||
|
||||
const Operator* CallForwardVarargs(size_t arity, uint32_t start_index);
|
||||
const Operator* Call(
|
||||
size_t arity, CallFrequency const& frequency = CallFrequency(),
|
||||
|
@ -156,7 +156,8 @@
|
||||
V(JSCreateObject) \
|
||||
V(JSCreatePromise) \
|
||||
V(JSCreateStringIterator) \
|
||||
V(JSCreateTypedArray)
|
||||
V(JSCreateTypedArray) \
|
||||
V(JSGetTemplateObject)
|
||||
|
||||
#define JS_OBJECT_OP_LIST(V) \
|
||||
JS_CREATE_OP_LIST(V) \
|
||||
|
@ -41,6 +41,7 @@ bool OperatorProperties::NeedsExactContext(const Operator* op) {
|
||||
case IrOpcode::kJSCreateEmptyLiteralObject:
|
||||
case IrOpcode::kJSCreateArrayFromIterable:
|
||||
case IrOpcode::kJSCreateLiteralRegExp:
|
||||
case IrOpcode::kJSGetTemplateObject:
|
||||
case IrOpcode::kJSForInEnumerate:
|
||||
case IrOpcode::kJSForInNext:
|
||||
case IrOpcode::kJSForInPrepare:
|
||||
|
@ -18,7 +18,10 @@ class ElementAccessFeedback;
|
||||
class ForInFeedback;
|
||||
class GlobalAccessFeedback;
|
||||
class InstanceOfFeedback;
|
||||
class LiteralFeedback;
|
||||
class NamedAccessFeedback;
|
||||
class RegExpLiteralFeedback;
|
||||
class TemplateObjectFeedback;
|
||||
|
||||
class ProcessedFeedback : public ZoneObject {
|
||||
public:
|
||||
@ -31,7 +34,10 @@ class ProcessedFeedback : public ZoneObject {
|
||||
kForIn,
|
||||
kGlobalAccess,
|
||||
kInstanceOf,
|
||||
kLiteral,
|
||||
kNamedAccess,
|
||||
kRegExpLiteral,
|
||||
kTemplateObject,
|
||||
};
|
||||
Kind kind() const { return kind_; }
|
||||
|
||||
@ -46,6 +52,9 @@ class ProcessedFeedback : public ZoneObject {
|
||||
GlobalAccessFeedback const& AsGlobalAccess() const;
|
||||
InstanceOfFeedback const& AsInstanceOf() const;
|
||||
NamedAccessFeedback const& AsNamedAccess() const;
|
||||
LiteralFeedback const& AsLiteral() const;
|
||||
RegExpLiteralFeedback const& AsRegExpLiteral() const;
|
||||
TemplateObjectFeedback const& AsTemplateObject() const;
|
||||
|
||||
protected:
|
||||
ProcessedFeedback(Kind kind, FeedbackSlotKind slot_kind);
|
||||
@ -187,7 +196,9 @@ class SingleValueFeedback : public ProcessedFeedback {
|
||||
(K == kBinaryOperation && slot_kind == FeedbackSlotKind::kBinaryOp) ||
|
||||
(K == kCompareOperation && slot_kind == FeedbackSlotKind::kCompareOp) ||
|
||||
(K == kForIn && slot_kind == FeedbackSlotKind::kForIn) ||
|
||||
(K == kInstanceOf && slot_kind == FeedbackSlotKind::kInstanceOf));
|
||||
(K == kInstanceOf && slot_kind == FeedbackSlotKind::kInstanceOf) ||
|
||||
((K == kLiteral || K == kRegExpLiteral || K == kTemplateObject) &&
|
||||
slot_kind == FeedbackSlotKind::kLiteral));
|
||||
}
|
||||
|
||||
T value() const { return value_; }
|
||||
@ -202,6 +213,24 @@ class InstanceOfFeedback
|
||||
using SingleValueFeedback::SingleValueFeedback;
|
||||
};
|
||||
|
||||
class LiteralFeedback
|
||||
: public SingleValueFeedback<AllocationSiteRef,
|
||||
ProcessedFeedback::kLiteral> {
|
||||
using SingleValueFeedback::SingleValueFeedback;
|
||||
};
|
||||
|
||||
class RegExpLiteralFeedback
|
||||
: public SingleValueFeedback<JSRegExpRef,
|
||||
ProcessedFeedback::kRegExpLiteral> {
|
||||
using SingleValueFeedback::SingleValueFeedback;
|
||||
};
|
||||
|
||||
class TemplateObjectFeedback
|
||||
: public SingleValueFeedback<JSArrayRef,
|
||||
ProcessedFeedback::kTemplateObject> {
|
||||
using SingleValueFeedback::SingleValueFeedback;
|
||||
};
|
||||
|
||||
class BinaryOperationFeedback
|
||||
: public SingleValueFeedback<BinaryOperationHint,
|
||||
ProcessedFeedback::kBinaryOperation> {
|
||||
|
@ -43,7 +43,6 @@ namespace compiler {
|
||||
V(CallRuntime) \
|
||||
V(CloneObject) \
|
||||
V(CreateArrayFromIterable) \
|
||||
V(CreateEmptyArrayLiteral) \
|
||||
V(CreateEmptyObjectLiteral) \
|
||||
V(CreateMappedArguments) \
|
||||
V(CreateRestParameter) \
|
||||
@ -162,6 +161,7 @@ namespace compiler {
|
||||
V(CreateBlockContext) \
|
||||
V(CreateCatchContext) \
|
||||
V(CreateClosure) \
|
||||
V(CreateEmptyArrayLiteral) \
|
||||
V(CreateEvalContext) \
|
||||
V(CreateFunctionContext) \
|
||||
V(CreateObjectLiteral) \
|
||||
@ -1111,14 +1111,13 @@ void SerializerForBackgroundCompilation::VisitGetSuperConstructor(
|
||||
|
||||
void SerializerForBackgroundCompilation::VisitGetTemplateObject(
|
||||
BytecodeArrayIterator* iterator) {
|
||||
ObjectRef description(
|
||||
TemplateObjectDescriptionRef description(
|
||||
broker(), iterator->GetConstantForIndexOperand(0, broker()->isolate()));
|
||||
FeedbackSlot slot = iterator->GetSlotOperand(1);
|
||||
FeedbackVectorRef feedback_vector_ref(broker(), feedback_vector());
|
||||
FeedbackSource source(feedback_vector(), slot);
|
||||
SharedFunctionInfoRef shared(broker(), environment()->function().shared());
|
||||
JSArrayRef template_object =
|
||||
shared.GetTemplateObject(description, feedback_vector_ref, slot,
|
||||
SerializationPolicy::kSerializeIfNeeded);
|
||||
JSArrayRef template_object = shared.GetTemplateObject(
|
||||
description, source, SerializationPolicy::kSerializeIfNeeded);
|
||||
environment()->accumulator_hints().Clear();
|
||||
environment()->accumulator_hints().AddConstant(template_object.object(),
|
||||
zone());
|
||||
@ -1467,6 +1466,9 @@ void SerializerForBackgroundCompilation::VisitCreateRegExpLiteral(
|
||||
Handle<String> constant_pattern = Handle<String>::cast(
|
||||
iterator->GetConstantForIndexOperand(0, broker()->isolate()));
|
||||
StringRef description(broker(), constant_pattern);
|
||||
FeedbackSlot slot = iterator->GetSlotOperand(1);
|
||||
FeedbackSource source(feedback_vector(), slot);
|
||||
broker()->ProcessFeedbackForRegExpLiteral(source);
|
||||
environment()->accumulator_hints().Clear();
|
||||
}
|
||||
|
||||
@ -1477,6 +1479,17 @@ void SerializerForBackgroundCompilation::VisitCreateArrayLiteral(
|
||||
iterator->GetConstantForIndexOperand(0, broker()->isolate()));
|
||||
ArrayBoilerplateDescriptionRef description(broker(),
|
||||
array_boilerplate_description);
|
||||
FeedbackSlot slot = iterator->GetSlotOperand(1);
|
||||
FeedbackSource source(feedback_vector(), slot);
|
||||
broker()->ProcessFeedbackForArrayOrObjectLiteral(source);
|
||||
environment()->accumulator_hints().Clear();
|
||||
}
|
||||
|
||||
void SerializerForBackgroundCompilation::VisitCreateEmptyArrayLiteral(
|
||||
BytecodeArrayIterator* iterator) {
|
||||
FeedbackSlot slot = iterator->GetSlotOperand(0);
|
||||
FeedbackSource source(feedback_vector(), slot);
|
||||
broker()->ProcessFeedbackForArrayOrObjectLiteral(source);
|
||||
environment()->accumulator_hints().Clear();
|
||||
}
|
||||
|
||||
@ -1486,6 +1499,9 @@ void SerializerForBackgroundCompilation::VisitCreateObjectLiteral(
|
||||
Handle<ObjectBoilerplateDescription>::cast(
|
||||
iterator->GetConstantForIndexOperand(0, broker()->isolate()));
|
||||
ObjectBoilerplateDescriptionRef description(broker(), constant_properties);
|
||||
FeedbackSlot slot = iterator->GetSlotOperand(1);
|
||||
FeedbackSource source(feedback_vector(), slot);
|
||||
broker()->ProcessFeedbackForArrayOrObjectLiteral(source);
|
||||
environment()->accumulator_hints().Clear();
|
||||
}
|
||||
|
||||
|
@ -1339,6 +1339,10 @@ Type Typer::Visitor::TypeJSCreateLiteralRegExp(Node* node) {
|
||||
return Type::OtherObject();
|
||||
}
|
||||
|
||||
Type Typer::Visitor::TypeJSGetTemplateObject(Node* node) {
|
||||
return Type::Array();
|
||||
}
|
||||
|
||||
Type Typer::Visitor::TypeJSLoadProperty(Node* node) {
|
||||
return Type::NonInternal();
|
||||
}
|
||||
|
@ -732,6 +732,10 @@ void Verifier::Visitor::Check(Node* node, const AllNodes& all) {
|
||||
// Type is OtherObject.
|
||||
CheckTypeIs(node, Type::OtherObject());
|
||||
break;
|
||||
case IrOpcode::kJSGetTemplateObject:
|
||||
// Type is Array
|
||||
CheckTypeIs(node, Type::Array());
|
||||
break;
|
||||
case IrOpcode::kJSLoadProperty:
|
||||
// Type can be anything.
|
||||
CheckTypeIs(node, Type::Any());
|
||||
|
Loading…
Reference in New Issue
Block a user