2015-08-05 14:06:43 +00:00
|
|
|
// Copyright 2015 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/context-measure.h"
|
|
|
|
|
2015-08-20 07:44:00 +00:00
|
|
|
#include "src/base/logging.h"
|
|
|
|
#include "src/contexts.h"
|
|
|
|
#include "src/objects-inl.h"
|
|
|
|
|
2015-08-05 14:06:43 +00:00
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
|
|
|
|
|
|
|
ContextMeasure::ContextMeasure(Context* context)
|
|
|
|
: context_(context),
|
|
|
|
root_index_map_(context->GetIsolate()),
|
|
|
|
recursion_depth_(0),
|
|
|
|
count_(0),
|
|
|
|
size_(0) {
|
|
|
|
DCHECK(context_->IsNativeContext());
|
2016-05-12 08:50:18 +00:00
|
|
|
Object* next_link = context_->next_context_link();
|
2015-08-05 14:06:43 +00:00
|
|
|
MeasureObject(context_);
|
|
|
|
MeasureDeferredObjects();
|
|
|
|
context_->set(Context::NEXT_CONTEXT_LINK, next_link);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool ContextMeasure::IsShared(HeapObject* object) {
|
|
|
|
if (object->IsScript()) return true;
|
|
|
|
if (object->IsSharedFunctionInfo()) return true;
|
|
|
|
if (object->IsScopeInfo()) return true;
|
|
|
|
if (object->IsCode() && !Code::cast(object)->is_optimized_code()) return true;
|
2016-01-18 15:08:36 +00:00
|
|
|
if (object->IsAccessorInfo()) return true;
|
2015-08-05 14:06:43 +00:00
|
|
|
if (object->IsWeakCell()) return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ContextMeasure::MeasureObject(HeapObject* object) {
|
2016-05-18 11:28:53 +00:00
|
|
|
if (reference_map_.Lookup(object).is_valid()) return;
|
2015-08-05 14:06:43 +00:00
|
|
|
if (root_index_map_.Lookup(object) != RootIndexMap::kInvalidRootIndex) return;
|
|
|
|
if (IsShared(object)) return;
|
2016-05-18 11:28:53 +00:00
|
|
|
reference_map_.Add(object, SerializerReference::DummyReference());
|
2015-08-05 14:06:43 +00:00
|
|
|
recursion_depth_++;
|
|
|
|
if (recursion_depth_ > kMaxRecursion) {
|
|
|
|
deferred_objects_.Add(object);
|
|
|
|
} else {
|
|
|
|
MeasureAndRecurse(object);
|
|
|
|
}
|
|
|
|
recursion_depth_--;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ContextMeasure::MeasureDeferredObjects() {
|
|
|
|
while (deferred_objects_.length() > 0) {
|
|
|
|
MeasureAndRecurse(deferred_objects_.RemoveLast());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ContextMeasure::MeasureAndRecurse(HeapObject* object) {
|
|
|
|
int size = object->Size();
|
|
|
|
count_++;
|
|
|
|
size_ += size;
|
|
|
|
Map* map = object->map();
|
|
|
|
MeasureObject(map);
|
|
|
|
object->IterateBody(map->instance_type(), size, this);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ContextMeasure::VisitPointers(Object** start, Object** end) {
|
|
|
|
for (Object** current = start; current < end; current++) {
|
|
|
|
if ((*current)->IsSmi()) continue;
|
|
|
|
MeasureObject(HeapObject::cast(*current));
|
|
|
|
}
|
|
|
|
}
|
2015-09-30 13:46:56 +00:00
|
|
|
} // namespace internal
|
|
|
|
} // namespace v8
|