mirror of
https://github.com/golang/go.git
synced 2025-12-08 06:10:04 +00:00
cmd/compile,runtime: implement stack objects
Rework how the compiler+runtime handles stack-allocated variables whose address is taken. Direct references to such variables work as before. References through pointers, however, use a new mechanism. The new mechanism is more precise than the old "ambiguously live" mechanism. It computes liveness at runtime based on the actual references among objects on the stack. Each function records all of its address-taken objects in a FUNCDATA. These are called "stack objects". The runtime then uses that information while scanning a stack to find all of the stack objects on a stack. It then does a mark phase on the stack objects, using all the pointers found on the stack (and ancillary structures, like defer records) as the root set. Only stack objects which are found to be live during this mark phase will be scanned and thus retain any heap objects they point to. A subsequent CL will remove all the "ambiguously live" logic from the compiler, so that the stack object tracing will be required. For this CL, the stack tracing is all redundant with the current ambiguously live logic. Update #22350 Change-Id: Ide19f1f71a5b6ec8c4d54f8f66f0e9a98344772f Reviewed-on: https://go-review.googlesource.com/c/134155 Reviewed-by: Austin Clements <austin@google.com>
This commit is contained in:
parent
433496615f
commit
cbafcc55e8
13 changed files with 607 additions and 27 deletions
|
|
@ -625,7 +625,7 @@ func adjustframe(frame *stkframe, arg unsafe.Pointer) bool {
|
|||
return true
|
||||
}
|
||||
|
||||
locals, args := getStackMap(frame, &adjinfo.cache, true)
|
||||
locals, args, objs := getStackMap(frame, &adjinfo.cache, true)
|
||||
|
||||
// Adjust local variables if stack frame has been allocated.
|
||||
if locals.n > 0 {
|
||||
|
|
@ -663,6 +663,42 @@ func adjustframe(frame *stkframe, arg unsafe.Pointer) bool {
|
|||
}
|
||||
adjustpointers(unsafe.Pointer(frame.argp), &args, adjinfo, funcInfo{})
|
||||
}
|
||||
|
||||
// Adjust pointers in all stack objects (whether they are live or not).
|
||||
// See comments in mgcmark.go:scanframeworker.
|
||||
if frame.varp != 0 {
|
||||
for _, obj := range objs {
|
||||
off := obj.off
|
||||
base := frame.varp // locals base pointer
|
||||
if off >= 0 {
|
||||
base = frame.argp // arguments and return values base pointer
|
||||
}
|
||||
p := base + uintptr(off)
|
||||
if p < frame.sp {
|
||||
// Object hasn't been allocated in the frame yet.
|
||||
// (Happens when the stack bounds check fails and
|
||||
// we call into morestack.)
|
||||
continue
|
||||
}
|
||||
t := obj.typ
|
||||
gcdata := t.gcdata
|
||||
var s *mspan
|
||||
if t.kind&kindGCProg != 0 {
|
||||
// See comments in mgcmark.go:scanstack
|
||||
s = materializeGCProg(t.ptrdata, gcdata)
|
||||
gcdata = (*byte)(unsafe.Pointer(s.startAddr))
|
||||
}
|
||||
for i := uintptr(0); i < t.ptrdata; i += sys.PtrSize {
|
||||
if *addb(gcdata, i/(8*sys.PtrSize))>>(i/sys.PtrSize&7)&1 != 0 {
|
||||
adjustpointer(adjinfo, unsafe.Pointer(p+i))
|
||||
}
|
||||
}
|
||||
if s != nil {
|
||||
dematerializeGCProg(s)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return true
|
||||
}
|
||||
|
||||
|
|
@ -1136,9 +1172,9 @@ func freeStackSpans() {
|
|||
unlock(&stackLarge.lock)
|
||||
}
|
||||
|
||||
// getStackMap returns the locals and arguments live pointer maps for
|
||||
// frame.
|
||||
func getStackMap(frame *stkframe, cache *pcvalueCache, debug bool) (locals, args bitvector) {
|
||||
// getStackMap returns the locals and arguments live pointer maps, and
|
||||
// stack object list for frame.
|
||||
func getStackMap(frame *stkframe, cache *pcvalueCache, debug bool) (locals, args bitvector, objs []stackObjectRecord) {
|
||||
targetpc := frame.continpc
|
||||
if targetpc == 0 {
|
||||
// Frame is dead. Return empty bitvectors.
|
||||
|
|
@ -1235,9 +1271,33 @@ func getStackMap(frame *stkframe, cache *pcvalueCache, debug bool) (locals, args
|
|||
}
|
||||
}
|
||||
}
|
||||
|
||||
// stack objects.
|
||||
p := funcdata(f, _FUNCDATA_StackObjects)
|
||||
if p != nil {
|
||||
n := *(*uintptr)(p)
|
||||
p = add(p, sys.PtrSize)
|
||||
*(*slice)(unsafe.Pointer(&objs)) = slice{array: noescape(p), len: int(n), cap: int(n)}
|
||||
// Note: the noescape above is needed to keep
|
||||
// getStackMap from from "leaking param content:
|
||||
// frame". That leak propagates up to getgcmask, then
|
||||
// GCMask, then verifyGCInfo, which converts the stack
|
||||
// gcinfo tests into heap gcinfo tests :(
|
||||
}
|
||||
|
||||
return
|
||||
}
|
||||
|
||||
// A stackObjectRecord is generated by the compiler for each stack object in a stack frame.
|
||||
// This record must match the generator code in cmd/compile/internal/gc/ssa.go:emitStackObjects.
|
||||
type stackObjectRecord struct {
|
||||
// offset in frame
|
||||
// if negative, offset from varp
|
||||
// if non-negative, offset from argp
|
||||
off int
|
||||
typ *_type
|
||||
}
|
||||
|
||||
//go:nosplit
|
||||
func morestackc() {
|
||||
throw("attempt to execute system stack code on user stack")
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue