Skip to content

Commit c40a373

Browse files
prattmicgopherbot
authored andcommitted
internal/godebugs: add decoratemappings as an opaque godebug setting
This adds a new godebug to control whether the runtime applies the anonymous memory mapping annotations added in https://go.dev/cl/646095. It is enabled by default. This has several effects: * The feature is only enabled by default when the main go.mod has go >= 1.25. * This feature can be disabled with GODEBUG=decoratemappings=0, or the equivalents in go.mod or package main. See https://go.dev/doc/godebug. * As an opaque setting, this option will not appear in runtime/metrics. * This setting is non-atomic, so it cannot be changed after startup. I am not 100% sure about my decision for the last two points. I've made this an opaque setting because it affects every memory mapping the runtime performs. Thus every mapping would report "non-default behavior", which doesn't seem useful. This setting could trivially be atomic and allow changes at run time, but those changes would only affect future mappings. That seems confusing and not helpful. On the other hand, going back to annotate or unannotate every previous mapping when the setting changes is unwarranted complexity. For #71546. Change-Id: I6a6a636c5ad551d76691cba2a6f668d5cff0e352 Reviewed-on: https://go-review.googlesource.com/c/go/+/655895 Reviewed-by: Michael Knyszek <[email protected]> Reviewed-by: Ian Lance Taylor <[email protected]> LUCI-TryBot-Result: Go LUCI <[email protected]> Auto-Submit: Michael Pratt <[email protected]>
1 parent c3950f8 commit c40a373

File tree

8 files changed

+97
-1
lines changed

8 files changed

+97
-1
lines changed

doc/godebug.md

+11
Original file line numberDiff line numberDiff line change
@@ -153,6 +153,17 @@ for example,
153153
see the [runtime documentation](/pkg/runtime#hdr-Environment_Variables)
154154
and the [go command documentation](/cmd/go#hdr-Build_and_test_caching).
155155

156+
### Go 1.25
157+
158+
Go 1.25 added a new `decoratemappings` setting that controls whether the Go
159+
runtime annotates OS anonymous memory mappings with context about their
160+
purpose. These annotations appear in /proc/self/maps and /proc/self/smaps as
161+
"[anon: Go: ...]". This setting is only used on Linux. For Go 1.25, it defaults
162+
to `decoratemappings=1`, enabling annotations. Using `decoratemappings=0`
163+
reverts to the pre-Go 1.25 behavior. This setting is fixed at program startup
164+
time, and can't be modified by changing the `GODEBUG` environment variable
165+
after the program starts.
166+
156167
### Go 1.24
157168

158169
Go 1.24 added a new `fips140` setting that controls whether the Go

doc/next/4-runtime.md

+8
Original file line numberDiff line numberDiff line change
@@ -16,3 +16,11 @@ value would print:
1616
This program will now print:
1717

1818
panic: PANIC [recovered, reraised]
19+
20+
<!-- go.dev/issue/71546 -->
21+
22+
On Linux systems with kernel support for anonymous VMA names
23+
(`CONFIG_ANON_VMA_NAME`), the Go runtime will annotate anonymous memory
24+
mappings with context about their purpose. e.g., `[anon: Go: heap]` for heap
25+
memory. This can be disabled with the [GODEBUG setting](/doc/godebug)
26+
`decoratemappings=0`.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
env GO111MODULE=on
2+
3+
# Go 1.24 module should disable decoratemappings.
4+
go list -f '{{.Module.GoVersion}} {{.DefaultGODEBUG}}'
5+
stdout decoratemappings=0
6+
7+
[!GOOS:linux] skip
8+
[short] skip
9+
10+
# Programs in Go 1.24 module should never see annotations. This ensures that
11+
# the runtime has not overridden the default.
12+
go run .
13+
14+
-- go.mod --
15+
go 1.24
16+
module m
17+
18+
-- main.go --
19+
package main
20+
21+
import (
22+
"log"
23+
"os"
24+
"strings"
25+
)
26+
27+
func main() {
28+
b, err := os.ReadFile("/proc/self/maps")
29+
if err != nil {
30+
log.Fatalf("Error reading: %v", err)
31+
}
32+
33+
if strings.Contains(string(b), "[anon: Go:") {
34+
log.Printf("/proc/self/maps:\n%s", string(b))
35+
log.Fatalf("/proc/self/maps contains Go annotation")
36+
}
37+
}

src/internal/godebugs/table.go

+1
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,7 @@ type Info struct {
2727
var All = []Info{
2828
{Name: "asynctimerchan", Package: "time", Changed: 23, Old: "1"},
2929
{Name: "dataindependenttiming", Package: "crypto/subtle", Opaque: true},
30+
{Name: "decoratemappings", Package: "runtime", Opaque: true, Changed: 25, Old: "0"},
3031
{Name: "execerrdot", Package: "os/exec"},
3132
{Name: "fips140", Package: "crypto/fips140", Opaque: true},
3233
{Name: "gocachehash", Package: "cmd/go"},

src/runtime/extern.go

+7
Original file line numberDiff line numberDiff line change
@@ -52,6 +52,13 @@ It is a comma-separated list of name=val pairs setting these named variables:
5252
cgocheck mode can be enabled using GOEXPERIMENT (which
5353
requires a rebuild), see https://pkg.go.dev/internal/goexperiment for details.
5454
55+
decoratemappings: controls whether the Go runtime annotates OS
56+
anonymous memory mappings with context about their purpose. These
57+
annotations appear in /proc/self/maps and /proc/self/smaps as
58+
"[anon: Go: ...]". This setting is only used on Linux. For Go 1.25, it
59+
defaults to `decoratemappings=1`, enabling annotations. Using
60+
`decoratemappings=0` reverts to the pre-Go 1.25 behavior.
61+
5562
disablethp: setting disablethp=1 on Linux disables transparent huge pages for the heap.
5663
It has no effect on other platforms. disablethp is meant for compatibility with versions
5764
of Go before 1.21, which stopped working around a Linux kernel default that can result

src/runtime/runtime1.go

+2
Original file line numberDiff line numberDiff line change
@@ -309,6 +309,7 @@ type dbgVar struct {
309309
var debug struct {
310310
cgocheck int32
311311
clobberfree int32
312+
decoratemappings int32
312313
disablethp int32
313314
dontfreezetheworld int32
314315
efence int32
@@ -369,6 +370,7 @@ var dbgvars = []*dbgVar{
369370
{name: "cgocheck", value: &debug.cgocheck},
370371
{name: "clobberfree", value: &debug.clobberfree},
371372
{name: "dataindependenttiming", value: &debug.dataindependenttiming},
373+
{name: "decoratemappings", value: &debug.decoratemappings, def: 1},
372374
{name: "disablethp", value: &debug.disablethp},
373375
{name: "dontfreezetheworld", value: &debug.dontfreezetheworld},
374376
{name: "efence", value: &debug.efence},

src/runtime/set_vma_name_linux.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ var prSetVMAUnsupported atomic.Bool
1616

1717
// setVMAName calls prctl(PR_SET_VMA, PR_SET_VMA_ANON_NAME, start, len, name)
1818
func setVMAName(start unsafe.Pointer, length uintptr, name string) {
19-
if unsupported := prSetVMAUnsupported.Load(); unsupported {
19+
if debug.decoratemappings == 0 || prSetVMAUnsupported.Load() {
2020
return
2121
}
2222

test/decoratemappingszero.go

+30
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
// run
2+
3+
// Copyright 2025 The Go Authors. All rights reserved.
4+
// Use of this source code is governed by a BSD-style
5+
// license that can be found in the LICENSE file.
6+
7+
// Disable mapping annotations, which only exists for Linux.
8+
9+
//go:debug decoratemappings=0
10+
//go:build linux
11+
12+
package main
13+
14+
import (
15+
"log"
16+
"os"
17+
"strings"
18+
)
19+
20+
func main() {
21+
b, err := os.ReadFile("/proc/self/maps")
22+
if err != nil {
23+
log.Fatalf("Error reading: %v", err)
24+
}
25+
26+
if strings.Contains(string(b), "[anon: Go:") {
27+
log.Printf("/proc/self/maps:\n%s", string(b))
28+
log.Fatalf("/proc/self/maps contains Go annotation")
29+
}
30+
}

0 commit comments

Comments
 (0)