diff options
author | 2024-06-03 18:18:32 +0800 | |
---|---|---|
committer | 2024-06-03 18:18:32 +0800 | |
commit | 69a2b6eaaa21354e78b5fe6d79b0e7e10e127dcc (patch) | |
tree | ac5614b4713862f2633098e6321b638d28256244 | |
download | bcsh-69a2b6eaaa21354e78b5fe6d79b0e7e10e127dcc.tar.gz bcsh-69a2b6eaaa21354e78b5fe6d79b0e7e10e127dcc.zip |
Initial commit: Basic found.
- I use flex and bison for lexical and syntax analysis, and main c
to run command
- basic CFG built to deal with easy command without redirecion (
it's written but have serious problem)
- it can now run command one by one, and use \ to change line
without ending the command.
Fighting forward!
-rw-r--r-- | .gitignore | 8 | ||||
-rw-r--r-- | Makefile | 32 | ||||
-rw-r--r-- | bcsh.c | 10 | ||||
-rw-r--r-- | cmd.h | 30 | ||||
-rw-r--r-- | lex.l | 33 | ||||
-rw-r--r-- | syntax.y | 50 |
6 files changed, 163 insertions, 0 deletions
diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..66c7e33 --- /dev/null +++ b/.gitignore | |||
@@ -0,0 +1,8 @@ | |||
1 | * | ||
2 | !*.* | ||
3 | !*/* | ||
4 | !Makefile | ||
5 | *.o | ||
6 | *.tab* | ||
7 | lex.yy.c | ||
8 | *.output \ No newline at end of file | ||
diff --git a/Makefile b/Makefile new file mode 100644 index 0000000..8410a24 --- /dev/null +++ b/Makefile | |||
@@ -0,0 +1,32 @@ | |||
1 | CC = gcc | ||
2 | FLEX = flex | ||
3 | BISON = bison | ||
4 | |||
5 | CFILES = $(shell find . -name "*.c") | ||
6 | OBJS = $(CFILES:.c=.o) | ||
7 | LFILE = $(wildcard *.l) | ||
8 | YFILE = $(wildcard *.y) | ||
9 | LFC = ./lex.yy.c | ||
10 | YFC = ./syntax.tab.c | ||
11 | LFO = $(LFC:.c=.o) | ||
12 | YFO = $(YFC:.c=.o) | ||
13 | |||
14 | bcsh: syntax $(filter-out $(LFO),$(OBJS)) | ||
15 | $(CC) -g -o bcsh $(filter-out $(LFO),$(OBJS)) -lfl | ||
16 | |||
17 | syntax: lexical syntax-c | ||
18 | $(CC) -c $(YFC) -o $(YFO) | ||
19 | |||
20 | lexical: $(LFILE) | ||
21 | $(FLEX) -o $(LFC) $(LFILE) | ||
22 | |||
23 | syntax-c: $(YFILE) | ||
24 | $(BISON) -o $(YFC) -d -v $(YFILE) -Wcounterexamples | ||
25 | |||
26 | -include $(patsubst %.o, %.d, $(OBJS)) | ||
27 | |||
28 | clean: | ||
29 | rm -f bcsh lex.yy.c syntax.tab.c syntax.tab.h syntax.output | ||
30 | rm -f $(OBJS) $(OBJS:.o=.d) | ||
31 | rm -f $(LFC) $(YFC) $(YFC:.c=.h) | ||
32 | rm -f *~ \ No newline at end of file | ||
@@ -0,0 +1,10 @@ | |||
1 | #include "cmd.h" | ||
2 | |||
3 | extern int yyparse(); | ||
4 | |||
5 | int main(int argc, char **argv) { | ||
6 | while (true) { | ||
7 | yyparse(); | ||
8 | } | ||
9 | return 0; | ||
10 | } \ No newline at end of file | ||
@@ -0,0 +1,30 @@ | |||
1 | #ifndef CMD_H | ||
2 | #define CMD_H | ||
3 | |||
4 | #include <stdbool.h> | ||
5 | #include <stdio.h> | ||
6 | #include <stdlib.h> | ||
7 | |||
8 | typedef enum { | ||
9 | CMD_TYPE_NORMAL, | ||
10 | CMD_TYPE_PIPE, | ||
11 | CMD_TYPE_AND, | ||
12 | CMD_TYPE_OR, | ||
13 | // add more command types as needed | ||
14 | } CommandType; | ||
15 | |||
16 | typedef struct Command { | ||
17 | CommandType type; // the type of the command | ||
18 | char **args; // an array of strings for the arguments of the command | ||
19 | struct Command *left; // a pointer to the left sub-command | ||
20 | struct Command *right; // a pointer to the right sub-command | ||
21 | char *redirect_in; // the file name for input redirection, or NULL if there | ||
22 | // is no input redirection | ||
23 | char *redirect_out; // the file name for output redirection, or NULL if | ||
24 | // there is no output redirection | ||
25 | char *append_out; // the file name for output appending, or NULL if there is | ||
26 | // no output appending | ||
27 | // add more fields as needed | ||
28 | } Command; | ||
29 | |||
30 | #endif \ No newline at end of file | ||
@@ -0,0 +1,33 @@ | |||
1 | %{ | ||
2 | #include "syntax.tab.h" // Bison 生成的头文件 | ||
3 | |||
4 | void yyerror(const char *s); | ||
5 | %} | ||
6 | |||
7 | %option noyywrap | ||
8 | |||
9 | %% | ||
10 | |||
11 | [\t ]+ { /* Ignore whitespace */ } | ||
12 | "\n" { printf("new line\n");return NEWLINE; } | ||
13 | "\\\n" { printf("> "); } | ||
14 | ">>" { return APPEND_OUT; } | ||
15 | "<<" { return HEREDOC; } | ||
16 | ">" { return REDIRECT_OUT; } | ||
17 | "<" { return REDIRECT_IN; } | ||
18 | ">>&" { return APPEND_OUTPUT_ERR; } | ||
19 | ">&" { return REDIRECT_OUTPUT_ERR; } | ||
20 | "|" { return PIPE; } | ||
21 | "&&" { return AND; } | ||
22 | "||" { return OR; } | ||
23 | "&" { return BACKGROUND; } | ||
24 | \"(\\.|[^\"])*\" { yylval.str = strdup(yytext); return STRING; } | ||
25 | [a-zA-Z0-9_\-\/.]+ { yylval.str = strdup(yytext); return WORD; } | ||
26 | |||
27 | |||
28 | %% | ||
29 | |||
30 | void yyerror(const char *s) | ||
31 | { | ||
32 | fprintf(stderr, "error: %s\n", s); | ||
33 | } | ||
diff --git a/syntax.y b/syntax.y new file mode 100644 index 0000000..aa0df71 --- /dev/null +++ b/syntax.y | |||
@@ -0,0 +1,50 @@ | |||
1 | %{ | ||
2 | #include "lex.yy.c" | ||
3 | #include "cmd.h" | ||
4 | |||
5 | %} | ||
6 | |||
7 | %union { | ||
8 | char *str; | ||
9 | } | ||
10 | |||
11 | %token <str> WORD STRING | ||
12 | %token NEWLINE CONTINUATION APPEND_OUT REDIRECT_OUT REDIRECT_IN APPEND_OUTPUT_ERR REDIRECT_OUTPUT_ERR PIPE AND OR BACKGROUND HEREDOC UNKNOWN | ||
13 | |||
14 | %% | ||
15 | |||
16 | line: | ||
17 | command NEWLINE { printf("Parsed a command.\n"); } | ||
18 | | NEWLINE { /* empty line */ } | ||
19 | ; | ||
20 | |||
21 | command: | ||
22 | part {printf("PART\n"); } | ||
23 | | part separator command { printf("COMMAND\n"); } | ||
24 | | part REDIRECT_OUT WORD tail { printf("REDIRECT_OUT: %s\n", $3); free($3); } | ||
25 | | part REDIRECT_IN WORD tail { printf("REDIRECT_IN: %s\n", $3); free($3); } | ||
26 | | part HEREDOC WORD tail { printf("HEREDOC: %s\n", $3); free($3); } | ||
27 | | part BACKGROUND { printf("BACKGROUND\n"); } | ||
28 | ; | ||
29 | |||
30 | tail: | ||
31 | part | ||
32 | | part REDIRECT_OUT WORD tail { printf("REDIRECT_OUT: %s\n", $3); free($3); } | ||
33 | | part REDIRECT_IN WORD tail { printf("REDIRECT_IN: %s\n", $3); free($3); } | ||
34 | | part HEREDOC WORD tail { printf("HEREDOC: %s\n", $3); free($3); } | ||
35 | ; | ||
36 | |||
37 | part: | ||
38 | part WORD { printf("WORD: %s\n", $2); free($2); } | ||
39 | | part STRING { printf("STRING: %s\n", $2); free($2); } | ||
40 | | WORD { printf("WORD: %s\n", $1); free($1); } | ||
41 | | STRING { printf("STRING: %s\n", $1); free($1); } | ||
42 | ; | ||
43 | |||
44 | separator: | ||
45 | PIPE { printf("PIPE\n"); } | ||
46 | | AND { printf("AND\n"); } | ||
47 | | OR { printf("OR\n"); } | ||
48 | ; | ||
49 | |||
50 | %% \ No newline at end of file | ||