#include #include #include #include #include "../boot/boot.h" #define PARTSRV "partfs.sdXX" enum { Dontpost, Post, }; char cputype[64]; char sys[2*64]; char reply[256]; int printcol; int mflag; int fflag; int kflag; int debugboot; int nousbboot; char *bargv[Nbarg]; int bargc; static void swapproc(void); static Method *rootserver(char*); static void kbmap(void); /* * we should inherit the standard fds all referring to /dev/cons, * but we're being paranoid. */ static void opencons(void) { close(0); close(1); close(2); bind("#c", "/dev", MBEFORE); open("/dev/cons", OREAD); open("/dev/cons", OWRITE); open("/dev/cons", OWRITE); } /* * init will reinitialize its namespace. * #ec gets us plan9.ini settings (*var variables). */ static void bindenvsrv(void) { bind("#ec", "/env", MREPL); bind("#e", "/env", MBEFORE|MCREATE); bind("#s", "/srv/", MREPL|MCREATE); } static void debuginit(int argc, char **argv) { int fd; if(getenv("debugboot")) debugboot = 1; if(getenv("nousbboot")) nousbboot = 1; #ifdef DEBUG print("argc=%d\n", argc); for(fd = 0; fd < argc; fd++) print("%#p %s ", argv[fd], argv[fd]); print("\n"); #endif /* DEBUG */ SET(fd); USED(argc, argv, fd); } /* * read disk partition tables here so that readnvram via factotum * can see them. ideally we would have this information in * environment variables before attaching #S, which would then * parse them and create partitions. */ static void partinit(void) { char *rdparts; rdparts = getenv("readparts"); if(rdparts) readparts(); free(rdparts); } /* * pick a method and initialize it */ static Method * pickmethod(int argc, char **argv) { Method *mp; if(method[0].name == nil) fatal("no boot methods"); mp = rootserver(argc ? *argv : 0); (*mp->config)(mp); return mp; } /* * authentication agent * sets hostowner, creating an auth discontinuity */ static void doauth(int cpuflag) { dprint("auth..."); authentication(cpuflag); } /* * connect to the root file system */ static int connectroot(Method *mp, int islocal, int ishybrid) { int fd, n; char buf[32]; fd = (*mp->connect)(); if(fd < 0) fatal("can't connect to file server"); if(getenv("srvold9p")) fd = old9p(fd); if(!islocal && !ishybrid){ if(cfs) fd = (*cfs)(fd); } print("version..."); buf[0] = '\0'; n = fversion(fd, 0, buf, sizeof buf); if(n < 0) fatal("can't init 9P"); srvcreate("boot", fd); return fd; } /* * create the name space, mount the root fs */ static int nsinit(int fd, char **rspp) { int afd; char *rp, *rsp; AuthInfo *ai; static char rootbuf[64]; if(bind("/", "/", MREPL) < 0) fatal("bind /"); rp = getenv("rootspec"); if(rp == nil) rp = ""; afd = fauth(fd, rp); if(afd >= 0){ ai = auth_proxy(afd, auth_getkey, "proto=p9any role=client"); if(ai == nil) print("authentication failed (%r), trying mount anyways\n"); } if(mount(fd, afd, "/root", MREPL|MCREATE, rp) < 0) fatal("mount /"); rsp = rp; rp = getenv("rootdir"); if(rp == nil) rp = rootdir; if(bind(rp, "/", MAFTER|MCREATE) < 0){ if(strncmp(rp, "/root", 5) == 0){ fprint(2, "boot: couldn't bind $rootdir=%s to root: %r\n", rp); fatal("second bind /"); } snprint(rootbuf, sizeof rootbuf, "/root/%s", rp); rp = rootbuf; if(bind(rp, "/", MAFTER|MCREATE) < 0){ fprint(2, "boot: couldn't bind $rootdir=%s to root: %r\n", rp); if(strcmp(rootbuf, "/root//plan9") != 0) fatal("second bind /"); /* undo installer's work */ fprint(2, "**** warning: remove rootdir=/plan9 " "entry from plan9.ini\n"); rp = "/root"; if(bind(rp, "/", MAFTER|MCREATE) < 0) fatal("second bind /"); } } setenv("rootdir", rp); *rspp = rsp; return afd; } static void execinit(void) { int iargc; char *cmd, cmdbuf[64], *iargv[16]; /* exec init */ cmd = getenv("init"); if(cmd == nil){ sprint(cmdbuf, "/%s/init -%s%s", cputype, cpuflag ? "c" : "t", mflag ? "m" : ""); cmd = cmdbuf; } iargc = tokenize(cmd, iargv, nelem(iargv)-1); cmd = iargv[0]; /* make iargv[0] basename(iargv[0]) */ if(iargv[0] = strrchr(iargv[0], '/')) iargv[0]++; else iargv[0] = cmd; iargv[iargc] = nil; chmod("/srv/" PARTSRV, 0600); exec(cmd, iargv); fatal(cmd); } void boot(int argc, char *argv[]) { int fd, afd, islocal, ishybrid; char *rsp; Method *mp; fmtinstall('r', errfmt); opencons(); bindenvsrv(); debuginit(argc, argv); ARGBEGIN{ case 'k': kflag = 1; break; case 'm': mflag = 1; break; case 'f': fflag = 1; break; }ARGEND readfile("#e/cputype", cputype, sizeof(cputype)); /* * set up usb keyboard & mouse, if any. * starts partfs on first disk, if any, to permit nvram on usb. */ if (!nousbboot) usbinit(Dontpost); if(getenv("pitft")) { fd = open("#P/pitft", ORDWR); fprint(fd, "init"); close(fd); } dprint("pickmethod..."); mp = pickmethod(argc, argv); islocal = strcmp(mp->name, "local") == 0; ishybrid = strcmp(mp->name, "hybrid") == 0; kbmap(); /* load keymap if it's there. */ /* don't trigger aoe until the network has been configured */ dprint("bind #æ..."); bind("#æ", "/dev", MAFTER); /* nvram could be here */ dprint("bind #S..."); bind("#S", "/dev", MAFTER); /* nvram could be here */ dprint("partinit..."); partinit(); doauth(cpuflag); /* authentication usually changes hostowner */ rfork(RFNAMEG); /* leave existing subprocs in own namespace */ if (!nousbboot) usbinit(Post); /* restart partfs under the new hostowner id */ fd = connectroot(mp, islocal, ishybrid); afd = nsinit(fd, &rsp); close(fd); settime(islocal, afd, rsp); if(afd > 0) close(afd); swapproc(); execinit(); exits("failed to exec init"); } static Method* findmethod(char *a) { Method *mp; int i, j; char *cp; if((i = strlen(a)) == 0) return nil; cp = strchr(a, '!'); if(cp) i = cp - a; for(mp = method; mp->name; mp++){ j = strlen(mp->name); if(j > i) j = i; if(strncmp(a, mp->name, j) == 0) break; } if(mp->name) return mp; return nil; } /* * ask user from whence cometh the root file system */ static Method* rootserver(char *arg) { char prompt[256]; Method *mp; char *cp; int n; /* look for required reply */ dprint("read #e/nobootprompt..."); readfile("#e/nobootprompt", reply, sizeof(reply)); if(reply[0]){ mp = findmethod(reply); if(mp) goto HaveMethod; print("boot method %s not found\n", reply); reply[0] = 0; } /* make list of methods */ mp = method; n = sprint(prompt, "root is from (%s", mp->name); for(mp++; mp->name; mp++) n += sprint(prompt+n, ", %s", mp->name); sprint(prompt+n, ")"); /* create default reply */ dprint("read #e/bootargs..."); readfile("#e/bootargs", reply, sizeof(reply)); if(reply[0] == 0 && arg != 0) strcpy(reply, arg); if(reply[0]){ mp = findmethod(reply); if(mp == 0) reply[0] = 0; } if(reply[0] == 0) strcpy(reply, method->name); /* parse replies */ do{ dprint("outin..."); outin(prompt, reply, sizeof(reply)); mp = findmethod(reply); }while(mp == nil); HaveMethod: bargc = tokenize(reply, bargv, Nbarg-2); bargv[bargc] = nil; cp = strchr(reply, '!'); if(cp) strcpy(sys, cp+1); dprint("pickmethod done\n"); return mp; } static void swapproc(void) { int fd; fd = open("#c/swap", OWRITE); if(fd < 0){ warning("opening #c/swap"); return; } if(write(fd, "start", 5) <= 0) warning("starting swap kproc"); close(fd); } int old9p(int fd) { int p[2]; if(pipe(p) < 0) fatal("pipe"); print("srvold9p..."); switch(fork()) { case -1: fatal("rfork srvold9p"); case 0: dup(fd, 1); close(fd); dup(p[0], 0); close(p[0]); close(p[1]); execl("/srvold9p", "srvold9p", "-s", 0); fatal("exec srvold9p"); default: close(fd); close(p[0]); } return p[1]; } static void kbmap(void) { char *f; int n, in, out; char buf[1024]; f = getenv("kbmap"); if(f == nil) return; if(bind("#κ", "/dev", MAFTER) < 0){ warning("can't bind #κ"); return; } in = open(f, OREAD); if(in < 0){ warning("can't open kbd map"); return; } out = open("/dev/kbmap", OWRITE); if(out < 0) { warning("can't open /dev/kbmap"); close(in); return; } while((n = read(in, buf, sizeof(buf))) > 0) if(write(out, buf, n) != n){ warning("write to /dev/kbmap failed"); break; } close(in); close(out); }