...
Source file
src/runtime/mcheckmark.go
Documentation: runtime
1
2
3
4
5
6
7
8
9
10
11
12
13 package runtime
14
15 import (
16 "internal/goarch"
17 "runtime/internal/atomic"
18 "runtime/internal/sys"
19 "unsafe"
20 )
21
22
23
24
25
26 type checkmarksMap struct {
27 _ sys.NotInHeap
28 b [heapArenaBytes / goarch.PtrSize / 8]uint8
29 }
30
31
32
33 var useCheckmark = false
34
35
36
37
38 func startCheckmarks() {
39 assertWorldStopped()
40
41
42 for _, ai := range mheap_.allArenas {
43 arena := mheap_.arenas[ai.l1()][ai.l2()]
44 bitmap := arena.checkmarks
45
46 if bitmap == nil {
47
48 bitmap = (*checkmarksMap)(persistentalloc(unsafe.Sizeof(*bitmap), 0, &memstats.gcMiscSys))
49 if bitmap == nil {
50 throw("out of memory allocating checkmarks bitmap")
51 }
52 arena.checkmarks = bitmap
53 } else {
54
55 for i := range bitmap.b {
56 bitmap.b[i] = 0
57 }
58 }
59 }
60
61 useCheckmark = true
62 }
63
64
65 func endCheckmarks() {
66 if gcMarkWorkAvailable(nil) {
67 throw("GC work not flushed")
68 }
69 useCheckmark = false
70 }
71
72
73
74
75 func setCheckmark(obj, base, off uintptr, mbits markBits) bool {
76 if !mbits.isMarked() {
77 printlock()
78 print("runtime: checkmarks found unexpected unmarked object obj=", hex(obj), "\n")
79 print("runtime: found obj at *(", hex(base), "+", hex(off), ")\n")
80
81
82 gcDumpObject("base", base, off)
83
84
85 gcDumpObject("obj", obj, ^uintptr(0))
86
87 getg().m.traceback = 2
88 throw("checkmark found unmarked object")
89 }
90
91 ai := arenaIndex(obj)
92 arena := mheap_.arenas[ai.l1()][ai.l2()]
93 arenaWord := (obj / heapArenaBytes / 8) % uintptr(len(arena.checkmarks.b))
94 mask := byte(1 << ((obj / heapArenaBytes) % 8))
95 bytep := &arena.checkmarks.b[arenaWord]
96
97 if atomic.Load8(bytep)&mask != 0 {
98
99 return true
100 }
101
102 atomic.Or8(bytep, mask)
103 return false
104 }
105
View as plain text