ref: 4aae319f76dd7157149aad6a67c3ca9b03a7f30a
dir: /sys/src/boot/pc/sub.c/
#include <u.h> #include <a.out.h> #include "fns.h" #include "mem.h" void memset(void *dst, int v, int n) { uchar *d = dst; while(n > 0){ *d++ = v; n--; } } void memmove(void *dst, void *src, int n) { uchar *d = dst; uchar *s = src; if(d < s){ while(n-- > 0) *d++ = *s++; } else if(d > s){ s += n; d += n; while(n-- > 0) *--d = *--s; } } int memcmp(void *src, void *dst, int n) { uchar *d = dst; uchar *s = src; int r = 0; while(n-- > 0) if(r = (*d++ - *s++)) break; return r; } int strlen(char *s) { char *p = s; while(*p) p++; return p - s; } char* strchr(char *s, int c) { for(; *s; s++) if(*s == c) return s; return 0; } void print(char *s) { while(*s) putc(*s++); } int readn(void *f, void *data, int len) { uchar *p, *e; putc(' '); p = data; e = p + len; while(p < e){ if(((ulong)p & 0xF000) == 0){ putc('\b'); putc(hex[((ulong)p>>16)&0xF]); } if((len = read(f, p, e - p)) <= 0) break; p += len; } putc('\b'); return p - (uchar*)data; } static int readline(void *f, char buf[64]) { static char white[] = "\t "; char *p; p = buf; do{ if(!f) putc('>'); for(;;){ if(!f){ putc(*p = getc()); if(*p == '\r') putc('\n'); else if(*p == '\b' && p > buf){ p--; continue; } }else if(read(f, p, 1) <= 0) return 0; if(strchr(crnl, *p)) break; if(p == buf && strchr(white, *p)) continue; /* whitespace on start of line */ if(p >= buf + 64-1){ if(!f){ putc('\b'); putc(' '); putc('\b'); } continue; /* line full do not advance */ } p++; } while(p > buf && strchr(white, p[-1])) p--; }while(p == buf); *p = 0; return p - buf; } static int timeout(int ms) { while(ms > 0){ if(gotc()) return 1; usleep(100000); ms -= 100; } return 0; } #define BOOTLINE ((char*)CONFADDR) #define BOOTLINELEN 64 #define BOOTARGS ((char*)(CONFADDR+BOOTLINELEN)) #define BOOTARGSLEN (4096-0x200-BOOTLINELEN) char *confend; static void apmconf(int); static void e820conf(void); static char* getconf(char *s, char *buf) { char *p, *e; int n; n = strlen(s); for(p = BOOTARGS; p < confend; p = e+1){ for(e = p+1; e < confend; e++) if(*e == '\n') break; if(!memcmp(p, s, n)){ p += n; n = e - p; buf[n] = 0; memmove(buf, p, n); return buf; } } return 0; } static int delconf(char *s) { char *p, *e; for(p = BOOTARGS; p < confend; p = e){ for(e = p+1; e < confend; e++){ if(*e == '\n'){ e++; break; } } if(!memcmp(p, s, strlen(s))){ memmove(p, e, confend - e); confend -= e - p; *confend = 0; return 1; } } return 0; } char* configure(void *f, char *path) { char line[64], *kern, *s, *p; int inblock, nowait, n; static int once = 1; if(once){ once = 0; Clear: memset(BOOTLINE, 0, BOOTLINELEN); confend = BOOTARGS; memset(confend, 0, BOOTARGSLEN); e820conf(); } nowait = 1; inblock = 0; Loop: while(readline(f, line) > 0){ if(*line == 0 || strchr("#;=", *line)) continue; if(*line == '['){ inblock = memcmp("[common]", line, 8); continue; } if(!memcmp("boot", line, 5)){ nowait=1; break; } if(!memcmp("wait", line, 5)){ nowait=0; continue; } if(!memcmp("show", line, 5)){ for(p = BOOTARGS; p < confend; p++){ if(*p == '\n') print(crnl); else putc(*p); } continue; } if(!memcmp("clear", line, 5)){ if(line[5] == 0){ print("ok"); print(crnl); goto Clear; } else if(line[5] == ' ' && delconf(line+6)){ print("ok"); print(crnl); } continue; } if(inblock || (p = strchr(line, '=')) == nil) continue; *p++ = 0; delconf(line); if(!memcmp("apm", line, 3)){ apmconf('0' - line[3]); continue; } s = confend; memmove(confend, line, n = strlen(line)); confend += n; *confend++ = '='; memmove(confend, p, n = strlen(p)); confend += n; *confend = 0; print(s); print(crnl); *confend++ = '\n'; *confend = 0; } kern = getconf("bootfile=", path); if(f){ close(f); f = 0; if(kern && (nowait==0 || timeout(1000))) goto Loop; } if(!kern){ print("no bootfile\r\n"); goto Loop; } while(p = strchr(kern, '!')) kern = p+1; return kern; } static void hexfmt(char *s, int i, ulong a) { s += i; while(i > 0){ *--s = hex[a&15]; a >>= 4; i--; } } static void addconfx(char *s, int w, ulong v) { int n; n = strlen(s); memmove(confend, s, n); hexfmt(confend+n, w, v); confend += n+w; *confend = 0; } static void apmconf(int id) { uchar *a; char *s; a = (uchar*)CONFADDR; memset(a, 0, 20); apm(id); if(memcmp(a, "APM", 4)) return; s = confend; addconfx("apm", 1, id); addconfx("=ax=", 4, *((ushort*)(a+4))); addconfx(" ebx=", 8, *((ulong*)(a+12))); addconfx(" cx=", 4, *((ushort*)(a+6))); addconfx(" dx=", 4, *((ushort*)(a+8))); addconfx(" di=", 4, *((ushort*)(a+10))); addconfx(" esi=", 8, *((ulong*)(a+16))); print(s); print(crnl); *confend++ = '\n'; *confend = 0; } static void e820conf(void) { struct { uvlong base; uvlong len; ulong typ; ulong ext; } e; uvlong v; ulong bx; char *s; bx=0; s = confend; do{ bx = e820(bx, &e); if(e.len != 0 && (e.ext & 3) == 1){ if(confend == s){ /* single entry <= 1MB is useless */ if(bx == 0 && e.typ == 1 && e.len <= 0x100000) break; memmove(confend, "*e820=", 6); confend += 6; } addconfx("", 1, e.typ); v = e.base; addconfx(" 0x", 8, v>>32); addconfx("", 8, v&0xffffffff); v += e.len; addconfx(" 0x", 8, v>>32); addconfx("", 8, v&0xffffffff); *confend++ = ' '; } } while(bx); if(confend == s) return; *confend = 0; print(s); print(crnl); *confend++ = '\n'; *confend = 0; } static ulong beswal(ulong l) { uchar *p = (uchar*)&l; return (p[0]<<24) | (p[1]<<16) | (p[2]<<8) | p[3]; } char* bootkern(void *f) { uchar *e, *d, *t; ulong n; Exec ex; while(a20() < 0) print("a20 enable failed\r\n"); if(readn(f, &ex, sizeof(ex)) != sizeof(ex)) return "bad header"; e = (uchar*)(beswal(ex.entry) & ~0xF0000000UL); switch(beswal(ex.magic)){ case S_MAGIC: if(readn(f, e, 8) != 8) goto Error; case I_MAGIC: break; default: return "bad magic"; } t = e; n = beswal(ex.text); if(readn(f, t, n) != n) goto Error; t += n; d = (uchar*)PGROUND((ulong)t); memset(t, 0, d - t); n = beswal(ex.data); if(readn(f, d, n) != n) goto Error; d += n; t = (uchar*)PGROUND((ulong)d); t += PGROUND(beswal(ex.bss)); memset(d, 0, t - d); close(f); unload(); print("boot"); print(crnl); jump(e); Error: return "i/o error"; }