aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorWe-unite <3205135446@qq.com>2024-06-03 18:18:32 +0800
committerWe-unite <3205135446@qq.com>2024-06-03 18:18:32 +0800
commit69a2b6eaaa21354e78b5fe6d79b0e7e10e127dcc (patch)
treeac5614b4713862f2633098e6321b638d28256244
downloadbcsh-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--.gitignore8
-rw-r--r--Makefile32
-rw-r--r--bcsh.c10
-rw-r--r--cmd.h30
-rw-r--r--lex.l33
-rw-r--r--syntax.y50
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*
7lex.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 @@
1CC = gcc
2FLEX = flex
3BISON = bison
4
5CFILES = $(shell find . -name "*.c")
6OBJS = $(CFILES:.c=.o)
7LFILE = $(wildcard *.l)
8YFILE = $(wildcard *.y)
9LFC = ./lex.yy.c
10YFC = ./syntax.tab.c
11LFO = $(LFC:.c=.o)
12YFO = $(YFC:.c=.o)
13
14bcsh: syntax $(filter-out $(LFO),$(OBJS))
15 $(CC) -g -o bcsh $(filter-out $(LFO),$(OBJS)) -lfl
16
17syntax: lexical syntax-c
18 $(CC) -c $(YFC) -o $(YFO)
19
20lexical: $(LFILE)
21 $(FLEX) -o $(LFC) $(LFILE)
22
23syntax-c: $(YFILE)
24 $(BISON) -o $(YFC) -d -v $(YFILE) -Wcounterexamples
25
26-include $(patsubst %.o, %.d, $(OBJS))
27
28clean:
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
diff --git a/bcsh.c b/bcsh.c
new file mode 100644
index 0000000..ca83402
--- /dev/null
+++ b/bcsh.c
@@ -0,0 +1,10 @@
1#include "cmd.h"
2
3extern int yyparse();
4
5int main(int argc, char **argv) {
6 while (true) {
7 yyparse();
8 }
9 return 0;
10} \ No newline at end of file
diff --git a/cmd.h b/cmd.h
new file mode 100644
index 0000000..adf03c7
--- /dev/null
+++ b/cmd.h
@@ -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
8typedef 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
16typedef 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
diff --git a/lex.l b/lex.l
new file mode 100644
index 0000000..036d58e
--- /dev/null
+++ b/lex.l
@@ -0,0 +1,33 @@
1%{
2#include "syntax.tab.h" // Bison 生成的头文件
3
4void 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
30void 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
16line:
17 command NEWLINE { printf("Parsed a command.\n"); }
18 | NEWLINE { /* empty line */ }
19 ;
20
21command:
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
30tail:
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
37part:
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
44separator:
45 PIPE { printf("PIPE\n"); }
46 | AND { printf("AND\n"); }
47 | OR { printf("OR\n"); }
48 ;
49
50%% \ No newline at end of file