-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsh.c
96 lines (82 loc) · 1.51 KB
/
sh.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
/* \file sh.c
*
* A simple shell
*/
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include "cmd.h"
#include "error.h"
#include "eval.h"
#include "input.h"
#include "lexer.h"
#include "mem.h"
#include "output.h"
#include "parser.h"
#include "redir.h"
#include "sh.h"
#include "trap.h"
#include "var.h"
int rootpid;
int main(int argc, char **argv)
{
int fd, exception;
struct stackmark mark;
struct jmploc jmploc;
if (argc > 1) {
if (strcmp(argv[1], "-c") == 0) {
if (argc > 2)
setinputstring(argv[2], 0);
else
die("-c requires an argument");
} else {
fd = setinputfile(argv[1], INPUT_NOFILE_OK);
if (fd < 0)
sdie(127, "%s:", argv[1]);
}
}
INTOFF;
if ((exception = setjmp(jmploc.loc))) {
/* reset the shell */
unwindredir();
unwindloops();
closescript();
yytoken = TNL;
popstackmark(&mark);
if (exception == EXINT)
fputc('\n', stderr);
} else {
pushstackmark(&mark);
rootpid = getpid();
initvar();
signal_init();
}
handler = &jmploc;
FORCEINTON;
return repl();
}
int repl(void)
{
int status = 0;
struct cmd *cmd;
struct stackmark mark;
pushstackmark(&mark);
for (; (cmd = parseline()) != CEOF; popstackmark(&mark))
if (cmd)
status = eval(cmd);
return status;
}
int source_builtin(struct cexec *cmd)
{
int status;
if (cmd->argc < 2) {
perrorf("%s: not enough arguments", cmd->argv[0]);
return 1;
}
setinputfile(cmd->argv[1], INPUT_PUSH_FILE);
status = repl();
popfile();
return status;
}