// // Created by servostar on 5/31/24. // #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)) { free(value); 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 = malloc(sizeof(Option)); option->is_opt = g_str_has_prefix(argv[i], "--"); option->string = strdup(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 = g_array_new(FALSE, FALSE, 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) { g_array_free(array, FALSE); return NULL; } return array; } TargetConfig* default_target_config() { DEBUG("generating default target config..."); TargetConfig* config = malloc(sizeof(TargetConfig)); config->name = strdup("out"); config->print_ast = false; config->print_asm = false; config->print_ir = false; config->mode = Application; config->archive_directory = strdup("archive"); config->output_directory = strdup("bin"); config->optimization_level = 1; config->root_module = NULL; config->link_search_paths = g_array_new(FALSE, FALSE, sizeof(char*)); return config; } TargetConfig* default_target_config_from_args() { DEBUG("generating default target from command line..."); TargetConfig* config = default_target_config(); 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 = strdup(opt->value); } } 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 = malloc(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 = malloc(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 = strdup( ((char**) files->data) [0]); g_array_free(files, TRUE); } 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