Skip to content
This repository was archived by the owner on Sep 9, 2020. It is now read-only.

Commit aa8e076

Browse files
authored
Merge pull request #1154 from tamird/remove-unnecessary-once
gps: simplify shutdown cleanup
2 parents 586f5a1 + 45da465 commit aa8e076

File tree

1 file changed

+41
-73
lines changed

1 file changed

+41
-73
lines changed

internal/gps/source_manager.go

+41-73
Original file line numberDiff line numberDiff line change
@@ -275,42 +275,22 @@ func (sm *SourceMgr) HandleSignals(sigch chan os.Signal) {
275275
// Run a new goroutine with the input sigch and the fresh qch
276276
go func(sch chan os.Signal, qch <-chan struct{}) {
277277
defer signal.Stop(sch)
278-
for {
279-
select {
280-
case <-sch:
281-
// Set up a timer to uninstall the signal handler after three
282-
// seconds, so that the user can easily force termination with a
283-
// second ctrl-c
284-
go func(c <-chan time.Time) {
285-
<-c
286-
signal.Stop(sch)
287-
}(time.After(3 * time.Second))
288-
289-
if !atomic.CompareAndSwapInt32(&sm.releasing, 0, 1) {
290-
// Something's already called Release() on this sm, so we
291-
// don't have to do anything, as we'd just be redoing
292-
// that work. Instead, deregister and return.
293-
return
294-
}
295-
296-
opc := sm.suprvsr.count()
297-
if opc > 0 {
298-
fmt.Printf("Signal received: waiting for %v ops to complete...\n", opc)
299-
}
300-
301-
// Mutex interaction in a signal handler is, as a general rule,
302-
// unsafe. I'm not clear on whether the guarantees Go provides
303-
// around signal handling, or having passed this through a
304-
// channel in general, obviate those concerns, but it's a lot
305-
// easier to just rely on the mutex contained in the Once right
306-
// now, so do that until it proves problematic or someone
307-
// provides a clear explanation.
308-
sm.relonce.Do(func() { sm.doRelease() })
309-
return
310-
case <-qch:
311-
// quit channel triggered - deregister our sigch and return
312-
return
278+
select {
279+
case <-sch:
280+
// Set up a timer to uninstall the signal handler after three
281+
// seconds, so that the user can easily force termination with a
282+
// second ctrl-c
283+
time.AfterFunc(3*time.Second, func() {
284+
signal.Stop(sch)
285+
})
286+
287+
if opc := sm.suprvsr.count(); opc > 0 {
288+
fmt.Printf("Signal received: waiting for %v ops to complete...\n", opc)
313289
}
290+
291+
sm.Release()
292+
case <-qch:
293+
// quit channel triggered - deregister our sigch and return
314294
}
315295
}(sigch, sm.qch)
316296
// Try to ensure handler is blocked in for-select before releasing the mutex
@@ -349,46 +329,34 @@ func (e CouldNotCreateLockError) Error() string {
349329
// longer safe to call methods against it; all method calls will immediately
350330
// result in errors.
351331
func (sm *SourceMgr) Release() {
352-
// Set sm.releasing before entering the Once func to guarantee that no
353-
// _more_ method calls will stack up if/while waiting.
354-
atomic.CompareAndSwapInt32(&sm.releasing, 0, 1)
332+
atomic.StoreInt32(&sm.releasing, 1)
355333

356-
// Whether 'releasing' is set or not, we don't want this function to return
357-
// until after the doRelease process is done, as doing so could cause the
358-
// process to terminate before a signal-driven doRelease() call has a chance
359-
// to finish its cleanup.
360-
sm.relonce.Do(sm.doRelease)
361-
}
334+
sm.relonce.Do(func() {
335+
// Send the signal to the supervisor to cancel all running calls.
336+
sm.cancelAll()
337+
sm.suprvsr.wait()
362338

363-
// doRelease actually releases physical resources (files on disk, etc.).
364-
//
365-
// This must be called only and exactly once. Calls to it should be wrapped in
366-
// the sm.relonce sync.Once instance.
367-
func (sm *SourceMgr) doRelease() {
368-
// Send the signal to the supervisor to cancel all running calls
369-
sm.cancelAll()
370-
sm.suprvsr.wait()
371-
372-
// Close the source coordinator.
373-
sm.srcCoord.close()
374-
375-
// Close the file handle for the lock file and remove it from disk
376-
sm.lf.Unlock()
377-
os.Remove(filepath.Join(sm.cachedir, "sm.lock"))
378-
379-
// Close the qch, if non-nil, so the signal handlers run out. This will
380-
// also deregister the sig channel, if any has been set up.
381-
if sm.qch != nil {
382-
close(sm.qch)
383-
}
339+
// Close the source coordinator.
340+
sm.srcCoord.close()
341+
342+
// Close the file handle for the lock file and remove it from disk
343+
sm.lf.Unlock()
344+
os.Remove(filepath.Join(sm.cachedir, "sm.lock"))
345+
346+
// Close the qch, if non-nil, so the signal handlers run out. This will
347+
// also deregister the sig channel, if any has been set up.
348+
if sm.qch != nil {
349+
close(sm.qch)
350+
}
351+
})
384352
}
385353

386354
// GetManifestAndLock returns manifest and lock information for the provided
387355
// ProjectIdentifier, at the provided Version. The work of producing the
388356
// manifest and lock is delegated to the provided ProjectAnalyzer's
389357
// DeriveManifestAndLock() method.
390358
func (sm *SourceMgr) GetManifestAndLock(id ProjectIdentifier, v Version, an ProjectAnalyzer) (Manifest, Lock, error) {
391-
if atomic.CompareAndSwapInt32(&sm.releasing, 1, 1) {
359+
if atomic.LoadInt32(&sm.releasing) == 1 {
392360
return nil, nil, smIsReleased{}
393361
}
394362

@@ -403,7 +371,7 @@ func (sm *SourceMgr) GetManifestAndLock(id ProjectIdentifier, v Version, an Proj
403371
// ListPackages parses the tree of the Go packages at and below the ProjectRoot
404372
// of the given ProjectIdentifier, at the given version.
405373
func (sm *SourceMgr) ListPackages(id ProjectIdentifier, v Version) (pkgtree.PackageTree, error) {
406-
if atomic.CompareAndSwapInt32(&sm.releasing, 1, 1) {
374+
if atomic.LoadInt32(&sm.releasing) == 1 {
407375
return pkgtree.PackageTree{}, smIsReleased{}
408376
}
409377

@@ -428,7 +396,7 @@ func (sm *SourceMgr) ListPackages(id ProjectIdentifier, v Version) (pkgtree.Pack
428396
// is not accessible (network outage, access issues, or the resource actually
429397
// went away), an error will be returned.
430398
func (sm *SourceMgr) ListVersions(id ProjectIdentifier) ([]PairedVersion, error) {
431-
if atomic.CompareAndSwapInt32(&sm.releasing, 1, 1) {
399+
if atomic.LoadInt32(&sm.releasing) == 1 {
432400
return nil, smIsReleased{}
433401
}
434402

@@ -444,7 +412,7 @@ func (sm *SourceMgr) ListVersions(id ProjectIdentifier) ([]PairedVersion, error)
444412
// RevisionPresentIn indicates whether the provided Revision is present in the given
445413
// repository.
446414
func (sm *SourceMgr) RevisionPresentIn(id ProjectIdentifier, r Revision) (bool, error) {
447-
if atomic.CompareAndSwapInt32(&sm.releasing, 1, 1) {
415+
if atomic.LoadInt32(&sm.releasing) == 1 {
448416
return false, smIsReleased{}
449417
}
450418

@@ -460,7 +428,7 @@ func (sm *SourceMgr) RevisionPresentIn(id ProjectIdentifier, r Revision) (bool,
460428
// SourceExists checks if a repository exists, either upstream or in the cache,
461429
// for the provided ProjectIdentifier.
462430
func (sm *SourceMgr) SourceExists(id ProjectIdentifier) (bool, error) {
463-
if atomic.CompareAndSwapInt32(&sm.releasing, 1, 1) {
431+
if atomic.LoadInt32(&sm.releasing) == 1 {
464432
return false, smIsReleased{}
465433
}
466434

@@ -478,7 +446,7 @@ func (sm *SourceMgr) SourceExists(id ProjectIdentifier) (bool, error) {
478446
//
479447
// The primary use case for this is prefetching.
480448
func (sm *SourceMgr) SyncSourceFor(id ProjectIdentifier) error {
481-
if atomic.CompareAndSwapInt32(&sm.releasing, 1, 1) {
449+
if atomic.LoadInt32(&sm.releasing) == 1 {
482450
return smIsReleased{}
483451
}
484452

@@ -493,7 +461,7 @@ func (sm *SourceMgr) SyncSourceFor(id ProjectIdentifier) error {
493461
// ExportProject writes out the tree of the provided ProjectIdentifier's
494462
// ProjectRoot, at the provided version, to the provided directory.
495463
func (sm *SourceMgr) ExportProject(id ProjectIdentifier, v Version, to string) error {
496-
if atomic.CompareAndSwapInt32(&sm.releasing, 1, 1) {
464+
if atomic.LoadInt32(&sm.releasing) == 1 {
497465
return smIsReleased{}
498466
}
499467

@@ -513,7 +481,7 @@ func (sm *SourceMgr) ExportProject(id ProjectIdentifier, v Version, to string) e
513481
// paths. (A special exception is written for gopkg.in to minimize network
514482
// activity, as its behavior is well-structured)
515483
func (sm *SourceMgr) DeduceProjectRoot(ip string) (ProjectRoot, error) {
516-
if atomic.CompareAndSwapInt32(&sm.releasing, 1, 1) {
484+
if atomic.LoadInt32(&sm.releasing) == 1 {
517485
return "", smIsReleased{}
518486
}
519487

0 commit comments

Comments
 (0)