/* Copyright holders: Sarah Walker see COPYING for more details */ #include #include #include #include "config.h" #include "ibm.h" char config_file_default[256]; static char config_file[256]; typedef struct list_t { struct list_t *next; } list_t; static list_t config_head; typedef struct section_t { struct list_t list; char name[256]; struct list_t entry_head; } section_t; typedef struct entry_t { struct list_t list; char name[256]; char data[256]; } entry_t; #define list_add(new, head) \ { \ struct list_t *next = head; \ \ while (next->next) \ next = next->next; \ \ (next)->next = new; \ (new)->next = NULL; \ } void config_dump() { section_t *current_section; pclog("Config data :\n"); current_section = (section_t *)config_head.next; while (current_section) { entry_t *current_entry; pclog("[%s]\n", current_section->name); current_entry = (entry_t *)current_section->entry_head.next; while (current_entry) { pclog("%s = %s\n", current_entry->name, current_entry->data); current_entry = (entry_t *)current_entry->list.next; } current_section = (section_t *)current_section->list.next; } } void config_free() { section_t *current_section; current_section = (section_t *)config_head.next; while (current_section) { section_t *next_section = (section_t *)current_section->list.next; entry_t *current_entry; current_entry = (entry_t *)current_section->entry_head.next; while (current_entry) { entry_t *next_entry = (entry_t *)current_entry->list.next; free(current_entry); current_entry = next_entry; } free(current_section); current_section = next_section; } } void config_load(char *fn) { FILE *f = fopen(fn, "rt"); section_t *current_section; memset(&config_head, 0, sizeof(list_t)); current_section = malloc(sizeof(section_t)); memset(current_section, 0, sizeof(section_t)); list_add(¤t_section->list, &config_head); if (!f) return; while (1) { int c; char buffer[256]; fgets(buffer, 255, f); if (feof(f)) break; c = 0; while (buffer[c] == ' ') c++; if (!buffer[c]) continue; if (buffer[c] == '#') /*Comment*/ continue; if (buffer[c] == '[') /*Section*/ { section_t *new_section; char name[256]; int d = 0; c++; while (buffer[c] != ']' && buffer[c]) name[d++] = buffer[c++]; if (buffer[c] != ']') continue; name[d] = 0; new_section = malloc(sizeof(section_t)); memset(new_section, 0, sizeof(section_t)); strncpy(new_section->name, name, 256); list_add(&new_section->list, &config_head); current_section = new_section; } else { entry_t *new_entry; char name[256]; int d = 0, data_pos; while (buffer[c] != '=' && buffer[c] != ' ' && buffer[c]) name[d++] = buffer[c++]; if (!buffer[c]) continue; name[d] = 0; while ((buffer[c] == '=' || buffer[c] == ' ') && buffer[c]) c++; if (!buffer[c]) continue; data_pos = c; while (buffer[c]) { if (buffer[c] == '\n') buffer[c] = 0; c++; } new_entry = malloc(sizeof(entry_t)); memset(new_entry, 0, sizeof(entry_t)); strncpy(new_entry->name, name, 256); strncpy(new_entry->data, &buffer[data_pos], 256); list_add(&new_entry->list, ¤t_section->entry_head); } } fclose(f); config_dump(); } void config_new() { FILE *f = fopen(config_file, "wt"); fclose(f); } static section_t *find_section(char *name) { section_t *current_section; char blank[] = ""; current_section = (section_t *)config_head.next; if (!name) name = blank; while (current_section) { if (!strncmp(current_section->name, name, 256)) return current_section; current_section = (section_t *)current_section->list.next; } return NULL; } static entry_t *find_entry(section_t *section, char *name) { entry_t *current_entry; current_entry = (entry_t *)section->entry_head.next; while (current_entry) { if (!strncmp(current_entry->name, name, 256)) return current_entry; current_entry = (entry_t *)current_entry->list.next; } return NULL; } static section_t *create_section(char *name) { section_t *new_section = malloc(sizeof(section_t)); memset(new_section, 0, sizeof(section_t)); strncpy(new_section->name, name, 256); list_add(&new_section->list, &config_head); return new_section; } static entry_t *create_entry(section_t *section, char *name) { entry_t *new_entry = malloc(sizeof(entry_t)); memset(new_entry, 0, sizeof(entry_t)); strncpy(new_entry->name, name, 256); list_add(&new_entry->list, §ion->entry_head); return new_entry; } int config_get_int(char *head, char *name, int def) { section_t *section; entry_t *entry; int value; section = find_section(head); if (!section) return def; entry = find_entry(section, name); if (!entry) return def; sscanf(entry->data, "%i", &value); return value; } char *config_get_string(char *head, char *name, char *def) { section_t *section; entry_t *entry; section = find_section(head); if (!section) return def; entry = find_entry(section, name); if (!entry) return def; return entry->data; } void config_set_int(char *head, char *name, int val) { section_t *section; entry_t *entry; section = find_section(head); if (!section) section = create_section(head); entry = find_entry(section, name); if (!entry) entry = create_entry(section, name); sprintf(entry->data, "%i", val); } void config_set_string(char *head, char *name, char *val) { section_t *section; entry_t *entry; section = find_section(head); if (!section) section = create_section(head); entry = find_entry(section, name); if (!entry) entry = create_entry(section, name); strncpy(entry->data, val, 256); } char *get_filename(char *s) { int c = strlen(s) - 1; while (c > 0) { if (s[c] == '/' || s[c] == '\\') return &s[c+1]; c--; } return s; } void append_filename(char *dest, char *s1, char *s2, int size) { sprintf(dest, "%s%s", s1, s2); } void put_backslash(char *s) { int c = strlen(s) - 1; if (s[c] != '/' && s[c] != '\\') s[c] = '/'; } char *get_extension(char *s) { int c = strlen(s) - 1; if (c <= 0) return s; while (c && s[c] != '.') c--; if (!c) return &s[strlen(s)]; return &s[c+1]; } void config_save(char *fn) { FILE *f = fopen(fn, "wt"); section_t *current_section; current_section = (section_t *)config_head.next; while (current_section) { entry_t *current_entry; if (current_section->name[0]) fprintf(f, "\n[%s]\n", current_section->name); current_entry = (entry_t *)current_section->entry_head.next; while (current_entry) { fprintf(f, "%s = %s\n", current_entry->name, current_entry->data); current_entry = (entry_t *)current_entry->list.next; } current_section = (section_t *)current_section->list.next; } fclose(f); }