X7ROOT File Manager
Current Path:
/opt/golang/1.22.0/src/runtime
opt
/
golang
/
1.22.0
/
src
/
runtime
/
📁
..
📄
HACKING.md
(13.85 KB)
📄
Makefile
(178 B)
📄
abi_test.go
(2.83 KB)
📄
alg.go
(10.99 KB)
📄
align_runtime_test.go
(1.82 KB)
📄
align_test.go
(5.37 KB)
📄
arena.go
(31.66 KB)
📄
arena_test.go
(13.39 KB)
📁
asan
📄
asan.go
(1.55 KB)
📄
asan0.go
(760 B)
📄
asan_amd64.s
(2.45 KB)
📄
asan_arm64.s
(2.14 KB)
📄
asan_loong64.s
(2.12 KB)
📄
asan_ppc64le.s
(2.75 KB)
📄
asan_riscv64.s
(1.92 KB)
📄
asm.s
(719 B)
📄
asm_386.s
(42.54 KB)
📄
asm_amd64.h
(631 B)
📄
asm_amd64.s
(60.09 KB)
📄
asm_arm.s
(31.58 KB)
📄
asm_arm64.s
(43.33 KB)
📄
asm_loong64.s
(27.58 KB)
📄
asm_mips64x.s
(24.34 KB)
📄
asm_mipsx.s
(25.82 KB)
📄
asm_ppc64x.h
(1.93 KB)
📄
asm_ppc64x.s
(45.19 KB)
📄
asm_riscv64.s
(26.97 KB)
📄
asm_s390x.s
(27.55 KB)
📄
asm_wasm.s
(11.82 KB)
📄
atomic_arm64.s
(259 B)
📄
atomic_loong64.s
(245 B)
📄
atomic_mips64x.s
(300 B)
📄
atomic_mipsx.s
(262 B)
📄
atomic_pointer.go
(3.66 KB)
📄
atomic_ppc64x.s
(437 B)
📄
atomic_riscv64.s
(275 B)
📄
auxv_none.go
(298 B)
📄
callers_test.go
(12.13 KB)
📁
cgo
📄
cgo.go
(2.5 KB)
📄
cgo_mmap.go
(2.42 KB)
📄
cgo_ppc64x.go
(418 B)
📄
cgo_sigaction.go
(3.28 KB)
📄
cgocall.go
(23.23 KB)
📄
cgocallback.go
(317 B)
📄
cgocheck.go
(7.97 KB)
📄
chan.go
(23.74 KB)
📄
chan_test.go
(23.44 KB)
📄
chanbarrier_test.go
(1.4 KB)
📄
checkptr.go
(3.29 KB)
📄
checkptr_test.go
(2.86 KB)
📄
closure_test.go
(937 B)
📄
compiler.go
(410 B)
📄
complex.go
(1.59 KB)
📄
complex_test.go
(1.05 KB)
📄
conv_wasm_test.go
(2.96 KB)
📄
coro.go
(4.92 KB)
📁
coverage
📄
covercounter.go
(749 B)
📄
covermeta.go
(2.4 KB)
📄
cpuflags.go
(810 B)
📄
cpuflags_amd64.go
(533 B)
📄
cpuflags_arm64.go
(312 B)
📄
cpuprof.go
(7.94 KB)
📄
cputicks.go
(437 B)
📄
crash_cgo_test.go
(23.35 KB)
📄
crash_test.go
(23.12 KB)
📄
crash_unix_test.go
(9.19 KB)
📄
create_file_nounix.go
(305 B)
📄
create_file_unix.go
(368 B)
📁
debug
📄
debug.go
(3.53 KB)
📄
debug_test.go
(7.99 KB)
📄
debugcall.go
(6.75 KB)
📄
debuglog.go
(18.24 KB)
📄
debuglog_off.go
(357 B)
📄
debuglog_on.go
(1.09 KB)
📄
debuglog_test.go
(4.9 KB)
📄
defer_test.go
(11.4 KB)
📄
defs1_linux.go
(845 B)
📄
defs1_netbsd_386.go
(2.91 KB)
📄
defs1_netbsd_amd64.go
(3.14 KB)
📄
defs1_netbsd_arm.go
(3.03 KB)
📄
defs1_netbsd_arm64.go
(3.25 KB)
📄
defs1_solaris_amd64.go
(4.01 KB)
📄
defs2_linux.go
(3.22 KB)
📄
defs3_linux.go
(1.09 KB)
📄
defs_aix.go
(4.17 KB)
📄
defs_aix_ppc64.go
(3.63 KB)
📄
defs_arm_linux.go
(2.67 KB)
📄
defs_darwin.go
(4.18 KB)
📄
defs_darwin_amd64.go
(6.34 KB)
📄
defs_darwin_arm64.go
(4.17 KB)
📄
defs_dragonfly.go
(2.73 KB)
📄
defs_dragonfly_amd64.go
(3.41 KB)
📄
defs_freebsd.go
(3.96 KB)
📄
defs_freebsd_386.go
(4.52 KB)
📄
defs_freebsd_amd64.go
(4.79 KB)
📄
defs_freebsd_arm.go
(3.92 KB)
📄
defs_freebsd_arm64.go
(4.18 KB)
📄
defs_freebsd_riscv64.go
(4.19 KB)
📄
defs_illumos_amd64.go
(285 B)
📄
defs_linux.go
(2.92 KB)
📄
defs_linux_386.go
(4.2 KB)
📄
defs_linux_amd64.go
(4.7 KB)
📄
defs_linux_arm.go
(3.89 KB)
📄
defs_linux_arm64.go
(3.62 KB)
📄
defs_linux_loong64.go
(3.43 KB)
📄
defs_linux_mips64x.go
(3.6 KB)
📄
defs_linux_mipsx.go
(3.6 KB)
📄
defs_linux_ppc64.go
(3.69 KB)
📄
defs_linux_ppc64le.go
(3.69 KB)
📄
defs_linux_riscv64.go
(3.81 KB)
📄
defs_linux_s390x.go
(3.16 KB)
📄
defs_netbsd.go
(2.83 KB)
📄
defs_netbsd_386.go
(855 B)
📄
defs_netbsd_amd64.go
(1.01 KB)
📄
defs_netbsd_arm.go
(764 B)
📄
defs_openbsd.go
(3.06 KB)
📄
defs_openbsd_386.go
(2.91 KB)
📄
defs_openbsd_amd64.go
(3.11 KB)
📄
defs_openbsd_arm.go
(3.03 KB)
📄
defs_openbsd_arm64.go
(2.78 KB)
📄
defs_openbsd_mips64.go
(2.75 KB)
📄
defs_openbsd_ppc64.go
(3 KB)
📄
defs_openbsd_riscv64.go
(2.89 KB)
📄
defs_plan9_386.go
(1.63 KB)
📄
defs_plan9_amd64.go
(1.82 KB)
📄
defs_plan9_arm.go
(1.73 KB)
📄
defs_solaris.go
(3.32 KB)
📄
defs_solaris_amd64.go
(1004 B)
📄
defs_windows.go
(2.25 KB)
📄
defs_windows_386.go
(2.28 KB)
📄
defs_windows_amd64.go
(3.19 KB)
📄
defs_windows_arm.go
(2.57 KB)
📄
defs_windows_arm64.go
(3.07 KB)
📄
duff_386.s
(8.24 KB)
📄
duff_amd64.s
(5.64 KB)
📄
duff_arm.s
(7.11 KB)
📄
duff_arm64.s
(5.27 KB)
📄
duff_loong64.s
(11.9 KB)
📄
duff_mips64x.s
(11.28 KB)
📄
duff_ppc64x.s
(7.06 KB)
📄
duff_riscv64.s
(11.4 KB)
📄
duff_s390x.s
(507 B)
📄
ehooks_test.go
(2.04 KB)
📄
env_plan9.go
(3 KB)
📄
env_posix.go
(1.56 KB)
📄
env_test.go
(1.16 KB)
📄
error.go
(9.29 KB)
📄
example_test.go
(1.55 KB)
📄
exithook.go
(2.32 KB)
📄
export_aix_test.go
(207 B)
📄
export_arm_test.go
(226 B)
📄
export_darwin_test.go
(207 B)
📄
export_debug_amd64_test.go
(3.6 KB)
📄
export_debug_arm64_test.go
(3.49 KB)
📄
export_debug_ppc64le_test.go
(3.5 KB)
📄
export_debug_test.go
(5.07 KB)
📄
export_debuglog_test.go
(1.27 KB)
📄
export_linux_test.go
(378 B)
📄
export_mmap_test.go
(429 B)
📄
export_pipe2_test.go
(310 B)
📄
export_pipe_test.go
(219 B)
📄
export_test.go
(50.52 KB)
📄
export_unix_test.go
(2.27 KB)
📄
export_windows_test.go
(903 B)
📄
extern.go
(18.58 KB)
📄
fastlog2.go
(1.22 KB)
📄
fastlog2_test.go
(784 B)
📄
fastlog2table.go
(904 B)
📄
fds_nonunix.go
(256 B)
📄
fds_test.go
(1.43 KB)
📄
fds_unix.go
(1.27 KB)
📄
float.go
(1.35 KB)
📄
float_test.go
(699 B)
📄
funcdata.h
(2.53 KB)
📄
gc_test.go
(20.32 KB)
📄
gcinfo_test.go
(5.95 KB)
📄
go_tls.h
(366 B)
📄
hash32.go
(1.58 KB)
📄
hash64.go
(1.95 KB)
📄
hash_test.go
(17.24 KB)
📄
heap_test.go
(529 B)
📄
heapdump.go
(17.88 KB)
📄
histogram.go
(7.3 KB)
📄
histogram_test.go
(3.51 KB)
📄
iface.go
(20.92 KB)
📄
iface_test.go
(7.45 KB)
📄
import_test.go
(1.42 KB)
📄
importx_test.go
(763 B)
📁
internal
📄
lfstack.go
(2.03 KB)
📄
lfstack_test.go
(2.74 KB)
📄
libfuzzer.go
(6.34 KB)
📄
libfuzzer_amd64.s
(5.03 KB)
📄
libfuzzer_arm64.s
(3.15 KB)
📄
lock_futex.go
(5.4 KB)
📄
lock_js.go
(7.28 KB)
📄
lock_sema.go
(6.75 KB)
📄
lock_wasip1.go
(2.01 KB)
📄
lockrank.go
(18.19 KB)
📄
lockrank_off.go
(1.17 KB)
📄
lockrank_on.go
(10.27 KB)
📄
lockrank_test.go
(856 B)
📄
malloc.go
(58.5 KB)
📄
malloc_test.go
(10.64 KB)
📄
map.go
(52.17 KB)
📄
map_benchmark_test.go
(10.59 KB)
📄
map_fast32.go
(12.74 KB)
📄
map_fast64.go
(12.92 KB)
📄
map_faststr.go
(14.32 KB)
📄
map_test.go
(31.75 KB)
📄
mbarrier.go
(13.78 KB)
📄
mbitmap.go
(22.54 KB)
📄
mbitmap_allocheaders.go
(44.67 KB)
📄
mbitmap_noallocheaders.go
(28.95 KB)
📄
mcache.go
(10 KB)
📄
mcentral.go
(8.05 KB)
📄
mcheckmark.go
(2.85 KB)
📄
mem.go
(6.72 KB)
📄
mem_aix.go
(2.01 KB)
📄
mem_bsd.go
(2.21 KB)
📄
mem_darwin.go
(1.96 KB)
📄
mem_js.go
(457 B)
📄
mem_linux.go
(4.98 KB)
📄
mem_plan9.go
(447 B)
📄
mem_sbrk.go
(4.19 KB)
📄
mem_wasip1.go
(392 B)
📄
mem_wasm.go
(488 B)
📄
mem_windows.go
(3.88 KB)
📄
memclr_386.s
(2.38 KB)
📄
memclr_amd64.s
(4.91 KB)
📄
memclr_arm.s
(2.6 KB)
📄
memclr_arm64.s
(3.62 KB)
📄
memclr_loong64.s
(843 B)
📄
memclr_mips64x.s
(1.72 KB)
📄
memclr_mipsx.s
(1.32 KB)
📄
memclr_plan9_386.s
(983 B)
📄
memclr_plan9_amd64.s
(511 B)
📄
memclr_ppc64x.s
(4.44 KB)
📄
memclr_riscv64.s
(1.71 KB)
📄
memclr_s390x.s
(1.96 KB)
📄
memclr_wasm.s
(485 B)
📄
memmove_386.s
(4.42 KB)
📄
memmove_amd64.s
(12.48 KB)
📄
memmove_arm.s
(5.9 KB)
📄
memmove_arm64.s
(5.96 KB)
📄
memmove_linux_amd64_test.go
(1.53 KB)
📄
memmove_loong64.s
(1.87 KB)
📄
memmove_mips64x.s
(1.83 KB)
📄
memmove_mipsx.s
(4.4 KB)
📄
memmove_plan9_386.s
(3.06 KB)
📄
memmove_plan9_amd64.s
(3.04 KB)
📄
memmove_ppc64x.s
(4.91 KB)
📄
memmove_riscv64.s
(5.46 KB)
📄
memmove_s390x.s
(2.92 KB)
📄
memmove_test.go
(21.23 KB)
📄
memmove_wasm.s
(479 B)
📁
metrics
📄
metrics.go
(26.01 KB)
📄
metrics_test.go
(42.46 KB)
📄
mfinal.go
(18.91 KB)
📄
mfinal_test.go
(5.57 KB)
📄
mfixalloc.go
(3.13 KB)
📄
mgc.go
(59.29 KB)
📄
mgclimit.go
(17.28 KB)
📄
mgclimit_test.go
(9.02 KB)
📄
mgcmark.go
(53.07 KB)
📄
mgcpacer.go
(55.36 KB)
📄
mgcpacer_test.go
(39.26 KB)
📄
mgcscavenge.go
(52.32 KB)
📄
mgcscavenge_test.go
(25.2 KB)
📄
mgcstack.go
(10.58 KB)
📄
mgcsweep.go
(32.26 KB)
📄
mgcwork.go
(12.89 KB)
📄
mheap.go
(72.64 KB)
📄
minmax.go
(1.46 KB)
📄
minmax_test.go
(3.31 KB)
📄
mkduff.go
(8.04 KB)
📄
mkfastlog2table.go
(3.08 KB)
📄
mklockrank.go
(9 KB)
📄
mkpreempt.go
(15.33 KB)
📄
mksizeclasses.go
(9.52 KB)
📄
mmap.go
(844 B)
📄
mpagealloc.go
(39.23 KB)
📄
mpagealloc_32bit.go
(4.56 KB)
📄
mpagealloc_64bit.go
(9.34 KB)
📄
mpagealloc_test.go
(32.59 KB)
📄
mpagecache.go
(5.59 KB)
📄
mpagecache_test.go
(10.79 KB)
📄
mpallocbits.go
(12.58 KB)
📄
mpallocbits_test.go
(13.69 KB)
📄
mprof.go
(47.4 KB)
📄
mranges.go
(14.46 KB)
📄
mranges_test.go
(5.68 KB)
📁
msan
📄
msan.go
(1.5 KB)
📄
msan0.go
(725 B)
📄
msan_amd64.s
(2.3 KB)
📄
msan_arm64.s
(1.98 KB)
📄
msan_loong64.s
(1.96 KB)
📄
msize_allocheaders.go
(1.32 KB)
📄
msize_noallocheaders.go
(915 B)
📄
mspanset.go
(13.12 KB)
📄
mstats.go
(33.81 KB)
📄
mwbbuf.go
(8.13 KB)
📄
nbpipe_pipe.go
(405 B)
📄
nbpipe_pipe2.go
(344 B)
📄
nbpipe_pipe_test.go
(706 B)
📄
nbpipe_test.go
(1.99 KB)
📄
net_plan9.go
(645 B)
📄
netpoll.go
(20.55 KB)
📄
netpoll_aix.go
(5.06 KB)
📄
netpoll_epoll.go
(4.4 KB)
📄
netpoll_fake.go
(664 B)
📄
netpoll_kqueue.go
(5.62 KB)
📄
netpoll_os_test.go
(520 B)
📄
netpoll_solaris.go
(11.2 KB)
📄
netpoll_stub.go
(1.48 KB)
📄
netpoll_wasip1.go
(6.08 KB)
📄
netpoll_windows.go
(4.01 KB)
📄
nonwindows_stub.go
(729 B)
📄
norace_linux_test.go
(915 B)
📄
norace_test.go
(983 B)
📄
numcpu_freebsd_test.go
(381 B)
📄
os2_aix.go
(20.88 KB)
📄
os2_freebsd.go
(302 B)
📄
os2_openbsd.go
(296 B)
📄
os2_plan9.go
(1.48 KB)
📄
os2_solaris.go
(320 B)
📄
os3_plan9.go
(3.94 KB)
📄
os3_solaris.go
(17.59 KB)
📄
os_aix.go
(8.89 KB)
📄
os_android.go
(463 B)
📄
os_darwin.go
(11.92 KB)
📄
os_darwin_arm64.go
(329 B)
📄
os_dragonfly.go
(7.14 KB)
📄
os_freebsd.go
(11.64 KB)
📄
os_freebsd2.go
(603 B)
📄
os_freebsd_amd64.go
(658 B)
📄
os_freebsd_arm.go
(1.45 KB)
📄
os_freebsd_arm64.go
(320 B)
📄
os_freebsd_noauxv.go
(241 B)
📄
os_freebsd_riscv64.go
(198 B)
📄
os_illumos.go
(3.93 KB)
📄
os_js.go
(767 B)
📄
os_linux.go
(25.71 KB)
📄
os_linux_arm.go
(1.51 KB)
📄
os_linux_arm64.go
(478 B)
📄
os_linux_be64.go
(806 B)
📄
os_linux_generic.go
(870 B)
📄
os_linux_loong64.go
(263 B)
📄
os_linux_mips64x.go
(996 B)
📄
os_linux_mipsx.go
(987 B)
📄
os_linux_noauxv.go
(337 B)
📄
os_linux_novdso.go
(347 B)
📄
os_linux_ppc64x.go
(526 B)
📄
os_linux_riscv64.go
(198 B)
📄
os_linux_s390x.go
(825 B)
📄
os_linux_x86.go
(234 B)
📄
os_netbsd.go
(10.12 KB)
📄
os_netbsd_386.go
(617 B)
📄
os_netbsd_amd64.go
(614 B)
📄
os_netbsd_arm.go
(1.07 KB)
📄
os_netbsd_arm64.go
(769 B)
📄
os_nonopenbsd.go
(437 B)
📄
os_only_solaris.go
(357 B)
📄
os_openbsd.go
(6.23 KB)
📄
os_openbsd_arm.go
(662 B)
📄
os_openbsd_arm64.go
(329 B)
📄
os_openbsd_libc.go
(1.49 KB)
📄
os_openbsd_mips64.go
(329 B)
📄
os_openbsd_syscall.go
(1.36 KB)
📄
os_openbsd_syscall1.go
(441 B)
📄
os_openbsd_syscall2.go
(2.51 KB)
📄
os_plan9.go
(10.18 KB)
📄
os_plan9_arm.go
(375 B)
📄
os_solaris.go
(6.62 KB)
📄
os_unix.go
(436 B)
📄
os_unix_nonlinux.go
(374 B)
📄
os_wasip1.go
(7 KB)
📄
os_wasm.go
(3.15 KB)
📄
os_windows.go
(41.39 KB)
📄
os_windows_arm.go
(511 B)
📄
os_windows_arm64.go
(339 B)
📄
pagetrace_off.go
(550 B)
📄
pagetrace_on.go
(10.36 KB)
📄
panic.go
(41.85 KB)
📄
panic32.go
(4.8 KB)
📄
panic_test.go
(1.71 KB)
📄
panicnil_test.go
(1.25 KB)
📄
pinner.go
(10.98 KB)
📄
pinner_test.go
(11.04 KB)
📄
plugin.go
(4.37 KB)
📁
pprof
📄
preempt.go
(15.03 KB)
📄
preempt_386.s
(824 B)
📄
preempt_amd64.s
(1.67 KB)
📄
preempt_arm.s
(1.49 KB)
📄
preempt_arm64.s
(1.97 KB)
📄
preempt_loong64.s
(2.41 KB)
📄
preempt_mips64x.s
(2.72 KB)
📄
preempt_mipsx.s
(2.68 KB)
📄
preempt_nonwindows.go
(290 B)
📄
preempt_ppc64x.s
(2.72 KB)
📄
preempt_riscv64.s
(2.26 KB)
📄
preempt_s390x.s
(1.01 KB)
📄
preempt_wasm.s
(176 B)
📄
print.go
(5.92 KB)
📄
proc.go
(204.28 KB)
📄
proc_runtime_test.go
(1.38 KB)
📄
proc_test.go
(25.85 KB)
📄
profbuf.go
(18.26 KB)
📄
profbuf_test.go
(8.54 KB)
📄
proflabel.go
(1.52 KB)
📁
race
📄
race.go
(18.81 KB)
📄
race0.go
(2.79 KB)
📄
race_amd64.s
(13.94 KB)
📄
race_arm64.s
(14.21 KB)
📄
race_ppc64le.s
(15.93 KB)
📄
race_s390x.s
(11.99 KB)
📄
rand.go
(7.11 KB)
📄
rand_test.go
(1.95 KB)
📄
rdebug.go
(550 B)
📄
retry.go
(760 B)
📄
rt0_aix_ppc64.s
(4.09 KB)
📄
rt0_android_386.s
(822 B)
📄
rt0_android_amd64.s
(754 B)
📄
rt0_android_arm.s
(843 B)
📄
rt0_android_arm64.s
(941 B)
📄
rt0_darwin_amd64.s
(399 B)
📄
rt0_darwin_arm64.s
(1.69 KB)
📄
rt0_dragonfly_amd64.s
(448 B)
📄
rt0_freebsd_386.s
(454 B)
📄
rt0_freebsd_amd64.s
(442 B)
📄
rt0_freebsd_arm.s
(298 B)
📄
rt0_freebsd_arm64.s
(1.88 KB)
📄
rt0_freebsd_riscv64.s
(2.72 KB)
📄
rt0_illumos_amd64.s
(311 B)
📄
rt0_ios_amd64.s
(425 B)
📄
rt0_ios_arm64.s
(425 B)
📄
rt0_js_wasm.s
(1.53 KB)
📄
rt0_linux_386.s
(450 B)
📄
rt0_linux_amd64.s
(307 B)
📄
rt0_linux_arm.s
(1007 B)
📄
rt0_linux_arm64.s
(1.81 KB)
📄
rt0_linux_loong64.s
(2.01 KB)
📄
rt0_linux_mips64x.s
(1014 B)
📄
rt0_linux_mipsx.s
(797 B)
📄
rt0_linux_ppc64.s
(847 B)
📄
rt0_linux_ppc64le.s
(2.89 KB)
📄
rt0_linux_riscv64.s
(2.65 KB)
📄
rt0_linux_s390x.s
(676 B)
📄
rt0_netbsd_386.s
(452 B)
📄
rt0_netbsd_amd64.s
(309 B)
📄
rt0_netbsd_arm.s
(296 B)
📄
rt0_netbsd_arm64.s
(1.8 KB)
📄
rt0_openbsd_386.s
(454 B)
📄
rt0_openbsd_amd64.s
(311 B)
📄
rt0_openbsd_arm.s
(298 B)
📄
rt0_openbsd_arm64.s
(1.96 KB)
📄
rt0_openbsd_mips64.s
(976 B)
📄
rt0_openbsd_ppc64.s
(370 B)
📄
rt0_openbsd_riscv64.s
(372 B)
📄
rt0_plan9_386.s
(523 B)
📄
rt0_plan9_amd64.s
(481 B)
📄
rt0_plan9_arm.s
(397 B)
📄
rt0_solaris_amd64.s
(311 B)
📄
rt0_wasip1_wasm.s
(387 B)
📄
rt0_windows_386.s
(1.28 KB)
📄
rt0_windows_amd64.s
(1.14 KB)
📄
rt0_windows_arm.s
(386 B)
📄
rt0_windows_arm64.s
(733 B)
📄
runtime-gdb.py
(15.44 KB)
📄
runtime-gdb_test.go
(24.04 KB)
📄
runtime-gdb_unix_test.go
(9.4 KB)
📄
runtime-lldb_test.go
(4.95 KB)
📄
runtime-seh_windows_test.go
(4.7 KB)
📄
runtime.go
(7.1 KB)
📄
runtime1.go
(16.66 KB)
📄
runtime2.go
(46.16 KB)
📄
runtime_boring.go
(606 B)
📄
runtime_linux_test.go
(1.77 KB)
📄
runtime_mmap_test.go
(1.77 KB)
📄
runtime_test.go
(11.81 KB)
📄
runtime_unix_test.go
(1.22 KB)
📄
rwmutex.go
(5 KB)
📄
rwmutex_test.go
(4.21 KB)
📄
security_aix.go
(449 B)
📄
security_issetugid.go
(502 B)
📄
security_linux.go
(335 B)
📄
security_nonunix.go
(256 B)
📄
security_test.go
(4.09 KB)
📄
security_unix.go
(818 B)
📄
select.go
(14.82 KB)
📄
sema.go
(18.43 KB)
📄
sema_test.go
(4.21 KB)
📄
semasleep_test.go
(3.46 KB)
📄
sigaction.go
(489 B)
📄
signal_386.go
(1.72 KB)
📄
signal_aix_ppc64.go
(3.54 KB)
📄
signal_amd64.go
(2.73 KB)
📄
signal_arm.go
(2.54 KB)
📄
signal_arm64.go
(3.83 KB)
📄
signal_darwin.go
(2.13 KB)
📄
signal_darwin_amd64.go
(4 KB)
📄
signal_darwin_arm64.go
(3.6 KB)
📄
signal_dragonfly.go
(2.17 KB)
📄
signal_dragonfly_amd64.go
(2.01 KB)
📄
signal_freebsd.go
(2.2 KB)
📄
signal_freebsd_386.go
(1.55 KB)
📄
signal_freebsd_amd64.go
(2.03 KB)
📄
signal_freebsd_arm.go
(2.18 KB)
📄
signal_freebsd_arm64.go
(3.24 KB)
📄
signal_freebsd_riscv64.go
(3.08 KB)
📄
signal_linux_386.go
(1.59 KB)
📄
signal_linux_amd64.go
(2.05 KB)
📄
signal_linux_arm.go
(2.12 KB)
📄
signal_linux_arm64.go
(2.95 KB)
📄
signal_linux_loong64.go
(3.22 KB)
📄
signal_linux_mips64x.go
(3.35 KB)
📄
signal_linux_mipsx.go
(3.67 KB)
📄
signal_linux_ppc64x.go
(3.5 KB)
📄
signal_linux_riscv64.go
(2.92 KB)
📄
signal_linux_s390x.go
(4.49 KB)
📄
signal_loong64.go
(3.01 KB)
📄
signal_mips64x.go
(3.18 KB)
📄
signal_mipsx.go
(3.06 KB)
📄
signal_netbsd.go
(2.18 KB)
📄
signal_netbsd_386.go
(1.76 KB)
📄
signal_netbsd_amd64.go
(2.33 KB)
📄
signal_netbsd_arm.go
(2.3 KB)
📄
signal_netbsd_arm64.go
(3.4 KB)
📄
signal_openbsd.go
(2.18 KB)
📄
signal_openbsd_386.go
(1.58 KB)
📄
signal_openbsd_amd64.go
(2.04 KB)
📄
signal_openbsd_arm.go
(2.12 KB)
📄
signal_openbsd_arm64.go
(3.39 KB)
📄
signal_openbsd_mips64.go
(3.28 KB)
📄
signal_openbsd_ppc64.go
(3.53 KB)
📄
signal_openbsd_riscv64.go
(3.12 KB)
📄
signal_plan9.go
(1.93 KB)
📄
signal_ppc64x.go
(3.71 KB)
📄
signal_riscv64.go
(2.91 KB)
📄
signal_solaris.go
(4.5 KB)
📄
signal_solaris_amd64.go
(2.47 KB)
📄
signal_unix.go
(44.11 KB)
📄
signal_windows.go
(14.48 KB)
📄
signal_windows_test.go
(8.98 KB)
📄
sigqueue.go
(7.62 KB)
📄
sigqueue_note.go
(648 B)
📄
sigqueue_plan9.go
(3.25 KB)
📄
sigtab_aix.go
(11.3 KB)
📄
sigtab_linux_generic.go
(3.52 KB)
📄
sigtab_linux_mipsx.go
(5.95 KB)
📄
sizeclasses.go
(9.17 KB)
📄
sizeof_test.go
(1003 B)
📄
slice.go
(11.23 KB)
📄
slice_test.go
(10.32 KB)
📄
softfloat64.go
(11.54 KB)
📄
softfloat64_test.go
(4.04 KB)
📄
stack.go
(40.02 KB)
📄
stack_test.go
(23.14 KB)
📄
start_line_amd64_test.go
(647 B)
📄
start_line_test.go
(2.58 KB)
📄
stkframe.go
(9.88 KB)
📄
string.go
(13.35 KB)
📄
string_test.go
(13.33 KB)
📄
stubs.go
(17.39 KB)
📄
stubs2.go
(1.15 KB)
📄
stubs3.go
(324 B)
📄
stubs_386.go
(708 B)
📄
stubs_amd64.go
(1.38 KB)
📄
stubs_arm.go
(689 B)
📄
stubs_arm64.go
(700 B)
📄
stubs_linux.go
(650 B)
📄
stubs_loong64.go
(638 B)
📄
stubs_mips64x.go
(522 B)
📄
stubs_mipsx.go
(441 B)
📄
stubs_nonlinux.go
(298 B)
📄
stubs_ppc64.go
(302 B)
📄
stubs_ppc64x.go
(688 B)
📄
stubs_riscv64.go
(695 B)
📄
stubs_s390x.go
(414 B)
📄
symtab.go
(34.76 KB)
📄
symtab_test.go
(7.48 KB)
📄
symtabinl.go
(3.81 KB)
📄
symtabinl_test.go
(2.97 KB)
📄
sys_aix_ppc64.s
(7.42 KB)
📄
sys_arm.go
(521 B)
📄
sys_arm64.go
(469 B)
📄
sys_darwin.go
(22.56 KB)
📄
sys_darwin_amd64.s
(19.65 KB)
📄
sys_darwin_arm64.go
(1.74 KB)
📄
sys_darwin_arm64.s
(18.28 KB)
📄
sys_dragonfly_amd64.s
(8.31 KB)
📄
sys_freebsd_386.s
(9.41 KB)
📄
sys_freebsd_amd64.s
(12.67 KB)
📄
sys_freebsd_arm.s
(10.38 KB)
📄
sys_freebsd_arm64.s
(9.49 KB)
📄
sys_freebsd_riscv64.s
(8.92 KB)
📄
sys_libc.go
(1.84 KB)
📄
sys_linux_386.s
(17.89 KB)
📄
sys_linux_amd64.s
(15.74 KB)
📄
sys_linux_arm.s
(13.5 KB)
📄
sys_linux_arm64.s
(16.71 KB)
📄
sys_linux_loong64.s
(14.16 KB)
📄
sys_linux_mips64x.s
(11.96 KB)
📄
sys_linux_mipsx.s
(9.69 KB)
📄
sys_linux_ppc64x.s
(18.09 KB)
📄
sys_linux_riscv64.s
(11.48 KB)
📄
sys_linux_s390x.s
(12.49 KB)
📄
sys_loong64.go
(489 B)
📄
sys_mips64x.go
(500 B)
📄
sys_mipsx.go
(496 B)
📄
sys_netbsd_386.s
(9.61 KB)
📄
sys_netbsd_amd64.s
(9.78 KB)
📄
sys_netbsd_arm.s
(10.58 KB)
📄
sys_netbsd_arm64.s
(9.47 KB)
📄
sys_nonppc64x.go
(245 B)
📄
sys_openbsd.go
(2.59 KB)
📄
sys_openbsd1.go
(1.23 KB)
📄
sys_openbsd2.go
(8.67 KB)
📄
sys_openbsd3.go
(3.37 KB)
📄
sys_openbsd_386.s
(20.4 KB)
📄
sys_openbsd_amd64.s
(15.54 KB)
📄
sys_openbsd_arm.s
(18.46 KB)
📄
sys_openbsd_arm64.s
(15.05 KB)
📄
sys_openbsd_mips64.s
(8.81 KB)
📄
sys_openbsd_ppc64.s
(15.3 KB)
📄
sys_openbsd_riscv64.s
(16.8 KB)
📄
sys_plan9_386.s
(4.48 KB)
📄
sys_plan9_amd64.s
(4.56 KB)
📄
sys_plan9_arm.s
(7.03 KB)
📄
sys_ppc64x.go
(532 B)
📄
sys_riscv64.go
(469 B)
📄
sys_s390x.go
(469 B)
📄
sys_solaris_amd64.s
(6.42 KB)
📄
sys_wasm.go
(758 B)
📄
sys_wasm.s
(1.43 KB)
📄
sys_windows_386.s
(6.46 KB)
📄
sys_windows_amd64.s
(8.41 KB)
📄
sys_windows_arm.s
(7.72 KB)
📄
sys_windows_arm64.s
(6.8 KB)
📄
sys_x86.go
(552 B)
📄
syscall2_solaris.go
(1.85 KB)
📄
syscall_aix.go
(6.33 KB)
📄
syscall_solaris.go
(8.38 KB)
📄
syscall_unix_test.go
(635 B)
📄
syscall_windows.go
(16.57 KB)
📄
syscall_windows_test.go
(32.45 KB)
📄
tagptr.go
(496 B)
📄
tagptr_32bit.go
(927 B)
📄
tagptr_64bit.go
(3.23 KB)
📄
test_amd64.go
(196 B)
📄
test_amd64.s
(316 B)
📄
test_stubs.go
(218 B)
📁
testdata
📄
textflag.h
(1.47 KB)
📄
time.go
(30.57 KB)
📄
time_fake.go
(2.5 KB)
📄
time_linux_amd64.s
(2.02 KB)
📄
time_nofake.go
(709 B)
📄
time_test.go
(2.04 KB)
📄
time_windows.h
(753 B)
📄
time_windows_386.s
(1.71 KB)
📄
time_windows_amd64.s
(786 B)
📄
time_windows_arm.s
(1.97 KB)
📄
time_windows_arm64.s
(906 B)
📄
timeasm.go
(418 B)
📄
timestub.go
(532 B)
📄
timestub2.go
(363 B)
📄
tls_arm.s
(3.45 KB)
📄
tls_arm64.h
(1.1 KB)
📄
tls_arm64.s
(1.2 KB)
📄
tls_loong64.s
(589 B)
📄
tls_mips64x.s
(733 B)
📄
tls_mipsx.s
(710 B)
📄
tls_ppc64x.s
(1.52 KB)
📄
tls_riscv64.s
(615 B)
📄
tls_s390x.s
(1.55 KB)
📄
tls_stub.go
(260 B)
📄
tls_windows_amd64.go
(294 B)
📁
trace
📄
trace.go
(62.76 KB)
📄
trace2.go
(33.81 KB)
📄
trace2buf.go
(6.58 KB)
📄
trace2cpu.go
(8.87 KB)
📄
trace2event.go
(8.39 KB)
📄
trace2map.go
(4.36 KB)
📄
trace2region.go
(1.68 KB)
📄
trace2runtime.go
(24.2 KB)
📄
trace2stack.go
(9.45 KB)
📄
trace2status.go
(7.24 KB)
📄
trace2string.go
(2.59 KB)
📄
trace2time.go
(3.21 KB)
📄
trace_cgo_test.go
(4.47 KB)
📄
traceback.go
(54.61 KB)
📄
traceback_test.go
(22.95 KB)
📄
tracebackx_test.go
(509 B)
📄
type.go
(12.1 KB)
📄
typekind.go
(743 B)
📄
unsafe.go
(2.97 KB)
📄
unsafepoint_test.go
(3.27 KB)
📄
utf8.go
(3.39 KB)
📄
vdso_elf32.go
(2.76 KB)
📄
vdso_elf64.go
(2.84 KB)
📄
vdso_freebsd.go
(2.44 KB)
📄
vdso_freebsd_arm.go
(454 B)
📄
vdso_freebsd_arm64.go
(454 B)
📄
vdso_freebsd_riscv64.go
(429 B)
📄
vdso_freebsd_x86.go
(1.86 KB)
📄
vdso_in_none.go
(443 B)
📄
vdso_linux.go
(7.77 KB)
📄
vdso_linux_386.go
(669 B)
📄
vdso_linux_amd64.go
(685 B)
📄
vdso_linux_arm.go
(669 B)
📄
vdso_linux_arm64.go
(670 B)
📄
vdso_linux_loong64.go
(793 B)
📄
vdso_linux_mips64x.go
(850 B)
📄
vdso_linux_ppc64x.go
(672 B)
📄
vdso_linux_riscv64.go
(666 B)
📄
vdso_linux_s390x.go
(659 B)
📄
vdso_test.go
(3.71 KB)
📄
vlop_386.s
(2.02 KB)
📄
vlop_arm.s
(7.06 KB)
📄
vlop_arm_test.go
(3.75 KB)
📄
vlrt.go
(6.71 KB)
📄
wincallback.go
(3.45 KB)
📄
write_err.go
(291 B)
📄
write_err_android.go
(4.65 KB)
📄
zcallback_windows.go
(155 B)
📄
zcallback_windows.s
(63.06 KB)
📄
zcallback_windows_arm.s
(89.32 KB)
📄
zcallback_windows_arm64.s
(89.32 KB)
Editing: mgc.go
// Copyright 2009 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. // Garbage collector (GC). // // The GC runs concurrently with mutator threads, is type accurate (aka precise), allows multiple // GC thread to run in parallel. It is a concurrent mark and sweep that uses a write barrier. It is // non-generational and non-compacting. Allocation is done using size segregated per P allocation // areas to minimize fragmentation while eliminating locks in the common case. // // The algorithm decomposes into several steps. // This is a high level description of the algorithm being used. For an overview of GC a good // place to start is Richard Jones' gchandbook.org. // // The algorithm's intellectual heritage includes Dijkstra's on-the-fly algorithm, see // Edsger W. Dijkstra, Leslie Lamport, A. J. Martin, C. S. Scholten, and E. F. M. Steffens. 1978. // On-the-fly garbage collection: an exercise in cooperation. Commun. ACM 21, 11 (November 1978), // 966-975. // For journal quality proofs that these steps are complete, correct, and terminate see // Hudson, R., and Moss, J.E.B. Copying Garbage Collection without stopping the world. // Concurrency and Computation: Practice and Experience 15(3-5), 2003. // // 1. GC performs sweep termination. // // a. Stop the world. This causes all Ps to reach a GC safe-point. // // b. Sweep any unswept spans. There will only be unswept spans if // this GC cycle was forced before the expected time. // // 2. GC performs the mark phase. // // a. Prepare for the mark phase by setting gcphase to _GCmark // (from _GCoff), enabling the write barrier, enabling mutator // assists, and enqueueing root mark jobs. No objects may be // scanned until all Ps have enabled the write barrier, which is // accomplished using STW. // // b. Start the world. From this point, GC work is done by mark // workers started by the scheduler and by assists performed as // part of allocation. The write barrier shades both the // overwritten pointer and the new pointer value for any pointer // writes (see mbarrier.go for details). Newly allocated objects // are immediately marked black. // // c. GC performs root marking jobs. This includes scanning all // stacks, shading all globals, and shading any heap pointers in // off-heap runtime data structures. Scanning a stack stops a // goroutine, shades any pointers found on its stack, and then // resumes the goroutine. // // d. GC drains the work queue of grey objects, scanning each grey // object to black and shading all pointers found in the object // (which in turn may add those pointers to the work queue). // // e. Because GC work is spread across local caches, GC uses a // distributed termination algorithm to detect when there are no // more root marking jobs or grey objects (see gcMarkDone). At this // point, GC transitions to mark termination. // // 3. GC performs mark termination. // // a. Stop the world. // // b. Set gcphase to _GCmarktermination, and disable workers and // assists. // // c. Perform housekeeping like flushing mcaches. // // 4. GC performs the sweep phase. // // a. Prepare for the sweep phase by setting gcphase to _GCoff, // setting up sweep state and disabling the write barrier. // // b. Start the world. From this point on, newly allocated objects // are white, and allocating sweeps spans before use if necessary. // // c. GC does concurrent sweeping in the background and in response // to allocation. See description below. // // 5. When sufficient allocation has taken place, replay the sequence // starting with 1 above. See discussion of GC rate below. // Concurrent sweep. // // The sweep phase proceeds concurrently with normal program execution. // The heap is swept span-by-span both lazily (when a goroutine needs another span) // and concurrently in a background goroutine (this helps programs that are not CPU bound). // At the end of STW mark termination all spans are marked as "needs sweeping". // // The background sweeper goroutine simply sweeps spans one-by-one. // // To avoid requesting more OS memory while there are unswept spans, when a // goroutine needs another span, it first attempts to reclaim that much memory // by sweeping. When a goroutine needs to allocate a new small-object span, it // sweeps small-object spans for the same object size until it frees at least // one object. When a goroutine needs to allocate large-object span from heap, // it sweeps spans until it frees at least that many pages into heap. There is // one case where this may not suffice: if a goroutine sweeps and frees two // nonadjacent one-page spans to the heap, it will allocate a new two-page // span, but there can still be other one-page unswept spans which could be // combined into a two-page span. // // It's critical to ensure that no operations proceed on unswept spans (that would corrupt // mark bits in GC bitmap). During GC all mcaches are flushed into the central cache, // so they are empty. When a goroutine grabs a new span into mcache, it sweeps it. // When a goroutine explicitly frees an object or sets a finalizer, it ensures that // the span is swept (either by sweeping it, or by waiting for the concurrent sweep to finish). // The finalizer goroutine is kicked off only when all spans are swept. // When the next GC starts, it sweeps all not-yet-swept spans (if any). // GC rate. // Next GC is after we've allocated an extra amount of memory proportional to // the amount already in use. The proportion is controlled by GOGC environment variable // (100 by default). If GOGC=100 and we're using 4M, we'll GC again when we get to 8M // (this mark is computed by the gcController.heapGoal method). This keeps the GC cost in // linear proportion to the allocation cost. Adjusting GOGC just changes the linear constant // (and also the amount of extra memory used). // Oblets // // In order to prevent long pauses while scanning large objects and to // improve parallelism, the garbage collector breaks up scan jobs for // objects larger than maxObletBytes into "oblets" of at most // maxObletBytes. When scanning encounters the beginning of a large // object, it scans only the first oblet and enqueues the remaining // oblets as new scan jobs. package runtime import ( "internal/cpu" "runtime/internal/atomic" "unsafe" ) const ( _DebugGC = 0 _FinBlockSize = 4 * 1024 // concurrentSweep is a debug flag. Disabling this flag // ensures all spans are swept while the world is stopped. concurrentSweep = true // debugScanConservative enables debug logging for stack // frames that are scanned conservatively. debugScanConservative = false // sweepMinHeapDistance is a lower bound on the heap distance // (in bytes) reserved for concurrent sweeping between GC // cycles. sweepMinHeapDistance = 1024 * 1024 ) // heapObjectsCanMove always returns false in the current garbage collector. // It exists for go4.org/unsafe/assume-no-moving-gc, which is an // unfortunate idea that had an even more unfortunate implementation. // Every time a new Go release happened, the package stopped building, // and the authors had to add a new file with a new //go:build line, and // then the entire ecosystem of packages with that as a dependency had to // explicitly update to the new version. Many packages depend on // assume-no-moving-gc transitively, through paths like // inet.af/netaddr -> go4.org/intern -> assume-no-moving-gc. // This was causing a significant amount of friction around each new // release, so we added this bool for the package to //go:linkname // instead. The bool is still unfortunate, but it's not as bad as // breaking the ecosystem on every new release. // // If the Go garbage collector ever does move heap objects, we can set // this to true to break all the programs using assume-no-moving-gc. // //go:linkname heapObjectsCanMove func heapObjectsCanMove() bool { return false } func gcinit() { if unsafe.Sizeof(workbuf{}) != _WorkbufSize { throw("size of Workbuf is suboptimal") } // No sweep on the first cycle. sweep.active.state.Store(sweepDrainedMask) // Initialize GC pacer state. // Use the environment variable GOGC for the initial gcPercent value. // Use the environment variable GOMEMLIMIT for the initial memoryLimit value. gcController.init(readGOGC(), readGOMEMLIMIT()) work.startSema = 1 work.markDoneSema = 1 lockInit(&work.sweepWaiters.lock, lockRankSweepWaiters) lockInit(&work.assistQueue.lock, lockRankAssistQueue) lockInit(&work.wbufSpans.lock, lockRankWbufSpans) } // gcenable is called after the bulk of the runtime initialization, // just before we're about to start letting user code run. // It kicks off the background sweeper goroutine, the background // scavenger goroutine, and enables GC. func gcenable() { // Kick off sweeping and scavenging. c := make(chan int, 2) go bgsweep(c) go bgscavenge(c) <-c <-c memstats.enablegc = true // now that runtime is initialized, GC is okay } // Garbage collector phase. // Indicates to write barrier and synchronization task to perform. var gcphase uint32 // The compiler knows about this variable. // If you change it, you must change builtin/runtime.go, too. // If you change the first four bytes, you must also change the write // barrier insertion code. var writeBarrier struct { enabled bool // compiler emits a check of this before calling write barrier pad [3]byte // compiler uses 32-bit load for "enabled" field alignme uint64 // guarantee alignment so that compiler can use a 32 or 64-bit load } // gcBlackenEnabled is 1 if mutator assists and background mark // workers are allowed to blacken objects. This must only be set when // gcphase == _GCmark. var gcBlackenEnabled uint32 const ( _GCoff = iota // GC not running; sweeping in background, write barrier disabled _GCmark // GC marking roots and workbufs: allocate black, write barrier ENABLED _GCmarktermination // GC mark termination: allocate black, P's help GC, write barrier ENABLED ) //go:nosplit func setGCPhase(x uint32) { atomic.Store(&gcphase, x) writeBarrier.enabled = gcphase == _GCmark || gcphase == _GCmarktermination } // gcMarkWorkerMode represents the mode that a concurrent mark worker // should operate in. // // Concurrent marking happens through four different mechanisms. One // is mutator assists, which happen in response to allocations and are // not scheduled. The other three are variations in the per-P mark // workers and are distinguished by gcMarkWorkerMode. type gcMarkWorkerMode int const ( // gcMarkWorkerNotWorker indicates that the next scheduled G is not // starting work and the mode should be ignored. gcMarkWorkerNotWorker gcMarkWorkerMode = iota // gcMarkWorkerDedicatedMode indicates that the P of a mark // worker is dedicated to running that mark worker. The mark // worker should run without preemption. gcMarkWorkerDedicatedMode // gcMarkWorkerFractionalMode indicates that a P is currently // running the "fractional" mark worker. The fractional worker // is necessary when GOMAXPROCS*gcBackgroundUtilization is not // an integer and using only dedicated workers would result in // utilization too far from the target of gcBackgroundUtilization. // The fractional worker should run until it is preempted and // will be scheduled to pick up the fractional part of // GOMAXPROCS*gcBackgroundUtilization. gcMarkWorkerFractionalMode // gcMarkWorkerIdleMode indicates that a P is running the mark // worker because it has nothing else to do. The idle worker // should run until it is preempted and account its time // against gcController.idleMarkTime. gcMarkWorkerIdleMode ) // gcMarkWorkerModeStrings are the strings labels of gcMarkWorkerModes // to use in execution traces. var gcMarkWorkerModeStrings = [...]string{ "Not worker", "GC (dedicated)", "GC (fractional)", "GC (idle)", } // pollFractionalWorkerExit reports whether a fractional mark worker // should self-preempt. It assumes it is called from the fractional // worker. func pollFractionalWorkerExit() bool { // This should be kept in sync with the fractional worker // scheduler logic in findRunnableGCWorker. now := nanotime() delta := now - gcController.markStartTime if delta <= 0 { return true } p := getg().m.p.ptr() selfTime := p.gcFractionalMarkTime + (now - p.gcMarkWorkerStartTime) // Add some slack to the utilization goal so that the // fractional worker isn't behind again the instant it exits. return float64(selfTime)/float64(delta) > 1.2*gcController.fractionalUtilizationGoal } var work workType type workType struct { full lfstack // lock-free list of full blocks workbuf _ cpu.CacheLinePad // prevents false-sharing between full and empty empty lfstack // lock-free list of empty blocks workbuf _ cpu.CacheLinePad // prevents false-sharing between empty and nproc/nwait wbufSpans struct { lock mutex // free is a list of spans dedicated to workbufs, but // that don't currently contain any workbufs. free mSpanList // busy is a list of all spans containing workbufs on // one of the workbuf lists. busy mSpanList } // Restore 64-bit alignment on 32-bit. _ uint32 // bytesMarked is the number of bytes marked this cycle. This // includes bytes blackened in scanned objects, noscan objects // that go straight to black, and permagrey objects scanned by // markroot during the concurrent scan phase. This is updated // atomically during the cycle. Updates may be batched // arbitrarily, since the value is only read at the end of the // cycle. // // Because of benign races during marking, this number may not // be the exact number of marked bytes, but it should be very // close. // // Put this field here because it needs 64-bit atomic access // (and thus 8-byte alignment even on 32-bit architectures). bytesMarked uint64 markrootNext uint32 // next markroot job markrootJobs uint32 // number of markroot jobs nproc uint32 tstart int64 nwait uint32 // Number of roots of various root types. Set by gcMarkRootPrepare. // // nStackRoots == len(stackRoots), but we have nStackRoots for // consistency. nDataRoots, nBSSRoots, nSpanRoots, nStackRoots int // Base indexes of each root type. Set by gcMarkRootPrepare. baseData, baseBSS, baseSpans, baseStacks, baseEnd uint32 // stackRoots is a snapshot of all of the Gs that existed // before the beginning of concurrent marking. The backing // store of this must not be modified because it might be // shared with allgs. stackRoots []*g // Each type of GC state transition is protected by a lock. // Since multiple threads can simultaneously detect the state // transition condition, any thread that detects a transition // condition must acquire the appropriate transition lock, // re-check the transition condition and return if it no // longer holds or perform the transition if it does. // Likewise, any transition must invalidate the transition // condition before releasing the lock. This ensures that each // transition is performed by exactly one thread and threads // that need the transition to happen block until it has // happened. // // startSema protects the transition from "off" to mark or // mark termination. startSema uint32 // markDoneSema protects transitions from mark to mark termination. markDoneSema uint32 bgMarkReady note // signal background mark worker has started bgMarkDone uint32 // cas to 1 when at a background mark completion point // Background mark completion signaling // mode is the concurrency mode of the current GC cycle. mode gcMode // userForced indicates the current GC cycle was forced by an // explicit user call. userForced bool // initialHeapLive is the value of gcController.heapLive at the // beginning of this GC cycle. initialHeapLive uint64 // assistQueue is a queue of assists that are blocked because // there was neither enough credit to steal or enough work to // do. assistQueue struct { lock mutex q gQueue } // sweepWaiters is a list of blocked goroutines to wake when // we transition from mark termination to sweep. sweepWaiters struct { lock mutex list gList } // cycles is the number of completed GC cycles, where a GC // cycle is sweep termination, mark, mark termination, and // sweep. This differs from memstats.numgc, which is // incremented at mark termination. cycles atomic.Uint32 // Timing/utilization stats for this cycle. stwprocs, maxprocs int32 tSweepTerm, tMark, tMarkTerm, tEnd int64 // nanotime() of phase start pauseNS int64 // total STW time this cycle // debug.gctrace heap sizes for this cycle. heap0, heap1, heap2 uint64 // Cumulative estimated CPU usage. cpuStats } // GC runs a garbage collection and blocks the caller until the // garbage collection is complete. It may also block the entire // program. func GC() { // We consider a cycle to be: sweep termination, mark, mark // termination, and sweep. This function shouldn't return // until a full cycle has been completed, from beginning to // end. Hence, we always want to finish up the current cycle // and start a new one. That means: // // 1. In sweep termination, mark, or mark termination of cycle // N, wait until mark termination N completes and transitions // to sweep N. // // 2. In sweep N, help with sweep N. // // At this point we can begin a full cycle N+1. // // 3. Trigger cycle N+1 by starting sweep termination N+1. // // 4. Wait for mark termination N+1 to complete. // // 5. Help with sweep N+1 until it's done. // // This all has to be written to deal with the fact that the // GC may move ahead on its own. For example, when we block // until mark termination N, we may wake up in cycle N+2. // Wait until the current sweep termination, mark, and mark // termination complete. n := work.cycles.Load() gcWaitOnMark(n) // We're now in sweep N or later. Trigger GC cycle N+1, which // will first finish sweep N if necessary and then enter sweep // termination N+1. gcStart(gcTrigger{kind: gcTriggerCycle, n: n + 1}) // Wait for mark termination N+1 to complete. gcWaitOnMark(n + 1) // Finish sweep N+1 before returning. We do this both to // complete the cycle and because runtime.GC() is often used // as part of tests and benchmarks to get the system into a // relatively stable and isolated state. for work.cycles.Load() == n+1 && sweepone() != ^uintptr(0) { Gosched() } // Callers may assume that the heap profile reflects the // just-completed cycle when this returns (historically this // happened because this was a STW GC), but right now the // profile still reflects mark termination N, not N+1. // // As soon as all of the sweep frees from cycle N+1 are done, // we can go ahead and publish the heap profile. // // First, wait for sweeping to finish. (We know there are no // more spans on the sweep queue, but we may be concurrently // sweeping spans, so we have to wait.) for work.cycles.Load() == n+1 && !isSweepDone() { Gosched() } // Now we're really done with sweeping, so we can publish the // stable heap profile. Only do this if we haven't already hit // another mark termination. mp := acquirem() cycle := work.cycles.Load() if cycle == n+1 || (gcphase == _GCmark && cycle == n+2) { mProf_PostSweep() } releasem(mp) } // gcWaitOnMark blocks until GC finishes the Nth mark phase. If GC has // already completed this mark phase, it returns immediately. func gcWaitOnMark(n uint32) { for { // Disable phase transitions. lock(&work.sweepWaiters.lock) nMarks := work.cycles.Load() if gcphase != _GCmark { // We've already completed this cycle's mark. nMarks++ } if nMarks > n { // We're done. unlock(&work.sweepWaiters.lock) return } // Wait until sweep termination, mark, and mark // termination of cycle N complete. work.sweepWaiters.list.push(getg()) goparkunlock(&work.sweepWaiters.lock, waitReasonWaitForGCCycle, traceBlockUntilGCEnds, 1) } } // gcMode indicates how concurrent a GC cycle should be. type gcMode int const ( gcBackgroundMode gcMode = iota // concurrent GC and sweep gcForceMode // stop-the-world GC now, concurrent sweep gcForceBlockMode // stop-the-world GC now and STW sweep (forced by user) ) // A gcTrigger is a predicate for starting a GC cycle. Specifically, // it is an exit condition for the _GCoff phase. type gcTrigger struct { kind gcTriggerKind now int64 // gcTriggerTime: current time n uint32 // gcTriggerCycle: cycle number to start } type gcTriggerKind int const ( // gcTriggerHeap indicates that a cycle should be started when // the heap size reaches the trigger heap size computed by the // controller. gcTriggerHeap gcTriggerKind = iota // gcTriggerTime indicates that a cycle should be started when // it's been more than forcegcperiod nanoseconds since the // previous GC cycle. gcTriggerTime // gcTriggerCycle indicates that a cycle should be started if // we have not yet started cycle number gcTrigger.n (relative // to work.cycles). gcTriggerCycle ) // test reports whether the trigger condition is satisfied, meaning // that the exit condition for the _GCoff phase has been met. The exit // condition should be tested when allocating. func (t gcTrigger) test() bool { if !memstats.enablegc || panicking.Load() != 0 || gcphase != _GCoff { return false } switch t.kind { case gcTriggerHeap: trigger, _ := gcController.trigger() return gcController.heapLive.Load() >= trigger case gcTriggerTime: if gcController.gcPercent.Load() < 0 { return false } lastgc := int64(atomic.Load64(&memstats.last_gc_nanotime)) return lastgc != 0 && t.now-lastgc > forcegcperiod case gcTriggerCycle: // t.n > work.cycles, but accounting for wraparound. return int32(t.n-work.cycles.Load()) > 0 } return true } // gcStart starts the GC. It transitions from _GCoff to _GCmark (if // debug.gcstoptheworld == 0) or performs all of GC (if // debug.gcstoptheworld != 0). // // This may return without performing this transition in some cases, // such as when called on a system stack or with locks held. func gcStart(trigger gcTrigger) { // Since this is called from malloc and malloc is called in // the guts of a number of libraries that might be holding // locks, don't attempt to start GC in non-preemptible or // potentially unstable situations. mp := acquirem() if gp := getg(); gp == mp.g0 || mp.locks > 1 || mp.preemptoff != "" { releasem(mp) return } releasem(mp) mp = nil // Pick up the remaining unswept/not being swept spans concurrently // // This shouldn't happen if we're being invoked in background // mode since proportional sweep should have just finished // sweeping everything, but rounding errors, etc, may leave a // few spans unswept. In forced mode, this is necessary since // GC can be forced at any point in the sweeping cycle. // // We check the transition condition continuously here in case // this G gets delayed in to the next GC cycle. for trigger.test() && sweepone() != ^uintptr(0) { } // Perform GC initialization and the sweep termination // transition. semacquire(&work.startSema) // Re-check transition condition under transition lock. if !trigger.test() { semrelease(&work.startSema) return } // In gcstoptheworld debug mode, upgrade the mode accordingly. // We do this after re-checking the transition condition so // that multiple goroutines that detect the heap trigger don't // start multiple STW GCs. mode := gcBackgroundMode if debug.gcstoptheworld == 1 { mode = gcForceMode } else if debug.gcstoptheworld == 2 { mode = gcForceBlockMode } // Ok, we're doing it! Stop everybody else semacquire(&gcsema) semacquire(&worldsema) // For stats, check if this GC was forced by the user. // Update it under gcsema to avoid gctrace getting wrong values. work.userForced = trigger.kind == gcTriggerCycle trace := traceAcquire() if trace.ok() { trace.GCStart() traceRelease(trace) } // Check that all Ps have finished deferred mcache flushes. for _, p := range allp { if fg := p.mcache.flushGen.Load(); fg != mheap_.sweepgen { println("runtime: p", p.id, "flushGen", fg, "!= sweepgen", mheap_.sweepgen) throw("p mcache not flushed") } } gcBgMarkStartWorkers() systemstack(gcResetMarkState) work.stwprocs, work.maxprocs = gomaxprocs, gomaxprocs if work.stwprocs > ncpu { // This is used to compute CPU time of the STW phases, // so it can't be more than ncpu, even if GOMAXPROCS is. work.stwprocs = ncpu } work.heap0 = gcController.heapLive.Load() work.pauseNS = 0 work.mode = mode now := nanotime() work.tSweepTerm = now var stw worldStop systemstack(func() { stw = stopTheWorldWithSema(stwGCSweepTerm) }) // Finish sweep before we start concurrent scan. systemstack(func() { finishsweep_m() }) // clearpools before we start the GC. If we wait the memory will not be // reclaimed until the next GC cycle. clearpools() work.cycles.Add(1) // Assists and workers can start the moment we start // the world. gcController.startCycle(now, int(gomaxprocs), trigger) // Notify the CPU limiter that assists may begin. gcCPULimiter.startGCTransition(true, now) // In STW mode, disable scheduling of user Gs. This may also // disable scheduling of this goroutine, so it may block as // soon as we start the world again. if mode != gcBackgroundMode { schedEnableUser(false) } // Enter concurrent mark phase and enable // write barriers. // // Because the world is stopped, all Ps will // observe that write barriers are enabled by // the time we start the world and begin // scanning. // // Write barriers must be enabled before assists are // enabled because they must be enabled before // any non-leaf heap objects are marked. Since // allocations are blocked until assists can // happen, we want to enable assists as early as // possible. setGCPhase(_GCmark) gcBgMarkPrepare() // Must happen before assists are enabled. gcMarkRootPrepare() // Mark all active tinyalloc blocks. Since we're // allocating from these, they need to be black like // other allocations. The alternative is to blacken // the tiny block on every allocation from it, which // would slow down the tiny allocator. gcMarkTinyAllocs() // At this point all Ps have enabled the write // barrier, thus maintaining the no white to // black invariant. Enable mutator assists to // put back-pressure on fast allocating // mutators. atomic.Store(&gcBlackenEnabled, 1) // In STW mode, we could block the instant systemstack // returns, so make sure we're not preemptible. mp = acquirem() // Concurrent mark. systemstack(func() { now = startTheWorldWithSema(0, stw) work.pauseNS += now - stw.start work.tMark = now sweepTermCpu := int64(work.stwprocs) * (work.tMark - work.tSweepTerm) work.cpuStats.gcPauseTime += sweepTermCpu work.cpuStats.gcTotalTime += sweepTermCpu // Release the CPU limiter. gcCPULimiter.finishGCTransition(now) }) // Release the world sema before Gosched() in STW mode // because we will need to reacquire it later but before // this goroutine becomes runnable again, and we could // self-deadlock otherwise. semrelease(&worldsema) releasem(mp) // Make sure we block instead of returning to user code // in STW mode. if mode != gcBackgroundMode { Gosched() } semrelease(&work.startSema) } // gcMarkDoneFlushed counts the number of P's with flushed work. // // Ideally this would be a captured local in gcMarkDone, but forEachP // escapes its callback closure, so it can't capture anything. // // This is protected by markDoneSema. var gcMarkDoneFlushed uint32 // gcMarkDone transitions the GC from mark to mark termination if all // reachable objects have been marked (that is, there are no grey // objects and can be no more in the future). Otherwise, it flushes // all local work to the global queues where it can be discovered by // other workers. // // This should be called when all local mark work has been drained and // there are no remaining workers. Specifically, when // // work.nwait == work.nproc && !gcMarkWorkAvailable(p) // // The calling context must be preemptible. // // Flushing local work is important because idle Ps may have local // work queued. This is the only way to make that work visible and // drive GC to completion. // // It is explicitly okay to have write barriers in this function. If // it does transition to mark termination, then all reachable objects // have been marked, so the write barrier cannot shade any more // objects. func gcMarkDone() { // Ensure only one thread is running the ragged barrier at a // time. semacquire(&work.markDoneSema) top: // Re-check transition condition under transition lock. // // It's critical that this checks the global work queues are // empty before performing the ragged barrier. Otherwise, // there could be global work that a P could take after the P // has passed the ragged barrier. if !(gcphase == _GCmark && work.nwait == work.nproc && !gcMarkWorkAvailable(nil)) { semrelease(&work.markDoneSema) return } // forEachP needs worldsema to execute, and we'll need it to // stop the world later, so acquire worldsema now. semacquire(&worldsema) // Flush all local buffers and collect flushedWork flags. gcMarkDoneFlushed = 0 forEachP(waitReasonGCMarkTermination, func(pp *p) { // Flush the write barrier buffer, since this may add // work to the gcWork. wbBufFlush1(pp) // Flush the gcWork, since this may create global work // and set the flushedWork flag. // // TODO(austin): Break up these workbufs to // better distribute work. pp.gcw.dispose() // Collect the flushedWork flag. if pp.gcw.flushedWork { atomic.Xadd(&gcMarkDoneFlushed, 1) pp.gcw.flushedWork = false } }) if gcMarkDoneFlushed != 0 { // More grey objects were discovered since the // previous termination check, so there may be more // work to do. Keep going. It's possible the // transition condition became true again during the // ragged barrier, so re-check it. semrelease(&worldsema) goto top } // There was no global work, no local work, and no Ps // communicated work since we took markDoneSema. Therefore // there are no grey objects and no more objects can be // shaded. Transition to mark termination. now := nanotime() work.tMarkTerm = now getg().m.preemptoff = "gcing" var stw worldStop systemstack(func() { stw = stopTheWorldWithSema(stwGCMarkTerm) }) // The gcphase is _GCmark, it will transition to _GCmarktermination // below. The important thing is that the wb remains active until // all marking is complete. This includes writes made by the GC. // There is sometimes work left over when we enter mark termination due // to write barriers performed after the completion barrier above. // Detect this and resume concurrent mark. This is obviously // unfortunate. // // See issue #27993 for details. // // Switch to the system stack to call wbBufFlush1, though in this case // it doesn't matter because we're non-preemptible anyway. restart := false systemstack(func() { for _, p := range allp { wbBufFlush1(p) if !p.gcw.empty() { restart = true break } } }) if restart { getg().m.preemptoff = "" systemstack(func() { now := startTheWorldWithSema(0, stw) work.pauseNS += now - stw.start }) semrelease(&worldsema) goto top } gcComputeStartingStackSize() // Disable assists and background workers. We must do // this before waking blocked assists. atomic.Store(&gcBlackenEnabled, 0) // Notify the CPU limiter that GC assists will now cease. gcCPULimiter.startGCTransition(false, now) // Wake all blocked assists. These will run when we // start the world again. gcWakeAllAssists() // Likewise, release the transition lock. Blocked // workers and assists will run when we start the // world again. semrelease(&work.markDoneSema) // In STW mode, re-enable user goroutines. These will be // queued to run after we start the world. schedEnableUser(true) // endCycle depends on all gcWork cache stats being flushed. // The termination algorithm above ensured that up to // allocations since the ragged barrier. gcController.endCycle(now, int(gomaxprocs), work.userForced) // Perform mark termination. This will restart the world. gcMarkTermination(stw) } // World must be stopped and mark assists and background workers must be // disabled. func gcMarkTermination(stw worldStop) { // Start marktermination (write barrier remains enabled for now). setGCPhase(_GCmarktermination) work.heap1 = gcController.heapLive.Load() startTime := nanotime() mp := acquirem() mp.preemptoff = "gcing" mp.traceback = 2 curgp := mp.curg // N.B. The execution tracer is not aware of this status // transition and handles it specially based on the // wait reason. casGToWaiting(curgp, _Grunning, waitReasonGarbageCollection) // Run gc on the g0 stack. We do this so that the g stack // we're currently running on will no longer change. Cuts // the root set down a bit (g0 stacks are not scanned, and // we don't need to scan gc's internal state). We also // need to switch to g0 so we can shrink the stack. systemstack(func() { gcMark(startTime) // Must return immediately. // The outer function's stack may have moved // during gcMark (it shrinks stacks, including the // outer function's stack), so we must not refer // to any of its variables. Return back to the // non-system stack to pick up the new addresses // before continuing. }) var stwSwept bool systemstack(func() { work.heap2 = work.bytesMarked if debug.gccheckmark > 0 { // Run a full non-parallel, stop-the-world // mark using checkmark bits, to check that we // didn't forget to mark anything during the // concurrent mark process. startCheckmarks() gcResetMarkState() gcw := &getg().m.p.ptr().gcw gcDrain(gcw, 0) wbBufFlush1(getg().m.p.ptr()) gcw.dispose() endCheckmarks() } // marking is complete so we can turn the write barrier off setGCPhase(_GCoff) stwSwept = gcSweep(work.mode) }) mp.traceback = 0 casgstatus(curgp, _Gwaiting, _Grunning) trace := traceAcquire() if trace.ok() { trace.GCDone() traceRelease(trace) } // all done mp.preemptoff = "" if gcphase != _GCoff { throw("gc done but gcphase != _GCoff") } // Record heapInUse for scavenger. memstats.lastHeapInUse = gcController.heapInUse.load() // Update GC trigger and pacing, as well as downstream consumers // of this pacing information, for the next cycle. systemstack(gcControllerCommit) // Update timing memstats now := nanotime() sec, nsec, _ := time_now() unixNow := sec*1e9 + int64(nsec) work.pauseNS += now - stw.start work.tEnd = now atomic.Store64(&memstats.last_gc_unix, uint64(unixNow)) // must be Unix time to make sense to user atomic.Store64(&memstats.last_gc_nanotime, uint64(now)) // monotonic time for us memstats.pause_ns[memstats.numgc%uint32(len(memstats.pause_ns))] = uint64(work.pauseNS) memstats.pause_end[memstats.numgc%uint32(len(memstats.pause_end))] = uint64(unixNow) memstats.pause_total_ns += uint64(work.pauseNS) markTermCpu := int64(work.stwprocs) * (work.tEnd - work.tMarkTerm) work.cpuStats.gcPauseTime += markTermCpu work.cpuStats.gcTotalTime += markTermCpu // Accumulate CPU stats. // // Pass gcMarkPhase=true so we can get all the latest GC CPU stats in there too. work.cpuStats.accumulate(now, true) // Compute overall GC CPU utilization. // Omit idle marking time from the overall utilization here since it's "free". memstats.gc_cpu_fraction = float64(work.cpuStats.gcTotalTime-work.cpuStats.gcIdleTime) / float64(work.cpuStats.totalTime) // Reset assist time and background time stats. // // Do this now, instead of at the start of the next GC cycle, because // these two may keep accumulating even if the GC is not active. scavenge.assistTime.Store(0) scavenge.backgroundTime.Store(0) // Reset idle time stat. sched.idleTime.Store(0) if work.userForced { memstats.numforcedgc++ } // Bump GC cycle count and wake goroutines waiting on sweep. lock(&work.sweepWaiters.lock) memstats.numgc++ injectglist(&work.sweepWaiters.list) unlock(&work.sweepWaiters.lock) // Increment the scavenge generation now. // // This moment represents peak heap in use because we're // about to start sweeping. mheap_.pages.scav.index.nextGen() // Release the CPU limiter. gcCPULimiter.finishGCTransition(now) // Finish the current heap profiling cycle and start a new // heap profiling cycle. We do this before starting the world // so events don't leak into the wrong cycle. mProf_NextCycle() // There may be stale spans in mcaches that need to be swept. // Those aren't tracked in any sweep lists, so we need to // count them against sweep completion until we ensure all // those spans have been forced out. // // If gcSweep fully swept the heap (for example if the sweep // is not concurrent due to a GODEBUG setting), then we expect // the sweepLocker to be invalid, since sweeping is done. // // N.B. Below we might duplicate some work from gcSweep; this is // fine as all that work is idempotent within a GC cycle, and // we're still holding worldsema so a new cycle can't start. sl := sweep.active.begin() if !stwSwept && !sl.valid { throw("failed to set sweep barrier") } else if stwSwept && sl.valid { throw("non-concurrent sweep failed to drain all sweep queues") } systemstack(func() { // The memstats updated above must be updated with the world // stopped to ensure consistency of some values, such as // sched.idleTime and sched.totaltime. memstats also include // the pause time (work,pauseNS), forcing computation of the // total pause time before the pause actually ends. // // Here we reuse the same now for start the world so that the // time added to /sched/pauses/total/gc:seconds will be // consistent with the value in memstats. startTheWorldWithSema(now, stw) }) // Flush the heap profile so we can start a new cycle next GC. // This is relatively expensive, so we don't do it with the // world stopped. mProf_Flush() // Prepare workbufs for freeing by the sweeper. We do this // asynchronously because it can take non-trivial time. prepareFreeWorkbufs() // Free stack spans. This must be done between GC cycles. systemstack(freeStackSpans) // Ensure all mcaches are flushed. Each P will flush its own // mcache before allocating, but idle Ps may not. Since this // is necessary to sweep all spans, we need to ensure all // mcaches are flushed before we start the next GC cycle. // // While we're here, flush the page cache for idle Ps to avoid // having pages get stuck on them. These pages are hidden from // the scavenger, so in small idle heaps a significant amount // of additional memory might be held onto. // // Also, flush the pinner cache, to avoid leaking that memory // indefinitely. forEachP(waitReasonFlushProcCaches, func(pp *p) { pp.mcache.prepareForSweep() if pp.status == _Pidle { systemstack(func() { lock(&mheap_.lock) pp.pcache.flush(&mheap_.pages) unlock(&mheap_.lock) }) } pp.pinnerCache = nil }) if sl.valid { // Now that we've swept stale spans in mcaches, they don't // count against unswept spans. // // Note: this sweepLocker may not be valid if sweeping had // already completed during the STW. See the corresponding // begin() call that produced sl. sweep.active.end(sl) } // Print gctrace before dropping worldsema. As soon as we drop // worldsema another cycle could start and smash the stats // we're trying to print. if debug.gctrace > 0 { util := int(memstats.gc_cpu_fraction * 100) var sbuf [24]byte printlock() print("gc ", memstats.numgc, " @", string(itoaDiv(sbuf[:], uint64(work.tSweepTerm-runtimeInitTime)/1e6, 3)), "s ", util, "%: ") prev := work.tSweepTerm for i, ns := range []int64{work.tMark, work.tMarkTerm, work.tEnd} { if i != 0 { print("+") } print(string(fmtNSAsMS(sbuf[:], uint64(ns-prev)))) prev = ns } print(" ms clock, ") for i, ns := range []int64{ int64(work.stwprocs) * (work.tMark - work.tSweepTerm), gcController.assistTime.Load(), gcController.dedicatedMarkTime.Load() + gcController.fractionalMarkTime.Load(), gcController.idleMarkTime.Load(), markTermCpu, } { if i == 2 || i == 3 { // Separate mark time components with /. print("/") } else if i != 0 { print("+") } print(string(fmtNSAsMS(sbuf[:], uint64(ns)))) } print(" ms cpu, ", work.heap0>>20, "->", work.heap1>>20, "->", work.heap2>>20, " MB, ", gcController.lastHeapGoal>>20, " MB goal, ", gcController.lastStackScan.Load()>>20, " MB stacks, ", gcController.globalsScan.Load()>>20, " MB globals, ", work.maxprocs, " P") if work.userForced { print(" (forced)") } print("\n") printunlock() } // Set any arena chunks that were deferred to fault. lock(&userArenaState.lock) faultList := userArenaState.fault userArenaState.fault = nil unlock(&userArenaState.lock) for _, lc := range faultList { lc.mspan.setUserArenaChunkToFault() } // Enable huge pages on some metadata if we cross a heap threshold. if gcController.heapGoal() > minHeapForMetadataHugePages { systemstack(func() { mheap_.enableMetadataHugePages() }) } semrelease(&worldsema) semrelease(&gcsema) // Careful: another GC cycle may start now. releasem(mp) mp = nil // now that gc is done, kick off finalizer thread if needed if !concurrentSweep { // give the queued finalizers, if any, a chance to run Gosched() } } // gcBgMarkStartWorkers prepares background mark worker goroutines. These // goroutines will not run until the mark phase, but they must be started while // the work is not stopped and from a regular G stack. The caller must hold // worldsema. func gcBgMarkStartWorkers() { // Background marking is performed by per-P G's. Ensure that each P has // a background GC G. // // Worker Gs don't exit if gomaxprocs is reduced. If it is raised // again, we can reuse the old workers; no need to create new workers. for gcBgMarkWorkerCount < gomaxprocs { go gcBgMarkWorker() notetsleepg(&work.bgMarkReady, -1) noteclear(&work.bgMarkReady) // The worker is now guaranteed to be added to the pool before // its P's next findRunnableGCWorker. gcBgMarkWorkerCount++ } } // gcBgMarkPrepare sets up state for background marking. // Mutator assists must not yet be enabled. func gcBgMarkPrepare() { // Background marking will stop when the work queues are empty // and there are no more workers (note that, since this is // concurrent, this may be a transient state, but mark // termination will clean it up). Between background workers // and assists, we don't really know how many workers there // will be, so we pretend to have an arbitrarily large number // of workers, almost all of which are "waiting". While a // worker is working it decrements nwait. If nproc == nwait, // there are no workers. work.nproc = ^uint32(0) work.nwait = ^uint32(0) } // gcBgMarkWorkerNode is an entry in the gcBgMarkWorkerPool. It points to a single // gcBgMarkWorker goroutine. type gcBgMarkWorkerNode struct { // Unused workers are managed in a lock-free stack. This field must be first. node lfnode // The g of this worker. gp guintptr // Release this m on park. This is used to communicate with the unlock // function, which cannot access the G's stack. It is unused outside of // gcBgMarkWorker(). m muintptr } func gcBgMarkWorker() { gp := getg() // We pass node to a gopark unlock function, so it can't be on // the stack (see gopark). Prevent deadlock from recursively // starting GC by disabling preemption. gp.m.preemptoff = "GC worker init" node := new(gcBgMarkWorkerNode) gp.m.preemptoff = "" node.gp.set(gp) node.m.set(acquirem()) notewakeup(&work.bgMarkReady) // After this point, the background mark worker is generally scheduled // cooperatively by gcController.findRunnableGCWorker. While performing // work on the P, preemption is disabled because we are working on // P-local work buffers. When the preempt flag is set, this puts itself // into _Gwaiting to be woken up by gcController.findRunnableGCWorker // at the appropriate time. // // When preemption is enabled (e.g., while in gcMarkDone), this worker // may be preempted and schedule as a _Grunnable G from a runq. That is // fine; it will eventually gopark again for further scheduling via // findRunnableGCWorker. // // Since we disable preemption before notifying bgMarkReady, we // guarantee that this G will be in the worker pool for the next // findRunnableGCWorker. This isn't strictly necessary, but it reduces // latency between _GCmark starting and the workers starting. for { // Go to sleep until woken by // gcController.findRunnableGCWorker. gopark(func(g *g, nodep unsafe.Pointer) bool { node := (*gcBgMarkWorkerNode)(nodep) if mp := node.m.ptr(); mp != nil { // The worker G is no longer running; release // the M. // // N.B. it is _safe_ to release the M as soon // as we are no longer performing P-local mark // work. // // However, since we cooperatively stop work // when gp.preempt is set, if we releasem in // the loop then the following call to gopark // would immediately preempt the G. This is // also safe, but inefficient: the G must // schedule again only to enter gopark and park // again. Thus, we defer the release until // after parking the G. releasem(mp) } // Release this G to the pool. gcBgMarkWorkerPool.push(&node.node) // Note that at this point, the G may immediately be // rescheduled and may be running. return true }, unsafe.Pointer(node), waitReasonGCWorkerIdle, traceBlockSystemGoroutine, 0) // Preemption must not occur here, or another G might see // p.gcMarkWorkerMode. // Disable preemption so we can use the gcw. If the // scheduler wants to preempt us, we'll stop draining, // dispose the gcw, and then preempt. node.m.set(acquirem()) pp := gp.m.p.ptr() // P can't change with preemption disabled. if gcBlackenEnabled == 0 { println("worker mode", pp.gcMarkWorkerMode) throw("gcBgMarkWorker: blackening not enabled") } if pp.gcMarkWorkerMode == gcMarkWorkerNotWorker { throw("gcBgMarkWorker: mode not set") } startTime := nanotime() pp.gcMarkWorkerStartTime = startTime var trackLimiterEvent bool if pp.gcMarkWorkerMode == gcMarkWorkerIdleMode { trackLimiterEvent = pp.limiterEvent.start(limiterEventIdleMarkWork, startTime) } decnwait := atomic.Xadd(&work.nwait, -1) if decnwait == work.nproc { println("runtime: work.nwait=", decnwait, "work.nproc=", work.nproc) throw("work.nwait was > work.nproc") } systemstack(func() { // Mark our goroutine preemptible so its stack // can be scanned. This lets two mark workers // scan each other (otherwise, they would // deadlock). We must not modify anything on // the G stack. However, stack shrinking is // disabled for mark workers, so it is safe to // read from the G stack. // // N.B. The execution tracer is not aware of this status // transition and handles it specially based on the // wait reason. casGToWaiting(gp, _Grunning, waitReasonGCWorkerActive) switch pp.gcMarkWorkerMode { default: throw("gcBgMarkWorker: unexpected gcMarkWorkerMode") case gcMarkWorkerDedicatedMode: gcDrainMarkWorkerDedicated(&pp.gcw, true) if gp.preempt { // We were preempted. This is // a useful signal to kick // everything out of the run // queue so it can run // somewhere else. if drainQ, n := runqdrain(pp); n > 0 { lock(&sched.lock) globrunqputbatch(&drainQ, int32(n)) unlock(&sched.lock) } } // Go back to draining, this time // without preemption. gcDrainMarkWorkerDedicated(&pp.gcw, false) case gcMarkWorkerFractionalMode: gcDrainMarkWorkerFractional(&pp.gcw) case gcMarkWorkerIdleMode: gcDrainMarkWorkerIdle(&pp.gcw) } casgstatus(gp, _Gwaiting, _Grunning) }) // Account for time and mark us as stopped. now := nanotime() duration := now - startTime gcController.markWorkerStop(pp.gcMarkWorkerMode, duration) if trackLimiterEvent { pp.limiterEvent.stop(limiterEventIdleMarkWork, now) } if pp.gcMarkWorkerMode == gcMarkWorkerFractionalMode { atomic.Xaddint64(&pp.gcFractionalMarkTime, duration) } // Was this the last worker and did we run out // of work? incnwait := atomic.Xadd(&work.nwait, +1) if incnwait > work.nproc { println("runtime: p.gcMarkWorkerMode=", pp.gcMarkWorkerMode, "work.nwait=", incnwait, "work.nproc=", work.nproc) throw("work.nwait > work.nproc") } // We'll releasem after this point and thus this P may run // something else. We must clear the worker mode to avoid // attributing the mode to a different (non-worker) G in // traceGoStart. pp.gcMarkWorkerMode = gcMarkWorkerNotWorker // If this worker reached a background mark completion // point, signal the main GC goroutine. if incnwait == work.nproc && !gcMarkWorkAvailable(nil) { // We don't need the P-local buffers here, allow // preemption because we may schedule like a regular // goroutine in gcMarkDone (block on locks, etc). releasem(node.m.ptr()) node.m.set(nil) gcMarkDone() } } } // gcMarkWorkAvailable reports whether executing a mark worker // on p is potentially useful. p may be nil, in which case it only // checks the global sources of work. func gcMarkWorkAvailable(p *p) bool { if p != nil && !p.gcw.empty() { return true } if !work.full.empty() { return true // global work available } if work.markrootNext < work.markrootJobs { return true // root scan work available } return false } // gcMark runs the mark (or, for concurrent GC, mark termination) // All gcWork caches must be empty. // STW is in effect at this point. func gcMark(startTime int64) { if debug.allocfreetrace > 0 { tracegc() } if gcphase != _GCmarktermination { throw("in gcMark expecting to see gcphase as _GCmarktermination") } work.tstart = startTime // Check that there's no marking work remaining. if work.full != 0 || work.markrootNext < work.markrootJobs { print("runtime: full=", hex(work.full), " next=", work.markrootNext, " jobs=", work.markrootJobs, " nDataRoots=", work.nDataRoots, " nBSSRoots=", work.nBSSRoots, " nSpanRoots=", work.nSpanRoots, " nStackRoots=", work.nStackRoots, "\n") panic("non-empty mark queue after concurrent mark") } if debug.gccheckmark > 0 { // This is expensive when there's a large number of // Gs, so only do it if checkmark is also enabled. gcMarkRootCheck() } // Drop allg snapshot. allgs may have grown, in which case // this is the only reference to the old backing store and // there's no need to keep it around. work.stackRoots = nil // Clear out buffers and double-check that all gcWork caches // are empty. This should be ensured by gcMarkDone before we // enter mark termination. // // TODO: We could clear out buffers just before mark if this // has a non-negligible impact on STW time. for _, p := range allp { // The write barrier may have buffered pointers since // the gcMarkDone barrier. However, since the barrier // ensured all reachable objects were marked, all of // these must be pointers to black objects. Hence we // can just discard the write barrier buffer. if debug.gccheckmark > 0 { // For debugging, flush the buffer and make // sure it really was all marked. wbBufFlush1(p) } else { p.wbBuf.reset() } gcw := &p.gcw if !gcw.empty() { printlock() print("runtime: P ", p.id, " flushedWork ", gcw.flushedWork) if gcw.wbuf1 == nil { print(" wbuf1=<nil>") } else { print(" wbuf1.n=", gcw.wbuf1.nobj) } if gcw.wbuf2 == nil { print(" wbuf2=<nil>") } else { print(" wbuf2.n=", gcw.wbuf2.nobj) } print("\n") throw("P has cached GC work at end of mark termination") } // There may still be cached empty buffers, which we // need to flush since we're going to free them. Also, // there may be non-zero stats because we allocated // black after the gcMarkDone barrier. gcw.dispose() } // Flush scanAlloc from each mcache since we're about to modify // heapScan directly. If we were to flush this later, then scanAlloc // might have incorrect information. // // Note that it's not important to retain this information; we know // exactly what heapScan is at this point via scanWork. for _, p := range allp { c := p.mcache if c == nil { continue } c.scanAlloc = 0 } // Reset controller state. gcController.resetLive(work.bytesMarked) } // gcSweep must be called on the system stack because it acquires the heap // lock. See mheap for details. // // Returns true if the heap was fully swept by this function. // // The world must be stopped. // //go:systemstack func gcSweep(mode gcMode) bool { assertWorldStopped() if gcphase != _GCoff { throw("gcSweep being done but phase is not GCoff") } lock(&mheap_.lock) mheap_.sweepgen += 2 sweep.active.reset() mheap_.pagesSwept.Store(0) mheap_.sweepArenas = mheap_.allArenas mheap_.reclaimIndex.Store(0) mheap_.reclaimCredit.Store(0) unlock(&mheap_.lock) sweep.centralIndex.clear() if !concurrentSweep || mode == gcForceBlockMode { // Special case synchronous sweep. // Record that no proportional sweeping has to happen. lock(&mheap_.lock) mheap_.sweepPagesPerByte = 0 unlock(&mheap_.lock) // Flush all mcaches. for _, pp := range allp { pp.mcache.prepareForSweep() } // Sweep all spans eagerly. for sweepone() != ^uintptr(0) { } // Free workbufs eagerly. prepareFreeWorkbufs() for freeSomeWbufs(false) { } // All "free" events for this mark/sweep cycle have // now happened, so we can make this profile cycle // available immediately. mProf_NextCycle() mProf_Flush() return true } // Background sweep. lock(&sweep.lock) if sweep.parked { sweep.parked = false ready(sweep.g, 0, true) } unlock(&sweep.lock) return false } // gcResetMarkState resets global state prior to marking (concurrent // or STW) and resets the stack scan state of all Gs. // // This is safe to do without the world stopped because any Gs created // during or after this will start out in the reset state. // // gcResetMarkState must be called on the system stack because it acquires // the heap lock. See mheap for details. // //go:systemstack func gcResetMarkState() { // This may be called during a concurrent phase, so lock to make sure // allgs doesn't change. forEachG(func(gp *g) { gp.gcscandone = false // set to true in gcphasework gp.gcAssistBytes = 0 }) // Clear page marks. This is just 1MB per 64GB of heap, so the // time here is pretty trivial. lock(&mheap_.lock) arenas := mheap_.allArenas unlock(&mheap_.lock) for _, ai := range arenas { ha := mheap_.arenas[ai.l1()][ai.l2()] for i := range ha.pageMarks { ha.pageMarks[i] = 0 } } work.bytesMarked = 0 work.initialHeapLive = gcController.heapLive.Load() } // Hooks for other packages var poolcleanup func() var boringCaches []unsafe.Pointer // for crypto/internal/boring //go:linkname sync_runtime_registerPoolCleanup sync.runtime_registerPoolCleanup func sync_runtime_registerPoolCleanup(f func()) { poolcleanup = f } //go:linkname boring_registerCache crypto/internal/boring/bcache.registerCache func boring_registerCache(p unsafe.Pointer) { boringCaches = append(boringCaches, p) } func clearpools() { // clear sync.Pools if poolcleanup != nil { poolcleanup() } // clear boringcrypto caches for _, p := range boringCaches { atomicstorep(p, nil) } // Clear central sudog cache. // Leave per-P caches alone, they have strictly bounded size. // Disconnect cached list before dropping it on the floor, // so that a dangling ref to one entry does not pin all of them. lock(&sched.sudoglock) var sg, sgnext *sudog for sg = sched.sudogcache; sg != nil; sg = sgnext { sgnext = sg.next sg.next = nil } sched.sudogcache = nil unlock(&sched.sudoglock) // Clear central defer pool. // Leave per-P pools alone, they have strictly bounded size. lock(&sched.deferlock) // disconnect cached list before dropping it on the floor, // so that a dangling ref to one entry does not pin all of them. var d, dlink *_defer for d = sched.deferpool; d != nil; d = dlink { dlink = d.link d.link = nil } sched.deferpool = nil unlock(&sched.deferlock) } // Timing // itoaDiv formats val/(10**dec) into buf. func itoaDiv(buf []byte, val uint64, dec int) []byte { i := len(buf) - 1 idec := i - dec for val >= 10 || i >= idec { buf[i] = byte(val%10 + '0') i-- if i == idec { buf[i] = '.' i-- } val /= 10 } buf[i] = byte(val + '0') return buf[i:] } // fmtNSAsMS nicely formats ns nanoseconds as milliseconds. func fmtNSAsMS(buf []byte, ns uint64) []byte { if ns >= 10e6 { // Format as whole milliseconds. return itoaDiv(buf, ns/1e6, 0) } // Format two digits of precision, with at most three decimal places. x := ns / 1e3 if x == 0 { buf[0] = '0' return buf[:1] } dec := 3 for x >= 100 { x /= 10 dec-- } return itoaDiv(buf, x, dec) } // Helpers for testing GC. // gcTestMoveStackOnNextCall causes the stack to be moved on a call // immediately following the call to this. It may not work correctly // if any other work appears after this call (such as returning). // Typically the following call should be marked go:noinline so it // performs a stack check. // // In rare cases this may not cause the stack to move, specifically if // there's a preemption between this call and the next. func gcTestMoveStackOnNextCall() { gp := getg() gp.stackguard0 = stackForceMove } // gcTestIsReachable performs a GC and returns a bit set where bit i // is set if ptrs[i] is reachable. func gcTestIsReachable(ptrs ...unsafe.Pointer) (mask uint64) { // This takes the pointers as unsafe.Pointers in order to keep // them live long enough for us to attach specials. After // that, we drop our references to them. if len(ptrs) > 64 { panic("too many pointers for uint64 mask") } // Block GC while we attach specials and drop our references // to ptrs. Otherwise, if a GC is in progress, it could mark // them reachable via this function before we have a chance to // drop them. semacquire(&gcsema) // Create reachability specials for ptrs. specials := make([]*specialReachable, len(ptrs)) for i, p := range ptrs { lock(&mheap_.speciallock) s := (*specialReachable)(mheap_.specialReachableAlloc.alloc()) unlock(&mheap_.speciallock) s.special.kind = _KindSpecialReachable if !addspecial(p, &s.special) { throw("already have a reachable special (duplicate pointer?)") } specials[i] = s // Make sure we don't retain ptrs. ptrs[i] = nil } semrelease(&gcsema) // Force a full GC and sweep. GC() // Process specials. for i, s := range specials { if !s.done { printlock() println("runtime: object", i, "was not swept") throw("IsReachable failed") } if s.reachable { mask |= 1 << i } lock(&mheap_.speciallock) mheap_.specialReachableAlloc.free(unsafe.Pointer(s)) unlock(&mheap_.speciallock) } return mask } // gcTestPointerClass returns the category of what p points to, one of: // "heap", "stack", "data", "bss", "other". This is useful for checking // that a test is doing what it's intended to do. // // This is nosplit simply to avoid extra pointer shuffling that may // complicate a test. // //go:nosplit func gcTestPointerClass(p unsafe.Pointer) string { p2 := uintptr(noescape(p)) gp := getg() if gp.stack.lo <= p2 && p2 < gp.stack.hi { return "stack" } if base, _, _ := findObject(p2, 0, 0); base != 0 { return "heap" } for _, datap := range activeModules() { if datap.data <= p2 && p2 < datap.edata || datap.noptrdata <= p2 && p2 < datap.enoptrdata { return "data" } if datap.bss <= p2 && p2 < datap.ebss || datap.noptrbss <= p2 && p2 <= datap.enoptrbss { return "bss" } } KeepAlive(p) return "other" }
Upload File
Create Folder