summaryrefslogtreecommitdiff
path: root/main.c
blob: 50df1384315038cd7ebdd8468a02f7df3dbd31e0 (plain)
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
/* TODO: multi-dimensional arrays are flattened in types.c */
/* TODO: remove all cgen errors */
/* TODO: don't eval consts in C */
/* TODO: Functions returning fixed-length arrays */
/* TODO: improve error for declaring a keyword, e.g. i8: int = 8123; */
#include "toc.c"

int main(int argc, char **argv) {
	if (argc < 2) {
		fprintf(stderr, "Please specify an input file.\n");
		return EXIT_FAILURE;
	}

	const char *in_filename = argv[1];
	FILE *in = fopen(in_filename, "r");
	if (!in) {
		fprintf(stderr, "Could not open file: %s.\n", argv[1]);
		return EXIT_FAILURE;
	}
	
	char *contents = err_malloc(4096);
	long contents_cap = 4095;
	long contents_len = 0;
	while (fgets(contents + contents_len, (int)(contents_cap - contents_len), in)) {
		contents_len += (long)strlen(contents + contents_len);
		
		if (contents_len >= (long)contents_cap - 1024) {
			contents_cap *= 2;
			/* allocate +1 so that pointers don't overflow */
			contents = err_realloc(contents, (size_t)contents_cap + 1);
		}
	}
	if (ferror(in)) {
		fprintf(stderr, "Error reading input file: %s.\n", argv[1]);
		return EXIT_FAILURE;
	}
	fclose(in);

	err_filename = in_filename;
	Identifiers file_idents;
	idents_create(&file_idents);
	Tokenizer t;
	tokr_create(&t, &file_idents);
	if (!tokenize_string(&t, contents)) {
		err_fprint(TEXT_IMPORTANT("Errors occured while preprocessing.\n"));
		return EXIT_FAILURE;
	}
	
	arr_foreach(&t.tokens, Token, token) {
		if (token != t.tokens.data)
			printf("    ");
		fprint_token(stdout, token);
	}
	printf("\n");
	Parser p;
	parser_from_tokenizer(&p, &t);
   	ParsedFile f;
	if (!parse_file(&p, &f)) {
		err_fprint(TEXT_IMPORTANT("Errors occured while parsing.\n"));
		return EXIT_FAILURE;
	}

	block_enter(NULL, &f.stmts); /* enter global scope */
	if (!types_file(&f)) {
		err_fprint(TEXT_IMPORTANT("Errors occured while determining types.\n"));
		return EXIT_FAILURE;
	}
	
	fprint_parsed_file(stdout, &f);
	
	tokr_free(&t);

	
	
	/* TODO (eventually): use a tmp file (don't overwrite old output if there's an error) */
	/* const char *c_out_filename = "out.c"; */
	/* const char *h_out_filename = "out.h"; */
	/* FILE *c_out = fopen(c_out_filename, "w"); */
	/* FILE *h_out = fopen(h_out_filename, "w"); */
	/* CGenerator cgen; */
	/* cgen_create(&cgen, &file_idents, c_out, h_out, h_out_filename); */
	/* if (!cgen_file(&cgen, &f)) { */
	/* 	err_fprint(TEXT_IMPORTANT("Errors occured while generating C code.\n")); */
	/* 	return EXIT_FAILURE; */
	/* } */
	
	block_exit(NULL, &f.stmts); /* exit global scope */
	free(contents);
	
	/* fclose(c_out); */
	/* fclose(h_out); */
	idents_free(&file_idents);
}