mirror of
https://github.com/golang/go.git
synced 2025-12-08 06:10:04 +00:00
runtime: make GC see object as allocated after it is initialized
When the GC is scanning some memory (possibly conservatively), finding a pointer, while concurrently another goroutine is allocating an object at the same address as the found pointer, the GC may see the pointer before the object and/or the heap bits are initialized. This may cause the GC to see bad pointers and possibly crash. To prevent this, we make it that the scanner can only see the object as allocated after the object and the heap bits are initialized. Currently the allocator uses freeindex to find the next available slot, and that code is coupled with updating the free index to a new slot past it. The scanner also uses the freeindex to determine if an object is allocated. This is somewhat racy. This CL makes the scanner use a different field, which is only updated after the object initialization (and a memory barrier). Fixes #54596. Change-Id: I2a57a226369926e7192c253dd0d21d3faf22297c Reviewed-on: https://go-review.googlesource.com/c/go/+/449017 Reviewed-by: Austin Clements <austin@google.com> Reviewed-by: Michael Knyszek <mknyszek@google.com> Run-TryBot: Cherry Mui <cherryyz@google.com> TryBot-Result: Gopher Robot <gobot@golang.org>
This commit is contained in:
parent
d52883f443
commit
febe7b8e2a
5 changed files with 24 additions and 1 deletions
|
|
@ -487,6 +487,14 @@ type mspan struct {
|
|||
speciallock mutex // guards specials list
|
||||
specials *special // linked list of special records sorted by offset.
|
||||
userArenaChunkFree addrRange // interval for managing chunk allocation
|
||||
|
||||
// freeIndexForScan is like freeindex, except that freeindex is
|
||||
// used by the allocator whereas freeIndexForScan is used by the
|
||||
// GC scanner. They are two fields so that the GC sees the object
|
||||
// is allocated only when the object and the heap bits are
|
||||
// initialized (see also the assignment of freeIndexForScan in
|
||||
// mallocgc, and issue 54596).
|
||||
freeIndexForScan uintptr
|
||||
}
|
||||
|
||||
func (s *mspan) base() uintptr {
|
||||
|
|
@ -1386,6 +1394,7 @@ func (h *mheap) initSpan(s *mspan, typ spanAllocType, spanclass spanClass, base,
|
|||
|
||||
// Initialize mark and allocation structures.
|
||||
s.freeindex = 0
|
||||
s.freeIndexForScan = 0
|
||||
s.allocCache = ^uint64(0) // all 1s indicating all free.
|
||||
s.gcmarkBits = newMarkBits(s.nelems)
|
||||
s.allocBits = newAllocBits(s.nelems)
|
||||
|
|
@ -1657,6 +1666,7 @@ func (span *mspan) init(base uintptr, npages uintptr) {
|
|||
span.specials = nil
|
||||
span.needzero = 0
|
||||
span.freeindex = 0
|
||||
span.freeIndexForScan = 0
|
||||
span.allocBits = nil
|
||||
span.gcmarkBits = nil
|
||||
span.state.set(mSpanDead)
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue