sh.c 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494
  1. // Shell.
  2. #include "types.h"
  3. #include "user.h"
  4. #include "fcntl.h"
  5. // Parsed command representation
  6. #define EXEC 1
  7. #define REDIR 2
  8. #define PIPE 3
  9. #define LIST 4
  10. #define BACK 5
  11. #define MAXARGS 10
  12. struct cmd {
  13. int type;
  14. };
  15. struct execcmd {
  16. int type;
  17. char *argv[MAXARGS];
  18. char *eargv[MAXARGS];
  19. };
  20. struct redircmd {
  21. int type;
  22. struct cmd *cmd;
  23. char *file;
  24. char *efile;
  25. int mode;
  26. int fd;
  27. };
  28. struct pipecmd {
  29. int type;
  30. struct cmd *left;
  31. struct cmd *right;
  32. };
  33. struct listcmd {
  34. int type;
  35. struct cmd *left;
  36. struct cmd *right;
  37. };
  38. struct backcmd {
  39. int type;
  40. struct cmd *cmd;
  41. };
  42. int fork1(void); // Fork but panics on failure.
  43. void panic(char*);
  44. struct cmd *parsecmd(char*);
  45. // Execute cmd. Never returns.
  46. void
  47. runcmd(struct cmd *cmd)
  48. {
  49. int p[2];
  50. struct backcmd *bcmd;
  51. struct execcmd *ecmd;
  52. struct listcmd *lcmd;
  53. struct pipecmd *pcmd;
  54. struct redircmd *rcmd;
  55. if(cmd == 0)
  56. exit();
  57. switch(cmd->type){
  58. default:
  59. panic("runcmd");
  60. case EXEC:
  61. ecmd = (struct execcmd*)cmd;
  62. if(ecmd->argv[0] == 0)
  63. exit();
  64. exec(ecmd->argv[0], ecmd->argv);
  65. printf(2, "exec %s failed\n", ecmd->argv[0]);
  66. break;
  67. case REDIR:
  68. rcmd = (struct redircmd*)cmd;
  69. close(rcmd->fd);
  70. if(open(rcmd->file, rcmd->mode) < 0){
  71. printf(2, "open %s failed\n", rcmd->file);
  72. exit();
  73. }
  74. runcmd(rcmd->cmd);
  75. break;
  76. case LIST:
  77. lcmd = (struct listcmd*)cmd;
  78. if(fork1() == 0)
  79. runcmd(lcmd->left);
  80. wait();
  81. runcmd(lcmd->right);
  82. break;
  83. case PIPE:
  84. pcmd = (struct pipecmd*)cmd;
  85. if(pipe(p) < 0)
  86. panic("pipe");
  87. if(fork1() == 0){
  88. close(1);
  89. dup(p[1]);
  90. close(p[0]);
  91. close(p[1]);
  92. runcmd(pcmd->left);
  93. }
  94. if(fork1() == 0){
  95. close(0);
  96. dup(p[0]);
  97. close(p[0]);
  98. close(p[1]);
  99. runcmd(pcmd->right);
  100. }
  101. close(p[0]);
  102. close(p[1]);
  103. wait();
  104. wait();
  105. break;
  106. case BACK:
  107. bcmd = (struct backcmd*)cmd;
  108. if(fork1() == 0)
  109. runcmd(bcmd->cmd);
  110. break;
  111. }
  112. exit();
  113. }
  114. int
  115. getcmd(char *buf, int nbuf)
  116. {
  117. printf(2, "$ ");
  118. memset(buf, 0, nbuf);
  119. gets(buf, nbuf);
  120. if(buf[0] == 0) // EOF
  121. return -1;
  122. return 0;
  123. }
  124. int
  125. main(void)
  126. {
  127. static char buf[100];
  128. int fd;
  129. // Assumes three file descriptors open.
  130. while((fd = open("console", O_RDWR)) >= 0){
  131. if(fd >= 3){
  132. close(fd);
  133. break;
  134. }
  135. }
  136. // Read and run input commands.
  137. while(getcmd(buf, sizeof(buf)) >= 0){
  138. if(buf[0] == 'c' && buf[1] == 'd' && buf[2] == ' '){
  139. // Clumsy but will have to do for now.
  140. // Chdir has no effect on the parent if run in the child.
  141. buf[strlen(buf)-1] = 0; // chop \n
  142. if(chdir(buf+3) < 0)
  143. printf(2, "cannot cd %s\n", buf+3);
  144. continue;
  145. }
  146. if(fork1() == 0)
  147. runcmd(parsecmd(buf));
  148. wait();
  149. }
  150. exit();
  151. }
  152. void
  153. panic(char *s)
  154. {
  155. printf(2, "%s\n", s);
  156. exit();
  157. }
  158. int
  159. fork1(void)
  160. {
  161. int pid;
  162. pid = fork();
  163. if(pid == -1)
  164. panic("fork");
  165. return pid;
  166. }
  167. //PAGEBREAK!
  168. // Constructors
  169. struct cmd*
  170. execcmd(void)
  171. {
  172. struct execcmd *cmd;
  173. cmd = malloc(sizeof(*cmd));
  174. memset(cmd, 0, sizeof(*cmd));
  175. cmd->type = EXEC;
  176. return (struct cmd*)cmd;
  177. }
  178. struct cmd*
  179. redircmd(struct cmd *subcmd, char *file, char *efile, int mode, int fd)
  180. {
  181. struct redircmd *cmd;
  182. cmd = malloc(sizeof(*cmd));
  183. memset(cmd, 0, sizeof(*cmd));
  184. cmd->type = REDIR;
  185. cmd->cmd = subcmd;
  186. cmd->file = file;
  187. cmd->efile = efile;
  188. cmd->mode = mode;
  189. cmd->fd = fd;
  190. return (struct cmd*)cmd;
  191. }
  192. struct cmd*
  193. pipecmd(struct cmd *left, struct cmd *right)
  194. {
  195. struct pipecmd *cmd;
  196. cmd = malloc(sizeof(*cmd));
  197. memset(cmd, 0, sizeof(*cmd));
  198. cmd->type = PIPE;
  199. cmd->left = left;
  200. cmd->right = right;
  201. return (struct cmd*)cmd;
  202. }
  203. struct cmd*
  204. listcmd(struct cmd *left, struct cmd *right)
  205. {
  206. struct listcmd *cmd;
  207. cmd = malloc(sizeof(*cmd));
  208. memset(cmd, 0, sizeof(*cmd));
  209. cmd->type = LIST;
  210. cmd->left = left;
  211. cmd->right = right;
  212. return (struct cmd*)cmd;
  213. }
  214. struct cmd*
  215. backcmd(struct cmd *subcmd)
  216. {
  217. struct backcmd *cmd;
  218. cmd = malloc(sizeof(*cmd));
  219. memset(cmd, 0, sizeof(*cmd));
  220. cmd->type = BACK;
  221. cmd->cmd = subcmd;
  222. return (struct cmd*)cmd;
  223. }
  224. //PAGEBREAK!
  225. // Parsing
  226. char whitespace[] = " \t\r\n\v";
  227. char symbols[] = "<|>&;()";
  228. int
  229. gettoken(char **ps, char *es, char **q, char **eq)
  230. {
  231. char *s;
  232. int ret;
  233. s = *ps;
  234. while(s < es && strchr(whitespace, *s))
  235. s++;
  236. if(q)
  237. *q = s;
  238. ret = *s;
  239. switch(*s){
  240. case 0:
  241. break;
  242. case '|':
  243. case '(':
  244. case ')':
  245. case ';':
  246. case '&':
  247. case '<':
  248. s++;
  249. break;
  250. case '>':
  251. s++;
  252. if(*s == '>'){
  253. ret = '+';
  254. s++;
  255. }
  256. break;
  257. default:
  258. ret = 'a';
  259. while(s < es && !strchr(whitespace, *s) && !strchr(symbols, *s))
  260. s++;
  261. break;
  262. }
  263. if(eq)
  264. *eq = s;
  265. while(s < es && strchr(whitespace, *s))
  266. s++;
  267. *ps = s;
  268. return ret;
  269. }
  270. int
  271. peek(char **ps, char *es, char *toks)
  272. {
  273. char *s;
  274. s = *ps;
  275. while(s < es && strchr(whitespace, *s))
  276. s++;
  277. *ps = s;
  278. return *s && strchr(toks, *s);
  279. }
  280. struct cmd *parseline(char**, char*);
  281. struct cmd *parsepipe(char**, char*);
  282. struct cmd *parseexec(char**, char*);
  283. struct cmd *nulterminate(struct cmd*);
  284. struct cmd*
  285. parsecmd(char *s)
  286. {
  287. char *es;
  288. struct cmd *cmd;
  289. es = s + strlen(s);
  290. cmd = parseline(&s, es);
  291. peek(&s, es, "");
  292. if(s != es){
  293. printf(2, "leftovers: %s\n", s);
  294. panic("syntax");
  295. }
  296. nulterminate(cmd);
  297. return cmd;
  298. }
  299. struct cmd*
  300. parseline(char **ps, char *es)
  301. {
  302. struct cmd *cmd;
  303. cmd = parsepipe(ps, es);
  304. while(peek(ps, es, "&")){
  305. gettoken(ps, es, 0, 0);
  306. cmd = backcmd(cmd);
  307. }
  308. if(peek(ps, es, ";")){
  309. gettoken(ps, es, 0, 0);
  310. cmd = listcmd(cmd, parseline(ps, es));
  311. }
  312. return cmd;
  313. }
  314. struct cmd*
  315. parsepipe(char **ps, char *es)
  316. {
  317. struct cmd *cmd;
  318. cmd = parseexec(ps, es);
  319. if(peek(ps, es, "|")){
  320. gettoken(ps, es, 0, 0);
  321. cmd = pipecmd(cmd, parsepipe(ps, es));
  322. }
  323. return cmd;
  324. }
  325. struct cmd*
  326. parseredirs(struct cmd *cmd, char **ps, char *es)
  327. {
  328. int tok;
  329. char *q, *eq;
  330. while(peek(ps, es, "<>")){
  331. tok = gettoken(ps, es, 0, 0);
  332. if(gettoken(ps, es, &q, &eq) != 'a')
  333. panic("missing file for redirection");
  334. switch(tok){
  335. case '<':
  336. cmd = redircmd(cmd, q, eq, O_RDONLY, 0);
  337. break;
  338. case '>':
  339. cmd = redircmd(cmd, q, eq, O_WRONLY|O_CREATE, 1);
  340. break;
  341. case '+': // >>
  342. cmd = redircmd(cmd, q, eq, O_WRONLY|O_CREATE, 1);
  343. break;
  344. }
  345. }
  346. return cmd;
  347. }
  348. struct cmd*
  349. parseblock(char **ps, char *es)
  350. {
  351. struct cmd *cmd;
  352. if(!peek(ps, es, "("))
  353. panic("parseblock");
  354. gettoken(ps, es, 0, 0);
  355. cmd = parseline(ps, es);
  356. if(!peek(ps, es, ")"))
  357. panic("syntax - missing )");
  358. gettoken(ps, es, 0, 0);
  359. cmd = parseredirs(cmd, ps, es);
  360. return cmd;
  361. }
  362. struct cmd*
  363. parseexec(char **ps, char *es)
  364. {
  365. char *q, *eq;
  366. int tok, argc;
  367. struct execcmd *cmd;
  368. struct cmd *ret;
  369. if(peek(ps, es, "("))
  370. return parseblock(ps, es);
  371. ret = execcmd();
  372. cmd = (struct execcmd*)ret;
  373. argc = 0;
  374. ret = parseredirs(ret, ps, es);
  375. while(!peek(ps, es, "|)&;")){
  376. if((tok=gettoken(ps, es, &q, &eq)) == 0)
  377. break;
  378. if(tok != 'a')
  379. panic("syntax");
  380. cmd->argv[argc] = q;
  381. cmd->eargv[argc] = eq;
  382. argc++;
  383. if(argc >= MAXARGS)
  384. panic("too many args");
  385. ret = parseredirs(ret, ps, es);
  386. }
  387. cmd->argv[argc] = 0;
  388. cmd->eargv[argc] = 0;
  389. return ret;
  390. }
  391. // NUL-terminate all the counted strings.
  392. struct cmd*
  393. nulterminate(struct cmd *cmd)
  394. {
  395. int i;
  396. struct backcmd *bcmd;
  397. struct execcmd *ecmd;
  398. struct listcmd *lcmd;
  399. struct pipecmd *pcmd;
  400. struct redircmd *rcmd;
  401. if(cmd == 0)
  402. return 0;
  403. switch(cmd->type){
  404. case EXEC:
  405. ecmd = (struct execcmd*)cmd;
  406. for(i=0; ecmd->argv[i]; i++)
  407. *ecmd->eargv[i] = 0;
  408. break;
  409. case REDIR:
  410. rcmd = (struct redircmd*)cmd;
  411. nulterminate(rcmd->cmd);
  412. *rcmd->efile = 0;
  413. break;
  414. case PIPE:
  415. pcmd = (struct pipecmd*)cmd;
  416. nulterminate(pcmd->left);
  417. nulterminate(pcmd->right);
  418. break;
  419. case LIST:
  420. lcmd = (struct listcmd*)cmd;
  421. nulterminate(lcmd->left);
  422. nulterminate(lcmd->right);
  423. break;
  424. case BACK:
  425. bcmd = (struct backcmd*)cmd;
  426. nulterminate(bcmd->cmd);
  427. break;
  428. }
  429. return cmd;
  430. }