diff options
| author | Than McIntosh <thanm@google.com> | 2021-10-08 11:51:40 -0400 |
|---|---|---|
| committer | Than McIntosh <thanm@google.com> | 2022-09-26 21:49:14 +0000 |
| commit | 07bdf1dc545c6f6e59f5780a9024e6f230d9b434 (patch) | |
| tree | e3695600b93acdc672438ebd432389b15bead6ef /src/runtime/testdata | |
| parent | cf83a490e495e5bfa7065cd97811e689101a687e (diff) | |
| download | go-07bdf1dc545c6f6e59f5780a9024e6f230d9b434.tar.xz | |
runtime: add an exit hook facility
Add a new API (not public/exported) for registering a function with
the runtime that should be called when program execution terminates,
to be used in the new code coverage re-implementation. The API looks
like
func addExitHook(f func(), runOnNonZeroExit bool)
The first argument is the function to be run, second argument controls
whether the function is invoked even if there is a call to os.Exit
with a non-zero status. Exit hooks are run in reverse order of
registration, e.g. the first hook to be registered will be the last to
run. Exit hook functions are not allowed to panic or to make calls to
os.Exit.
Updates #51430.
Change-Id: I906f8c5184b7c1666f05a62cfc7833bf1a4300c4
Reviewed-on: https://go-review.googlesource.com/c/go/+/354790
TryBot-Result: Gopher Robot <gobot@golang.org>
Run-TryBot: Than McIntosh <thanm@google.com>
Reviewed-by: Michael Knyszek <mknyszek@google.com>
Diffstat (limited to 'src/runtime/testdata')
| -rw-r--r-- | src/runtime/testdata/testexithooks/testexithooks.go | 87 |
1 files changed, 87 insertions, 0 deletions
diff --git a/src/runtime/testdata/testexithooks/testexithooks.go b/src/runtime/testdata/testexithooks/testexithooks.go new file mode 100644 index 0000000000..57561faf8e --- /dev/null +++ b/src/runtime/testdata/testexithooks/testexithooks.go @@ -0,0 +1,87 @@ +// Copyright 2022 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 + +import ( + "flag" + "os" + _ "unsafe" +) + +import "C" + +var modeflag = flag.String("mode", "", "mode to run in") + +func main() { + flag.Parse() + switch *modeflag { + case "simple": + testSimple() + case "goodexit": + testGoodExit() + case "badexit": + testBadExit() + case "panics": + testPanics() + case "callsexit": + testHookCallsExit() + default: + panic("unknown mode") + } +} + +//go:linkname runtime_addExitHook runtime.addExitHook +func runtime_addExitHook(f func(), runOnNonZeroExit bool) + +func testSimple() { + f1 := func() { println("foo") } + f2 := func() { println("bar") } + runtime_addExitHook(f1, false) + runtime_addExitHook(f2, false) + // no explicit call to os.Exit +} + +func testGoodExit() { + f1 := func() { println("apple") } + f2 := func() { println("orange") } + runtime_addExitHook(f1, false) + runtime_addExitHook(f2, false) + // explicit call to os.Exit + os.Exit(0) +} + +func testBadExit() { + f1 := func() { println("blog") } + f2 := func() { println("blix") } + f3 := func() { println("blek") } + f4 := func() { println("blub") } + f5 := func() { println("blat") } + runtime_addExitHook(f1, false) + runtime_addExitHook(f2, true) + runtime_addExitHook(f3, false) + runtime_addExitHook(f4, true) + runtime_addExitHook(f5, false) + os.Exit(1) +} + +func testPanics() { + f1 := func() { println("ok") } + f2 := func() { panic("BADBADBAD") } + f3 := func() { println("good") } + runtime_addExitHook(f1, true) + runtime_addExitHook(f2, true) + runtime_addExitHook(f3, true) + os.Exit(0) +} + +func testHookCallsExit() { + f1 := func() { println("ok") } + f2 := func() { os.Exit(1) } + f3 := func() { println("good") } + runtime_addExitHook(f1, true) + runtime_addExitHook(f2, true) + runtime_addExitHook(f3, true) + os.Exit(1) +} |
