Workshop o mikrokontrolérech na SKSP 2024.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

261 lines
5.8 KiB

3 months ago
/*
* Insane tester of reading configuration files
*
* (c) 2006 Robert Spalek <robert@ucw.cz>
* (c) 2012--2014 Martin Mares <mj@ucw.cz>
* (c) 2014 Pavel Charvat <pchar@ucw.cz>
*/
#include <ucw/lib.h>
#include <ucw/conf.h>
#include <ucw/getopt.h>
#include <ucw/clists.h>
#include <ucw/fastbuf.h>
#include <ucw/gary.h>
#include <ucw/xtypes.h>
#include <stdlib.h>
#include <stdio.h>
#include <time.h>
static int verbose;
static int reload;
struct sub_sect_1 {
cnode n;
char *name;
time_t t;
char *level;
int confidence[2];
double *list;
};
static struct sub_sect_1 sec1 = { {}, "Charlie", 0, "WBAFC", { 0, -1}, NULL };
static char *
init_sec_1(struct sub_sect_1 *s)
{
if (s == &sec1) // this is a static variable; skip clearing
return NULL;
s->name = "unknown";
s->level = "default";
s->confidence[0] = 5;
s->confidence[1] = 6;
// leave s->list==NULL
return NULL;
}
static char *
commit_sec_1(struct sub_sect_1 *s)
{
if (s->confidence[0] < 0 || s->confidence[0] > 10)
return "Well, this can't be";
return NULL;
}
static char *
time_parser(uint number, char **pars, time_t *ptr)
{
*ptr = number ? atoi(pars[0]) : time(NULL);
return NULL;
}
static struct cf_section cf_sec_1 = {
CF_TYPE(struct sub_sect_1),
CF_INIT(init_sec_1),
CF_COMMIT(commit_sec_1),
#define F(x) PTR_TO(struct sub_sect_1, x)
CF_ITEMS {
CF_STRING("name", F(name)),
//CF_PARSER("t", F(t), time_parser, 0),
CF_STRING("level", F(level)),
CF_INT_ARY("confidence", F(confidence[0]), 2), // XXX: the [0] is needed for the sake of type checking
CF_DOUBLE_DYN("list", F(list), 100),
CF_END
}
#undef F
};
static uint nr1 = 15;
static int *nrs1;
static int nrs2[5];
static char *str1 = "no worries";
static char **str2;
static u64 u1 = 0xCafeBeefDeadC00ll;
static double d1 = -1.1;
static clist secs;
static time_t t1, t2;
static u32 ip;
static int *look;
static u16 numbers[10] = { 2, 100, 1, 5 };
static u32 bitmap1 = 0xff;
static u32 bitmap2 = 3;
static intmax_t intmax;
static char *
parse_u16(char *string, u16 *ptr)
{
uint a;
char *msg = cf_parse_int(string, &a);
if (msg)
return msg;
if (a >= (1<<16))
return "Come on, man, this doesn't fit to 16 bits";
*ptr = a;
return NULL;
}
static void
dump_u16(struct fastbuf *fb, u16 *ptr)
{
bprintf(fb, "%d ", *ptr);
}
static struct cf_user_type u16_type = {
.size = sizeof(u16),
.name = "u16",
.parser = (cf_parser1*) parse_u16,
.dumper = (cf_dumper1*) dump_u16
};
static char *
init_top(void *ptr UNUSED)
{
for (uint i=0; i<5; i++)
{
struct sub_sect_1 *s = xmalloc(sizeof(struct sub_sect_1)); // XXX: cannot by cf_malloc(), because it's deleted when cf_reload()'ed
cf_init_section("slaves", &cf_sec_1, s, 1);
s->confidence[1] = i;
clist_add_tail(&secs, &s->n);
}
return NULL;
}
static char *
commit_top(void *ptr UNUSED)
{
if (nr1 != 15)
return "Don't touch my variable!";
return NULL;
}
static const char * const alphabet[] = { "alpha", "beta", "gamma", "delta", NULL };
static struct cf_section cf_top = {
CF_INIT(init_top),
CF_COMMIT(commit_top),
CF_ITEMS {
CF_UINT("nr1", &nr1),
CF_INT_DYN("nrs1", &nrs1, 1000),
CF_INT_ARY("nrs2", nrs2, 5),
CF_STRING("str1", &str1),
CF_STRING_DYN("str2", &str2, 20),
CF_U64("u1", &u1),
CF_DOUBLE("d1", &d1),
CF_PARSER("FirstTime", &t1, time_parser, -1),
CF_PARSER("SecondTime", &t2, time_parser, 1),
CF_SECTION("master", &sec1, &cf_sec_1),
CF_LIST("slaves", &secs, &cf_sec_1),
CF_IP("ip", &ip),
CF_LOOKUP_DYN("look", &look, alphabet, 1000),
CF_USER_ARY("numbers", numbers, &u16_type, 10),
CF_XTYPE("intmax", &intmax, &xt_intmax),
CF_BITMAP_INT("bitmap1", &bitmap1),
CF_BITMAP_LOOKUP("bitmap2", &bitmap2, ((const char* const[]) {
"one", "two", "three", "four", "five", "six", "seven", "eight",
"nine", "ten", "eleven", "twelve", "thirteen", "fourteen", "fifteen", "seventeen",
"eighteen", "nineteen", "twenty", NULL // hidden joke here
})),
CF_END
}
};
static char short_opts[] = CF_SHORT_OPTS "rv";
static struct option long_opts[] = {
CF_LONG_OPTS
{"reload", 0, 0, 'r'},
{"verbose", 0, 0, 'v'},
{NULL, 0, 0, 0}
};
static char *help = "\
Usage: conf-test [ctxt] [nojournal] <options>\n\
\n\
Options:\n" CF_USAGE "\
-r, --reload\t\tReload configuration\n\
-v, --verbose\t\tBe verbose\n\
";
static void NONRET
usage(char *msg, ...)
{
va_list va;
va_start(va, msg);
if (msg)
vfprintf(stderr, msg, va);
fputs(help, stderr);
exit(1);
}
int
main(int argc, char *argv[])
{
log_init(argv[0]);
struct cf_context *cc = NULL, *prev = NULL;
// Special arguments which have to be parsed before cf_getopt()
while (argc > 1) {
if (!strcmp(argv[1], "ctxt")) {
cc = cf_new_context();
prev = cf_switch_context(cc);
argc--, argv++;
} else if (!strcmp(argv[1], "nojournal")) {
cf_set_journalling(0);
argc--, argv++;
} else
break;
}
cf_declare_section("top", &cf_top, 0);
cf_def_file = "ucw/conf-test.cf";
// Create and initialize dynamic arrays
GARY_INIT(nrs1, 6);
memcpy(nrs1, (int []) { 5, 5, 4, 3, 2, 1 }, 6 * sizeof(int));
GARY_INIT(str2, 2);
str2[0] = "Alice";
str2[1] = "Bob";
GARY_INIT(look, 2);
look[0] = 2;
look[1] = 1;
GARY_INIT(sec1.list, 3);
memcpy(sec1.list, (double []) { 1e4, -1e-4, 8 }, 3 * sizeof(double));
int opt;
while ((opt = cf_getopt(argc, argv, short_opts, long_opts, NULL)) >= 0)
switch (opt) {
case 'r': reload++; break;
case 'v': verbose++; break;
default: usage("unknown option %c\n", opt);
}
if (optind < argc)
usage("too many parameters (%d more)\n", argc-optind);
if (reload) {
cf_reload(NULL);
cf_reload(NULL);
}
if (verbose) {
struct fastbuf *out = bfdopen(1, 1<<14);
cf_dump_sections(out);
bclose(out);
}
if (cc) {
cf_switch_context(prev);
cf_delete_context(cc);
}
return 0;
}