* This file contains some common code for the uci library
*/
+#define _GNU_SOURCE
#include <sys/types.h>
#include <stdbool.h>
#include <string.h>
#include <stdlib.h>
#include <stdio.h>
+#include "uci.h"
+#include "err.h"
+
+static const char *uci_confdir = UCI_CONFDIR;
+static const char *uci_savedir = UCI_SAVEDIR;
+
+static const char *uci_errstr[] = {
+ [UCI_OK] = "Success",
+ [UCI_ERR_MEM] = "Out of memory",
+ [UCI_ERR_INVAL] = "Invalid argument",
+ [UCI_ERR_NOTFOUND] = "Entry not found",
+ [UCI_ERR_IO] = "I/O error",
+ [UCI_ERR_PARSE] = "Parse error",
+ [UCI_ERR_DUPLICATE] = "Duplicate entry",
+ [UCI_ERR_UNKNOWN] = "Unknown error",
+};
-#include "libuci.h"
-
-#define DEBUG
+#include "util.c"
+#include "list.c"
+#include "file.c"
-#ifdef DEBUG
-#define DPRINTF(...) fprintf(stderr, __VA_ARGS__)
-#else
-#define DPRINTF(...)
-#endif
-/*
- * throw an uci exception and store the error number
- * in the context.
- */
-#define UCI_THROW(ctx, err) do { \
- longjmp(ctx->trap, err); \
-} while (0)
+/* exported functions */
+struct uci_context *uci_alloc_context(void)
+{
+ struct uci_context *ctx;
-/*
- * store the return address for handling exceptions
- * needs to be called in every externally visible library function
- *
- * NB: this does not handle recursion at all. Calling externally visible
- * functions from other uci functions is only allowed at the end of the
- * calling function.
- */
-#define UCI_HANDLE_ERR(ctx) do { \
- int __val; \
- if (!ctx) \
- return UCI_ERR_INVAL; \
- __val = setjmp(ctx->trap); \
- if (__val) { \
- ctx->errno = __val; \
- return __val; \
- } \
-} while (0)
+ ctx = (struct uci_context *) malloc(sizeof(struct uci_context));
+ memset(ctx, 0, sizeof(struct uci_context));
+ uci_list_init(&ctx->root);
+ uci_list_init(&ctx->history_path);
+ ctx->flags = UCI_FLAG_STRICT;
-/*
- * check the specified condition.
- * throw an invalid argument exception if it's false
- */
-#define UCI_ASSERT(ctx, expr) do { \
- if (!(expr)) { \
- DPRINTF("[%s:%d] Assertion failed\n", __FILE__, __LINE__); \
- UCI_THROW(ctx, UCI_ERR_INVAL); \
- } \
-} while (0)
-
-
-static char *uci_errstr[] = {
- [UCI_OK] = "Success",
- [UCI_ERR_MEM] = "Out of memory",
- [UCI_ERR_INVAL] = "Invalid argument",
- [UCI_ERR_NOTFOUND] = "Entry not found",
- [UCI_ERR_PARSE] = "Parse error",
- [UCI_ERR_UNKNOWN] = "Unknown error",
-};
+ ctx->confdir = (char *) uci_confdir;
+ ctx->savedir = (char *) uci_savedir;
+ return ctx;
+}
-/*
- * UCI wrapper for malloc, which uses exception handling
- */
-static void *uci_malloc(struct uci_context *ctx, size_t size)
+void uci_free_context(struct uci_context *ctx)
{
- void *ptr;
-
- ptr = malloc(size);
- if (!ptr)
- UCI_THROW(ctx, UCI_ERR_MEM);
- memset(ptr, 0, size);
-
- return ptr;
+ struct uci_element *e, *tmp;
+
+ if (ctx->confdir != uci_confdir)
+ free(ctx->confdir);
+ if (ctx->savedir != uci_savedir)
+ free(ctx->savedir);
+
+ UCI_TRAP_SAVE(ctx, ignore);
+ uci_cleanup(ctx);
+ uci_foreach_element_safe(&ctx->root, tmp, e) {
+ struct uci_package *p = uci_to_package(e);
+ uci_free_package(&p);
+ }
+ uci_foreach_element_safe(&ctx->history_path, tmp, e) {
+ uci_free_element(e);
+ }
+ free(ctx);
+ UCI_TRAP_RESTORE(ctx);
+
+ignore:
+ return;
}
-/*
- * UCI wrapper for realloc, which uses exception handling
- */
-static void *uci_realloc(struct uci_context *ctx, void *ptr, size_t size)
+int uci_add_history_path(struct uci_context *ctx, const char *dir)
{
- ptr = realloc(ptr, size);
- if (!ptr)
- UCI_THROW(ctx, UCI_ERR_MEM);
+ struct uci_element *e;
- return ptr;
+ UCI_HANDLE_ERR(ctx);
+ UCI_ASSERT(ctx, dir != NULL);
+ e = uci_alloc_generic(ctx, UCI_TYPE_PATH, dir, sizeof(struct uci_element));
+ uci_list_add(&ctx->history_path, &e->list);
+
+ return 0;
}
-#include "hash.c"
-#include "parse.c"
+int uci_set_confdir(struct uci_context *ctx, const char *dir)
+{
+ char *cdir;
+
+ UCI_HANDLE_ERR(ctx);
+ UCI_ASSERT(ctx, dir != NULL);
-/* externally visible functions */
+ cdir = uci_strdup(ctx, dir);
+ if (ctx->confdir != uci_confdir)
+ free(ctx->confdir);
+ ctx->confdir = cdir;
+ return 0;
+}
-struct uci_context *uci_alloc(void)
+int uci_set_savedir(struct uci_context *ctx, const char *dir)
{
- struct uci_context *ctx;
-
- ctx = (struct uci_context *) malloc(sizeof(struct uci_context));
- memset(ctx, 0, sizeof(struct uci_context));
-
- return ctx;
+ char *sdir;
+
+ UCI_HANDLE_ERR(ctx);
+ UCI_ASSERT(ctx, dir != NULL);
+
+ sdir = uci_strdup(ctx, dir);
+ if (ctx->savedir != uci_savedir)
+ free(ctx->savedir);
+ ctx->savedir = sdir;
+ return 0;
}
int uci_cleanup(struct uci_context *ctx)
{
UCI_HANDLE_ERR(ctx);
- uci_parse_cleanup(ctx);
+ uci_file_cleanup(ctx);
return 0;
}
-void uci_perror(struct uci_context *ctx, const char *str)
+void uci_perror(struct uci_context *ctx, const char *prefix)
{
int err;
err = UCI_ERR_INVAL;
else
err = ctx->errno;
-
+
if ((err < 0) || (err >= UCI_ERR_LAST))
err = UCI_ERR_UNKNOWN;
+ if (prefix)
+ fprintf(stderr, "%s: ", prefix);
+ if (ctx->func)
+ fprintf(stderr, "%s: ", ctx->func);
+
switch (err) {
case UCI_ERR_PARSE:
if (ctx->pctx) {
- fprintf(stderr, "%s: %s at line %d, byte %d\n", str, uci_errstr[err], ctx->pctx->line, ctx->pctx->byte);
+ fprintf(stderr, "%s (%s) at line %d, byte %d\n", uci_errstr[err], (ctx->pctx->reason ? ctx->pctx->reason : "unknown"), ctx->pctx->line, ctx->pctx->byte);
break;
}
/* fall through */
default:
- fprintf(stderr, "%s: %s\n", str, uci_errstr[err]);
+ fprintf(stderr, "%s\n", uci_errstr[err]);
break;
}
}