Commit ab244401 authored by David Gibson's avatar David Gibson

strmap: Convert to using TCON_WRAP() instead of plain TCON()

The usual way of construction strmap objects is to use the STRMAP_MEMBERS()
macro which expands to both a raw strmap structure and a tcon type canary.
However, the tcon type canary involves a flexible array member which means
that in standard C99 STRMAP_MEMBERS() must appear only at the end of a
structure definition.  But worse, that structure can then only appear at
the end of any other structure it is included in, which is pretty
inconvenient for the intended purpose of creating type specific strmaps.

gcc extensions allow this to work (somehow), but clang complains loudly
about it.

The tcon module already includes the TCON_WRAP() mechanism, which already
provides this same sort of type-specific definitions in a more general way.
So convert strmap (and its users) to that approach.

This removes STRMAP_MEMBERS() entirely, breaking compatibility.  I'm hoping
strmap is used in few enough places that we can get away with that.
Signed-off-by: default avatarDavid Gibson <david@gibson.dropbear.id.au>
parent 0229ddbf
...@@ -213,7 +213,7 @@ static struct cdump_type *get_type(struct cdump_definitions *defs, ...@@ -213,7 +213,7 @@ static struct cdump_type *get_type(struct cdump_definitions *defs,
enum cdump_type_kind kind, enum cdump_type_kind kind,
const char *name) const char *name)
{ {
struct cdump_map *m; cdump_map_t *m;
struct cdump_type *t; struct cdump_type *t;
switch (kind) { switch (kind) {
...@@ -609,7 +609,7 @@ static bool tok_take_enum(struct parse_state *ps) ...@@ -609,7 +609,7 @@ static bool tok_take_enum(struct parse_state *ps)
static bool gather_undefines(const char *name, static bool gather_undefines(const char *name,
struct cdump_type *t, struct cdump_type *t,
struct cdump_map *undefs) cdump_map_t *undefs)
{ {
if (!type_defined(t)) if (!type_defined(t))
strmap_add(undefs, name, t); strmap_add(undefs, name, t);
...@@ -618,15 +618,15 @@ static bool gather_undefines(const char *name, ...@@ -618,15 +618,15 @@ static bool gather_undefines(const char *name,
static bool remove_from_map(const char *name, static bool remove_from_map(const char *name,
struct cdump_type *t, struct cdump_type *t,
struct cdump_map *map) cdump_map_t *map)
{ {
strmap_del(map, name, NULL); strmap_del(map, name, NULL);
return true; return true;
} }
static void remove_undefined(struct cdump_map *map) static void remove_undefined(cdump_map_t *map)
{ {
struct cdump_map undefs; cdump_map_t undefs;
/* We can't delete inside iterator, so gather all the undefs /* We can't delete inside iterator, so gather all the undefs
* then remove them. */ * then remove them. */
......
...@@ -50,14 +50,12 @@ struct cdump_type { ...@@ -50,14 +50,12 @@ struct cdump_type {
}; };
/* The map of typenames to definitions */ /* The map of typenames to definitions */
struct cdump_map { typedef STRMAP(struct cdump_type *) cdump_map_t;
STRMAP_MEMBERS(struct cdump_type *);
};
struct cdump_definitions { struct cdump_definitions {
struct cdump_map enums; cdump_map_t enums;
struct cdump_map structs; cdump_map_t structs;
struct cdump_map unions; cdump_map_t unions;
}; };
/** /**
......
...@@ -29,7 +29,7 @@ ...@@ -29,7 +29,7 @@
* int main(int argc, char *argv[]) * int main(int argc, char *argv[])
* { * {
* size_t i; * size_t i;
* struct { STRMAP_MEMBERS(size_t); } map; * STRMAP(size_t) map;
* *
* strmap_init(&map); * strmap_init(&map);
* for (i = 1; i < argc; i++) * for (i = 1; i < argc; i++)
......
...@@ -21,7 +21,7 @@ struct strmap { ...@@ -21,7 +21,7 @@ struct strmap {
}; };
/** /**
* STRMAP_MEMBERS - declare members for a type-specific strmap. * STRMAP - declare a type-specific strmap
* @type: type for this map's values, or void * for any pointer. * @type: type for this map's values, or void * for any pointer.
* *
* You use this to create your own typed strmap for a particular type. * You use this to create your own typed strmap for a particular type.
...@@ -29,14 +29,11 @@ struct strmap { ...@@ -29,14 +29,11 @@ struct strmap {
* value! * value!
* *
* Example: * Example:
* struct strmap_intp { * STRMAP(int *) int_strmap;
* STRMAP_MEMBERS(int *); * strmap_init(&int_strmap);
* };
*/ */
#define STRMAP_MEMBERS(type) \ #define STRMAP(type) \
struct strmap raw; \ TCON_WRAP(struct strmap, type canary)
TCON(type canary)
/** /**
* strmap_init - initialize a string map (empty) * strmap_init - initialize a string map (empty)
...@@ -46,11 +43,11 @@ struct strmap { ...@@ -46,11 +43,11 @@ struct strmap {
* need this. * need this.
* *
* Example: * Example:
* struct strmap_intp map; * STRMAP(int *) map;
* *
* strmap_init(&map); * strmap_init(&map);
*/ */
#define strmap_init(map) strmap_init_(&(map)->raw) #define strmap_init(map) strmap_init_(tcon_unwrap(map))
static inline void strmap_init_(struct strmap *map) static inline void strmap_init_(struct strmap *map)
{ {
...@@ -65,7 +62,7 @@ static inline void strmap_init_(struct strmap *map) ...@@ -65,7 +62,7 @@ static inline void strmap_init_(struct strmap *map)
* if (!strmap_empty(&map)) * if (!strmap_empty(&map))
* abort(); * abort();
*/ */
#define strmap_empty(map) strmap_empty_(&(map)->raw) #define strmap_empty(map) strmap_empty_(tcon_unwrap(map))
static inline bool strmap_empty_(const struct strmap *map) static inline bool strmap_empty_(const struct strmap *map)
{ {
...@@ -85,7 +82,7 @@ static inline bool strmap_empty_(const struct strmap *map) ...@@ -85,7 +82,7 @@ static inline bool strmap_empty_(const struct strmap *map)
* printf("hello => %i\n", *val); * printf("hello => %i\n", *val);
*/ */
#define strmap_get(map, member) \ #define strmap_get(map, member) \
tcon_cast((map), canary, strmap_get_(&(map)->raw, (member))) tcon_cast((map), canary, strmap_get_(tcon_unwrap(map), (member)))
void *strmap_get_(const struct strmap *map, const char *member); void *strmap_get_(const struct strmap *map, const char *member);
/** /**
...@@ -107,7 +104,7 @@ void *strmap_get_(const struct strmap *map, const char *member); ...@@ -107,7 +104,7 @@ void *strmap_get_(const struct strmap *map, const char *member);
* printf("goodbye was already in the map\n"); * printf("goodbye was already in the map\n");
*/ */
#define strmap_add(map, member, value) \ #define strmap_add(map, member, value) \
strmap_add_(&tcon_check((map), canary, (value))->raw, \ strmap_add_(tcon_unwrap(tcon_check((map), canary, (value))), \
(member), (void *)(value)) (member), (void *)(value))
bool strmap_add_(struct strmap *map, const char *member, const void *value); bool strmap_add_(struct strmap *map, const char *member, const void *value);
...@@ -130,7 +127,7 @@ bool strmap_add_(struct strmap *map, const char *member, const void *value); ...@@ -130,7 +127,7 @@ bool strmap_add_(struct strmap *map, const char *member, const void *value);
* printf("goodbye was not in the map?\n"); * printf("goodbye was not in the map?\n");
*/ */
#define strmap_del(map, member, valuep) \ #define strmap_del(map, member, valuep) \
strmap_del_(&tcon_check_ptr((map), canary, valuep)->raw, \ strmap_del_(tcon_unwrap(tcon_check_ptr((map), canary, valuep)), \
(member), (void **)valuep) (member), (void **)valuep)
char *strmap_del_(struct strmap *map, const char *member, void **valuep); char *strmap_del_(struct strmap *map, const char *member, void **valuep);
...@@ -143,7 +140,7 @@ char *strmap_del_(struct strmap *map, const char *member, void **valuep); ...@@ -143,7 +140,7 @@ char *strmap_del_(struct strmap *map, const char *member, void **valuep);
* Example: * Example:
* strmap_clear(&map); * strmap_clear(&map);
*/ */
#define strmap_clear(map) strmap_clear_(&(map)->raw) #define strmap_clear(map) strmap_clear_(tcon_unwrap(map))
void strmap_clear_(struct strmap *map); void strmap_clear_(struct strmap *map);
...@@ -160,9 +157,7 @@ void strmap_clear_(struct strmap *map); ...@@ -160,9 +157,7 @@ void strmap_clear_(struct strmap *map);
* You should not alter the map within the @handle function! * You should not alter the map within the @handle function!
* *
* Example: * Example:
* struct strmap_intp { * typedef STRMAP(int *) strmap_intp;
* STRMAP_MEMBERS(int *);
* };
* static bool dump_some(const char *member, int *value, int *num) * static bool dump_some(const char *member, int *value, int *num)
* { * {
* // Only dump out num nodes. * // Only dump out num nodes.
...@@ -172,7 +167,7 @@ void strmap_clear_(struct strmap *map); ...@@ -172,7 +167,7 @@ void strmap_clear_(struct strmap *map);
* return true; * return true;
* } * }
* *
* static void dump_map(const struct strmap_intp *map) * static void dump_map(const strmap_intp *map)
* { * {
* int max = 100; * int max = 100;
* strmap_iterate(map, dump_some, &max); * strmap_iterate(map, dump_some, &max);
...@@ -181,7 +176,7 @@ void strmap_clear_(struct strmap *map); ...@@ -181,7 +176,7 @@ void strmap_clear_(struct strmap *map);
* } * }
*/ */
#define strmap_iterate(map, handle, arg) \ #define strmap_iterate(map, handle, arg) \
strmap_iterate_(&(map)->raw, \ strmap_iterate_(tcon_unwrap(map), \
typesafe_cb_cast(bool (*)(const char *, \ typesafe_cb_cast(bool (*)(const char *, \
void *, void *), \ void *, void *), \
bool (*)(const char *, \ bool (*)(const char *, \
...@@ -202,7 +197,7 @@ void strmap_iterate_(const struct strmap *map, ...@@ -202,7 +197,7 @@ void strmap_iterate_(const struct strmap *map,
* strmap_empty() on the returned pointer. * strmap_empty() on the returned pointer.
* *
* Example: * Example:
* static void dump_prefix(const struct strmap_intp *map, * static void dump_prefix(const strmap_intp *map,
* const char *prefix) * const char *prefix)
* { * {
* int max = 100; * int max = 100;
...@@ -214,10 +209,10 @@ void strmap_iterate_(const struct strmap *map, ...@@ -214,10 +209,10 @@ void strmap_iterate_(const struct strmap *map,
*/ */
#if HAVE_TYPEOF #if HAVE_TYPEOF
#define strmap_prefix(map, prefix) \ #define strmap_prefix(map, prefix) \
((const __typeof__(map))strmap_prefix_(&(map)->raw, (prefix))) ((const __typeof__(map))strmap_prefix_(tcon_unwrap(map), (prefix)))
#else #else
#define strmap_prefix(map, prefix) \ #define strmap_prefix(map, prefix) \
((const void *)strmap_prefix_(&(map)->raw, (prefix))) ((const void *)strmap_prefix_(tcon_unwrap(map), (prefix)))
#endif #endif
const struct strmap *strmap_prefix_(const struct strmap *map, const struct strmap *strmap_prefix_(const struct strmap *map,
......
...@@ -14,9 +14,7 @@ static bool find_string(const char *str, char *member, const char *cmp) ...@@ -14,9 +14,7 @@ static bool find_string(const char *str, char *member, const char *cmp)
int main(void) int main(void)
{ {
struct strmap_charp { STRMAP(char *) map;
STRMAP_MEMBERS(char *);
} map;
plan_tests(3); plan_tests(3);
......
...@@ -33,9 +33,7 @@ static bool dump(const char *member, char *value, bool *ok) ...@@ -33,9 +33,7 @@ static bool dump(const char *member, char *value, bool *ok)
int main(void) int main(void)
{ {
struct strmap_charp { STRMAP(char *) map;
STRMAP_MEMBERS(char *);
} map;
unsigned int i; unsigned int i;
char *str[NUM]; char *str[NUM];
bool dump_ok; bool dump_ok;
......
...@@ -24,11 +24,9 @@ static bool find_empty(const char *index, char *value, char *empty) ...@@ -24,11 +24,9 @@ static bool find_empty(const char *index, char *value, char *empty)
int main(void) int main(void)
{ {
struct map { typedef STRMAP(char *) map_t;
STRMAP_MEMBERS(char *); map_t map;
}; const map_t *sub;
struct map map;
const struct map *sub;
unsigned int i; unsigned int i;
char *str[NUM], *empty; char *str[NUM], *empty;
...@@ -56,9 +54,9 @@ int main(void) ...@@ -56,9 +54,9 @@ int main(void)
/* Everything */ /* Everything */
sub = strmap_prefix(&map, "0"); sub = strmap_prefix(&map, "0");
ok1(sub->raw.u.n == map.raw.u.n); ok1(tcon_unwrap(sub)->u.n == tcon_unwrap(&map)->u.n);
sub = strmap_prefix(&map, ""); sub = strmap_prefix(&map, "");
ok1(sub->raw.u.n == map.raw.u.n); ok1(tcon_unwrap(sub)->u.n == tcon_unwrap(&map)->u.n);
/* Single. */ /* Single. */
sub = strmap_prefix(&map, "00000000"); sub = strmap_prefix(&map, "00000000");
......
...@@ -4,9 +4,7 @@ ...@@ -4,9 +4,7 @@
int main(void) int main(void)
{ {
struct strmap_charp { STRMAP(char *) map;
STRMAP_MEMBERS(char *);
} map;
const char str[] = "hello"; const char str[] = "hello";
const char val[] = "there"; const char val[] = "there";
const char none[] = ""; const char none[] = "";
......
...@@ -35,12 +35,10 @@ ...@@ -35,12 +35,10 @@
#include <ccan/tal/path/path.h> #include <ccan/tal/path/path.h>
#include <ccan/strmap/strmap.h> #include <ccan/strmap/strmap.h>
struct ccanlint_map { typedef STRMAP(struct ccanlint *) ccanlint_map_t;
STRMAP_MEMBERS(struct ccanlint *);
};
int verbose = 0; int verbose = 0;
static struct ccanlint_map tests; static ccanlint_map_t tests;
bool safe_mode = false; bool safe_mode = false;
bool keep_results = false; bool keep_results = false;
bool non_ccan_deps = false; bool non_ccan_deps = false;
...@@ -273,7 +271,7 @@ static bool init_deps(const char *member, struct ccanlint *c, void *unused) ...@@ -273,7 +271,7 @@ static bool init_deps(const char *member, struct ccanlint *c, void *unused)
} }
static bool check_names(const char *member, struct ccanlint *c, static bool check_names(const char *member, struct ccanlint *c,
struct ccanlint_map *names) ccanlint_map_t *names)
{ {
if (!strmap_add(names, c->name, c)) if (!strmap_add(names, c->name, c))
err(1, "Duplicate name %s", c->name); err(1, "Duplicate name %s", c->name);
...@@ -282,7 +280,7 @@ static bool check_names(const char *member, struct ccanlint *c, ...@@ -282,7 +280,7 @@ static bool check_names(const char *member, struct ccanlint *c,
static void init_tests(void) static void init_tests(void)
{ {
struct ccanlint_map names; ccanlint_map_t names;
struct ccanlint **table; struct ccanlint **table;
size_t i, num; size_t i, num;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment