github.com/u-root/u-root@v7.0.1-0.20200915234505-ad7babab0a8e+incompatible/cmds/exp/console/console.go (about)

     1  // Copyright 2015-2020 the u-root Authors. All rights reserved
     2  // Use of this source code is governed by a BSD-style
     3  // license that can be found in the LICENSE file.
     4  
     5  // console implements a basic console. It establishes a pair of files
     6  // to read from, the default being a UART at 0x3f8, but an alternative
     7  // being just stdin and stdout. It will also set up a root file system
     8  // using util.Rootfs, although this can be disabled as well.
     9  // Console uses a Go version of fork_pty to start up a shell, default
    10  // sh. Console runs until the shell exits and then exits itself.
    11  package main
    12  
    13  import (
    14  	"flag"
    15  	"fmt"
    16  	"io"
    17  	"log"
    18  	"os"
    19  
    20  	"github.com/u-root/u-root/pkg/libinit"
    21  	"github.com/u-root/u-root/pkg/pty"
    22  )
    23  
    24  var (
    25  	serial    = flag.String("serial", "0x3f8", "which IO device: stdio, i8042, or serial port starting with 0")
    26  	setupRoot = flag.Bool("setuproot", false, "Set up a root file system")
    27  )
    28  
    29  func main() {
    30  	fmt.Printf("console -- starting")
    31  	flag.Parse()
    32  
    33  	a := flag.Args()
    34  	if len(a) == 0 {
    35  		a = []string{"/bin/sh"}
    36  	}
    37  
    38  	p, err := pty.New()
    39  	if err != nil {
    40  		log.Fatalf("Console exits: can't open pty: %v", err)
    41  	}
    42  	p.Command(a[0], a[1:]...)
    43  	// Make a good faith effort to set up root. This being
    44  	// a kind of init program, we do our best and keep going.
    45  	if *setupRoot {
    46  		libinit.SetEnv()
    47  		libinit.CreateRootfs()
    48  	}
    49  
    50  	in, out, err := console(*serial)
    51  
    52  	if err != nil {
    53  		log.Fatal(err)
    54  	}
    55  
    56  	err = p.Start()
    57  	if err != nil {
    58  		log.Fatalf("Console exits: can't start %v: %v", a, err)
    59  	}
    60  	kid := p.C.Process.Pid
    61  
    62  	// You need the \r\n as we are now in raw mode!
    63  	fmt.Printf("Started %d\r\n", kid)
    64  
    65  	go io.Copy(out, p.Ptm)
    66  
    67  	go func() {
    68  		var data = make([]byte, 1)
    69  		for {
    70  			if _, err := in.Read(data); err != nil {
    71  				fmt.Printf("kid stdin: done\n")
    72  			}
    73  			if data[0] == '\r' {
    74  				if _, err := out.Write(data); err != nil {
    75  					log.Printf("error on echo %v: %v", data, err)
    76  				}
    77  				data[0] = '\n'
    78  			}
    79  			if _, err := p.Ptm.Write(data); err != nil {
    80  				log.Printf("Error writing input to ptm: %v: give up\n", err)
    81  				break
    82  			}
    83  		}
    84  	}()
    85  
    86  	if err := p.Wait(); err != nil {
    87  		log.Fatalf("Console exits: %v", err)
    88  	}
    89  	log.Printf("Console all done")
    90  	os.Exit(0)
    91  }