// // Created by servostar on 5/31/24. // #include #include #include #include #include #include #include #include static GHashTable* args = NULL; static void clean(void) { GHashTableIter iter; gpointer key, value; g_hash_table_iter_init(&iter, args); while (g_hash_table_iter_next(&iter, &key, &value)) { mem_free(value); mem_free(key); } g_hash_table_destroy(args); } void parse_options(int argc, char* argv[]) { args = g_hash_table_new(g_str_hash, g_str_equal); atexit(clean); for (int i = 0; i < argc; i++) { Option* option = mem_alloc(MemoryNamespaceOpt, sizeof(Option)); option->is_opt = g_str_has_prefix(argv[i], "--"); option->string = mem_strdup(MemoryNamespaceOpt, argv[i] + (option->is_opt ? 2 : 0)); option->index = i; option->value = NULL; char* equals = strchr(option->string, '='); if (equals != NULL) { option->value = equals + 1; *equals = 0; } g_hash_table_insert(args, (gpointer) option->string, (gpointer) option); } } bool is_option_set(const char* option) { assert(option != NULL); assert(args != NULL); return g_hash_table_contains(args, option); } const Option* get_option(const char* option) { if (g_hash_table_contains(args, option)) { return g_hash_table_lookup(args, option); } return NULL; } GArray* get_non_options_after(const char* command) { const Option* command_option = get_option(command); if (command_option == NULL) { return NULL; } GArray* array = mem_new_g_array(MemoryNamespaceOpt, sizeof(const char*)); GHashTableIter iter; gpointer key, value; g_hash_table_iter_init(&iter, args); while (g_hash_table_iter_next(&iter, &key, &value)) { Option* option = (Option*) value; if (!option->is_opt && command_option->index < option->index) { g_array_append_val(array, option->string); } } if (array->len == 0) { return NULL; } return array; } TargetConfig* default_target_config() { DEBUG("generating default target config..."); TargetConfig* config = mem_alloc(MemoryNamespaceOpt, sizeof(TargetConfig)); config->name = mem_strdup(MemoryNamespaceOpt, "out"); config->print_ast = false; config->print_asm = false; config->print_ir = false; config->driver = mem_strdup(MemoryNamespaceOpt, DEFAULT_DRIVER); config->mode = Application; config->archive_directory = mem_strdup(MemoryNamespaceOpt, "archive"); config->output_directory = mem_strdup(MemoryNamespaceOpt, "bin"); config->optimization_level = 1; config->root_module = NULL; config->link_search_paths = mem_new_g_array(MemoryNamespaceOpt, sizeof(char*)); config->lld_fatal_warnings = FALSE; config->gsc_fatal_warnings = FALSE; config->import_paths = mem_new_g_array(MemoryNamespaceOpt, sizeof(char*)); return config; } TargetConfig* default_target_config_from_args() { DEBUG("generating default target from command line..."); TargetConfig* config = default_target_config(); gboolean fatal_warnings = is_option_set("all-fatal-warnings"); if (fatal_warnings || is_option_set("lld-fatal-warnings")) { config->lld_fatal_warnings = true; } if (fatal_warnings || is_option_set("gsc-fatal-warnings")) { config->gsc_fatal_warnings = true; } if (is_option_set("print-ast")) { config->print_ast = true; } if (is_option_set("print-asm")) { config->print_asm = true; } if (is_option_set("print-ir")) { config->print_ir = true; } if (is_option_set("mode")) { const Option* opt = get_option("mode"); if (opt->value != NULL) { if (strcmp(opt->value, "app") == 0) { config->mode = Application; } else if (strcmp(opt->value, "lib") == 0) { config->mode = Library; } else { print_message(Warning, "Invalid compilation mode: %s", opt->value); } } } if (is_option_set("output")) { const Option* opt = get_option("output"); if (opt->value != NULL) { config->name = mem_strdup(MemoryNamespaceOpt, (char*) opt->value); } } if (is_option_set("driver")) { const Option* opt = get_option("driver"); if (opt->value != NULL) { config->driver = mem_strdup(MemoryNamespaceOpt, (char*) opt->value); } } char* cwd = g_get_current_dir(); char* cached_cwd = mem_strdup(MemoryNamespaceOpt, cwd); g_array_append_val(config->link_search_paths, cached_cwd); free(cwd); if (is_option_set("link-paths")) { const Option* opt = get_option("link-paths"); if (opt->value != NULL) { const char* start = opt->value; const char* end = NULL; while((end = strchr(start, ',')) != NULL) { const int len = end - start; char* link_path = mem_alloc(MemoryNamespaceOpt, len + 1); memcpy(link_path, start, len); link_path[len] = 0; g_array_append_val(config->link_search_paths, link_path); start = end; } const int len = strlen(start); if (len > 0) { char* link_path = mem_alloc(MemoryNamespaceOpt, len + 1); memcpy(link_path, start, len); link_path[len] = 0; g_array_append_val(config->link_search_paths, link_path); } } } GArray* files = get_non_options_after("compile"); if (files == NULL) { print_message(Error, "No input file provided."); } else { if (files->len > 1) { print_message(Warning, "Got more than one file to compile, using first, ignoring others."); } config->root_module = mem_strdup(MemoryNamespaceOpt, g_array_index(files, char*, 0)); } char* default_import_path = mem_strdup(MemoryNamespaceOpt, "."); g_array_append_val(config->import_paths, default_import_path); if (is_option_set("import-paths")) { const Option* opt = get_option("import-paths"); if (opt->value != NULL) { const char* start = opt->value; const char* end = NULL; while((end = strchr(start, ',')) != NULL) { const int len = end - start; char* import_path = mem_alloc(MemoryNamespaceOpt, len + 1); memcpy(import_path, start, len); import_path[len] = 0; g_array_append_val(config->import_paths, import_path); start = end; } const int len = strlen(start); if (len > 0) { char* import_path = mem_alloc(MemoryNamespaceOpt, len + 1); memcpy(import_path, start, len); import_path[len] = 0; g_array_append_val(config->import_paths, import_path); } } } return config; } void print_help(void) { DEBUG("printing help dialog..."); const char *lines[] = { "Gemstone Compiler (c) GPL-2.0", "Build a project target: gsc build [target]|all", "Compile non-project file: gsc compile [file]", "Output information: gsc