/* Copyright (C) 2019, 2020 Leo Tenenbaum. This file is part of toc. toc is distributed under version 3 of the GNU General Public License, without any warranty whatsoever. You should have received a copy of the GNU General Public License along with toc. If not, see . */ /* see development.md for development information */ /* @TODO: a..b should probably only go up to b-1 figure out how printf is gonna work if we do #include "foo.toc", bar; and foo.toc fails, bar should be declared as TYPE_UNKNOWN (right now it's undeclared) fix #foreign not at global scope - right now the cgen'd definition doesn't use the proper type find out why file output is really slow at compile time; try to improve it improve type_to_str: Foo ::= struct(t::Type) {} type_to_str(Foo(int)) switch - #fallthrough enums unions bitwise operations --- #compile_only for functions only used at compile time warn about x : u8 = 1283; switch to / add as an alternative: libffi - better yet, inline assembly don't bother generating ret_ if nothing's deferred X ::= newtype(int); or something any odd number of "s for a string give each warning a number; #no_warn(warning), #no_warn_throughout_this_file(warning) test various parse errors; see if they can be improved, e.g. if else { 3; } use point #except x; optional -Wshadow format errors so that vim/emacs can jump to them show include stack--especially for redeclarations with #include #force stuff like __builtin_sqrt --- make sure that floating point literals are as exact as possible have some way of doing Infinity and s/qNaN (you can have them be in std/math.toc) once you have a bunch of test code: - analyze memory usage by secretly passing __FILE__, __LINE__ to allocr_m/c/realloc - try making more Expression members pointers - try making Value.slice a pointer - should val_stack be on the allocator? what about temporary arrays? -->on the contrary, should in_decls be off the allocator? maybe macros are just inline functions passing untyped expressions to macros */ #if defined __unix__ || (defined __APPLE__ && defined __MACH__) #define _POSIX_C_SOURCE 200112L #include #define UNISTD_AVAILABLE 1 #endif #include "toc.c" #if defined TOC_DEBUG && defined __GNU_LIBRARY__ && defined UNISTD_AVAILABLE && !defined BACKTRACE #define BACKTRACE 1 #endif #if BACKTRACE #include #include static char *program_name; static void signal_handler(int num) { switch (num) { case SIGABRT: fprintf(stderr, "Aborted.\n"); break; case SIGSEGV: fprintf(stderr, "Segmentation fault.\n"); break; case SIGFPE: fprintf(stderr, "Floating point exception.\n"); break; case SIGINT: fprintf(stderr, "Interrupted.\n"); break; case SIGTERM: fprintf(stderr, "Terminated.\n"); break; case SIGILL: fprintf(stderr, "Illegal instruction.\n"); break; default: fprintf(stderr, "Terminated for unknown reason (signal %d).\n", num); break; } fprintf(stderr, "Stack trace:\n"); static void *addrs[300]; int naddrs = (int)(sizeof addrs / sizeof *addrs); naddrs = backtrace(addrs, naddrs); /* char **syms = backtrace_symbols(addrs, naddrs); */ char command[2048] = "addr2line -p -f -a -e "; strcat(command, program_name); strcat(command, " "); for (int i = 4; i < naddrs; ++i) { snprintf(command + strlen(command), sizeof command - strlen(command), "%p ", addrs[i]); } system(command); /* free(syms); */ signal(SIGABRT, SIG_DFL); abort(); } #endif static const char *replace_extension(Allocator *a, const char *filename, const char *new_extension, const char *dflt) { char *new_filename = allocr_malloc(a, strlen(filename) + strlen(new_extension) + 1); strcpy(new_filename, filename); char *dot = strchr(new_filename, '.'); if (dot) { strcpy(dot, new_extension); return new_filename; } else { return dflt; } } int main(int argc, char **argv) { #if BACKTRACE program_name = argv[0]; signal(SIGABRT, signal_handler); signal(SIGSEGV, signal_handler); signal(SIGINT, signal_handler); signal(SIGTERM, signal_handler); signal(SIGILL, signal_handler); signal(SIGFPE, signal_handler); #endif #if RUN_TESTS printf("running tests...\n"); test_all(); #endif const char *in_filename = NULL; const char *out_filename = NULL, *c_filename = NULL; const char *c_compiler = "cc"; char *env_CC = getenv("CC"); if (env_CC) c_compiler = env_CC; bool verbose = false; bool debug_build = true; bool compile_c = true; ErrCtx err_ctx = {0}; err_ctx.enabled = true; bool default_color_enabled; #if UNISTD_AVAILABLE #if defined _POSIX_VERSION && _POSIX_VERSION >= 200112L /* isatty available */ default_color_enabled = (bool)isatty(2); /* is /dev/stderr a tty? */ #else default_color_enabled = false; /* old posix version */ #endif #else default_color_enabled = false; /* probably windows */ #endif err_ctx.color_enabled = default_color_enabled; for (int i = 1; i < argc; ++i) { char *arg = argv[i]; if (streq(arg, "-no-color")) { err_ctx.color_enabled = false; } else if (streq(arg, "-color")) { err_ctx.color_enabled = true; } else if (streq(arg, "-o")) { if (i == argc-1) { fprintf(stderr, "-o cannot be the last argument to toc.\n"); return EXIT_FAILURE; } out_filename = argv[i+1]; ++i; } else if (streq(arg, "-O")) { if (i == argc-1) { fprintf(stderr, "-O cannot be the last argument to toc.\n"); return EXIT_FAILURE; } c_filename = argv[i+1]; ++i; } else if (streq(arg, "-v") || streq(arg, "-verbose")) { printf("Verbose mode enabled\n"); verbose = true; } else if (streq(arg, "-d") || streq(arg, "-debug")) { debug_build = true; } else if (streq(arg, "-r") || streq(arg, "-release")) { debug_build = false; } else if (streq(arg, "-cc")) { if (i == argc-1) { fprintf(stderr, "-cc cannot be the last argument to toc.\n"); return EXIT_FAILURE; } c_compiler = argv[i+1]; ++i; } else if (streq(arg, "-c")) { compile_c = false; } else { if (arg[0] == '-') { fprintf(stderr, "Unrecognized option: %s.\n", argv[i]); return EXIT_FAILURE; } else { in_filename = arg; } } } if (!in_filename) { #ifdef TOC_DEBUG in_filename = "test.toc"; #else fprintf(stderr, "Please specify an input file.\n"); return EXIT_FAILURE; #endif } Allocator main_allocr; allocr_create(&main_allocr); if (c_filename && out_filename && !compile_c) { fprintf(stderr, "You can't set both -o and -O unless you're compiling the outputted C code."); return EXIT_FAILURE; } if (!c_filename) { if (out_filename && !compile_c) { c_filename = out_filename; } else { c_filename = replace_extension(&main_allocr, in_filename, ".c", "out.c"); } } if (!out_filename) { out_filename = replace_extension(&main_allocr, in_filename, "", "a.out"); } File file = {0}; file.filename = in_filename; Location file_where = {0}; file_where.file = &file; file.ctx = &err_ctx; if (verbose) printf("Reading contents of %s...\n", in_filename); char *contents = read_file_contents(&main_allocr, in_filename, file_where); if (!contents) return EXIT_FAILURE; Identifiers globals; idents_create(&globals, &main_allocr, NULL); Tokenizer t; file.contents = contents; if (verbose) printf("Tokenizing...\n"); tokr_create(&t, &err_ctx, &main_allocr); if (!tokenize_file(&t, &file)) { err_text_important(&err_ctx, "Errors occured during preprocessing.\n"); allocr_free_all(&main_allocr); return EXIT_FAILURE; } if (verbose) { printf("Tokens:\n"); arr_foreach(t.tokens, Token, token) { if (token != t.tokens) printf(" "); fprint_token(stdout, token); } printf("\n-----\n"); } if (verbose) printf("Parsing...\n"); GlobalCtx global_ctx = {0}; str_hash_table_create(&global_ctx.included_files, sizeof(IncludedFile), &main_allocr); global_ctx.main_file = &file; global_ctx.err_ctx = &err_ctx; global_ctx.debug_build = debug_build; Parser p; parser_create(&p, &globals, &t, &main_allocr, &global_ctx); ParsedFile f; if (!parse_file(&p, &f)) { err_text_important(&err_ctx, "Errors occured during parsing.\n"); allocr_free_all(&main_allocr); return EXIT_FAILURE; } if (verbose) { printf("Parsed file:\n"); fprint_parsed_file(stdout, &f); printf("\n\n-----\n\n"); } if (verbose) printf("Determining types...\n"); Typer tr; Evaluator ev; evalr_create(&ev, &tr, &main_allocr); typer_create(&tr, &ev, &main_allocr, &globals, &global_ctx); if (!types_file(&tr, &f)) { err_text_important(&err_ctx, "Errors occured while determining types.\n"); allocr_free_all(&main_allocr); return EXIT_FAILURE; } if (verbose) { printf("Typed file:\n"); fprint_parsed_file(stdout, &f); printf("\n\n-----\n\n"); } if (verbose) printf("Opening output file %s...\n", c_filename); FILE *out = fopen(c_filename, "w"); if (!out) { err_text_important(&err_ctx, "Could not open output file: "); err_fprint(&err_ctx, "%s\n", c_filename); allocr_free_all(&main_allocr); return EXIT_FAILURE; } if (verbose) printf("Generating C code...\n"); CGenerator g; cgen_create(&g, out, &globals, &main_allocr); cgen_file(&g, &f, &tr); fclose(out); if (compile_c) { if (verbose) printf("Compiling C code...\n"); char cmd[2048] = {0}; char const *cflags = debug_build ? "-O0 -g" : "-O3 -s"; snprintf(cmd, sizeof cmd, "%s -w %s %s -o %s", c_compiler, cflags, c_filename, out_filename); int ret = system(cmd); if (ret) { fprintf(stderr, "Uh oh... C compiler failed...\n"); return EXIT_FAILURE; } } if (verbose) printf("Cleaning up...\n"); allocr_free_all(&main_allocr); return 0; }