| // 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 |