package termios // #include // typedef struct termios termios_t; import "C" import ( "golang.org/x/sys/unix" "unsafe" ) const ( TCSETS = 0x5402 TCSETSW = 0x5403 TCSETSF = 0x5404 TCSBRK = 0x5409 TCSBRKP = 0x5425 IXON = 0x00000400 IXANY = 0x00000800 IXOFF = 0x00001000 CRTSCTS = 0x80000000 ) // Tcgetattr gets the current serial port settings. func Tcgetattr(fd uintptr) (*unix.Termios, error) { return unix.IoctlGetTermios(int(fd), unix.TCGETS) } // Tcsetattr sets the current serial port settings. func Tcsetattr(fd, action uintptr, argp *unix.Termios) error { return unix.IoctlSetTermios(int(fd), uint(action), tiosToUnix(argp)) } // Tcsendbreak transmits a continuous stream of zero-valued bits for a specific // duration, if the terminal is using asynchronous serial data transmission. If // duration is zero, it transmits zero-valued bits for at least 0.25 seconds, and not more that 0.5 seconds. // If duration is not zero, it sends zero-valued bits for some // implementation-defined length of time. func Tcsendbreak(fd uintptr, duration int) error { return ioctl(fd, TCSBRKP, uintptr(duration)) } // Tcdrain waits until all output written to the object referred to by fd has been transmitted. func Tcdrain(fd uintptr) error { // simulate drain with TCSADRAIN var attr unix.Termios if err := Tcgetattr(fd, &attr); err != nil { return err } return Tcsetattr(fd, TCSADRAIN, &attr) } // Tcflush discards data written to the object referred to by fd but not transmitted, or data received but not read, depending on the value of selector. func Tcflush(fd, selector uintptr) error { return ioctl(fd, unix.TCFLSH, selector) } // Tiocinq returns the number of bytes in the input buffer. func Tiocinq(fd uintptr) (int, error) { return unix.IoctlGetInt(int(fd), unix.FIORDCHK) } // Cfgetispeed returns the input baud rate stored in the termios structure. func Cfgetispeed(attr *unix.Termios) uint32 { solTermios := tiosToUnix(attr) return uint32(C.cfgetispeed((*C.termios_t)(unsafe.Pointer(solTermios)))) } // Cfsetispeed sets the input baud rate stored in the termios structure. func Cfsetispeed(attr *unix.Termios, speed uintptr) error { solTermios := tiosToUnix(attr) _, err := C.cfsetispeed((*C.termios_t)(unsafe.Pointer(solTermios)), C.speed_t(speed)) return err } // Cfgetospeed returns the output baud rate stored in the termios structure. func Cfgetospeed(attr *unix.Termios) uint32 { solTermios := tiosToUnix(attr) return uint32(C.cfgetospeed((*C.termios_t)(unsafe.Pointer(solTermios)))) } // Cfsetospeed sets the output baud rate stored in the termios structure. func Cfsetospeed(attr *unix.Termios, speed uintptr) error { solTermios := tiosToUnix(attr) _, err := C.cfsetospeed((*C.termios_t)(unsafe.Pointer(solTermios)), C.speed_t(speed)) return err } // tiosToUnix copies a unix.Termios to a x/sys/unix.Termios. // This is needed since type conversions between the two fail due to // more recent x/sys/unix.Termios renaming the padding field. func tiosToUnix(st *unix.Termios) *unix.Termios { return &unix.Termios{ Iflag: st.Iflag, Oflag: st.Oflag, Cflag: st.Cflag, Lflag: st.Lflag, Cc: st.Cc, } } // tiosTounix copies a x/sys/unix.Termios to a unix.Termios. func tiosTounix(ut *unix.Termios) *unix.Termios { return &unix.Termios{ Iflag: ut.Iflag, Oflag: ut.Oflag, Cflag: ut.Cflag, Lflag: ut.Lflag, Cc: ut.Cc, } }