@@ -275,42 +275,22 @@ func (sm *SourceMgr) HandleSignals(sigch chan os.Signal) {
275
275
// Run a new goroutine with the input sigch and the fresh qch
276
276
go func (sch chan os.Signal , qch <- chan struct {}) {
277
277
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 )
313
289
}
290
+
291
+ sm .Release ()
292
+ case <- qch :
293
+ // quit channel triggered - deregister our sigch and return
314
294
}
315
295
}(sigch , sm .qch )
316
296
// Try to ensure handler is blocked in for-select before releasing the mutex
@@ -349,46 +329,34 @@ func (e CouldNotCreateLockError) Error() string {
349
329
// longer safe to call methods against it; all method calls will immediately
350
330
// result in errors.
351
331
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 )
355
333
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 ()
362
338
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
+ })
384
352
}
385
353
386
354
// GetManifestAndLock returns manifest and lock information for the provided
387
355
// ProjectIdentifier, at the provided Version. The work of producing the
388
356
// manifest and lock is delegated to the provided ProjectAnalyzer's
389
357
// DeriveManifestAndLock() method.
390
358
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 {
392
360
return nil , nil , smIsReleased {}
393
361
}
394
362
@@ -403,7 +371,7 @@ func (sm *SourceMgr) GetManifestAndLock(id ProjectIdentifier, v Version, an Proj
403
371
// ListPackages parses the tree of the Go packages at and below the ProjectRoot
404
372
// of the given ProjectIdentifier, at the given version.
405
373
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 {
407
375
return pkgtree.PackageTree {}, smIsReleased {}
408
376
}
409
377
@@ -428,7 +396,7 @@ func (sm *SourceMgr) ListPackages(id ProjectIdentifier, v Version) (pkgtree.Pack
428
396
// is not accessible (network outage, access issues, or the resource actually
429
397
// went away), an error will be returned.
430
398
func (sm * SourceMgr ) ListVersions (id ProjectIdentifier ) ([]PairedVersion , error ) {
431
- if atomic .CompareAndSwapInt32 (& sm .releasing , 1 , 1 ) {
399
+ if atomic .LoadInt32 (& sm .releasing ) == 1 {
432
400
return nil , smIsReleased {}
433
401
}
434
402
@@ -444,7 +412,7 @@ func (sm *SourceMgr) ListVersions(id ProjectIdentifier) ([]PairedVersion, error)
444
412
// RevisionPresentIn indicates whether the provided Revision is present in the given
445
413
// repository.
446
414
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 {
448
416
return false , smIsReleased {}
449
417
}
450
418
@@ -460,7 +428,7 @@ func (sm *SourceMgr) RevisionPresentIn(id ProjectIdentifier, r Revision) (bool,
460
428
// SourceExists checks if a repository exists, either upstream or in the cache,
461
429
// for the provided ProjectIdentifier.
462
430
func (sm * SourceMgr ) SourceExists (id ProjectIdentifier ) (bool , error ) {
463
- if atomic .CompareAndSwapInt32 (& sm .releasing , 1 , 1 ) {
431
+ if atomic .LoadInt32 (& sm .releasing ) == 1 {
464
432
return false , smIsReleased {}
465
433
}
466
434
@@ -478,7 +446,7 @@ func (sm *SourceMgr) SourceExists(id ProjectIdentifier) (bool, error) {
478
446
//
479
447
// The primary use case for this is prefetching.
480
448
func (sm * SourceMgr ) SyncSourceFor (id ProjectIdentifier ) error {
481
- if atomic .CompareAndSwapInt32 (& sm .releasing , 1 , 1 ) {
449
+ if atomic .LoadInt32 (& sm .releasing ) == 1 {
482
450
return smIsReleased {}
483
451
}
484
452
@@ -493,7 +461,7 @@ func (sm *SourceMgr) SyncSourceFor(id ProjectIdentifier) error {
493
461
// ExportProject writes out the tree of the provided ProjectIdentifier's
494
462
// ProjectRoot, at the provided version, to the provided directory.
495
463
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 {
497
465
return smIsReleased {}
498
466
}
499
467
@@ -513,7 +481,7 @@ func (sm *SourceMgr) ExportProject(id ProjectIdentifier, v Version, to string) e
513
481
// paths. (A special exception is written for gopkg.in to minimize network
514
482
// activity, as its behavior is well-structured)
515
483
func (sm * SourceMgr ) DeduceProjectRoot (ip string ) (ProjectRoot , error ) {
516
- if atomic .CompareAndSwapInt32 (& sm .releasing , 1 , 1 ) {
484
+ if atomic .LoadInt32 (& sm .releasing ) == 1 {
517
485
return "" , smIsReleased {}
518
486
}
519
487
0 commit comments