xref: /freebsd-src/usr.sbin/iovctl/validate.c (revision 4d65a7c6951cea0333f1a0c1b32c38489cdfa6c5)
1dba9ec34SRyan Stone /*-
2*faf139ccSRyan Stone  * Copyright (c) 2014-2015 Sandvine Inc.
3dba9ec34SRyan Stone  * All rights reserved.
4dba9ec34SRyan Stone  *
5dba9ec34SRyan Stone  * Redistribution and use in source and binary forms, with or without
6dba9ec34SRyan Stone  * modification, are permitted provided that the following conditions
7dba9ec34SRyan Stone  * are met:
8dba9ec34SRyan Stone  * 1. Redistributions of source code must retain the above copyright
9dba9ec34SRyan Stone  *    notice, this list of conditions and the following disclaimer.
10dba9ec34SRyan Stone  * 2. Redistributions in binary form must reproduce the above copyright
11dba9ec34SRyan Stone  *    notice, this list of conditions and the following disclaimer in the
12dba9ec34SRyan Stone  *    documentation and/or other materials provided with the distribution.
13dba9ec34SRyan Stone  *
14dba9ec34SRyan Stone  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15dba9ec34SRyan Stone  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16dba9ec34SRyan Stone  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17dba9ec34SRyan Stone  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18dba9ec34SRyan Stone  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19dba9ec34SRyan Stone  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20dba9ec34SRyan Stone  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21dba9ec34SRyan Stone  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22dba9ec34SRyan Stone  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23dba9ec34SRyan Stone  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24dba9ec34SRyan Stone  * SUCH DAMAGE.
25dba9ec34SRyan Stone  */
26dba9ec34SRyan Stone 
27dba9ec34SRyan Stone #include <sys/param.h>
28dba9ec34SRyan Stone #include <sys/iov.h>
29c36e54bbSMariusz Zaborski #include <sys/dnv.h>
30c36e54bbSMariusz Zaborski #include <sys/nv.h>
31dba9ec34SRyan Stone 
32dba9ec34SRyan Stone #include <err.h>
33dba9ec34SRyan Stone #include <regex.h>
34dba9ec34SRyan Stone #include <stdlib.h>
35dba9ec34SRyan Stone 
36dba9ec34SRyan Stone #include "iovctl.h"
37dba9ec34SRyan Stone 
38dba9ec34SRyan Stone /*
39dba9ec34SRyan Stone  * Returns a writeable pointer to the configuration for the given device.
40dba9ec34SRyan Stone  * If no configuration exists, a new nvlist with empty driver and iov
41dba9ec34SRyan Stone  * sections is allocated and returned.
42dba9ec34SRyan Stone  *
43dba9ec34SRyan Stone  * Returning a writeable pointer requires removing the configuration from config
44dba9ec34SRyan Stone  * using nvlist_take.  It is the responsibility of the caller to re-insert the
45dba9ec34SRyan Stone  * nvlist in config with nvlist_move_nvlist.
46dba9ec34SRyan Stone  */
47dba9ec34SRyan Stone static nvlist_t *
find_config(nvlist_t * config,const char * device)48dba9ec34SRyan Stone find_config(nvlist_t *config, const char * device)
49dba9ec34SRyan Stone {
50dba9ec34SRyan Stone 	nvlist_t *subsystem, *empty_driver, *empty_iov;
51dba9ec34SRyan Stone 
52dba9ec34SRyan Stone 	subsystem = dnvlist_take_nvlist(config, device, NULL);
53dba9ec34SRyan Stone 
54dba9ec34SRyan Stone 	if (subsystem != NULL)
55dba9ec34SRyan Stone 		return (subsystem);
56dba9ec34SRyan Stone 
57dba9ec34SRyan Stone 	empty_driver = nvlist_create(NV_FLAG_IGNORE_CASE);
58dba9ec34SRyan Stone 	if (empty_driver == NULL)
59dba9ec34SRyan Stone 		err(1, "Could not allocate config nvlist");
60dba9ec34SRyan Stone 
61dba9ec34SRyan Stone 	empty_iov = nvlist_create(NV_FLAG_IGNORE_CASE);
62dba9ec34SRyan Stone 	if (empty_iov == NULL)
63dba9ec34SRyan Stone 		err(1, "Could not allocate config nvlist");
64dba9ec34SRyan Stone 
65dba9ec34SRyan Stone 	subsystem = nvlist_create(NV_FLAG_IGNORE_CASE);
66dba9ec34SRyan Stone 	if (subsystem == NULL)
67dba9ec34SRyan Stone 		err(1, "Could not allocate config nvlist");
68dba9ec34SRyan Stone 
69dba9ec34SRyan Stone 	nvlist_move_nvlist(subsystem, DRIVER_CONFIG_NAME, empty_driver);
70dba9ec34SRyan Stone 	nvlist_move_nvlist(subsystem, IOV_CONFIG_NAME, empty_iov);
71dba9ec34SRyan Stone 
72dba9ec34SRyan Stone 	return (subsystem);
73dba9ec34SRyan Stone }
74dba9ec34SRyan Stone 
75dba9ec34SRyan Stone static uint16_t
parse_vf_num(const char * key,regmatch_t * matches)76dba9ec34SRyan Stone parse_vf_num(const char *key, regmatch_t *matches)
77dba9ec34SRyan Stone {
78dba9ec34SRyan Stone 	u_long vf_num;
79dba9ec34SRyan Stone 
80dba9ec34SRyan Stone 	vf_num = strtoul(key + matches[1].rm_so, NULL, 10);
81dba9ec34SRyan Stone 
82dba9ec34SRyan Stone 	if (vf_num > UINT16_MAX)
83dba9ec34SRyan Stone 		errx(1, "VF number %lu is too large to be valid",
84dba9ec34SRyan Stone 		    vf_num);
85dba9ec34SRyan Stone 
86dba9ec34SRyan Stone 	return (vf_num);
87dba9ec34SRyan Stone }
88dba9ec34SRyan Stone 
89dba9ec34SRyan Stone /*
90dba9ec34SRyan Stone  * Apply the default values specified in device_defaults to the specified
91dba9ec34SRyan Stone  * subsystem in the given device_config.
92dba9ec34SRyan Stone  *
93dba9ec34SRyan Stone  * This function assumes that the values specified in device_defaults have
94dba9ec34SRyan Stone  * already been validated.
95dba9ec34SRyan Stone  */
96dba9ec34SRyan Stone static void
apply_subsystem_defaults(nvlist_t * device_config,const char * subsystem,const nvlist_t * device_defaults)97dba9ec34SRyan Stone apply_subsystem_defaults(nvlist_t *device_config, const char *subsystem,
98dba9ec34SRyan Stone     const nvlist_t *device_defaults)
99dba9ec34SRyan Stone {
100dba9ec34SRyan Stone 	nvlist_t *config;
101dba9ec34SRyan Stone 	const nvlist_t *defaults;
102dba9ec34SRyan Stone 	const char *name;
103dba9ec34SRyan Stone 	void *cookie;
104dba9ec34SRyan Stone 	size_t len;
105dba9ec34SRyan Stone 	const void *bin;
106dba9ec34SRyan Stone 	int type;
107dba9ec34SRyan Stone 
108dba9ec34SRyan Stone 	config = nvlist_take_nvlist(device_config, subsystem);
109dba9ec34SRyan Stone 	defaults = nvlist_get_nvlist(device_defaults, subsystem);
110dba9ec34SRyan Stone 
111dba9ec34SRyan Stone 	cookie = NULL;
112dba9ec34SRyan Stone 	while ((name = nvlist_next(defaults, &type, &cookie)) != NULL) {
113dba9ec34SRyan Stone 		if (nvlist_exists(config, name))
114dba9ec34SRyan Stone 			continue;
115dba9ec34SRyan Stone 
116dba9ec34SRyan Stone 		switch (type) {
117dba9ec34SRyan Stone 		case NV_TYPE_BOOL:
118dba9ec34SRyan Stone 			nvlist_add_bool(config, name,
119dba9ec34SRyan Stone 			    nvlist_get_bool(defaults, name));
120dba9ec34SRyan Stone 			break;
121dba9ec34SRyan Stone 		case NV_TYPE_NUMBER:
122dba9ec34SRyan Stone 			nvlist_add_number(config, name,
123dba9ec34SRyan Stone 			    nvlist_get_number(defaults, name));
124dba9ec34SRyan Stone 			break;
125dba9ec34SRyan Stone 		case NV_TYPE_STRING:
126dba9ec34SRyan Stone 			nvlist_add_string(config, name,
127dba9ec34SRyan Stone 			    nvlist_get_string(defaults, name));
128dba9ec34SRyan Stone 			break;
129dba9ec34SRyan Stone 		case NV_TYPE_NVLIST:
130dba9ec34SRyan Stone 			nvlist_add_nvlist(config, name,
131dba9ec34SRyan Stone 			    nvlist_get_nvlist(defaults, name));
132dba9ec34SRyan Stone 			break;
133dba9ec34SRyan Stone 		case NV_TYPE_BINARY:
134dba9ec34SRyan Stone 			bin = nvlist_get_binary(defaults, name, &len);
135dba9ec34SRyan Stone 			nvlist_add_binary(config, name, bin, len);
136dba9ec34SRyan Stone 			break;
137dba9ec34SRyan Stone 		default:
138dba9ec34SRyan Stone 			errx(1, "Unexpected type '%d'", type);
139dba9ec34SRyan Stone 		}
140dba9ec34SRyan Stone 	}
141dba9ec34SRyan Stone 	nvlist_move_nvlist(device_config, subsystem, config);
142dba9ec34SRyan Stone }
143dba9ec34SRyan Stone 
144dba9ec34SRyan Stone /*
145dba9ec34SRyan Stone  * Iterate over every subsystem in the given VF device and apply default values
146dba9ec34SRyan Stone  * for parameters that were not configured with a value.
147dba9ec34SRyan Stone  *
148dba9ec34SRyan Stone  * This function assumes that the values specified in defaults have already been
149dba9ec34SRyan Stone  * validated.
150dba9ec34SRyan Stone  */
151dba9ec34SRyan Stone static void
apply_defaults(nvlist_t * vf,const nvlist_t * defaults)152dba9ec34SRyan Stone apply_defaults(nvlist_t *vf, const nvlist_t *defaults)
153dba9ec34SRyan Stone {
154dba9ec34SRyan Stone 
155dba9ec34SRyan Stone 	apply_subsystem_defaults(vf, DRIVER_CONFIG_NAME, defaults);
156dba9ec34SRyan Stone 	apply_subsystem_defaults(vf, IOV_CONFIG_NAME, defaults);
157dba9ec34SRyan Stone }
158dba9ec34SRyan Stone 
159dba9ec34SRyan Stone /*
160dba9ec34SRyan Stone  * Validate that all required parameters have been configured in the specified
161dba9ec34SRyan Stone  * subsystem.
162dba9ec34SRyan Stone  */
163dba9ec34SRyan Stone static void
validate_subsystem(const nvlist_t * device,const nvlist_t * device_schema,const char * subsystem_name,const char * config_name)164dba9ec34SRyan Stone validate_subsystem(const nvlist_t *device, const nvlist_t *device_schema,
165dba9ec34SRyan Stone     const char *subsystem_name, const char *config_name)
166dba9ec34SRyan Stone {
167dba9ec34SRyan Stone 	const nvlist_t *subsystem, *schema, *config;
168dba9ec34SRyan Stone 	const char *name;
169dba9ec34SRyan Stone 	void *cookie;
170dba9ec34SRyan Stone 	int type;
171dba9ec34SRyan Stone 
172dba9ec34SRyan Stone 	subsystem = nvlist_get_nvlist(device, subsystem_name);
173dba9ec34SRyan Stone 	schema = nvlist_get_nvlist(device_schema, subsystem_name);
174dba9ec34SRyan Stone 
175dba9ec34SRyan Stone 	cookie = NULL;
176dba9ec34SRyan Stone 	while ((name = nvlist_next(schema, &type, &cookie)) != NULL) {
177dba9ec34SRyan Stone 		config = nvlist_get_nvlist(schema, name);
178dba9ec34SRyan Stone 
179dba9ec34SRyan Stone 		if (dnvlist_get_bool(config, REQUIRED_SCHEMA_NAME, false)) {
180dba9ec34SRyan Stone 			if (!nvlist_exists(subsystem, name))
181dba9ec34SRyan Stone 				errx(1,
182dba9ec34SRyan Stone 				    "Required parameter '%s' not found in '%s'",
183dba9ec34SRyan Stone 				    name, config_name);
184dba9ec34SRyan Stone 		}
185dba9ec34SRyan Stone 	}
186dba9ec34SRyan Stone }
187dba9ec34SRyan Stone 
188dba9ec34SRyan Stone /*
189dba9ec34SRyan Stone  * Validate that all required parameters have been configured in all subsystems
190dba9ec34SRyan Stone  * in the device.
191dba9ec34SRyan Stone  */
192dba9ec34SRyan Stone static void
validate_device(const nvlist_t * device,const nvlist_t * schema,const char * config_name)193dba9ec34SRyan Stone validate_device(const nvlist_t *device, const nvlist_t *schema,
194dba9ec34SRyan Stone     const char *config_name)
195dba9ec34SRyan Stone {
196dba9ec34SRyan Stone 
197dba9ec34SRyan Stone 	validate_subsystem(device, schema, DRIVER_CONFIG_NAME, config_name);
198dba9ec34SRyan Stone 	validate_subsystem(device, schema, IOV_CONFIG_NAME, config_name);
199dba9ec34SRyan Stone }
200dba9ec34SRyan Stone 
201dba9ec34SRyan Stone static uint16_t
get_num_vfs(const nvlist_t * pf)202dba9ec34SRyan Stone get_num_vfs(const nvlist_t *pf)
203dba9ec34SRyan Stone {
204dba9ec34SRyan Stone 	const nvlist_t *iov;
205dba9ec34SRyan Stone 
206dba9ec34SRyan Stone 	iov = nvlist_get_nvlist(pf, IOV_CONFIG_NAME);
207dba9ec34SRyan Stone 	return (nvlist_get_number(iov, "num_vfs"));
208dba9ec34SRyan Stone }
209dba9ec34SRyan Stone 
210dba9ec34SRyan Stone /*
211dba9ec34SRyan Stone  * Validates the configuration that has been parsed into config using the given
212dba9ec34SRyan Stone  * config schema.  Note that the parser is required to not insert configuration
213dba9ec34SRyan Stone  * keys that are not valid in the schema, and to not insert configuration values
214dba9ec34SRyan Stone  * that are of the incorrect type.  Therefore this function will not validate
215dba9ec34SRyan Stone  * either condition.  This function is only responsible for inserting config
216dba9ec34SRyan Stone  * file defaults in individual VF sections and removing the DEFAULT_SCHEMA_NAME
217dba9ec34SRyan Stone  * subsystem from config, validating that all required parameters in the schema
218dba9ec34SRyan Stone  * are present in each PF and VF subsystem, and that there is no VF subsystem
219dba9ec34SRyan Stone  * section whose number exceeds num_vfs.
220dba9ec34SRyan Stone  */
221dba9ec34SRyan Stone void
validate_config(nvlist_t * config,const nvlist_t * schema,const regex_t * vf_pat)222dba9ec34SRyan Stone validate_config(nvlist_t *config, const nvlist_t *schema, const regex_t *vf_pat)
223dba9ec34SRyan Stone {
224dba9ec34SRyan Stone 	char device_name[VF_MAX_NAME];
225dba9ec34SRyan Stone 	regmatch_t matches[2];
226dba9ec34SRyan Stone 	nvlist_t *defaults, *pf, *vf;
227dba9ec34SRyan Stone 	const nvlist_t *vf_schema;
228dba9ec34SRyan Stone 	const char *key;
229dba9ec34SRyan Stone 	void *cookie;
230dba9ec34SRyan Stone 	int i, type;
231dba9ec34SRyan Stone 	uint16_t vf_num, num_vfs;
232dba9ec34SRyan Stone 
233dba9ec34SRyan Stone 	pf = find_config(config, PF_CONFIG_NAME);
234dba9ec34SRyan Stone 	validate_device(pf, nvlist_get_nvlist(schema, PF_CONFIG_NAME),
235dba9ec34SRyan Stone 	    PF_CONFIG_NAME);
236dba9ec34SRyan Stone 	nvlist_move_nvlist(config, PF_CONFIG_NAME, pf);
237dba9ec34SRyan Stone 
238dba9ec34SRyan Stone 	num_vfs = get_num_vfs(pf);
239dba9ec34SRyan Stone 	vf_schema = nvlist_get_nvlist(schema, VF_SCHEMA_NAME);
240dba9ec34SRyan Stone 
241dba9ec34SRyan Stone 	if (num_vfs == 0)
242dba9ec34SRyan Stone 		errx(1, "PF.num_vfs must be at least 1");
243dba9ec34SRyan Stone 
244dba9ec34SRyan Stone 	defaults = dnvlist_take_nvlist(config, DEFAULT_SCHEMA_NAME, NULL);
245dba9ec34SRyan Stone 
246dba9ec34SRyan Stone 	for (i = 0; i < num_vfs; i++) {
247dba9ec34SRyan Stone 		snprintf(device_name, sizeof(device_name), VF_PREFIX"%d",
248dba9ec34SRyan Stone 		    i);
249dba9ec34SRyan Stone 
250dba9ec34SRyan Stone 		vf = find_config(config, device_name);
251dba9ec34SRyan Stone 
252dba9ec34SRyan Stone 		if (defaults != NULL)
253dba9ec34SRyan Stone 			apply_defaults(vf, defaults);
254dba9ec34SRyan Stone 
255dba9ec34SRyan Stone 		validate_device(vf, vf_schema, device_name);
256dba9ec34SRyan Stone 		nvlist_move_nvlist(config, device_name, vf);
257dba9ec34SRyan Stone 	}
258dba9ec34SRyan Stone 	nvlist_destroy(defaults);
259dba9ec34SRyan Stone 
260dba9ec34SRyan Stone 	cookie = NULL;
261dba9ec34SRyan Stone 	while ((key = nvlist_next(config, &type, &cookie)) != NULL) {
262dba9ec34SRyan Stone 		if (regexec(vf_pat, key, nitems(matches), matches, 0) == 0) {
263dba9ec34SRyan Stone 			vf_num = parse_vf_num(key, matches);
264dba9ec34SRyan Stone 			if (vf_num >= num_vfs)
265dba9ec34SRyan Stone 				errx(1,
266dba9ec34SRyan Stone 				   "VF number %d is out of bounds (num_vfs=%d)",
267dba9ec34SRyan Stone 				    vf_num, num_vfs);
268dba9ec34SRyan Stone 		}
269dba9ec34SRyan Stone 	}
270dba9ec34SRyan Stone }
271dba9ec34SRyan Stone 
272