add libucimap
[project/uci.git] / ucimap.h
1 /*
2 * ucimap - library for mapping uci sections into data structures
3 * Copyright (C) 2008 Felix Fietkau <nbd@openwrt.org>
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License version 2
7 * as published by the Free Software Foundation
8 *
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
13 */
14 #include <stdbool.h>
15 #include "uci_list.h"
16 #include "uci.h"
17
18 #ifndef ARRAY_SIZE
19 #define ARRAY_SIZE(x) (sizeof(x) / sizeof((x)[0]))
20 #endif
21
22 #define BITFIELD_SIZE(_fields) (((_fields) / 8) + 1)
23
24 #define CLR_BIT(_name, _bit) do { \
25 _name[(_bit) / 8] &= ~(1 << ((_bit) % 8)); \
26 } while (0)
27
28 #define SET_BIT(_name, _bit) do { \
29 _name[(_bit) / 8] |= (1 << ((_bit) % 8)); \
30 } while (0)
31
32 #define TEST_BIT(_name, _bit) \
33 (_name[(_bit) / 8] & (1 << ((_bit) % 8)))
34
35 #define OPTMAP_OPTION(_maptype, _type, _field, ...) \
36 { \
37 .type = _maptype, \
38 .name = #_field, \
39 .offset = offsetof(_type, _field), \
40 __VA_ARGS__ \
41 }
42
43 struct uci_sectmap;
44 struct uci_optmap;
45
46 struct uci_map {
47 struct uci_sectmap *sections;
48 unsigned int n_sections;
49 struct list_head sdata;
50
51 void *priv; /* user data */
52 };
53
54 enum ucimap_type {
55 UCIMAP_STRING,
56 UCIMAP_BOOL,
57 UCIMAP_INT,
58 };
59
60 /* ucimap internal */
61 struct uci_sectmap_data {
62 struct list_head list;
63 struct uci_sectmap *sm;
64 const char *section_name;
65 unsigned long allocmap_len;
66
67 /* list of allocations done by ucimap */
68 void **allocmap;
69
70 /* map for changed fields */
71 unsigned char *cmap;
72 };
73
74 struct uci_sectmap {
75 /* type string for the uci section */
76 const char *type;
77
78 /* length of the struct to map into */
79 unsigned int alloc_len;
80
81 /* give the caller time to initialize the preallocated struct */
82 int (*init_section)(struct uci_map *map, void *section, struct uci_section *s);
83
84 /* pass the fully processed struct to the callback after the section end */
85 int (*add_section)(struct uci_map *map, void *section);
86
87 /* let the callback clean up its own stuff in the section */
88 int (*free_section)(struct uci_map *map, void *section);
89
90 /* list of option mappings for this section */
91 struct uci_optmap *options;
92 unsigned int n_options;
93 };
94
95 struct uci_optmap {
96 unsigned int offset;
97 const char *name;
98 enum ucimap_type type;
99 union {
100 struct {
101 int base;
102 int min;
103 int max;
104 } i;
105 struct {
106 int maxlen;
107 } s;
108 } data;
109 };
110
111 extern int ucimap_init(struct uci_map *map);
112 extern void ucimap_cleanup(struct uci_map *map);
113 extern void ucimap_set_changed(void *section, void *field);
114 extern int ucimap_store_section(struct uci_map *map, struct uci_package *p, void *section);
115 extern void ucimap_parse(struct uci_map *map, struct uci_package *pkg);
116