-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathatexit.go
90 lines (79 loc) · 2.49 KB
/
atexit.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
// Copyright 2015 Manlio Perillo. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
// Package atexit implements support for running deferred functions in case of
// abnormal exit (in contrast to a normal exit when the program returns from
// the main function).
//
// Since calling os.Exit, e.g. during a signal handler, does not call deferred
// functions, a complementary mechanism is required when the program acquires
// resources that are not automatically released by the operating system at
// program termination, e.g. SYSV shared memory.
//
// atexit is designed to work with, and complement, Go standard deferred
// mechanism.
// The Exit function provided by this package must be used, in order to run
// registered deferred functions.
//
// The Exit function SHOULD only be called in case of abnormal program
// termination.
package atexit
import (
"os"
"sync"
)
type deferred struct {
f func()
once sync.Once
}
// call calls the deferred function once, either during standard deferred
// mechanism or during an abnormal program termination.
func (d *deferred) call() {
d.once.Do(d.f)
}
// Do registers the function f to be called in case of abnormal program
// termination, caused by calling the Exit function, and returns a function
// that can be used in a defer statement.
//
// Do SHOULD only be used to release resources that are not automatically
// released by the operating system at program termination.
// The idiomatic use of Do is
// AcquireResource(...)
// ...
// defer atexit.Do(func() {
// ReleaseResource(...)
// })()
//
// The function f is called exactly once, either during standard deferred
// mechanism or as part of atexit termination.
//
// If f panics, atexit considers it to have returned; it will not be called again.
func Do(f func()) func() {
d := &deferred{f: f}
mu.Lock()
dl = append(dl, d)
mu.Unlock()
return d.call
}
// Registered deferred functions.
var (
mu sync.Mutex // guards dl
dl = make([]*deferred, 0, 10)
)
// exit runs all registered deferred functions, in FIFO order. It is used for
// testing.
func exit() {
mu.Lock()
defer mu.Unlock()
for _, d := range dl {
defer d.call()
}
}
// Exit runs all registered deferred functions, in FIFO order, and then causes
// the current program to exit with the given status code.
// In case one of the deferred functions panics, the exit status is ignored and
// control passes to Go runtime.
func Exit(code int) {
exit()
os.Exit(code)
}