bcom.c 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460
  1. #include "u.h"
  2. #include "lib.h"
  3. #include "mem.h"
  4. #include "dat.h"
  5. #include "fns.h"
  6. #include "io.h"
  7. #include "fs.h"
  8. Type types[] = {
  9. { Tfloppy,
  10. Fini|Ffs,
  11. floppyinit, floppyinitdev,
  12. floppygetfspart, 0, floppyboot,
  13. },
  14. { Tsd,
  15. Fini|Ffs,
  16. sdinit, sdinitdev,
  17. sdgetfspart, sdaddconf, sdboot,
  18. },
  19. { Tnil,
  20. 0,
  21. 0, 0,
  22. 0, 0, 0,
  23. },
  24. };
  25. #include "sd.h"
  26. extern SDifc sdataifc;
  27. extern SDifc sdmylexifc;
  28. extern SDifc sd53c8xxifc;
  29. SDifc* sdifc[] = {
  30. &sdataifc,
  31. // &sdmylexifc,
  32. // &sd53c8xxifc,
  33. nil,
  34. };
  35. typedef struct Mode Mode;
  36. enum {
  37. Maxdev = 7,
  38. Dany = -1,
  39. Nmedia = 16,
  40. Nini = 10,
  41. };
  42. enum { /* mode */
  43. Mauto = 0x00,
  44. Mlocal = 0x01,
  45. Manual = 0x02,
  46. NMode = 0x03,
  47. };
  48. typedef struct Medium Medium;
  49. struct Medium {
  50. Type* type;
  51. int flag;
  52. int dev;
  53. char name[NAMELEN];
  54. Fs* inifs;
  55. Medium* next;
  56. };
  57. typedef struct Mode {
  58. char* name;
  59. int mode;
  60. } Mode;
  61. static Medium media[Nmedia];
  62. static Medium *curmedium = media;
  63. static Mode modes[NMode+1] = {
  64. [Mauto] { "auto", Mauto, },
  65. [Mlocal] { "local", Mlocal, },
  66. [Manual] { "manual", Manual, },
  67. };
  68. char *defaultpartition = "new";
  69. static Medium*
  70. parse(char *line, char **file)
  71. {
  72. char *p;
  73. Type *tp;
  74. Medium *mp;
  75. if(p = strchr(line, '!')) {
  76. *p++ = 0;
  77. *file = p;
  78. } else
  79. *file = "";
  80. for(tp = types; tp->type != Tnil; tp++)
  81. for(mp = tp->media; mp; mp = mp->next)
  82. if(strcmp(mp->name, line) == 0)
  83. return mp;
  84. return nil;
  85. }
  86. static int
  87. boot(Medium *mp, char *file)
  88. {
  89. static Boot b;
  90. memset(&b, 0, sizeof b);
  91. b.state = INIT9LOAD;
  92. // sprint(BOOTLINE, "%s!%s", mp->name, file);
  93. return (*mp->type->boot)(mp->dev, file, &b);
  94. }
  95. static Medium*
  96. allocm(Type *tp)
  97. {
  98. Medium **l;
  99. if(curmedium >= &media[Nmedia])
  100. return 0;
  101. for(l = &tp->media; *l; l = &(*l)->next)
  102. ;
  103. *l = curmedium++;
  104. return *l;
  105. }
  106. char *parts[] = { "dos", "9fat", "fs", 0 };
  107. Medium*
  108. probe(int type, int flag, int dev)
  109. {
  110. Type *tp;
  111. int i;
  112. Medium *mp;
  113. for(tp = types; tp->type != Tnil; tp++){
  114. if(type != Tany && type != tp->type)
  115. continue;
  116. if(flag != Fnone){
  117. for(mp = tp->media; mp; mp = mp->next){
  118. if((flag & mp->flag) && (dev == Dany || dev == mp->dev))
  119. return mp;
  120. }
  121. }
  122. if((tp->flag & Fprobe) == 0){
  123. tp->flag |= Fprobe;
  124. tp->mask = (*tp->init)();
  125. }
  126. for(i = 0; tp->mask; i++){
  127. if((tp->mask & (1<<i)) == 0)
  128. continue;
  129. tp->mask &= ~(1<<i);
  130. if((mp = allocm(tp)) == 0)
  131. continue;
  132. mp->dev = i;
  133. mp->flag = tp->flag;
  134. mp->type = tp;
  135. (*tp->initdev)(i, mp->name);
  136. if((flag & mp->flag) && (dev == Dany || dev == i))
  137. return mp;
  138. }
  139. }
  140. return 0;
  141. }
  142. extern int loopconst;
  143. void
  144. main(void)
  145. {
  146. Medium *mp;
  147. int flag;
  148. char def[2*NAMELEN], line[80], *p, *file;
  149. Type *tp;
  150. i8042a20();
  151. memset(m, 0, sizeof(Mach));
  152. trapinit();
  153. clockinit();
  154. alarminit();
  155. spllo();
  156. kbdinit();
  157. if((ulong)&end > (KZERO|(640*1024)))
  158. panic("i'm too big");
  159. /*
  160. * If there were any arguments, MS-DOS leaves a character
  161. * count followed by the arguments in the runtime header.
  162. * Step over the leading space.
  163. */
  164. p = (char*)0x80080080;
  165. if(p[0]){
  166. p[p[0]+1] = 0;
  167. p += 2;
  168. }
  169. else
  170. p = 0;
  171. /*
  172. * Advance command line to first option, if any
  173. */
  174. if(p) {
  175. while(*p==' ' || *p=='\t')
  176. p++;
  177. if(*p == 0)
  178. p = nil;
  179. }
  180. /*
  181. * Probe everything, to collect device names.
  182. */
  183. probe(Tany, Fnone, Dany);
  184. if(p != 0) {
  185. if((mp = parse(p, &file)) == nil) {
  186. print("bad loadfile syntax: %s\n", p);
  187. goto done;
  188. }
  189. boot(mp, file);
  190. }
  191. done:
  192. flag = 0;
  193. for(tp = types; tp->type != Tnil; tp++){
  194. for(mp = tp->media; mp; mp = mp->next){
  195. if(flag == 0){
  196. flag = 1;
  197. print("Load devices:");
  198. }
  199. print(" %s", mp->name);
  200. }
  201. }
  202. if(flag)
  203. print("\n");
  204. for(;;){
  205. if(getstr("load from", line, sizeof(line), nil, 0) >= 0)
  206. if(mp = parse(line, &file))
  207. boot(mp, file);
  208. def[0] = 0;
  209. }
  210. }
  211. int
  212. getfields(char *lp, char **fields, int n, char sep)
  213. {
  214. int i;
  215. for(i = 0; lp && *lp && i < n; i++){
  216. while(*lp == sep)
  217. *lp++ = 0;
  218. if(*lp == 0)
  219. break;
  220. fields[i] = lp;
  221. while(*lp && *lp != sep){
  222. if(*lp == '\\' && *(lp+1) == '\n')
  223. *lp++ = ' ';
  224. lp++;
  225. }
  226. }
  227. return i;
  228. }
  229. int
  230. cistrcmp(char *a, char *b)
  231. {
  232. int ac, bc;
  233. for(;;){
  234. ac = *a++;
  235. bc = *b++;
  236. if(ac >= 'A' && ac <= 'Z')
  237. ac = 'a' + (ac - 'A');
  238. if(bc >= 'A' && bc <= 'Z')
  239. bc = 'a' + (bc - 'A');
  240. ac -= bc;
  241. if(ac)
  242. return ac;
  243. if(bc == 0)
  244. break;
  245. }
  246. return 0;
  247. }
  248. int
  249. cistrncmp(char *a, char *b, int n)
  250. {
  251. unsigned ac, bc;
  252. while(n > 0){
  253. ac = *a++;
  254. bc = *b++;
  255. n--;
  256. if(ac >= 'A' && ac <= 'Z')
  257. ac = 'a' + (ac - 'A');
  258. if(bc >= 'A' && bc <= 'Z')
  259. bc = 'a' + (bc - 'A');
  260. ac -= bc;
  261. if(ac)
  262. return ac;
  263. if(bc == 0)
  264. break;
  265. }
  266. return 0;
  267. }
  268. void*
  269. ialloc(ulong n, int align)
  270. {
  271. static ulong palloc;
  272. ulong p;
  273. int a;
  274. if(palloc == 0)
  275. palloc = 3*1024*1024;
  276. p = palloc;
  277. if(align <= 0)
  278. align = 4;
  279. if(a = n % align)
  280. n += align - a;
  281. if(a = p % align)
  282. p += align - a;
  283. palloc = p+n;
  284. return memset((void*)(p|KZERO), 0, n);
  285. }
  286. void*
  287. xspanalloc(ulong size, int align, ulong span)
  288. {
  289. ulong a, v;
  290. a = (ulong)ialloc(size+align+span, 0);
  291. if(span > 2)
  292. v = (a + span) & ~(span-1);
  293. else
  294. v = a;
  295. if(align > 1)
  296. v = (v + align) & ~(align-1);
  297. return (void*)v;
  298. }
  299. static Block *allocbp;
  300. Block*
  301. allocb(int size)
  302. {
  303. Block *bp, **lbp;
  304. ulong addr;
  305. lbp = &allocbp;
  306. for(bp = *lbp; bp; bp = bp->next){
  307. if((bp->lim - bp->base) >= size){
  308. *lbp = bp->next;
  309. break;
  310. }
  311. lbp = &bp->next;
  312. }
  313. if(bp == 0){
  314. bp = ialloc(sizeof(Block)+size+64, 0);
  315. addr = (ulong)bp;
  316. addr = ROUNDUP(addr + sizeof(Block), 8);
  317. bp->base = (uchar*)addr;
  318. bp->lim = ((uchar*)bp) + sizeof(Block)+size+64;
  319. }
  320. if(bp->flag)
  321. panic("allocb reuse\n");
  322. bp->rp = bp->base;
  323. bp->wp = bp->rp;
  324. bp->next = 0;
  325. bp->flag = 1;
  326. return bp;
  327. }
  328. void
  329. freeb(Block* bp)
  330. {
  331. bp->next = allocbp;
  332. allocbp = bp;
  333. bp->flag = 0;
  334. }
  335. enum {
  336. Paddr= 0x70, /* address port */
  337. Pdata= 0x71, /* data port */
  338. };
  339. uchar
  340. nvramread(int offset)
  341. {
  342. outb(Paddr, offset);
  343. return inb(Pdata);
  344. }
  345. void (*etherdetach)(void);
  346. void (*floppydetach)(void);
  347. void (*sddetach)(void);
  348. void
  349. warp9(ulong entry)
  350. {
  351. if(etherdetach)
  352. etherdetach();
  353. consdrain();
  354. (*(void(*)(void))(PADDR(entry)))();
  355. }
  356. char*
  357. getconf(char*)
  358. {
  359. return nil;
  360. }
  361. void
  362. addconf(char*, ...)
  363. {
  364. }
  365. void
  366. uartspecial(int, void(*)(int), int(*)(void), int)
  367. {
  368. }
  369. void
  370. uartputs(IOQ*, char*, int)
  371. {
  372. }
  373. void
  374. uartputc(int)
  375. {}
  376. void
  377. uartdrain(void)
  378. {
  379. }