// Copyright (c) 2013-2014 Conformal Systems LLC. // Use of this source code is governed by an ISC // license that can be found in the LICENSE file. package main import ( "os" "os/signal" ) // interruptChannel is used to receive SIGINT (Ctrl+C) signals. var interruptChannel chan os.Signal // addHandlerChannel is used to add an interrupt handler to the list of handlers // to be invoked on SIGINT (Ctrl+C) signals. var addHandlerChannel = make(chan func()) // mainInterruptHandler listens for SIGINT (Ctrl+C) signals on the // interruptChannel and invokes the registered interruptCallbacks accordingly. // It also listens for callback registration. It must be run as a goroutine. func mainInterruptHandler() { // interruptCallbacks is a list of callbacks to invoke when a // SIGINT (Ctrl+C) is received. var interruptCallbacks []func() for { select { case <-interruptChannel: btcdLog.Infof("Received SIGINT (Ctrl+C). Shutting down...") // run handlers in LIFO order. for i := range interruptCallbacks { idx := len(interruptCallbacks) - 1 - i callback := interruptCallbacks[idx] callback() } // Signal the main goroutine to shutdown. shutdownChannel <- true case handler := <-addHandlerChannel: interruptCallbacks = append(interruptCallbacks, handler) } } } // addInterruptHandler adds a handler to call when a SIGINT (Ctrl+C) is // received. func addInterruptHandler(handler func()) { // Create the channel and start the main interrupt handler which invokes // all other callbacks and exits if not already done. if interruptChannel == nil { interruptChannel = make(chan os.Signal, 1) signal.Notify(interruptChannel, os.Interrupt) go mainInterruptHandler() } addHandlerChannel <- handler }