rulimine/stage23/lib/config.c

284 lines
6.7 KiB
C
Raw Normal View History

2021-11-26 15:09:09 +03:00
#include "lib/uri.h"
2020-01-22 09:13:19 +03:00
#include <stddef.h>
2020-10-17 06:37:39 +03:00
#include <stdbool.h>
2020-01-22 09:13:19 +03:00
#include <lib/config.h>
#include <lib/libc.h>
2020-03-30 22:24:36 +03:00
#include <lib/blib.h>
2020-09-20 13:03:44 +03:00
#include <mm/pmm.h>
2020-04-14 06:20:55 +03:00
#include <fs/file.h>
2020-11-05 03:37:45 +03:00
#include <lib/print.h>
#include <pxe/tftp.h>
2020-01-22 09:13:19 +03:00
#include <sys/cpu.h>
2020-01-22 09:13:19 +03:00
#define SEPARATOR '\n'
2021-12-11 21:58:00 +03:00
extern bool *bad_config;
2020-10-20 03:38:44 +03:00
bool config_ready = false;
2020-03-30 22:24:36 +03:00
static char *config_addr;
int init_config_disk(struct volume *part) {
2021-10-21 02:27:05 +03:00
struct file_handle *f;
2020-03-30 22:24:36 +03:00
2021-10-21 02:27:05 +03:00
if ((f = fopen(part, "/limine.cfg")) == NULL
&& (f = fopen(part, "/boot/limine.cfg")) == NULL
2021-11-26 15:09:09 +03:00
&& (f = fopen(part, "/EFI/BOOT/limine.cfg")) == NULL
2021-11-27 14:01:47 +03:00
&& (f = uri_open("fwcfg:///opt/org.limine-bootloader.config")) == NULL) {
return -1;
2020-03-30 22:24:36 +03:00
}
2021-10-21 02:27:05 +03:00
size_t config_size = f->size + 1;
config_addr = ext_mem_alloc(config_size);
2020-03-30 22:24:36 +03:00
2021-10-21 02:27:05 +03:00
fread(f, config_addr, 0, f->size);
fclose(f);
2020-03-30 22:24:36 +03:00
2020-11-05 03:37:45 +03:00
return init_config(config_size);
}
#if bios == 1
2020-11-05 03:37:45 +03:00
int init_config_pxe(void) {
2021-08-12 08:40:29 +03:00
struct file_handle f;
if (tftp_open(&f, 0, 69, "limine.cfg")) {
2020-11-05 03:37:45 +03:00
return -1;
}
2021-08-12 08:40:29 +03:00
size_t config_size = f.size + 1;
config_addr = ext_mem_alloc(config_size);
fread(&f, config_addr, 0, f.size);
return init_config(config_size);
2020-11-05 03:37:45 +03:00
}
2021-03-02 12:23:43 +03:00
#endif
2020-11-05 03:37:45 +03:00
#define NOT_CHILD (-1)
#define DIRECT_CHILD 0
#define INDIRECT_CHILD 1
static int is_child(char *buf, size_t limit,
size_t current_depth, size_t index) {
if (!config_get_entry_name(buf, index, limit))
return NOT_CHILD;
if (strlen(buf) < current_depth + 1)
return NOT_CHILD;
for (size_t j = 0; j < current_depth; j++)
if (buf[j] != ':')
return NOT_CHILD;
if (buf[current_depth] == ':')
return INDIRECT_CHILD;
return DIRECT_CHILD;
}
static bool is_directory(char *buf, size_t limit,
size_t current_depth, size_t index) {
switch (is_child(buf, limit, current_depth + 1, index + 1)) {
default:
case NOT_CHILD:
return false;
case INDIRECT_CHILD:
2021-12-11 21:58:00 +03:00
*bad_config = true;
panic(true, "config: Malformed config file. Parentless child.");
case DIRECT_CHILD:
return true;
}
}
static struct menu_entry *create_menu_tree(struct menu_entry *parent,
size_t current_depth, size_t index) {
struct menu_entry *root = NULL, *prev = NULL;
for (size_t i = index; ; i++) {
static char name[64];
switch (is_child(name, 64, current_depth, i)) {
case NOT_CHILD:
return root;
case INDIRECT_CHILD:
continue;
case DIRECT_CHILD:
break;
}
struct menu_entry *entry = ext_mem_alloc(sizeof(struct menu_entry));
if (root == NULL)
root = entry;
config_get_entry_name(name, i, 64);
2020-12-01 04:52:03 +03:00
bool default_expanded = name[current_depth] == '+';
strcpy(entry->name, name + current_depth + default_expanded);
entry->parent = parent;
2021-07-08 17:57:40 +03:00
size_t entry_size;
char *config_entry = config_get_entry(&entry_size, i);
entry->body = ext_mem_alloc(entry_size + 1);
memcpy(entry->body, config_entry, entry_size);
entry->body[entry_size] = 0;
if (is_directory(name, 64, current_depth, i)) {
entry->sub = create_menu_tree(entry, current_depth + 1, i + 1);
2020-12-01 04:52:03 +03:00
entry->expanded = default_expanded;
2021-07-08 17:57:40 +03:00
}
char *comment = config_get_value(entry->body, 0, "COMMENT");
if (comment != NULL) {
entry->comment = comment;
}
if (prev != NULL)
prev->next = entry;
prev = entry;
}
}
struct menu_entry *menu_tree = NULL;
2020-11-05 03:37:45 +03:00
int init_config(size_t config_size) {
// remove windows carriage returns, if any
2020-07-02 17:14:09 +03:00
for (size_t i = 0; i < config_size; i++) {
if (config_addr[i] == '\r') {
2020-07-02 17:14:09 +03:00
for (size_t j = i; j < config_size - 1; j++)
config_addr[j] = config_addr[j+1];
2020-07-02 17:14:09 +03:00
config_size--;
}
}
2020-10-20 03:38:44 +03:00
config_ready = true;
menu_tree = create_menu_tree(NULL, 1, 0);
2020-03-30 22:24:36 +03:00
return 0;
}
bool config_get_entry_name(char *ret, size_t index, size_t limit) {
if (!config_ready)
return false;
2020-04-21 17:23:30 +03:00
char *p = config_addr;
for (size_t i = 0; i <= index; i++) {
while (*p != ':') {
if (!*p)
return false;
2020-04-21 17:23:30 +03:00
p++;
}
p++;
2020-05-27 10:24:59 +03:00
if ((p - 1) != config_addr && *(p - 2) != '\n')
i--;
2020-04-21 17:23:30 +03:00
}
p--;
2020-04-21 17:23:30 +03:00
size_t i;
for (i = 0; i < (limit - 1); i++) {
if (p[i] == SEPARATOR)
break;
ret[i] = p[i];
}
ret[i] = 0;
return true;
2020-04-21 17:23:30 +03:00
}
char *config_get_entry(size_t *size, size_t index) {
if (!config_ready)
return NULL;
char *ret;
2020-04-21 17:23:30 +03:00
char *p = config_addr;
for (size_t i = 0; i <= index; i++) {
while (*p != ':') {
if (!*p)
return NULL;
2020-04-21 17:23:30 +03:00
p++;
}
2020-04-21 19:27:05 +03:00
p++;
2020-05-27 10:24:59 +03:00
if ((p - 1) != config_addr && *(p - 2) != '\n')
i--;
2020-04-21 17:23:30 +03:00
}
do {
p++;
} while (*p != '\n');
ret = p;
2020-04-21 17:23:30 +03:00
cont:
2020-04-21 19:27:05 +03:00
while (*p != ':' && *p)
p++;
2020-04-21 17:23:30 +03:00
if (*p && *(p - 1) != '\n') {
p++;
goto cont;
}
*size = p - ret;
2020-04-21 17:23:30 +03:00
return ret;
2020-04-21 17:23:30 +03:00
}
static const char *lastkey;
struct conf_tuple config_get_tuple(const char *config, size_t index,
const char *key1, const char *key2) {
struct conf_tuple conf_tuple;
conf_tuple.value1 = config_get_value(config, index, key1);
if (conf_tuple.value1 == NULL) {
return (struct conf_tuple){0};
}
conf_tuple.value2 = config_get_value(lastkey, 0, key2);
const char *lk1 = lastkey;
const char *next_value1 = config_get_value(config, index + 1, key1);
const char *lk2 = lastkey;
if (conf_tuple.value2 != NULL && next_value1 != NULL) {
if ((uintptr_t)lk1 > (uintptr_t)lk2) {
conf_tuple.value2 = NULL;
}
}
return conf_tuple;
}
char *config_get_value(const char *config, size_t index, const char *key) {
2021-04-11 03:30:48 +03:00
if (!key || !config_ready)
2020-01-22 09:13:19 +03:00
return NULL;
if (config == NULL)
config = config_addr;
2020-01-22 09:13:19 +03:00
size_t key_len = strlen(key);
for (size_t i = 0; config[i]; i++) {
if (!strncmp(&config[i], key, key_len) && config[i + key_len] == '=') {
if (i && config[i - 1] != SEPARATOR)
2020-01-22 09:13:19 +03:00
continue;
2020-03-30 23:27:15 +03:00
if (index--)
continue;
2020-01-22 09:13:19 +03:00
i += key_len + 1;
size_t value_len;
for (value_len = 0;
config[i + value_len] != SEPARATOR && config[i + value_len];
value_len++);
char *buf = ext_mem_alloc(value_len + 1);
memcpy(buf, config + i, value_len);
lastkey = config + i;
2020-01-22 09:13:19 +03:00
return buf;
}
}
return NULL;
}