mirror of
https://github.com/golang/go.git
synced 2025-12-08 06:10:04 +00:00
runtime,runtime/metrics: add memory metrics
This change adds support for a variety of runtime memory metrics and contains the base implementation of Read for the runtime/metrics package, which lives in the runtime. It also adds testing infrastructure for the metrics package, and a bunch of format and documentation tests. For #37112. Change-Id: I16a2c4781eeeb2de0abcb045c15105f1210e2d8a Reviewed-on: https://go-review.googlesource.com/c/go/+/247041 Run-TryBot: Michael Knyszek <mknyszek@google.com> TryBot-Result: Go Bot <gobot@golang.org> Reviewed-by: Michael Pratt <mpratt@google.com> Trust: Michael Knyszek <mknyszek@google.com>
This commit is contained in:
parent
79781e8dd3
commit
b08dfbaa43
9 changed files with 781 additions and 6 deletions
114
src/runtime/metrics_test.go
Normal file
114
src/runtime/metrics_test.go
Normal file
|
|
@ -0,0 +1,114 @@
|
|||
// Copyright 2020 The Go Authors. All rights reserved.
|
||||
// Use of this source code is governed by a BSD-style
|
||||
// license that can be found in the LICENSE file.
|
||||
|
||||
package runtime_test
|
||||
|
||||
import (
|
||||
"runtime"
|
||||
"runtime/metrics"
|
||||
"strings"
|
||||
"testing"
|
||||
"unsafe"
|
||||
)
|
||||
|
||||
func prepareAllMetricsSamples() (map[string]metrics.Description, []metrics.Sample) {
|
||||
all := metrics.All()
|
||||
samples := make([]metrics.Sample, len(all))
|
||||
descs := make(map[string]metrics.Description)
|
||||
for i := range all {
|
||||
samples[i].Name = all[i].Name
|
||||
descs[all[i].Name] = all[i]
|
||||
}
|
||||
return descs, samples
|
||||
}
|
||||
|
||||
func TestReadMetrics(t *testing.T) {
|
||||
// Tests whether readMetrics produces values aligning
|
||||
// with ReadMemStats while the world is stopped.
|
||||
var mstats runtime.MemStats
|
||||
_, samples := prepareAllMetricsSamples()
|
||||
runtime.ReadMetricsSlow(&mstats, unsafe.Pointer(&samples[0]), len(samples), cap(samples))
|
||||
|
||||
checkUint64 := func(t *testing.T, m string, got, want uint64) {
|
||||
t.Helper()
|
||||
if got != want {
|
||||
t.Errorf("metric %q: got %d, want %d", m, got, want)
|
||||
}
|
||||
}
|
||||
|
||||
// Check to make sure the values we read line up with other values we read.
|
||||
for i := range samples {
|
||||
switch name := samples[i].Name; name {
|
||||
case "/memory/classes/heap/free:bytes":
|
||||
checkUint64(t, name, samples[i].Value.Uint64(), mstats.HeapIdle-mstats.HeapReleased)
|
||||
case "/memory/classes/heap/released:bytes":
|
||||
checkUint64(t, name, samples[i].Value.Uint64(), mstats.HeapReleased)
|
||||
case "/memory/classes/heap/objects:bytes":
|
||||
checkUint64(t, name, samples[i].Value.Uint64(), mstats.HeapAlloc)
|
||||
case "/memory/classes/heap/unused:bytes":
|
||||
checkUint64(t, name, samples[i].Value.Uint64(), mstats.HeapInuse-mstats.HeapAlloc)
|
||||
case "/memory/classes/heap/stacks:bytes":
|
||||
checkUint64(t, name, samples[i].Value.Uint64(), mstats.StackInuse)
|
||||
case "/memory/classes/metadata/mcache/free:bytes":
|
||||
checkUint64(t, name, samples[i].Value.Uint64(), mstats.MCacheSys-mstats.MCacheInuse)
|
||||
case "/memory/classes/metadata/mcache/inuse:bytes":
|
||||
checkUint64(t, name, samples[i].Value.Uint64(), mstats.MCacheInuse)
|
||||
case "/memory/classes/metadata/mspan/free:bytes":
|
||||
checkUint64(t, name, samples[i].Value.Uint64(), mstats.MSpanSys-mstats.MSpanInuse)
|
||||
case "/memory/classes/metadata/mspan/inuse:bytes":
|
||||
checkUint64(t, name, samples[i].Value.Uint64(), mstats.MSpanInuse)
|
||||
case "/memory/classes/metadata/other:bytes":
|
||||
checkUint64(t, name, samples[i].Value.Uint64(), mstats.GCSys)
|
||||
case "/memory/classes/os-stacks:bytes":
|
||||
checkUint64(t, name, samples[i].Value.Uint64(), mstats.StackSys-mstats.StackInuse)
|
||||
case "/memory/classes/other:bytes":
|
||||
checkUint64(t, name, samples[i].Value.Uint64(), mstats.OtherSys)
|
||||
case "/memory/classes/profiling/buckets:bytes":
|
||||
checkUint64(t, name, samples[i].Value.Uint64(), mstats.BuckHashSys)
|
||||
case "/memory/classes/total:bytes":
|
||||
checkUint64(t, name, samples[i].Value.Uint64(), mstats.Sys)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func TestReadMetricsConsistency(t *testing.T) {
|
||||
// Tests whether readMetrics produces consistent, sensible values.
|
||||
// The values are read concurrently with the runtime doing other
|
||||
// things (e.g. allocating) so what we read can't reasonably compared
|
||||
// to runtime values.
|
||||
|
||||
// Read all the supported metrics through the metrics package.
|
||||
descs, samples := prepareAllMetricsSamples()
|
||||
metrics.Read(samples)
|
||||
|
||||
// Check to make sure the values we read make sense.
|
||||
var totalVirtual struct {
|
||||
got, want uint64
|
||||
}
|
||||
for i := range samples {
|
||||
kind := samples[i].Value.Kind()
|
||||
if want := descs[samples[i].Name].Kind; kind != want {
|
||||
t.Errorf("supported metric %q has unexpected kind: got %d, want %d", samples[i].Name, kind, want)
|
||||
continue
|
||||
}
|
||||
if samples[i].Name != "/memory/classes/total:bytes" && strings.HasPrefix(samples[i].Name, "/memory/classes") {
|
||||
v := samples[i].Value.Uint64()
|
||||
totalVirtual.want += v
|
||||
|
||||
// None of these stats should ever get this big.
|
||||
// If they do, there's probably overflow involved,
|
||||
// usually due to bad accounting.
|
||||
if int64(v) < 0 {
|
||||
t.Errorf("%q has high/negative value: %d", samples[i].Name, v)
|
||||
}
|
||||
}
|
||||
switch samples[i].Name {
|
||||
case "/memory/classes/total:bytes":
|
||||
totalVirtual.got = samples[i].Value.Uint64()
|
||||
}
|
||||
}
|
||||
if totalVirtual.got != totalVirtual.want {
|
||||
t.Errorf(`"/memory/classes/total:bytes" does not match sum of /memory/classes/**: got %d, want %d`, totalVirtual.got, totalVirtual.want)
|
||||
}
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue