xref: /titanic_44/usr/src/lib/libdladm/common/linkprop.c (revision f645252839e7ff6d25cadf6a45b2ae9099943c5a)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 /*
22  * Copyright 2008 Sun Microsystems, Inc.  All rights reserved.
23  * Use is subject to license terms.
24  */
25 
26 #include <stdlib.h>
27 #include <strings.h>
28 #include <errno.h>
29 #include <ctype.h>
30 #include <stddef.h>
31 #include <sys/types.h>
32 #include <sys/stat.h>
33 #include <sys/dld.h>
34 #include <sys/zone.h>
35 #include <fcntl.h>
36 #include <unistd.h>
37 #include <libdevinfo.h>
38 #include <zone.h>
39 #include <libdllink.h>
40 #include <libdladm_impl.h>
41 #include <libdlwlan_impl.h>
42 #include <libdlwlan.h>
43 #include <libdlvlan.h>
44 #include <libdlvnic.h>
45 #include <libintl.h>
46 #include <dlfcn.h>
47 #include <link.h>
48 #include <inet/wifi_ioctl.h>
49 #include <libdladm.h>
50 #include <libdlstat.h>
51 #include <sys/param.h>
52 #include <sys/debug.h>
53 #include <sys/dld.h>
54 #include <sys/mac_flow.h>
55 #include <inttypes.h>
56 #include <sys/ethernet.h>
57 #include <net/wpa.h>
58 #include <sys/sysmacros.h>
59 
60 /*
61  * The linkprop get() callback.
62  * - pd: 	pointer to the prop_desc_t
63  * - propstrp:	a property string array to keep the returned property.
64  *		Caller allocated.
65  * - cntp:	number of returned properties.
66  *		Caller also uses it to indicate how many it expects.
67  */
68 struct prop_desc;
69 typedef struct prop_desc prop_desc_t;
70 
71 typedef dladm_status_t	pd_getf_t(prop_desc_t *pdp,
72 			datalink_id_t, char **propstp, uint_t *cntp,
73 			datalink_media_t, uint_t, uint_t *);
74 
75 /*
76  * The linkprop set() callback.
77  * - propval:	a val_desc_t array which keeps the property values to be set.
78  * - cnt:	number of properties to be set.
79  * - flags: 	additional flags passed down the system call.
80  *
81  * pd_set takes val_desc_t given by pd_check(), translates it into
82  * a format suitable for kernel consumption. This may require allocation
83  * of ioctl buffers etc. pd_set() may call another common routine (used
84  * by all other pd_sets) which invokes the ioctl.
85  */
86 typedef dladm_status_t	pd_setf_t(prop_desc_t *, datalink_id_t,
87 			    val_desc_t *propval, uint_t cnt, uint_t flags,
88 			    datalink_media_t);
89 
90 /*
91  * The linkprop check() callback.
92  * - propstrp:	property string array which keeps the property to be checked.
93  * - cnt:	number of properties.
94  * - propval:	return value; the property values of the given property strings.
95  *
96  * pd_check checks that the input values are valid. It does so by
97  * iteraring through the pd_modval list for the property. If
98  * the modifiable values cannot be expressed as a list, a pd_check
99  * specific to this property can be used. If the input values are
100  * verified to be valid, pd_check allocates a val_desc_t and fills it
101  * with either a val_desc_t found on the pd_modval list or something
102  * generated on the fly.
103  */
104 typedef dladm_status_t	pd_checkf_t(prop_desc_t *pdp, datalink_id_t,
105 			    char **propstrp, uint_t cnt, val_desc_t *propval,
106 			    datalink_media_t);
107 
108 typedef struct link_attr_s {
109 	mac_prop_id_t	pp_id;
110 	size_t		pp_valsize;
111 	char		*pp_name;
112 } link_attr_t;
113 
114 static dld_ioc_macprop_t *i_dladm_buf_alloc_by_name(size_t, datalink_id_t,
115 			    const char *, uint_t, dladm_status_t *);
116 static dld_ioc_macprop_t *i_dladm_buf_alloc_by_id(size_t, datalink_id_t,
117 			    mac_prop_id_t, uint_t, dladm_status_t *);
118 static dld_ioc_macprop_t *i_dladm_get_public_prop(datalink_id_t, char *, uint_t,
119 			    dladm_status_t *, uint_t *);
120 
121 static dladm_status_t i_dladm_set_prop(datalink_id_t, const char *, char **,
122 					uint_t, uint_t);
123 static dladm_status_t i_dladm_get_prop(datalink_id_t, const char *, char **,
124 					uint_t *, dladm_prop_type_t, uint_t);
125 static link_attr_t *dladm_name2prop(const char *);
126 static link_attr_t *dladm_id2prop(mac_prop_id_t);
127 
128 static pd_getf_t	do_get_zone, do_get_autopush, do_get_rate_mod,
129 			do_get_rate_prop, do_get_channel_prop,
130 			do_get_powermode_prop, do_get_radio_prop,
131 			i_dladm_duplex_get, i_dladm_status_get,
132 			i_dladm_binary_get, i_dladm_uint32_get,
133 			i_dladm_flowctl_get, dld_maxbw_get, dld_cpus_get,
134 			dld_priority_get;
135 
136 static pd_setf_t	do_set_zone, do_set_rate_prop,
137 			do_set_powermode_prop, do_set_radio_prop,
138 			i_dladm_set_public_prop, do_set_res, do_set_cpus;
139 
140 static pd_checkf_t	do_check_zone, do_check_autopush, do_check_rate,
141 			i_dladm_defmtu_check, do_check_maxbw, do_check_cpus,
142 			do_check_priority;
143 
144 static dladm_status_t	i_dladm_speed_get(prop_desc_t *, datalink_id_t,
145 			char **, uint_t *, uint_t, uint_t *);
146 static dladm_status_t	i_dladm_wlan_get_legacy_ioctl(datalink_id_t, void *,
147 			    uint_t, uint_t);
148 static dladm_status_t	i_dladm_wlan_set_legacy_ioctl(datalink_id_t, void *,
149 			    uint_t, uint_t);
150 static dladm_status_t	i_dladm_macprop(void *, boolean_t);
151 static const char	*dladm_perm2str(uint_t, char *);
152 
153 struct prop_desc {
154 	/*
155 	 * link property name
156 	 */
157 	char			*pd_name;
158 
159 	/*
160 	 * default property value, can be set to { "", NULL }
161 	 */
162 	val_desc_t		pd_defval;
163 
164 	/*
165 	 * list of optional property values, can be NULL.
166 	 *
167 	 * This is set to non-NULL if there is a list of possible property
168 	 * values.  pd_optval would point to the array of possible values.
169 	 */
170 	val_desc_t		*pd_optval;
171 
172 	/*
173 	 * count of the above optional property values. 0 if pd_optval is NULL.
174 	 */
175 	uint_t			pd_noptval;
176 
177 	/*
178 	 * callback to set link property;
179 	 * set to NULL if this property is read-only
180 	 */
181 	pd_setf_t		*pd_set;
182 
183 	/*
184 	 * callback to get modifiable link property
185 	 */
186 	pd_getf_t		*pd_getmod;
187 
188 	/*
189 	 * callback to get current link property
190 	 */
191 	pd_getf_t		*pd_get;
192 
193 	/*
194 	 * callback to validate link property value, set to NULL if pd_optval
195 	 * is not NULL. In that case, validate the value by comparing it with
196 	 * the pd_optval. Return a val_desc_t array pointer if the value is
197 	 * valid.
198 	 */
199 	pd_checkf_t		*pd_check;
200 
201 	uint_t			pd_flags;
202 #define	PD_TEMPONLY	0x1	/* property is temporary only */
203 #define	PD_CHECK_ALLOC	0x2	/* alloc vd_val as part of pd_check */
204 	/*
205 	 * indicate link classes this property applies to.
206 	 */
207 	datalink_class_t	pd_class;
208 
209 	/*
210 	 * indicate link media type this property applies to.
211 	 */
212 	datalink_media_t	pd_dmedia;
213 };
214 
215 #define	MAC_PROP_BUFSIZE(v)	sizeof (dld_ioc_macprop_t) + (v) - 1
216 
217 /*
218  * Supported link properties enumerated in the prop_table[] array are
219  * computed using the callback functions in that array. To compute the
220  * property value, multiple distinct system calls may be needed (e.g.,
221  * for wifi speed, we need to issue system calls to get desired/supported
222  * rates). The link_attr[] table enumerates the interfaces to the kernel,
223  * and the type/size of the data passed in the user-kernel interface.
224  */
225 static link_attr_t link_attr[] = {
226 	{ MAC_PROP_DUPLEX,	sizeof (link_duplex_t),	"duplex"},
227 
228 	{ MAC_PROP_SPEED,	sizeof (uint64_t),	"speed"},
229 
230 	{ MAC_PROP_STATUS,	sizeof (link_state_t),	"state"},
231 
232 	{ MAC_PROP_AUTONEG,	sizeof (uint8_t),	"adv_autoneg_cap"},
233 
234 	{ MAC_PROP_MTU,		sizeof (uint32_t),	"mtu"},
235 
236 	{ MAC_PROP_FLOWCTRL,	sizeof (link_flowctrl_t), "flowctrl"},
237 
238 	{ MAC_PROP_ZONE,	sizeof (dld_ioc_zid_t),	"zone"},
239 
240 	{ MAC_PROP_AUTOPUSH,	sizeof (struct dlautopush), "autopush"},
241 
242 	{ MAC_PROP_ADV_1000FDX_CAP, sizeof (uint8_t),	"adv_1000fdx_cap"},
243 
244 	{ MAC_PROP_EN_1000FDX_CAP, sizeof (uint8_t),	"en_1000fdx_cap"},
245 
246 	{ MAC_PROP_ADV_1000HDX_CAP, sizeof (uint8_t),	"adv_1000hdx_cap"},
247 
248 	{ MAC_PROP_EN_1000HDX_CAP, sizeof (uint8_t),	"en_1000hdx_cap"},
249 
250 	{ MAC_PROP_ADV_100FDX_CAP, sizeof (uint8_t),	"adv_100fdx_cap"},
251 
252 	{ MAC_PROP_EN_100FDX_CAP, sizeof (uint8_t),	"en_100fdx_cap"},
253 
254 	{ MAC_PROP_ADV_100HDX_CAP, sizeof (uint8_t),	"adv_100hdx_cap"},
255 
256 	{ MAC_PROP_EN_100HDX_CAP, sizeof (uint8_t),	"en_100hdx_cap"},
257 
258 	{ MAC_PROP_ADV_10FDX_CAP, sizeof (uint8_t),	"adv_10fdx_cap"},
259 
260 	{ MAC_PROP_EN_10FDX_CAP, sizeof (uint8_t),	"en_10fdx_cap"},
261 
262 	{ MAC_PROP_ADV_10HDX_CAP, sizeof (uint8_t),	"adv_10hdx_cap"},
263 
264 	{ MAC_PROP_EN_10HDX_CAP, sizeof (uint8_t),	"en_10hdx_cap"},
265 
266 	{ MAC_PROP_WL_ESSID,	sizeof (wl_linkstatus_t), "essid"},
267 
268 	{ MAC_PROP_WL_BSSID,	sizeof (wl_bssid_t),	"bssid"},
269 
270 	{ MAC_PROP_WL_BSSTYPE,	sizeof (wl_bss_type_t),	"bsstype"},
271 
272 	{ MAC_PROP_WL_LINKSTATUS, sizeof (wl_linkstatus_t), "wl_linkstatus"},
273 
274 	/* wl_rates_t has variable length */
275 	{ MAC_PROP_WL_DESIRED_RATES, sizeof (wl_rates_t), "desired_rates"},
276 
277 	/* wl_rates_t has variable length */
278 	{ MAC_PROP_WL_SUPPORTED_RATES, sizeof (wl_rates_t), "supported_rates"},
279 
280 	{ MAC_PROP_WL_AUTH_MODE, sizeof (wl_authmode_t), "authmode"},
281 
282 	{ MAC_PROP_WL_ENCRYPTION, sizeof (wl_encryption_t), "encryption"},
283 
284 	{ MAC_PROP_WL_RSSI,	sizeof (wl_rssi_t),	"signal"},
285 
286 	{ MAC_PROP_WL_PHY_CONFIG, sizeof (wl_phy_conf_t), "phy_conf"},
287 
288 	{ MAC_PROP_WL_CAPABILITY, sizeof (wl_capability_t), "capability"},
289 
290 	{ MAC_PROP_WL_WPA,	sizeof (wl_wpa_t),	"wpa"},
291 
292 	/*  wl_wpa_ess_t has variable length */
293 	{ MAC_PROP_WL_SCANRESULTS, sizeof (wl_wpa_ess_t), "scan_results"},
294 
295 	{ MAC_PROP_WL_POWER_MODE, sizeof (wl_ps_mode_t), "powermode"},
296 
297 	{ MAC_PROP_WL_RADIO,	sizeof (dladm_wlan_radio_t), "wl_radio"},
298 
299 	{ MAC_PROP_WL_ESS_LIST, sizeof (wl_ess_list_t),	"wl_ess_list"},
300 
301 	{ MAC_PROP_WL_KEY_TAB,	sizeof (wl_wep_key_tab_t), "wl_wep_key"},
302 
303 	{ MAC_PROP_WL_CREATE_IBSS, sizeof (wl_create_ibss_t), "createibss"},
304 
305 	/* wl_wpa_ie_t has variable length */
306 	{ MAC_PROP_WL_SETOPTIE,	sizeof (wl_wpa_ie_t),	"set_ie"},
307 
308 	{ MAC_PROP_WL_DELKEY,	sizeof (wl_del_key_t),	"wpa_del_key"},
309 
310 	{ MAC_PROP_WL_KEY,	sizeof (wl_key_t),	"wl_key"},
311 
312 	{ MAC_PROP_WL_MLME,	sizeof (wl_mlme_t),	"mlme"},
313 
314 	{ MAC_PROP_MAXBW,	sizeof (mac_resource_props_t),	"maxbw"},
315 
316 	{ MAC_PROP_PRIO,	sizeof (mac_resource_props_t),	"priority"},
317 
318 	{ MAC_PROP_BIND_CPU,	sizeof (mac_resource_props_t),	"cpus"},
319 
320 	{ MAC_PROP_PRIVATE,	0,			"driver-private"}
321 
322 };
323 
324 static  val_desc_t	link_duplex_vals[] = {
325 	{ "half", 	LINK_DUPLEX_HALF	},
326 	{ "full", 	LINK_DUPLEX_HALF	}
327 };
328 static  val_desc_t	link_status_vals[] = {
329 	{ "up",		LINK_STATE_UP		},
330 	{ "down",	LINK_STATE_DOWN		}
331 };
332 static  val_desc_t	link_01_vals[] = {
333 	{ "1",		1			},
334 	{ "0",		0			}
335 };
336 static  val_desc_t	link_flow_vals[] = {
337 	{ "no",		LINK_FLOWCTRL_NONE	},
338 	{ "tx",		LINK_FLOWCTRL_TX	},
339 	{ "rx",		LINK_FLOWCTRL_RX	},
340 	{ "bi",		LINK_FLOWCTRL_BI	}
341 };
342 static  val_desc_t	link_priority_vals[] = {
343 	{ "low",	MPL_LOW	},
344 	{ "medium",	MPL_MEDIUM	},
345 	{ "high",	MPL_HIGH	}
346 };
347 
348 static val_desc_t	dladm_wlan_radio_vals[] = {
349 	{ "on",		DLADM_WLAN_RADIO_ON	},
350 	{ "off",	DLADM_WLAN_RADIO_OFF	}
351 };
352 
353 static val_desc_t	dladm_wlan_powermode_vals[] = {
354 	{ "off",	DLADM_WLAN_PM_OFF	},
355 	{ "fast",	DLADM_WLAN_PM_FAST	},
356 	{ "max",	DLADM_WLAN_PM_MAX	}
357 };
358 
359 #define	VALCNT(vals)    (sizeof ((vals)) / sizeof (val_desc_t))
360 #define	RESET_VAL	((uintptr_t)-1)
361 
362 static prop_desc_t	prop_table[] = {
363 	{ "channel",	{ NULL, 0 },
364 	    NULL, 0, NULL, NULL,
365 	    do_get_channel_prop, NULL, 0,
366 	    DATALINK_CLASS_PHYS, DL_WIFI },
367 
368 	{ "powermode",	{ "off", DLADM_WLAN_PM_OFF },
369 	    dladm_wlan_powermode_vals, VALCNT(dladm_wlan_powermode_vals),
370 	    do_set_powermode_prop, NULL,
371 	    do_get_powermode_prop, NULL, 0,
372 	    DATALINK_CLASS_PHYS, DL_WIFI },
373 
374 	{ "radio",	{ "on", DLADM_WLAN_RADIO_ON },
375 	    dladm_wlan_radio_vals, VALCNT(dladm_wlan_radio_vals),
376 	    do_set_radio_prop, NULL,
377 	    do_get_radio_prop, NULL, 0,
378 	    DATALINK_CLASS_PHYS, DL_WIFI },
379 
380 	{ "speed",	{ "", 0 }, NULL, 0,
381 	    do_set_rate_prop, do_get_rate_mod,
382 	    do_get_rate_prop, do_check_rate, 0,
383 	    DATALINK_CLASS_PHYS, DATALINK_ANY_MEDIATYPE },
384 
385 	{ "autopush",	{ "", 0 }, NULL, 0,
386 	    i_dladm_set_public_prop, NULL,
387 	    do_get_autopush, do_check_autopush, PD_CHECK_ALLOC,
388 	    DATALINK_CLASS_ALL, DATALINK_ANY_MEDIATYPE },
389 
390 	{ "zone",	{ "", 0 }, NULL, 0,
391 	    do_set_zone, NULL,
392 	    do_get_zone, do_check_zone, PD_TEMPONLY|PD_CHECK_ALLOC,
393 	    DATALINK_CLASS_ALL, DATALINK_ANY_MEDIATYPE },
394 
395 	{ "duplex",	{ "", 0 },
396 	    link_duplex_vals, VALCNT(link_duplex_vals),
397 	    NULL, NULL, i_dladm_duplex_get, NULL,
398 	    0, DATALINK_CLASS_PHYS, DL_ETHER },
399 
400 	{ "state",	{ "up", LINK_STATE_UP },
401 	    link_status_vals, VALCNT(link_status_vals),
402 	    NULL, NULL, i_dladm_status_get, NULL,
403 	    0, DATALINK_CLASS_ALL, DATALINK_ANY_MEDIATYPE },
404 
405 	{ "adv_autoneg_cap", { "1", 1 },
406 	    link_01_vals, VALCNT(link_01_vals),
407 	    i_dladm_set_public_prop, NULL, i_dladm_binary_get, NULL,
408 	    0, DATALINK_CLASS_PHYS, DL_ETHER },
409 
410 	{ "mtu", { "", 0 }, NULL, 0,
411 	    i_dladm_set_public_prop, NULL, i_dladm_uint32_get,
412 	    i_dladm_defmtu_check, 0, DATALINK_CLASS_ALL,
413 	    DATALINK_ANY_MEDIATYPE },
414 
415 	{ "flowctrl", { "", 0 },
416 	    link_flow_vals, VALCNT(link_flow_vals),
417 	    i_dladm_set_public_prop, NULL, i_dladm_flowctl_get, NULL,
418 	    0, DATALINK_CLASS_PHYS, DL_ETHER },
419 
420 	{ "adv_1000fdx_cap", { "", 0 },
421 	    link_01_vals, VALCNT(link_01_vals),
422 	    NULL, NULL, i_dladm_binary_get, NULL,
423 	    0, DATALINK_CLASS_PHYS, DL_ETHER },
424 
425 	{ "en_1000fdx_cap", { "", 0 },
426 	    link_01_vals, VALCNT(link_01_vals),
427 	    i_dladm_set_public_prop, NULL, i_dladm_binary_get, NULL,
428 	    0, DATALINK_CLASS_PHYS, DL_ETHER },
429 
430 	{ "adv_1000hdx_cap", { "", 0 },
431 	    link_01_vals, VALCNT(link_01_vals),
432 	    NULL, NULL, i_dladm_binary_get, NULL,
433 	    0, DATALINK_CLASS_PHYS, DL_ETHER },
434 
435 	{ "en_1000hdx_cap", { "", 0 },
436 	    link_01_vals, VALCNT(link_01_vals),
437 	    i_dladm_set_public_prop, NULL, i_dladm_binary_get, NULL,
438 	    0, DATALINK_CLASS_PHYS, DL_ETHER },
439 
440 	{ "adv_100fdx_cap", { "", 0 },
441 	    link_01_vals, VALCNT(link_01_vals),
442 	    NULL, NULL, i_dladm_binary_get, NULL,
443 	    0, DATALINK_CLASS_PHYS, DL_ETHER },
444 
445 	{ "en_100fdx_cap", { "", 0 },
446 	    link_01_vals, VALCNT(link_01_vals),
447 	    i_dladm_set_public_prop, NULL, i_dladm_binary_get, NULL,
448 	    0, DATALINK_CLASS_PHYS, DL_ETHER },
449 
450 	{ "adv_100hdx_cap", { "", 0 },
451 	    link_01_vals, VALCNT(link_01_vals),
452 	    NULL, NULL, i_dladm_binary_get, NULL,
453 	    0, DATALINK_CLASS_PHYS, DL_ETHER },
454 
455 	{ "en_100hdx_cap", { "", 0 },
456 	    link_01_vals, VALCNT(link_01_vals),
457 	    i_dladm_set_public_prop, NULL, i_dladm_binary_get, NULL,
458 	    0, DATALINK_CLASS_PHYS, DL_ETHER },
459 
460 	{ "adv_10fdx_cap", { "", 0 },
461 	    link_01_vals, VALCNT(link_01_vals),
462 	    NULL, NULL, i_dladm_binary_get, NULL,
463 	    0, DATALINK_CLASS_PHYS, DL_ETHER },
464 
465 	{ "en_10fdx_cap", { "", 0 },
466 	    link_01_vals, VALCNT(link_01_vals),
467 	    i_dladm_set_public_prop, NULL, i_dladm_binary_get, NULL,
468 	    0, DATALINK_CLASS_PHYS, DL_ETHER },
469 
470 	{ "adv_10hdx_cap", { "", 0 },
471 	    link_01_vals, VALCNT(link_01_vals),
472 	    NULL, NULL, i_dladm_binary_get, NULL,
473 	    0, DATALINK_CLASS_PHYS, DL_ETHER },
474 
475 	{ "en_10hdx_cap", { "", 0 },
476 	    link_01_vals, VALCNT(link_01_vals),
477 	    i_dladm_set_public_prop, NULL, i_dladm_binary_get, NULL,
478 	    0, DATALINK_CLASS_PHYS, DL_ETHER },
479 
480 	{ "maxbw", { "--", RESET_VAL }, NULL, 0,
481 	    do_set_res, NULL,
482 	    dld_maxbw_get, do_check_maxbw, PD_CHECK_ALLOC,
483 	    DATALINK_CLASS_ALL, DATALINK_ANY_MEDIATYPE },
484 
485 	{ "cpus", { "--", RESET_VAL }, NULL, 0,
486 	    do_set_cpus, NULL,
487 	    dld_cpus_get, do_check_cpus, 0,
488 	    DATALINK_CLASS_ALL, DATALINK_ANY_MEDIATYPE },
489 
490 	{ "priority", { "high", RESET_VAL },
491 	    link_priority_vals, VALCNT(link_priority_vals), do_set_res, NULL,
492 	    dld_priority_get, do_check_priority, PD_CHECK_ALLOC,
493 	    DATALINK_CLASS_ALL, DATALINK_ANY_MEDIATYPE },
494 };
495 
496 #define	DLADM_MAX_PROPS	(sizeof (prop_table) / sizeof (prop_desc_t))
497 
498 static resource_prop_t rsrc_prop_table[] = {
499 	{"maxbw",	do_extract_maxbw},
500 	{"priority",	do_extract_priority},
501 	{"cpus",	do_extract_cpus}
502 };
503 #define	DLADM_MAX_RSRC_PROP (sizeof (rsrc_prop_table) / \
504 	sizeof (resource_prop_t))
505 
506 /*
507  * when retrieving  private properties, we pass down a buffer with
508  * DLADM_PROP_BUF_CHUNK of space for the driver to return the property value.
509  */
510 #define	DLADM_PROP_BUF_CHUNK	1024
511 
512 static dladm_status_t	i_dladm_set_linkprop_db(datalink_id_t, const char *,
513 			    char **, uint_t);
514 static dladm_status_t	i_dladm_get_linkprop_db(datalink_id_t, const char *,
515 			    char **, uint_t *);
516 static dladm_status_t	i_dladm_set_single_prop(datalink_id_t, datalink_class_t,
517 			    uint32_t, prop_desc_t *, char **, uint_t, uint_t);
518 static dladm_status_t	i_dladm_set_linkprop(datalink_id_t, const char *,
519 			    char **, uint_t, uint_t);
520 static dladm_status_t	i_dladm_getset_defval(prop_desc_t *, datalink_id_t,
521 			    datalink_media_t, uint_t);
522 
523 static dladm_status_t	link_proplist_check(dladm_arg_list_t *);
524 
525 /*
526  * Unfortunately, MAX_SCAN_SUPPORT_RATES is too small to allow all
527  * rates to be retrieved. However, we cannot increase it at this
528  * time because it will break binary compatibility with unbundled
529  * WiFi drivers and utilities. So for now we define an additional
530  * constant, MAX_SUPPORT_RATES, to allow all rates to be retrieved.
531  */
532 #define	MAX_SUPPORT_RATES	64
533 
534 #define	AP_ANCHOR	"[anchor]"
535 #define	AP_DELIMITER	'.'
536 
537 static dladm_status_t
538 do_check_prop(prop_desc_t *pdp, char **prop_val, uint_t val_cnt,
539     val_desc_t *vdp)
540 {
541 	int		i, j;
542 	dladm_status_t	status = DLADM_STATUS_OK;
543 
544 	for (j = 0; j < val_cnt; j++) {
545 		for (i = 0; i < pdp->pd_noptval; i++) {
546 			if (strcasecmp(*prop_val,
547 			    pdp->pd_optval[i].vd_name) == 0) {
548 				break;
549 			}
550 		}
551 		if (i == pdp->pd_noptval) {
552 			status = DLADM_STATUS_BADVAL;
553 			goto done;
554 		}
555 		(void) memcpy(vdp + j, &pdp->pd_optval[i], sizeof (val_desc_t));
556 	}
557 
558 done:
559 	return (status);
560 }
561 
562 static dladm_status_t
563 i_dladm_set_single_prop(datalink_id_t linkid, datalink_class_t class,
564     uint32_t media, prop_desc_t *pdp, char **prop_val, uint_t val_cnt,
565     uint_t flags)
566 {
567 	dladm_status_t	status = DLADM_STATUS_OK;
568 	val_desc_t	*vdp = NULL;
569 	boolean_t	needfree = B_FALSE;
570 	uint_t		cnt, i;
571 
572 	if (!(pdp->pd_class & class))
573 		return (DLADM_STATUS_BADARG);
574 
575 	if (!DATALINK_MEDIA_ACCEPTED(pdp->pd_dmedia, media))
576 		return (DLADM_STATUS_BADARG);
577 
578 	if ((flags & DLADM_OPT_PERSIST) && (pdp->pd_flags & PD_TEMPONLY))
579 		return (DLADM_STATUS_TEMPONLY);
580 
581 	if (!(flags & DLADM_OPT_ACTIVE))
582 		return (DLADM_STATUS_OK);
583 
584 	if (pdp->pd_set == NULL)
585 		return (DLADM_STATUS_PROPRDONLY);
586 
587 	if (prop_val != NULL) {
588 		vdp = malloc(sizeof (val_desc_t) * val_cnt);
589 		if (vdp == NULL)
590 			return (DLADM_STATUS_NOMEM);
591 
592 		if (pdp->pd_check != NULL) {
593 			needfree = ((pdp->pd_flags & PD_CHECK_ALLOC) != 0);
594 			status = pdp->pd_check(pdp, linkid, prop_val, val_cnt,
595 			    vdp, media);
596 		} else if (pdp->pd_optval != NULL) {
597 			status = do_check_prop(pdp, prop_val, val_cnt, vdp);
598 		} else {
599 			status = DLADM_STATUS_BADARG;
600 		}
601 
602 		if (status != DLADM_STATUS_OK)
603 			goto done;
604 
605 		cnt = val_cnt;
606 	} else {
607 		boolean_t	defval = B_FALSE;
608 
609 		if (pdp->pd_defval.vd_name == NULL)
610 			return (DLADM_STATUS_NOTSUP);
611 
612 		cnt = 1;
613 		defval = (strlen(pdp->pd_defval.vd_name) > 0);
614 		if ((pdp->pd_flags & PD_CHECK_ALLOC) != 0 || defval) {
615 			if ((vdp = malloc(sizeof (val_desc_t))) == NULL)
616 				return (DLADM_STATUS_NOMEM);
617 
618 			if (defval) {
619 				(void) memcpy(vdp, &pdp->pd_defval,
620 				    sizeof (val_desc_t));
621 			} else if (pdp->pd_check != NULL) {
622 				status = pdp->pd_check(pdp, linkid, prop_val,
623 				    cnt, vdp, media);
624 				if (status != DLADM_STATUS_OK)
625 					goto done;
626 			}
627 		} else {
628 			status = i_dladm_getset_defval(pdp, linkid,
629 			    media, flags);
630 			return (status);
631 		}
632 	}
633 	status = pdp->pd_set(pdp, linkid, vdp, cnt, flags, media);
634 	if (needfree) {
635 		for (i = 0; i < cnt; i++)
636 			free((void *)((val_desc_t *)vdp + i)->vd_val);
637 	}
638 done:
639 	free(vdp);
640 	return (status);
641 }
642 
643 static dladm_status_t
644 i_dladm_set_linkprop(datalink_id_t linkid, const char *prop_name,
645     char **prop_val, uint_t val_cnt, uint_t flags)
646 {
647 	int			i;
648 	boolean_t		found = B_FALSE;
649 	datalink_class_t	class;
650 	uint32_t		media;
651 	dladm_status_t		status = DLADM_STATUS_OK;
652 
653 	status = dladm_datalink_id2info(linkid, NULL, &class, &media, NULL, 0);
654 	if (status != DLADM_STATUS_OK)
655 		return (status);
656 
657 	for (i = 0; i < DLADM_MAX_PROPS; i++) {
658 		prop_desc_t	*pdp = &prop_table[i];
659 		dladm_status_t	s;
660 
661 		if (prop_name != NULL &&
662 		    (strcasecmp(prop_name, pdp->pd_name) != 0))
663 			continue;
664 		found = B_TRUE;
665 		s = i_dladm_set_single_prop(linkid, class, media, pdp, prop_val,
666 		    val_cnt, flags);
667 
668 		if (prop_name != NULL) {
669 			status = s;
670 			break;
671 		} else {
672 			if (s != DLADM_STATUS_OK &&
673 			    s != DLADM_STATUS_NOTSUP)
674 				status = s;
675 		}
676 	}
677 	if (!found) {
678 		if (prop_name[0] == '_') {
679 			/* other private properties */
680 			status = i_dladm_set_prop(linkid, prop_name, prop_val,
681 			    val_cnt, flags);
682 		} else  {
683 			status = DLADM_STATUS_NOTFOUND;
684 		}
685 	}
686 
687 	return (status);
688 }
689 
690 /*
691  * Set/reset link property for specific link
692  */
693 dladm_status_t
694 dladm_set_linkprop(datalink_id_t linkid, const char *prop_name, char **prop_val,
695     uint_t val_cnt, uint_t flags)
696 {
697 	dladm_status_t	status = DLADM_STATUS_OK;
698 
699 	if ((linkid == DATALINK_INVALID_LINKID) || (flags == 0) ||
700 	    (prop_val == NULL && val_cnt > 0) ||
701 	    (prop_val != NULL && val_cnt == 0) ||
702 	    (prop_name == NULL && prop_val != NULL)) {
703 		return (DLADM_STATUS_BADARG);
704 	}
705 
706 	status = i_dladm_set_linkprop(linkid, prop_name, prop_val,
707 	    val_cnt, flags);
708 	if (status != DLADM_STATUS_OK)
709 		return (status);
710 
711 	if (flags & DLADM_OPT_PERSIST) {
712 		status = i_dladm_set_linkprop_db(linkid, prop_name,
713 		    prop_val, val_cnt);
714 	}
715 	return (status);
716 }
717 
718 /*
719  * Walk link properties of the given specific link.
720  */
721 dladm_status_t
722 dladm_walk_linkprop(datalink_id_t linkid, void *arg,
723     int (*func)(datalink_id_t, const char *, void *))
724 {
725 	dladm_status_t		status;
726 	datalink_class_t	class;
727 	uint_t			media;
728 	int			i;
729 
730 	if (linkid == DATALINK_INVALID_LINKID || func == NULL)
731 		return (DLADM_STATUS_BADARG);
732 
733 	status = dladm_datalink_id2info(linkid, NULL, &class, &media, NULL, 0);
734 	if (status != DLADM_STATUS_OK)
735 		return (status);
736 
737 	for (i = 0; i < DLADM_MAX_PROPS; i++) {
738 		if (!(prop_table[i].pd_class & class))
739 			continue;
740 
741 		if (!DATALINK_MEDIA_ACCEPTED(prop_table[i].pd_dmedia, media))
742 			continue;
743 
744 		if (func(linkid, prop_table[i].pd_name, arg) ==
745 		    DLADM_WALK_TERMINATE) {
746 			break;
747 		}
748 	}
749 
750 	return (DLADM_STATUS_OK);
751 }
752 
753 /*
754  * Get linkprop of the given specific link.
755  */
756 dladm_status_t
757 dladm_get_linkprop(datalink_id_t linkid, dladm_prop_type_t type,
758     const char *prop_name, char **prop_val, uint_t *val_cntp)
759 {
760 	dladm_status_t		status = DLADM_STATUS_OK;
761 	datalink_class_t	class;
762 	uint_t			media;
763 	prop_desc_t		*pdp;
764 	uint_t			cnt, dld_flags = 0;
765 	int			i;
766 	uint_t			perm_flags;
767 
768 	if (type == DLADM_PROP_VAL_DEFAULT)
769 		dld_flags = MAC_PROP_DEFAULT;
770 
771 	if (linkid == DATALINK_INVALID_LINKID || prop_name == NULL ||
772 	    prop_val == NULL || val_cntp == NULL || *val_cntp == 0)
773 		return (DLADM_STATUS_BADARG);
774 
775 	for (i = 0; i < DLADM_MAX_PROPS; i++)
776 		if (strcasecmp(prop_name, prop_table[i].pd_name) == 0)
777 			break;
778 
779 	if (i == DLADM_MAX_PROPS) {
780 		if (prop_name[0] == '_') {
781 			/*
782 			 * private property.
783 			 */
784 			return (i_dladm_get_prop(linkid, prop_name,
785 			    prop_val, val_cntp, type, dld_flags));
786 		} else {
787 			return (DLADM_STATUS_NOTFOUND);
788 		}
789 	}
790 
791 	pdp = &prop_table[i];
792 
793 	status = dladm_datalink_id2info(linkid, NULL, &class, &media, NULL, 0);
794 	if (status != DLADM_STATUS_OK)
795 		return (status);
796 
797 	if (!(pdp->pd_class & class))
798 		return (DLADM_STATUS_BADARG);
799 
800 	if (!DATALINK_MEDIA_ACCEPTED(pdp->pd_dmedia, media))
801 		return (DLADM_STATUS_BADARG);
802 
803 	switch (type) {
804 	case DLADM_PROP_VAL_CURRENT:
805 		status = pdp->pd_get(pdp, linkid, prop_val, val_cntp, media,
806 		    dld_flags, &perm_flags);
807 		break;
808 
809 	case DLADM_PROP_VAL_PERM:
810 		if (pdp->pd_set == NULL) {
811 			perm_flags = MAC_PROP_PERM_READ;
812 			*val_cntp = 1;
813 		} else {
814 			status = pdp->pd_get(pdp, linkid, prop_val, val_cntp,
815 			    media, dld_flags, &perm_flags);
816 		}
817 
818 		*prop_val[0] = '\0';
819 		if (status == DLADM_STATUS_OK)
820 			(void) dladm_perm2str(perm_flags, *prop_val);
821 		break;
822 
823 	case DLADM_PROP_VAL_DEFAULT:
824 		/*
825 		 * If defaults are not defined for the property,
826 		 * pd_defval.vd_name should be null. If the driver
827 		 * has to be contacted for the value, vd_name should
828 		 * be the empty string (""). Otherwise, dladm will
829 		 * just print whatever is in the table.
830 		 */
831 		if (pdp->pd_defval.vd_name == NULL) {
832 			status = DLADM_STATUS_NOTSUP;
833 			break;
834 		}
835 
836 		if (strlen(pdp->pd_defval.vd_name) == 0) {
837 			status = pdp->pd_get(pdp, linkid, prop_val, val_cntp,
838 			    media, dld_flags, &perm_flags);
839 		} else {
840 			(void) strcpy(*prop_val, pdp->pd_defval.vd_name);
841 		}
842 		*val_cntp = 1;
843 		break;
844 
845 	case DLADM_PROP_VAL_MODIFIABLE:
846 		if (pdp->pd_getmod != NULL) {
847 			status = pdp->pd_getmod(pdp, linkid, prop_val,
848 			    val_cntp, media, dld_flags, &perm_flags);
849 			break;
850 		}
851 		cnt = pdp->pd_noptval;
852 		if (cnt == 0) {
853 			status = DLADM_STATUS_NOTSUP;
854 		} else if (cnt > *val_cntp) {
855 			status = DLADM_STATUS_TOOSMALL;
856 		} else {
857 			for (i = 0; i < cnt; i++) {
858 				(void) strcpy(prop_val[i],
859 				    pdp->pd_optval[i].vd_name);
860 			}
861 			*val_cntp = cnt;
862 		}
863 		break;
864 	case DLADM_PROP_VAL_PERSISTENT:
865 		if (pdp->pd_flags & PD_TEMPONLY)
866 			return (DLADM_STATUS_TEMPONLY);
867 		status = i_dladm_get_linkprop_db(linkid, prop_name,
868 		    prop_val, val_cntp);
869 		break;
870 	default:
871 		status = DLADM_STATUS_BADARG;
872 		break;
873 	}
874 
875 	return (status);
876 }
877 
878 /*ARGSUSED*/
879 static int
880 i_dladm_init_one_prop(datalink_id_t linkid, const char *prop_name, void *arg)
881 {
882 	char	*buf, **propvals;
883 	uint_t	i, valcnt = DLADM_MAX_PROP_VALCNT;
884 
885 	if ((buf = malloc((sizeof (char *) + DLADM_PROP_VAL_MAX) *
886 	    DLADM_MAX_PROP_VALCNT)) == NULL) {
887 		return (DLADM_WALK_CONTINUE);
888 	}
889 
890 	propvals = (char **)(void *)buf;
891 	for (i = 0; i < valcnt; i++) {
892 		propvals[i] = buf +
893 		    sizeof (char *) * DLADM_MAX_PROP_VALCNT +
894 		    i * DLADM_PROP_VAL_MAX;
895 	}
896 
897 	if (dladm_get_linkprop(linkid, DLADM_PROP_VAL_PERSISTENT, prop_name,
898 	    propvals, &valcnt) != DLADM_STATUS_OK) {
899 		goto done;
900 	}
901 
902 	(void) dladm_set_linkprop(linkid, prop_name, propvals, valcnt,
903 	    DLADM_OPT_ACTIVE);
904 
905 done:
906 	if (buf != NULL)
907 		free(buf);
908 
909 	return (DLADM_WALK_CONTINUE);
910 }
911 
912 /*ARGSUSED*/
913 static int
914 i_dladm_init_linkprop(datalink_id_t linkid, void *arg)
915 {
916 	datalink_class_t	class;
917 	dladm_status_t		status;
918 
919 	status = dladm_datalink_id2info(linkid, NULL, &class, NULL, NULL, 0);
920 	if (status != DLADM_STATUS_OK)
921 		return (DLADM_WALK_TERMINATE);
922 
923 	if ((class & (DATALINK_CLASS_VNIC | DATALINK_CLASS_VLAN)) == 0)
924 		(void) dladm_init_linkprop(linkid, B_TRUE);
925 
926 	return (DLADM_WALK_CONTINUE);
927 }
928 
929 dladm_status_t
930 dladm_init_linkprop(datalink_id_t linkid, boolean_t any_media)
931 {
932 	datalink_media_t	dmedia;
933 	uint32_t		media;
934 
935 	dmedia = any_media ? DATALINK_ANY_MEDIATYPE : DL_WIFI;
936 
937 	if (linkid == DATALINK_ALL_LINKID) {
938 		(void) dladm_walk_datalink_id(i_dladm_init_linkprop, NULL,
939 		    DATALINK_CLASS_ALL, dmedia, DLADM_OPT_PERSIST);
940 	} else if (any_media || ((dladm_datalink_id2info(linkid, NULL, NULL,
941 	    &media, NULL, 0) == DLADM_STATUS_OK) &&
942 	    DATALINK_MEDIA_ACCEPTED(dmedia, media))) {
943 		(void) dladm_walk_linkprop(linkid, NULL, i_dladm_init_one_prop);
944 	}
945 	return (DLADM_STATUS_OK);
946 }
947 
948 /* ARGSUSED */
949 static dladm_status_t
950 do_get_zone(prop_desc_t *pdp, datalink_id_t linkid,
951     char **prop_val, uint_t *val_cnt, datalink_media_t media,
952     uint_t flags, uint_t *perm_flags)
953 {
954 	char			zone_name[ZONENAME_MAX];
955 	zoneid_t		zid;
956 	dladm_status_t		status;
957 	char			*cp;
958 	dld_ioc_macprop_t	*dip;
959 
960 	if (flags != 0)
961 		return (DLADM_STATUS_NOTSUP);
962 
963 	dip = i_dladm_get_public_prop(linkid, pdp->pd_name, flags,
964 	    &status, perm_flags);
965 	if (status != DLADM_STATUS_OK)
966 		return (status);
967 
968 	cp = dip->pr_val;
969 	(void) memcpy(&zid, cp, sizeof (zid));
970 	free(dip);
971 
972 	*val_cnt = 1;
973 	if (zid != GLOBAL_ZONEID) {
974 		if (getzonenamebyid(zid, zone_name, sizeof (zone_name)) < 0) {
975 			return (dladm_errno2status(errno));
976 		}
977 
978 		(void) strncpy(*prop_val, zone_name, DLADM_PROP_VAL_MAX);
979 	} else {
980 		*prop_val[0] = '\0';
981 	}
982 
983 	return (DLADM_STATUS_OK);
984 }
985 
986 typedef int (*zone_get_devroot_t)(char *, char *, size_t);
987 
988 static int
989 i_dladm_get_zone_dev(char *zone_name, char *dev, size_t devlen)
990 {
991 	char			root[MAXPATHLEN];
992 	zone_get_devroot_t	real_zone_get_devroot;
993 	void			*dlhandle;
994 	void			*sym;
995 	int			ret;
996 
997 	if ((dlhandle = dlopen("libzonecfg.so.1", RTLD_LAZY)) == NULL)
998 		return (-1);
999 
1000 	if ((sym = dlsym(dlhandle, "zone_get_devroot")) == NULL) {
1001 		(void) dlclose(dlhandle);
1002 		return (-1);
1003 	}
1004 
1005 	real_zone_get_devroot = (zone_get_devroot_t)sym;
1006 
1007 	if ((ret = real_zone_get_devroot(zone_name, root, sizeof (root))) == 0)
1008 		(void) snprintf(dev, devlen, "%s%s", root, "/dev");
1009 	(void) dlclose(dlhandle);
1010 	return (ret);
1011 }
1012 
1013 static dladm_status_t
1014 i_dladm_update_deventry(zoneid_t zid, datalink_id_t linkid, boolean_t add)
1015 {
1016 	char		path[MAXPATHLEN];
1017 	char		name[MAXLINKNAMELEN];
1018 	di_prof_t	prof = NULL;
1019 	char		zone_name[ZONENAME_MAX];
1020 	dladm_status_t	status;
1021 	int		ret;
1022 
1023 	if (getzonenamebyid(zid, zone_name, sizeof (zone_name)) < 0)
1024 		return (dladm_errno2status(errno));
1025 	if (i_dladm_get_zone_dev(zone_name, path, sizeof (path)) != 0)
1026 		return (dladm_errno2status(errno));
1027 	if (di_prof_init(path, &prof) != 0)
1028 		return (dladm_errno2status(errno));
1029 
1030 	status = dladm_linkid2legacyname(linkid, name, MAXLINKNAMELEN);
1031 	if (status != DLADM_STATUS_OK)
1032 		goto cleanup;
1033 
1034 	if (add)
1035 		ret = di_prof_add_dev(prof, name);
1036 	else
1037 		ret = di_prof_add_exclude(prof, name);
1038 
1039 	if (ret != 0) {
1040 		status = dladm_errno2status(errno);
1041 		goto cleanup;
1042 	}
1043 
1044 	if (di_prof_commit(prof) != 0)
1045 		status = dladm_errno2status(errno);
1046 cleanup:
1047 	if (prof)
1048 		di_prof_fini(prof);
1049 
1050 	return (status);
1051 }
1052 
1053 /* ARGSUSED */
1054 static dladm_status_t
1055 do_set_zone(prop_desc_t *pdp, datalink_id_t linkid, val_desc_t *vdp,
1056     uint_t val_cnt, uint_t flags, datalink_media_t media)
1057 {
1058 	dladm_status_t		status = DLADM_STATUS_OK;
1059 	zoneid_t		zid_old, zid_new;
1060 	char			link[MAXLINKNAMELEN];
1061 	char			*cp;
1062 	dld_ioc_macprop_t	*dip;
1063 	dld_ioc_zid_t		*dzp;
1064 
1065 	if (val_cnt != 1)
1066 		return (DLADM_STATUS_BADVALCNT);
1067 
1068 	dzp = (dld_ioc_zid_t *)vdp->vd_val;
1069 
1070 	dip = i_dladm_get_public_prop(linkid, pdp->pd_name, flags,
1071 	    &status, NULL);
1072 	if (status != DLADM_STATUS_OK)
1073 		return (status);
1074 
1075 	cp = dip->pr_val;
1076 	(void) memcpy(&zid_old, cp, sizeof (zid_old));
1077 	free(dip);
1078 
1079 	zid_new = dzp->diz_zid;
1080 	(void) strlcpy(link, dzp->diz_link, MAXLINKNAMELEN);
1081 
1082 	/* Do nothing if setting to current value */
1083 	if (zid_new == zid_old)
1084 		return (status);
1085 
1086 	if (zid_new != GLOBAL_ZONEID) {
1087 		/*
1088 		 * If the new zoneid is the global zone, we could destroy
1089 		 * the link (in the case of an implicitly-created VLAN) as a
1090 		 * result of setting the zoneid. In that case, we defer the
1091 		 * operation to the end of this function to avoid recreating
1092 		 * the VLAN and getting a different linkid during the rollback
1093 		 * if other operation fails.
1094 		 *
1095 		 * Otherwise, this operation will hold a reference to the
1096 		 * link and prevent a link renaming, so we need to do it
1097 		 * before other operations.
1098 		 */
1099 		status = i_dladm_set_public_prop(pdp, linkid, vdp, val_cnt,
1100 		    flags, media);
1101 		if (status != DLADM_STATUS_OK)
1102 			return (status);
1103 	}
1104 
1105 	if (zid_old != GLOBAL_ZONEID) {
1106 		if (zone_remove_datalink(zid_old, link) != 0 &&
1107 		    errno != ENXIO) {
1108 			status = dladm_errno2status(errno);
1109 			goto rollback1;
1110 		}
1111 
1112 		/*
1113 		 * It is okay to fail to update the /dev entry (some
1114 		 * vanity-named links do not have a /dev entry).
1115 		 */
1116 		(void) i_dladm_update_deventry(zid_old, linkid, B_FALSE);
1117 	}
1118 
1119 	if (zid_new != GLOBAL_ZONEID) {
1120 		if (zone_add_datalink(zid_new, link) != 0) {
1121 			status = dladm_errno2status(errno);
1122 			goto rollback2;
1123 		}
1124 
1125 		(void) i_dladm_update_deventry(zid_new, linkid, B_TRUE);
1126 	} else {
1127 		status = i_dladm_set_public_prop(pdp, linkid, vdp, val_cnt,
1128 		    flags, media);
1129 		if (status != DLADM_STATUS_OK)
1130 			goto rollback2;
1131 	}
1132 
1133 	return (DLADM_STATUS_OK);
1134 
1135 rollback2:
1136 	if (zid_old != GLOBAL_ZONEID)
1137 		(void) i_dladm_update_deventry(zid_old, linkid, B_TRUE);
1138 	if (zid_old != GLOBAL_ZONEID)
1139 		(void) zone_add_datalink(zid_old, link);
1140 rollback1:
1141 	if (zid_new != GLOBAL_ZONEID) {
1142 		dzp->diz_zid = zid_old;
1143 		(void) i_dladm_set_public_prop(pdp, linkid, vdp, val_cnt,
1144 		    flags, media);
1145 	}
1146 
1147 	return (status);
1148 }
1149 
1150 /* ARGSUSED */
1151 static dladm_status_t
1152 do_check_zone(prop_desc_t *pdp, datalink_id_t linkid, char **prop_val,
1153     uint_t val_cnt, val_desc_t *vdp, datalink_media_t media)
1154 {
1155 	char		*zone_name;
1156 	char		linkname[MAXLINKNAMELEN];
1157 	zoneid_t	zoneid;
1158 	dladm_status_t	status = DLADM_STATUS_OK;
1159 	dld_ioc_zid_t	*dzp;
1160 
1161 	if (val_cnt != 1)
1162 		return (DLADM_STATUS_BADVALCNT);
1163 
1164 	dzp = malloc(sizeof (dld_ioc_zid_t));
1165 	if (dzp == NULL)
1166 		return (DLADM_STATUS_NOMEM);
1167 
1168 	if ((status = dladm_datalink_id2info(linkid, NULL, NULL, NULL,
1169 	    linkname, MAXLINKNAMELEN)) != DLADM_STATUS_OK) {
1170 		goto done;
1171 	}
1172 
1173 	zone_name = (prop_val != NULL) ? *prop_val : GLOBAL_ZONENAME;
1174 	if (strlen(linkname) > MAXLINKNAMELEN) {
1175 		status = DLADM_STATUS_BADVAL;
1176 		goto done;
1177 	}
1178 
1179 	if ((zoneid = getzoneidbyname(zone_name)) == -1) {
1180 		status = DLADM_STATUS_BADVAL;
1181 		goto done;
1182 	}
1183 
1184 	if (zoneid != GLOBAL_ZONEID) {
1185 		ushort_t	flags;
1186 
1187 		if (zone_getattr(zoneid, ZONE_ATTR_FLAGS, &flags,
1188 		    sizeof (flags)) < 0) {
1189 			status = dladm_errno2status(errno);
1190 			goto done;
1191 		}
1192 
1193 		if (!(flags & ZF_NET_EXCL)) {
1194 			status = DLADM_STATUS_BADVAL;
1195 			goto done;
1196 		}
1197 	}
1198 
1199 	(void) memset(dzp, 0, sizeof (dld_ioc_zid_t));
1200 
1201 	dzp->diz_zid = zoneid;
1202 	(void) strlcpy(dzp->diz_link, linkname, MAXLINKNAMELEN);
1203 
1204 	vdp->vd_val = (uintptr_t)dzp;
1205 	return (DLADM_STATUS_OK);
1206 done:
1207 	free(dzp);
1208 	return (status);
1209 }
1210 
1211 /* ARGSUSED */
1212 static dladm_status_t
1213 dld_maxbw_get(prop_desc_t *pdp, datalink_id_t linkid,
1214     char **prop_val, uint_t *val_cnt, datalink_media_t media,
1215     uint_t flags, uint_t *perm_flags)
1216 {
1217 	dld_ioc_macprop_t	*dip;
1218 	mac_resource_props_t	mrp;
1219 	dladm_status_t		status;
1220 
1221 	dip = i_dladm_get_public_prop(linkid, pdp->pd_name, flags,
1222 	    &status, perm_flags);
1223 	if (dip == NULL)
1224 		return (status);
1225 
1226 	bcopy(dip->pr_val, &mrp, sizeof (mac_resource_props_t));
1227 	free(dip);
1228 
1229 	if ((mrp.mrp_mask & MRP_MAXBW) == 0) {
1230 		(*prop_val)[0] = '\0';
1231 	} else {
1232 		(void) dladm_bw2str(mrp.mrp_maxbw, prop_val[0]);
1233 	}
1234 	*val_cnt = 1;
1235 	return (DLADM_STATUS_OK);
1236 }
1237 
1238 /* ARGSUSED */
1239 static dladm_status_t
1240 do_check_maxbw(prop_desc_t *pdp, datalink_id_t linkid, char **prop_val,
1241     uint_t val_cnt, val_desc_t *vdp, datalink_media_t media)
1242 {
1243 	uint64_t	*maxbw;
1244 	dladm_status_t	status = DLADM_STATUS_OK;
1245 
1246 	if (val_cnt != 1)
1247 		return (DLADM_STATUS_BADVALCNT);
1248 
1249 	maxbw = malloc(sizeof (uint64_t));
1250 	if (maxbw == NULL)
1251 		return (DLADM_STATUS_NOMEM);
1252 
1253 	status = dladm_str2bw(*prop_val, maxbw);
1254 	if (status != DLADM_STATUS_OK) {
1255 		free(maxbw);
1256 		return (status);
1257 	}
1258 
1259 	if ((*maxbw < MRP_MAXBW_MINVAL) && (*maxbw != 0)) {
1260 		free(maxbw);
1261 		return (DLADM_STATUS_MINMAXBW);
1262 	}
1263 
1264 	vdp->vd_val = (uintptr_t)maxbw;
1265 	return (DLADM_STATUS_OK);
1266 }
1267 
1268 /* ARGSUSED */
1269 dladm_status_t
1270 do_extract_maxbw(val_desc_t *vdp, void *arg, uint_t cnt)
1271 {
1272 	mac_resource_props_t *mrp = (mac_resource_props_t *)arg;
1273 
1274 	bcopy((char *)vdp->vd_val, &mrp->mrp_maxbw, sizeof (uint64_t));
1275 	mrp->mrp_mask |= MRP_MAXBW;
1276 
1277 	return (DLADM_STATUS_OK);
1278 }
1279 
1280 /* ARGSUSED */
1281 static dladm_status_t
1282 dld_cpus_get(prop_desc_t *pdp, datalink_id_t linkid,
1283     char **prop_val, uint_t *val_cnt, datalink_media_t media,
1284     uint_t flags, uint_t *perm_flags)
1285 {
1286 	dld_ioc_macprop_t	*dip;
1287 	mac_resource_props_t	mrp;
1288 	int			i;
1289 	uint32_t		ncpus;
1290 	uchar_t			*cp;
1291 	dladm_status_t		status;
1292 
1293 	dip = i_dladm_get_public_prop(linkid, pdp->pd_name, flags,
1294 	    &status, perm_flags);
1295 	if (dip == NULL)
1296 		return (status);
1297 
1298 	cp = (uchar_t *)dip->pr_val;
1299 	(void) memcpy(&mrp, cp, sizeof (mac_resource_props_t));
1300 	free(dip);
1301 
1302 	ncpus = mrp.mrp_ncpus;
1303 
1304 	if (ncpus > *val_cnt)
1305 		return (DLADM_STATUS_TOOSMALL);
1306 
1307 	if (ncpus == 0) {
1308 		(*prop_val)[0] = '\0';
1309 		*val_cnt = 1;
1310 		return (DLADM_STATUS_OK);
1311 	}
1312 
1313 	*val_cnt = ncpus;
1314 	for (i = 0; i < ncpus; i++) {
1315 		(void) snprintf(prop_val[i], DLADM_PROP_VAL_MAX,
1316 		    "%u", mrp.mrp_cpu[i]);
1317 	}
1318 	return (DLADM_STATUS_OK);
1319 }
1320 
1321 /* ARGSUSED */
1322 static dladm_status_t
1323 do_set_res(prop_desc_t *pdp, datalink_id_t linkid, val_desc_t *vdp,
1324     uint_t val_cnt, uint_t flags, datalink_media_t media)
1325 {
1326 	mac_resource_props_t	mrp;
1327 	dladm_status_t		status = DLADM_STATUS_OK;
1328 	dld_ioc_macprop_t	*dip;
1329 
1330 	bzero(&mrp, sizeof (mac_resource_props_t));
1331 	dip = i_dladm_buf_alloc_by_name(0, linkid, pdp->pd_name,
1332 	    flags, &status);
1333 
1334 	if (dip == NULL)
1335 		return (status);
1336 
1337 	if (vdp->vd_val == RESET_VAL) {
1338 		switch (dip->pr_num) {
1339 		case MAC_PROP_MAXBW:
1340 			mrp.mrp_maxbw = MRP_MAXBW_RESETVAL;
1341 			mrp.mrp_mask = MRP_MAXBW;
1342 			break;
1343 		case MAC_PROP_PRIO:
1344 			mrp.mrp_priority = MPL_RESET;
1345 			mrp.mrp_mask = MRP_PRIORITY;
1346 			break;
1347 		default:
1348 			free(dip);
1349 			return (DLADM_STATUS_BADARG);
1350 		}
1351 	} else {
1352 		switch (dip->pr_num) {
1353 		case MAC_PROP_MAXBW:
1354 			bcopy((void *)vdp->vd_val, &mrp.mrp_maxbw,
1355 			    sizeof (uint64_t));
1356 			mrp.mrp_mask = MRP_MAXBW;
1357 			break;
1358 		case MAC_PROP_PRIO:
1359 			bcopy((void *)vdp->vd_val, &mrp.mrp_priority,
1360 			    sizeof (mac_priority_level_t));
1361 			mrp.mrp_mask = MRP_PRIORITY;
1362 			break;
1363 		default:
1364 			free(dip);
1365 			return (DLADM_STATUS_BADARG);
1366 		}
1367 	}
1368 
1369 	(void) memcpy(dip->pr_val, &mrp, dip->pr_valsize);
1370 	status = i_dladm_macprop(dip, B_TRUE);
1371 	free(dip);
1372 	return (status);
1373 }
1374 
1375 /* ARGSUSED */
1376 static dladm_status_t
1377 do_set_cpus(prop_desc_t *pdp, datalink_id_t linkid, val_desc_t *vdp,
1378     uint_t val_cnt, uint_t flags, datalink_media_t media)
1379 {
1380 	mac_resource_props_t	mrp;
1381 	dladm_status_t		status;
1382 	dld_ioc_macprop_t	*dip;
1383 	datalink_class_t	class;
1384 
1385 	/*
1386 	 * CPU bindings can be set on VNIC and regular physical links.
1387 	 * However VNICs fails the dladm_phys_info test(). So apply
1388 	 * the phys_info test only on physical links.
1389 	 */
1390 	if ((status = dladm_datalink_id2info(linkid, NULL, &class,
1391 	    NULL, NULL, 0)) != DLADM_STATUS_OK) {
1392 		return (status);
1393 	}
1394 
1395 	/*
1396 	 * We set intr_cpu to -1. The interrupt will be retargetted,
1397 	 * if possible when the setup is complete in MAC.
1398 	 */
1399 	bzero(&mrp, sizeof (mac_resource_props_t));
1400 	mrp.mrp_mask = MRP_CPUS;
1401 	if (vdp != NULL && vdp->vd_val != RESET_VAL) {
1402 		mac_resource_props_t	*vmrp;
1403 
1404 		vmrp = (mac_resource_props_t *)vdp->vd_val;
1405 		if (vmrp->mrp_ncpus > 0) {
1406 			bcopy(vmrp, &mrp, sizeof (mac_resource_props_t));
1407 			mrp.mrp_mask = MRP_CPUS;
1408 		}
1409 		mrp.mrp_mask |= MRP_CPUS_USERSPEC;
1410 		mrp.mrp_fanout_mode = MCM_CPUS;
1411 		mrp.mrp_intr_cpu = -1;
1412 	}
1413 
1414 	dip = i_dladm_buf_alloc_by_name(0, linkid, pdp->pd_name,
1415 	    flags, &status);
1416 	if (dip == NULL)
1417 		return (status);
1418 
1419 	(void) memcpy(dip->pr_val, &mrp, dip->pr_valsize);
1420 	status = i_dladm_macprop(dip, B_TRUE);
1421 	free(dip);
1422 	return (status);
1423 }
1424 
1425 /* ARGSUSED */
1426 static dladm_status_t
1427 do_check_cpus(prop_desc_t *pdp, datalink_id_t linkid, char **prop_val,
1428     uint_t val_cnt, val_desc_t *vdp, datalink_media_t media)
1429 {
1430 	uint32_t		cpuid;
1431 	int			i, j, rc;
1432 	long			nproc = sysconf(_SC_NPROCESSORS_CONF);
1433 	mac_resource_props_t	*mrp;
1434 
1435 	mrp = malloc(sizeof (mac_resource_props_t));
1436 	if (mrp == NULL)
1437 		return (DLADM_STATUS_NOMEM);
1438 
1439 	for (i = 0; i < val_cnt; i++) {
1440 		errno = 0;
1441 		cpuid = strtol(prop_val[i], (char **)NULL, 10);
1442 		if (errno != 0 || cpuid >= nproc) {
1443 			free(mrp);
1444 			return (DLADM_STATUS_CPUMAX);
1445 		}
1446 		rc = p_online(cpuid, P_STATUS);
1447 		if (rc < 1) {
1448 			free(mrp);
1449 			return (DLADM_STATUS_CPUERR);
1450 		}
1451 		if (rc != P_ONLINE) {
1452 			free(mrp);
1453 			return (DLADM_STATUS_CPUNOTONLINE);
1454 		}
1455 		mrp->mrp_cpu[i] = cpuid;
1456 	}
1457 	mrp->mrp_ncpus = (uint32_t)val_cnt;
1458 
1459 	/* Check for duplicates */
1460 	for (i = 0; i < val_cnt; i++) {
1461 		for (j = 0; j < val_cnt; j++) {
1462 			if (i != j && mrp->mrp_cpu[i] == mrp->mrp_cpu[j]) {
1463 				free(mrp);
1464 				return (DLADM_STATUS_BADARG);
1465 			}
1466 		}
1467 	}
1468 	vdp->vd_val = (uintptr_t)mrp;
1469 
1470 	return (DLADM_STATUS_OK);
1471 }
1472 
1473 /* ARGSUSED */
1474 dladm_status_t
1475 do_extract_cpus(val_desc_t *vdp, void *arg, uint_t cnt)
1476 {
1477 	mac_resource_props_t	*mrp = (mac_resource_props_t *)arg;
1478 	mac_resource_props_t	*vmrp = (mac_resource_props_t *)vdp->vd_val;
1479 	int			i;
1480 
1481 	for (i = 0; i < vmrp->mrp_ncpus; i++) {
1482 		mrp->mrp_cpu[i] = vmrp->mrp_cpu[i];
1483 	}
1484 	mrp->mrp_ncpus = vmrp->mrp_ncpus;
1485 	mrp->mrp_mask |= (MRP_CPUS|MRP_CPUS_USERSPEC);
1486 	mrp->mrp_fanout_mode = MCM_CPUS;
1487 
1488 	return (DLADM_STATUS_OK);
1489 }
1490 
1491 /* ARGSUSED */
1492 static dladm_status_t
1493 dld_priority_get(prop_desc_t *pdp, datalink_id_t linkid,
1494     char **prop_val, uint_t *val_cnt, datalink_media_t media,
1495     uint_t flags, uint_t *perm_flags)
1496 {
1497 	dld_ioc_macprop_t	*dip;
1498 	mac_resource_props_t	mrp;
1499 	mac_priority_level_t	pri;
1500 	dladm_status_t		status;
1501 
1502 	dip = i_dladm_get_public_prop(linkid, pdp->pd_name, flags,
1503 	    &status, perm_flags);
1504 	if (dip == NULL)
1505 		return (status);
1506 
1507 	bcopy(dip->pr_val, &mrp, sizeof (mac_resource_props_t));
1508 	free(dip);
1509 
1510 	pri = ((mrp.mrp_mask & MRP_PRIORITY) == 0) ? MPL_HIGH :
1511 	    mrp.mrp_priority;
1512 
1513 	(void) dladm_pri2str(pri, prop_val[0]);
1514 	*val_cnt = 1;
1515 	return (DLADM_STATUS_OK);
1516 }
1517 
1518 /* ARGSUSED */
1519 static dladm_status_t
1520 do_check_priority(prop_desc_t *pdp, datalink_id_t linkid, char **prop_val,
1521     uint_t val_cnt, val_desc_t *vdp, datalink_media_t media)
1522 {
1523 	mac_priority_level_t	*pri;
1524 	dladm_status_t	status = DLADM_STATUS_OK;
1525 
1526 	if (val_cnt != 1)
1527 		return (DLADM_STATUS_BADVALCNT);
1528 
1529 	pri = malloc(sizeof (mac_priority_level_t));
1530 	if (pri == NULL)
1531 		return (DLADM_STATUS_NOMEM);
1532 
1533 	status = dladm_str2pri(*prop_val, pri);
1534 	if (status != DLADM_STATUS_OK) {
1535 		free(pri);
1536 		return (status);
1537 	}
1538 
1539 	if (*pri < MPL_LOW || *pri > MPL_HIGH) {
1540 		free(pri);
1541 		return (DLADM_STATUS_BADVAL);
1542 	}
1543 
1544 	vdp->vd_val = (uintptr_t)pri;
1545 	return (DLADM_STATUS_OK);
1546 }
1547 
1548 /* ARGSUSED */
1549 dladm_status_t
1550 do_extract_priority(val_desc_t *vdp, void *arg, uint_t cnt)
1551 {
1552 	mac_resource_props_t *mrp = (mac_resource_props_t *)arg;
1553 
1554 	bcopy((char *)vdp->vd_val, &mrp->mrp_priority,
1555 	    sizeof (mac_priority_level_t));
1556 	mrp->mrp_mask |= MRP_PRIORITY;
1557 
1558 	return (DLADM_STATUS_OK);
1559 }
1560 
1561 /* ARGSUSED */
1562 static dladm_status_t
1563 do_get_autopush(prop_desc_t *pdp, datalink_id_t linkid,
1564     char **prop_val, uint_t *val_cnt, datalink_media_t media,
1565     uint_t flags, uint_t *perm_flags)
1566 {
1567 	struct		dlautopush dlap;
1568 	int		i, len;
1569 	dladm_status_t	status;
1570 	dld_ioc_macprop_t	*dip;
1571 
1572 	if (flags & MAC_PROP_DEFAULT)
1573 		return (DLADM_STATUS_NOTDEFINED);
1574 
1575 	*val_cnt = 1;
1576 	dip = i_dladm_get_public_prop(linkid, pdp->pd_name, flags,
1577 	    &status, perm_flags);
1578 	if (dip == NULL) {
1579 		(*prop_val)[0] = '\0';
1580 		return (DLADM_STATUS_OK);
1581 	}
1582 	(void) memcpy(&dlap, dip->pr_val, sizeof (dlap));
1583 
1584 	for (i = 0, len = 0; i < dlap.dap_npush; i++) {
1585 		if (i != 0) {
1586 			(void) snprintf(*prop_val + len,
1587 			    DLADM_PROP_VAL_MAX - len, "%c", AP_DELIMITER);
1588 			len += 1;
1589 		}
1590 		(void) snprintf(*prop_val + len, DLADM_PROP_VAL_MAX - len,
1591 		    "%s", dlap.dap_aplist[i]);
1592 		len += strlen(dlap.dap_aplist[i]);
1593 		if (dlap.dap_anchor - 1 == i) {
1594 			(void) snprintf(*prop_val + len,
1595 			    DLADM_PROP_VAL_MAX - len, "%c%s", AP_DELIMITER,
1596 			    AP_ANCHOR);
1597 			len += (strlen(AP_ANCHOR) + 1);
1598 		}
1599 	}
1600 	free(dip);
1601 done:
1602 	return (DLADM_STATUS_OK);
1603 }
1604 
1605 /*
1606  * Add the specified module to the dlautopush structure; returns a
1607  * DLADM_STATUS_* code.
1608  */
1609 dladm_status_t
1610 i_dladm_add_ap_module(const char *module, struct dlautopush *dlap)
1611 {
1612 	if ((strlen(module) == 0) || (strlen(module) > FMNAMESZ))
1613 		return (DLADM_STATUS_BADVAL);
1614 
1615 	if (strncasecmp(module, AP_ANCHOR, strlen(AP_ANCHOR)) == 0) {
1616 		/*
1617 		 * We don't allow multiple anchors, and the anchor must
1618 		 * be after at least one module.
1619 		 */
1620 		if (dlap->dap_anchor != 0)
1621 			return (DLADM_STATUS_BADVAL);
1622 		if (dlap->dap_npush == 0)
1623 			return (DLADM_STATUS_BADVAL);
1624 
1625 		dlap->dap_anchor = dlap->dap_npush;
1626 		return (DLADM_STATUS_OK);
1627 	}
1628 	if (dlap->dap_npush > MAXAPUSH)
1629 		return (DLADM_STATUS_BADVALCNT);
1630 
1631 	(void) strlcpy(dlap->dap_aplist[dlap->dap_npush++], module,
1632 	    FMNAMESZ + 1);
1633 
1634 	return (DLADM_STATUS_OK);
1635 }
1636 
1637 /*
1638  * Currently, both '.' and ' '(space) can be used as the delimiters between
1639  * autopush modules. The former is used in dladm set-linkprop, and the
1640  * latter is used in the autopush(1M) file.
1641  */
1642 /* ARGSUSED */
1643 static dladm_status_t
1644 do_check_autopush(prop_desc_t *pdp, datalink_id_t linkid, char **prop_val,
1645     uint_t val_cnt, val_desc_t *vdp, datalink_media_t media)
1646 {
1647 	char			*module;
1648 	struct dlautopush	*dlap;
1649 	dladm_status_t		status;
1650 	char			val[DLADM_PROP_VAL_MAX];
1651 	char			delimiters[4];
1652 
1653 	if (val_cnt != 1)
1654 		return (DLADM_STATUS_BADVALCNT);
1655 
1656 	if (prop_val != NULL) {
1657 		dlap = malloc(sizeof (struct dlautopush));
1658 		if (dlap == NULL)
1659 			return (DLADM_STATUS_NOMEM);
1660 
1661 		(void) memset(dlap, 0, sizeof (struct dlautopush));
1662 		(void) snprintf(delimiters, 4, " %c\n", AP_DELIMITER);
1663 		bcopy(*prop_val, val, DLADM_PROP_VAL_MAX);
1664 		module = strtok(val, delimiters);
1665 		while (module != NULL) {
1666 			status = i_dladm_add_ap_module(module, dlap);
1667 			if (status != DLADM_STATUS_OK)
1668 				return (status);
1669 			module = strtok(NULL, delimiters);
1670 		}
1671 
1672 		vdp->vd_val = (uintptr_t)dlap;
1673 	} else {
1674 		vdp->vd_val = 0;
1675 	}
1676 	return (DLADM_STATUS_OK);
1677 }
1678 
1679 #define	WLDP_BUFSIZE (MAX_BUF_LEN - WIFI_BUF_OFFSET)
1680 
1681 /* ARGSUSED */
1682 static dladm_status_t
1683 do_get_rate_common(prop_desc_t *pdp, datalink_id_t linkid,
1684     char **prop_val, uint_t *val_cnt, uint_t id, uint_t *perm_flags)
1685 {
1686 	wl_rates_t	*wrp;
1687 	uint_t		i;
1688 	dladm_status_t	status = DLADM_STATUS_OK;
1689 
1690 	wrp = malloc(WLDP_BUFSIZE);
1691 	if (wrp == NULL)
1692 		return (DLADM_STATUS_NOMEM);
1693 
1694 	status = i_dladm_wlan_param(linkid, wrp, id, WLDP_BUFSIZE, B_FALSE);
1695 	if (status != DLADM_STATUS_OK)
1696 		goto done;
1697 
1698 	if (wrp->wl_rates_num > *val_cnt) {
1699 		status = DLADM_STATUS_TOOSMALL;
1700 		goto done;
1701 	}
1702 
1703 	if (wrp->wl_rates_rates[0] == 0) {
1704 		prop_val[0][0] = '\0';
1705 		*val_cnt = 1;
1706 		goto done;
1707 	}
1708 
1709 	for (i = 0; i < wrp->wl_rates_num; i++) {
1710 		(void) snprintf(prop_val[i], DLADM_STRSIZE, "%.*f",
1711 		    wrp->wl_rates_rates[i] % 2,
1712 		    (float)wrp->wl_rates_rates[i] / 2);
1713 	}
1714 	*val_cnt = wrp->wl_rates_num;
1715 	*perm_flags = MAC_PROP_PERM_RW;
1716 
1717 done:
1718 	free(wrp);
1719 	return (status);
1720 }
1721 
1722 static dladm_status_t
1723 do_get_rate_prop(prop_desc_t *pdp, datalink_id_t linkid,
1724     char **prop_val, uint_t *val_cnt, datalink_media_t media,
1725     uint_t flags, uint_t *perm_flags)
1726 {
1727 	if (media != DL_WIFI) {
1728 		return (i_dladm_speed_get(pdp, linkid, prop_val,
1729 		    val_cnt, flags, perm_flags));
1730 	}
1731 
1732 	return (do_get_rate_common(pdp, linkid, prop_val, val_cnt,
1733 	    MAC_PROP_WL_DESIRED_RATES, perm_flags));
1734 }
1735 
1736 /* ARGSUSED */
1737 static dladm_status_t
1738 do_get_rate_mod(prop_desc_t *pdp, datalink_id_t linkid,
1739     char **prop_val, uint_t *val_cnt, datalink_media_t media,
1740     uint_t flags, uint_t *perm_flags)
1741 {
1742 	switch (media) {
1743 	case DL_ETHER:
1744 		/*
1745 		 * Speed for ethernet links is unbounded. E.g., 802.11b
1746 		 * links can have a speed of 5.5 Gbps.
1747 		 */
1748 		return (DLADM_STATUS_NOTSUP);
1749 
1750 	case DL_WIFI:
1751 		return (do_get_rate_common(pdp, linkid, prop_val, val_cnt,
1752 		    MAC_PROP_WL_SUPPORTED_RATES, perm_flags));
1753 	default:
1754 		return (DLADM_STATUS_BADARG);
1755 	}
1756 }
1757 
1758 static dladm_status_t
1759 do_set_rate(datalink_id_t linkid, dladm_wlan_rates_t *rates)
1760 {
1761 	int		i;
1762 	uint_t		len;
1763 	wl_rates_t	*wrp;
1764 	dladm_status_t	status = DLADM_STATUS_OK;
1765 
1766 	wrp = malloc(WLDP_BUFSIZE);
1767 	if (wrp == NULL)
1768 		return (DLADM_STATUS_NOMEM);
1769 
1770 	bzero(wrp, WLDP_BUFSIZE);
1771 	for (i = 0; i < rates->wr_cnt; i++)
1772 		wrp->wl_rates_rates[i] = rates->wr_rates[i];
1773 	wrp->wl_rates_num = rates->wr_cnt;
1774 
1775 	len = offsetof(wl_rates_t, wl_rates_rates) +
1776 	    (rates->wr_cnt * sizeof (char)) + WIFI_BUF_OFFSET;
1777 	status = i_dladm_wlan_param(linkid, wrp, MAC_PROP_WL_DESIRED_RATES,
1778 	    len, B_TRUE);
1779 
1780 	free(wrp);
1781 	return (status);
1782 }
1783 
1784 /* ARGSUSED */
1785 static dladm_status_t
1786 do_set_rate_prop(prop_desc_t *pdp, datalink_id_t linkid,
1787     val_desc_t *vdp, uint_t val_cnt, uint_t flags, datalink_media_t media)
1788 {
1789 	dladm_wlan_rates_t	rates;
1790 	dladm_status_t		status;
1791 
1792 	/*
1793 	 * can currently set rate on WIFI links only.
1794 	 */
1795 	if (media != DL_WIFI)
1796 		return (DLADM_STATUS_PROPRDONLY);
1797 
1798 	if (val_cnt != 1)
1799 		return (DLADM_STATUS_BADVALCNT);
1800 
1801 	rates.wr_cnt = 1;
1802 	rates.wr_rates[0] = vdp[0].vd_val;
1803 
1804 	status = do_set_rate(linkid, &rates);
1805 
1806 done:
1807 	return (status);
1808 }
1809 
1810 /* ARGSUSED */
1811 static dladm_status_t
1812 do_check_rate(prop_desc_t *pdp, datalink_id_t linkid, char **prop_val,
1813     uint_t val_cnt, val_desc_t *vdp, datalink_media_t media)
1814 {
1815 	int		i;
1816 	uint_t		modval_cnt = MAX_SUPPORT_RATES;
1817 	char		*buf, **modval;
1818 	dladm_status_t	status;
1819 	uint_t 		perm_flags;
1820 
1821 	if (val_cnt != 1)
1822 		return (DLADM_STATUS_BADVALCNT);
1823 
1824 	buf = malloc((sizeof (char *) + DLADM_STRSIZE) *
1825 	    MAX_SUPPORT_RATES);
1826 	if (buf == NULL) {
1827 		status = DLADM_STATUS_NOMEM;
1828 		goto done;
1829 	}
1830 
1831 	modval = (char **)(void *)buf;
1832 	for (i = 0; i < MAX_SUPPORT_RATES; i++) {
1833 		modval[i] = buf + sizeof (char *) * MAX_SUPPORT_RATES +
1834 		    i * DLADM_STRSIZE;
1835 	}
1836 
1837 	status = do_get_rate_mod(NULL, linkid, modval, &modval_cnt, media,
1838 	    0, &perm_flags);
1839 	if (status != DLADM_STATUS_OK)
1840 		goto done;
1841 
1842 	for (i = 0; i < modval_cnt; i++) {
1843 		if (strcasecmp(*prop_val, modval[i]) == 0) {
1844 			vdp->vd_val = (uintptr_t)(uint_t)
1845 			    (atof(*prop_val) * 2);
1846 			status = DLADM_STATUS_OK;
1847 			break;
1848 		}
1849 	}
1850 	if (i == modval_cnt)
1851 		status = DLADM_STATUS_BADVAL;
1852 done:
1853 	free(buf);
1854 	return (status);
1855 }
1856 
1857 static dladm_status_t
1858 do_get_phyconf(datalink_id_t linkid, void *buf, int buflen)
1859 {
1860 	return (i_dladm_wlan_param(linkid, buf, MAC_PROP_WL_PHY_CONFIG,
1861 	    buflen, B_FALSE));
1862 }
1863 
1864 /* ARGSUSED */
1865 static dladm_status_t
1866 do_get_channel_prop(prop_desc_t *pdp, datalink_id_t linkid,
1867     char **prop_val, uint_t *val_cnt, datalink_media_t media,
1868     uint_t flags, uint_t *perm_flags)
1869 {
1870 	uint32_t	channel;
1871 	char		buf[WLDP_BUFSIZE];
1872 	dladm_status_t	status = DLADM_STATUS_OK;
1873 	wl_phy_conf_t	wl_phy_conf;
1874 
1875 	if ((status = do_get_phyconf(linkid, buf, sizeof (buf)))
1876 	    != DLADM_STATUS_OK)
1877 		goto done;
1878 
1879 	(void) memcpy(&wl_phy_conf, buf, sizeof (wl_phy_conf));
1880 	if (!i_dladm_wlan_convert_chan(&wl_phy_conf, &channel)) {
1881 		status = DLADM_STATUS_NOTFOUND;
1882 		goto done;
1883 	}
1884 
1885 	(void) snprintf(*prop_val, DLADM_STRSIZE, "%u", channel);
1886 	*val_cnt = 1;
1887 	*perm_flags = MAC_PROP_PERM_READ;
1888 done:
1889 	return (status);
1890 }
1891 
1892 static dladm_status_t
1893 do_get_powermode(datalink_id_t linkid, void *buf, int buflen)
1894 {
1895 	return (i_dladm_wlan_param(linkid, buf, MAC_PROP_WL_POWER_MODE,
1896 	    buflen, B_FALSE));
1897 }
1898 
1899 /* ARGSUSED */
1900 static dladm_status_t
1901 do_get_powermode_prop(prop_desc_t *pdp, datalink_id_t linkid,
1902     char **prop_val, uint_t *val_cnt, datalink_media_t media,
1903     uint_t flags, uint_t *perm_flags)
1904 {
1905 	wl_ps_mode_t	mode;
1906 	const char	*s;
1907 	char		buf[WLDP_BUFSIZE];
1908 	dladm_status_t	status = DLADM_STATUS_OK;
1909 
1910 	if ((status = do_get_powermode(linkid, buf, sizeof (buf)))
1911 	    != DLADM_STATUS_OK)
1912 		goto done;
1913 
1914 	(void) memcpy(&mode, buf, sizeof (mode));
1915 	switch (mode.wl_ps_mode) {
1916 	case WL_PM_AM:
1917 		s = "off";
1918 		break;
1919 	case WL_PM_MPS:
1920 		s = "max";
1921 		break;
1922 	case WL_PM_FAST:
1923 		s = "fast";
1924 		break;
1925 	default:
1926 		status = DLADM_STATUS_NOTFOUND;
1927 		goto done;
1928 	}
1929 	(void) snprintf(*prop_val, DLADM_STRSIZE, "%s", s);
1930 	*val_cnt = 1;
1931 	*perm_flags = MAC_PROP_PERM_RW;
1932 done:
1933 	return (status);
1934 }
1935 
1936 static dladm_status_t
1937 do_set_powermode(datalink_id_t linkid, dladm_wlan_powermode_t *pm)
1938 {
1939 	wl_ps_mode_t    ps_mode;
1940 
1941 	(void) memset(&ps_mode, 0xff, sizeof (ps_mode));
1942 
1943 	switch (*pm) {
1944 	case DLADM_WLAN_PM_OFF:
1945 		ps_mode.wl_ps_mode = WL_PM_AM;
1946 		break;
1947 	case DLADM_WLAN_PM_MAX:
1948 		ps_mode.wl_ps_mode = WL_PM_MPS;
1949 		break;
1950 	case DLADM_WLAN_PM_FAST:
1951 		ps_mode.wl_ps_mode = WL_PM_FAST;
1952 		break;
1953 	default:
1954 		return (DLADM_STATUS_NOTSUP);
1955 	}
1956 	return (i_dladm_wlan_param(linkid, &ps_mode, MAC_PROP_WL_POWER_MODE,
1957 	    sizeof (ps_mode), B_TRUE));
1958 }
1959 
1960 /* ARGSUSED */
1961 static dladm_status_t
1962 do_set_powermode_prop(prop_desc_t *pdp, datalink_id_t linkid,
1963     val_desc_t *vdp, uint_t val_cnt, uint_t flags, datalink_media_t media)
1964 {
1965 	dladm_wlan_powermode_t powermode = (dladm_wlan_powermode_t)vdp->vd_val;
1966 	dladm_status_t status;
1967 
1968 	if (val_cnt != 1)
1969 		return (DLADM_STATUS_BADVALCNT);
1970 
1971 	status = do_set_powermode(linkid, &powermode);
1972 
1973 	return (status);
1974 }
1975 
1976 static dladm_status_t
1977 do_get_radio(datalink_id_t linkid, void *buf, int buflen)
1978 {
1979 	return (i_dladm_wlan_param(linkid, buf, MAC_PROP_WL_RADIO, buflen,
1980 	    B_FALSE));
1981 }
1982 
1983 /* ARGSUSED */
1984 static dladm_status_t
1985 do_get_radio_prop(prop_desc_t *pdp, datalink_id_t linkid,
1986     char **prop_val, uint_t *val_cnt, datalink_media_t media,
1987     uint_t flags, uint_t *perm_flags)
1988 {
1989 	wl_radio_t	radio;
1990 	const char	*s;
1991 	char		buf[WLDP_BUFSIZE];
1992 	dladm_status_t	status = DLADM_STATUS_OK;
1993 
1994 	if ((status = do_get_radio(linkid, buf, sizeof (buf)))
1995 	    != DLADM_STATUS_OK)
1996 		goto done;
1997 
1998 	(void) memcpy(&radio, buf, sizeof (radio));
1999 	switch (radio) {
2000 	case B_TRUE:
2001 		s = "on";
2002 		break;
2003 	case B_FALSE:
2004 		s = "off";
2005 		break;
2006 	default:
2007 		status = DLADM_STATUS_NOTFOUND;
2008 		goto done;
2009 	}
2010 	(void) snprintf(*prop_val, DLADM_STRSIZE, "%s", s);
2011 	*val_cnt = 1;
2012 	*perm_flags = MAC_PROP_PERM_RW;
2013 done:
2014 	return (status);
2015 }
2016 
2017 static dladm_status_t
2018 do_set_radio(datalink_id_t linkid, dladm_wlan_radio_t *radio)
2019 {
2020 	wl_radio_t r;
2021 
2022 	switch (*radio) {
2023 	case DLADM_WLAN_RADIO_ON:
2024 		r = B_TRUE;
2025 		break;
2026 	case DLADM_WLAN_RADIO_OFF:
2027 		r = B_FALSE;
2028 		break;
2029 	default:
2030 		return (DLADM_STATUS_NOTSUP);
2031 	}
2032 	return (i_dladm_wlan_param(linkid, &r, MAC_PROP_WL_RADIO,
2033 	    sizeof (r), B_TRUE));
2034 }
2035 
2036 /* ARGSUSED */
2037 static dladm_status_t
2038 do_set_radio_prop(prop_desc_t *pdp, datalink_id_t linkid,
2039     val_desc_t *vdp, uint_t val_cnt, uint_t fags, datalink_media_t media)
2040 {
2041 	dladm_wlan_radio_t radio = (dladm_wlan_radio_t)vdp->vd_val;
2042 	dladm_status_t status;
2043 
2044 	if (val_cnt != 1)
2045 		return (DLADM_STATUS_BADVALCNT);
2046 
2047 	status = do_set_radio(linkid, &radio);
2048 
2049 	return (status);
2050 }
2051 
2052 static dladm_status_t
2053 i_dladm_set_linkprop_db(datalink_id_t linkid, const char *prop_name,
2054     char **prop_val, uint_t val_cnt)
2055 {
2056 	char		buf[MAXLINELEN];
2057 	int		i;
2058 	dladm_conf_t	conf;
2059 	dladm_status_t	status;
2060 
2061 	status = dladm_read_conf(linkid, &conf);
2062 	if (status != DLADM_STATUS_OK)
2063 		return (status);
2064 
2065 	/*
2066 	 * reset case.
2067 	 */
2068 	if (val_cnt == 0) {
2069 		status = dladm_unset_conf_field(conf, prop_name);
2070 		if (status == DLADM_STATUS_OK)
2071 			status = dladm_write_conf(conf);
2072 		goto done;
2073 	}
2074 
2075 	buf[0] = '\0';
2076 	for (i = 0; i < val_cnt; i++) {
2077 		(void) strlcat(buf, prop_val[i], MAXLINELEN);
2078 		if (i != val_cnt - 1)
2079 			(void) strlcat(buf, ",", MAXLINELEN);
2080 	}
2081 
2082 	status = dladm_set_conf_field(conf, prop_name, DLADM_TYPE_STR, buf);
2083 	if (status == DLADM_STATUS_OK)
2084 		status = dladm_write_conf(conf);
2085 
2086 done:
2087 	dladm_destroy_conf(conf);
2088 	return (status);
2089 }
2090 
2091 static dladm_status_t
2092 i_dladm_get_linkprop_db(datalink_id_t linkid, const char *prop_name,
2093     char **prop_val, uint_t *val_cntp)
2094 {
2095 	char		buf[MAXLINELEN], *str;
2096 	uint_t		cnt = 0;
2097 	dladm_conf_t	conf;
2098 	dladm_status_t	status;
2099 
2100 	status = dladm_read_conf(linkid, &conf);
2101 	if (status != DLADM_STATUS_OK)
2102 		return (status);
2103 
2104 	status = dladm_get_conf_field(conf, prop_name, buf, MAXLINELEN);
2105 	if (status != DLADM_STATUS_OK)
2106 		goto done;
2107 
2108 	str = strtok(buf, ",");
2109 	while (str != NULL) {
2110 		if (cnt == *val_cntp) {
2111 			status = DLADM_STATUS_TOOSMALL;
2112 			goto done;
2113 		}
2114 		(void) strlcpy(prop_val[cnt++], str, DLADM_PROP_VAL_MAX);
2115 		str = strtok(NULL, ",");
2116 	}
2117 
2118 	*val_cntp = cnt;
2119 
2120 done:
2121 	dladm_destroy_conf(conf);
2122 	return (status);
2123 }
2124 
2125 static link_attr_t *
2126 dladm_name2prop(const char *prop_name)
2127 {
2128 	link_attr_t *p;
2129 
2130 	for (p = link_attr; p->pp_id != MAC_PROP_PRIVATE; p++) {
2131 		if (strcmp(p->pp_name, prop_name) == 0)
2132 			break;
2133 	}
2134 	return (p);
2135 }
2136 
2137 static link_attr_t *
2138 dladm_id2prop(mac_prop_id_t propid)
2139 {
2140 	link_attr_t *p;
2141 
2142 	for (p = link_attr; p->pp_id != MAC_PROP_PRIVATE; p++) {
2143 		if (p->pp_id == propid)
2144 			break;
2145 	}
2146 	return (p);
2147 }
2148 
2149 static dld_ioc_macprop_t *
2150 i_dladm_buf_alloc_impl(size_t valsize, datalink_id_t linkid,
2151     const char *prop_name, mac_prop_id_t propid, uint_t flags,
2152     dladm_status_t *status)
2153 {
2154 	int dsize;
2155 	dld_ioc_macprop_t *dip;
2156 
2157 	*status = DLADM_STATUS_OK;
2158 	dsize = MAC_PROP_BUFSIZE(valsize);
2159 	dip = malloc(dsize);
2160 	if (dip == NULL) {
2161 		*status = DLADM_STATUS_NOMEM;
2162 		return (NULL);
2163 	}
2164 	bzero(dip, dsize);
2165 	dip->pr_valsize = valsize;
2166 	(void) strlcpy(dip->pr_name, prop_name, sizeof (dip->pr_name));
2167 	dip->pr_version = MAC_PROP_VERSION;
2168 	dip->pr_linkid = linkid;
2169 	dip->pr_num = propid;
2170 	dip->pr_flags = flags;
2171 	return (dip);
2172 }
2173 
2174 static dld_ioc_macprop_t *
2175 i_dladm_buf_alloc_by_name(size_t valsize, datalink_id_t linkid,
2176     const char *prop_name, uint_t flags, dladm_status_t *status)
2177 {
2178 	link_attr_t *p;
2179 
2180 	p = dladm_name2prop(prop_name);
2181 	valsize = MAX(p->pp_valsize, valsize);
2182 	return (i_dladm_buf_alloc_impl(valsize, linkid, prop_name, p->pp_id,
2183 	    flags, status));
2184 }
2185 
2186 static dld_ioc_macprop_t *
2187 i_dladm_buf_alloc_by_id(size_t valsize, datalink_id_t linkid,
2188     mac_prop_id_t propid, uint_t flags, dladm_status_t *status)
2189 {
2190 	link_attr_t *p;
2191 
2192 	p = dladm_id2prop(propid);
2193 	valsize = MAX(p->pp_valsize, valsize);
2194 	return (i_dladm_buf_alloc_impl(valsize, linkid, p->pp_name, propid,
2195 	    flags, status));
2196 }
2197 
2198 /* ARGSUSED */
2199 static dladm_status_t
2200 i_dladm_set_public_prop(prop_desc_t *pdp, datalink_id_t linkid,
2201     val_desc_t *vdp, uint_t val_cnt, uint_t flags, datalink_media_t media)
2202 {
2203 	dld_ioc_macprop_t	*dip;
2204 	dladm_status_t	status = DLADM_STATUS_OK;
2205 	uint8_t		u8;
2206 	uint16_t	u16;
2207 	uint32_t	u32;
2208 	void		*val;
2209 
2210 	dip = i_dladm_buf_alloc_by_name(0, linkid, pdp->pd_name, 0, &status);
2211 	if (dip == NULL)
2212 		return (status);
2213 
2214 	if (pdp->pd_flags & PD_CHECK_ALLOC)
2215 		val = (void *)vdp->vd_val;
2216 	else {
2217 		/*
2218 		 * Currently all 1/2/4-byte size properties are byte/word/int.
2219 		 * No need (yet) to distinguish these from arrays of same size.
2220 		 */
2221 		switch (dip->pr_valsize) {
2222 		case 1:
2223 			u8 = vdp->vd_val;
2224 			val = &u8;
2225 			break;
2226 		case 2:
2227 			u16 = vdp->vd_val;
2228 			val = &u16;
2229 			break;
2230 		case 4:
2231 			u32 = vdp->vd_val;
2232 			val = &u32;
2233 			break;
2234 		default:
2235 			val = &vdp->vd_val;
2236 			break;
2237 		}
2238 	}
2239 
2240 	if (val != NULL)
2241 		(void) memcpy(dip->pr_val, val, dip->pr_valsize);
2242 	else
2243 		dip->pr_valsize = 0;
2244 
2245 	status = i_dladm_macprop(dip, B_TRUE);
2246 
2247 done:
2248 	free(dip);
2249 	return (status);
2250 }
2251 
2252 dladm_status_t
2253 i_dladm_macprop(void *dip, boolean_t set)
2254 {
2255 	int fd;
2256 	dladm_status_t status = DLADM_STATUS_OK;
2257 
2258 	if ((fd = open(DLD_CONTROL_DEV, O_RDWR)) < 0) {
2259 		status = dladm_errno2status(errno);
2260 		return (status);
2261 	}
2262 	if (ioctl(fd, (set ? DLDIOC_SETMACPROP : DLDIOC_GETMACPROP), dip))
2263 		status = dladm_errno2status(errno);
2264 
2265 	(void) close(fd);
2266 	return (status);
2267 }
2268 
2269 static dld_ioc_macprop_t *
2270 i_dladm_get_public_prop(datalink_id_t linkid, char *prop_name, uint_t flags,
2271     dladm_status_t *status, uint_t *perm_flags)
2272 {
2273 	dld_ioc_macprop_t *dip = NULL;
2274 
2275 	dip = i_dladm_buf_alloc_by_name(0, linkid, prop_name, flags, status);
2276 	if (dip == NULL)
2277 		return (NULL);
2278 
2279 	*status = i_dladm_macprop(dip, B_FALSE);
2280 	if (*status != DLADM_STATUS_OK) {
2281 		free(dip);
2282 		return (NULL);
2283 	}
2284 	if (perm_flags != NULL)
2285 		*perm_flags = dip->pr_perm_flags;
2286 
2287 	return (dip);
2288 }
2289 
2290 /* ARGSUSED */
2291 static dladm_status_t
2292 i_dladm_defmtu_check(prop_desc_t *pdp, datalink_id_t linkid,
2293     char **prop_val, uint_t val_cnt, val_desc_t *v, datalink_media_t media)
2294 {
2295 	if (val_cnt != 1)
2296 		return (DLADM_STATUS_BADVAL);
2297 	v->vd_val = atoi(prop_val[0]);
2298 	return (DLADM_STATUS_OK);
2299 }
2300 
2301 /* ARGSUSED */
2302 static dladm_status_t
2303 i_dladm_duplex_get(prop_desc_t *pdp, datalink_id_t linkid,
2304     char **prop_val, uint_t *val_cnt, datalink_media_t media,
2305     uint_t flags, uint_t *perm_flags)
2306 {
2307 	link_duplex_t   link_duplex;
2308 	dladm_status_t  status;
2309 
2310 	if ((status = dladm_get_single_mac_stat(linkid, "link_duplex",
2311 	    KSTAT_DATA_UINT32, &link_duplex)) != 0)
2312 		return (status);
2313 
2314 	switch (link_duplex) {
2315 	case LINK_DUPLEX_FULL:
2316 		(void) strcpy(*prop_val, "full");
2317 		break;
2318 	case LINK_DUPLEX_HALF:
2319 		(void) strcpy(*prop_val, "half");
2320 		break;
2321 	default:
2322 		(void) strcpy(*prop_val, "unknown");
2323 		break;
2324 	}
2325 	*val_cnt = 1;
2326 	return (DLADM_STATUS_OK);
2327 }
2328 
2329 /* ARGSUSED */
2330 static dladm_status_t
2331 i_dladm_speed_get(prop_desc_t *pdp, datalink_id_t linkid,
2332     char **prop_val, uint_t *val_cnt, uint_t flags, uint_t *perm_flags)
2333 {
2334 	uint64_t	ifspeed = 0;
2335 	dladm_status_t status;
2336 
2337 	if ((status = dladm_get_single_mac_stat(linkid, "ifspeed",
2338 	    KSTAT_DATA_UINT64, &ifspeed)) != 0)
2339 		return (status);
2340 
2341 	if ((ifspeed % 1000000) != 0) {
2342 		(void) snprintf(*prop_val, DLADM_PROP_VAL_MAX,
2343 		    "%llf", ifspeed / (float)1000000); /* Mbps */
2344 	} else {
2345 		(void) snprintf(*prop_val, DLADM_PROP_VAL_MAX,
2346 		    "%llu", ifspeed / 1000000); /* Mbps */
2347 	}
2348 	*val_cnt = 1;
2349 	*perm_flags = MAC_PROP_PERM_READ;
2350 	return (DLADM_STATUS_OK);
2351 }
2352 
2353 /* ARGSUSED */
2354 static dladm_status_t
2355 i_dladm_status_get(prop_desc_t *pdp, datalink_id_t linkid,
2356     char **prop_val, uint_t *val_cnt, datalink_media_t media,
2357     uint_t flags, uint_t *perm_flags)
2358 {
2359 	link_state_t		link_state;
2360 	dladm_status_t		status;
2361 	uchar_t 		*cp;
2362 	dld_ioc_macprop_t	*dip;
2363 
2364 	dip = i_dladm_get_public_prop(linkid, pdp->pd_name, flags,
2365 	    &status, perm_flags);
2366 	if (status != DLADM_STATUS_OK)
2367 		return (status);
2368 
2369 	cp = (uchar_t *)dip->pr_val;
2370 	(void) memcpy(&link_state, cp, sizeof (link_state));
2371 
2372 	switch (link_state) {
2373 	case LINK_STATE_UP:
2374 		(void) strcpy(*prop_val, "up");
2375 		break;
2376 	case LINK_STATE_DOWN:
2377 		(void) strcpy(*prop_val, "down");
2378 		break;
2379 	default:
2380 		(void) strcpy(*prop_val, "unknown");
2381 		break;
2382 	}
2383 	*val_cnt = 1;
2384 	free(dip);
2385 	return (DLADM_STATUS_OK);
2386 }
2387 
2388 /* ARGSUSED */
2389 static dladm_status_t
2390 i_dladm_binary_get(prop_desc_t *pdp, datalink_id_t linkid,
2391     char **prop_val, uint_t *val_cnt, datalink_media_t media,
2392     uint_t flags, uint_t *perm_flags)
2393 {
2394 	dld_ioc_macprop_t *dip;
2395 	dladm_status_t status;
2396 
2397 	dip = i_dladm_get_public_prop(linkid, pdp->pd_name, flags,
2398 	    &status, perm_flags);
2399 	if (dip == NULL)
2400 		return (status);
2401 
2402 	(void) snprintf(*prop_val, DLADM_PROP_VAL_MAX, "%x", dip->pr_val[0]);
2403 	free(dip);
2404 	*val_cnt = 1;
2405 	return (DLADM_STATUS_OK);
2406 }
2407 
2408 /* ARGSUSED */
2409 static dladm_status_t
2410 i_dladm_uint32_get(prop_desc_t *pdp, datalink_id_t linkid,
2411     char **prop_val, uint_t *val_cnt, datalink_media_t media,
2412     uint_t flags, uint_t *perm_flags)
2413 {
2414 	dld_ioc_macprop_t *dip;
2415 	uint32_t v = 0;
2416 	uchar_t *cp;
2417 	dladm_status_t status;
2418 
2419 	dip = i_dladm_get_public_prop(linkid, pdp->pd_name, flags,
2420 	    &status, perm_flags);
2421 	if (dip == NULL)
2422 		return (status);
2423 
2424 	cp = (uchar_t *)dip->pr_val;
2425 	(void) memcpy(&v, cp, sizeof (v));
2426 	(void) snprintf(*prop_val, DLADM_PROP_VAL_MAX, "%ld", v);
2427 	free(dip);
2428 	*val_cnt = 1;
2429 	return (DLADM_STATUS_OK);
2430 }
2431 
2432 /* ARGSUSED */
2433 static dladm_status_t
2434 i_dladm_flowctl_get(prop_desc_t *pdp, datalink_id_t linkid,
2435     char **prop_val, uint_t *val_cnt, datalink_media_t media,
2436     uint_t flags, uint_t *perm_flags)
2437 {
2438 	dld_ioc_macprop_t *dip;
2439 	link_flowctrl_t v;
2440 	dladm_status_t status;
2441 	uchar_t *cp;
2442 
2443 	dip = i_dladm_get_public_prop(linkid, pdp->pd_name, flags,
2444 	    &status, perm_flags);
2445 	if (dip == NULL)
2446 		return (status);
2447 
2448 	cp = (uchar_t *)dip->pr_val;
2449 	(void) memcpy(&v, cp, sizeof (v));
2450 	switch (v) {
2451 	case LINK_FLOWCTRL_NONE:
2452 		(void) sprintf(*prop_val, "no");
2453 		break;
2454 	case LINK_FLOWCTRL_RX:
2455 		(void) sprintf(*prop_val, "rx");
2456 		break;
2457 	case LINK_FLOWCTRL_TX:
2458 		(void) sprintf(*prop_val, "tx");
2459 		break;
2460 	case LINK_FLOWCTRL_BI:
2461 		(void) sprintf(*prop_val, "bi");
2462 		break;
2463 	}
2464 	free(dip);
2465 	*val_cnt = 1;
2466 	return (DLADM_STATUS_OK);
2467 }
2468 
2469 
2470 /* ARGSUSED */
2471 static dladm_status_t
2472 i_dladm_set_prop(datalink_id_t linkid, const char *prop_name,
2473     char **prop_val, uint_t val_cnt, uint_t flags)
2474 {
2475 	int		i, slen;
2476 	int 		bufsize = 0;
2477 	dld_ioc_macprop_t *dip = NULL;
2478 	uchar_t 	*dp;
2479 	link_attr_t *p;
2480 	dladm_status_t	status = DLADM_STATUS_OK;
2481 
2482 	if ((prop_name == NULL && prop_val != NULL) ||
2483 	    (prop_val != NULL && val_cnt == 0))
2484 		return (DLADM_STATUS_BADARG);
2485 	p = dladm_name2prop(prop_name);
2486 	if (p->pp_id != MAC_PROP_PRIVATE)
2487 		return (DLADM_STATUS_BADARG);
2488 
2489 	/*
2490 	 * private properties: all parsing is done in the kernel.
2491 	 * allocate a enough space for each property + its separator (',').
2492 	 */
2493 	for (i = 0; i < val_cnt; i++) {
2494 		bufsize += strlen(prop_val[i]) + 1;
2495 	}
2496 
2497 	if (prop_val == NULL) {
2498 		/*
2499 		 * getting default value. so use more buffer space.
2500 		 */
2501 		bufsize += DLADM_PROP_BUF_CHUNK;
2502 	}
2503 
2504 	dip = i_dladm_buf_alloc_by_name(bufsize + 1, linkid, prop_name,
2505 	    (prop_val != NULL ? 0 : MAC_PROP_DEFAULT), &status);
2506 	if (dip == NULL)
2507 		return (status);
2508 
2509 	dp = (uchar_t *)dip->pr_val;
2510 	slen = 0;
2511 
2512 	if (prop_val == NULL) {
2513 		status = i_dladm_macprop(dip, B_FALSE);
2514 	} else {
2515 		for (i = 0; i < val_cnt; i++) {
2516 			int plen = 0;
2517 
2518 			plen = strlen(prop_val[i]);
2519 			bcopy(prop_val[i], dp, plen);
2520 			slen += plen;
2521 			/*
2522 			 * add a "," separator and update dp.
2523 			 */
2524 			if (i != (val_cnt -1))
2525 				dp[slen++] = ',';
2526 			dp += (plen + 1);
2527 		}
2528 		status = i_dladm_macprop(dip, B_TRUE);
2529 	}
2530 
2531 	free(dip);
2532 	return (status);
2533 }
2534 
2535 static dladm_status_t
2536 i_dladm_get_prop(datalink_id_t linkid, const char *prop_name,
2537     char **prop_val, uint_t *val_cnt, dladm_prop_type_t type, uint_t dld_flags)
2538 {
2539 	dladm_status_t	status = DLADM_STATUS_OK;
2540 	dld_ioc_macprop_t *dip = NULL;
2541 	link_attr_t *p;
2542 	char tmp = '\0';
2543 
2544 	if ((prop_name == NULL && prop_val != NULL) ||
2545 	    (prop_val != NULL && val_cnt == 0))
2546 		return (DLADM_STATUS_BADARG);
2547 
2548 	p = dladm_name2prop(prop_name);
2549 	if (p->pp_id != MAC_PROP_PRIVATE)
2550 		return (DLADM_STATUS_BADARG);
2551 
2552 	if (type == DLADM_PROP_VAL_MODIFIABLE) {
2553 		*prop_val = &tmp;
2554 		*val_cnt = 1;
2555 		return (DLADM_STATUS_OK);
2556 	}
2557 
2558 	/*
2559 	 * private properties: all parsing is done in the kernel.
2560 	 */
2561 	dip = i_dladm_buf_alloc_by_name(DLADM_PROP_BUF_CHUNK, linkid, prop_name,
2562 	    dld_flags, &status);
2563 	if (dip == NULL)
2564 		return (status);
2565 
2566 	if ((status = i_dladm_macprop(dip, B_FALSE)) == DLADM_STATUS_OK) {
2567 		if (type == DLADM_PROP_VAL_PERM) {
2568 			(void) dladm_perm2str(dip->pr_perm_flags, *prop_val);
2569 		} else {
2570 			(void) strncpy(*prop_val, dip->pr_val,
2571 			    DLADM_PROP_VAL_MAX);
2572 		}
2573 		*val_cnt = 1;
2574 	}
2575 	free(dip);
2576 	return (status);
2577 }
2578 
2579 
2580 static dladm_status_t
2581 i_dladm_getset_defval(prop_desc_t *pdp, datalink_id_t linkid,
2582     datalink_media_t media, uint_t flags)
2583 {
2584 	dladm_status_t status;
2585 	char **prop_vals = NULL, *buf;
2586 	size_t bufsize;
2587 	uint_t cnt;
2588 	int i;
2589 	uint_t perm_flags;
2590 
2591 	/*
2592 	 * Allocate buffer needed for prop_vals array. We can have at most
2593 	 * DLADM_MAX_PROP_VALCNT char *prop_vals[] entries, where
2594 	 * each entry has max size DLADM_PROP_VAL_MAX
2595 	 */
2596 	bufsize =
2597 	    (sizeof (char *) + DLADM_PROP_VAL_MAX) * DLADM_MAX_PROP_VALCNT;
2598 	buf = malloc(bufsize);
2599 	prop_vals = (char **)(void *)buf;
2600 	for (i = 0; i < DLADM_MAX_PROP_VALCNT; i++) {
2601 		prop_vals[i] = buf +
2602 		    sizeof (char *) * DLADM_MAX_PROP_VALCNT +
2603 		    i * DLADM_PROP_VAL_MAX;
2604 	}
2605 
2606 	/*
2607 	 * For properties which have pdp->pd_defval.vd_name as a non-empty
2608 	 * string, the "" itself is used to reset the property (exceptions
2609 	 * are zone and autopush, which populate vdp->vd_val). So
2610 	 * libdladm can copy pdp->pd_defval over to the val_desc_t passed
2611 	 * down on the setprop using the global values in the table. For
2612 	 * other cases (vd_name is ""), doing reset-linkprop will cause
2613 	 * libdladm to do a getprop to find the default value and then do
2614 	 * a setprop to reset the value to default.
2615 	 */
2616 	status = pdp->pd_get(pdp, linkid, prop_vals, &cnt, media,
2617 	    MAC_PROP_DEFAULT, &perm_flags);
2618 	if (status == DLADM_STATUS_OK) {
2619 		if (perm_flags == MAC_PROP_PERM_RW) {
2620 			status = i_dladm_set_single_prop(linkid, pdp->pd_class,
2621 			    media, pdp, prop_vals, cnt, flags);
2622 		}
2623 		else
2624 			status = DLADM_STATUS_NOTSUP;
2625 	}
2626 	free(buf);
2627 	return (status);
2628 }
2629 
2630 int
2631 macprop_to_wifi(mac_prop_id_t wl_prop)
2632 {
2633 	switch (wl_prop) {
2634 	case MAC_PROP_WL_ESSID:
2635 		return (WL_ESSID);
2636 	case MAC_PROP_WL_BSSID:
2637 		return (WL_BSSID);
2638 	case MAC_PROP_WL_BSSTYPE:
2639 		return (WL_BSS_TYPE);
2640 	case MAC_PROP_WL_LINKSTATUS:
2641 		return (WL_LINKSTATUS);
2642 	case MAC_PROP_WL_DESIRED_RATES:
2643 		return (WL_DESIRED_RATES);
2644 	case MAC_PROP_WL_SUPPORTED_RATES:
2645 		return (WL_SUPPORTED_RATES);
2646 	case MAC_PROP_WL_AUTH_MODE:
2647 		return (WL_AUTH_MODE);
2648 	case MAC_PROP_WL_ENCRYPTION:
2649 		return (WL_ENCRYPTION);
2650 	case MAC_PROP_WL_RSSI:
2651 		return (WL_RSSI);
2652 	case MAC_PROP_WL_PHY_CONFIG:
2653 		return (WL_PHY_CONFIG);
2654 	case MAC_PROP_WL_CAPABILITY:
2655 		return (WL_CAPABILITY);
2656 	case MAC_PROP_WL_WPA:
2657 		return (WL_WPA);
2658 	case MAC_PROP_WL_SCANRESULTS:
2659 		return (WL_SCANRESULTS);
2660 	case MAC_PROP_WL_POWER_MODE:
2661 		return (WL_POWER_MODE);
2662 	case MAC_PROP_WL_RADIO:
2663 		return (WL_RADIO);
2664 	case MAC_PROP_WL_ESS_LIST:
2665 		return (WL_ESS_LIST);
2666 	case MAC_PROP_WL_KEY_TAB:
2667 		return (WL_WEP_KEY_TAB);
2668 	case MAC_PROP_WL_CREATE_IBSS:
2669 		return (WL_CREATE_IBSS);
2670 	case MAC_PROP_WL_SETOPTIE:
2671 		return (WL_SETOPTIE);
2672 	case MAC_PROP_WL_DELKEY:
2673 		return (WL_DELKEY);
2674 	case MAC_PROP_WL_KEY:
2675 		return (WL_KEY);
2676 	case MAC_PROP_WL_MLME:
2677 		return (WL_MLME);
2678 	default:
2679 		return (-1);
2680 	}
2681 }
2682 
2683 dladm_status_t
2684 i_dladm_wlan_param(datalink_id_t linkid, void *buf, mac_prop_id_t cmd,
2685     size_t len, boolean_t set)
2686 {
2687 	uint32_t		flags;
2688 	dladm_status_t		status;
2689 	uint32_t		media;
2690 	dld_ioc_macprop_t	*dip;
2691 	void			*dp;
2692 
2693 	if ((status = dladm_datalink_id2info(linkid, &flags, NULL, &media,
2694 	    NULL, 0)) != DLADM_STATUS_OK) {
2695 		return (status);
2696 	}
2697 
2698 	if (media != DL_WIFI)
2699 		return (DLADM_STATUS_BADARG);
2700 
2701 	if (!(flags & DLADM_OPT_ACTIVE))
2702 		return (DLADM_STATUS_TEMPONLY);
2703 
2704 	if (len == (MAX_BUF_LEN - WIFI_BUF_OFFSET))
2705 		len = MAX_BUF_LEN - sizeof (dld_ioc_macprop_t) - 1;
2706 
2707 	dip = i_dladm_buf_alloc_by_id(len, linkid, cmd, 0, &status);
2708 	if (dip == NULL)
2709 		return (DLADM_STATUS_NOMEM);
2710 
2711 	dp = (uchar_t *)dip->pr_val;
2712 	if (set)
2713 		(void) memcpy(dp, buf, len);
2714 
2715 	status = i_dladm_macprop(dip, set);
2716 	if (status == DLADM_STATUS_NOTSUP) {
2717 		if (set) {
2718 			status = i_dladm_wlan_set_legacy_ioctl(linkid,
2719 			    buf, len, macprop_to_wifi(cmd));
2720 		} else {
2721 			status = i_dladm_wlan_get_legacy_ioctl(linkid,
2722 			    buf, len, macprop_to_wifi(cmd));
2723 		}
2724 	} else if (status == DLADM_STATUS_OK) {
2725 		if (!set)
2726 			(void) memcpy(buf, dp, len);
2727 	}
2728 
2729 	free(dip);
2730 	return (status);
2731 }
2732 
2733 static dladm_status_t
2734 i_dladm_wlan_get_legacy_ioctl(datalink_id_t linkid, void *buf, uint_t buflen,
2735     uint_t id)
2736 {
2737 	wldp_t *gbuf;
2738 	dladm_status_t status;
2739 
2740 	if ((gbuf = malloc(MAX_BUF_LEN)) == NULL)
2741 		return (DLADM_STATUS_NOMEM);
2742 
2743 	(void) memset(gbuf, 0, MAX_BUF_LEN);
2744 	status = i_dladm_wlan_legacy_ioctl(linkid, gbuf, id, MAX_BUF_LEN,
2745 	    WLAN_GET_PARAM, sizeof (wldp_t));
2746 	if (status == DLADM_STATUS_OK)
2747 		(void) memcpy(buf, gbuf->wldp_buf, buflen);
2748 
2749 	free(gbuf);
2750 	return (status);
2751 }
2752 
2753 static dladm_status_t
2754 i_dladm_wlan_set_legacy_ioctl(datalink_id_t linkid,  void *buf, uint_t buflen,
2755     uint_t id)
2756 {
2757 	wldp_t *gbuf;
2758 	dladm_status_t status = DLADM_STATUS_OK;
2759 
2760 	if ((gbuf = malloc(MAX_BUF_LEN)) == NULL)
2761 		return (DLADM_STATUS_NOMEM);
2762 
2763 	(void) memset(gbuf, 0, MAX_BUF_LEN);
2764 	(void) memcpy(gbuf->wldp_buf, buf, buflen);
2765 	buflen += WIFI_BUF_OFFSET;
2766 	status = i_dladm_wlan_legacy_ioctl(linkid, gbuf, id, buflen,
2767 	    WLAN_SET_PARAM, buflen);
2768 
2769 	free(gbuf);
2770 	return (status);
2771 }
2772 
2773 static dladm_status_t
2774 link_proplist_check(dladm_arg_list_t *proplist)
2775 {
2776 	int		i, j;
2777 	boolean_t	matched;
2778 
2779 	for (i = 0; i < proplist->al_count; i++) {
2780 		matched = B_FALSE;
2781 		for (j = 0; j < DLADM_MAX_PROPS; j++) {
2782 			if (strcmp(proplist->al_info[i].ai_name,
2783 			    prop_table[j].pd_name) == 0)
2784 				matched = B_TRUE;
2785 		}
2786 		if (!matched)
2787 			return (DLADM_STATUS_BADPROP);
2788 	}
2789 	return (DLADM_STATUS_OK);
2790 }
2791 
2792 dladm_status_t
2793 dladm_parse_link_props(char *str, dladm_arg_list_t **listp, boolean_t novalues)
2794 {
2795 	dladm_status_t	status;
2796 
2797 	status = dladm_parse_args(str, listp, novalues);
2798 	if (status != DLADM_STATUS_OK)
2799 		return (status);
2800 
2801 	status = link_proplist_check(*listp);
2802 	if (status != DLADM_STATUS_OK) {
2803 		dladm_free_props(*listp);
2804 		return (status);
2805 	}
2806 
2807 	return (DLADM_STATUS_OK);
2808 }
2809 
2810 /*
2811  * Retrieve the one link property from the database
2812  */
2813 /*ARGSUSED*/
2814 static int
2815 i_dladm_get_one_prop(datalink_id_t linkid, const char *prop_name, void *arg)
2816 {
2817 	dladm_arg_list_t	*proplist = arg;
2818 	dladm_arg_info_t	*aip = NULL;
2819 
2820 	aip = &proplist->al_info[proplist->al_count];
2821 	/*
2822 	 * it is fine to point to prop_name since prop_name points to the
2823 	 * prop_table[n].pd_name.
2824 	 */
2825 	aip->ai_name = prop_name;
2826 
2827 	(void) dladm_get_linkprop(linkid, DLADM_PROP_VAL_PERSISTENT, prop_name,
2828 	    aip->ai_val, &aip->ai_count);
2829 
2830 	if (aip->ai_count != 0)
2831 		proplist->al_count++;
2832 
2833 	return (DLADM_WALK_CONTINUE);
2834 }
2835 
2836 
2837 /*
2838  * Retrieve all link properties for a link from the database and
2839  * return a property list.
2840  */
2841 dladm_status_t
2842 dladm_link_get_proplist(datalink_id_t linkid, dladm_arg_list_t **listp)
2843 {
2844 	dladm_arg_list_t	*list;
2845 	dladm_status_t		status = DLADM_STATUS_OK;
2846 
2847 	list = calloc(1, sizeof (dladm_arg_list_t));
2848 	if (list == NULL)
2849 		return (dladm_errno2status(errno));
2850 
2851 	status = dladm_walk_linkprop(linkid, list, i_dladm_get_one_prop);
2852 
2853 	*listp = list;
2854 	return (status);
2855 }
2856 
2857 /*
2858  * Retrieve the named property from a proplist, check the value and
2859  * convert to a kernel structure.
2860  */
2861 static dladm_status_t
2862 i_dladm_link_proplist_extract_one(dladm_arg_list_t *proplist,
2863     const char *name, void *val)
2864 {
2865 	dladm_status_t		status;
2866 	dladm_arg_info_t	*aip = NULL;
2867 	int			i, j;
2868 
2869 	/* Find named property in proplist */
2870 	for (i = 0; i < proplist->al_count; i++) {
2871 		aip = &proplist->al_info[i];
2872 		if (strcasecmp(aip->ai_name, name) == 0)
2873 			break;
2874 	}
2875 
2876 	/* Property not in list */
2877 	if (i == proplist->al_count)
2878 		return (DLADM_STATUS_OK);
2879 
2880 	for (i = 0; i < DLADM_MAX_PROPS; i++) {
2881 		prop_desc_t	*pdp = &prop_table[i];
2882 		val_desc_t	*vdp;
2883 
2884 		vdp = malloc(sizeof (val_desc_t) * aip->ai_count);
2885 		if (vdp == NULL)
2886 			return (DLADM_STATUS_NOMEM);
2887 
2888 		if (strcasecmp(aip->ai_name, pdp->pd_name) != 0)
2889 			continue;
2890 
2891 		if (aip->ai_val == NULL)
2892 			return (DLADM_STATUS_BADARG);
2893 
2894 		/* Check property value */
2895 		if (pdp->pd_check != NULL) {
2896 			status = pdp->pd_check(pdp, 0, aip->ai_val,
2897 			    aip->ai_count, vdp, 0);
2898 		} else {
2899 			status = DLADM_STATUS_BADARG;
2900 		}
2901 
2902 		if (status != DLADM_STATUS_OK)
2903 			return (status);
2904 
2905 		for (j = 0; j < DLADM_MAX_RSRC_PROP; j++) {
2906 			resource_prop_t	*rpp = &rsrc_prop_table[j];
2907 
2908 			if (strcasecmp(aip->ai_name, rpp->rp_name) != 0)
2909 				continue;
2910 
2911 			/* Extract kernel structure */
2912 			if (rpp->rp_extract != NULL) {
2913 				status = rpp->rp_extract(vdp, val,
2914 				    aip->ai_count);
2915 			} else {
2916 				status = DLADM_STATUS_BADARG;
2917 			}
2918 			break;
2919 		}
2920 
2921 		if (status != DLADM_STATUS_OK)
2922 			return (status);
2923 
2924 		break;
2925 	}
2926 	return (status);
2927 }
2928 
2929 /*
2930  * Extract properties from a proplist and convert to mac_resource_props_t.
2931  */
2932 dladm_status_t
2933 dladm_link_proplist_extract(dladm_arg_list_t *proplist,
2934     mac_resource_props_t *mrp)
2935 {
2936 	dladm_status_t	status = DLADM_STATUS_OK;
2937 
2938 	status = i_dladm_link_proplist_extract_one(proplist, "maxbw", mrp);
2939 	if (status != DLADM_STATUS_OK)
2940 		return (status);
2941 	status = i_dladm_link_proplist_extract_one(proplist, "priority", mrp);
2942 	if (status != DLADM_STATUS_OK)
2943 		return (status);
2944 	status = i_dladm_link_proplist_extract_one(proplist, "cpus", mrp);
2945 	if (status != DLADM_STATUS_OK)
2946 		return (status);
2947 	return (status);
2948 }
2949 
2950 static const char *
2951 dladm_perm2str(uint_t perm, char *buf)
2952 {
2953 	(void) snprintf(buf, DLADM_STRSIZE, "%c%c",
2954 	    ((perm & MAC_PROP_PERM_READ) != 0) ? 'r' : '-',
2955 	    ((perm & MAC_PROP_PERM_WRITE) != 0) ? 'w' : '-');
2956 	return (buf);
2957 }
2958