]> CyberLeo.Net >> Repos - FreeBSD/FreeBSD.git/blob - usr.sbin/iovctl/parse.c
THIS BRANCH IS OBSOLETE, PLEASE READ:
[FreeBSD/FreeBSD.git] / usr.sbin / iovctl / parse.c
1 /*-
2  * Copyright (c) 2014-2015 Sandvine Inc.
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions and the following disclaimer.
10  * 2. Redistributions in binary form must reproduce the above copyright
11  *    notice, this list of conditions and the following disclaimer in the
12  *    documentation and/or other materials provided with the distribution.
13  *
14  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24  * SUCH DAMAGE.
25  */
26
27 #include <sys/cdefs.h>
28 __FBSDID("$FreeBSD$");
29
30 #include <sys/param.h>
31 #include <sys/iov.h>
32 #include <sys/nv.h>
33 #include <net/ethernet.h>
34
35 #include <err.h>
36 #include <errno.h>
37 #include <fcntl.h>
38 #include <regex.h>
39 #include <stdint.h>
40 #include <stdio.h>
41 #include <stdlib.h>
42 #include <string.h>
43 #include <ucl.h>
44 #include <unistd.h>
45
46 #include "iovctl.h"
47
48 static void
49 report_config_error(const char *key, const ucl_object_t *obj, const char *type)
50 {
51
52         errx(1, "Value '%s' of key '%s' is not of type %s",
53             ucl_object_tostring(obj), key, type);
54 }
55
56 /*
57  * Verifies that the value specified in the config file is a boolean value, and
58  * then adds the value to the configuration.
59  */
60 static void
61 add_bool_config(const char *key, const ucl_object_t *obj, nvlist_t *config)
62 {
63         bool val;
64
65         if (!ucl_object_toboolean_safe(obj, &val))
66                 report_config_error(key, obj, "bool");
67
68         nvlist_add_bool(config, key, val);
69 }
70
71 /*
72  * Verifies that the value specified in the config file is a string, and then
73  * adds the value to the configuration.
74  */
75 static void
76 add_string_config(const char *key, const ucl_object_t *obj, nvlist_t *config)
77 {
78         const char *val;
79
80         if (!ucl_object_tostring_safe(obj, &val))
81                 report_config_error(key, obj, "string");
82
83         nvlist_add_string(config, key, val);
84 }
85
86 /*
87  * Verifies that the value specified in the config file is a integer value
88  * within the specified range, and then adds the value to the configuration.
89  */
90 static void
91 add_uint_config(const char *key, const ucl_object_t *obj, nvlist_t *config,
92     const char *type, uint64_t max)
93 {
94         int64_t val;
95         uint64_t uval;
96
97         /* I must use a signed type here as libucl doesn't provide unsigned. */
98         if (!ucl_object_toint_safe(obj, &val))
99                 report_config_error(key, obj, type);
100
101         if (val < 0)
102                 report_config_error(key, obj, type);
103
104         uval = val;
105         if (uval > max)
106                 report_config_error(key, obj, type);
107
108         nvlist_add_number(config, key, uval);
109 }
110
111 /*
112  * Verifies that the value specified in the config file is a unicast MAC
113  * address, and then adds the value to the configuration.
114  */
115 static void
116 add_unicast_mac_config(const char *key, const ucl_object_t *obj, nvlist_t *config)
117 {
118         uint8_t mac[ETHER_ADDR_LEN];
119         const char *val, *token;
120         char *parse, *orig_parse, *tokpos, *endpos;
121         size_t len;
122         u_long value;
123         int i;
124
125         if (!ucl_object_tostring_safe(obj, &val))
126                 report_config_error(key, obj, "unicast-mac");
127
128         parse = strdup(val);
129         orig_parse = parse;
130
131         i = 0;
132         while ((token = strtok_r(parse, ":", &tokpos)) != NULL) {
133                 parse = NULL;
134
135                 len = strlen(token);
136                 if (len < 1 || len > 2)
137                         report_config_error(key, obj, "unicast-mac");
138
139                 value = strtoul(token, &endpos, 16);
140
141                 if (*endpos != '\0')
142                         report_config_error(key, obj, "unicast-mac");
143
144                 if (value > UINT8_MAX)
145                         report_config_error(key, obj, "unicast-mac");
146
147                 if (i >= ETHER_ADDR_LEN)
148                         report_config_error(key, obj, "unicast-mac");
149
150                 mac[i] = value;
151                 i++;
152         }
153
154         free(orig_parse);
155
156         if (i != ETHER_ADDR_LEN)
157                 report_config_error(key, obj, "unicast-mac");
158
159         if (ETHER_IS_MULTICAST(mac))
160                 errx(1, "Value '%s' of key '%s' is a multicast address",
161                     ucl_object_tostring(obj), key);
162
163         nvlist_add_binary(config, key, mac, ETHER_ADDR_LEN);
164 }
165
166 /*
167  * Validates that the given configuation value has the right type as specified
168  * in the schema, and then adds the value to the configuation node.
169  */
170 static void
171 add_config(const char *key, const ucl_object_t *obj, nvlist_t *config,
172     const nvlist_t *schema)
173 {
174         const char *type;
175
176         type = nvlist_get_string(schema, TYPE_SCHEMA_NAME);
177
178         if (strcasecmp(type, "bool") == 0)
179                 add_bool_config(key, obj, config);
180         else if (strcasecmp(type, "string") == 0)
181                 add_string_config(key, obj, config);
182         else if (strcasecmp(type, "uint8_t") == 0)
183                 add_uint_config(key, obj, config, type, UINT8_MAX);
184         else if (strcasecmp(type, "uint16_t") == 0)
185                 add_uint_config(key, obj, config, type, UINT16_MAX);
186         else if (strcasecmp(type, "uint32_t") == 0)
187                 add_uint_config(key, obj, config, type, UINT32_MAX);
188         else if (strcasecmp(type, "uint64_t") == 0)
189                 add_uint_config(key, obj, config, type, UINT64_MAX);
190         else if (strcasecmp(type, "unicast-mac") == 0)
191                 add_unicast_mac_config(key, obj, config);
192         else
193                 errx(1, "Unexpected type '%s' in schema", type);
194 }
195
196 /*
197  * Parses all values specified in a device section in the configuration file,
198  * validates that the key/value pair is valid in the schema, and then adds
199  * the key/value pair to the correct subsystem in the config.
200  */
201 static void
202 parse_device_config(const ucl_object_t *top, nvlist_t *config,
203     const char *subsystem, const nvlist_t *schema)
204 {
205         ucl_object_iter_t it;
206         const ucl_object_t *obj;
207         nvlist_t *subsystem_config, *driver_config, *iov_config;
208         const nvlist_t *driver_schema, *iov_schema;
209         const char *key;
210
211         if (nvlist_exists(config, subsystem))
212                 errx(1, "Multiple definitions of '%s' in config file",
213                     subsystem);
214
215         driver_schema = nvlist_get_nvlist(schema, DRIVER_CONFIG_NAME);
216         iov_schema = nvlist_get_nvlist(schema, IOV_CONFIG_NAME);
217
218         driver_config = nvlist_create(NV_FLAG_IGNORE_CASE);
219         if (driver_config == NULL)
220                 err(1, "Could not allocate config nvlist");
221
222         iov_config = nvlist_create(NV_FLAG_IGNORE_CASE);
223         if (iov_config == NULL)
224                 err(1, "Could not allocate config nvlist");
225
226         subsystem_config = nvlist_create(NV_FLAG_IGNORE_CASE);
227         if (subsystem_config == NULL)
228                 err(1, "Could not allocate config nvlist");
229
230         it = NULL;
231         while ((obj = ucl_iterate_object(top, &it, true)) != NULL) {
232                 key = ucl_object_key(obj);
233
234                 if (nvlist_exists_nvlist(iov_schema, key))
235                         add_config(key, obj, iov_config,
236                             nvlist_get_nvlist(iov_schema, key));
237                 else if (nvlist_exists_nvlist(driver_schema, key))
238                         add_config(key, obj, driver_config,
239                             nvlist_get_nvlist(driver_schema, key));
240                 else
241                         errx(1, "%s: Invalid config key '%s'", subsystem, key);
242         }
243
244         nvlist_move_nvlist(subsystem_config, DRIVER_CONFIG_NAME, driver_config);
245         nvlist_move_nvlist(subsystem_config, IOV_CONFIG_NAME, iov_config);
246         nvlist_move_nvlist(config, subsystem, subsystem_config);
247 }
248
249 /*
250  * Parses the specified config file using the given schema, and returns an
251  * nvlist containing the configuration specified by the file.
252  *
253  * Exits with a message to stderr and an error if any config validation fails.
254  */
255 nvlist_t *
256 parse_config_file(const char *filename, const nvlist_t *schema)
257 {
258         ucl_object_iter_t it;
259         struct ucl_parser *parser;
260         ucl_object_t *top;
261         const ucl_object_t *obj;
262         nvlist_t *config;
263         const nvlist_t *pf_schema, *vf_schema;
264         const char *errmsg, *key;
265         regex_t vf_pat;
266         int regex_err, processed_vf;
267
268         regex_err = regcomp(&vf_pat, "^"VF_PREFIX"([1-9][0-9]*|0)$",
269             REG_EXTENDED | REG_ICASE);
270         if (regex_err != 0)
271                 errx(1, "Could not compile VF regex");
272
273         parser = ucl_parser_new(0);
274         if (parser == NULL)
275                 err(1, "Could not allocate parser");
276
277         if (!ucl_parser_add_file(parser, filename))
278                 err(1, "Could not open '%s' for reading", filename);
279
280         errmsg = ucl_parser_get_error(parser);
281         if (errmsg != NULL)
282                 errx(1, "Could not parse '%s': %s", filename, errmsg);
283
284         config = nvlist_create(NV_FLAG_IGNORE_CASE);
285         if (config == NULL)
286                 err(1, "Could not allocate config nvlist");
287
288         pf_schema = nvlist_get_nvlist(schema, PF_CONFIG_NAME);
289         vf_schema = nvlist_get_nvlist(schema, VF_SCHEMA_NAME);
290
291         processed_vf = 0;
292         top = ucl_parser_get_object(parser);
293         it = NULL;
294         while ((obj = ucl_iterate_object(top, &it, true)) != NULL) {
295                 key = ucl_object_key(obj);
296
297                 if (strcasecmp(key, PF_CONFIG_NAME) == 0)
298                         parse_device_config(obj, config, key, pf_schema);
299                 else if (strcasecmp(key, DEFAULT_SCHEMA_NAME) == 0) {
300                         /*
301                          * Enforce that the default section must come before all
302                          * VF sections.  This will hopefully prevent confusing
303                          * the user by having a default value apply to a VF
304                          * that was declared earlier in the file.
305                          *
306                          * This also gives us the flexibility to extend the file
307                          * format in the future to allow for multiple default
308                          * sections that do only apply to subsequent VF
309                          * sections.
310                          */
311                         if (processed_vf)
312                                 errx(1,
313                         "'default' section must precede all VF sections");
314
315                         parse_device_config(obj, config, key, vf_schema);
316                 } else if (regexec(&vf_pat, key, 0, NULL, 0) == 0) {
317                         processed_vf = 1;
318                         parse_device_config(obj, config, key, vf_schema);
319                 } else
320                         errx(1, "Unexpected top-level node: %s", key);
321         }
322
323         validate_config(config, schema, &vf_pat);
324
325         ucl_object_unref(top);
326         ucl_parser_free(parser);
327         regfree(&vf_pat);
328
329         return (config);
330 }
331
332 /*
333  * Parse the PF configuration section for and return the value specified for
334  * the device parameter, or NULL if the device is not specified.
335  */
336 static const char *
337 find_pf_device(const ucl_object_t *pf)
338 {
339         ucl_object_iter_t it;
340         const ucl_object_t *obj;
341         const char *key, *device;
342
343         it = NULL;
344         while ((obj = ucl_iterate_object(pf, &it, true)) != NULL) {
345                 key = ucl_object_key(obj);
346
347                 if (strcasecmp(key, "device") == 0) {
348                         if (!ucl_object_tostring_safe(obj, &device))
349                                 err(1,
350                                     "Config PF.device must be a string");
351
352                         return (device);
353                 }
354         }
355
356         return (NULL);
357 }
358
359 /*
360  * Manually parse the config file looking for the name of the PF device.  We
361  * have to do this separately because we need the config schema to call the
362  * normal config file parsing code, and we need to know the name of the PF
363  * device so that we can fetch the schema from it.
364  *
365  * This will always exit on failure, so if it returns then it is guaranteed to
366  * have returned a valid device name.
367  */
368 char *
369 find_device(const char *filename)
370 {
371         char *device;
372         const char *deviceName;
373         ucl_object_iter_t it;
374         struct ucl_parser *parser;
375         ucl_object_t *top;
376         const ucl_object_t *obj;
377         const char *errmsg, *key;
378         int error;
379
380         device = NULL;
381         deviceName = NULL;
382
383         parser = ucl_parser_new(0);
384         if (parser == NULL)
385                 err(1, "Could not allocate parser");
386
387         if (!ucl_parser_add_file(parser, filename))
388                 err(1, "Could not open '%s' for reading", filename);
389
390         errmsg = ucl_parser_get_error(parser);
391         if (errmsg != NULL)
392                 errx(1, "Could not parse '%s': %s", filename, errmsg);
393
394         top = ucl_parser_get_object (parser);
395         it = NULL;
396         while ((obj = ucl_iterate_object(top, &it, true)) != NULL) {
397                 key = ucl_object_key(obj);
398
399                 if (strcasecmp(key, PF_CONFIG_NAME) == 0) {
400                         deviceName = find_pf_device(obj);
401                         break;
402                 }
403         }
404
405         if (deviceName == NULL)
406                 errx(1, "Config file does not specify device");
407
408         error = asprintf(&device, "/dev/iov/%s", deviceName);
409         if (error < 0)
410                 err(1, "Could not allocate memory for device");
411
412         ucl_object_unref(top);
413         ucl_parser_free(parser);
414
415         return (device);
416 }