2013-03-13 03:29:40 -04:00
|
|
|
package term
|
2013-02-13 20:10:00 -05:00
|
|
|
|
|
|
|
import (
|
2013-05-14 18:37:35 -04:00
|
|
|
"os"
|
|
|
|
"os/signal"
|
2013-02-13 20:10:00 -05:00
|
|
|
"syscall"
|
|
|
|
"unsafe"
|
|
|
|
)
|
|
|
|
|
|
|
|
type State struct {
|
2013-02-26 20:26:46 -05:00
|
|
|
termios Termios
|
2013-02-13 20:10:00 -05:00
|
|
|
}
|
|
|
|
|
2013-05-24 17:44:16 -04:00
|
|
|
type Winsize struct {
|
|
|
|
Height uint16
|
2013-07-08 13:20:13 -04:00
|
|
|
Width uint16
|
2013-05-24 17:44:16 -04:00
|
|
|
x uint16
|
|
|
|
y uint16
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetWinsize(fd uintptr) (*Winsize, error) {
|
|
|
|
ws := &Winsize{}
|
|
|
|
_, _, err := syscall.Syscall(syscall.SYS_IOCTL, fd, uintptr(syscall.TIOCGWINSZ), uintptr(unsafe.Pointer(ws)))
|
2013-10-16 19:05:50 -04:00
|
|
|
// Skipp errno = 0
|
|
|
|
if err == 0 {
|
|
|
|
return ws, nil
|
|
|
|
}
|
2013-05-24 17:44:16 -04:00
|
|
|
return ws, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func SetWinsize(fd uintptr, ws *Winsize) error {
|
|
|
|
_, _, err := syscall.Syscall(syscall.SYS_IOCTL, fd, uintptr(syscall.TIOCSWINSZ), uintptr(unsafe.Pointer(ws)))
|
2013-10-16 19:05:50 -04:00
|
|
|
// Skipp errno = 0
|
|
|
|
if err == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
2013-05-24 17:44:16 -04:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2013-02-13 20:10:00 -05:00
|
|
|
// IsTerminal returns true if the given file descriptor is a terminal.
|
2013-06-01 18:55:05 -04:00
|
|
|
func IsTerminal(fd uintptr) bool {
|
2013-02-26 20:26:46 -05:00
|
|
|
var termios Termios
|
2013-06-01 18:55:05 -04:00
|
|
|
_, _, err := syscall.Syscall(syscall.SYS_IOCTL, fd, uintptr(getTermios), uintptr(unsafe.Pointer(&termios)))
|
2013-02-26 20:26:46 -05:00
|
|
|
return err == 0
|
2013-02-13 20:10:00 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// Restore restores the terminal connected to the given file descriptor to a
|
|
|
|
// previous state.
|
2013-06-24 17:52:02 -04:00
|
|
|
func RestoreTerminal(fd uintptr, state *State) error {
|
2013-06-01 18:55:05 -04:00
|
|
|
_, _, err := syscall.Syscall(syscall.SYS_IOCTL, fd, uintptr(setTermios), uintptr(unsafe.Pointer(&state.termios)))
|
2013-02-26 20:26:46 -05:00
|
|
|
return err
|
2013-02-13 20:10:00 -05:00
|
|
|
}
|
2013-05-14 18:37:35 -04:00
|
|
|
|
2013-08-03 19:43:20 -04:00
|
|
|
func SaveState(fd uintptr) (*State, error) {
|
|
|
|
var oldState State
|
|
|
|
if _, _, err := syscall.Syscall(syscall.SYS_IOCTL, fd, getTermios, uintptr(unsafe.Pointer(&oldState.termios))); err != 0 {
|
2013-05-14 18:37:35 -04:00
|
|
|
return nil, err
|
|
|
|
}
|
2013-08-03 19:43:20 -04:00
|
|
|
|
|
|
|
return &oldState, nil
|
|
|
|
}
|
|
|
|
|
2013-08-18 00:29:37 -04:00
|
|
|
func DisableEcho(fd uintptr, state *State) error {
|
2013-08-03 19:43:20 -04:00
|
|
|
newState := state.termios
|
|
|
|
newState.Lflag &^= syscall.ECHO
|
|
|
|
|
|
|
|
if _, _, err := syscall.Syscall(syscall.SYS_IOCTL, fd, setTermios, uintptr(unsafe.Pointer(&newState))); err != 0 {
|
|
|
|
return err
|
|
|
|
}
|
2013-08-18 00:29:37 -04:00
|
|
|
handleInterrupt(fd, state)
|
2013-08-03 19:43:20 -04:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2013-08-18 00:29:37 -04:00
|
|
|
func SetRawTerminal(fd uintptr) (*State, error) {
|
|
|
|
oldState, err := MakeRaw(fd)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
handleInterrupt(fd, oldState)
|
|
|
|
return oldState, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func handleInterrupt(fd uintptr, state *State) {
|
2013-08-03 19:43:20 -04:00
|
|
|
sigchan := make(chan os.Signal, 1)
|
|
|
|
signal.Notify(sigchan, os.Interrupt)
|
|
|
|
|
2013-05-14 18:37:35 -04:00
|
|
|
go func() {
|
2013-08-03 19:43:20 -04:00
|
|
|
_ = <-sigchan
|
|
|
|
RestoreTerminal(fd, state)
|
2013-05-14 18:37:35 -04:00
|
|
|
os.Exit(0)
|
|
|
|
}()
|
2013-08-03 19:43:20 -04:00
|
|
|
}
|