blob: 2b7c7a9c551f46cbaa0321d1e6553c6bb1b6bff1 [file] [log] [blame]
// 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.
/*
#cgo noescape runCWithNoEscape
#cgo nocallback 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")
}