summaryrefslogtreecommitdiffhomepage
path: root/src/loader/cfg.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/loader/cfg.c')
-rw-r--r--src/loader/cfg.c173
1 files changed, 173 insertions, 0 deletions
diff --git a/src/loader/cfg.c b/src/loader/cfg.c
new file mode 100644
index 0000000..37794e6
--- /dev/null
+++ b/src/loader/cfg.c
@@ -0,0 +1,173 @@
+// MIT License, Copyright (c) 2021 Marvin Borner
+
+#include <cfg.h>
+#include <dev.h>
+#include <lib.h>
+#include <log.h>
+#include <pnc.h>
+
+// Keys
+#define TIMEOUT "TIMEOUT"
+#define PATH "PATH"
+
+// Config struct (gettable using cfg_get)
+struct {
+ u32 timeout;
+ struct {
+ u8 exists : 1;
+ char name[64];
+ char path[64];
+ } elem[16]; // Up to 16 different selections
+} cfg = { 0 };
+
+// Config file contents (if found)
+static char file[1024] = { 0 };
+
+// Find config file
+static u8 cfg_find(struct dev *dev)
+{
+ if (!dev->fs.read)
+ return 0; // No fs found or not readable - continue!
+
+ s32 res = dev->fs.read("/boot/segelboot.cfg", file, 0, sizeof(file), dev);
+ if (res > 0)
+ return 1; // Break foreach
+
+ // Continue foreach
+ return 0;
+}
+
+// Checks if index is appropriate as some key/value need to be in element
+static void cfg_in_element(u8 index)
+{
+ if (index == 0xff)
+ panic("No element name given\n");
+}
+
+// Add/overwrite value by key and element index
+static void cfg_add(u8 index, enum cfg_key key, const char *value)
+{
+ cfg.elem[index].exists = 1;
+
+ switch (key) {
+ case CFG_NAME:
+ cfg_in_element(index);
+ strlcpy(cfg.elem[index].name, value, sizeof(cfg.elem[index].name));
+ break;
+ case CFG_TIMEOUT:
+ cfg.timeout = atoi(value);
+ break;
+ case CFG_PATH:
+ cfg_in_element(index);
+ strlcpy(cfg.elem[index].path, value, sizeof(cfg.elem[index].path));
+ break;
+ case CFG_NONE:
+ default:
+ panic("Invalid config\n");
+ }
+}
+
+// TODO: This code is kind of messy
+// Structure per line: KEY=VALUE
+static void cfg_parse(void)
+{
+ // Element index
+ u8 elem = 0xff;
+
+ // Value per key
+ char value[64] = { 0 };
+ u8 value_index = 0;
+
+ // States
+ enum cfg_key current = CFG_NONE; // Value key type
+ u8 state = 0; // 0 is key, 1 is value, 2 is elem
+
+ const char *start = file; // Start is at the beginning of the key
+ for (const char *p = start; *p; p++) {
+ if (state == 0) {
+ // We're at key parsing
+ if (*p == '\n') { // A key can't just end but ok
+ start = p + 1;
+ } else if (*p == '#') {
+ state = 2; // Let's parse the element name
+ p++;
+ continue;
+ } else if (*p != '=') {
+ continue;
+ }
+
+ // The key is now at start until p
+ u8 diff = p - start;
+
+ // Timeout key
+ if (diff == sizeof(TIMEOUT) - 1 && memcmp(start, TIMEOUT, diff) == 0) {
+ current = CFG_TIMEOUT;
+ state = 1;
+ continue;
+ }
+
+ // Path key
+ if (diff == sizeof(PATH) - 1 && memcmp(start, PATH, diff) == 0) {
+ current = CFG_PATH;
+ state = 1;
+ continue;
+ }
+ } else if (state == 1) {
+ // We're at value parsing
+ assert(value_index + 1 < (u8)sizeof(value));
+ if (*p == '\n') { // Finished
+ value[value_index] = 0;
+ cfg_add(elem, current, value);
+ value_index = 0;
+ state = 0;
+ p--; // Repeat parse normally
+ } else {
+ value[value_index++] = *p;
+ }
+ } else if (state == 2) {
+ // We're at element name parsing
+ assert(value_index + 1 < (u8)sizeof(value));
+ if (*p == '\n') { // Finished
+ elem = elem == 0xff ? 0 : elem + 1;
+ value[value_index] = 0;
+ cfg_add(elem, CFG_NAME, value);
+ value_index = 0;
+ state = 0;
+ p--; // Repeat parse normally
+ } else {
+ value[value_index++] = *p;
+ }
+ }
+ }
+}
+
+const void *cfg_get(u8 index, enum cfg_key key)
+{
+ switch (key) {
+ case CFG_NAME:
+ return &cfg.elem[index].path;
+ case CFG_TIMEOUT:
+ return &cfg.timeout;
+ case CFG_PATH:
+ return &cfg.elem[index].path;
+ case CFG_NONE:
+ default:
+ return NULL;
+ }
+}
+
+void cfg_print(void)
+{
+ log("[CFG] Global: %d\n", cfg.timeout);
+
+ for (u8 i = 0; i < COUNT(cfg.elem) && cfg.elem[i].exists; i++)
+ log("[CFG] Element: %s at %s\n", cfg.elem[i].name, cfg.elem[i].path);
+}
+
+void cfg_exec(void)
+{
+ dev_foreach(DEV_DISK, &cfg_find);
+ if (!file[0])
+ panic("No config found\n");
+ cfg_parse();
+}