blob: f5eebac677ad1c9a669b9c3fd3a328928a4d30cb (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
|
// Copyright 2023 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 main
// #cgo noescape annotations for a C function means its arguments won't escape to heap.
// We assume that there won't be 100 new allocated heap objects in other places,
// i.e. runtime.ReadMemStats or other runtime background works.
// So, the tests are:
// 1. at least 100 new allocated heap objects after invoking withoutNoEscape 100 times.
// 2. less than 100 new allocated heap objects after invoking withoutNoEscape 100 times.
/*
// TODO(#56378): #cgo noescape runCWithNoEscape
void runCWithNoEscape(void *p) {
}
void runCWithoutNoEscape(void *p) {
}
*/
import "C"
import (
"fmt"
"runtime"
"runtime/debug"
"unsafe"
)
const num = 100
func init() {
register("CgoNoEscape", CgoNoEscape)
}
//go:noinline
func withNoEscape() {
var str string
C.runCWithNoEscape(unsafe.Pointer(&str))
}
//go:noinline
func withoutNoEscape() {
var str string
C.runCWithoutNoEscape(unsafe.Pointer(&str))
}
func CgoNoEscape() {
// make GC stop to see the heap objects allocated
debug.SetGCPercent(-1)
var stats runtime.MemStats
runtime.ReadMemStats(&stats)
preHeapObjects := stats.HeapObjects
for i := 0; i < num; i++ {
withNoEscape()
}
runtime.ReadMemStats(&stats)
nowHeapObjects := stats.HeapObjects
if nowHeapObjects-preHeapObjects >= num {
fmt.Printf("too many heap objects allocated, pre: %v, now: %v\n", preHeapObjects, nowHeapObjects)
}
runtime.ReadMemStats(&stats)
preHeapObjects = stats.HeapObjects
for i := 0; i < num; i++ {
withoutNoEscape()
}
runtime.ReadMemStats(&stats)
nowHeapObjects = stats.HeapObjects
if nowHeapObjects-preHeapObjects < num {
fmt.Printf("too few heap objects allocated, pre: %v, now: %v\n", preHeapObjects, nowHeapObjects)
}
fmt.Println("OK")
}
|