e16/src/econfig.c

475 lines
10 KiB
C

/*
* Copyright (C) 2004-2015 Kim Woelders
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to
* deal in the Software without restriction, including without limitation the
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
* sell copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies of the Software, its documentation and marketing & publicity
* materials, and acknowledgment shall be given in the documentation, materials
* and software packages that this Software was used.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
* THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
* IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
#include "config.h"
#include <ctype.h>
#include "E.h"
#include "econfig.h"
#include "emodule.h"
/*
* Braindead flat ASCII config file implementation
*/
typedef struct {
char *key;
char *value;
} ECfgFileItem;
typedef struct {
FILE *fs;
int nitms;
ECfgFileItem *pitms;
} ECfgFile;
static void CfgItemSetFromString(const CfgItem * ci, const char *str,
int set_dflt);
static ECfgFile *
e16_db_open(const char *name)
{
ECfgFile *ecf;
FILE *fs;
fs = fopen(name, "w");
if (!fs)
return NULL;
ecf = ECALLOC(ECfgFile, 1);
if (!ecf)
goto done;
ecf->fs = fs;
done:
if (!ecf)
fclose(fs);
return ecf;
}
static ECfgFile *
e16_db_open_read(const char *name)
{
ECfgFile *ecf;
FILE *fs;
char buf[4096], key[128], *s;
int i, len;
fs = fopen(name, "r");
if (!fs)
return NULL;
ecf = ECALLOC(ECfgFile, 1);
if (!ecf)
goto done;
for (;;)
{
s = fgets(buf, sizeof(buf), fs);
if (!s)
break;
/* Strip comment and trailing whitespace */
i = strcspn(s, "#\r\n");
for (; i > 0; i--)
if (!isspace(s[i - 1]))
break;
s[i] = '\0';
len = 0;
i = sscanf(s, "%100s = %n", key, &len);
if (i <= 0 || len <= 0)
continue; /* Ignore bad format */
i = ecf->nitms++;
ecf->pitms = EREALLOC(ECfgFileItem, ecf->pitms, ecf->nitms);
ecf->pitms[i].key = Estrdup(key);
ecf->pitms[i].value = Estrdup(s + len);
}
done:
fclose(fs);
return ecf;
}
static void
e16_db_close(ECfgFile *ecf)
{
int i;
if (ecf->pitms)
{
for (i = 0; i < ecf->nitms; i++)
{
Efree(ecf->pitms[i].key);
Efree(ecf->pitms[i].value);
}
Efree(ecf->pitms);
}
if (ecf->fs)
fclose(ecf->fs);
Efree(ecf);
}
static const char *
ECfgFileFindValue(ECfgFile *ecf, const char *key)
{
int i;
for (i = 0; i < ecf->nitms; i++)
if (!strcmp(key, ecf->pitms[i].key))
return ecf->pitms[i].value;
return NULL;
}
/*
* Configuration handling.
*/
static void
CfgItemLoad(ECfgFile *ecf, const char *prefix, const CfgItem *ci, int dflt)
{
char buf[1024];
const char *name = buf;
const char *value;
if (prefix)
Esnprintf(buf, sizeof(buf), "%s.%s", prefix, ci->name);
else
name = ci->name;
if (EDebug(EDBUG_TYPE_CONFIG) > 1)
Eprintf("%s: %s\n", __func__, name);
if (!ci->ptr)
return;
value = (ecf) ? ECfgFileFindValue(ecf, name) : NULL;
if (value || dflt)
CfgItemSetFromString(ci, value, 1);
}
static void
CfgItemSave(ECfgFile *ecf, const char *prefix, const CfgItem *ci)
{
char buf[1024], buf2[1024];
const char *name = buf;
if (prefix)
Esnprintf(buf, sizeof(buf), "%s.%s", prefix, ci->name);
else
name = ci->name;
if (EDebug(EDBUG_TYPE_CONFIG) > 1)
Eprintf("%s: %s\n", __func__, name);
if (!ci->ptr)
return;
CfgItemToString(ci, buf2, sizeof(buf2));
fprintf(ecf->fs, "%s = %s\n", name, buf2);
}
static const char *
ConfigurationGetFile(char *buf, int len)
{
Esnprintf(buf, len, "%s.cfg", EGetSavePrefix());
return buf;
}
static void
_ConfigurationLoad(const char *file, int dflt)
{
int i, nml, j, ncl;
const EModule *const *pml, *pm;
const CfgItem *pcl;
ECfgFile *ecf;
if (EDebug(EDBUG_TYPE_CONFIG))
Eprintf("%s\n", __func__);
ecf = e16_db_open_read(file);
/* NB! We have to assign the defaults even if it doesn't exist */
if (!ecf && !dflt)
return;
/* Load module configs */
MODULE_LIST_GET(pml, nml);
for (i = 0; i < nml; i++)
{
pm = pml[i];
ncl = pm->cfg.num;
pcl = pm->cfg.lst;
for (j = 0; j < ncl; j++)
CfgItemLoad(ecf, pm->name, pcl + j, dflt);
}
MODULE_LIST_FREE(pml);
if (ecf)
e16_db_close(ecf);
}
void
ConfigurationLoad(void)
{
char buf[4096];
memset(&Conf, 0, sizeof(EConf));
_ConfigurationLoad(ConfigurationGetFile(buf, sizeof(buf)), 1);
}
void
ConfigurationSave(void)
{
int i, nml, j, ncl;
const EModule *const *pml, *pm;
const CfgItem *pcl;
char buf[4096];
ECfgFile *ecf;
if (EDebug(EDBUG_TYPE_CONFIG))
Eprintf("%s\n", __func__);
ecf = e16_db_open(ConfigurationGetFile(buf, sizeof(buf)));
if (!ecf)
return;
/* Load module configs */
MODULE_LIST_GET(pml, nml);
for (i = 0; i < nml; i++)
{
pm = pml[i];
ncl = pm->cfg.num;
pcl = pm->cfg.lst;
for (j = 0; j < ncl; j++)
CfgItemSave(ecf, pm->name, pcl + j);
}
MODULE_LIST_FREE(pml);
e16_db_close(ecf);
}
const CfgItem *
CfgItemFind(const CfgItem *pcl, int ncl, const char *name)
{
int i;
for (i = 0; i < ncl; i++, pcl++)
if (!strcmp(name, pcl->name))
return pcl;
return NULL;
}
static void
CfgItemSetFromString(const CfgItem *ci, const char *str, int set_dflt)
{
int ival;
#ifdef ITEM_TYPE_FLOAT
int n;
float fval;
#endif
char *ptr;
ptr = (char *)str;
switch (ci->type)
{
case ITEM_TYPE_BOOL:
ival = (str) ? strtoul(str, &ptr, 0) : 0;
if (ptr <= str)
{
if (!set_dflt)
break;
ival = (ci->dflt) ? 1 : 0;
}
*((char *)ci->ptr) = ival;
break;
case ITEM_TYPE_INT:
case ITEM_TYPE_HEX:
ival = (str) ? strtoul(str, &ptr, 0) : 0;
if (ptr <= str)
{
if (!set_dflt)
break;
ival = ci->dflt;
}
*((int *)ci->ptr) = ival;
break;
#ifdef ITEM_TYPE_FLOAT
case ITEM_TYPE_FLOAT:
n = (str) ? sscanf(str, "%f", &fval) : 0;
if (n <= 0)
{
if (!set_dflt)
break;
fval = ci->dflt;
}
*((float *)ci->ptr) = fval;
break;
#endif
case ITEM_TYPE_STRING:
Efree(*(char **)ci->ptr);
if (str && *str == '\0')
str = NULL;
*((char **)ci->ptr) = Estrdup(str);
break;
}
}
void
CfgItemToString(const CfgItem *ci, char *buf, int len)
{
buf[0] = '\0';
switch (ci->type)
{
case ITEM_TYPE_BOOL:
Esnprintf(buf, len, "%d", *((char *)ci->ptr));
break;
case ITEM_TYPE_INT:
Esnprintf(buf, len, "%d", *((int *)ci->ptr));
break;
case ITEM_TYPE_HEX:
Esnprintf(buf, len, "%#x", *((unsigned int *)ci->ptr));
break;
#ifdef ITEM_TYPE_FLOAT
case ITEM_TYPE_FLOAT:
Esnprintf(buf, len, "%.3f", *((float *)ci->ptr));
break;
#endif
case ITEM_TYPE_STRING:
if (*((char **)ci->ptr))
Esnprintf(buf, len, "%s", *((char **)ci->ptr));
break;
}
}
int
CfgItemListNamedItemSet(const CfgItem *pcl, int ncl, const char *item,
const char *value)
{
const CfgItem *ci;
ci = CfgItemFind(pcl, ncl, item);
if (!ci)
return -1;
if (ci->func)
ci->func(ci->ptr, value);
else
CfgItemSetFromString(ci, value, 0);
return 0;
}
#if 0 /* Unused */
int
CfgItemListNamedItemToString(const CfgItem *pcl, int ncl, const char *item,
char *buf, int len)
{
const CfgItem *ci;
ci = CfgItemFind(pcl, ncl, item);
if (!ci)
return -1;
CfgItemToString(ci, buf, len);
return 0;
}
#endif
/*
* Set <module>.<item> <value>
*/
void
ConfigurationSet(const char *params)
{
const char *p;
char name[1024];
char item[1024];
unsigned int len;
if (!params)
return;
p = strchr(params, '.');
if (!p)
{
Eprintf("%s: missed: %s\n", __func__, params);
return;
}
len = p - params;
if (len >= sizeof(name))
len = sizeof(name) - 1;
memcpy(name, params, len);
name[len] = '\0';
p++;
len = 0;
sscanf(p, "%1000s %n", item, &len);
p += len;
ModuleConfigSet(name, item, p);
/* Save changed configuration */
autosave();
}
/*
* Show <module>.<item> <value>
*/
void
ConfigurationShow(const char *params)
{
const char *p;
char name[1024];
char item[1024];
unsigned int len;
/* No parameters - All */
if (!params || params[0] == '\0')
{
ModulesConfigShow();
return;
}
/* No '.' - All for module */
p = strchr(params, '.');
if (!p)
{
ModuleConfigShow(params, NULL);
return;
}
/* Specific module, specific item. */
len = p - params;
if (len >= sizeof(name))
len = sizeof(name) - 1;
memcpy(name, params, len);
name[len] = '\0';
p++;
sscanf(p, "%s", item);
ModuleConfigShow(name, item);
}