2015-02-10 06:07:18 +08:00
|
|
|
package libcontainer
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"unsafe"
|
2017-03-28 05:46:57 +08:00
|
|
|
|
|
|
|
"golang.org/x/sys/unix"
|
2015-02-10 06:07:18 +08:00
|
|
|
)
|
|
|
|
|
2017-03-03 04:53:06 +08:00
|
|
|
func ConsoleFromFile(f *os.File) Console {
|
|
|
|
return &linuxConsole{
|
|
|
|
master: f,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-03 23:29:34 +08:00
|
|
|
// newConsole returns an initialized console that can be used within a container by copying bytes
|
2015-02-12 09:12:03 +08:00
|
|
|
// from the master side to the slave that is attached as the tty for the container's init process.
|
2016-09-14 17:02:53 +08:00
|
|
|
func newConsole() (Console, error) {
|
2017-03-28 05:46:57 +08:00
|
|
|
master, err := os.OpenFile("/dev/ptmx", unix.O_RDWR|unix.O_NOCTTY|unix.O_CLOEXEC, 0)
|
2015-02-10 06:07:18 +08:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
console, err := ptsname(master)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if err := unlockpt(master); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &linuxConsole{
|
|
|
|
slavePath: console,
|
|
|
|
master: master,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2016-10-12 07:22:48 +08:00
|
|
|
// linuxConsole is a linux pseudo TTY for use within a container.
|
2015-02-10 06:07:18 +08:00
|
|
|
type linuxConsole struct {
|
|
|
|
master *os.File
|
|
|
|
slavePath string
|
|
|
|
}
|
|
|
|
|
2016-09-03 01:31:54 +08:00
|
|
|
func (c *linuxConsole) File() *os.File {
|
|
|
|
return c.master
|
2015-02-10 06:07:18 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *linuxConsole) Path() string {
|
|
|
|
return c.slavePath
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *linuxConsole) Read(b []byte) (int, error) {
|
|
|
|
return c.master.Read(b)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *linuxConsole) Write(b []byte) (int, error) {
|
|
|
|
return c.master.Write(b)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *linuxConsole) Close() error {
|
|
|
|
if m := c.master; m != nil {
|
|
|
|
return m.Close()
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// mount initializes the console inside the rootfs mounting with the specified mount label
|
|
|
|
// and applying the correct ownership of the console.
|
2016-09-14 06:09:09 +08:00
|
|
|
func (c *linuxConsole) mount() error {
|
2017-03-28 05:46:57 +08:00
|
|
|
oldMask := unix.Umask(0000)
|
|
|
|
defer unix.Umask(oldMask)
|
2016-06-03 23:29:34 +08:00
|
|
|
f, err := os.Create("/dev/console")
|
2015-02-10 06:07:18 +08:00
|
|
|
if err != nil && !os.IsExist(err) {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if f != nil {
|
|
|
|
f.Close()
|
|
|
|
}
|
2017-03-28 05:46:57 +08:00
|
|
|
return unix.Mount(c.slavePath, "/dev/console", "bind", unix.MS_BIND, "")
|
2015-02-10 06:07:18 +08:00
|
|
|
}
|
|
|
|
|
2015-06-10 06:19:47 +08:00
|
|
|
// dupStdio opens the slavePath for the console and dups the fds to the current
|
2015-02-10 06:07:18 +08:00
|
|
|
// processes stdio, fd 0,1,2.
|
|
|
|
func (c *linuxConsole) dupStdio() error {
|
2017-03-28 05:46:57 +08:00
|
|
|
slave, err := c.open(unix.O_RDWR)
|
2015-02-10 06:07:18 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
fd := int(slave.Fd())
|
|
|
|
for _, i := range []int{0, 1, 2} {
|
2017-03-28 05:46:57 +08:00
|
|
|
if err := unix.Dup3(fd, i, 0); err != nil {
|
2015-02-10 06:07:18 +08:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// open is a clone of os.OpenFile without the O_CLOEXEC used to open the pty slave.
|
|
|
|
func (c *linuxConsole) open(flag int) (*os.File, error) {
|
2017-03-28 05:46:57 +08:00
|
|
|
r, e := unix.Open(c.slavePath, flag, 0)
|
2015-02-10 06:07:18 +08:00
|
|
|
if e != nil {
|
|
|
|
return nil, &os.PathError{
|
|
|
|
Op: "open",
|
|
|
|
Path: c.slavePath,
|
|
|
|
Err: e,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return os.NewFile(uintptr(r), c.slavePath), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func ioctl(fd uintptr, flag, data uintptr) error {
|
2017-03-28 05:46:57 +08:00
|
|
|
if _, _, err := unix.Syscall(unix.SYS_IOCTL, fd, flag, data); err != 0 {
|
2015-02-10 06:07:18 +08:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// unlockpt unlocks the slave pseudoterminal device corresponding to the master pseudoterminal referred to by f.
|
|
|
|
// unlockpt should be called before opening the slave side of a pty.
|
|
|
|
func unlockpt(f *os.File) error {
|
|
|
|
var u int32
|
2017-03-28 05:46:57 +08:00
|
|
|
return ioctl(f.Fd(), unix.TIOCSPTLCK, uintptr(unsafe.Pointer(&u)))
|
2015-02-10 06:07:18 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// ptsname retrieves the name of the first available pts for the given master.
|
|
|
|
func ptsname(f *os.File) (string, error) {
|
|
|
|
var n int32
|
2017-03-28 05:46:57 +08:00
|
|
|
if err := ioctl(f.Fd(), unix.TIOCGPTN, uintptr(unsafe.Pointer(&n))); err != nil {
|
2015-02-10 06:07:18 +08:00
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
return fmt.Sprintf("/dev/pts/%d", n), nil
|
|
|
|
}
|
2016-10-24 19:09:01 +08:00
|
|
|
|
2017-06-08 12:08:44 +08:00
|
|
|
// SaneTerminal sets the necessary tty_ioctl(4)s to ensure that a pty pair
|
2016-10-24 19:09:01 +08:00
|
|
|
// created by us acts normally. In particular, a not-very-well-known default of
|
|
|
|
// Linux unix98 ptys is that they have +onlcr by default. While this isn't a
|
|
|
|
// problem for terminal emulators, because we relay data from the terminal we
|
|
|
|
// also relay that funky line discipline.
|
2017-06-08 12:08:44 +08:00
|
|
|
func SaneTerminal(terminal *os.File) error {
|
2016-10-24 19:09:01 +08:00
|
|
|
// Go doesn't have a wrapper for any of the termios ioctls.
|
2017-03-28 05:46:57 +08:00
|
|
|
var termios unix.Termios
|
2016-10-24 19:09:01 +08:00
|
|
|
|
2017-03-28 05:46:57 +08:00
|
|
|
if err := ioctl(terminal.Fd(), unix.TCGETS, uintptr(unsafe.Pointer(&termios))); err != nil {
|
2016-10-24 19:09:01 +08:00
|
|
|
return fmt.Errorf("ioctl(tty, tcgets): %s", err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
// Set -onlcr so we don't have to deal with \r.
|
2017-03-28 05:46:57 +08:00
|
|
|
termios.Oflag &^= unix.ONLCR
|
2016-10-24 19:09:01 +08:00
|
|
|
|
2017-03-28 05:46:57 +08:00
|
|
|
if err := ioctl(terminal.Fd(), unix.TCSETS, uintptr(unsafe.Pointer(&termios))); err != nil {
|
2016-10-24 19:09:01 +08:00
|
|
|
return fmt.Errorf("ioctl(tty, tcsets): %s", err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|