mirror of
https://github.com/dart-lang/sdk
synced 2024-11-02 12:24:24 +00:00
86fdeab739
This flag doesn't seem to be used anywhere, neither of dart/flutter/golem/g3. Main motivation is to remove the GC triggering code in Isolate::MakeRunnable. Issue https://github.com/dart-lang/sdk/issues/36097 Change-Id: If48072f62974dec5f7c19cc08be7904401e38a61 Reviewed-on: https://dart-review.googlesource.com/c/sdk/+/171282 Reviewed-by: Ryan Macnak <rmacnak@google.com> Commit-Queue: Martin Kustermann <kustermann@google.com>
244 lines
6.7 KiB
C++
244 lines
6.7 KiB
C++
// Copyright (c) 2014, the Dart project authors. Please see the AUTHORS file
|
|
// for details. All rights reserved. Use of this source code is governed by a
|
|
// BSD-style license that can be found in the LICENSE file.
|
|
|
|
#include "vm/metrics.h"
|
|
|
|
#include "vm/isolate.h"
|
|
#include "vm/json_stream.h"
|
|
#include "vm/log.h"
|
|
#include "vm/native_entry.h"
|
|
#include "vm/object.h"
|
|
#include "vm/runtime_entry.h"
|
|
|
|
namespace dart {
|
|
|
|
DEFINE_FLAG(bool,
|
|
print_metrics,
|
|
false,
|
|
"Print metrics when isolates (and the VM) are shutdown.");
|
|
|
|
Metric* Metric::vm_list_head_ = NULL;
|
|
|
|
Metric::Metric() : unit_(kCounter), value_(0) {}
|
|
Metric::~Metric() {}
|
|
|
|
void Metric::InitInstance(IsolateGroup* isolate_group,
|
|
const char* name,
|
|
const char* description,
|
|
Unit unit) {
|
|
// Only called once.
|
|
ASSERT(name != NULL);
|
|
isolate_group_ = isolate_group;
|
|
name_ = name;
|
|
description_ = description;
|
|
unit_ = unit;
|
|
}
|
|
|
|
#if !defined(PRODUCT)
|
|
void Metric::InitInstance(Isolate* isolate,
|
|
const char* name,
|
|
const char* description,
|
|
Unit unit) {
|
|
// Only called once.
|
|
ASSERT(name != NULL);
|
|
isolate_ = isolate;
|
|
name_ = name;
|
|
description_ = description;
|
|
unit_ = unit;
|
|
}
|
|
|
|
void Metric::InitInstance(const char* name,
|
|
const char* description,
|
|
Unit unit) {
|
|
// Only called once.
|
|
ASSERT(name != NULL);
|
|
name_ = name;
|
|
description_ = description;
|
|
unit_ = unit;
|
|
}
|
|
|
|
static const char* UnitString(intptr_t unit) {
|
|
switch (unit) {
|
|
case Metric::kCounter:
|
|
return "counter";
|
|
case Metric::kByte:
|
|
return "byte";
|
|
case Metric::kMicrosecond:
|
|
return "us";
|
|
default:
|
|
UNREACHABLE();
|
|
}
|
|
UNREACHABLE();
|
|
return NULL;
|
|
}
|
|
|
|
void Metric::PrintJSON(JSONStream* stream) {
|
|
JSONObject obj(stream);
|
|
obj.AddProperty("type", "Counter");
|
|
obj.AddProperty("name", name_);
|
|
obj.AddProperty("description", description_);
|
|
obj.AddProperty("unit", UnitString(unit()));
|
|
if (isolate_ == nullptr && isolate_group_ == nullptr) {
|
|
obj.AddFixedServiceId("vm/metrics/%s", name_);
|
|
} else {
|
|
obj.AddFixedServiceId("metrics/native/%s", name_);
|
|
}
|
|
// TODO(johnmccutchan): Overflow?
|
|
double value_as_double = static_cast<double>(Value());
|
|
obj.AddProperty("value", value_as_double);
|
|
}
|
|
#endif // !defined(PRODUCT)
|
|
|
|
char* Metric::ValueToString(int64_t value, Unit unit) {
|
|
Thread* thread = Thread::Current();
|
|
ASSERT(thread != NULL);
|
|
Zone* zone = thread->zone();
|
|
ASSERT(zone != NULL);
|
|
switch (unit) {
|
|
case kCounter:
|
|
return zone->PrintToString("%" Pd64 "", value);
|
|
case kByte: {
|
|
const char* scaled_suffix = "B";
|
|
double scaled_value = static_cast<double>(value);
|
|
if (value > GB) {
|
|
scaled_suffix = "GB";
|
|
scaled_value /= GB;
|
|
} else if (value > MB) {
|
|
scaled_suffix = "MB";
|
|
scaled_value /= MB;
|
|
} else if (value > KB) {
|
|
scaled_suffix = "kB";
|
|
scaled_value /= KB;
|
|
}
|
|
return zone->PrintToString("%.3f %s (%" Pd64 " B)", scaled_value,
|
|
scaled_suffix, value);
|
|
}
|
|
case kMicrosecond: {
|
|
const char* scaled_suffix = "us";
|
|
double scaled_value = static_cast<double>(value);
|
|
if (value > kMicrosecondsPerSecond) {
|
|
scaled_suffix = "s";
|
|
scaled_value /= kMicrosecondsPerSecond;
|
|
} else if (value > kMicrosecondsPerMillisecond) {
|
|
scaled_suffix = "ms";
|
|
scaled_value /= kMicrosecondsPerMillisecond;
|
|
}
|
|
return zone->PrintToString("%.3f %s (%" Pd64 " us)", scaled_value,
|
|
scaled_suffix, value);
|
|
}
|
|
default:
|
|
UNREACHABLE();
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
char* Metric::ToString() {
|
|
Thread* thread = Thread::Current();
|
|
ASSERT(thread != NULL);
|
|
Zone* zone = thread->zone();
|
|
ASSERT(zone != NULL);
|
|
return zone->PrintToString("%s %s", name(), ValueToString(Value(), unit()));
|
|
}
|
|
|
|
int64_t MetricHeapOldUsed::Value() const {
|
|
ASSERT(isolate_group() == IsolateGroup::Current());
|
|
return isolate_group()->heap()->UsedInWords(Heap::kOld) * kWordSize;
|
|
}
|
|
|
|
int64_t MetricHeapOldCapacity::Value() const {
|
|
ASSERT(isolate_group() == IsolateGroup::Current());
|
|
return isolate_group()->heap()->CapacityInWords(Heap::kOld) * kWordSize;
|
|
}
|
|
|
|
int64_t MetricHeapOldExternal::Value() const {
|
|
ASSERT(isolate_group() == IsolateGroup::Current());
|
|
return isolate_group()->heap()->ExternalInWords(Heap::kOld) * kWordSize;
|
|
}
|
|
|
|
int64_t MetricHeapNewUsed::Value() const {
|
|
ASSERT(isolate_group() == IsolateGroup::Current());
|
|
return isolate_group()->heap()->UsedInWords(Heap::kNew) * kWordSize;
|
|
}
|
|
|
|
int64_t MetricHeapNewCapacity::Value() const {
|
|
ASSERT(isolate_group() == IsolateGroup::Current());
|
|
return isolate_group()->heap()->CapacityInWords(Heap::kNew) * kWordSize;
|
|
}
|
|
|
|
int64_t MetricHeapNewExternal::Value() const {
|
|
ASSERT(isolate_group() == IsolateGroup::Current());
|
|
return isolate_group()->heap()->ExternalInWords(Heap::kNew) * kWordSize;
|
|
}
|
|
|
|
int64_t MetricHeapUsed::Value() const {
|
|
ASSERT(isolate_group() == IsolateGroup::Current());
|
|
return isolate_group()->heap()->UsedInWords(Heap::kNew) * kWordSize +
|
|
isolate_group()->heap()->UsedInWords(Heap::kOld) * kWordSize;
|
|
}
|
|
|
|
#if !defined(PRODUCT)
|
|
int64_t MetricIsolateCount::Value() const {
|
|
return Isolate::IsolateListLength();
|
|
}
|
|
|
|
int64_t MetricCurrentRSS::Value() const {
|
|
return Service::CurrentRSS();
|
|
}
|
|
|
|
int64_t MetricPeakRSS::Value() const {
|
|
return Service::MaxRSS();
|
|
}
|
|
#endif // !defined(PRODUCT)
|
|
|
|
#if !defined(PRODUCT)
|
|
|
|
#define VM_METRIC_VARIABLE(type, variable, name, unit) \
|
|
type vm_metric_##variable;
|
|
VM_METRIC_LIST(VM_METRIC_VARIABLE);
|
|
#undef VM_METRIC_VARIABLE
|
|
|
|
void Metric::Init() {
|
|
#define VM_METRIC_INIT(type, variable, name, unit) \
|
|
vm_metric_##variable.InitInstance(name, NULL, Metric::unit);
|
|
VM_METRIC_LIST(VM_METRIC_INIT);
|
|
#undef VM_METRIC_INIT
|
|
}
|
|
|
|
void Metric::Cleanup() {
|
|
if (FLAG_print_metrics) {
|
|
// Create a zone to allocate temporary strings in.
|
|
StackZone sz(Thread::Current());
|
|
OS::PrintErr("Printing metrics for VM\n");
|
|
|
|
#define VM_METRIC_INIT(type, variable, name, unit) \
|
|
OS::PrintErr("%s\n", vm_metric_##variable.ToString());
|
|
VM_METRIC_LIST(VM_METRIC_INIT);
|
|
#undef VM_METRIC_INIT
|
|
OS::PrintErr("\n");
|
|
}
|
|
}
|
|
|
|
#endif // !defined(PRODUCT)
|
|
|
|
MaxMetric::MaxMetric() : Metric() {
|
|
set_value(kMinInt64);
|
|
}
|
|
|
|
void MaxMetric::SetValue(int64_t new_value) {
|
|
if (new_value > value()) {
|
|
set_value(new_value);
|
|
}
|
|
}
|
|
|
|
MinMetric::MinMetric() : Metric() {
|
|
set_value(kMaxInt64);
|
|
}
|
|
|
|
void MinMetric::SetValue(int64_t new_value) {
|
|
if (new_value < value()) {
|
|
set_value(new_value);
|
|
}
|
|
}
|
|
|
|
} // namespace dart
|