123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108 |
- // +build linux darwin freebsd netbsd openbsd solaris dragonfly
- // +build !appengine
- package pb
- import (
- "errors"
- "fmt"
- "os"
- "os/signal"
- "sync"
- "syscall"
- "golang.org/x/sys/unix"
- )
- var ErrPoolWasStarted = errors.New("Bar pool was started")
- var (
- echoLockMutex sync.Mutex
- origTermStatePtr *unix.Termios
- tty *os.File
- )
- func init() {
- echoLockMutex.Lock()
- defer echoLockMutex.Unlock()
- var err error
- tty, err = os.Open("/dev/tty")
- if err != nil {
- tty = os.Stdin
- }
- }
- // terminalWidth returns width of the terminal.
- func terminalWidth() (int, error) {
- echoLockMutex.Lock()
- defer echoLockMutex.Unlock()
- fd := int(tty.Fd())
- ws, err := unix.IoctlGetWinsize(fd, unix.TIOCGWINSZ)
- if err != nil {
- return 0, err
- }
- return int(ws.Col), nil
- }
- func lockEcho() (shutdownCh chan struct{}, err error) {
- echoLockMutex.Lock()
- defer echoLockMutex.Unlock()
- if origTermStatePtr != nil {
- return shutdownCh, ErrPoolWasStarted
- }
- fd := int(tty.Fd())
- origTermStatePtr, err = unix.IoctlGetTermios(fd, ioctlReadTermios)
- if err != nil {
- return nil, fmt.Errorf("Can't get terminal settings: %v", err)
- }
- oldTermios := *origTermStatePtr
- newTermios := oldTermios
- newTermios.Lflag &^= syscall.ECHO
- newTermios.Lflag |= syscall.ICANON | syscall.ISIG
- newTermios.Iflag |= syscall.ICRNL
- if err := unix.IoctlSetTermios(fd, ioctlWriteTermios, &newTermios); err != nil {
- return nil, fmt.Errorf("Can't set terminal settings: %v", err)
- }
- shutdownCh = make(chan struct{})
- go catchTerminate(shutdownCh)
- return
- }
- func unlockEcho() error {
- echoLockMutex.Lock()
- defer echoLockMutex.Unlock()
- if origTermStatePtr == nil {
- return nil
- }
- fd := int(tty.Fd())
- if err := unix.IoctlSetTermios(fd, ioctlWriteTermios, origTermStatePtr); err != nil {
- return fmt.Errorf("Can't set terminal settings: %v", err)
- }
- origTermStatePtr = nil
- return nil
- }
- // listen exit signals and restore terminal state
- func catchTerminate(shutdownCh chan struct{}) {
- sig := make(chan os.Signal, 1)
- signal.Notify(sig, os.Interrupt, syscall.SIGQUIT, syscall.SIGTERM, syscall.SIGKILL)
- defer signal.Stop(sig)
- select {
- case <-shutdownCh:
- unlockEcho()
- case <-sig:
- unlockEcho()
- }
- }
|