xref: /titanic_44/usr/src/uts/common/io/usb/usba/usba.c (revision 1db2880b3a411e3c56e50c7dc42d3b137fcc4e48)
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 2009 Sun Microsystems, Inc.  All rights reserved.
23  * Use is subject to license terms.
24  *
25  * Copyright 2014 Garrett D'Amore <garrett@damore.org>
26  */
27 
28 
29 /*
30  * USBA: Solaris USB Architecture support
31  */
32 #define	USBA_FRAMEWORK
33 #include <sys/usb/usba/usba_impl.h>
34 #include <sys/usb/usba/hcdi_impl.h>
35 #include <sys/usb/hubd/hub.h>
36 #include <sys/fs/dv_node.h>
37 
38 static int usba_str_startcmp(char *, char *);
39 
40 /*
41  * USBA private variables and tunables
42  */
43 static kmutex_t	usba_mutex;
44 
45 /* mutex to protect usba_root_hubs */
46 static kmutex_t usba_hub_mutex;
47 
48 typedef struct usba_root_hub_ent {
49 	dev_info_t *dip;
50 	struct usba_root_hub_ent *next;
51 }usba_root_hub_ent_t;
52 
53 static usba_root_hub_ent_t *usba_root_hubs = NULL;
54 
55 /*
56  * ddivs forced binding:
57  *
58  *    usbc usbc_xhubs usbc_xaddress  node name
59  *
60  *	0	x	x	class name or "device"
61  *
62  *	1	0	0	ddivs_usbc
63  *	1	0	>1	ddivs_usbc except device
64  *				at usbc_xaddress
65  *	1	1	0	ddivs_usbc except hubs
66  *	1	1	>1	ddivs_usbc except hubs and
67  *				device at usbc_xaddress
68  */
69 uint_t usba_ddivs_usbc;
70 uint_t usba_ddivs_usbc_xhubs;
71 uint_t usba_ddivs_usbc_xaddress;
72 
73 uint_t usba_ugen_force_binding;
74 
75 /*
76  * compatible name handling
77  */
78 /*
79  * allowing for 15 compat names, plus one force bind name and
80  * one possible specified client driver name
81  */
82 #define	USBA_MAX_COMPAT_NAMES		17
83 #define	USBA_MAX_COMPAT_NAME_LEN	64
84 
85 /* double linked list for usba_devices */
86 usba_list_entry_t	usba_device_list;
87 
88 _NOTE(MUTEX_PROTECTS_DATA(usba_mutex, usba_device_list))
89 
90 /*
91  * modload support
92  */
93 
94 static struct modlmisc modlmisc	= {
95 	&mod_miscops,	/* Type	of module */
96 	"USBA: USB Architecture 2.0 1.66"
97 };
98 
99 static struct modlinkage modlinkage = {
100 	MODREV_1, (void	*)&modlmisc, NULL
101 };
102 
103 
104 static usb_log_handle_t	usba_log_handle;
105 uint_t		usba_errlevel = USB_LOG_L4;
106 uint_t		usba_errmask = (uint_t)-1;
107 
108 extern usb_log_handle_t	hubdi_log_handle;
109 
110 int
111 _init(void)
112 {
113 	int rval;
114 
115 	/*
116 	 * usbai providing log support needs to be init'ed first
117 	 * and destroyed last
118 	 */
119 	usba_usbai_initialization();
120 	usba_usba_initialization();
121 	usba_usbai_register_initialization();
122 	usba_hcdi_initialization();
123 	usba_hubdi_initialization();
124 	usba_devdb_initialization();
125 
126 	if ((rval = mod_install(&modlinkage)) != 0) {
127 		usba_devdb_destroy();
128 		usba_hubdi_destroy();
129 		usba_hcdi_destroy();
130 		usba_usbai_register_destroy();
131 		usba_usba_destroy();
132 		usba_usbai_destroy();
133 	}
134 
135 	return (rval);
136 }
137 
138 int
139 _fini()
140 {
141 	int rval;
142 
143 	if ((rval = mod_remove(&modlinkage)) == 0) {
144 		usba_devdb_destroy();
145 		usba_hubdi_destroy();
146 		usba_hcdi_destroy();
147 		usba_usbai_register_destroy();
148 		usba_usba_destroy();
149 		usba_usbai_destroy();
150 	}
151 
152 	return (rval);
153 }
154 
155 int
156 _info(struct modinfo *modinfop)
157 {
158 	return (mod_info(&modlinkage, modinfop));
159 }
160 
161 boolean_t
162 usba_owns_ia(dev_info_t *dip)
163 {
164 	int if_count = ddi_prop_get_int(DDI_DEV_T_ANY, dip, DDI_PROP_DONTPASS,
165 	    "interface-count", 0);
166 
167 	return ((if_count) ? B_TRUE : B_FALSE);
168 }
169 
170 /*
171  * common bus ctl for hcd, usb_mid, and hubd
172  */
173 int
174 usba_bus_ctl(dev_info_t	*dip,
175 	dev_info_t		*rdip,
176 	ddi_ctl_enum_t		op,
177 	void			*arg,
178 	void			*result)
179 {
180 	dev_info_t		*child_dip = (dev_info_t *)arg;
181 	usba_device_t		*usba_device;
182 	usba_hcdi_t		*usba_hcdi;
183 	usba_hcdi_ops_t		*usba_hcdi_ops;
184 
185 	USB_DPRINTF_L4(DPRINT_MASK_USBA, hubdi_log_handle,
186 	    "usba_bus_ctl: %s%d %s%d op=%d", ddi_node_name(rdip),
187 	    ddi_get_instance(rdip), ddi_node_name(dip),
188 	    ddi_get_instance(dip), op);
189 
190 	switch (op) {
191 
192 	case DDI_CTLOPS_REPORTDEV:
193 	{
194 		char *name, compat_name[64], *speed;
195 		usba_device_t	*hub_usba_device;
196 		dev_info_t	*hubdip;
197 
198 		usba_device = usba_get_usba_device(rdip);
199 
200 		/* find the parent hub */
201 		hubdip = ddi_get_parent(rdip);
202 		while ((strcmp(ddi_driver_name(hubdip), "hubd") != 0) &&
203 		    !(usba_is_root_hub(hubdip))) {
204 			hubdip = ddi_get_parent(hubdip);
205 		}
206 
207 		hub_usba_device = usba_get_usba_device(hubdip);
208 
209 		if (usba_device) {
210 			if (usb_owns_device(rdip)) {
211 				(void) snprintf(compat_name,
212 				    sizeof (compat_name),
213 				    "usb%x,%x",
214 				    usba_device->usb_dev_descr->idVendor,
215 				    usba_device->usb_dev_descr->idProduct);
216 			} else if (usba_owns_ia(rdip)) {
217 				(void) snprintf(compat_name,
218 				    sizeof (compat_name),
219 				    "usbia%x,%x.config%x.%x",
220 				    usba_device->usb_dev_descr->idVendor,
221 				    usba_device->usb_dev_descr->idProduct,
222 				    usba_device->usb_cfg_value,
223 				    usb_get_if_number(rdip));
224 			} else {
225 				(void) snprintf(compat_name,
226 				    sizeof (compat_name),
227 				    "usbif%x,%x.config%x.%x",
228 				    usba_device->usb_dev_descr->idVendor,
229 				    usba_device->usb_dev_descr->idProduct,
230 				    usba_device->usb_cfg_value,
231 				    usb_get_if_number(rdip));
232 			}
233 			switch (usba_device->usb_port_status) {
234 			case USBA_HIGH_SPEED_DEV:
235 				speed = "hi speed (USB 2.x)";
236 
237 				break;
238 			case USBA_LOW_SPEED_DEV:
239 				speed = "low speed (USB 1.x)";
240 
241 				break;
242 			case USBA_FULL_SPEED_DEV:
243 			default:
244 				speed = "full speed (USB 1.x)";
245 
246 				break;
247 			}
248 
249 			cmn_err(CE_CONT,
250 			    "?USB %x.%x %s (%s) operating at %s on "
251 			    "USB %x.%x %s hub: "
252 			    "%s@%s, %s%d at bus address %d\n",
253 			    (usba_device->usb_dev_descr->bcdUSB & 0xff00) >> 8,
254 			    usba_device->usb_dev_descr->bcdUSB & 0xff,
255 			    (usb_owns_device(rdip) ? "device" :
256 			    ((usba_owns_ia(rdip) ? "interface-association" :
257 			    "interface"))),
258 			    compat_name, speed,
259 			    (hub_usba_device->usb_dev_descr->bcdUSB &
260 			    0xff00) >> 8,
261 			    hub_usba_device->usb_dev_descr->bcdUSB & 0xff,
262 			    usba_is_root_hub(hubdip) ? "root" : "external",
263 			    ddi_node_name(rdip), ddi_get_name_addr(rdip),
264 			    ddi_driver_name(rdip),
265 			    ddi_get_instance(rdip), usba_device->usb_addr);
266 
267 			name = kmem_alloc(MAXNAMELEN, KM_SLEEP);
268 			(void) usba_get_mfg_prod_sn_str(rdip, name, MAXNAMELEN);
269 			if (name[0] != '\0') {
270 				cmn_err(CE_CONT, "?\t%s\n", name);
271 			}
272 			kmem_free(name, MAXNAMELEN);
273 
274 		} else { /* harden USBA against this case; if it happens */
275 
276 			cmn_err(CE_CONT,
277 			    "?USB-device: %s@%s, %s%d\n",
278 			    ddi_node_name(rdip), ddi_get_name_addr(rdip),
279 			    ddi_driver_name(rdip), ddi_get_instance(rdip));
280 		}
281 
282 		return (DDI_SUCCESS);
283 	}
284 
285 	case DDI_CTLOPS_INITCHILD:
286 	{
287 		int			usb_addr;
288 		uint_t			n;
289 		char			name[32];
290 		int			*data;
291 		int			rval;
292 		int			len = sizeof (usb_addr);
293 
294 		usba_hcdi	= usba_hcdi_get_hcdi(dip);
295 		usba_hcdi_ops	= usba_hcdi->hcdi_ops;
296 		ASSERT(usba_hcdi_ops != NULL);
297 
298 		/*
299 		 * as long as the dip exists, it should have
300 		 * usba_device structure associated with it
301 		 */
302 		usba_device = usba_get_usba_device(child_dip);
303 		if (usba_device == NULL) {
304 
305 			USB_DPRINTF_L2(DPRINT_MASK_USBA, hubdi_log_handle,
306 			    "usba_bus_ctl: DDI_NOT_WELL_FORMED (%s (0x%p))",
307 			    ddi_node_name(child_dip), (void *)child_dip);
308 
309 			return (DDI_NOT_WELL_FORMED);
310 		}
311 
312 		/* the dip should have an address and reg property */
313 		if (ddi_prop_op(DDI_DEV_T_NONE, child_dip, PROP_LEN_AND_VAL_BUF,
314 		    DDI_PROP_DONTPASS |	DDI_PROP_CANSLEEP, "assigned-address",
315 		    (caddr_t)&usb_addr,	&len) != DDI_SUCCESS) {
316 
317 			USB_DPRINTF_L2(DPRINT_MASK_USBA, hubdi_log_handle,
318 			    "usba_bus_ctl:\n\t"
319 			    "%s%d %s%d op=%d rdip = 0x%p dip = 0x%p",
320 			    ddi_node_name(rdip), ddi_get_instance(rdip),
321 			    ddi_node_name(dip), ddi_get_instance(dip), op,
322 			    (void *)rdip, (void *)dip);
323 
324 			USB_DPRINTF_L2(DPRINT_MASK_USBA, hubdi_log_handle,
325 			    "usba_bus_ctl: DDI_NOT_WELL_FORMED (%s (0x%p))",
326 			    ddi_node_name(child_dip), (void *)child_dip);
327 
328 			return (DDI_NOT_WELL_FORMED);
329 		}
330 
331 		if ((rval = ddi_prop_lookup_int_array(DDI_DEV_T_ANY, child_dip,
332 		    DDI_PROP_DONTPASS, "reg",
333 		    &data, &n)) != DDI_SUCCESS) {
334 
335 			USB_DPRINTF_L2(DPRINT_MASK_USBA, hubdi_log_handle,
336 			    "usba_bus_ctl: %d, DDI_NOT_WELL_FORMED", rval);
337 
338 			return (DDI_NOT_WELL_FORMED);
339 		}
340 
341 
342 		/*
343 		 * if the configuration is 1, the unit address is
344 		 * just the interface number
345 		 */
346 		if ((n == 1) || ((n > 1) && (data[1] == 1))) {
347 			(void) sprintf(name, "%x", data[0]);
348 		} else {
349 			(void) sprintf(name, "%x,%x", data[0], data[1]);
350 		}
351 
352 		USB_DPRINTF_L3(DPRINT_MASK_USBA,
353 		    hubdi_log_handle, "usba_bus_ctl: name = %s", name);
354 
355 		ddi_prop_free(data);
356 		ddi_set_name_addr(child_dip, name);
357 
358 		/*
359 		 * increment the reference count for each child using this
360 		 * usba_device structure
361 		 */
362 		mutex_enter(&usba_device->usb_mutex);
363 		usba_device->usb_ref_count++;
364 
365 		USB_DPRINTF_L3(DPRINT_MASK_USBA, hubdi_log_handle,
366 		    "usba_bus_ctl: init usba_device = 0x%p ref_count = %d",
367 		    (void *)usba_device, usba_device->usb_ref_count);
368 
369 		mutex_exit(&usba_device->usb_mutex);
370 
371 		return (DDI_SUCCESS);
372 	}
373 
374 	case DDI_CTLOPS_UNINITCHILD:
375 	{
376 		usba_device = usba_get_usba_device(child_dip);
377 
378 		if (usba_device != NULL) {
379 			/*
380 			 * decrement the reference count for each child
381 			 * using this  usba_device structure
382 			 */
383 			mutex_enter(&usba_device->usb_mutex);
384 			usba_device->usb_ref_count--;
385 
386 			USB_DPRINTF_L3(DPRINT_MASK_USBA, hubdi_log_handle,
387 			    "usba_hcdi_bus_ctl: uninit usba_device=0x%p "
388 			    "ref_count=%d",
389 			    (void *)usba_device, usba_device->usb_ref_count);
390 
391 			mutex_exit(&usba_device->usb_mutex);
392 		}
393 		ddi_set_name_addr(child_dip, NULL);
394 
395 		return (DDI_SUCCESS);
396 	}
397 
398 	case DDI_CTLOPS_IOMIN:
399 		/* Do nothing */
400 		return (DDI_SUCCESS);
401 
402 	/*
403 	 * These ops correspond	to functions that "shouldn't" be called
404 	 * by a	USB client driver.  So	we whine when we're called.
405 	 */
406 	case DDI_CTLOPS_DMAPMAPC:
407 	case DDI_CTLOPS_REPORTINT:
408 	case DDI_CTLOPS_REGSIZE:
409 	case DDI_CTLOPS_NREGS:
410 	case DDI_CTLOPS_SIDDEV:
411 	case DDI_CTLOPS_SLAVEONLY:
412 	case DDI_CTLOPS_AFFINITY:
413 	case DDI_CTLOPS_POKE:
414 	case DDI_CTLOPS_PEEK:
415 		cmn_err(CE_CONT, "%s%d:	invalid	op (%d)	from %s%d",
416 		    ddi_node_name(dip), ddi_get_instance(dip),
417 		    op, ddi_node_name(rdip), ddi_get_instance(rdip));
418 		return (DDI_FAILURE);
419 
420 	/*
421 	 * Everything else (e.g. PTOB/BTOP/BTOPR requests) we pass up
422 	 */
423 	default:
424 		return (ddi_ctlops(dip,	rdip, op, arg, result));
425 	}
426 }
427 
428 
429 /*
430  * initialize and destroy USBA module
431  */
432 void
433 usba_usba_initialization()
434 {
435 	usba_log_handle = usb_alloc_log_hdl(NULL, "usba", &usba_errlevel,
436 	    &usba_errmask, NULL, 0);
437 
438 	USB_DPRINTF_L4(DPRINT_MASK_USBA,
439 	    usba_log_handle, "usba_usba_initialization");
440 
441 	mutex_init(&usba_mutex, NULL, MUTEX_DRIVER, NULL);
442 	mutex_init(&usba_hub_mutex, NULL, MUTEX_DRIVER, NULL);
443 	usba_init_list(&usba_device_list, NULL, NULL);
444 }
445 
446 
447 void
448 usba_usba_destroy()
449 {
450 	USB_DPRINTF_L4(DPRINT_MASK_USBA, usba_log_handle, "usba_usba_destroy");
451 
452 	mutex_destroy(&usba_hub_mutex);
453 	mutex_destroy(&usba_mutex);
454 	usba_destroy_list(&usba_device_list);
455 
456 	usb_free_log_hdl(usba_log_handle);
457 }
458 
459 
460 /*
461  * usba_set_usb_address:
462  *	set usb address in usba_device structure
463  */
464 int
465 usba_set_usb_address(usba_device_t *usba_device)
466 {
467 	usb_addr_t address;
468 	uchar_t s = 8;
469 	usba_hcdi_t *hcdi;
470 	char *usb_address_in_use;
471 
472 	mutex_enter(&usba_device->usb_mutex);
473 
474 	hcdi = usba_hcdi_get_hcdi(usba_device->usb_root_hub_dip);
475 
476 	mutex_enter(&hcdi->hcdi_mutex);
477 	usb_address_in_use = hcdi->hcdi_usb_address_in_use;
478 
479 	for (address = ROOT_HUB_ADDR + 1;
480 	    address <= USBA_MAX_ADDRESS; address++) {
481 		if (usb_address_in_use[address/s] & (1 << (address % s))) {
482 			continue;
483 		}
484 		usb_address_in_use[address/s] |= (1 << (address % s));
485 		hcdi->hcdi_device_count++;
486 		HCDI_HOTPLUG_STATS_DATA(hcdi)->hcdi_device_count.value.ui64++;
487 		mutex_exit(&hcdi->hcdi_mutex);
488 
489 		USB_DPRINTF_L3(DPRINT_MASK_USBA, usba_log_handle,
490 		    "usba_set_usb_address: %d", address);
491 
492 		usba_device->usb_addr = address;
493 
494 		mutex_exit(&usba_device->usb_mutex);
495 
496 		return (USB_SUCCESS);
497 	}
498 
499 	usba_device->usb_addr = 0;
500 
501 	USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
502 	    "no usb address available");
503 
504 	mutex_exit(&hcdi->hcdi_mutex);
505 	mutex_exit(&usba_device->usb_mutex);
506 
507 	return (USB_FAILURE);
508 }
509 
510 
511 /*
512  * usba_unset_usb_address:
513  *	unset usb_address in usba_device structure
514  */
515 void
516 usba_unset_usb_address(usba_device_t *usba_device)
517 {
518 	usb_addr_t address;
519 	usba_hcdi_t *hcdi;
520 	uchar_t s = 8;
521 	char *usb_address_in_use;
522 
523 	mutex_enter(&usba_device->usb_mutex);
524 	address = usba_device->usb_addr;
525 	hcdi = usba_hcdi_get_hcdi(usba_device->usb_root_hub_dip);
526 
527 	if (address > ROOT_HUB_ADDR) {
528 		USB_DPRINTF_L3(DPRINT_MASK_USBA, usba_log_handle,
529 		    "usba_unset_usb_address: address=%d", address);
530 
531 		mutex_enter(&hcdi->hcdi_mutex);
532 		usb_address_in_use = hcdi->hcdi_usb_address_in_use;
533 
534 		ASSERT(usb_address_in_use[address/s] & (1 << (address % s)));
535 
536 		usb_address_in_use[address/s] &= ~(1 << (address % s));
537 
538 		hcdi->hcdi_device_count--;
539 		HCDI_HOTPLUG_STATS_DATA(hcdi)->hcdi_device_count.value.ui64--;
540 
541 		mutex_exit(&hcdi->hcdi_mutex);
542 
543 		usba_device->usb_addr = 0;
544 	}
545 	mutex_exit(&usba_device->usb_mutex);
546 }
547 
548 
549 struct usba_evdata *
550 usba_get_evdata(dev_info_t *dip)
551 {
552 	usba_evdata_t *evdata;
553 	usba_device_t *usba_device = usba_get_usba_device(dip);
554 
555 	/* called when dip attaches */
556 	ASSERT(usba_device != NULL);
557 
558 	mutex_enter(&usba_device->usb_mutex);
559 	evdata = usba_device->usb_evdata;
560 	while (evdata) {
561 		if (evdata->ev_dip == dip) {
562 			mutex_exit(&usba_device->usb_mutex);
563 
564 			return (evdata);
565 		}
566 		evdata = evdata->ev_next;
567 	}
568 
569 	evdata = kmem_zalloc(sizeof (usba_evdata_t), KM_SLEEP);
570 	evdata->ev_dip = dip;
571 	evdata->ev_next = usba_device->usb_evdata;
572 	usba_device->usb_evdata = evdata;
573 	mutex_exit(&usba_device->usb_mutex);
574 
575 	return (evdata);
576 }
577 
578 
579 /*
580  * allocate a usb device structure and link it in the list
581  */
582 usba_device_t *
583 usba_alloc_usba_device(dev_info_t *root_hub_dip)
584 {
585 	usba_device_t	*usba_device;
586 	int		ep_idx;
587 	ddi_iblock_cookie_t iblock_cookie =
588 	    usba_hcdi_get_hcdi(root_hub_dip)->hcdi_iblock_cookie;
589 
590 	/*
591 	 * create a new usba_device structure
592 	 */
593 	usba_device = kmem_zalloc(sizeof (usba_device_t), KM_SLEEP);
594 
595 	/*
596 	 * initialize usba_device
597 	 */
598 	mutex_init(&usba_device->usb_mutex, NULL, MUTEX_DRIVER,
599 	    iblock_cookie);
600 
601 	usba_init_list(&usba_device->usb_device_list, (usb_opaque_t)usba_device,
602 	    iblock_cookie);
603 	usba_init_list(&usba_device->usb_allocated, (usb_opaque_t)usba_device,
604 	    iblock_cookie);
605 	mutex_enter(&usba_device->usb_mutex);
606 	usba_device->usb_root_hub_dip = root_hub_dip;
607 
608 	/*
609 	 * add to list of usba_devices
610 	 */
611 	usba_add_to_list(&usba_device_list, &usba_device->usb_device_list);
612 
613 	/* init mutex in each usba_ph_impl structure */
614 	for (ep_idx = 0; ep_idx < USBA_N_ENDPOINTS; ep_idx++) {
615 		mutex_init(&usba_device->usb_ph_list[ep_idx].usba_ph_mutex,
616 		    NULL, MUTEX_DRIVER, iblock_cookie);
617 	}
618 
619 	USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
620 	    "allocated usba_device 0x%p", (void *)usba_device);
621 
622 	mutex_exit(&usba_device->usb_mutex);
623 
624 	return (usba_device);
625 }
626 
627 
628 /* free NDI event data associated with usba_device */
629 void
630 usba_free_evdata(usba_evdata_t *evdata)
631 {
632 	usba_evdata_t *next;
633 
634 	while (evdata) {
635 		next = evdata->ev_next;
636 		kmem_free(evdata, sizeof (usba_evdata_t));
637 		evdata = next;
638 	}
639 }
640 
641 
642 /*
643  * free usb device structure
644  */
645 void
646 usba_free_usba_device(usba_device_t *usba_device)
647 {
648 	int			i, ep_idx;
649 	usb_pipe_handle_t	def_ph;
650 
651 	if (usba_device == NULL) {
652 
653 		return;
654 	}
655 
656 	mutex_enter(&usba_device->usb_mutex);
657 	if (usba_device->usb_ref_count) {
658 		mutex_exit(&usba_device->usb_mutex);
659 
660 		return;
661 	}
662 
663 	USB_DPRINTF_L3(DPRINT_MASK_USBA, usba_log_handle,
664 	    "usba_free_usba_device 0x%p, address=0x%x, ref cnt=%d",
665 	    (void *)usba_device, usba_device->usb_addr,
666 	    usba_device->usb_ref_count);
667 
668 	usba_free_evdata(usba_device->usb_evdata);
669 	mutex_exit(&usba_device->usb_mutex);
670 
671 	def_ph = usba_usbdev_to_dflt_pipe_handle(usba_device);
672 	if (def_ph != NULL) {
673 		usba_pipe_handle_data_t	*ph_data = usba_get_ph_data(def_ph);
674 
675 		if (ph_data) {
676 			usb_pipe_close(ph_data->p_dip, def_ph,
677 			    USB_FLAGS_SLEEP | USBA_FLAGS_PRIVILEGED,
678 			    NULL, NULL);
679 		}
680 	}
681 
682 	mutex_enter(&usba_mutex);
683 
684 	/* destroy mutex in each usba_ph_impl structure */
685 	for (ep_idx = 0; ep_idx < USBA_N_ENDPOINTS; ep_idx++) {
686 		mutex_destroy(&usba_device->usb_ph_list[ep_idx].usba_ph_mutex);
687 	}
688 
689 	(void) usba_rm_from_list(&usba_device_list,
690 	    &usba_device->usb_device_list);
691 
692 	mutex_exit(&usba_mutex);
693 
694 	usba_destroy_list(&usba_device->usb_device_list);
695 	usba_destroy_list(&usba_device->usb_allocated);
696 
697 	USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
698 	    "deallocating usba_device = 0x%p, address = 0x%x",
699 	    (void *)usba_device, usba_device->usb_addr);
700 
701 	/*
702 	 * ohci allocates descriptors for root hub so we can't
703 	 * deallocate these here
704 	 */
705 
706 	if (usba_device->usb_addr != ROOT_HUB_ADDR) {
707 		if (usba_device->usb_cfg_array) {
708 			USB_DPRINTF_L3(DPRINT_MASK_USBA, usba_log_handle,
709 			    "deallocating usb_config_array: 0x%p",
710 			    (void *)usba_device->usb_cfg_array);
711 			mutex_enter(&usba_device->usb_mutex);
712 			for (i = 0;
713 			    i < usba_device->usb_dev_descr->bNumConfigurations;
714 			    i++) {
715 				if (usba_device->usb_cfg_array[i]) {
716 					kmem_free(
717 					    usba_device->usb_cfg_array[i],
718 					    usba_device->usb_cfg_array_len[i]);
719 				}
720 			}
721 
722 			/* free the array pointers */
723 			kmem_free(usba_device->usb_cfg_array,
724 			    usba_device->usb_cfg_array_length);
725 			kmem_free(usba_device->usb_cfg_array_len,
726 			    usba_device->usb_cfg_array_len_length);
727 
728 			mutex_exit(&usba_device->usb_mutex);
729 		}
730 
731 		if (usba_device->usb_cfg_str_descr) {
732 			USB_DPRINTF_L3(DPRINT_MASK_USBA, usba_log_handle,
733 			    "deallocating usb_cfg_str_descr: 0x%p",
734 			    (void *)usba_device->usb_cfg_str_descr);
735 			for (i = 0;
736 			    i < usba_device->usb_dev_descr->bNumConfigurations;
737 			    i++) {
738 				if (usba_device->usb_cfg_str_descr[i]) {
739 					kmem_free(
740 					    usba_device->usb_cfg_str_descr[i],
741 					    strlen(usba_device->
742 					    usb_cfg_str_descr[i]) + 1);
743 				}
744 			}
745 			/* free the array pointers */
746 			kmem_free(usba_device->usb_cfg_str_descr,
747 			    sizeof (uchar_t *) * usba_device->usb_n_cfgs);
748 		}
749 
750 		if (usba_device->usb_dev_descr) {
751 			kmem_free(usba_device->usb_dev_descr,
752 			    sizeof (usb_dev_descr_t));
753 		}
754 
755 		if (usba_device->usb_mfg_str) {
756 			kmem_free(usba_device->usb_mfg_str,
757 			    strlen(usba_device->usb_mfg_str) + 1);
758 		}
759 
760 		if (usba_device->usb_product_str) {
761 			kmem_free(usba_device->usb_product_str,
762 			    strlen(usba_device->usb_product_str) + 1);
763 		}
764 
765 		if (usba_device->usb_serialno_str) {
766 			kmem_free(usba_device->usb_serialno_str,
767 			    strlen(usba_device->usb_serialno_str) + 1);
768 		}
769 
770 		usba_unset_usb_address(usba_device);
771 	}
772 
773 #ifndef __lock_lint
774 	ASSERT(usba_device->usb_client_dev_data_list.cddl_next == NULL);
775 #endif
776 
777 	if (usba_device->usb_client_flags) {
778 #ifndef __lock_lint
779 		int i;
780 
781 		for (i = 0; i < usba_device->usb_n_ifs; i++) {
782 			ASSERT(usba_device->usb_client_flags[i] == 0);
783 		}
784 #endif
785 		kmem_free(usba_device->usb_client_flags,
786 		    usba_device->usb_n_ifs * USBA_CLIENT_FLAG_SIZE);
787 	}
788 
789 
790 	if (usba_device->usb_client_attach_list) {
791 		kmem_free(usba_device->usb_client_attach_list,
792 		    usba_device->usb_n_ifs *
793 		    sizeof (*usba_device->usb_client_attach_list));
794 	}
795 	if (usba_device->usb_client_ev_cb_list) {
796 		kmem_free(usba_device->usb_client_ev_cb_list,
797 		    usba_device->usb_n_ifs *
798 		    sizeof (*usba_device->usb_client_ev_cb_list));
799 	}
800 
801 	/*
802 	 * finally ready to destroy the structure
803 	 */
804 	mutex_destroy(&usba_device->usb_mutex);
805 
806 	kmem_free((caddr_t)usba_device, sizeof (usba_device_t));
807 }
808 
809 
810 /* clear the data toggle for all endpoints on this device */
811 void
812 usba_clear_data_toggle(usba_device_t *usba_device)
813 {
814 	int	i;
815 
816 	if (usba_device != NULL) {
817 		mutex_enter(&usba_device->usb_mutex);
818 		for (i = 0; i < USBA_N_ENDPOINTS; i++) {
819 			usba_device->usb_ph_list[i].usba_ph_flags &=
820 			    ~USBA_PH_DATA_TOGGLE;
821 		}
822 		mutex_exit(&usba_device->usb_mutex);
823 	}
824 }
825 
826 
827 /*
828  * usba_create_child_devi():
829  *	create a child devinfo node, usba_device, attach properties.
830  *	the usba_device structure is shared between all interfaces
831  */
832 int
833 usba_create_child_devi(dev_info_t	*dip,
834 		char			*node_name,
835 		usba_hcdi_ops_t		*usba_hcdi_ops,
836 		dev_info_t		*usb_root_hub_dip,
837 		usb_port_status_t	port_status,
838 		usba_device_t		*usba_device,
839 		dev_info_t		**child_dip)
840 {
841 	int rval = USB_FAILURE;
842 	int usba_device_allocated = 0;
843 	usb_addr_t	address;
844 
845 	USB_DPRINTF_L4(DPRINT_MASK_USBA, usba_log_handle,
846 	    "usba_create_child_devi: %s usba_device=0x%p "
847 	    "port status=0x%x", node_name,
848 	    (void *)usba_device, port_status);
849 
850 	ndi_devi_alloc_sleep(dip, node_name, (pnode_t)DEVI_SID_NODEID,
851 	    child_dip);
852 
853 	USB_DPRINTF_L3(DPRINT_MASK_USBA, usba_log_handle,
854 	    "child dip=0x%p", (void *)*child_dip);
855 
856 	if (usba_device == NULL) {
857 
858 		usba_device = usba_alloc_usba_device(usb_root_hub_dip);
859 
860 		/* grab the mutex to keep warlock happy */
861 		mutex_enter(&usba_device->usb_mutex);
862 		usba_device->usb_hcdi_ops	= usba_hcdi_ops;
863 		usba_device->usb_port_status	= port_status;
864 		mutex_exit(&usba_device->usb_mutex);
865 
866 		usba_device_allocated++;
867 	} else {
868 		mutex_enter(&usba_device->usb_mutex);
869 		if (usba_hcdi_ops) {
870 			ASSERT(usba_device->usb_hcdi_ops == usba_hcdi_ops);
871 		}
872 		if (usb_root_hub_dip) {
873 			ASSERT(usba_device->usb_root_hub_dip ==
874 			    usb_root_hub_dip);
875 		}
876 
877 		usba_device->usb_port_status	= port_status;
878 
879 		mutex_exit(&usba_device->usb_mutex);
880 	}
881 
882 	if (usba_device->usb_addr == 0) {
883 		if (usba_set_usb_address(usba_device) == USB_FAILURE) {
884 			address = 0;
885 
886 			USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
887 			    "cannot set usb address for dip=0x%p",
888 			    (void *)*child_dip);
889 
890 			goto fail;
891 		}
892 	}
893 	address = usba_device->usb_addr;
894 
895 	/* attach properties */
896 	rval = ndi_prop_update_int(DDI_DEV_T_NONE, *child_dip,
897 	    "assigned-address", address);
898 	if (rval != DDI_PROP_SUCCESS) {
899 		USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
900 		    "cannot set usb address property for dip=0x%p",
901 		    (void *)*child_dip);
902 		rval = USB_FAILURE;
903 
904 		goto fail;
905 	}
906 
907 	/*
908 	 * store the usba_device point in the dip
909 	 */
910 	usba_set_usba_device(*child_dip, usba_device);
911 
912 	USB_DPRINTF_L4(DPRINT_MASK_USBA, usba_log_handle,
913 	    "usba_create_child_devi: devi=0x%p (%s) ud=0x%p",
914 	    (void *)*child_dip, ddi_driver_name(*child_dip),
915 	    (void *)usba_device);
916 
917 	return (USB_SUCCESS);
918 
919 fail:
920 	if (*child_dip) {
921 		int rval = usba_destroy_child_devi(*child_dip, NDI_DEVI_REMOVE);
922 		ASSERT(rval == USB_SUCCESS);
923 		*child_dip = NULL;
924 	}
925 
926 	if (usba_device_allocated) {
927 		usba_free_usba_device(usba_device);
928 	} else if (address && usba_device) {
929 		usba_unset_usb_address(usba_device);
930 	}
931 
932 	USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
933 	    "usba_create_child_devi failed: rval=%d", rval);
934 
935 	return (rval);
936 }
937 
938 
939 int
940 usba_destroy_child_devi(dev_info_t *dip, uint_t flag)
941 {
942 	usba_device_t	*usba_device;
943 	int		rval = NDI_SUCCESS;
944 
945 	USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
946 	    "usba_destroy_child_devi: %s%d (0x%p)",
947 	    ddi_driver_name(dip), ddi_get_instance(dip), (void *)dip);
948 
949 	usba_device = usba_get_usba_device(dip);
950 
951 	/*
952 	 * if the child hasn't been bound yet, we can just
953 	 * free the dip
954 	 */
955 	if (i_ddi_node_state(dip) < DS_INITIALIZED) {
956 		/*
957 		 * do not call ndi_devi_free() since it might
958 		 * deadlock
959 		 */
960 		rval = ddi_remove_child(dip, 0);
961 
962 	} else {
963 		char *devnm = kmem_alloc(MAXNAMELEN + 1, KM_SLEEP);
964 		dev_info_t *pdip = ddi_get_parent(dip);
965 
966 		(void) ddi_deviname(dip, devnm);
967 
968 		USB_DPRINTF_L3(DPRINT_MASK_USBA, usba_log_handle,
969 		    "usba_destroy_child_devi:\n\t"
970 		    "offlining dip 0x%p usba_device=0x%p (%s)", (void *)dip,
971 		    (void *)usba_device, devnm);
972 
973 		(void) devfs_clean(pdip, NULL, DV_CLEAN_FORCE);
974 		rval =	ndi_devi_unconfig_one(pdip, devnm + 1, NULL,
975 		    flag | NDI_UNCONFIG | NDI_DEVI_OFFLINE);
976 		if (rval != NDI_SUCCESS) {
977 			USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
978 			    " ndi_devi_unconfig_one %s%d failed (%d)",
979 			    ddi_driver_name(dip), ddi_get_instance(dip),
980 			    rval);
981 		}
982 		kmem_free(devnm, MAXNAMELEN + 1);
983 	}
984 
985 	USB_DPRINTF_L4(DPRINT_MASK_USBA, usba_log_handle,
986 	    "usba_destroy_child_devi: rval=%d", rval);
987 
988 	return (rval == NDI_SUCCESS ? USB_SUCCESS : USB_FAILURE);
989 }
990 
991 
992 /*
993  * list management
994  */
995 void
996 usba_init_list(usba_list_entry_t *element, usb_opaque_t private,
997 	ddi_iblock_cookie_t	iblock_cookie)
998 {
999 	mutex_init(&element->list_mutex, NULL, MUTEX_DRIVER,
1000 	    iblock_cookie);
1001 	mutex_enter(&element->list_mutex);
1002 	element->private = private;
1003 	mutex_exit(&element->list_mutex);
1004 }
1005 
1006 
1007 void
1008 usba_destroy_list(usba_list_entry_t *head)
1009 {
1010 	mutex_enter(&head->list_mutex);
1011 	ASSERT(head->next == NULL);
1012 	ASSERT(head->prev == NULL);
1013 	mutex_exit(&head->list_mutex);
1014 
1015 	mutex_destroy(&head->list_mutex);
1016 }
1017 
1018 
1019 void
1020 usba_add_to_list(usba_list_entry_t *head, usba_list_entry_t *element)
1021 {
1022 	usba_list_entry_t *next;
1023 	int		remaining;
1024 
1025 	mutex_enter(&head->list_mutex);
1026 	mutex_enter(&element->list_mutex);
1027 
1028 	remaining = head->count;
1029 
1030 	/* check if it is not in another list */
1031 	ASSERT(element->next == NULL);
1032 	ASSERT(element->prev == NULL);
1033 
1034 #ifdef DEBUG
1035 	/*
1036 	 * only verify the list when not in interrupt context, we
1037 	 * have to trust the HCD
1038 	 */
1039 	if (!servicing_interrupt()) {
1040 
1041 		/* check if not already in this list */
1042 		for (next = head->next; (next != NULL);
1043 		    next = next->next) {
1044 			if (next == element) {
1045 				USB_DPRINTF_L0(DPRINT_MASK_USBA,
1046 				    usba_log_handle,
1047 				    "Attempt to corrupt USB list at 0x%p",
1048 				    (void *)head);
1049 				ASSERT(next == element);
1050 
1051 				goto done;
1052 			}
1053 			remaining--;
1054 
1055 			/*
1056 			 * Detect incorrect circ links or found
1057 			 * unexpected elements.
1058 			 */
1059 			if ((next->next && (remaining == 0)) ||
1060 			    ((next->next == NULL) && remaining)) {
1061 				panic("Corrupted USB list at 0x%p",
1062 				    (void *)head);
1063 				/*NOTREACHED*/
1064 			}
1065 		}
1066 	}
1067 #endif
1068 
1069 	if (head->next == NULL) {
1070 		head->prev = head->next = element;
1071 	} else {
1072 		/* add to tail */
1073 		head->prev->next = element;
1074 		element->prev = head->prev;
1075 		head->prev = element;
1076 	}
1077 
1078 	head->count++;
1079 
1080 	USB_DPRINTF_L4(DPRINT_MASK_USBA, usba_log_handle,
1081 	    "usba_add_to_list: head=0x%p element=0x%p count=%d",
1082 	    (void *)head, (void *)element, head->count);
1083 
1084 done:
1085 	mutex_exit(&head->list_mutex);
1086 	mutex_exit(&element->list_mutex);
1087 }
1088 
1089 
1090 int
1091 usba_rm_from_list(usba_list_entry_t *head, usba_list_entry_t *element)
1092 {
1093 	usba_list_entry_t *e;
1094 	int		found = 0;
1095 	int		remaining;
1096 
1097 	/* find the element in the list first */
1098 	mutex_enter(&head->list_mutex);
1099 
1100 	USB_DPRINTF_L4(DPRINT_MASK_USBA, usba_log_handle,
1101 	    "usba_rm_from_list: head=0x%p element=0x%p count=%d",
1102 	    (void *)head, (void *)element, head->count);
1103 
1104 	remaining = head->count;
1105 	e = head->next;
1106 
1107 	while (e) {
1108 		if (e == element) {
1109 			found++;
1110 			break;
1111 		}
1112 		e = e->next;
1113 
1114 		remaining--;
1115 
1116 		/* Detect incorrect circ links or found unexpected elements. */
1117 		if ((e && (remaining == 0)) ||
1118 		    ((e == NULL) && (remaining))) {
1119 			panic("Corrupted USB list at 0x%p", (void *)head);
1120 			/*NOTREACHED*/
1121 		}
1122 	}
1123 
1124 	if (!found) {
1125 		mutex_exit(&head->list_mutex);
1126 
1127 		return (USB_FAILURE);
1128 	}
1129 
1130 	/* now remove the element */
1131 	mutex_enter(&element->list_mutex);
1132 
1133 	if (element->next) {
1134 		element->next->prev = element->prev;
1135 	}
1136 	if (element->prev) {
1137 		element->prev->next = element->next;
1138 	}
1139 	if (head->next == element) {
1140 		head->next = element->next;
1141 	}
1142 	if (head->prev == element) {
1143 		head->prev = element->prev;
1144 	}
1145 
1146 	element->prev = element->next = NULL;
1147 	if (head->next == NULL) {
1148 		ASSERT(head->prev == NULL);
1149 	} else {
1150 		ASSERT(head->next->prev == NULL);
1151 	}
1152 	if (head->prev == NULL) {
1153 		ASSERT(head->next == NULL);
1154 	} else {
1155 		ASSERT(head->prev->next == NULL);
1156 	}
1157 
1158 	head->count--;
1159 
1160 	USB_DPRINTF_L4(DPRINT_MASK_USBA, usba_log_handle,
1161 	    "usba_rm_from_list success: head=0x%p element=0x%p cnt=%d",
1162 	    (void *)head, (void *)element, head->count);
1163 
1164 	mutex_exit(&element->list_mutex);
1165 	mutex_exit(&head->list_mutex);
1166 
1167 	return (USB_SUCCESS);
1168 }
1169 
1170 
1171 usba_list_entry_t *
1172 usba_rm_first_from_list(usba_list_entry_t *head)
1173 {
1174 	usba_list_entry_t *element = NULL;
1175 
1176 	if (head) {
1177 		mutex_enter(&head->list_mutex);
1178 		element = head->next;
1179 		if (element) {
1180 			/* now remove the element */
1181 			mutex_enter(&element->list_mutex);
1182 			head->next = element->next;
1183 			if (head->next) {
1184 				head->next->prev = NULL;
1185 			}
1186 			if (head->prev == element) {
1187 				head->prev = element->next;
1188 			}
1189 			element->prev = element->next = NULL;
1190 			mutex_exit(&element->list_mutex);
1191 			head->count--;
1192 		}
1193 		if (head->next == NULL) {
1194 			ASSERT(head->prev == NULL);
1195 		} else {
1196 			ASSERT(head->next->prev == NULL);
1197 		}
1198 		if (head->prev == NULL) {
1199 			ASSERT(head->next == NULL);
1200 		} else {
1201 			ASSERT(head->prev->next == NULL);
1202 		}
1203 		USB_DPRINTF_L4(DPRINT_MASK_USBA, usba_log_handle,
1204 		    "usba_rm_first_from_list: head=0x%p el=0x%p cnt=%d",
1205 		    (void *)head, (void *)element, head->count);
1206 
1207 		mutex_exit(&head->list_mutex);
1208 	}
1209 
1210 	return (element);
1211 }
1212 
1213 
1214 usb_opaque_t
1215 usba_rm_first_pvt_from_list(usba_list_entry_t *head)
1216 {
1217 	usba_list_entry_t *element = usba_rm_first_from_list(head);
1218 	usb_opaque_t private = NULL;
1219 
1220 	if (element) {
1221 		mutex_enter(&element->list_mutex);
1222 		private = element->private;
1223 		mutex_exit(&element->list_mutex);
1224 	}
1225 
1226 	return (private);
1227 }
1228 
1229 
1230 /*
1231  * move list to new list and zero original list
1232  */
1233 void
1234 usba_move_list(usba_list_entry_t *head, usba_list_entry_t *new,
1235 	ddi_iblock_cookie_t iblock_cookie)
1236 {
1237 	usba_init_list(new, NULL, iblock_cookie);
1238 	mutex_enter(&head->list_mutex);
1239 	mutex_enter(&new->list_mutex);
1240 
1241 	new->next = head->next;
1242 	new->prev = head->prev;
1243 	new->count = head->count;
1244 	new->private = head->private;
1245 
1246 	head->next = NULL;
1247 	head->prev = NULL;
1248 	head->count = 0;
1249 	head->private = NULL;
1250 	mutex_exit(&head->list_mutex);
1251 	mutex_exit(&new->list_mutex);
1252 }
1253 
1254 
1255 int
1256 usba_check_in_list(usba_list_entry_t *head, usba_list_entry_t *element)
1257 {
1258 	int		rval = USB_FAILURE;
1259 	int		remaining;
1260 	usba_list_entry_t *next;
1261 
1262 	mutex_enter(&head->list_mutex);
1263 	remaining = head->count;
1264 
1265 	mutex_enter(&element->list_mutex);
1266 	for (next = head->next; next != NULL; next = next->next) {
1267 		if (next == element) {
1268 			rval = USB_SUCCESS;
1269 			break;
1270 		}
1271 		remaining--;
1272 
1273 		/* Detect incorrect circ links or found unexpected elements. */
1274 		if ((next->next && (remaining == 0)) ||
1275 		    ((next->next == NULL) && remaining)) {
1276 			panic("Corrupted USB list at 0x%p", (void *)head);
1277 			/*NOTREACHED*/
1278 		}
1279 	}
1280 	mutex_exit(&element->list_mutex);
1281 	mutex_exit(&head->list_mutex);
1282 
1283 	return (rval);
1284 }
1285 
1286 
1287 int
1288 usba_list_entry_leaks(usba_list_entry_t *head, char *what)
1289 {
1290 	int		count = 0;
1291 	int		remaining;
1292 	usba_list_entry_t *next;
1293 
1294 	mutex_enter(&head->list_mutex);
1295 	remaining = head->count;
1296 	for (next = head->next; next != NULL; next = next->next) {
1297 		USB_DPRINTF_L2(DPRINT_MASK_HCDI, usba_log_handle,
1298 		    "leaking %s 0x%p", what, (void *)next->private);
1299 		count++;
1300 
1301 		remaining--;
1302 
1303 		/* Detect incorrect circ links or found unexpected elements. */
1304 		if ((next->next && (remaining == 0)) ||
1305 		    ((next->next == NULL) && remaining)) {
1306 			panic("Corrupted USB list at 0x%p", (void *)head);
1307 			/*NOTREACHED*/
1308 		}
1309 	}
1310 	ASSERT(count == head->count);
1311 	mutex_exit(&head->list_mutex);
1312 
1313 	if (count) {
1314 		USB_DPRINTF_L2(DPRINT_MASK_HCDI, usba_log_handle,
1315 		    "usba_list_entry_count: leaking %d", count);
1316 	}
1317 
1318 	return (count);
1319 }
1320 
1321 
1322 int
1323 usba_list_entry_count(usba_list_entry_t *head)
1324 {
1325 	int count;
1326 
1327 	mutex_enter(&head->list_mutex);
1328 	count = head->count;
1329 	mutex_exit(&head->list_mutex);
1330 
1331 	return (count);
1332 }
1333 
1334 /* add a new root hub to the usba_root_hubs list */
1335 
1336 void
1337 usba_add_root_hub(dev_info_t *dip)
1338 {
1339 	usba_root_hub_ent_t *hub;
1340 
1341 	hub = (usba_root_hub_ent_t *)
1342 	    kmem_zalloc(sizeof (usba_root_hub_ent_t), KM_SLEEP);
1343 
1344 	mutex_enter(&usba_hub_mutex);
1345 	hub->dip = dip;
1346 	hub->next = usba_root_hubs;
1347 	usba_root_hubs = hub;
1348 	mutex_exit(&usba_hub_mutex);
1349 }
1350 
1351 /* remove a root hub from the usba_root_hubs list */
1352 
1353 void
1354 usba_rem_root_hub(dev_info_t *dip)
1355 {
1356 	usba_root_hub_ent_t **hubp, *hub;
1357 
1358 	mutex_enter(&usba_hub_mutex);
1359 	hubp = &usba_root_hubs;
1360 	while (*hubp) {
1361 		if ((*hubp)->dip == dip) {
1362 			hub = *hubp;
1363 			*hubp = hub->next;
1364 			kmem_free(hub, sizeof (struct usba_root_hub_ent));
1365 			mutex_exit(&usba_hub_mutex);
1366 
1367 			return;
1368 		}
1369 		hubp = &(*hubp)->next;
1370 	}
1371 	mutex_exit(&usba_hub_mutex);
1372 }
1373 
1374 /*
1375  * check whether this dip is the root hub. Any root hub known by
1376  * usba is recorded in the linked list pointed to by usba_root_hubs
1377  */
1378 int
1379 usba_is_root_hub(dev_info_t *dip)
1380 {
1381 	usba_root_hub_ent_t *hub;
1382 
1383 	mutex_enter(&usba_hub_mutex);
1384 	hub = usba_root_hubs;
1385 	while (hub) {
1386 		if (hub->dip == dip) {
1387 			mutex_exit(&usba_hub_mutex);
1388 
1389 			return (1);
1390 		}
1391 		hub = hub->next;
1392 	}
1393 	mutex_exit(&usba_hub_mutex);
1394 
1395 	return (0);
1396 }
1397 
1398 /*
1399  * get and store usba_device pointer in the devi
1400  */
1401 usba_device_t *
1402 usba_get_usba_device(dev_info_t *dip)
1403 {
1404 	/*
1405 	 * we cannot use parent_data in the usb node because its
1406 	 * bus parent (eg. PCI nexus driver) uses this data
1407 	 *
1408 	 * we cannot use driver data in the other usb nodes since
1409 	 * usb drivers may need to use this
1410 	 */
1411 	if (usba_is_root_hub(dip)) {
1412 		usba_hcdi_t *hcdi = usba_hcdi_get_hcdi(dip);
1413 
1414 		return (hcdi->hcdi_usba_device);
1415 	} else {
1416 
1417 		return (ddi_get_parent_data(dip));
1418 	}
1419 }
1420 
1421 
1422 /*
1423  * Retrieve the usba_device pointer from the dev without checking for
1424  * the root hub first.	This function is only used in polled mode.
1425  */
1426 usba_device_t *
1427 usba_polled_get_usba_device(dev_info_t *dip)
1428 {
1429 	/*
1430 	 * Don't call usba_is_root_hub() to find out if this is
1431 	 * the root hub  usba_is_root_hub() calls into the DDI
1432 	 * where there are locking issues. The dip sent in during
1433 	 * polled mode will never be the root hub, so just get
1434 	 * the usba_device pointer from the dip.
1435 	 */
1436 	return (ddi_get_parent_data(dip));
1437 }
1438 
1439 
1440 void
1441 usba_set_usba_device(dev_info_t *dip, usba_device_t *usba_device)
1442 {
1443 	if (usba_is_root_hub(dip)) {
1444 		usba_hcdi_t *hcdi = usba_hcdi_get_hcdi(dip);
1445 		/* no locking is needed here */
1446 		_NOTE(NOW_INVISIBLE_TO_OTHER_THREADS(hcdi->hcdi_usba_device))
1447 		hcdi->hcdi_usba_device = usba_device;
1448 		_NOTE(NOW_VISIBLE_TO_OTHER_THREADS(hcdi->hcdi_usba_device))
1449 	} else {
1450 		ddi_set_parent_data(dip, usba_device);
1451 	}
1452 }
1453 
1454 
1455 /*
1456  * usba_set_node_name() according to class, subclass, and protocol
1457  * following the 1275 USB binding tables.
1458  */
1459 
1460 /* device node table, refer to section 3.2.2.1 of 1275 binding */
1461 static node_name_entry_t device_node_name_table[] = {
1462 { USB_CLASS_COMM,	DONTCARE,	DONTCARE,	"communications" },
1463 { USB_CLASS_HUB,	DONTCARE,	DONTCARE,	"hub" },
1464 { USB_CLASS_DIAG,	DONTCARE,	DONTCARE,	"diagnostics" },
1465 { USB_CLASS_MISC,	DONTCARE,	DONTCARE,	"miscellaneous" },
1466 { DONTCARE,		DONTCARE,	DONTCARE,	"device" }
1467 };
1468 
1469 /* interface-association node table */
1470 static node_name_entry_t ia_node_name_table[] = {
1471 { USB_CLASS_AUDIO,	DONTCARE,	DONTCARE, "audio" },
1472 { USB_CLASS_VIDEO,	DONTCARE,	DONTCARE, "video" },
1473 { USB_CLASS_WIRELESS,	USB_SUBCLS_WUSB_2, USB_PROTO_WUSB_DWA,
1474 						"device-wire-adaptor" },
1475 { USB_CLASS_WIRELESS,	DONTCARE,	DONTCARE, "wireless-controller" },
1476 { DONTCARE,		DONTCARE,	DONTCARE, "interface-association" }
1477 };
1478 
1479 /* interface node table, refer to section 3.3.2.1 */
1480 static node_name_entry_t if_node_name_table[] = {
1481 { USB_CLASS_AUDIO, USB_SUBCLS_AUD_CONTROL, DONTCARE,	"sound-control" },
1482 { USB_CLASS_AUDIO, USB_SUBCLS_AUD_STREAMING, DONTCARE, "sound" },
1483 { USB_CLASS_AUDIO, USB_SUBCLS_AUD_MIDI_STREAMING, DONTCARE, "midi" },
1484 { USB_CLASS_AUDIO, DONTCARE,		DONTCARE,	"sound" },
1485 
1486 { USB_CLASS_COMM, USB_SUBCLS_CDCC_DIRECT_LINE,	DONTCARE, "line" },
1487 { USB_CLASS_COMM, USB_SUBCLS_CDCC_ABSTRCT_CTRL,	DONTCARE, "modem" },
1488 { USB_CLASS_COMM, USB_SUBCLS_CDCC_PHONE_CTRL, DONTCARE, "telephone" },
1489 { USB_CLASS_COMM, USB_SUBCLS_CDCC_MULTCNL_ISDN, DONTCARE, "isdn" },
1490 { USB_CLASS_COMM, USB_SUBCLS_CDCC_ISDN,		DONTCARE, "isdn" },
1491 { USB_CLASS_COMM, USB_SUBCLS_CDCC_ETHERNET,	DONTCARE, "ethernet" },
1492 { USB_CLASS_COMM, USB_SUBCLS_CDCC_ATM_NETWORK, DONTCARE, "atm-network" },
1493 { USB_CLASS_COMM, DONTCARE,		DONTCARE,	"communications" },
1494 
1495 { USB_CLASS_HID, USB_SUBCLS_HID_1, USB_PROTO_HID_KEYBOARD,	"keyboard" },
1496 { USB_CLASS_HID, USB_SUBCLS_HID_1, USB_PROTO_HID_MOUSE,	"mouse" },
1497 { USB_CLASS_HID,	DONTCARE,	DONTCARE,	"input" },
1498 
1499 { USB_CLASS_HUB,	DONTCARE,	DONTCARE,	"hub" },
1500 
1501 { USB_CLASS_PHYSICAL,	DONTCARE,	DONTCARE,	"physical" },
1502 
1503 { USB_CLASS_IMAGE,	DONTCARE,	DONTCARE,	"image" },
1504 
1505 { USB_CLASS_PRINTER,	DONTCARE,	DONTCARE,	"printer" },
1506 
1507 { USB_CLASS_MASS_STORAGE, DONTCARE,	DONTCARE,	"storage" },
1508 
1509 { USB_CLASS_CDC_DATA,	DONTCARE,	DONTCARE,	"data" },
1510 
1511 { USB_CLASS_SECURITY,	DONTCARE,	DONTCARE,	"security" },
1512 
1513 { USB_CLASS_VIDEO, USB_SUBCLS_VIDEO_CONTROL, DONTCARE,	"video-control" },
1514 { USB_CLASS_VIDEO, USB_SUBCLS_VIDEO_STREAM,  DONTCARE,	"video-stream" },
1515 { USB_CLASS_VIDEO,	DONTCARE,	DONTCARE,	"video" },
1516 
1517 { USB_CLASS_APP,	USB_SUBCLS_APP_FIRMWARE, DONTCARE, "firmware" },
1518 { USB_CLASS_APP,	USB_SUBCLS_APP_IRDA,	DONTCARE, "IrDa" },
1519 { USB_CLASS_APP,	USB_SUBCLS_APP_TEST,	DONTCARE, "test" },
1520 
1521 { USB_CLASS_MISC,	USB_SUBCLS_CBAF, USB_PROTO_CBAF,  "wusb_ca"},
1522 { USB_CLASS_WIRELESS, USB_SUBCLS_WUSB_1, USB_PROTO_WUSB_RC, "hwa-radio" },
1523 { USB_CLASS_WIRELESS, USB_SUBCLS_WUSB_2, USB_PROTO_WUSB_HWA, "hwa-host" },
1524 { USB_CLASS_WIRELESS, USB_SUBCLS_WUSB_2, USB_PROTO_WUSB_DWA, "dwa-control" },
1525 { USB_CLASS_WIRELESS, USB_SUBCLS_WUSB_2, USB_PROTO_WUSB_DWA_ISO, "dwa-isoc" },
1526 { USB_CLASS_WIRELESS, DONTCARE, DONTCARE, "wireless" },
1527 
1528 { DONTCARE,		DONTCARE,	DONTCARE,	"interface" },
1529 
1530 };
1531 
1532 /* combined node table, refer to section 3.4.2.1 */
1533 static node_name_entry_t combined_node_name_table[] = {
1534 { USB_CLASS_AUDIO, USB_SUBCLS_AUD_CONTROL, DONTCARE,	"sound-control" },
1535 { USB_CLASS_AUDIO, USB_SUBCLS_AUD_STREAMING, DONTCARE, "sound" },
1536 { USB_CLASS_AUDIO, USB_SUBCLS_AUD_MIDI_STREAMING, DONTCARE, "midi" },
1537 { USB_CLASS_AUDIO, DONTCARE,		DONTCARE,	"sound" },
1538 
1539 { USB_CLASS_COMM, USB_SUBCLS_CDCC_DIRECT_LINE,	DONTCARE, "line" },
1540 { USB_CLASS_COMM, USB_SUBCLS_CDCC_ABSTRCT_CTRL,	DONTCARE, "modem" },
1541 { USB_CLASS_COMM, USB_SUBCLS_CDCC_PHONE_CTRL, DONTCARE, "telephone" },
1542 { USB_CLASS_COMM, USB_SUBCLS_CDCC_MULTCNL_ISDN, DONTCARE, "isdn" },
1543 { USB_CLASS_COMM, USB_SUBCLS_CDCC_ISDN,		DONTCARE, "isdn" },
1544 { USB_CLASS_COMM, USB_SUBCLS_CDCC_ETHERNET,	DONTCARE, "ethernet" },
1545 { USB_CLASS_COMM, USB_SUBCLS_CDCC_ATM_NETWORK, DONTCARE, "atm-network" },
1546 { USB_CLASS_COMM, DONTCARE,		DONTCARE,	"communications" },
1547 
1548 { USB_CLASS_HID, USB_SUBCLS_HID_1, USB_PROTO_HID_KEYBOARD, "keyboard" },
1549 { USB_CLASS_HID, USB_SUBCLS_HID_1, USB_PROTO_HID_MOUSE,	"mouse" },
1550 { USB_CLASS_HID,	DONTCARE,	DONTCARE,	"input" },
1551 
1552 { USB_CLASS_PHYSICAL,	DONTCARE,	DONTCARE,	"physical" },
1553 
1554 { USB_CLASS_IMAGE,	DONTCARE,	DONTCARE,	"image" },
1555 
1556 { USB_CLASS_PRINTER,	DONTCARE,	DONTCARE,	"printer" },
1557 
1558 { USB_CLASS_MASS_STORAGE, USB_SUBCLS_MS_RBC_T10,	DONTCARE, "storage" },
1559 { USB_CLASS_MASS_STORAGE, USB_SUBCLS_MS_SFF8020I,	DONTCARE, "cdrom" },
1560 { USB_CLASS_MASS_STORAGE, USB_SUBCLS_MS_QIC_157,	DONTCARE, "tape" },
1561 { USB_CLASS_MASS_STORAGE, USB_SUBCLS_MS_UFI,		DONTCARE, "floppy" },
1562 { USB_CLASS_MASS_STORAGE, USB_SUBCLS_MS_SFF8070I,	DONTCARE, "storage" },
1563 { USB_CLASS_MASS_STORAGE, USB_SUBCLS_MS_SCSI,		DONTCARE, "storage" },
1564 { USB_CLASS_MASS_STORAGE, DONTCARE,	DONTCARE,	"storage" },
1565 
1566 { USB_CLASS_CDC_DATA,	DONTCARE,	DONTCARE,	"data" },
1567 
1568 { USB_CLASS_SECURITY,	DONTCARE,	DONTCARE,	"security" },
1569 
1570 { USB_CLASS_VIDEO, USB_SUBCLS_VIDEO_CONTROL, DONTCARE,	"video-control" },
1571 { USB_CLASS_VIDEO, USB_SUBCLS_VIDEO_STREAM,  DONTCARE,	"video-stream" },
1572 { USB_CLASS_VIDEO,	DONTCARE,	DONTCARE,	"video" },
1573 
1574 { USB_CLASS_APP,	USB_SUBCLS_APP_FIRMWARE, DONTCARE, "firmware" },
1575 { USB_CLASS_APP,	USB_SUBCLS_APP_IRDA,	DONTCARE, "IrDa" },
1576 { USB_CLASS_APP,	USB_SUBCLS_APP_TEST,	DONTCARE, "test" },
1577 
1578 { USB_CLASS_COMM,	DONTCARE,	DONTCARE,	"communications" },
1579 { USB_CLASS_HUB,	DONTCARE,	DONTCARE,	"hub" },
1580 { USB_CLASS_DIAG,	DONTCARE,	DONTCARE,	"diagnostics" },
1581 { USB_CLASS_MISC,	DONTCARE,	DONTCARE,	"miscellaneous" },
1582 { DONTCARE,		DONTCARE,	DONTCARE,	"device" }
1583 };
1584 
1585 static size_t device_node_name_table_size =
1586 	sizeof (device_node_name_table)/sizeof (struct node_name_entry);
1587 static size_t ia_node_name_table_size =
1588 	sizeof (ia_node_name_table)/sizeof (struct node_name_entry);
1589 static size_t if_node_name_table_size =
1590 	sizeof (if_node_name_table)/sizeof (struct node_name_entry);
1591 static size_t combined_node_name_table_size =
1592 	sizeof (combined_node_name_table)/sizeof (struct node_name_entry);
1593 
1594 
1595 static void
1596 usba_set_node_name(dev_info_t *dip, uint8_t class, uint8_t subclass,
1597     uint8_t protocol, uint_t flag)
1598 {
1599 	int i;
1600 	size_t size;
1601 	node_name_entry_t *node_name_table;
1602 
1603 	switch (flag) {
1604 	/* interface share node names with interface-association */
1605 	case FLAG_INTERFACE_ASSOCIATION_NODE:
1606 		node_name_table = ia_node_name_table;
1607 		size = ia_node_name_table_size;
1608 		break;
1609 	case FLAG_INTERFACE_NODE:
1610 		node_name_table = if_node_name_table;
1611 		size = if_node_name_table_size;
1612 		break;
1613 	case FLAG_DEVICE_NODE:
1614 		node_name_table = device_node_name_table;
1615 		size = device_node_name_table_size;
1616 		break;
1617 	case FLAG_COMBINED_NODE:
1618 		node_name_table = combined_node_name_table;
1619 		size = combined_node_name_table_size;
1620 		break;
1621 	default:
1622 
1623 		return;
1624 	}
1625 
1626 	for (i = 0; i < size; i++) {
1627 		int16_t c = node_name_table[i].class;
1628 		int16_t s = node_name_table[i].subclass;
1629 		int16_t p = node_name_table[i].protocol;
1630 
1631 		if (((c == DONTCARE) || (c == class)) &&
1632 		    ((s == DONTCARE) || (s == subclass)) &&
1633 		    ((p == DONTCARE) || (p == protocol))) {
1634 			char *name = node_name_table[i].name;
1635 
1636 			(void) ndi_devi_set_nodename(dip, name, 0);
1637 			break;
1638 		}
1639 	}
1640 }
1641 
1642 
1643 #ifdef DEBUG
1644 /*
1645  * walk the children of the parent of this devi and compare the
1646  * name and  reg property of each child. If there is a match
1647  * return this node
1648  */
1649 static dev_info_t *
1650 usba_find_existing_node(dev_info_t *odip)
1651 {
1652 	dev_info_t *ndip, *child, *pdip;
1653 	int	*odata, *ndata;
1654 	uint_t	n_odata, n_ndata;
1655 	int	circular;
1656 
1657 	pdip = ddi_get_parent(odip);
1658 	if (ddi_prop_lookup_int_array(DDI_DEV_T_ANY,
1659 	    odip, DDI_PROP_DONTPASS, "reg",
1660 	    &odata, &n_odata) != DDI_SUCCESS) {
1661 		USB_DPRINTF_L2(DPRINT_MASK_HCDI, usba_log_handle,
1662 		    "usba_find_existing_node: "
1663 		    "%s: DDI_NOT_WELL_FORMED", ddi_driver_name(odip));
1664 
1665 		return (NULL);
1666 	}
1667 
1668 	ndi_devi_enter(pdip, &circular);
1669 	ndip = (dev_info_t *)(DEVI(pdip)->devi_child);
1670 	while ((child = ndip) != NULL) {
1671 
1672 		ndip = (dev_info_t *)(DEVI(child)->devi_sibling);
1673 
1674 		if (child == odip) {
1675 			continue;
1676 		}
1677 
1678 		if (strcmp(DEVI(child)->devi_node_name,
1679 		    DEVI(odip)->devi_node_name)) {
1680 			continue;
1681 		}
1682 
1683 		if (ddi_prop_lookup_int_array(DDI_DEV_T_ANY,
1684 		    child, DDI_PROP_DONTPASS, "reg",
1685 		    &ndata, &n_ndata) != DDI_SUCCESS) {
1686 
1687 			USB_DPRINTF_L2(DPRINT_MASK_HCDI, usba_log_handle,
1688 			    "usba_find_existing_node: "
1689 			    "%s DDI_NOT_WELL_FORMED", ddi_driver_name(child));
1690 
1691 		} else if (n_ndata && n_odata && (bcmp(odata, ndata,
1692 		    max(n_odata, n_ndata) * sizeof (int)) == 0)) {
1693 
1694 			USB_DPRINTF_L3(DPRINT_MASK_HCDI, usba_log_handle,
1695 			    "usba_find_existing_node: found %s%d (%p)",
1696 			    ddi_driver_name(child),
1697 			    ddi_get_instance(child), (void *)child);
1698 
1699 			USB_DPRINTF_L3(DPRINT_MASK_HCDI, usba_log_handle,
1700 			    "usba_find_existing_node: "
1701 			    "reg: %x %x %x - %x %x %x",
1702 			    n_odata, odata[0], odata[1],
1703 			    n_ndata, ndata[0], ndata[1]);
1704 
1705 			ddi_prop_free(ndata);
1706 			break;
1707 
1708 		} else {
1709 			ddi_prop_free(ndata);
1710 		}
1711 	}
1712 
1713 	ndi_devi_exit(pdip, circular);
1714 
1715 	ddi_prop_free(odata);
1716 
1717 	return (child);
1718 }
1719 #endif
1720 
1721 /* change all unprintable characters to spaces */
1722 static void
1723 usba_filter_string(char *instr, char *outstr)
1724 {
1725 	while (*instr) {
1726 		if ((*instr >= ' ') && (*instr <= '~')) {
1727 			*outstr = *instr;
1728 		} else {
1729 			*outstr = ' ';
1730 		}
1731 		outstr++;
1732 		instr++;
1733 	}
1734 	*outstr = '\0';
1735 }
1736 
1737 
1738 /*
1739  * lookup ugen binding specified in property in
1740  * hcd.conf files
1741  */
1742 int
1743 usba_get_ugen_binding(dev_info_t *dip)
1744 {
1745 	usba_device_t	*usba_device = usba_get_usba_device(dip);
1746 	usba_hcdi_t	*hcdi =
1747 	    usba_hcdi_get_hcdi(usba_device->usb_root_hub_dip);
1748 
1749 	return (hcdi->hcdi_ugen_default_binding);
1750 }
1751 
1752 
1753 /*
1754  * driver binding support at device level
1755  */
1756 dev_info_t *
1757 usba_ready_device_node(dev_info_t *child_dip)
1758 {
1759 	int		rval, i;
1760 	int		n = 0;
1761 	usba_device_t	*usba_device = usba_get_usba_device(child_dip);
1762 	usb_dev_descr_t	*usb_dev_descr;
1763 	uint_t		n_cfgs;	/* number of configs */
1764 	uint_t		n_ifs;	/* number of interfaces */
1765 	uint_t		port, bus_num;
1766 	size_t		usb_config_length;
1767 	uchar_t 	*usb_config;
1768 	int		reg[1];
1769 	usb_addr_t	address = usb_get_addr(child_dip);
1770 	usb_if_descr_t	if_descr;
1771 	size_t		size;
1772 	int		combined_node = 0;
1773 	int		is_hub;
1774 	char		*devprop_str;
1775 	char		*force_bind = NULL;
1776 	char		*usba_name_buf = NULL;
1777 	char		*usba_name[USBA_MAX_COMPAT_NAMES];
1778 
1779 	usb_config = usb_get_raw_cfg_data(child_dip, &usb_config_length);
1780 
1781 	mutex_enter(&usba_device->usb_mutex);
1782 	mutex_enter(&usba_mutex);
1783 
1784 	USB_DPRINTF_L4(DPRINT_MASK_USBA, usba_log_handle,
1785 	    "usba_ready_device_node: child=0x%p", (void *)child_dip);
1786 
1787 	port = usba_device->usb_port;
1788 	usb_dev_descr = usba_device->usb_dev_descr;
1789 	n_cfgs = usba_device->usb_n_cfgs;
1790 	n_ifs = usba_device->usb_n_ifs;
1791 	bus_num = usba_device->usb_addr;
1792 
1793 	if (address != ROOT_HUB_ADDR) {
1794 		size = usb_parse_if_descr(
1795 		    usb_config,
1796 		    usb_config_length,
1797 		    0,		/* interface index */
1798 		    0,		/* alt interface index */
1799 		    &if_descr,
1800 		    USB_IF_DESCR_SIZE);
1801 
1802 		if (size != USB_IF_DESCR_SIZE) {
1803 			USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
1804 			    "parsing interface: "
1805 			    "size (%lu) != USB_IF_DESCR_SIZE (%d)",
1806 			    size, USB_IF_DESCR_SIZE);
1807 
1808 			mutex_exit(&usba_mutex);
1809 			mutex_exit(&usba_device->usb_mutex);
1810 
1811 			return (child_dip);
1812 		}
1813 	} else {
1814 		/* fake an interface descriptor for the root hub */
1815 		bzero(&if_descr, sizeof (if_descr));
1816 
1817 		if_descr.bInterfaceClass = USB_CLASS_HUB;
1818 	}
1819 
1820 	reg[0] = port;
1821 
1822 	mutex_exit(&usba_mutex);
1823 	mutex_exit(&usba_device->usb_mutex);
1824 
1825 	rval = ndi_prop_update_int_array(
1826 	    DDI_DEV_T_NONE, child_dip, "reg", reg, 1);
1827 
1828 	if (rval != DDI_PROP_SUCCESS) {
1829 		USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
1830 		    "usba_ready_device_node: property update failed");
1831 
1832 		return (child_dip);
1833 	}
1834 
1835 	combined_node = ((n_cfgs == 1) && (n_ifs == 1) &&
1836 	    ((usb_dev_descr->bDeviceClass == USB_CLASS_HUB) ||
1837 	    (usb_dev_descr->bDeviceClass == 0)));
1838 
1839 	is_hub = (if_descr.bInterfaceClass == USB_CLASS_HUB) ||
1840 	    (usb_dev_descr->bDeviceClass == USB_CLASS_HUB);
1841 
1842 	/* set node name */
1843 	if (combined_node) {
1844 		usba_set_node_name(child_dip,
1845 		    if_descr.bInterfaceClass,
1846 		    if_descr.bInterfaceSubClass,
1847 		    if_descr.bInterfaceProtocol,
1848 		    FLAG_COMBINED_NODE);
1849 	} else {
1850 		usba_set_node_name(child_dip,
1851 		    usb_dev_descr->bDeviceClass,
1852 		    usb_dev_descr->bDeviceSubClass,
1853 		    usb_dev_descr->bDeviceProtocol,
1854 		    FLAG_DEVICE_NODE);
1855 	}
1856 
1857 	/*
1858 	 * check force binding rules
1859 	 */
1860 	if ((address != ROOT_HUB_ADDR) && usba_ddivs_usbc &&
1861 	    (address != usba_ddivs_usbc_xaddress) &&
1862 	    (!(usba_ddivs_usbc_xhubs && is_hub))) {
1863 		force_bind = "ddivs_usbc";
1864 		(void) ndi_devi_set_nodename(child_dip, "ddivs_usbc", 0);
1865 
1866 	} else if (usba_device->usb_preferred_driver) {
1867 		force_bind = usba_device->usb_preferred_driver;
1868 
1869 	} else if ((address != ROOT_HUB_ADDR) &&
1870 	    ((usba_ugen_force_binding == USBA_UGEN_DEVICE_BINDING) ||
1871 	    ((usba_ugen_force_binding == USBA_UGEN_INTERFACE_BINDING) &&
1872 	    combined_node)) && (!is_hub)) {
1873 		force_bind = "ugen";
1874 	}
1875 
1876 #ifdef DEBUG
1877 	/*
1878 	 * check whether there is another dip with this name and address
1879 	 * If the dip contains usba_device, it is held by the previous
1880 	 * round of configuration.
1881 	 */
1882 	ASSERT(usba_find_existing_node(child_dip) == NULL);
1883 #endif
1884 
1885 	usba_name_buf = kmem_zalloc(USBA_MAX_COMPAT_NAMES *
1886 	    USBA_MAX_COMPAT_NAME_LEN, KM_SLEEP);
1887 
1888 	for (i = 0; i < USBA_MAX_COMPAT_NAMES; i++) {
1889 		usba_name[i] = usba_name_buf + (i * USBA_MAX_COMPAT_NAME_LEN);
1890 	}
1891 
1892 	if (force_bind) {
1893 		(void) ndi_devi_set_nodename(child_dip, force_bind, 0);
1894 		(void) strncpy(usba_name[n++], force_bind,
1895 		    USBA_MAX_COMPAT_NAME_LEN);
1896 	}
1897 
1898 	/*
1899 	 * If the callback function of specified driver is registered,
1900 	 * it will be called here to check whether to take over the device.
1901 	 */
1902 	if (usb_cap.usba_dev_driver_cb != NULL) {
1903 		char		*dev_drv = NULL;
1904 		usb_dev_str_t	dev_str;
1905 		char		*pathname = kmem_alloc(MAXPATHLEN, KM_SLEEP);
1906 
1907 		dev_str.usb_mfg = usba_device->usb_mfg_str;
1908 		dev_str.usb_product = usba_device->usb_product_str;
1909 		dev_str.usb_serialno = usba_device->usb_serialno_str;
1910 
1911 		(void) ddi_pathname(child_dip, pathname);
1912 
1913 		if ((usb_cap.usba_dev_driver_cb(usb_dev_descr, &dev_str,
1914 		    pathname, bus_num, port, &dev_drv, NULL) == USB_SUCCESS) &&
1915 		    (dev_drv != NULL)) {
1916 			USB_DPRINTF_L3(DPRINT_MASK_USBA, usba_log_handle,
1917 			    "usba_ready_device_node: dev_driver=%s, port =%d,"
1918 			    "bus =%d, path=%s\n\t",
1919 			    dev_drv, port, bus_num, pathname);
1920 
1921 			(void) strncpy(usba_name[n++], dev_drv,
1922 			    USBA_MAX_COMPAT_NAME_LEN);
1923 		}
1924 		kmem_free(pathname, MAXPATHLEN);
1925 	}
1926 
1927 	/* create compatible names */
1928 	if (combined_node) {
1929 
1930 		/* 1. usbVID,PID.REV */
1931 		(void) sprintf(usba_name[n++],
1932 		    "usb%x,%x.%x",
1933 		    usb_dev_descr->idVendor,
1934 		    usb_dev_descr->idProduct,
1935 		    usb_dev_descr->bcdDevice);
1936 
1937 		/* 2. usbVID,PID */
1938 		(void) sprintf(usba_name[n++],
1939 		    "usb%x,%x",
1940 		    usb_dev_descr->idVendor,
1941 		    usb_dev_descr->idProduct);
1942 
1943 		if (usb_dev_descr->bDeviceClass != 0) {
1944 			/* 3. usbVID,classDC.DSC.DPROTO */
1945 			(void) sprintf(usba_name[n++],
1946 			    "usb%x,class%x.%x.%x",
1947 			    usb_dev_descr->idVendor,
1948 			    usb_dev_descr->bDeviceClass,
1949 			    usb_dev_descr->bDeviceSubClass,
1950 			    usb_dev_descr->bDeviceProtocol);
1951 
1952 			/* 4. usbVID,classDC.DSC */
1953 			(void) sprintf(usba_name[n++],
1954 			    "usb%x,class%x.%x",
1955 			    usb_dev_descr->idVendor,
1956 			    usb_dev_descr->bDeviceClass,
1957 			    usb_dev_descr->bDeviceSubClass);
1958 
1959 			/* 5. usbVID,classDC */
1960 			(void) sprintf(usba_name[n++],
1961 			    "usb%x,class%x",
1962 			    usb_dev_descr->idVendor,
1963 			    usb_dev_descr->bDeviceClass);
1964 
1965 			/* 6. usb,classDC.DSC.DPROTO */
1966 			(void) sprintf(usba_name[n++],
1967 			    "usb,class%x.%x.%x",
1968 			    usb_dev_descr->bDeviceClass,
1969 			    usb_dev_descr->bDeviceSubClass,
1970 			    usb_dev_descr->bDeviceProtocol);
1971 
1972 			/* 7. usb,classDC.DSC */
1973 			(void) sprintf(usba_name[n++],
1974 			    "usb,class%x.%x",
1975 			    usb_dev_descr->bDeviceClass,
1976 			    usb_dev_descr->bDeviceSubClass);
1977 
1978 			/* 8. usb,classDC */
1979 			(void) sprintf(usba_name[n++],
1980 			    "usb,class%x",
1981 			    usb_dev_descr->bDeviceClass);
1982 		}
1983 
1984 		if (if_descr.bInterfaceClass != 0) {
1985 			/* 9. usbifVID,classIC.ISC.IPROTO */
1986 			(void) sprintf(usba_name[n++],
1987 			    "usbif%x,class%x.%x.%x",
1988 			    usb_dev_descr->idVendor,
1989 			    if_descr.bInterfaceClass,
1990 			    if_descr.bInterfaceSubClass,
1991 			    if_descr.bInterfaceProtocol);
1992 
1993 			/* 10. usbifVID,classIC.ISC */
1994 			(void) sprintf(usba_name[n++],
1995 			    "usbif%x,class%x.%x",
1996 			    usb_dev_descr->idVendor,
1997 			    if_descr.bInterfaceClass,
1998 			    if_descr.bInterfaceSubClass);
1999 
2000 			/* 11. usbifVID,classIC */
2001 			(void) sprintf(usba_name[n++],
2002 			    "usbif%x,class%x",
2003 			    usb_dev_descr->idVendor,
2004 			    if_descr.bInterfaceClass);
2005 
2006 			/* 12. usbif,classIC.ISC.IPROTO */
2007 			(void) sprintf(usba_name[n++],
2008 			    "usbif,class%x.%x.%x",
2009 			    if_descr.bInterfaceClass,
2010 			    if_descr.bInterfaceSubClass,
2011 			    if_descr.bInterfaceProtocol);
2012 
2013 			/* 13. usbif,classIC.ISC */
2014 			(void) sprintf(usba_name[n++],
2015 			    "usbif,class%x.%x",
2016 			    if_descr.bInterfaceClass,
2017 			    if_descr.bInterfaceSubClass);
2018 
2019 			/* 14. usbif,classIC */
2020 			(void) sprintf(usba_name[n++],
2021 			    "usbif,class%x",
2022 			    if_descr.bInterfaceClass);
2023 		}
2024 
2025 		/* 15. ugen or usb_mid */
2026 		if (usba_get_ugen_binding(child_dip) ==
2027 		    USBA_UGEN_DEVICE_BINDING) {
2028 			(void) sprintf(usba_name[n++], "ugen");
2029 		} else {
2030 			(void) sprintf(usba_name[n++], "usb,device");
2031 		}
2032 
2033 	} else {
2034 		if (n_cfgs > 1) {
2035 			/* 1. usbVID,PID.REV.configCN */
2036 			(void) sprintf(usba_name[n++],
2037 			    "usb%x,%x.%x.config%x",
2038 			    usb_dev_descr->idVendor,
2039 			    usb_dev_descr->idProduct,
2040 			    usb_dev_descr->bcdDevice,
2041 			    usba_device->usb_cfg_value);
2042 		}
2043 
2044 		/* 2. usbVID,PID.REV */
2045 		(void) sprintf(usba_name[n++],
2046 		    "usb%x,%x.%x",
2047 		    usb_dev_descr->idVendor,
2048 		    usb_dev_descr->idProduct,
2049 		    usb_dev_descr->bcdDevice);
2050 
2051 		/* 3. usbVID,PID.configCN */
2052 		if (n_cfgs > 1) {
2053 			(void) sprintf(usba_name[n++],
2054 			    "usb%x,%x.%x",
2055 			    usb_dev_descr->idVendor,
2056 			    usb_dev_descr->idProduct,
2057 			    usba_device->usb_cfg_value);
2058 		}
2059 
2060 		/* 4. usbVID,PID */
2061 		(void) sprintf(usba_name[n++],
2062 		    "usb%x,%x",
2063 		    usb_dev_descr->idVendor,
2064 		    usb_dev_descr->idProduct);
2065 
2066 		if (usb_dev_descr->bDeviceClass != 0) {
2067 			/* 5. usbVID,classDC.DSC.DPROTO */
2068 			(void) sprintf(usba_name[n++],
2069 			    "usb%x,class%x.%x.%x",
2070 			    usb_dev_descr->idVendor,
2071 			    usb_dev_descr->bDeviceClass,
2072 			    usb_dev_descr->bDeviceSubClass,
2073 			    usb_dev_descr->bDeviceProtocol);
2074 
2075 			/* 6. usbVID,classDC.DSC */
2076 			(void) sprintf(usba_name[n++],
2077 			    "usb%x.class%x.%x",
2078 			    usb_dev_descr->idVendor,
2079 			    usb_dev_descr->bDeviceClass,
2080 			    usb_dev_descr->bDeviceSubClass);
2081 
2082 			/* 7. usbVID,classDC */
2083 			(void) sprintf(usba_name[n++],
2084 			    "usb%x.class%x",
2085 			    usb_dev_descr->idVendor,
2086 			    usb_dev_descr->bDeviceClass);
2087 
2088 			/* 8. usb,classDC.DSC.DPROTO */
2089 			(void) sprintf(usba_name[n++],
2090 			    "usb,class%x.%x.%x",
2091 			    usb_dev_descr->bDeviceClass,
2092 			    usb_dev_descr->bDeviceSubClass,
2093 			    usb_dev_descr->bDeviceProtocol);
2094 
2095 			/* 9. usb,classDC.DSC */
2096 			(void) sprintf(usba_name[n++],
2097 			    "usb,class%x.%x",
2098 			    usb_dev_descr->bDeviceClass,
2099 			    usb_dev_descr->bDeviceSubClass);
2100 
2101 			/* 10. usb,classDC */
2102 			(void) sprintf(usba_name[n++],
2103 			    "usb,class%x",
2104 			    usb_dev_descr->bDeviceClass);
2105 		}
2106 
2107 		if (usba_get_ugen_binding(child_dip) ==
2108 		    USBA_UGEN_DEVICE_BINDING) {
2109 			/* 11. ugen */
2110 			(void) sprintf(usba_name[n++], "ugen");
2111 		} else {
2112 			/* 11. usb,device */
2113 			(void) sprintf(usba_name[n++], "usb,device");
2114 		}
2115 	}
2116 
2117 	for (i = 0; i < n; i += 2) {
2118 		USB_DPRINTF_L3(DPRINT_MASK_USBA, usba_log_handle,
2119 		    "compatible name:\t%s\t%s", usba_name[i],
2120 		    (((i+1) < n)? usba_name[i+1] : ""));
2121 	}
2122 
2123 	rval = ndi_prop_update_string_array(DDI_DEV_T_NONE, child_dip,
2124 	    "compatible", (char **)usba_name, n);
2125 
2126 	kmem_free(usba_name_buf, USBA_MAX_COMPAT_NAMES *
2127 	    USBA_MAX_COMPAT_NAME_LEN);
2128 
2129 	if (rval != DDI_PROP_SUCCESS) {
2130 
2131 		USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
2132 		    "usba_ready_device_node: property update failed");
2133 
2134 		return (child_dip);
2135 	}
2136 
2137 	/* update the address property */
2138 	rval = ndi_prop_update_int(DDI_DEV_T_NONE, child_dip,
2139 	    "assigned-address", usba_device->usb_addr);
2140 	if (rval != DDI_PROP_SUCCESS) {
2141 		USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
2142 		    "usba_ready_device_node: address update failed");
2143 	}
2144 
2145 	/* update the usb device properties (PSARC/2000/454) */
2146 	rval = ndi_prop_update_int(DDI_DEV_T_NONE, child_dip,
2147 	    "usb-vendor-id", usb_dev_descr->idVendor);
2148 	if (rval != DDI_PROP_SUCCESS) {
2149 		USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
2150 		    "usba_ready_device_node: usb-vendor-id update failed");
2151 	}
2152 
2153 	rval = ndi_prop_update_int(DDI_DEV_T_NONE, child_dip,
2154 	    "usb-product-id", usb_dev_descr->idProduct);
2155 	if (rval != DDI_PROP_SUCCESS) {
2156 		USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
2157 		    "usba_ready_device_node: usb-product-id update failed");
2158 	}
2159 
2160 	rval = ndi_prop_update_int(DDI_DEV_T_NONE, child_dip,
2161 	    "usb-revision-id", usb_dev_descr->bcdDevice);
2162 	if (rval != DDI_PROP_SUCCESS) {
2163 		USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
2164 		    "usba_ready_device_node: usb-revision-id update failed");
2165 	}
2166 
2167 	rval = ndi_prop_update_int(DDI_DEV_T_NONE, child_dip,
2168 	    "usb-num-configs", usb_dev_descr->bNumConfigurations);
2169 	if (rval != DDI_PROP_SUCCESS) {
2170 		USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
2171 		    "usba_ready_device_node: usb-num-configs update failed");
2172 	}
2173 
2174 	rval = ndi_prop_update_int(DDI_DEV_T_NONE, child_dip,
2175 	    "usb-release", usb_dev_descr->bcdUSB);
2176 	if (rval != DDI_PROP_SUCCESS) {
2177 		USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
2178 		    "usba_ready_device_node: usb-release update failed");
2179 	}
2180 
2181 	rval = ndi_prop_update_byte_array(DDI_DEV_T_NONE, child_dip,
2182 	    "usb-dev-descriptor", (uchar_t *)usb_dev_descr,
2183 	    sizeof (usb_dev_descr_t));
2184 	if (rval != DDI_PROP_SUCCESS) {
2185 		USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
2186 		    "usba_ready_device_node: usb-descriptor update failed");
2187 	}
2188 
2189 	rval = ndi_prop_update_byte_array(DDI_DEV_T_NONE, child_dip,
2190 	    "usb-raw-cfg-descriptors", usb_config, usb_config_length);
2191 	if (rval != DDI_PROP_SUCCESS) {
2192 		USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
2193 		    "usba_ready_device_node: usb-raw-cfg-descriptors update "
2194 		    "failed");
2195 	}
2196 
2197 	devprop_str = kmem_zalloc(USB_MAXSTRINGLEN, KM_SLEEP);
2198 
2199 	if (usba_device->usb_serialno_str) {
2200 		usba_filter_string(usba_device->usb_serialno_str, devprop_str);
2201 		rval = ndi_prop_update_string(DDI_DEV_T_NONE, child_dip,
2202 		    "usb-serialno", devprop_str);
2203 		if (rval != DDI_PROP_SUCCESS) {
2204 			USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
2205 			    "usba_ready_device_node: "
2206 			    "usb-serialno update failed");
2207 		}
2208 	}
2209 
2210 	if (usba_device->usb_mfg_str) {
2211 		usba_filter_string(usba_device->usb_mfg_str, devprop_str);
2212 		rval = ndi_prop_update_string(DDI_DEV_T_NONE, child_dip,
2213 		    "usb-vendor-name", devprop_str);
2214 		if (rval != DDI_PROP_SUCCESS) {
2215 			USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
2216 			    "usba_ready_device_node: "
2217 			    "usb-vendor-name update failed");
2218 		}
2219 	}
2220 
2221 	if (usba_device->usb_product_str) {
2222 		usba_filter_string(usba_device->usb_product_str, devprop_str);
2223 		rval = ndi_prop_update_string(DDI_DEV_T_NONE, child_dip,
2224 		    "usb-product-name", devprop_str);
2225 		if (rval != DDI_PROP_SUCCESS) {
2226 			USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
2227 			    "usba_ready_device_node: "
2228 			    "usb-product-name update failed");
2229 		}
2230 	}
2231 
2232 	kmem_free(devprop_str, USB_MAXSTRINGLEN);
2233 
2234 	if (!combined_node) {
2235 		/* update the configuration property */
2236 		rval = ndi_prop_update_int(DDI_DEV_T_NONE, child_dip,
2237 		    "configuration#", usba_device->usb_cfg_value);
2238 		if (rval != DDI_PROP_SUCCESS) {
2239 			USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
2240 			    "usba_ready_device_node: "
2241 			    "config prop update failed");
2242 		}
2243 	}
2244 
2245 	if (usba_device->usb_port_status == USBA_LOW_SPEED_DEV) {
2246 		/* create boolean property */
2247 		rval = ndi_prop_create_boolean(DDI_DEV_T_NONE, child_dip,
2248 		    "low-speed");
2249 		if (rval != DDI_PROP_SUCCESS) {
2250 			USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
2251 			    "usba_ready_device_node: "
2252 			    "low speed prop update failed");
2253 		}
2254 	}
2255 
2256 	if (usba_device->usb_port_status == USBA_HIGH_SPEED_DEV) {
2257 		/* create boolean property */
2258 		rval = ndi_prop_create_boolean(DDI_DEV_T_NONE, child_dip,
2259 		    "high-speed");
2260 		if (rval != DDI_PROP_SUCCESS) {
2261 			USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
2262 			    "usba_ready_device_node: "
2263 			    "high speed prop update failed");
2264 		}
2265 	}
2266 
2267 	USB_DPRINTF_L4(DPRINT_MASK_USBA, usba_log_handle,
2268 	    "%s%d at port %d: %s, dip=0x%p",
2269 	    ddi_node_name(ddi_get_parent(child_dip)),
2270 	    ddi_get_instance(ddi_get_parent(child_dip)),
2271 	    port, ddi_node_name(child_dip), (void *)child_dip);
2272 
2273 	usba_set_usba_device(child_dip, usba_device);
2274 
2275 	ASSERT(!mutex_owned(&(usba_get_usba_device(child_dip)->usb_mutex)));
2276 
2277 	return (child_dip);
2278 }
2279 
2280 
2281 /*
2282  * driver binding at interface association level. the first arg is the parent
2283  * dip. if_count returns amount of interfaces which are associated within
2284  * this interface-association that starts from first_if.
2285  */
2286 /*ARGSUSED*/
2287 dev_info_t *
2288 usba_ready_interface_association_node(dev_info_t	*dip,
2289 					uint_t		first_if,
2290 					uint_t		*if_count)
2291 {
2292 	dev_info_t		*child_dip = NULL;
2293 	usba_device_t		*child_ud = usba_get_usba_device(dip);
2294 	usb_dev_descr_t		*usb_dev_descr;
2295 	size_t			usb_cfg_length;
2296 	uchar_t			*usb_cfg;
2297 	usb_ia_descr_t		ia_descr;
2298 	int			i, n, rval;
2299 	int			reg[2];
2300 	size_t			size;
2301 	usb_port_status_t	port_status;
2302 	char			*force_bind = NULL;
2303 	char			*usba_name_buf = NULL;
2304 	char			*usba_name[USBA_MAX_COMPAT_NAMES];
2305 
2306 	usb_cfg = usb_get_raw_cfg_data(dip, &usb_cfg_length);
2307 
2308 	mutex_enter(&child_ud->usb_mutex);
2309 
2310 	usb_dev_descr = child_ud->usb_dev_descr;
2311 
2312 	/*
2313 	 * for each interface association, determine all compatible names
2314 	 */
2315 	USB_DPRINTF_L3(DPRINT_MASK_USBA, usba_log_handle,
2316 	    "usba_ready_ia_node: "
2317 	    "port %d, interface = %d, port_status = %x",
2318 	    child_ud->usb_port, first_if, child_ud->usb_port_status);
2319 
2320 	/* Parse the interface descriptor */
2321 	size = usb_parse_ia_descr(
2322 	    usb_cfg,
2323 	    usb_cfg_length,
2324 	    first_if,	/* interface index */
2325 	    &ia_descr,
2326 	    USB_IA_DESCR_SIZE);
2327 
2328 	*if_count = 1;
2329 	if (size != USB_IA_DESCR_SIZE) {
2330 		USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
2331 		    "parsing ia: size (%lu) != USB_IA_DESCR_SIZE (%d)",
2332 		    size, USB_IA_DESCR_SIZE);
2333 		mutex_exit(&child_ud->usb_mutex);
2334 
2335 		return (NULL);
2336 	}
2337 
2338 	port_status = child_ud->usb_port_status;
2339 
2340 	/* create reg property */
2341 	reg[0] = first_if;
2342 	reg[1] = child_ud->usb_cfg_value;
2343 
2344 	mutex_exit(&child_ud->usb_mutex);
2345 
2346 	/* clone this dip */
2347 	rval =	usba_create_child_devi(dip,
2348 	    "interface-association",
2349 	    NULL,		/* usba_hcdi ops */
2350 	    NULL,		/* root hub dip */
2351 	    port_status,	/* port status */
2352 	    child_ud,	/* share this usba_device */
2353 	    &child_dip);
2354 
2355 	if (rval != USB_SUCCESS) {
2356 
2357 		goto fail;
2358 	}
2359 
2360 	rval = ndi_prop_update_int_array(
2361 	    DDI_DEV_T_NONE, child_dip, "reg", reg, 2);
2362 
2363 	if (rval != DDI_PROP_SUCCESS) {
2364 
2365 		goto fail;
2366 	}
2367 
2368 	usba_set_node_name(child_dip, ia_descr.bFunctionClass,
2369 	    ia_descr.bFunctionSubClass, ia_descr.bFunctionProtocol,
2370 	    FLAG_INTERFACE_ASSOCIATION_NODE);
2371 
2372 	/* check force binding */
2373 	if (usba_ugen_force_binding ==
2374 	    USBA_UGEN_INTERFACE_ASSOCIATION_BINDING) {
2375 		force_bind = "ugen";
2376 	}
2377 
2378 	/*
2379 	 * check whether there is another dip with this name and address
2380 	 */
2381 	ASSERT(usba_find_existing_node(child_dip) == NULL);
2382 
2383 	usba_name_buf = kmem_zalloc(USBA_MAX_COMPAT_NAMES *
2384 	    USBA_MAX_COMPAT_NAME_LEN, KM_SLEEP);
2385 
2386 	for (i = 0; i < USBA_MAX_COMPAT_NAMES; i++) {
2387 		usba_name[i] = usba_name_buf + (i * USBA_MAX_COMPAT_NAME_LEN);
2388 	}
2389 
2390 	n = 0;
2391 
2392 	if (force_bind) {
2393 		(void) ndi_devi_set_nodename(child_dip, force_bind, 0);
2394 		(void) strncpy(usba_name[n++], force_bind,
2395 		    USBA_MAX_COMPAT_NAME_LEN);
2396 	}
2397 
2398 	/* 1) usbiaVID,PID.REV.configCN.FN */
2399 	(void) sprintf(usba_name[n++],
2400 	    "usbia%x,%x.%x.config%x.%x",
2401 	    usb_dev_descr->idVendor,
2402 	    usb_dev_descr->idProduct,
2403 	    usb_dev_descr->bcdDevice,
2404 	    child_ud->usb_cfg_value,
2405 	    first_if);
2406 
2407 	/* 2) usbiaVID,PID.configCN.FN */
2408 	(void) sprintf(usba_name[n++],
2409 	    "usbia%x,%x.config%x.%x",
2410 	    usb_dev_descr->idVendor,
2411 	    usb_dev_descr->idProduct,
2412 	    child_ud->usb_cfg_value,
2413 	    first_if);
2414 
2415 
2416 	if (ia_descr.bFunctionClass) {
2417 		/* 3) usbiaVID,classFC.FSC.FPROTO */
2418 		(void) sprintf(usba_name[n++],
2419 		    "usbia%x,class%x.%x.%x",
2420 		    usb_dev_descr->idVendor,
2421 		    ia_descr.bFunctionClass,
2422 		    ia_descr.bFunctionSubClass,
2423 		    ia_descr.bFunctionProtocol);
2424 
2425 		/* 4) usbiaVID,classFC.FSC */
2426 		(void) sprintf(usba_name[n++],
2427 		    "usbia%x,class%x.%x",
2428 		    usb_dev_descr->idVendor,
2429 		    ia_descr.bFunctionClass,
2430 		    ia_descr.bFunctionSubClass);
2431 
2432 		/* 5) usbiaVID,classFC */
2433 		(void) sprintf(usba_name[n++],
2434 		    "usbia%x,class%x",
2435 		    usb_dev_descr->idVendor,
2436 		    ia_descr.bFunctionClass);
2437 
2438 		/* 6) usbia,classFC.FSC.FPROTO */
2439 		(void) sprintf(usba_name[n++],
2440 		    "usbia,class%x.%x.%x",
2441 		    ia_descr.bFunctionClass,
2442 		    ia_descr.bFunctionSubClass,
2443 		    ia_descr.bFunctionProtocol);
2444 
2445 		/* 7) usbia,classFC.FSC */
2446 		(void) sprintf(usba_name[n++],
2447 		    "usbia,class%x.%x",
2448 		    ia_descr.bFunctionClass,
2449 		    ia_descr.bFunctionSubClass);
2450 
2451 		/* 8) usbia,classFC */
2452 		(void) sprintf(usba_name[n++],
2453 		    "usbia,class%x",
2454 		    ia_descr.bFunctionClass);
2455 	}
2456 
2457 	if (usba_get_ugen_binding(child_dip) ==
2458 	    USBA_UGEN_INTERFACE_ASSOCIATION_BINDING) {
2459 		/* 9) ugen */
2460 		(void) sprintf(usba_name[n++], "ugen");
2461 	} else {
2462 
2463 		(void) sprintf(usba_name[n++], "usb,ia");
2464 	}
2465 
2466 	for (i = 0; i < n; i += 2) {
2467 		USB_DPRINTF_L3(DPRINT_MASK_USBA, usba_log_handle,
2468 		    "compatible name:\t%s\t%s", usba_name[i],
2469 		    (((i+1) < n)? usba_name[i+1] : ""));
2470 	}
2471 
2472 	/* create compatible property */
2473 	rval = ndi_prop_update_string_array(DDI_DEV_T_NONE, child_dip,
2474 	    "compatible", (char **)usba_name, n);
2475 
2476 	kmem_free(usba_name_buf, USBA_MAX_COMPAT_NAMES *
2477 	    USBA_MAX_COMPAT_NAME_LEN);
2478 
2479 	if (rval != DDI_PROP_SUCCESS) {
2480 
2481 		goto fail;
2482 	}
2483 
2484 	/* update the address property */
2485 	rval = ndi_prop_update_int(DDI_DEV_T_NONE, child_dip,
2486 	    "assigned-address", child_ud->usb_addr);
2487 	if (rval != DDI_PROP_SUCCESS) {
2488 		USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
2489 		    "usba_ready_interface_node: address update failed");
2490 	}
2491 
2492 	/* create property with first interface number */
2493 	rval = ndi_prop_update_int(DDI_DEV_T_NONE, child_dip,
2494 	    "interface", ia_descr.bFirstInterface);
2495 
2496 	if (rval != DDI_PROP_SUCCESS) {
2497 
2498 		goto fail;
2499 	}
2500 
2501 	/* create property with the count of interfaces in this ia */
2502 	rval = ndi_prop_update_int(DDI_DEV_T_NONE, child_dip,
2503 	    "interface-count", ia_descr.bInterfaceCount);
2504 
2505 	if (rval != DDI_PROP_SUCCESS) {
2506 
2507 		goto fail;
2508 	}
2509 
2510 	USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
2511 	    "%s%d port %d: %s, dip = 0x%p",
2512 	    ddi_node_name(ddi_get_parent(dip)),
2513 	    ddi_get_instance(ddi_get_parent(dip)),
2514 	    child_ud->usb_port, ddi_node_name(child_dip), (void *)child_dip);
2515 
2516 	*if_count = ia_descr.bInterfaceCount;
2517 	usba_set_usba_device(child_dip, child_ud);
2518 	ASSERT(!mutex_owned(&(usba_get_usba_device(child_dip)->usb_mutex)));
2519 
2520 	return (child_dip);
2521 
2522 fail:
2523 	(void) usba_destroy_child_devi(child_dip, NDI_DEVI_REMOVE);
2524 
2525 	return (NULL);
2526 }
2527 
2528 
2529 /*
2530  * driver binding at interface level, the first arg will be the
2531  * the parent dip
2532  */
2533 /*ARGSUSED*/
2534 dev_info_t *
2535 usba_ready_interface_node(dev_info_t *dip, uint_t intf)
2536 {
2537 	dev_info_t		*child_dip = NULL;
2538 	usba_device_t		*child_ud = usba_get_usba_device(dip);
2539 	usb_dev_descr_t	*usb_dev_descr;
2540 	size_t			usb_cfg_length;
2541 	uchar_t 		*usb_cfg;
2542 	usb_if_descr_t	if_descr;
2543 	int			i, n, rval;
2544 	int			reg[2];
2545 	size_t			size;
2546 	usb_port_status_t	port_status;
2547 	char			*force_bind = NULL;
2548 	char			*usba_name_buf = NULL;
2549 	char			*usba_name[USBA_MAX_COMPAT_NAMES];
2550 
2551 	usb_cfg = usb_get_raw_cfg_data(dip, &usb_cfg_length);
2552 
2553 	mutex_enter(&child_ud->usb_mutex);
2554 
2555 	usb_dev_descr = child_ud->usb_dev_descr;
2556 
2557 	/*
2558 	 * for each interface, determine all compatible names
2559 	 */
2560 	USB_DPRINTF_L3(DPRINT_MASK_USBA, usba_log_handle,
2561 	    "usba_ready_interface_node: "
2562 	    "port %d, interface = %d port status = %x",
2563 	    child_ud->usb_port, intf, child_ud->usb_port_status);
2564 
2565 	/* Parse the interface descriptor */
2566 	size = usb_parse_if_descr(
2567 	    usb_cfg,
2568 	    usb_cfg_length,
2569 	    intf,		/* interface index */
2570 	    0,		/* alt interface index */
2571 	    &if_descr,
2572 	    USB_IF_DESCR_SIZE);
2573 
2574 	if (size != USB_IF_DESCR_SIZE) {
2575 		USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
2576 		    "parsing interface: size (%lu) != USB_IF_DESCR_SIZE (%d)",
2577 		    size, USB_IF_DESCR_SIZE);
2578 		mutex_exit(&child_ud->usb_mutex);
2579 
2580 		return (NULL);
2581 	}
2582 
2583 	port_status = child_ud->usb_port_status;
2584 
2585 	/* create reg property */
2586 	reg[0] = intf;
2587 	reg[1] = child_ud->usb_cfg_value;
2588 
2589 	mutex_exit(&child_ud->usb_mutex);
2590 
2591 	/* clone this dip */
2592 	rval =	usba_create_child_devi(dip,
2593 	    "interface",
2594 	    NULL,		/* usba_hcdi ops */
2595 	    NULL,		/* root hub dip */
2596 	    port_status,	/* port status */
2597 	    child_ud,	/* share this usba_device */
2598 	    &child_dip);
2599 
2600 	if (rval != USB_SUCCESS) {
2601 
2602 		goto fail;
2603 	}
2604 
2605 	rval = ndi_prop_update_int_array(
2606 	    DDI_DEV_T_NONE, child_dip, "reg", reg, 2);
2607 
2608 	if (rval != DDI_PROP_SUCCESS) {
2609 
2610 		goto fail;
2611 	}
2612 
2613 	usba_set_node_name(child_dip, if_descr.bInterfaceClass,
2614 	    if_descr.bInterfaceSubClass, if_descr.bInterfaceProtocol,
2615 	    FLAG_INTERFACE_NODE);
2616 
2617 	/* check force binding */
2618 	if (usba_ugen_force_binding == USBA_UGEN_INTERFACE_BINDING) {
2619 		force_bind = "ugen";
2620 	}
2621 
2622 	/*
2623 	 * check whether there is another dip with this name and address
2624 	 */
2625 	ASSERT(usba_find_existing_node(child_dip) == NULL);
2626 
2627 	usba_name_buf = kmem_zalloc(USBA_MAX_COMPAT_NAMES *
2628 	    USBA_MAX_COMPAT_NAME_LEN, KM_SLEEP);
2629 
2630 	for (i = 0; i < USBA_MAX_COMPAT_NAMES; i++) {
2631 		usba_name[i] = usba_name_buf + (i * USBA_MAX_COMPAT_NAME_LEN);
2632 	}
2633 
2634 	n = 0;
2635 
2636 	if (force_bind) {
2637 		(void) ndi_devi_set_nodename(child_dip, force_bind, 0);
2638 		(void) strncpy(usba_name[n++], force_bind,
2639 		    USBA_MAX_COMPAT_NAME_LEN);
2640 	}
2641 
2642 	/* 1) usbifVID,PID.REV.configCN.IN */
2643 	(void) sprintf(usba_name[n++],
2644 	    "usbif%x,%x.%x.config%x.%x",
2645 	    usb_dev_descr->idVendor,
2646 	    usb_dev_descr->idProduct,
2647 	    usb_dev_descr->bcdDevice,
2648 	    child_ud->usb_cfg_value,
2649 	    intf);
2650 
2651 	/* 2) usbifVID,PID.configCN.IN */
2652 	(void) sprintf(usba_name[n++],
2653 	    "usbif%x,%x.config%x.%x",
2654 	    usb_dev_descr->idVendor,
2655 	    usb_dev_descr->idProduct,
2656 	    child_ud->usb_cfg_value,
2657 	    intf);
2658 
2659 
2660 	if (if_descr.bInterfaceClass) {
2661 		/* 3) usbifVID,classIC.ISC.IPROTO */
2662 		(void) sprintf(usba_name[n++],
2663 		    "usbif%x,class%x.%x.%x",
2664 		    usb_dev_descr->idVendor,
2665 		    if_descr.bInterfaceClass,
2666 		    if_descr.bInterfaceSubClass,
2667 		    if_descr.bInterfaceProtocol);
2668 
2669 		/* 4) usbifVID,classIC.ISC */
2670 		(void) sprintf(usba_name[n++],
2671 		    "usbif%x,class%x.%x",
2672 		    usb_dev_descr->idVendor,
2673 		    if_descr.bInterfaceClass,
2674 		    if_descr.bInterfaceSubClass);
2675 
2676 		/* 5) usbifVID,classIC */
2677 		(void) sprintf(usba_name[n++],
2678 		    "usbif%x,class%x",
2679 		    usb_dev_descr->idVendor,
2680 		    if_descr.bInterfaceClass);
2681 
2682 		/* 6) usbif,classIC.ISC.IPROTO */
2683 		(void) sprintf(usba_name[n++],
2684 		    "usbif,class%x.%x.%x",
2685 		    if_descr.bInterfaceClass,
2686 		    if_descr.bInterfaceSubClass,
2687 		    if_descr.bInterfaceProtocol);
2688 
2689 		/* 7) usbif,classIC.ISC */
2690 		(void) sprintf(usba_name[n++],
2691 		    "usbif,class%x.%x",
2692 		    if_descr.bInterfaceClass,
2693 		    if_descr.bInterfaceSubClass);
2694 
2695 		/* 8) usbif,classIC */
2696 		(void) sprintf(usba_name[n++],
2697 		    "usbif,class%x",
2698 		    if_descr.bInterfaceClass);
2699 	}
2700 
2701 	if (usba_get_ugen_binding(child_dip) ==
2702 	    USBA_UGEN_INTERFACE_BINDING) {
2703 		/* 9) ugen */
2704 		(void) sprintf(usba_name[n++], "ugen");
2705 	}
2706 
2707 	for (i = 0; i < n; i += 2) {
2708 		USB_DPRINTF_L3(DPRINT_MASK_USBA, usba_log_handle,
2709 		    "compatible name:\t%s\t%s", usba_name[i],
2710 		    (((i+1) < n)? usba_name[i+1] : ""));
2711 	}
2712 
2713 	/* create compatible property */
2714 	rval = ndi_prop_update_string_array(DDI_DEV_T_NONE, child_dip,
2715 	    "compatible", (char **)usba_name, n);
2716 
2717 	kmem_free(usba_name_buf, USBA_MAX_COMPAT_NAMES *
2718 	    USBA_MAX_COMPAT_NAME_LEN);
2719 
2720 	if (rval != DDI_PROP_SUCCESS) {
2721 
2722 		goto fail;
2723 	}
2724 
2725 	/* update the address property */
2726 	rval = ndi_prop_update_int(DDI_DEV_T_NONE, child_dip,
2727 	    "assigned-address", child_ud->usb_addr);
2728 	if (rval != DDI_PROP_SUCCESS) {
2729 		USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
2730 		    "usba_ready_interface_node: address update failed");
2731 	}
2732 
2733 	/* create property with if number */
2734 	rval = ndi_prop_update_int(DDI_DEV_T_NONE, child_dip,
2735 	    "interface", intf);
2736 
2737 	if (rval != DDI_PROP_SUCCESS) {
2738 
2739 		goto fail;
2740 	}
2741 
2742 	USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
2743 	    "%s%d port %d: %s, dip = 0x%p",
2744 	    ddi_node_name(ddi_get_parent(dip)),
2745 	    ddi_get_instance(ddi_get_parent(dip)),
2746 	    child_ud->usb_port, ddi_node_name(child_dip), (void *)child_dip);
2747 
2748 	usba_set_usba_device(child_dip, child_ud);
2749 	ASSERT(!mutex_owned(&(usba_get_usba_device(child_dip)->usb_mutex)));
2750 
2751 	return (child_dip);
2752 
2753 fail:
2754 	(void) usba_destroy_child_devi(child_dip, NDI_DEVI_REMOVE);
2755 
2756 	return (NULL);
2757 }
2758 
2759 
2760 /*
2761  * retrieve string descriptors for manufacturer, vendor and serial
2762  * number
2763  */
2764 void
2765 usba_get_dev_string_descrs(dev_info_t *dip, usba_device_t *ud)
2766 {
2767 	char	*tmpbuf, *str;
2768 	int	l;
2769 	usb_dev_descr_t *usb_dev_descr = ud->usb_dev_descr;
2770 
2771 
2772 	USB_DPRINTF_L4(DPRINT_MASK_USBA, usba_log_handle,
2773 	    "usba_get_usb_string_descr: m=%d, p=%d, s=%d",
2774 	    usb_dev_descr->iManufacturer,
2775 	    usb_dev_descr->iProduct,
2776 	    usb_dev_descr->iSerialNumber);
2777 
2778 	tmpbuf = kmem_zalloc(USB_MAXSTRINGLEN, KM_SLEEP);
2779 
2780 	/* fetch manufacturer string */
2781 	if ((ud->usb_mfg_str == NULL) && usb_dev_descr->iManufacturer &&
2782 	    (usb_get_string_descr(dip, USB_LANG_ID,
2783 	    usb_dev_descr->iManufacturer, tmpbuf, USB_MAXSTRINGLEN) ==
2784 	    USB_SUCCESS)) {
2785 
2786 		l = strlen(tmpbuf);
2787 		if (l > 0) {
2788 			str = kmem_zalloc(l + 1, KM_SLEEP);
2789 			mutex_enter(&ud->usb_mutex);
2790 			ud->usb_mfg_str = str;
2791 			(void) strcpy(ud->usb_mfg_str, tmpbuf);
2792 			mutex_exit(&ud->usb_mutex);
2793 		}
2794 	}
2795 
2796 	/* fetch product string */
2797 	if ((ud->usb_product_str == NULL) && usb_dev_descr->iProduct &&
2798 	    (usb_get_string_descr(dip, USB_LANG_ID, usb_dev_descr->iProduct,
2799 	    tmpbuf, USB_MAXSTRINGLEN) ==
2800 	    USB_SUCCESS)) {
2801 
2802 		l = strlen(tmpbuf);
2803 		if (l > 0) {
2804 			str = kmem_zalloc(l + 1, KM_SLEEP);
2805 			mutex_enter(&ud->usb_mutex);
2806 			ud->usb_product_str = str;
2807 			(void) strcpy(ud->usb_product_str, tmpbuf);
2808 			mutex_exit(&ud->usb_mutex);
2809 		}
2810 	}
2811 
2812 	/* fetch device serial number string */
2813 	if ((ud->usb_serialno_str == NULL) && usb_dev_descr->iSerialNumber &&
2814 	    (usb_get_string_descr(dip, USB_LANG_ID,
2815 	    usb_dev_descr->iSerialNumber, tmpbuf, USB_MAXSTRINGLEN) ==
2816 	    USB_SUCCESS)) {
2817 
2818 		l = strlen(tmpbuf);
2819 		if (l > 0) {
2820 			str = kmem_zalloc(l + 1, KM_SLEEP);
2821 			mutex_enter(&ud->usb_mutex);
2822 			ud->usb_serialno_str = str;
2823 			(void) strcpy(ud->usb_serialno_str, tmpbuf);
2824 			mutex_exit(&ud->usb_mutex);
2825 		}
2826 	}
2827 
2828 	kmem_free(tmpbuf, USB_MAXSTRINGLEN);
2829 }
2830 
2831 
2832 /*
2833  * usba_str_startcmp:
2834  *	Return the number of characters duplicated from the beginning of the
2835  *	string.  Return -1 if a complete duplicate.
2836  *
2837  * Arguments:
2838  *	Two strings to compare.
2839  */
2840 static int usba_str_startcmp(char *first, char *second)
2841 {
2842 	int num_same_chars = 0;
2843 	while (*first == *second++) {
2844 		if (*first++ == '\0') {
2845 			return (-1);
2846 		}
2847 		num_same_chars++;
2848 	}
2849 
2850 	return (num_same_chars);
2851 }
2852 
2853 
2854 /*
2855  * usba_get_mfg_prod_sn_str:
2856  *	Return a string containing mfg, product, serial number strings.
2857  *	Remove duplicates if some strings are the same.
2858  *
2859  * Arguments:
2860  *	dip	- pointer to dev info
2861  *	buffer	- Where string is returned
2862  *	buflen	- Length of buffer
2863  *
2864  * Returns:
2865  *	Same as second arg.
2866  */
2867 char *
2868 usba_get_mfg_prod_sn_str(
2869     dev_info_t	*dip,
2870     char	*buffer,
2871     int		buflen)
2872 {
2873 	usba_device_t *usba_device = usba_get_usba_device(dip);
2874 	int return_len = 0;
2875 	int len = 0;
2876 	int duplen;
2877 
2878 	buffer[0] = '\0';
2879 	buffer[buflen-1] = '\0';
2880 
2881 	if ((usba_device->usb_mfg_str) &&
2882 	    ((len = strlen(usba_device->usb_mfg_str)) != 0)) {
2883 		(void) strncpy(buffer, usba_device->usb_mfg_str, buflen - 1);
2884 		return_len = min(buflen - 1, len);
2885 	}
2886 
2887 	/* Product string exists to append. */
2888 	if ((usba_device->usb_product_str) &&
2889 	    ((len = strlen(usba_device->usb_product_str)) != 0)) {
2890 
2891 		/* Append only parts of string that don't match mfg string. */
2892 		duplen = usba_str_startcmp(buffer,
2893 		    usba_device->usb_product_str);
2894 
2895 		if (duplen != -1) {		/* Not a complete match. */
2896 			if (return_len > 0) {
2897 				buffer[return_len++] = ' ';
2898 			}
2899 
2900 			/* Skip over the dup part of the concat'ed string. */
2901 			len -= duplen;
2902 			(void) strncpy(&buffer[return_len],
2903 			    &usba_device->usb_product_str[duplen],
2904 			    buflen - return_len - 1);
2905 			return_len = min(buflen - 1, return_len + len);
2906 		}
2907 	}
2908 
2909 	if ((usba_device->usb_serialno_str) &&
2910 	    ((len = strlen(usba_device->usb_serialno_str)) != 0)) {
2911 		if (return_len > 0) {
2912 			buffer[return_len++] = ' ';
2913 		}
2914 		(void) strncpy(&buffer[return_len],
2915 		    usba_device->usb_serialno_str,
2916 		    buflen - return_len - 1);
2917 	}
2918 
2919 	return (buffer);
2920 }
2921 
2922 
2923 /*
2924  * USB enumeration statistic functions
2925  */
2926 
2927 /*
2928  * Increments the hotplug statistics based on flags.
2929  */
2930 void
2931 usba_update_hotplug_stats(dev_info_t *dip, usb_flags_t flags)
2932 {
2933 	usba_device_t	*usba_device = usba_get_usba_device(dip);
2934 	usba_hcdi_t	*hcdi =
2935 	    usba_hcdi_get_hcdi(usba_device->usb_root_hub_dip);
2936 
2937 	mutex_enter(&hcdi->hcdi_mutex);
2938 	if (flags & USBA_TOTAL_HOTPLUG_SUCCESS) {
2939 		hcdi->hcdi_total_hotplug_success++;
2940 		HCDI_HOTPLUG_STATS_DATA(hcdi)->
2941 		    hcdi_hotplug_total_success.value.ui64++;
2942 	}
2943 	if (flags & USBA_HOTPLUG_SUCCESS) {
2944 		hcdi->hcdi_hotplug_success++;
2945 		HCDI_HOTPLUG_STATS_DATA(hcdi)->
2946 		    hcdi_hotplug_success.value.ui64++;
2947 	}
2948 	if (flags & USBA_TOTAL_HOTPLUG_FAILURE) {
2949 		hcdi->hcdi_total_hotplug_failure++;
2950 		HCDI_HOTPLUG_STATS_DATA(hcdi)->
2951 		    hcdi_hotplug_total_failure.value.ui64++;
2952 	}
2953 	if (flags & USBA_HOTPLUG_FAILURE) {
2954 		hcdi->hcdi_hotplug_failure++;
2955 		HCDI_HOTPLUG_STATS_DATA(hcdi)->
2956 		    hcdi_hotplug_failure.value.ui64++;
2957 	}
2958 	mutex_exit(&hcdi->hcdi_mutex);
2959 }
2960 
2961 
2962 /*
2963  * Retrieve the current enumeration statistics
2964  */
2965 void
2966 usba_get_hotplug_stats(dev_info_t *dip, ulong_t *total_success,
2967     ulong_t *success, ulong_t *total_failure, ulong_t *failure,
2968     uchar_t *device_count)
2969 {
2970 	usba_device_t	*usba_device = usba_get_usba_device(dip);
2971 	usba_hcdi_t	*hcdi =
2972 	    usba_hcdi_get_hcdi(usba_device->usb_root_hub_dip);
2973 
2974 	mutex_enter(&hcdi->hcdi_mutex);
2975 	*total_success = hcdi->hcdi_total_hotplug_success;
2976 	*success = hcdi->hcdi_hotplug_success;
2977 	*total_failure = hcdi->hcdi_total_hotplug_failure;
2978 	*failure = hcdi->hcdi_hotplug_failure;
2979 	*device_count = hcdi->hcdi_device_count;
2980 	mutex_exit(&hcdi->hcdi_mutex);
2981 }
2982 
2983 
2984 /*
2985  * Reset the resetable hotplug stats
2986  */
2987 void
2988 usba_reset_hotplug_stats(dev_info_t *dip)
2989 {
2990 	usba_device_t	*usba_device = usba_get_usba_device(dip);
2991 	usba_hcdi_t	*hcdi =
2992 	    usba_hcdi_get_hcdi(usba_device->usb_root_hub_dip);
2993 	hcdi_hotplug_stats_t *hsp;
2994 
2995 	mutex_enter(&hcdi->hcdi_mutex);
2996 	hcdi->hcdi_hotplug_success = 0;
2997 	hcdi->hcdi_hotplug_failure = 0;
2998 
2999 	hsp = HCDI_HOTPLUG_STATS_DATA(hcdi);
3000 	hsp->hcdi_hotplug_success.value.ui64 = 0;
3001 	hsp->hcdi_hotplug_failure.value.ui64 = 0;
3002 	mutex_exit(&hcdi->hcdi_mutex);
3003 }
3004 
3005 
3006 /*
3007  * usba_bind_driver():
3008  *	This function calls ndi_devi_bind_driver() which tries to
3009  *	bind a driver to the device.  If the driver binding fails
3010  *	we get an rval of NDI_UNBOUD and report an error to the
3011  *	syslog that the driver failed binding.
3012  *	If rval is something other than NDI_UNBOUND we report an
3013  *	error to the console.
3014  *
3015  *	This function returns USB_SUCCESS if no errors were
3016  *	encountered while binding.
3017  */
3018 int
3019 usba_bind_driver(dev_info_t *dip)
3020 {
3021 	int	rval;
3022 	char	*name;
3023 	uint8_t if_num = usba_get_ifno(dip);
3024 
3025 	USB_DPRINTF_L4(DPRINT_MASK_USBA, usba_log_handle,
3026 	    "usba_bind_driver: dip = 0x%p, if_num = 0x%x", (void *)dip, if_num);
3027 
3028 	name = kmem_zalloc(MAXNAMELEN, KM_SLEEP);
3029 
3030 	/* bind device to the driver */
3031 	if ((rval = ndi_devi_bind_driver(dip, 0)) != NDI_SUCCESS) {
3032 		/* if we fail to bind report an error */
3033 		(void) usba_get_mfg_prod_sn_str(dip, name, MAXNAMELEN);
3034 		if (name[0] != '\0') {
3035 			if (!usb_owns_device(dip)) {
3036 				USB_DPRINTF_L1(DPRINT_MASK_USBA,
3037 				    usba_log_handle,
3038 				    "no driver found for "
3039 				    "interface %d (nodename: '%s') of %s",
3040 				    if_num, ddi_node_name(dip), name);
3041 			} else {
3042 				USB_DPRINTF_L1(DPRINT_MASK_USBA,
3043 				    usba_log_handle,
3044 				    "no driver found for device %s", name);
3045 			}
3046 		} else {
3047 			(void) ddi_pathname(dip, name);
3048 			USB_DPRINTF_L1(DPRINT_MASK_USBA,
3049 			    usba_log_handle,
3050 			    "no driver found for device %s", name);
3051 		}
3052 
3053 		kmem_free(name, MAXNAMELEN);
3054 
3055 		return (USB_FAILURE);
3056 	}
3057 	kmem_free(name, MAXNAMELEN);
3058 
3059 	return ((rval == NDI_SUCCESS) ? USB_SUCCESS : USB_FAILURE);
3060 }
3061 
3062 
3063 /*
3064  * usba_get_hc_dma_attr:
3065  *	function returning dma attributes of the HCD
3066  *
3067  * Arguments:
3068  *	dip	- pointer to devinfo of the client
3069  *
3070  * Return Values:
3071  *	hcdi_dma_attr
3072  */
3073 ddi_dma_attr_t *
3074 usba_get_hc_dma_attr(dev_info_t *dip)
3075 {
3076 	usba_device_t *usba_device = usba_get_usba_device(dip);
3077 	usba_hcdi_t *hcdi = usba_hcdi_get_hcdi(usba_device->usb_root_hub_dip);
3078 
3079 	return (hcdi->hcdi_dma_attr);
3080 }
3081 
3082 
3083 /*
3084  * usba_check_for_leaks:
3085  *	check usba_device structure for leaks
3086  *
3087  * Arguments:
3088  *	usba_device	- usba_device structure pointer
3089  */
3090 void
3091 usba_check_for_leaks(usba_device_t *usba_device)
3092 {
3093 	int i, ph_open_cnt, req_wrp_leaks, iface;
3094 	int leaks = 0;
3095 
3096 	USB_DPRINTF_L4(DPRINT_MASK_USBA, usba_log_handle,
3097 	    "usba_check_for_leaks: %s%d usba_device=0x%p",
3098 	    ddi_driver_name(usba_device->usb_dip),
3099 	    ddi_get_instance(usba_device->usb_dip), (void *)usba_device);
3100 
3101 	/*
3102 	 * default pipe is still open
3103 	 * all other pipes should be closed
3104 	 */
3105 	for (ph_open_cnt = 0, i = 1; i < USBA_N_ENDPOINTS; i++) {
3106 		usba_ph_impl_t *ph_impl =
3107 		    &usba_device->usb_ph_list[i];
3108 		if (ph_impl->usba_ph_data) {
3109 			USB_DPRINTF_L2(DPRINT_MASK_USBA,
3110 			    usba_log_handle,
3111 			    "%s%d: leaking pipehandle=0x%p (0x%p) ep_addr=0x%x",
3112 			    ddi_driver_name(ph_impl->usba_ph_data->p_dip),
3113 			    ddi_get_instance(ph_impl->usba_ph_data->p_dip),
3114 			    (void *)ph_impl,
3115 			    (void *)ph_impl->usba_ph_data,
3116 			    ph_impl->usba_ph_ep.bEndpointAddress);
3117 			ph_open_cnt++;
3118 			leaks++;
3119 #ifndef DEBUG
3120 			usb_pipe_close(ph_impl->usba_ph_data->p_dip,
3121 			    (usb_pipe_handle_t)ph_impl, USB_FLAGS_SLEEP,
3122 			    NULL, NULL);
3123 #endif
3124 		}
3125 	}
3126 	req_wrp_leaks =  usba_list_entry_leaks(&usba_device->
3127 	    usb_allocated, "request wrappers");
3128 
3129 	ASSERT(ph_open_cnt == 0);
3130 	ASSERT(req_wrp_leaks == 0);
3131 
3132 	if (req_wrp_leaks) {
3133 		usba_list_entry_t *entry;
3134 
3135 		while ((entry = usba_rm_first_from_list(
3136 		    &usba_device->usb_allocated)) != NULL) {
3137 			usba_req_wrapper_t *wrp;
3138 
3139 			mutex_enter(&entry->list_mutex);
3140 			wrp = (usba_req_wrapper_t *)entry->private;
3141 			mutex_exit(&entry->list_mutex);
3142 			leaks++;
3143 
3144 			USB_DPRINTF_L2(DPRINT_MASK_USBA,
3145 			    usba_log_handle,
3146 			    "%s%d: leaking request 0x%p",
3147 			    ddi_driver_name(wrp->wr_dip),
3148 			    ddi_get_instance(wrp->wr_dip),
3149 			    (void *)wrp->wr_req);
3150 
3151 			/*
3152 			 * put it back, usba_req_wrapper_free
3153 			 * expects it on the list
3154 			 */
3155 			usba_add_to_list(&usba_device->usb_allocated,
3156 			    &wrp->wr_allocated_list);
3157 
3158 			usba_req_wrapper_free(wrp);
3159 		}
3160 	}
3161 
3162 	mutex_enter(&usba_device->usb_mutex);
3163 	for (iface = 0; iface < usba_device->usb_n_ifs; iface++) {
3164 		USB_DPRINTF_L3(DPRINT_MASK_USBA, usba_log_handle,
3165 		    "usba_check_for_leaks: if=%d client_flags=0x%x",
3166 		    iface, usba_device->usb_client_flags[iface]);
3167 
3168 		if (usba_device->usb_client_flags[iface] &
3169 		    USBA_CLIENT_FLAG_DEV_DATA) {
3170 			usb_client_dev_data_list_t *entry =
3171 			    usba_device->usb_client_dev_data_list.cddl_next;
3172 			usb_client_dev_data_list_t *next;
3173 			usb_client_dev_data_t *dev_data;
3174 
3175 			while (entry) {
3176 				dev_info_t *dip = entry->cddl_dip;
3177 				next = entry->cddl_next;
3178 				dev_data = entry->cddl_dev_data;
3179 
3180 
3181 				if (!i_ddi_devi_attached(dip)) {
3182 					USB_DPRINTF_L2(DPRINT_MASK_USBA,
3183 					    usba_log_handle,
3184 					    "%s%d: leaking dev_data 0x%p",
3185 					    ddi_driver_name(dip),
3186 					    ddi_get_instance(dip),
3187 					    (void *)dev_data);
3188 
3189 					leaks++;
3190 
3191 					mutex_exit(&usba_device->usb_mutex);
3192 					usb_free_dev_data(dip, dev_data);
3193 					mutex_enter(&usba_device->usb_mutex);
3194 				}
3195 
3196 				entry = next;
3197 			}
3198 		}
3199 		if (usba_device->usb_client_flags[iface] &
3200 		    USBA_CLIENT_FLAG_ATTACH) {
3201 			dev_info_t *dip = usba_device->
3202 			    usb_client_attach_list[iface].dip;
3203 
3204 			USB_DPRINTF_L2(DPRINT_MASK_USBA,
3205 			    usba_log_handle,
3206 			    "%s%d: did no usb_client_detach",
3207 			    ddi_driver_name(dip), ddi_get_instance(dip));
3208 			leaks++;
3209 
3210 			mutex_exit(&usba_device->usb_mutex);
3211 			usb_client_detach(dip, NULL);
3212 			mutex_enter(&usba_device->usb_mutex);
3213 
3214 			usba_device->
3215 			    usb_client_attach_list[iface].dip = NULL;
3216 
3217 			usba_device->usb_client_flags[iface] &=
3218 			    ~USBA_CLIENT_FLAG_ATTACH;
3219 
3220 		}
3221 		if (usba_device->usb_client_flags[iface] &
3222 		    USBA_CLIENT_FLAG_EV_CBS) {
3223 			dev_info_t *dip =
3224 			    usba_device->usb_client_ev_cb_list[iface].
3225 			    dip;
3226 			usb_event_t *ev_data =
3227 			    usba_device->usb_client_ev_cb_list[iface].
3228 			    ev_data;
3229 
3230 			USB_DPRINTF_L2(DPRINT_MASK_USBA,
3231 			    usba_log_handle,
3232 			    "%s%d: did no usb_unregister_event_cbs",
3233 			    ddi_driver_name(dip), ddi_get_instance(dip));
3234 			leaks++;
3235 
3236 			mutex_exit(&usba_device->usb_mutex);
3237 			usb_unregister_event_cbs(dip, ev_data);
3238 			mutex_enter(&usba_device->usb_mutex);
3239 
3240 			usba_device->usb_client_ev_cb_list[iface].
3241 			    dip = NULL;
3242 			usba_device->usb_client_ev_cb_list[iface].
3243 			    ev_data = NULL;
3244 			usba_device->usb_client_flags[iface] &=
3245 			    ~USBA_CLIENT_FLAG_EV_CBS;
3246 		}
3247 	}
3248 	mutex_exit(&usba_device->usb_mutex);
3249 
3250 	if (leaks) {
3251 		USB_DPRINTF_L2(DPRINT_MASK_USBA, usba_log_handle,
3252 		    "all %d leaks fixed", leaks);
3253 	}
3254 }
3255