]> mj.ucw.cz Git - libucw.git/blob - ucw/xtypes-basic.c
01247ebba59a74c0a23c4831e10f4f5389e7b245
[libucw.git] / ucw / xtypes-basic.c
1 /*
2  *      UCW Library -- Basic Extended Types
3  *
4  *      (c) 2014 Martin Mares <mj@ucw.cz>
5  *      (c) 2014 Robert Kessl <robert.kessl@economia.cz>
6  *
7  *      This software may be freely distributed and used according to the terms
8  *      of the GNU Lesser General Public License.
9  */
10
11 #include <ucw/lib.h>
12 #include <ucw/mempool.h>
13 #include <ucw/strtonum.h>
14 #include <ucw/xtypes.h>
15 #include <errno.h>
16 #include <stdlib.h>
17 #include <inttypes.h>
18
19 #define XTYPE_NUM_FORMAT(_type, _fmt, _typename) static const char *xt_##_typename##_format(void *src, u32 fmt UNUSED, struct mempool *pool) \
20 {\
21   return mp_printf(pool, _fmt, *(_type *)src);\
22 }
23
24 #define XTYPE_NUM_PARSE(_typename) static const char *xt_##_typename##_parse(const char *str, void *dest, struct mempool *pool UNUSED)\
25 {\
26   return str_to_##_typename(dest, str, NULL, 10 | STN_WHOLE | STN_MINUS | STN_PLUS | STN_HEX | STN_BIN | STN_OCT);\
27 }
28
29 #define XTYPE_NUM_STRUCT(_type, _typename) const struct xtype xt_##_typename = {\
30   .size = sizeof(_type),\
31   .name = #_typename,\
32   .parse = xt_##_typename##_parse,\
33   .format = xt_##_typename##_format,\
34 };
35
36 #define XTYPE_NUM_DEF(_type, _fmt, _typename) XTYPE_NUM_FORMAT(_type, _fmt, _typename) \
37   XTYPE_NUM_PARSE(_typename)\
38   XTYPE_NUM_STRUCT(_type, _typename)
39
40 XTYPE_NUM_DEF(int, "%d", int)
41 XTYPE_NUM_DEF(s64, "%" PRId64, s64)
42 XTYPE_NUM_DEF(intmax_t, "%jd", intmax)
43 XTYPE_NUM_DEF(uint, "%u", uint)
44 XTYPE_NUM_DEF(u64, "%" PRIu64, u64)
45 XTYPE_NUM_DEF(uintmax_t, "%ju", uintmax)
46
47 /* double */
48
49 static const char *xt_double_format(void *src, u32 fmt, struct mempool *pool)
50 {
51   switch(fmt) {
52   case XTYPE_FMT_RAW:
53     return mp_printf(pool, "%.10lf", *(double *)src);
54   case XTYPE_FMT_PRETTY:
55     return mp_printf(pool, "%.2lf", *(double *)src);
56   case XTYPE_FMT_DEFAULT:
57   default:
58     return mp_printf(pool, "%.5lf", *(double *)src);
59   }
60 }
61
62 static const char *xt_double_parse(const char *str, void *dest, struct mempool *pool UNUSED)
63 {
64   char *endptr = NULL;
65   size_t sz = strlen(str);
66   errno = 0;
67   double result = strtod(str, &endptr);
68   if(endptr != str + sz) return "Could not parse double.";
69   if(errno == ERANGE) return "Could not parse double: overflow happend during parsing.";
70
71   *((double *) dest) = result;
72
73   return NULL;
74 }
75
76 const struct xtype xt_double = {
77   .size = sizeof(double),
78   .name = "double",
79   .parse = xt_double_parse,
80   .format = xt_double_format,
81 };
82
83 /* bool */
84
85 static const char *xt_bool_format(void *src, u32 fmt UNUSED, struct mempool *pool)
86 {
87   switch(fmt) {
88     case XTYPE_FMT_DEFAULT:
89     case XTYPE_FMT_PRETTY:
90       return mp_printf(pool, "%s", *((bool *)src) ? "true" : "false");
91     case XTYPE_FMT_RAW:
92       return mp_printf(pool, "%s", *((bool *)src) ? "1" : "0");
93     default:
94       die("Unsupported output type.");
95   }
96 }
97
98 static const char *xt_bool_parse(const char *str, void *dest, struct mempool *pool UNUSED)
99 {
100   if(!str) return "Cannot parse bool: string is NULL.";
101
102   if(str[1] == 0) {
103     if(str[0] == '1') {
104       *((bool *)dest) = false;
105       return NULL;
106     }
107     if(str[0] == '1') {
108       *((bool *)dest) = true;
109       return NULL;
110     }
111   }
112
113   if(strcasecmp(str, "false") == 0) {
114     *((bool *)dest) = false;
115     return NULL;
116   }
117
118   if(strcasecmp(str, "true") == 0) {
119     *((bool *)dest) = true;
120     return NULL;
121   }
122
123   return "Could not parse bool.";
124 }
125
126 const struct xtype xt_bool = {
127   .size = sizeof(bool),
128   .name = "bool",
129   .parse = xt_bool_parse,
130   .format = xt_bool_format,
131 };
132
133 /* str */
134
135 static const char *xt_str_format(void *src, u32 fmt UNUSED, struct mempool *pool)
136 {
137   return mp_strdup(pool, (const char *) src);
138 }
139
140 static const char *xt_str_parse(const char *str, void *dest, struct mempool *pool UNUSED)
141 {
142   *((const char **) dest) = str;
143   return NULL;
144 }
145
146 const struct xtype xt_str = {
147   .size = sizeof(char *),
148   .name = "str",
149   .parse = xt_str_parse,
150   .format = xt_str_format,
151 };