xref: /illumos-gate/usr/src/uts/common/io/usb/clients/printer/usbprn.c (revision dcafa541382944b24abd3a40c357b47e04f314e2)
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  * Copyright 2008 Sun Microsystems, Inc.  All rights reserved.
22  * Use is subject to license terms.
23  */
24 
25 #pragma ident	"%Z%%M%	%I%	%E% SMI"
26 
27 /*
28  * Printer Class Driver for USB
29  *
30  * This driver supports devices that adhere to the USB Printer Class
31  * specification 1.0.
32  *
33  * NOTE: This driver is not DDI compliant in that it uses undocumented
34  * functions for logging (USB_DPRINTF_L*, usb_alloc_log_hdl, usb_free_log_hdl),
35  * and serialization (usb_serialize_access, usb_release_access,
36  * usb_init_serialization, usb_fini_serialization)
37  *
38  * Undocumented functions may go away in a future Solaris OS release.
39  *
40  * Please see the DDK for sample code of these functions, and for the usbskel
41  * skeleton template driver which contains scaled-down versions of these
42  * functions written in a DDI-compliant way.
43  */
44 
45 #if defined(lint) && !defined(DEBUG)
46 #define	DEBUG
47 #endif
48 #ifdef __lock_lint
49 #define	_MULTI_DATAMODEL
50 #endif
51 
52 #define	USBDRV_MAJOR_VER	2
53 #define	USBDRV_MINOR_VER	0
54 
55 #include <sys/usb/usba.h>
56 #include <sys/usb/usba/usba_ugen.h>
57 #include <sys/bpp_io.h>
58 #include <sys/ecppsys.h>
59 #include <sys/prnio.h>
60 #include <sys/errno.h>
61 #include <sys/usb/clients/printer/usb_printer.h>
62 #include <sys/usb/clients/printer/usbprn.h>
63 
64 /* Debugging support */
65 uint_t	usbprn_errmask		= (uint_t)PRINT_MASK_ALL;
66 uint_t	usbprn_errlevel 	= USB_LOG_L4;
67 uint_t	usbprn_instance_debug	= (uint_t)-1;
68 
69 /* local variables */
70 static uint_t usbprn_ifcap =
71 	PRN_HOTPLUG | PRN_1284_DEVID | PRN_1284_STATUS | PRN_TIMEOUTS;
72 
73 /*
74  * Function Prototypes
75  */
76 static int	usbprn_attach(dev_info_t *, ddi_attach_cmd_t);
77 static int	usbprn_detach(dev_info_t *, ddi_detach_cmd_t);
78 static int	usbprn_info(dev_info_t *, ddi_info_cmd_t, void *, void **);
79 static void	usbprn_cleanup(dev_info_t *, usbprn_state_t *);
80 
81 static int	usbprn_get_descriptors(usbprn_state_t *);
82 static int	usbprn_get_device_id(usbprn_state_t *);
83 static int	usbprn_get_port_status(usbprn_state_t *);
84 
85 static int	usbprn_open(dev_t *, int, int, cred_t *);
86 static int	usbprn_close(dev_t, int, int, cred_t *);
87 static int	usbprn_open_usb_pipes(usbprn_state_t *);
88 static void	usbprn_close_usb_pipes(usbprn_state_t *);
89 static int	usbprn_write(dev_t, struct uio *, cred_t *);
90 static int	usbprn_read(dev_t, struct uio *, cred_t *);
91 static int	usbprn_poll(dev_t, short, int, short *, struct pollhead **);
92 
93 static int	usbprn_ioctl(dev_t, int, intptr_t, int, cred_t *, int *);
94 static void	usbprn_minphys(struct buf *);
95 static int	usbprn_strategy(struct buf *);
96 static int	usbprn_setparms(usbprn_state_t *, intptr_t arg, int);
97 static int	usbprn_getparms(usbprn_state_t *, intptr_t, int);
98 static void	usbprn_geterr(usbprn_state_t *, intptr_t, int);
99 static int	usbprn_testio(usbprn_state_t  *, int);
100 static int	usbprn_ioctl_get_status(usbprn_state_t *);
101 static int	usbprn_prnio_get_status(usbprn_state_t *, intptr_t, int);
102 static int	usbprn_prnio_get_1284_status(usbprn_state_t *, intptr_t, int);
103 static int	usbprn_prnio_get_ifcap(usbprn_state_t *, intptr_t, int);
104 static int	usbprn_prnio_set_ifcap(usbprn_state_t *, intptr_t, int);
105 static int	usbprn_prnio_get_ifinfo(usbprn_state_t *, intptr_t, int);
106 static int	usbprn_prnio_get_1284_devid(usbprn_state_t *, intptr_t, int);
107 static int	usbprn_prnio_get_timeouts(usbprn_state_t *, intptr_t, int);
108 static int	usbprn_prnio_set_timeouts(usbprn_state_t *, intptr_t, int);
109 
110 static void	usbprn_send_async_bulk_data(usbprn_state_t *);
111 
112 static void	usbprn_bulk_xfer_cb(usb_pipe_handle_t, usb_bulk_req_t *);
113 static void	usbprn_bulk_xfer_exc_cb(usb_pipe_handle_t,
114 		    usb_bulk_req_t *);
115 
116 static void	usbprn_biodone(usbprn_state_t *, int, int);
117 static char	usbprn_error_state(uchar_t);
118 static void	usbprn_print_long(usbprn_state_t *, char *, int);
119 
120 /* event handling */
121 static	void	usbprn_restore_device_state(dev_info_t *, usbprn_state_t *);
122 static	int	usbprn_disconnect_event_cb(dev_info_t *);
123 static	int	usbprn_reconnect_event_cb(dev_info_t *);
124 static	int	usbprn_cpr_suspend(dev_info_t *);
125 static	void	usbprn_cpr_resume(dev_info_t *);
126 
127 static usb_event_t usbprn_events = {
128 	usbprn_disconnect_event_cb,
129 	usbprn_reconnect_event_cb,
130 	NULL, NULL
131 };
132 
133 /* PM handling */
134 static	void	usbprn_create_pm_components(dev_info_t *, usbprn_state_t *);
135 static	int	usbprn_power(dev_info_t *, int comp, int level);
136 static	int	usbprn_pwrlvl0(usbprn_state_t *);
137 static	int	usbprn_pwrlvl1(usbprn_state_t *);
138 static	int	usbprn_pwrlvl2(usbprn_state_t *);
139 static	int	usbprn_pwrlvl3(usbprn_state_t *);
140 static	void	usbprn_pm_busy_component(usbprn_state_t *);
141 static	void	usbprn_pm_idle_component(usbprn_state_t *);
142 
143 /* module loading stuff */
144 struct cb_ops usbprn_cb_ops = {
145 	usbprn_open,		/* open  */
146 	usbprn_close,		/* close */
147 	nulldev,		/* strategy */
148 	nulldev,		/* print */
149 	nulldev,		/* dump */
150 	usbprn_read,		/* read */
151 	usbprn_write,		/* write */
152 	usbprn_ioctl,		/* ioctl */
153 	nulldev,		/* devmap */
154 	nulldev,		/* mmap */
155 	nulldev,		/* segmap */
156 	usbprn_poll,		/* poll */
157 	ddi_prop_op,		/* cb_prop_op */
158 	NULL,			/* streamtab  */
159 	D_64BIT | D_MP
160 };
161 
162 static struct dev_ops usbprn_ops = {
163 	DEVO_REV,		/* devo_rev, */
164 	0,			/* refcnt  */
165 	usbprn_info,		/* info */
166 	nulldev,		/* identify */
167 	nulldev,		/* probe */
168 	usbprn_attach,		/* attach */
169 	usbprn_detach,		/* detach */
170 	nodev,			/* reset */
171 	&usbprn_cb_ops,		/* driver operations */
172 	NULL,			/* bus operations */
173 	usbprn_power		/* power */
174 };
175 
176 static struct modldrv usbprnmodldrv =	{
177 	&mod_driverops,
178 	"USB printer client driver %I%",
179 	&usbprn_ops
180 };
181 
182 static struct modlinkage modlinkage = {
183 	MODREV_1,
184 	&usbprnmodldrv,
185 	NULL,
186 };
187 
188 /* local variables */
189 
190 /* soft state structures */
191 #define	USBPRN_INITIAL_SOFT_SPACE	1
192 static void *usbprn_statep;
193 
194 static int usbprn_max_xfer_size = USBPRN_MAX_XFER_SIZE;
195 
196 /* prnio support */
197 static const char usbprn_prnio_ifinfo[] = PRN_USB;
198 
199 
200 int
201 _init(void)
202 {
203 	int rval;
204 
205 	if ((rval = ddi_soft_state_init(&usbprn_statep,
206 	    sizeof (usbprn_state_t), USBPRN_INITIAL_SOFT_SPACE)) != 0) {
207 
208 		return (rval);
209 	}
210 
211 	if ((rval = mod_install(&modlinkage)) != 0) {
212 		ddi_soft_state_fini(&usbprn_statep);
213 	}
214 
215 	return (rval);
216 }
217 
218 
219 int
220 _fini(void)
221 {
222 	int rval;
223 
224 	if ((rval = mod_remove(&modlinkage)) != 0) {
225 
226 		return (rval);
227 	}
228 
229 	ddi_soft_state_fini(&usbprn_statep);
230 
231 	return (rval);
232 }
233 
234 
235 int
236 _info(struct modinfo *modinfop)
237 {
238 	return (mod_info(&modlinkage, modinfop));
239 }
240 
241 
242 /*
243  * usbprn_info:
244  *	Get minor number, soft state structure, etc.
245  */
246 /*ARGSUSED*/
247 static int
248 usbprn_info(dev_info_t *dip, ddi_info_cmd_t infocmd,
249 			void *arg, void **result)
250 {
251 	usbprn_state_t	*usbprnp;
252 	int		error = DDI_FAILURE;
253 	minor_t		minor = getminor((dev_t)arg);
254 	int		instance = USBPRN_MINOR_TO_INSTANCE(minor);
255 
256 	switch (infocmd) {
257 	case DDI_INFO_DEVT2DEVINFO:
258 		if ((usbprnp = ddi_get_soft_state(usbprn_statep,
259 		    instance)) != NULL) {
260 			*result = usbprnp->usbprn_dip;
261 			if (*result != NULL) {
262 				error = DDI_SUCCESS;
263 			}
264 		} else {
265 			*result = NULL;
266 		}
267 
268 		break;
269 	case DDI_INFO_DEVT2INSTANCE:
270 		*result = (void *)(uintptr_t)instance;
271 		error = DDI_SUCCESS;
272 
273 		break;
274 	default:
275 
276 		break;
277 	}
278 
279 	return (error);
280 }
281 
282 
283 /*
284  * usbprn_attach:
285  *	Attach driver
286  *	Get the descriptor information
287  *	Get the device id
288  *	Reset the device
289  *	Get the port status
290  */
291 static int
292 usbprn_attach(dev_info_t *dip, ddi_attach_cmd_t cmd)
293 {
294 	int			instance = ddi_get_instance(dip);
295 	usbprn_state_t		*usbprnp = NULL;
296 	size_t			sz;
297 	usb_ugen_info_t 	usb_ugen_info;
298 
299 	switch (cmd) {
300 	case DDI_ATTACH:
301 
302 		break;
303 	case DDI_RESUME:
304 		usbprn_cpr_resume(dip);
305 
306 		return (DDI_SUCCESS);
307 	default:
308 
309 		return (DDI_FAILURE);
310 	}
311 
312 	if (ddi_soft_state_zalloc(usbprn_statep, instance) == DDI_SUCCESS) {
313 		usbprnp = ddi_get_soft_state(usbprn_statep, instance);
314 	}
315 	if (usbprnp == NULL)  {
316 
317 		return (DDI_FAILURE);
318 	}
319 
320 	usbprnp->usbprn_instance = instance;
321 	usbprnp->usbprn_dip	= dip;
322 	usbprnp->usbprn_log_handle = usb_alloc_log_hdl(dip,
323 	    "prn", &usbprn_errlevel,
324 	    &usbprn_errmask, &usbprn_instance_debug, 0);
325 
326 	USB_DPRINTF_L4(PRINT_MASK_ATTA, usbprnp->usbprn_log_handle,
327 	    "usbprn_attach: cmd=%x", cmd);
328 
329 	if (usb_client_attach(dip, USBDRV_VERSION, 0) != USB_SUCCESS) {
330 		USB_DPRINTF_L2(PRINT_MASK_ATTA, usbprnp->usbprn_log_handle,
331 		    "usb_client_attach failed");
332 
333 		goto fail;
334 	}
335 	if (usb_get_dev_data(dip, &usbprnp->usbprn_dev_data,
336 	    USB_PARSE_LVL_IF, 0) != USB_SUCCESS) {
337 		USB_DPRINTF_L2(PRINT_MASK_ATTA, usbprnp->usbprn_log_handle,
338 		    "usb_get_dev_data failed");
339 
340 		goto fail;
341 	}
342 
343 	/* Initialize locks and conditional variables */
344 	mutex_init(&usbprnp->usbprn_mutex, NULL, MUTEX_DRIVER,
345 	    usbprnp->usbprn_dev_data->dev_iblock_cookie);
346 	usbprnp->usbprn_write_acc = usb_init_serialization(dip,
347 	    USB_INIT_SER_CHECK_SAME_THREAD);
348 	usbprnp->usbprn_ser_acc = usb_init_serialization(dip,
349 	    USB_INIT_SER_CHECK_SAME_THREAD);
350 	usbprnp->usbprn_dev_acc = usb_init_serialization(dip, 0);
351 
352 	usbprnp->usbprn_flags |= USBPRN_LOCKS_INIT_DONE;
353 
354 	/* Obtain all the relevant descriptors */
355 	if (usbprn_get_descriptors(usbprnp) != USB_SUCCESS) {
356 		USB_DPRINTF_L2(PRINT_MASK_ATTA, usbprnp->usbprn_log_handle,
357 		    "usb get descriptors failed");
358 
359 		goto fail;
360 	}
361 
362 	usbprnp->usbprn_def_ph = usbprnp->usbprn_dev_data->dev_default_ph;
363 
364 	/* Obtain the device id */
365 	(void) usbprn_get_device_id(usbprnp);
366 
367 	/* Get the port status */
368 	if (usbprn_get_port_status(usbprnp) != USB_SUCCESS) {
369 		/* some printers fail on the first */
370 		if (usbprn_get_port_status(usbprnp) != USB_SUCCESS) {
371 			USB_DPRINTF_L2(PRINT_MASK_ATTA,
372 			    usbprnp->usbprn_log_handle,
373 			    "usb get port status failed");
374 
375 			goto fail;
376 		}
377 	}
378 
379 	USB_DPRINTF_L3(PRINT_MASK_ATTA, usbprnp->usbprn_log_handle,
380 	    "usbprn_attach: printer status=0x%x", usbprnp->usbprn_last_status);
381 
382 	if ((usbprnp->usbprn_last_status & USB_PRINTER_PORT_NO_ERROR) == 0) {
383 		USB_DPRINTF_L2(PRINT_MASK_ATTA, usbprnp->usbprn_log_handle,
384 		    "usbprn_attach: error occurred with the printer");
385 	}
386 
387 	/*
388 	 * Create minor node based on information from the
389 	 * descriptors
390 	 */
391 	if ((ddi_create_minor_node(dip, "printer", S_IFCHR,
392 	    instance << USBPRN_MINOR_INSTANCE_SHIFT,
393 	    DDI_NT_PRINTER, 0)) != DDI_SUCCESS) {
394 		USB_DPRINTF_L2(PRINT_MASK_ATTA, usbprnp->usbprn_log_handle,
395 		    "usbprn_attach: cannot create minor node");
396 
397 		goto fail;
398 	}
399 
400 	usbprnp->usbprn_setparms.write_timeout = USBPRN_XFER_TIMEOUT;
401 	usbprnp->usbprn_setparms.mode =  ECPP_CENTRONICS;
402 	usbprnp->usbprn_dev_state = USB_DEV_ONLINE;
403 
404 	if (usb_pipe_get_max_bulk_transfer_size(usbprnp->usbprn_dip, &sz)) {
405 
406 		goto fail;
407 	}
408 
409 	usbprnp->usbprn_max_bulk_xfer_size = sz;
410 
411 	USB_DPRINTF_L4(PRINT_MASK_OPEN, usbprnp->usbprn_log_handle,
412 	    "usbprn_attach: xfer_size=0x%lx", sz);
413 
414 	/* enable PM */
415 	usbprn_create_pm_components(dip, usbprnp);
416 
417 	/* Register for events */
418 	if (usb_register_event_cbs(dip, &usbprn_events, 0) != USB_SUCCESS) {
419 		USB_DPRINTF_L2(PRINT_MASK_ATTA, usbprnp->usbprn_log_handle,
420 		    "usbprn_attach: usb_register_event_cbs failed");
421 
422 		goto fail;
423 	}
424 
425 	usb_free_dev_data(dip, usbprnp->usbprn_dev_data);
426 	usbprnp->usbprn_dev_data = NULL;
427 
428 	if (usb_owns_device(dip)) {
429 		/* get a ugen handle */
430 		bzero(&usb_ugen_info, sizeof (usb_ugen_info));
431 
432 		usb_ugen_info.usb_ugen_flags = 0;
433 		usb_ugen_info.usb_ugen_minor_node_ugen_bits_mask =
434 		    (dev_t)USBPRN_MINOR_UGEN_BITS_MASK;
435 		usb_ugen_info.usb_ugen_minor_node_instance_mask =
436 		    (dev_t)~USBPRN_MINOR_UGEN_BITS_MASK;
437 		usbprnp->usbprn_ugen_hdl =
438 		    usb_ugen_get_hdl(dip, &usb_ugen_info);
439 
440 		if (usb_ugen_attach(usbprnp->usbprn_ugen_hdl, cmd) !=
441 		    USB_SUCCESS) {
442 			USB_DPRINTF_L2(PRINT_MASK_ATTA,
443 			    usbprnp->usbprn_log_handle,
444 			    "usb_ugen_attach failed");
445 
446 			usb_ugen_release_hdl(usbprnp->usbprn_ugen_hdl);
447 			usbprnp->usbprn_ugen_hdl = NULL;
448 		}
449 	}
450 
451 	/* Report device */
452 	ddi_report_dev(dip);
453 
454 	USB_DPRINTF_L4(PRINT_MASK_ATTA, usbprnp->usbprn_log_handle,
455 	    "usbprn_attach: done");
456 
457 	return (DDI_SUCCESS);
458 
459 fail:
460 	if (usbprnp) {
461 		usbprn_cleanup(dip, usbprnp);
462 	}
463 
464 	return (DDI_FAILURE);
465 }
466 
467 
468 /*
469  * usbprn_detach:
470  *	detach or suspend driver instance
471  */
472 static int
473 usbprn_detach(dev_info_t *dip, ddi_detach_cmd_t cmd)
474 {
475 	int		instance = ddi_get_instance(dip);
476 	usbprn_state_t	*usbprnp;
477 	int		rval = DDI_FAILURE;
478 
479 	usbprnp = ddi_get_soft_state(usbprn_statep, instance);
480 
481 	USB_DPRINTF_L4(PRINT_MASK_ATTA, usbprnp->usbprn_log_handle,
482 	    "usbprn_detach: cmd=%x", cmd);
483 
484 	switch (cmd) {
485 	case DDI_DETACH:
486 		ASSERT((usbprnp->usbprn_flags & USBPRN_OPEN) == 0);
487 		usbprn_cleanup(dip, usbprnp);
488 
489 		return (DDI_SUCCESS);
490 	case DDI_SUSPEND:
491 		rval = usbprn_cpr_suspend(dip);
492 
493 		return ((rval == USB_SUCCESS) ? DDI_SUCCESS :
494 		    DDI_FAILURE);
495 	default:
496 
497 		return (rval);
498 	}
499 }
500 
501 
502 /*
503  * usbprn_cleanup:
504  *	clean up the driver state
505  */
506 static void
507 usbprn_cleanup(dev_info_t *dip, usbprn_state_t *usbprnp)
508 {
509 	usbprn_power_t	*usbprnpm = usbprnp->usbprn_pm;
510 	int		rval = 0;
511 
512 	USB_DPRINTF_L4(PRINT_MASK_ATTA, usbprnp->usbprn_log_handle,
513 	    "usbprn_cleanup: Start");
514 
515 	ASSERT(usbprnp != NULL);
516 
517 	if (usbprnp->usbprn_flags & USBPRN_LOCKS_INIT_DONE) {
518 		/*
519 		 * Disable the event callbacks first, after this point, event
520 		 * callbacks will never get called. Note we shouldn't hold
521 		 * mutex while unregistering events because there may be a
522 		 * competing event callback thread. Event callbacks are done
523 		 * with ndi mutex held and this can cause a potential deadlock.
524 		 */
525 		usb_unregister_event_cbs(dip, &usbprn_events);
526 
527 		mutex_enter(&usbprnp->usbprn_mutex);
528 		if ((usbprnpm) &&
529 		    (usbprnp->usbprn_dev_state != USB_DEV_DISCONNECTED)) {
530 
531 			mutex_exit(&usbprnp->usbprn_mutex);
532 			usbprn_pm_busy_component(usbprnp);
533 			mutex_enter(&usbprnp->usbprn_mutex);
534 
535 			if (usbprnpm->usbprn_wakeup_enabled) {
536 
537 				mutex_exit(&usbprnp->usbprn_mutex);
538 
539 				(void) pm_raise_power(dip, 0,
540 				    USB_DEV_OS_FULL_PWR);
541 
542 				if ((rval = usb_handle_remote_wakeup(dip,
543 				    USB_REMOTE_WAKEUP_DISABLE)) !=
544 				    USB_SUCCESS) {
545 					USB_DPRINTF_L2(PRINT_MASK_ALL,
546 					    usbprnp->usbprn_log_handle,
547 					    "usbprn_cleanup: "
548 					    "disable remote wakeup "
549 					    "failed, rval=%d", rval);
550 				}
551 			} else {
552 				mutex_exit(&usbprnp->usbprn_mutex);
553 			}
554 
555 			(void) pm_lower_power(dip, 0, USB_DEV_OS_PWR_OFF);
556 			usbprn_pm_idle_component(usbprnp);
557 
558 			mutex_enter(&usbprnp->usbprn_mutex);
559 		}
560 
561 		ddi_remove_minor_node(dip, NULL);
562 
563 		mutex_exit(&usbprnp->usbprn_mutex);
564 
565 		if (usbprnp->usbprn_device_id) {
566 			kmem_free(usbprnp->usbprn_device_id,
567 			    usbprnp->usbprn_device_id_len + 1);
568 		}
569 
570 		mutex_destroy(&usbprnp->usbprn_mutex);
571 		usb_fini_serialization(usbprnp->usbprn_dev_acc);
572 		usb_fini_serialization(usbprnp->usbprn_ser_acc);
573 		usb_fini_serialization(usbprnp->usbprn_write_acc);
574 	}
575 
576 	if (usbprnpm) {
577 		kmem_free(usbprnpm, sizeof (usbprn_power_t));
578 	}
579 
580 	USB_DPRINTF_L4(PRINT_MASK_ATTA, usbprnp->usbprn_log_handle,
581 	    "usbprn_cleanup: End");
582 
583 	if (usbprnp->usbprn_ugen_hdl) {
584 		(void) usb_ugen_detach(usbprnp->usbprn_ugen_hdl, DDI_DETACH);
585 		usb_ugen_release_hdl(usbprnp->usbprn_ugen_hdl);
586 	}
587 
588 	/* unregister with USBA */
589 	usb_client_detach(dip, usbprnp->usbprn_dev_data);
590 
591 	usb_free_log_hdl(usbprnp->usbprn_log_handle);
592 	ddi_prop_remove_all(dip);
593 	ddi_soft_state_free(usbprn_statep, usbprnp->usbprn_instance);
594 }
595 
596 
597 /*
598  * usbprn_cpr_suspend:
599  *	prepare to be suspended
600  */
601 static int
602 usbprn_cpr_suspend(dev_info_t *dip)
603 {
604 	usbprn_state_t	*usbprnp;
605 	int		instance = ddi_get_instance(dip);
606 	int		rval = USB_FAILURE;
607 
608 	usbprnp = ddi_get_soft_state(usbprn_statep, instance);
609 
610 	USB_DPRINTF_L4(PRINT_MASK_CPR, usbprnp->usbprn_log_handle,
611 	    "usbprn_cpr_suspend");
612 
613 	(void) usb_serialize_access(usbprnp->usbprn_ser_acc, USB_WAIT, 0);
614 
615 	mutex_enter(&usbprnp->usbprn_mutex);
616 
617 	if ((usbprnp->usbprn_flags & USBPRN_OPEN) != 0) {
618 		mutex_exit(&usbprnp->usbprn_mutex);
619 
620 		USB_DPRINTF_L2(PRINT_MASK_CPR,
621 		    usbprnp->usbprn_log_handle,
622 		    "usbprn_cpr_suspend: "
623 		    "Device is open.  Can't suspend");
624 
625 	} else {
626 		usbprnp->usbprn_dev_state = USB_DEV_SUSPENDED;
627 		mutex_exit(&usbprnp->usbprn_mutex);
628 
629 		USB_DPRINTF_L4(PRINT_MASK_CPR, usbprnp->usbprn_log_handle,
630 		    "usbprn_cpr_suspend: SUCCESS");
631 		rval = USB_SUCCESS;
632 	}
633 	usb_release_access(usbprnp->usbprn_ser_acc);
634 
635 	if ((rval == USB_SUCCESS) && usbprnp->usbprn_ugen_hdl) {
636 		rval = usb_ugen_detach(usbprnp->usbprn_ugen_hdl,
637 		    DDI_SUSPEND);
638 	}
639 
640 	return (rval);
641 }
642 
643 
644 static void
645 usbprn_cpr_resume(dev_info_t *dip)
646 {
647 	int		instance = ddi_get_instance(dip);
648 	usbprn_state_t	*usbprnp = ddi_get_soft_state(usbprn_statep, instance);
649 
650 	USB_DPRINTF_L4(PRINT_MASK_CPR, usbprnp->usbprn_log_handle,
651 	    "usbprn_cpr_resume");
652 
653 	/* Needed as power up state of dev is "unknown" to system */
654 	usbprn_pm_busy_component(usbprnp);
655 	(void) pm_raise_power(dip, 0, USB_DEV_OS_FULL_PWR);
656 
657 	usbprn_restore_device_state(dip, usbprnp);
658 
659 	usbprn_pm_idle_component(usbprnp);
660 
661 	if (usbprnp->usbprn_ugen_hdl) {
662 		(void) usb_ugen_attach(usbprnp->usbprn_ugen_hdl,
663 		    DDI_RESUME);
664 	}
665 }
666 
667 
668 /*
669  * usbprn_get_descriptors:
670  *	Obtain all the descriptors for the device
671  */
672 static int
673 usbprn_get_descriptors(usbprn_state_t *usbprnp)
674 {
675 	int			interface;
676 	usb_client_dev_data_t	*dev_data =
677 	    usbprnp->usbprn_dev_data;
678 	usb_alt_if_data_t	*altif_data;
679 	usb_cfg_data_t		*cfg_data;
680 	usb_ep_data_t		*ep_data;
681 	dev_info_t		*dip = usbprnp->usbprn_dip;
682 	int			alt, rval;
683 
684 	ASSERT(!mutex_owned(&usbprnp->usbprn_mutex));
685 
686 	/*
687 	 * Section 4.2.1 of the spec says the printer could have
688 	 * multiple configurations.  This driver is just for one
689 	 * configuration interface and one interface.
690 	 */
691 	interface = dev_data->dev_curr_if;
692 	cfg_data = dev_data->dev_curr_cfg;
693 
694 	/* find alternate that supports BI/UNI protocol */
695 	for (alt = 0; alt < cfg_data->cfg_if[interface].if_n_alt; alt++) {
696 		altif_data = &cfg_data->cfg_if[interface].if_alt[alt];
697 
698 		if ((altif_data->altif_descr.bInterfaceProtocol ==
699 		    USB_PROTO_PRINTER_UNI) ||
700 		    (altif_data->altif_descr.bInterfaceProtocol ==
701 		    USB_PROTO_PRINTER_BI)) {
702 
703 			break;
704 		} else {
705 			USB_DPRINTF_L3(PRINT_MASK_ATTA,
706 			    usbprnp->usbprn_log_handle,
707 			    "alternate %d not supported", alt);
708 		}
709 	}
710 
711 	if (alt == cfg_data->cfg_if[interface].if_n_alt) {
712 		USB_DPRINTF_L2(PRINT_MASK_ATTA, usbprnp->usbprn_log_handle,
713 		    "usbprn_get_descriptors: no alternate");
714 
715 		return (USB_FAILURE);
716 	}
717 
718 
719 	if ((rval = usb_set_alt_if(dip, interface, alt, USB_FLAGS_SLEEP,
720 	    NULL, NULL)) != USB_SUCCESS) {
721 		USB_DPRINTF_L2(PRINT_MASK_ATTA, usbprnp->usbprn_log_handle,
722 		    "usbprn_get_descriptors: set alternate failed (%d)",
723 		    rval);
724 
725 		return (rval);
726 	}
727 
728 	usbprnp->usbprn_config_descr = cfg_data->cfg_descr;
729 	usbprnp->usbprn_if_descr = altif_data->altif_descr;
730 
731 	/*
732 	 * find the endpoint descriptors. There will be a bulk-out endpoint
733 	 * and an optional bulk-in endpoint.
734 	 */
735 	if ((ep_data = usb_lookup_ep_data(dip, dev_data, interface, alt, 0,
736 	    USB_EP_ATTR_BULK, USB_EP_DIR_OUT)) != NULL) {
737 		usbprnp->usbprn_bulk_out.ps_ept_descr = ep_data->ep_descr;
738 	}
739 	if ((ep_data = usb_lookup_ep_data(dip, dev_data, interface, alt, 0,
740 	    USB_EP_ATTR_BULK, USB_EP_DIR_IN)) != NULL) {
741 		usbprnp->usbprn_bulk_in.ps_ept_descr = ep_data->ep_descr;
742 	}
743 
744 	return (USB_SUCCESS);
745 }
746 
747 
748 /*
749  * usbprn_get_device_id:
750  *	Get the device id as described in 4.2.1 of the specification
751  *	Lexmark printer returns 2 bytes when asked for 8 bytes
752  *	We are ignoring data over and underrun.
753  *	This is a synchronous function
754  */
755 static int
756 usbprn_get_device_id(usbprn_state_t *usbprnp)
757 {
758 	int			len, n;
759 	mblk_t			*data = NULL;
760 	usb_cr_t		completion_reason;
761 	usb_cb_flags_t		cb_flags;
762 	int			rval = USB_FAILURE;
763 	usb_ctrl_setup_t setup = {
764 	    USB_DEV_REQ_DEV_TO_HOST |	/* bmRequestType */
765 	    USB_DEV_REQ_TYPE_CLASS |
766 	    USB_DEV_REQ_RCPT_IF,
767 	    USB_PRINTER_GET_DEVICE_ID,	/* bRequest */
768 	    0,				/* wValue: fill in later */
769 	    0,				/* wIndex: fill in later  */
770 	    0,				/* wLength: fill in later */
771 	    0				/* attributes */
772 	    };
773 	void			*ptr;
774 
775 	USB_DPRINTF_L4(PRINT_MASK_ATTA, usbprnp->usbprn_log_handle,
776 	    "usbprn_get_device_id: Begin");
777 
778 	ASSERT(!mutex_owned(&usbprnp->usbprn_mutex));
779 
780 	setup.wIndex = (usbprnp->usbprn_if_descr.bInterfaceNumber << 0x8) |
781 	    (usbprnp->usbprn_if_descr.bAlternateSetting);
782 	setup.wLength = USBPRN_MAX_DEVICE_ID_LENGTH;
783 	setup.wValue = usbprnp->usbprn_config_descr.iConfiguration;
784 
785 	/*
786 	 * This is always a sync request as this will never
787 	 * be called in interrupt context.
788 	 * First get the first two bytes that gives the length
789 	 * of the device id string; then get the whole string
790 	 */
791 	if (usb_pipe_ctrl_xfer_wait(usbprnp->usbprn_def_ph, &setup,
792 	    &data, &completion_reason, &cb_flags, 0) != USB_SUCCESS) {
793 
794 		USB_DPRINTF_L2(PRINT_MASK_ATTA, usbprnp->usbprn_log_handle,
795 		    "usbprn_get_device_id: First sync command failed, cr=%d ",
796 		    completion_reason);
797 
798 		/*
799 		 * some devices return more than requested. as long as
800 		 * we get the first two bytes, we can continue
801 		 */
802 		if (((completion_reason != USB_CR_DATA_OVERRUN) &&
803 		    (completion_reason != USB_CR_DATA_UNDERRUN)) ||
804 		    (data == NULL)) {
805 
806 			goto done;
807 		}
808 	}
809 
810 	ASSERT(data);
811 	n = data->b_wptr - data->b_rptr;
812 
813 	if (n < 2) {
814 
815 		goto done;
816 	}
817 
818 	len = (((*data->b_rptr) << 0x8) | (*(data->b_rptr+1)));
819 
820 	/*
821 	 * Std 1284-1994, chapter 7.6:
822 	 *	Length values of x'0000', x'0001' and x'0002' are reserved
823 	 */
824 	if (len < 3) {
825 
826 		goto done;
827 	}
828 
829 	USB_DPRINTF_L3(PRINT_MASK_ATTA, usbprnp->usbprn_log_handle,
830 	    "usbprn_get_device_id: device id length=%d", len);
831 
832 	/* did we get enough data */
833 	if (len > n) {
834 		freemsg(data);
835 		data = NULL;
836 
837 		setup.wLength = (uint16_t)len;
838 		if ((rval = usb_pipe_ctrl_xfer_wait(usbprnp->usbprn_def_ph,
839 		    &setup, &data, &completion_reason, &cb_flags, 0)) !=
840 		    USB_SUCCESS) {
841 			USB_DPRINTF_L2(PRINT_MASK_ATTA,
842 			    usbprnp->usbprn_log_handle,
843 			    "usbprn_get_device_id: 2nd command failed "
844 			    "cr=%d cb_flags=0x%x",
845 			    completion_reason, cb_flags);
846 
847 			goto done;
848 		}
849 
850 		ASSERT(len == (data->b_wptr - data->b_rptr));
851 	}
852 
853 	USB_DPRINTF_L3(PRINT_MASK_ATTA, usbprnp->usbprn_log_handle,
854 	    "usbprn_get_device_id: returned data length=%ld",
855 	    (long)(data->b_wptr - data->b_rptr));
856 
857 	ptr = kmem_zalloc(len + 1, KM_SLEEP);
858 
859 	mutex_enter(&usbprnp->usbprn_mutex);
860 	usbprnp->usbprn_device_id_len = len;
861 	usbprnp->usbprn_device_id = ptr;
862 
863 	bcopy(data->b_rptr, usbprnp->usbprn_device_id,
864 	    usbprnp->usbprn_device_id_len);
865 	usbprnp->usbprn_device_id[usbprnp->usbprn_device_id_len] = '\0';
866 
867 	/* Length is in the first two bytes, dump string in logbuf */
868 	usbprn_print_long(usbprnp, usbprnp->usbprn_device_id + 2,
869 	    usbprnp->usbprn_device_id_len - 2);
870 	mutex_exit(&usbprnp->usbprn_mutex);
871 
872 	rval = USB_SUCCESS;
873 done:
874 	freemsg(data);
875 
876 	USB_DPRINTF_L4(PRINT_MASK_ATTA, usbprnp->usbprn_log_handle,
877 	    "usbprn_get_device_id: rval=%d", rval);
878 
879 	return (rval);
880 }
881 
882 
883 /*
884  * usbprn_get_port_status:
885  *	Get the port status.
886  *	This is a synchronous function
887  */
888 static int
889 usbprn_get_port_status(usbprn_state_t  *usbprnp)
890 {
891 	mblk_t			*data = NULL;
892 	usb_cr_t		completion_reason;
893 	usb_cb_flags_t		cb_flags;
894 	usb_ctrl_setup_t setup = {
895 	    USB_DEV_REQ_DEV_TO_HOST |	/* bmRequestType */
896 	    USB_DEV_REQ_TYPE_CLASS |
897 	    USB_DEV_REQ_RCPT_IF,
898 	    USB_PRINTER_GET_PORT_STATUS, /* bRequest */
899 	    0,				/* wValue */
900 	    0,				/* wIndex: fill in later  */
901 	    1,				/* wLength */
902 	    0				/* attributes */
903 	    };
904 	ASSERT(!mutex_owned(&usbprnp->usbprn_mutex));
905 
906 	USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
907 	    "usbprn_get_port_status: Begin");
908 
909 	setup.wIndex = usbprnp->usbprn_if_descr.bInterfaceNumber;
910 	if (usb_pipe_ctrl_xfer_wait(usbprnp->usbprn_def_ph,
911 	    &setup, &data, &completion_reason, &cb_flags, 0) !=
912 	    USB_SUCCESS) {
913 		USB_DPRINTF_L2(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
914 		    "usbprn_get_port_status: Sync command failed "
915 		    "cr=%d cb_flags=0x%x", completion_reason, cb_flags);
916 
917 		freemsg(data);
918 
919 		return (USB_FAILURE);
920 	} else {
921 		mutex_enter(&usbprnp->usbprn_mutex);
922 
923 		ASSERT(data);
924 		ASSERT((data->b_wptr - data->b_rptr) == 1);
925 
926 		usbprnp->usbprn_last_status = *data->b_rptr;
927 
928 		USB_DPRINTF_L3(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
929 		    "usbprn_get_port_status(sync): status=0x%x",
930 		    usbprnp->usbprn_last_status);
931 
932 		mutex_exit(&usbprnp->usbprn_mutex);
933 		freemsg(data);
934 
935 		return (USB_SUCCESS);
936 	}
937 }
938 
939 
940 /*
941  * usbprn_open:
942  *	Open the pipes
943  */
944 /*ARGSUSED*/
945 static int
946 usbprn_open(dev_t *devp, int flag, int sflag, cred_t *credp)
947 {
948 	usbprn_state_t *usbprnp = ddi_get_soft_state(usbprn_statep,
949 	    USBPRN_MINOR_TO_INSTANCE(getminor(*devp)));
950 	int rval = 0;
951 
952 	if (usbprnp == NULL) {
953 
954 		return (ENXIO);
955 	}
956 
957 	USB_DPRINTF_L4(PRINT_MASK_OPEN, usbprnp->usbprn_log_handle,
958 	    "usbprn_open:");
959 
960 	(void) usb_serialize_access(usbprnp->usbprn_ser_acc, USB_WAIT, 0);
961 
962 	/* Fail open on a disconnected device */
963 	mutex_enter(&usbprnp->usbprn_mutex);
964 	if (usbprnp->usbprn_dev_state == USB_DEV_DISCONNECTED) {
965 		mutex_exit(&usbprnp->usbprn_mutex);
966 		usb_release_access(usbprnp->usbprn_ser_acc);
967 
968 		return (ENODEV);
969 	}
970 
971 	/* cannot happen? but just in case */
972 	if (usbprnp->usbprn_dev_state == USB_DEV_SUSPENDED) {
973 		mutex_exit(&usbprnp->usbprn_mutex);
974 		usb_release_access(usbprnp->usbprn_ser_acc);
975 
976 		return (EIO);
977 	}
978 
979 	if (getminor(*devp) & USBPRN_MINOR_UGEN_BITS_MASK) {
980 		mutex_exit(&usbprnp->usbprn_mutex);
981 
982 		rval = usb_ugen_open(usbprnp->usbprn_ugen_hdl,
983 		    devp, flag, sflag, credp);
984 
985 		usb_release_access(usbprnp->usbprn_ser_acc);
986 
987 		return (rval);
988 	}
989 
990 	/* Exit if this instance is already open */
991 	if (usbprnp->usbprn_flags & USBPRN_OPEN) {
992 		mutex_exit(&usbprnp->usbprn_mutex);
993 		usb_release_access(usbprnp->usbprn_ser_acc);
994 
995 		return (EBUSY);
996 	}
997 	mutex_exit(&usbprnp->usbprn_mutex);
998 
999 	/* raise power */
1000 	usbprn_pm_busy_component(usbprnp);
1001 	(void) pm_raise_power(usbprnp->usbprn_dip,
1002 	    0, USB_DEV_OS_FULL_PWR);
1003 	/* initialize some softstate data */
1004 	mutex_enter(&usbprnp->usbprn_mutex);
1005 	usbprnp->usbprn_prn_timeouts.tmo_forward =
1006 	    usbprnp->usbprn_setparms.write_timeout;
1007 	usbprnp->usbprn_prn_timeouts.tmo_reverse = 0;
1008 	mutex_exit(&usbprnp->usbprn_mutex);
1009 
1010 	if (usbprn_open_usb_pipes(usbprnp) != USB_SUCCESS) {
1011 
1012 		USB_DPRINTF_L2(PRINT_MASK_ATTA, usbprnp->usbprn_log_handle,
1013 		    "usbprn_open: pipe open failed");
1014 
1015 		usb_release_access(usbprnp->usbprn_ser_acc);
1016 		usbprn_pm_idle_component(usbprnp);
1017 
1018 		return (EIO);
1019 	}
1020 
1021 	mutex_enter(&usbprnp->usbprn_mutex);
1022 	usbprnp->usbprn_flags |= USBPRN_OPEN;
1023 
1024 	/* set last status to online */
1025 	usbprnp->usbprn_last_status &= ~USB_PRINTER_PORT_NO_SELECT;
1026 	mutex_exit(&usbprnp->usbprn_mutex);
1027 
1028 	usb_release_access(usbprnp->usbprn_ser_acc);
1029 
1030 	USB_DPRINTF_L4(PRINT_MASK_OPEN, usbprnp->usbprn_log_handle,
1031 	    "usbprn_open: End");
1032 
1033 	return (rval);
1034 }
1035 
1036 
1037 /*
1038  * usbprn_close:
1039  *	Close the pipes
1040  */
1041 /*ARGSUSED*/
1042 static int
1043 usbprn_close(dev_t dev, int flag, int otyp, cred_t *credp)
1044 {
1045 	usbprn_state_t	*usbprnp = ddi_get_soft_state(usbprn_statep,
1046 	    USBPRN_MINOR_TO_INSTANCE(getminor(dev)));
1047 	int		rval = 0;
1048 
1049 	if (usbprnp == NULL) {
1050 
1051 		return (ENXIO);
1052 	}
1053 
1054 	USB_DPRINTF_L4(PRINT_MASK_CLOSE, usbprnp->usbprn_log_handle,
1055 	    "usbprn_close:");
1056 
1057 	if (getminor(dev) & USBPRN_MINOR_UGEN_BITS_MASK) {
1058 		rval = usb_ugen_close(usbprnp->usbprn_ugen_hdl,
1059 		    dev, flag, otyp, credp);
1060 
1061 		return (rval);
1062 	}
1063 
1064 	/* avoid races with connect/disconnect */
1065 	(void) usb_serialize_access(usbprnp->usbprn_ser_acc, USB_WAIT, 0);
1066 	(void) usb_serialize_access(usbprnp->usbprn_dev_acc, USB_WAIT, 0);
1067 
1068 	/* Close all usb pipes */
1069 	usbprn_close_usb_pipes(usbprnp);
1070 
1071 	/* prevent any accesses by setting flags to closed */
1072 	mutex_enter(&usbprnp->usbprn_mutex);
1073 	usbprnp->usbprn_flags &= ~USBPRN_OPEN;
1074 	mutex_exit(&usbprnp->usbprn_mutex);
1075 
1076 	usb_release_access(usbprnp->usbprn_dev_acc);
1077 	usb_release_access(usbprnp->usbprn_ser_acc);
1078 
1079 	usbprn_pm_idle_component(usbprnp);
1080 
1081 	USB_DPRINTF_L4(PRINT_MASK_CLOSE, usbprnp->usbprn_log_handle,
1082 	    "usbprn_close: End");
1083 
1084 	return (rval);
1085 }
1086 
1087 
1088 /*
1089  * usbprn_read:
1090  *	Read entry point (TBD)
1091  */
1092 /* ARGSUSED */
1093 static int
1094 usbprn_read(dev_t dev, struct uio *uiop, cred_t *credp)
1095 {
1096 	usbprn_state_t *usbprnp = ddi_get_soft_state(usbprn_statep,
1097 	    USBPRN_MINOR_TO_INSTANCE(getminor(dev)));
1098 
1099 	if (usbprnp == NULL) {
1100 
1101 		return (ENXIO);
1102 	}
1103 
1104 	if (getminor(dev) & USBPRN_MINOR_UGEN_BITS_MASK) {
1105 		int rval;
1106 
1107 		/* raise power */
1108 		usbprn_pm_busy_component(usbprnp);
1109 		(void) pm_raise_power(usbprnp->usbprn_dip,
1110 		    0, USB_DEV_OS_FULL_PWR);
1111 
1112 		if (usb_serialize_access(usbprnp->usbprn_write_acc,
1113 		    USB_WAIT_SIG, 0) == 0) {
1114 			usbprn_pm_idle_component(usbprnp);
1115 
1116 			return (EINTR);
1117 		}
1118 
1119 		rval = usb_ugen_read(usbprnp->usbprn_ugen_hdl, dev,
1120 		    uiop, credp);
1121 
1122 		usb_release_access(usbprnp->usbprn_write_acc);
1123 
1124 		usbprn_pm_idle_component(usbprnp);
1125 
1126 		return (rval);
1127 	}
1128 
1129 	/* Do a bulk-in from the printer */
1130 
1131 	return (EIO);
1132 }
1133 
1134 
1135 /*
1136  * usbprn_write:
1137  *	Write to the printer
1138  */
1139 /* ARGSUSED2 */
1140 static int
1141 usbprn_write(dev_t dev, struct uio *uiop, cred_t *credp)
1142 {
1143 	usbprn_state_t *usbprnp = ddi_get_soft_state(usbprn_statep,
1144 	    USBPRN_MINOR_TO_INSTANCE(getminor(dev)));
1145 	usbprn_ps_t	*bulk_in = &usbprnp->usbprn_bulk_in;
1146 	usbprn_ps_t	*bulk_out = &usbprnp->usbprn_bulk_out;
1147 	int		rval;
1148 
1149 	if (usbprnp == NULL) {
1150 
1151 		return (ENXIO);
1152 	}
1153 
1154 	USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1155 	    "usbprn_write: Begin usbprnp=0x%p ", (void *)usbprnp);
1156 
1157 	if (getminor(dev) & USBPRN_MINOR_UGEN_BITS_MASK) {
1158 		/* raise power */
1159 		usbprn_pm_busy_component(usbprnp);
1160 		(void) pm_raise_power(usbprnp->usbprn_dip,
1161 		    0, USB_DEV_OS_FULL_PWR);
1162 
1163 		if (usb_serialize_access(usbprnp->usbprn_write_acc,
1164 		    USB_WAIT_SIG, 0) == 0) {
1165 			usbprn_pm_idle_component(usbprnp);
1166 
1167 			return (EINTR);
1168 		}
1169 
1170 		rval = usb_ugen_write(usbprnp->usbprn_ugen_hdl, dev,
1171 		    uiop, credp);
1172 
1173 		usb_release_access(usbprnp->usbprn_write_acc);
1174 
1175 		usbprn_pm_idle_component(usbprnp);
1176 
1177 		return (rval);
1178 	}
1179 
1180 	/*
1181 	 * serialize writes
1182 	 * we cannot use usbprn_ser_acc sync object at this point because
1183 	 * that would block out the ioctls for the full duration of the write.
1184 	 */
1185 	if (usb_serialize_access(usbprnp->usbprn_write_acc,
1186 	    USB_WAIT_SIG, 0) == 0) {
1187 
1188 		return (EINTR);
1189 	}
1190 
1191 	/*
1192 	 * Check the status of the pipe.  If it's not idle,
1193 	 * then wait.
1194 	 */
1195 	mutex_enter(&usbprnp->usbprn_mutex);
1196 
1197 	/* if device is disconnected or pipes closed, fail immediately */
1198 	if (!(USBPRN_DEVICE_ACCESS_OK(usbprnp))) {
1199 		mutex_exit(&usbprnp->usbprn_mutex);
1200 
1201 		USB_DPRINTF_L2(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1202 		    "usbprn_write: device can't be accessed");
1203 
1204 		usb_release_access(usbprnp->usbprn_write_acc);
1205 
1206 		return (EIO);
1207 	}
1208 
1209 	/* all pipes must be idle */
1210 	ASSERT(bulk_out->ps_flags == USBPRN_PS_IDLE);
1211 	ASSERT(bulk_in->ps_flags == USBPRN_PS_IDLE);
1212 
1213 	mutex_exit(&usbprnp->usbprn_mutex);
1214 
1215 	/*
1216 	 * Call physio to do the transfer.  physio will
1217 	 * call the strategy routine, and then call
1218 	 * biowait() to block until the transfer completes.
1219 	 */
1220 	rval = physio(usbprn_strategy, (struct buf *)0, dev,
1221 	    B_WRITE, usbprn_minphys, uiop);
1222 
1223 	usb_release_access(usbprnp->usbprn_write_acc);
1224 
1225 	USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1226 	    "usbprn_write: End");
1227 
1228 	return (rval);
1229 }
1230 
1231 
1232 /*
1233  * usbprn_poll
1234  */
1235 static int
1236 usbprn_poll(dev_t dev, short events,
1237     int anyyet,  short *reventsp, struct pollhead **phpp)
1238 {
1239 	usbprn_state_t *usbprnp = ddi_get_soft_state(usbprn_statep,
1240 	    USBPRN_MINOR_TO_INSTANCE(getminor(dev)));
1241 
1242 	if (usbprnp == NULL) {
1243 
1244 		return (ENXIO);
1245 	}
1246 
1247 	if (getminor(dev) & USBPRN_MINOR_UGEN_BITS_MASK) {
1248 		return (usb_ugen_poll(usbprnp->usbprn_ugen_hdl, dev, events,
1249 		    anyyet, reventsp, phpp));
1250 	}
1251 
1252 	return (ENXIO);
1253 }
1254 
1255 
1256 /*
1257  * usbprn_strategy:
1258  *	service a request to the device.
1259  */
1260 static int
1261 usbprn_strategy(struct buf *bp)
1262 {
1263 	usbprn_state_t *usbprnp = ddi_get_soft_state(usbprn_statep,
1264 	    USBPRN_MINOR_TO_INSTANCE(getminor(bp->b_edev)));
1265 	usbprn_ps_t	*bulk_out = &usbprnp->usbprn_bulk_out;
1266 
1267 	bp_mapin(bp);
1268 
1269 	/*
1270 	 * serialize to avoid races
1271 	 * access is released in usbprn_biodone()
1272 	 */
1273 	(void) usb_serialize_access(usbprnp->usbprn_dev_acc, USB_WAIT, 0);
1274 
1275 	mutex_enter(&usbprnp->usbprn_mutex);
1276 	if (!(USBPRN_DEVICE_ACCESS_OK(usbprnp))) {
1277 		usbprn_biodone(usbprnp, EIO, 0);
1278 		mutex_exit(&usbprnp->usbprn_mutex);
1279 
1280 		USB_DPRINTF_L2(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1281 		    "usbprn_strategy: device can't be accessed");
1282 
1283 		return (0);
1284 	}
1285 
1286 	bulk_out->ps_flags = USBPRN_PS_NEED_TO_XFER;
1287 
1288 	ASSERT(usbprnp->usbprn_bp == NULL);
1289 	usbprnp->usbprn_bp = bp;
1290 
1291 	USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1292 	    "usbprn_strategy: usbprnp=0x%p bp=0x%p count=%lu",
1293 	    (void *)usbprnp, (void *)bp, bp->b_bcount);
1294 
1295 	ASSERT(usbprnp->usbprn_bulk_mp == NULL);
1296 
1297 	usbprnp->usbprn_bulk_mp = allocb(bp->b_bcount, BPRI_HI);
1298 
1299 	if (usbprnp->usbprn_bulk_mp == NULL) {
1300 		bulk_out->ps_flags = USBPRN_PS_IDLE;
1301 		usbprn_biodone(usbprnp, EIO, 0);
1302 		mutex_exit(&usbprnp->usbprn_mutex);
1303 
1304 		USB_DPRINTF_L2(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1305 		    "usbprn_strategy: allocb failed");
1306 
1307 		return (0);
1308 	}
1309 
1310 	bcopy((caddr_t)bp->b_un.b_addr,
1311 	    usbprnp->usbprn_bulk_mp->b_datap->db_base, bp->b_bcount);
1312 	usbprnp->usbprn_bulk_mp->b_wptr += bp->b_bcount;
1313 	mutex_exit(&usbprnp->usbprn_mutex);
1314 
1315 	usbprn_send_async_bulk_data(usbprnp);
1316 
1317 	return (0);
1318 }
1319 
1320 
1321 /*
1322  * usbprn_ioctl:
1323  *	handle the ioctl
1324  */
1325 /*ARGSUSED4*/
1326 static int
1327 usbprn_ioctl(dev_t dev, int cmd, intptr_t arg, int flag,
1328 		cred_t *credp, int *rvalp)
1329 {
1330 	int		err = 0;
1331 	usbprn_state_t	*usbprnp = ddi_get_soft_state(usbprn_statep,
1332 	    USBPRN_MINOR_TO_INSTANCE(getminor(dev)));
1333 	struct ecpp_device_id	usbprn_devid;
1334 	int		len;
1335 
1336 	USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1337 	    "usbprn_ioctl: Begin ");
1338 
1339 	(void) usb_serialize_access(usbprnp->usbprn_ser_acc, USB_WAIT, 0);
1340 	mutex_enter(&usbprnp->usbprn_mutex);
1341 
1342 	/*
1343 	 * only for PRNIOC_GET_STATUS cmd:
1344 	 * if device is disconnected or pipes closed, fail immediately
1345 	 */
1346 	if ((cmd == PRNIOC_GET_STATUS) &&
1347 	    !(USBPRN_DEVICE_ACCESS_OK(usbprnp))) {
1348 		mutex_exit(&usbprnp->usbprn_mutex);
1349 
1350 		USB_DPRINTF_L2(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1351 		    "usbprn_write: device can't be accessed");
1352 
1353 		usb_release_access(usbprnp->usbprn_ser_acc);
1354 
1355 		return (EIO);
1356 	}
1357 	mutex_exit(&usbprnp->usbprn_mutex);
1358 
1359 	switch (cmd) {
1360 	case ECPPIOC_GETDEVID:
1361 		/*
1362 		 * With genericized ioctls this interface should change.
1363 		 * We ignore the mode in USB printer driver because
1364 		 * it need not be in nibble mode in usb driver unlike
1365 		 * ecpp to retrieve the device id string. Also we do
1366 		 * not expect the application to call this twice since
1367 		 * it doesn't change since attach time and we take care
1368 		 * of calling it twice: once for getting the length and
1369 		 * once for getting the actual device id string. So we
1370 		 * set both the lengths to actual device id string length.
1371 		 * Ref: PSARC/2000/018
1372 		 */
1373 		USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1374 		    "usbprn_ioctl: ECPPIOC_GETDEVID(0x%x)", cmd);
1375 
1376 		bzero(&usbprn_devid, sizeof (usbprn_devid));
1377 
1378 		ASSERT(!(mutex_owned(&usbprnp->usbprn_mutex)));
1379 #ifdef _MULTI_DATAMODEL
1380 		switch (ddi_model_convert_from(flag & FMODELS)) {
1381 		case DDI_MODEL_ILP32: {
1382 			struct ecpp_device_id32	usbprn_devid32;
1383 
1384 			if (ddi_copyin((caddr_t)arg, &usbprn_devid32,
1385 			    sizeof (struct ecpp_device_id32), flag)) {
1386 				err = EFAULT;
1387 
1388 				break;
1389 			}
1390 
1391 			if (usbprnp->usbprn_device_id == NULL) {
1392 				err = EIO;
1393 
1394 				break;
1395 			}
1396 			ASSERT(usbprnp->usbprn_device_id_len > 2);
1397 
1398 			usbprn_devid32.rlen = usbprnp->usbprn_device_id_len - 2;
1399 			len = min(usbprn_devid32.len, usbprn_devid32.rlen);
1400 
1401 			if (ddi_copyout(usbprnp->usbprn_device_id + 2,
1402 			    (caddr_t)(uintptr_t)usbprn_devid32.addr,
1403 			    len, flag)) {
1404 				err = EFAULT;
1405 
1406 				break;
1407 			}
1408 
1409 			if (ddi_copyout(&usbprn_devid32, (caddr_t)arg,
1410 			    sizeof (struct ecpp_device_id32), flag)) {
1411 				err = EFAULT;
1412 
1413 				break;
1414 			}
1415 
1416 			break;
1417 		}
1418 		case DDI_MODEL_NONE:
1419 			if (ddi_copyin((caddr_t)arg, &usbprn_devid,
1420 			    sizeof (struct ecpp_device_id), flag)) {
1421 				err = EFAULT;
1422 
1423 				break;
1424 			}
1425 
1426 			if (usbprnp->usbprn_device_id == NULL) {
1427 				err = EIO;
1428 
1429 				break;
1430 			}
1431 			ASSERT(usbprnp->usbprn_device_id_len > 2);
1432 
1433 			usbprn_devid.rlen = usbprnp->usbprn_device_id_len - 2;
1434 			len = min(usbprn_devid.len, usbprn_devid.rlen);
1435 
1436 			if (ddi_copyout(usbprnp->usbprn_device_id + 2,
1437 			    usbprn_devid.addr, len, flag)) {
1438 				err = EFAULT;
1439 
1440 				break;
1441 			}
1442 
1443 			if (ddi_copyout(&usbprn_devid, (caddr_t)arg,
1444 			    sizeof (struct ecpp_device_id), flag)) {
1445 				err = EFAULT;
1446 
1447 				break;
1448 			}
1449 
1450 			break;
1451 		}
1452 
1453 		break;
1454 #else
1455 		if (ddi_copyin((caddr_t)arg, &usbprn_devid,
1456 		    sizeof (struct ecpp_device_id), flag)) {
1457 			err = EFAULT;
1458 
1459 			break;
1460 		}
1461 
1462 
1463 		if (usbprnp->usbprn_device_id == NULL) {
1464 			err = EIO;
1465 
1466 			break;
1467 		}
1468 		ASSERT(usbprnp->usbprn_device_id_len > 2);
1469 
1470 		usbprn_devid.rlen = usbprnp->usbprn_device_id_len - 2;
1471 		len = min(usbprn_devid.len, usbprn_devid.rlen);
1472 
1473 		if (ddi_copyout(usbprnp->usbprn_device_id + 2,
1474 		    usbprn_devid.addr, len, flag)) {
1475 			err = EFAULT;
1476 
1477 			break;
1478 		}
1479 
1480 		if (ddi_copyout(&usbprn_devid, (caddr_t)arg,
1481 		    sizeof (struct ecpp_device_id), flag)) {
1482 			err = EFAULT;
1483 
1484 			break;
1485 		}
1486 
1487 		break;
1488 #endif
1489 	case ECPPIOC_SETPARMS:
1490 		err = usbprn_setparms(usbprnp, arg, flag);
1491 
1492 		break;
1493 	case ECPPIOC_GETPARMS:
1494 		USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1495 		    "usbprn_ioctl: ECPPIOC_GETPARMS(0x%x)", cmd);
1496 
1497 		/* Get the parameters */
1498 		err = usbprn_getparms(usbprnp, arg, flag);
1499 
1500 		break;
1501 	case BPPIOC_GETERR:
1502 		USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1503 		    "usbprn_ioctl: ECPPIOC_GETERR(0x%x)", cmd);
1504 
1505 		/* Get the error state */
1506 		usbprn_geterr(usbprnp, arg, flag);
1507 
1508 		break;
1509 	case BPPIOC_TESTIO:
1510 		USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1511 		    "usbprn_ioctl: BPPIOC_TESTIO(0x%x)",  cmd);
1512 
1513 		/* Get the port status */
1514 		err = usbprn_testio(usbprnp, flag);
1515 
1516 		break;
1517 	case PRNIOC_GET_IFCAP:
1518 		USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1519 		    "usbprn_ioctl : PRNIOC_GET_IFCAP(0x%x)",  cmd);
1520 
1521 		/* get interface capabilities */
1522 		err = usbprn_prnio_get_ifcap(usbprnp, arg, flag);
1523 
1524 		break;
1525 	case PRNIOC_SET_IFCAP:
1526 		USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1527 		    "usbprn_ioctl : PRNIOC_SET_IFCAP(0x%x)",  cmd);
1528 
1529 		/* get interface capabilities */
1530 		err = usbprn_prnio_set_ifcap(usbprnp, arg, flag);
1531 
1532 		break;
1533 	case PRNIOC_GET_IFINFO:
1534 		USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1535 		    "usbprn_ioctl : PRNIOC_GET_IFINFO(0x%x)",  cmd);
1536 
1537 		/* get interface information */
1538 		err = usbprn_prnio_get_ifinfo(usbprnp, arg, flag);
1539 
1540 		break;
1541 	case PRNIOC_GET_STATUS:
1542 		USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1543 		    "usbprn_ioctl : PRNIOC_GET_STATUS(0x%x)",  cmd);
1544 
1545 		/* get prnio status */
1546 		err = usbprn_prnio_get_status(usbprnp, arg, flag);
1547 
1548 		break;
1549 	case PRNIOC_GET_1284_DEVID:
1550 		USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1551 		    "usbprn_ioctl : PRNIOC_GET_1284_DEVID(0x%x)",  cmd);
1552 
1553 		/* get device ID */
1554 		err = usbprn_prnio_get_1284_devid(usbprnp, arg, flag);
1555 
1556 		break;
1557 	case PRNIOC_GET_1284_STATUS:
1558 		USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1559 		    "usbprn_ioctl : PRNIOC_GET_1284_STATUS(0x%x)",  cmd);
1560 
1561 		/* get prnio status */
1562 		err = usbprn_prnio_get_1284_status(usbprnp, arg, flag);
1563 
1564 		break;
1565 	case PRNIOC_GET_TIMEOUTS:
1566 		USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1567 		    "usbprn_ioctl : PRNIOC_GET_TIMEOUTS(0x%x)", cmd);
1568 
1569 		/* Get the parameters */
1570 		err = usbprn_prnio_get_timeouts(usbprnp, arg, flag);
1571 
1572 		break;
1573 	case PRNIOC_SET_TIMEOUTS:
1574 		USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1575 		    "usbprn_ioctl : PRNIOC_SET_TIMEOUTS(0x%x)", cmd);
1576 
1577 		/* Get the parameters */
1578 		err = usbprn_prnio_set_timeouts(usbprnp, arg, flag);
1579 
1580 		break;
1581 	case PRNIOC_RESET:
1582 		USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1583 		    "usbprn_ioctl : PRNIOC_RESET(0x%x)",  cmd);
1584 
1585 		/* nothing */
1586 		err = 0;
1587 
1588 		break;
1589 	default:
1590 		USB_DPRINTF_L2(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1591 		    "usbprn_ioctl: unknown(0x%x)", cmd);
1592 		err = EINVAL;
1593 	}
1594 
1595 	usb_release_access(usbprnp->usbprn_ser_acc);
1596 
1597 	USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1598 	    "usbprn_ioctl: End ");
1599 
1600 	return (err);
1601 }
1602 
1603 
1604 /*
1605  * breakup by physio
1606  */
1607 static void
1608 usbprn_minphys(struct buf *bp)
1609 {
1610 	usbprn_state_t *usbprnp = ddi_get_soft_state(usbprn_statep,
1611 	    USBPRN_MINOR_TO_INSTANCE(getminor(bp->b_edev)));
1612 
1613 	mutex_enter(&usbprnp->usbprn_mutex);
1614 	USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1615 	    "usbprn_minphys: bcount=%lu", bp->b_bcount);
1616 
1617 	if (bp->b_bcount > usbprnp->usbprn_max_bulk_xfer_size) {
1618 		bp->b_bcount = min(usbprn_max_xfer_size,
1619 		    usbprnp->usbprn_max_bulk_xfer_size);
1620 	} else {
1621 		bp->b_bcount = min(usbprn_max_xfer_size, bp->b_bcount);
1622 	}
1623 	mutex_exit(&usbprnp->usbprn_mutex);
1624 }
1625 
1626 
1627 /*
1628  * usbprn_open_usb_pipes:
1629  *	Open all pipes on the device
1630  */
1631 static int
1632 usbprn_open_usb_pipes(usbprn_state_t *usbprnp)
1633 {
1634 	usb_pipe_policy_t *policy;
1635 	usbprn_ps_t	*bulk_in = &usbprnp->usbprn_bulk_in;
1636 	usbprn_ps_t	*bulk_out = &usbprnp->usbprn_bulk_out;
1637 
1638 	USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1639 	    "usbprn_open_usb_pipes:");
1640 
1641 	/*
1642 	 * Intitialize the pipe policy for the bulk out pipe
1643 	 */
1644 	mutex_enter(&usbprnp->usbprn_mutex);
1645 	policy = &(bulk_out->ps_policy);
1646 	policy->pp_max_async_reqs = 1;
1647 	mutex_exit(&usbprnp->usbprn_mutex);
1648 
1649 	/* Open bulk_out pipe */
1650 	if (usb_pipe_open(usbprnp->usbprn_dip, &bulk_out->ps_ept_descr,
1651 	    policy, USB_FLAGS_SLEEP, &bulk_out->ps_handle) != USB_SUCCESS) {
1652 
1653 		return (USB_FAILURE);
1654 	}
1655 
1656 #ifdef LATER
1657 	mutex_enter(&usbprnp->usbprn_mutex);
1658 	/* Open the bulk in pipe if one exists */
1659 	if (bulk_in->ps_ept_descr->bLength) {
1660 		/*
1661 		 * Initialize the pipe policy for the Bulk In pipe
1662 		 */
1663 		policy = &bulk_in->ps_policy;
1664 		bulk_in->ps_flags = USBPRN_PS_IDLE;
1665 		policy->pp_max_async_reqs = 1;
1666 		mutex_exit(&usbprnp->usbprn_mutex);
1667 
1668 		/* Open bulk_in pipe */
1669 		if (usb_pipe_open(usbprnp->usbprn_dip, bulk_in->ps_ept_descr,
1670 		    policy, USB_FLAGS_SLEEP, &bulk_in->ps_handle) !=
1671 		    USB_SUCCESS) {
1672 
1673 			return (USB_FAILURE);
1674 		}
1675 	} else {
1676 		mutex_exit(&usbprnp->usbprn_mutex);
1677 	}
1678 #else
1679 	mutex_enter(&usbprnp->usbprn_mutex);
1680 	bulk_in->ps_flags = USBPRN_PS_IDLE;
1681 	mutex_exit(&usbprnp->usbprn_mutex);
1682 #endif
1683 
1684 	USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1685 	    "usbprn_open_usb_pipes: success");
1686 
1687 	return (USB_SUCCESS);
1688 }
1689 
1690 
1691 /*
1692  * usbprn_close_usb_pipes:
1693  *	Close the default/bulk in/out pipes synchronously
1694  */
1695 static void
1696 usbprn_close_usb_pipes(usbprn_state_t *usbprnp)
1697 {
1698 	usbprn_ps_t	*bulk_in = &usbprnp->usbprn_bulk_in;
1699 	usbprn_ps_t	*bulk_out = &usbprnp->usbprn_bulk_out;
1700 
1701 	USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1702 	    "usbprn_close_usb_pipes:");
1703 #ifdef DEBUG
1704 	mutex_enter(&usbprnp->usbprn_mutex);
1705 	ASSERT(bulk_out->ps_flags == USBPRN_PS_IDLE);
1706 	ASSERT(bulk_in->ps_flags == USBPRN_PS_IDLE);
1707 	mutex_exit(&usbprnp->usbprn_mutex);
1708 #endif
1709 
1710 	/*
1711 	 * close the pipe, if another thread is already closing the
1712 	 * pipe, we get USB_INVALID_PIPE
1713 	 */
1714 	if (bulk_out->ps_handle) {
1715 
1716 		USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1717 		    "usbprn_close_usb_pipes: Closing bulk out pipe");
1718 
1719 		usb_pipe_close(usbprnp->usbprn_dip, bulk_out->ps_handle,
1720 		    USB_FLAGS_SLEEP, NULL, NULL);
1721 		bulk_out->ps_handle = NULL;
1722 	}
1723 	if (bulk_in->ps_handle) {
1724 
1725 		USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1726 		    "usbprn_close_usb_pipes: Closing bulk in pipe");
1727 
1728 		usb_pipe_close(usbprnp->usbprn_dip, bulk_in->ps_handle,
1729 		    USB_FLAGS_SLEEP, NULL, NULL);
1730 		bulk_in->ps_handle = NULL;
1731 	}
1732 }
1733 
1734 
1735 /*
1736  * usbprn_getparms:
1737  *	Get the parameters for the device
1738  */
1739 static int
1740 usbprn_getparms(usbprn_state_t *usbprnp, intptr_t arg, int flag)
1741 {
1742 	ASSERT(!(mutex_owned(&usbprnp->usbprn_mutex)));
1743 
1744 	if (ddi_copyout(&usbprnp->usbprn_setparms,
1745 	    (caddr_t)arg, sizeof (struct ecpp_transfer_parms), flag)) {
1746 
1747 		return (EFAULT);
1748 	}
1749 
1750 	return (0);
1751 }
1752 
1753 
1754 /*
1755  * usbprn_setparms:
1756  *	Set the parameters for the device
1757  */
1758 static int
1759 usbprn_setparms(usbprn_state_t *usbprnp, intptr_t arg, int flag)
1760 {
1761 	struct ecpp_transfer_parms xfer;
1762 
1763 	ASSERT(!(mutex_owned(&usbprnp->usbprn_mutex)));
1764 
1765 	if (ddi_copyin((caddr_t)arg, &xfer,
1766 	    sizeof (struct ecpp_transfer_parms), flag)) {
1767 
1768 		return (EFAULT);
1769 	}
1770 	if ((xfer.write_timeout < USBPRN_XFER_TIMEOUT_MIN) ||
1771 	    (xfer.write_timeout > USBPRN_XFER_TIMEOUT_MAX)) {
1772 
1773 		return (EINVAL);
1774 	}
1775 	if (!((xfer.mode == ECPP_CENTRONICS) ||
1776 	    (xfer.mode == ECPP_COMPAT_MODE) ||
1777 	    (xfer.mode == ECPP_NIBBLE_MODE) ||
1778 	    (xfer.mode == ECPP_ECP_MODE) ||
1779 	    (xfer.mode == ECPP_DIAG_MODE))) {
1780 
1781 		return (EINVAL);
1782 
1783 	}
1784 	if (xfer.mode != ECPP_CENTRONICS) {
1785 
1786 		return (EPROTONOSUPPORT);
1787 	}
1788 
1789 	mutex_enter(&usbprnp->usbprn_mutex);
1790 	usbprnp->usbprn_setparms = xfer;
1791 	usbprnp->usbprn_prn_timeouts.tmo_forward = xfer.write_timeout;
1792 	mutex_exit(&usbprnp->usbprn_mutex);
1793 
1794 	return (0);
1795 }
1796 
1797 
1798 /*
1799  * usbprn_geterr:
1800  *	Return the any device error state
1801  */
1802 static void
1803 usbprn_geterr(usbprn_state_t *usbprnp, intptr_t arg, int flag)
1804 {
1805 	struct bpp_error_status bpp_status;
1806 
1807 	bzero(&bpp_status, sizeof (bpp_status));
1808 
1809 	mutex_enter(&usbprnp->usbprn_mutex);
1810 	bpp_status.bus_error = 0;
1811 	bpp_status.timeout_occurred = 0;
1812 	bpp_status.pin_status = usbprn_error_state(usbprnp->usbprn_last_status);
1813 
1814 	USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1815 	    "usbprn_geterr: status=0x%x", usbprnp->usbprn_last_status);
1816 
1817 	mutex_exit(&usbprnp->usbprn_mutex);
1818 
1819 	(void) ddi_copyout(&bpp_status,
1820 	    (caddr_t)arg, sizeof (struct bpp_error_status), flag);
1821 }
1822 
1823 
1824 /*
1825  * usbprn_error_state:
1826  *	Map the driver error state to that of the application
1827  */
1828 static char
1829 usbprn_error_state(uchar_t status)
1830 {
1831 	uchar_t app_err_status = 0;
1832 
1833 	if (!(status & USB_PRINTER_PORT_NO_ERROR)) {
1834 		app_err_status |= USB_PRINTER_ERR_ERR;
1835 	}
1836 	if (status & USB_PRINTER_PORT_EMPTY) {
1837 		app_err_status |= USB_PRINTER_PE_ERR;
1838 	}
1839 	if (!(status & USB_PRINTER_PORT_NO_SELECT)) {
1840 		app_err_status |= USB_PRINTER_SLCT_ERR;
1841 	}
1842 
1843 	return (app_err_status);
1844 }
1845 
1846 
1847 static int
1848 usbprn_ioctl_get_status(usbprn_state_t *usbprnp)
1849 {
1850 	/* Check the transfer mode */
1851 	mutex_enter(&usbprnp->usbprn_mutex);
1852 
1853 	/* if device is disconnected or pipes closed, fail immediately */
1854 	if (!(USBPRN_DEVICE_ACCESS_OK(usbprnp))) {
1855 		mutex_exit(&usbprnp->usbprn_mutex);
1856 
1857 		USB_DPRINTF_L2(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1858 		    "usbprn_ioctl_get_status: device can't be accessed");
1859 
1860 		return (EIO);
1861 	}
1862 	mutex_exit(&usbprnp->usbprn_mutex);
1863 
1864 	if (usbprn_get_port_status(usbprnp) != USB_SUCCESS) {
1865 
1866 		return (EIO);
1867 	}
1868 
1869 	return (0);
1870 }
1871 
1872 
1873 /*
1874  * usbprn_testio:
1875  *	Execute the ECPP_TESTIO ioctl
1876  */
1877 /* ARGSUSED1 */
1878 static int
1879 usbprn_testio(usbprn_state_t *usbprnp, int flag)
1880 {
1881 	int	err;
1882 
1883 	USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1884 	    "usbprn_testio: begin");
1885 
1886 	if ((err = usbprn_ioctl_get_status(usbprnp)) != 0) {
1887 
1888 		return (err);
1889 	}
1890 
1891 	/* There is an error.  Return it to the user */
1892 	mutex_enter(&usbprnp->usbprn_mutex);
1893 
1894 	if (usbprn_error_state(usbprnp->usbprn_last_status) != 0) {
1895 		mutex_exit(&usbprnp->usbprn_mutex);
1896 
1897 		return (EIO);
1898 
1899 	} else {
1900 		mutex_exit(&usbprnp->usbprn_mutex);
1901 
1902 		return (0);
1903 	}
1904 }
1905 
1906 
1907 /*
1908  * usbprn_prnio_get_status:
1909  *	Execute the PRNIOC_GET_STATUS ioctl
1910  */
1911 static int
1912 usbprn_prnio_get_status(usbprn_state_t *usbprnp, intptr_t arg, int flag)
1913 {
1914 	uint_t	prnio_status = 0;
1915 	int	err;
1916 
1917 	USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1918 	    "usbprn_prnio_get_status: begin");
1919 
1920 	/* capture printer status */
1921 	err = usbprn_ioctl_get_status(usbprnp);
1922 
1923 	mutex_enter(&usbprnp->usbprn_mutex);
1924 
1925 	if (usbprnp->usbprn_dev_state == USB_DEV_ONLINE) {
1926 		prnio_status |= PRN_ONLINE;
1927 	}
1928 	if ((err == 0) &&
1929 	    (usbprnp->usbprn_last_status & USB_PRINTER_PORT_NO_ERROR)) {
1930 		prnio_status |= PRN_READY;
1931 	}
1932 
1933 	mutex_exit(&usbprnp->usbprn_mutex);
1934 
1935 	if (ddi_copyout(&prnio_status,
1936 	    (caddr_t)arg, sizeof (prnio_status), flag)) {
1937 
1938 		return (EFAULT);
1939 	}
1940 
1941 	return (0);
1942 }
1943 
1944 
1945 /*
1946  * usbprn_prnio_get_1284_status:
1947  *	Execute the PRNIOC_GET_1284_STATUS ioctl
1948  */
1949 static int
1950 usbprn_prnio_get_1284_status(usbprn_state_t *usbprnp, intptr_t arg, int flag)
1951 {
1952 	uchar_t		status;
1953 	int		err;
1954 
1955 	USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
1956 	    "usbprn_prnio_get_1284_status: begin");
1957 
1958 	if ((err = usbprn_ioctl_get_status(usbprnp)) != 0) {
1959 
1960 		return (err);
1961 	}
1962 
1963 	/* status was captured successfully */
1964 	mutex_enter(&usbprnp->usbprn_mutex);
1965 
1966 	status = usbprnp->usbprn_last_status & (USB_PRINTER_PORT_NO_ERROR |
1967 	    USB_PRINTER_PORT_NO_SELECT | USB_PRINTER_PORT_EMPTY);
1968 
1969 	mutex_exit(&usbprnp->usbprn_mutex);
1970 
1971 	if (ddi_copyout(&status, (caddr_t)arg, sizeof (status), flag)) {
1972 
1973 		return (EFAULT);
1974 	}
1975 
1976 	return (0);
1977 }
1978 
1979 
1980 /*
1981  * usbprn_prnio_get_ifcap:
1982  *	Execute the PRNIOC_GET_IFCAP ioctl
1983  */
1984 /* ARGSUSED */
1985 static int
1986 usbprn_prnio_get_ifcap(usbprn_state_t *usbprnp, intptr_t arg, int flag)
1987 {
1988 	ASSERT(!(mutex_owned(&usbprnp->usbprn_mutex)));
1989 
1990 	if (ddi_copyout(&usbprn_ifcap, (caddr_t)arg, sizeof (usbprn_ifcap),
1991 	    flag)) {
1992 
1993 		return (EFAULT);
1994 	}
1995 
1996 	return (0);
1997 }
1998 
1999 
2000 /*
2001  * usbprn_prnio_get_ifcap:
2002  *	Execute the PRNIOC_SET_IFCAP ioctl
2003  */
2004 /* ARGSUSED */
2005 static int
2006 usbprn_prnio_set_ifcap(usbprn_state_t *usbprnp, intptr_t arg, int flag)
2007 {
2008 	uint_t	new_ifcap;
2009 
2010 	ASSERT(!(mutex_owned(&usbprnp->usbprn_mutex)));
2011 
2012 	if (ddi_copyin((caddr_t)arg, &new_ifcap, sizeof (new_ifcap), flag)) {
2013 
2014 		return (EFAULT);
2015 	}
2016 
2017 	/* no settable capabilities */
2018 	if (usbprn_ifcap != new_ifcap) {
2019 
2020 		return (EINVAL);
2021 	}
2022 
2023 	return (0);
2024 }
2025 
2026 
2027 /*
2028  * usbprn_prnio_get_ifinfo:
2029  *	Execute the PRNIOC_GET_IFINFO ioctl
2030  */
2031 /* ARGSUSED */
2032 static int
2033 usbprn_prnio_get_ifinfo(usbprn_state_t *usbprnp, intptr_t arg, int flag)
2034 {
2035 	struct prn_interface_info	prn_info;
2036 	int	rlen, len;
2037 
2038 	rlen = strlen(usbprn_prnio_ifinfo);
2039 
2040 #ifdef _MULTI_DATAMODEL
2041 	ASSERT(!(mutex_owned(&usbprnp->usbprn_mutex)));
2042 
2043 	switch (ddi_model_convert_from(flag & FMODELS)) {
2044 	case DDI_MODEL_ILP32: {
2045 		struct prn_interface_info32	prn_info32;
2046 
2047 		if (ddi_copyin((caddr_t)arg, &prn_info32,
2048 		    sizeof (struct prn_interface_info32), flag)) {
2049 
2050 			return (EFAULT);
2051 		}
2052 
2053 		prn_info32.if_rlen = rlen;
2054 		len = min(rlen, prn_info32.if_len);
2055 
2056 		if (ddi_copyout(&usbprn_prnio_ifinfo[0],
2057 		    (caddr_t)(uintptr_t)prn_info32.if_data, len, flag)) {
2058 
2059 			return (EFAULT);
2060 		}
2061 
2062 		if (ddi_copyout(&prn_info32, (caddr_t)arg,
2063 		    sizeof (struct prn_interface_info32), flag)) {
2064 
2065 			return (EFAULT);
2066 		}
2067 
2068 		break;
2069 	}
2070 	case DDI_MODEL_NONE:
2071 #endif /* _MULTI_DATAMODEL */
2072 		ASSERT(!(mutex_owned(&usbprnp->usbprn_mutex)));
2073 
2074 		if (ddi_copyin((caddr_t)arg, &prn_info,
2075 		    sizeof (struct prn_interface_info), flag)) {
2076 
2077 			return (EFAULT);
2078 		}
2079 
2080 		prn_info.if_rlen = rlen;
2081 		len = min(rlen, prn_info.if_len);
2082 
2083 		if (ddi_copyout(&usbprn_prnio_ifinfo[0],
2084 		    prn_info.if_data, len, flag)) {
2085 
2086 			return (EFAULT);
2087 		}
2088 
2089 		if (ddi_copyout(&prn_info, (caddr_t)arg,
2090 		    sizeof (struct prn_interface_info), flag)) {
2091 
2092 			return (EFAULT);
2093 		}
2094 #ifdef _MULTI_DATAMODEL
2095 
2096 		break;
2097 	}
2098 #endif /* _MULTI_DATAMODEL */
2099 
2100 	return (0);
2101 }
2102 
2103 
2104 /*
2105  * usbprn_prnio_getdevid:
2106  *	Execute the PRNIOC_GET_1284_DEVID ioctl
2107  */
2108 static int
2109 usbprn_prnio_get_1284_devid(usbprn_state_t *usbprnp, intptr_t arg, int flag)
2110 {
2111 	struct prn_1284_device_id prn_devid;
2112 	int	len;
2113 
2114 	ASSERT(!(mutex_owned(&usbprnp->usbprn_mutex)));
2115 
2116 #ifdef _MULTI_DATAMODEL
2117 	switch (ddi_model_convert_from(flag & FMODELS)) {
2118 	case DDI_MODEL_ILP32: {
2119 		struct prn_1284_device_id32	prn_devid32;
2120 
2121 		if (ddi_copyin((caddr_t)arg, &prn_devid32,
2122 		    sizeof (struct prn_1284_device_id32), flag)) {
2123 
2124 			return (EFAULT);
2125 		}
2126 
2127 		prn_devid32.id_rlen = usbprnp->usbprn_device_id_len - 2;
2128 		len = min(prn_devid32.id_rlen, prn_devid32.id_len);
2129 
2130 		if (ddi_copyout(usbprnp->usbprn_device_id + 2,
2131 		    (caddr_t)(uintptr_t)prn_devid32.id_data, len, flag)) {
2132 
2133 			return (EFAULT);
2134 		}
2135 
2136 		if (ddi_copyout(&prn_devid32, (caddr_t)arg,
2137 		    sizeof (struct prn_1284_device_id32), flag)) {
2138 
2139 			return (EFAULT);
2140 		}
2141 
2142 		break;
2143 	}
2144 	case DDI_MODEL_NONE:
2145 #endif /* _MULTI_DATAMODEL */
2146 		if (ddi_copyin((caddr_t)arg, &prn_devid,
2147 		    sizeof (struct prn_1284_device_id), flag)) {
2148 
2149 			return (EFAULT);
2150 		}
2151 
2152 		prn_devid.id_rlen = usbprnp->usbprn_device_id_len - 2;
2153 		len = min(prn_devid.id_rlen, prn_devid.id_len);
2154 
2155 		if (ddi_copyout(usbprnp->usbprn_device_id + 2,
2156 		    prn_devid.id_data, len, flag)) {
2157 
2158 			return (EFAULT);
2159 		}
2160 
2161 		if (ddi_copyout(&prn_devid, (caddr_t)arg,
2162 		    sizeof (struct prn_1284_device_id), flag)) {
2163 
2164 			return (EFAULT);
2165 		}
2166 #ifdef _MULTI_DATAMODEL
2167 
2168 		break;
2169 	}
2170 #endif /* _MULTI_DATAMODEL */
2171 
2172 	return (0);
2173 }
2174 
2175 
2176 /*
2177  * usbprn_prnio_get_timeouts:
2178  *	Return timeout
2179  */
2180 static int
2181 usbprn_prnio_get_timeouts(usbprn_state_t *usbprnp, intptr_t arg, int flag)
2182 {
2183 	ASSERT(!(mutex_owned(&usbprnp->usbprn_mutex)));
2184 
2185 	if (ddi_copyout(&usbprnp->usbprn_prn_timeouts,
2186 	    (caddr_t)arg, sizeof (struct prn_timeouts), flag)) {
2187 
2188 		return (EFAULT);
2189 	}
2190 
2191 	return (0);
2192 }
2193 
2194 
2195 /*
2196  * usbprn_prnio_set_timeouts:
2197  *	Set write timeout and prn timeout
2198  */
2199 static int
2200 usbprn_prnio_set_timeouts(usbprn_state_t *usbprnp, intptr_t arg, int flag)
2201 {
2202 	struct prn_timeouts prn_timeouts;
2203 
2204 	ASSERT(!(mutex_owned(&usbprnp->usbprn_mutex)));
2205 
2206 	if (ddi_copyin((caddr_t)arg, &prn_timeouts,
2207 	    sizeof (struct prn_timeouts), flag)) {
2208 
2209 		return (EFAULT);
2210 	}
2211 
2212 	if ((prn_timeouts.tmo_forward < USBPRN_XFER_TIMEOUT_MIN) ||
2213 	    (prn_timeouts.tmo_forward > USBPRN_XFER_TIMEOUT_MAX)) {
2214 
2215 		return (EINVAL);
2216 	}
2217 
2218 	mutex_enter(&usbprnp->usbprn_mutex);
2219 
2220 	usbprnp->usbprn_prn_timeouts = prn_timeouts;
2221 	usbprnp->usbprn_setparms.write_timeout = prn_timeouts.tmo_forward;
2222 
2223 	mutex_exit(&usbprnp->usbprn_mutex);
2224 
2225 	return (0);
2226 }
2227 
2228 
2229 /*
2230  * usbprn_biodone:
2231  *	If there is a bp, complete it
2232  */
2233 static void
2234 usbprn_biodone(usbprn_state_t *usbprnp, int err, int bytes_remaining)
2235 {
2236 	struct buf *bp = usbprnp->usbprn_bp;
2237 	usbprn_ps_t	*bulk_out = &usbprnp->usbprn_bulk_out;
2238 	usbprn_ps_t	*bulk_in = &usbprnp->usbprn_bulk_in;
2239 
2240 	ASSERT(mutex_owned(&usbprnp->usbprn_mutex));
2241 
2242 	/* all pipes must be idle now */
2243 	ASSERT(bulk_out->ps_flags == USBPRN_PS_IDLE);
2244 	ASSERT(bulk_in->ps_flags == USBPRN_PS_IDLE);
2245 
2246 	if (bp) {
2247 		bp->b_resid = bytes_remaining;
2248 
2249 		USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
2250 		    "usbprn_biodone: "
2251 		    "bp=0x%p bcount=0x%lx resid=0x%lx remaining=0x%x err=%d",
2252 		    (void *)bp, bp->b_bcount, bp->b_resid, bytes_remaining,
2253 		    err);
2254 
2255 		if (err) {
2256 			bioerror(bp, err);
2257 		}
2258 
2259 		usbprnp->usbprn_bp = NULL;
2260 		biodone(bp);
2261 	}
2262 
2263 	/* release access */
2264 	usb_release_access(usbprnp->usbprn_dev_acc);
2265 }
2266 
2267 
2268 /*
2269  * usbprn_send_async_bulk_data:
2270  *	Send bulk data down to the device through the bulk out pipe
2271  */
2272 static void
2273 usbprn_send_async_bulk_data(usbprn_state_t *usbprnp)
2274 {
2275 	int		rval;
2276 	int		timeout;
2277 	mblk_t		*mp;
2278 	size_t		max_xfer_count, xfer_count;
2279 	usbprn_ps_t	*bulk_out = &usbprnp->usbprn_bulk_out;
2280 	usb_bulk_req_t *req;
2281 
2282 	mutex_enter(&usbprnp->usbprn_mutex);
2283 	ASSERT(bulk_out->ps_flags == USBPRN_PS_NEED_TO_XFER);
2284 
2285 	timeout = usbprnp->usbprn_setparms.write_timeout;
2286 	max_xfer_count = usbprnp->usbprn_bp->b_bcount;
2287 	mp = usbprnp->usbprn_bulk_mp;
2288 	ASSERT(mp != NULL);
2289 	xfer_count = mp->b_wptr - mp->b_rptr;
2290 	mutex_exit(&usbprnp->usbprn_mutex);
2291 
2292 	req = usb_alloc_bulk_req(usbprnp->usbprn_dip, 0, USB_FLAGS_SLEEP);
2293 	req->bulk_len		= xfer_count;
2294 	req->bulk_data		= mp;
2295 	req->bulk_timeout	= timeout;
2296 	req->bulk_cb		= usbprn_bulk_xfer_cb;
2297 	req->bulk_exc_cb	= usbprn_bulk_xfer_exc_cb;
2298 	req->bulk_client_private = (usb_opaque_t)usbprnp;
2299 	req->bulk_attributes	= USB_ATTRS_AUTOCLEARING;
2300 
2301 	USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
2302 	    "usbprn_send_async_bulk_data: req = 0x%p "
2303 	    "max_bulk_xfer_size=%lu mp=0x%p xfer_cnt=%lu timeout=%x",
2304 	    (void *)req, max_xfer_count, (void *)mp, xfer_count, timeout);
2305 
2306 	ASSERT(xfer_count <= max_xfer_count);
2307 
2308 
2309 	if ((rval = usb_pipe_bulk_xfer(bulk_out->ps_handle, req, 0)) !=
2310 	    USB_SUCCESS) {
2311 
2312 		USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
2313 		    "usbprn_send_async_bulk_data: Bulk mp=0x%p "
2314 		    "rval=%d", (void *)mp, rval);
2315 
2316 		mutex_enter(&usbprnp->usbprn_mutex);
2317 		bulk_out->ps_flags = USBPRN_PS_IDLE;
2318 		usbprnp->usbprn_bulk_mp = NULL;
2319 		usbprn_biodone(usbprnp, EIO, 0);
2320 		mutex_exit(&usbprnp->usbprn_mutex);
2321 
2322 		usb_free_bulk_req(req);
2323 	} else {
2324 		mutex_enter(&usbprnp->usbprn_mutex);
2325 		usbprnp->usbprn_bulk_mp = NULL;
2326 		mutex_exit(&usbprnp->usbprn_mutex);
2327 	}
2328 }
2329 
2330 
2331 /*
2332  * usbprn_bulk_xfer_cb
2333  *	Callback for a normal transfer for both bulk pipes.
2334  */
2335 /*ARGSUSED*/
2336 static void
2337 usbprn_bulk_xfer_cb(usb_pipe_handle_t pipe, usb_bulk_req_t *req)
2338 {
2339 	usbprn_state_t	*usbprnp = (usbprn_state_t *)req->bulk_client_private;
2340 	usbprn_ps_t	*bulk_out = &usbprnp->usbprn_bulk_out;
2341 
2342 	ASSERT(usbprnp != NULL);
2343 	ASSERT(!mutex_owned(&usbprnp->usbprn_mutex));
2344 
2345 	mutex_enter(&usbprnp->usbprn_mutex);
2346 
2347 	USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
2348 	    "usbprn_bulk_xfer_cb: mp=0x%p ", (void *)usbprnp->usbprn_bulk_mp);
2349 
2350 	ASSERT(bulk_out->ps_flags == USBPRN_PS_NEED_TO_XFER);
2351 	ASSERT(usbprnp->usbprn_bp != NULL);
2352 	ASSERT((req->bulk_cb_flags & USB_CB_INTR_CONTEXT) == 0);
2353 
2354 	/*
2355 	 * if device is disconnected or driver close called, return
2356 	 * The pipe could be closed, or a timeout could have
2357 	 * come in and the pipe is being reset.  If the
2358 	 * state isn't transferring, then return
2359 	 */
2360 	if (!(USBPRN_DEVICE_ACCESS_OK(usbprnp)) ||
2361 	    (bulk_out->ps_flags != USBPRN_PS_NEED_TO_XFER)) {
2362 		USB_DPRINTF_L3(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
2363 		    "usbprn_bulk_xfer_cb: no access or pipe closed");
2364 
2365 		bulk_out->ps_flags = USBPRN_PS_IDLE;
2366 		usbprn_biodone(usbprnp, EIO, 0);
2367 	} else {
2368 
2369 		/*
2370 		 * data has been xferred, complete the bp.
2371 		 */
2372 		USB_DPRINTF_L3(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
2373 		    "usbprn_bulk_xfer_cb: transaction over");
2374 
2375 		bulk_out->ps_flags = USBPRN_PS_IDLE;
2376 		usbprn_biodone(usbprnp, 0, 0);
2377 	}
2378 
2379 	mutex_exit(&usbprnp->usbprn_mutex);
2380 
2381 	usb_free_bulk_req(req);
2382 }
2383 
2384 
2385 /*
2386  * usbprn_bulk_xfer_exc_cb:
2387  *	Exception callback for the bulk pipes
2388  */
2389 static void
2390 usbprn_bulk_xfer_exc_cb(usb_pipe_handle_t pipe, usb_bulk_req_t *req)
2391 {
2392 	usbprn_state_t	*usbprnp = (usbprn_state_t *)req->bulk_client_private;
2393 	usbprn_ps_t	*bulk_out = &usbprnp->usbprn_bulk_out;
2394 	int		bytes_remaining = 0;
2395 	mblk_t		*data = req->bulk_data;
2396 	usb_cr_t	completion_reason = req->bulk_completion_reason;
2397 	usb_cb_flags_t	cb_flags = req->bulk_cb_flags;
2398 
2399 	USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
2400 	    "usbprn_bulk_xfer_exc_cb: "
2401 	    "pipe=0x%p req=0x%p cr=%d cb_flags=0x%x data=0x%p",
2402 	    (void *)pipe, (void *)req, completion_reason, cb_flags,
2403 	    (void *)data);
2404 
2405 	ASSERT((req->bulk_cb_flags & USB_CB_INTR_CONTEXT) == 0);
2406 	ASSERT(data != NULL);
2407 	mutex_enter(&usbprnp->usbprn_mutex);
2408 
2409 	ASSERT(bulk_out->ps_flags == USBPRN_PS_NEED_TO_XFER);
2410 	bulk_out->ps_flags = USBPRN_PS_IDLE;
2411 	bulk_out->ps_cr = completion_reason;
2412 
2413 	if (data) {
2414 		bytes_remaining = data->b_wptr - data->b_rptr;
2415 	}
2416 
2417 	/*
2418 	 * If the pipe is closed or device not responding or not in
2419 	 * need of transfer, just give up on this bp.
2420 	 */
2421 	if (!(USBPRN_DEVICE_ACCESS_OK(usbprnp)) ||
2422 	    (req->bulk_completion_reason == USB_CR_DEV_NOT_RESP)) {
2423 		USB_DPRINTF_L2(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
2424 		    "usbprn_bulk_xfer_exc_cb: "
2425 		    "device not accesible or wrong state");
2426 		usbprn_biodone(usbprnp, EIO, 0);
2427 	} else {
2428 		if (completion_reason == USB_CR_TIMEOUT) {
2429 			USB_DPRINTF_L2(PRINT_MASK_ALL,
2430 			    usbprnp->usbprn_log_handle,
2431 			    "usbprn_bulk_xfer_exc_cb: timeout error, "
2432 			    "xferred %lu bytes",
2433 			    ((usbprnp->usbprn_bp->b_bcount) -
2434 			    bytes_remaining));
2435 			usbprn_biodone(usbprnp, 0, bytes_remaining);
2436 		} else {
2437 			usbprn_biodone(usbprnp, EIO, 0);
2438 		}
2439 
2440 	}
2441 
2442 	mutex_exit(&usbprnp->usbprn_mutex);
2443 
2444 	usb_free_bulk_req(req);
2445 }
2446 
2447 
2448 /*
2449  * usbprn_reconnect_event_cb:
2450  *	Called upon when the device is hotplugged back; event handling
2451  */
2452 /*ARGSUSED*/
2453 static int
2454 usbprn_reconnect_event_cb(dev_info_t *dip)
2455 {
2456 	usbprn_state_t	*usbprnp =
2457 	    (usbprn_state_t *)ddi_get_soft_state(usbprn_statep,
2458 	    ddi_get_instance(dip));
2459 
2460 	ASSERT(usbprnp != NULL);
2461 
2462 	USB_DPRINTF_L3(PRINT_MASK_EVENTS, usbprnp->usbprn_log_handle,
2463 	    "usbprn_reconnect_event_cb:");
2464 
2465 	(void) usb_serialize_access(usbprnp->usbprn_ser_acc, USB_WAIT, 0);
2466 
2467 	mutex_enter(&usbprnp->usbprn_mutex);
2468 	ASSERT(usbprnp->usbprn_dev_state == USB_DEV_DISCONNECTED);
2469 
2470 	mutex_exit(&usbprnp->usbprn_mutex);
2471 
2472 	usbprn_restore_device_state(dip, usbprnp);
2473 
2474 	if (usbprnp->usbprn_ugen_hdl) {
2475 		(void) usb_ugen_reconnect_ev_cb(usbprnp->usbprn_ugen_hdl);
2476 	}
2477 
2478 	usb_release_access(usbprnp->usbprn_ser_acc);
2479 
2480 	return (USB_SUCCESS);
2481 }
2482 
2483 
2484 /*
2485  * usbprn_disconnect_event_cb:
2486  *	callback for disconnect events
2487  */
2488 /*ARGSUSED*/
2489 static int
2490 usbprn_disconnect_event_cb(dev_info_t *dip)
2491 {
2492 	usbprn_state_t	*usbprnp = (usbprn_state_t *)ddi_get_soft_state(
2493 	    usbprn_statep, ddi_get_instance(dip));
2494 
2495 	USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
2496 	    "usbprn_disconnect_event_cb: Begin");
2497 
2498 	(void) usb_serialize_access(usbprnp->usbprn_ser_acc, USB_WAIT, 0);
2499 
2500 	mutex_enter(&usbprnp->usbprn_mutex);
2501 	usbprnp->usbprn_dev_state = USB_DEV_DISCONNECTED;
2502 
2503 	if (usbprnp->usbprn_flags & USBPRN_OPEN) {
2504 		USB_DPRINTF_L0(PRINT_MASK_EVENTS, usbprnp->usbprn_log_handle,
2505 		    "device was disconnected while open. "
2506 		    "Data may have been lost");
2507 	}
2508 
2509 	/* For now, we set the offline bit in usbprn_last_status */
2510 	usbprnp->usbprn_last_status |= USB_PRINTER_PORT_NO_SELECT;
2511 
2512 	mutex_exit(&usbprnp->usbprn_mutex);
2513 
2514 	if (usbprnp->usbprn_ugen_hdl) {
2515 		(void) usb_ugen_disconnect_ev_cb(usbprnp->usbprn_ugen_hdl);
2516 	}
2517 
2518 	usb_release_access(usbprnp->usbprn_ser_acc);
2519 
2520 	USB_DPRINTF_L4(PRINT_MASK_EVENTS, usbprnp->usbprn_log_handle,
2521 	    "usbprn_disconnect_event_cb: End");
2522 
2523 	return (USB_SUCCESS);
2524 }
2525 
2526 
2527 /*
2528  * usbprn_restore_device_state:
2529  *	set original configuration of the device
2530  *	Restores data xfer
2531  */
2532 static void
2533 usbprn_restore_device_state(dev_info_t *dip, usbprn_state_t *usbprnp)
2534 {
2535 	int alt, rval, iface;
2536 
2537 	USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
2538 	    "usbprn_restore_device_state:");
2539 
2540 	mutex_enter(&usbprnp->usbprn_mutex);
2541 	ASSERT((usbprnp->usbprn_dev_state == USB_DEV_DISCONNECTED) ||
2542 	    (usbprnp->usbprn_dev_state == USB_DEV_SUSPENDED));
2543 
2544 	mutex_exit(&usbprnp->usbprn_mutex);
2545 
2546 	/* Check if we are talking to the same device */
2547 	if (usb_check_same_device(dip, usbprnp->usbprn_log_handle,
2548 	    USB_LOG_L0, PRINT_MASK_ALL,
2549 	    USB_CHK_ALL, NULL) != USB_SUCCESS) {
2550 
2551 		/* change the device state from suspended to disconnected */
2552 		mutex_enter(&usbprnp->usbprn_mutex);
2553 		usbprnp->usbprn_dev_state = USB_DEV_DISCONNECTED;
2554 		mutex_exit(&usbprnp->usbprn_mutex);
2555 
2556 		return;
2557 	}
2558 
2559 	USB_DPRINTF_L0(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
2560 	    "Printer has been reconnected but data may have been lost");
2561 
2562 	mutex_enter(&usbprnp->usbprn_mutex);
2563 
2564 	/* set last status to online */
2565 	usbprnp->usbprn_last_status &= ~USB_PRINTER_PORT_NO_SELECT;
2566 	mutex_exit(&usbprnp->usbprn_mutex);
2567 
2568 	/* Get the port status */
2569 	if (usbprn_get_port_status(usbprnp) != USB_SUCCESS) {
2570 		USB_DPRINTF_L2(PRINT_MASK_ATTA, usbprnp->usbprn_log_handle,
2571 		    "usbprn_restore_device_state: port status failed");
2572 
2573 		return;
2574 	}
2575 
2576 	mutex_enter(&usbprnp->usbprn_mutex);
2577 
2578 	if ((usbprnp->usbprn_last_status & USB_PRINTER_PORT_NO_ERROR) == 0) {
2579 		USB_DPRINTF_L2(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
2580 		    "usbprn_restore_device_state: An error with the printer");
2581 	}
2582 
2583 	if (usbprnp->usbprn_flags & USBPRN_OPEN) {
2584 		mutex_exit(&usbprnp->usbprn_mutex);
2585 		usbprn_close_usb_pipes(usbprnp);
2586 		mutex_enter(&usbprnp->usbprn_mutex);
2587 	}
2588 
2589 	/* restore alternate */
2590 	alt = usbprnp->usbprn_if_descr.bAlternateSetting,
2591 	    mutex_exit(&usbprnp->usbprn_mutex);
2592 
2593 	iface = usb_owns_device(dip) ? 0 :  usb_get_if_number(dip);
2594 	if ((rval = usb_set_alt_if(dip, iface, alt,
2595 	    USB_FLAGS_SLEEP, NULL, NULL)) != USB_SUCCESS) {
2596 		USB_DPRINTF_L2(PRINT_MASK_ATTA, usbprnp->usbprn_log_handle,
2597 		    "usbprn_restore_device_state: set alternate failed (%d)",
2598 		    rval);
2599 
2600 		return;
2601 	}
2602 
2603 	mutex_enter(&usbprnp->usbprn_mutex);
2604 
2605 	if (usbprnp->usbprn_flags & USBPRN_OPEN) {
2606 
2607 		mutex_exit(&usbprnp->usbprn_mutex);
2608 		(void) usbprn_open_usb_pipes(usbprnp);
2609 		mutex_enter(&usbprnp->usbprn_mutex);
2610 	}
2611 
2612 	if (usbprnp->usbprn_pm && usbprnp->usbprn_pm->usbprn_wakeup_enabled) {
2613 		mutex_exit(&usbprnp->usbprn_mutex);
2614 		(void) usb_handle_remote_wakeup(usbprnp->usbprn_dip,
2615 		    USB_REMOTE_WAKEUP_ENABLE);
2616 		mutex_enter(&usbprnp->usbprn_mutex);
2617 	}
2618 
2619 	usbprnp->usbprn_dev_state = USB_DEV_ONLINE;
2620 	mutex_exit(&usbprnp->usbprn_mutex);
2621 
2622 	USB_DPRINTF_L4(PRINT_MASK_ALL, usbprnp->usbprn_log_handle,
2623 	    "usbprn_restore_device_state: End");
2624 }
2625 
2626 
2627 /*
2628  *	Create power managements components
2629  */
2630 static void
2631 usbprn_create_pm_components(dev_info_t *dip, usbprn_state_t *usbprnp)
2632 {
2633 	usbprn_power_t	*usbprnpm;
2634 	uint_t		pwr_states;
2635 
2636 	USB_DPRINTF_L4(PRINT_MASK_PM, usbprnp->usbprn_log_handle,
2637 	    "usbprn_create_pm_components: Begin");
2638 
2639 	/* Allocate the state structure */
2640 	usbprnpm = kmem_zalloc(sizeof (usbprn_power_t),
2641 	    KM_SLEEP);
2642 	usbprnp->usbprn_pm = usbprnpm;
2643 	usbprnpm->usbprn_pm_capabilities = 0;
2644 	usbprnpm->usbprn_current_power = USB_DEV_OS_FULL_PWR;
2645 
2646 	if (usb_create_pm_components(dip, &pwr_states) ==
2647 	    USB_SUCCESS) {
2648 		USB_DPRINTF_L4(PRINT_MASK_PM,
2649 		    usbprnp->usbprn_log_handle,
2650 		    "usbprn_create_pm_components: "
2651 		    "created PM components");
2652 
2653 		if (usb_handle_remote_wakeup(dip,
2654 		    USB_REMOTE_WAKEUP_ENABLE) == USB_SUCCESS) {
2655 			usbprnpm->usbprn_wakeup_enabled = 1;
2656 		}
2657 		usbprnpm->usbprn_pwr_states = (uint8_t)pwr_states;
2658 		(void) pm_raise_power(usbprnp->usbprn_dip, 0,
2659 		    USB_DEV_OS_FULL_PWR);
2660 	} else {
2661 		USB_DPRINTF_L2(PRINT_MASK_PM,
2662 		    usbprnp->usbprn_log_handle,
2663 		    "usbprn_create_pm_components: Failed");
2664 	}
2665 
2666 	USB_DPRINTF_L4(PRINT_MASK_PM, usbprnp->usbprn_log_handle,
2667 	    "usbprn_create_pm_components: END");
2668 }
2669 
2670 
2671 /*
2672  * usbprn_pwrlvl0:
2673  * Functions to handle power transition for OS levels 0 -> 3
2674  */
2675 static int
2676 usbprn_pwrlvl0(usbprn_state_t *usbprnp)
2677 {
2678 	int	rval;
2679 
2680 	USB_DPRINTF_L4(PRINT_MASK_PM, usbprnp->usbprn_log_handle,
2681 	    "usbprn_pwrlvl0:");
2682 
2683 	switch (usbprnp->usbprn_dev_state) {
2684 	case USB_DEV_ONLINE:
2685 		/* Deny the powerdown request if the device is busy */
2686 		if (usbprnp->usbprn_pm->usbprn_pm_busy != 0) {
2687 
2688 			return (USB_FAILURE);
2689 		}
2690 
2691 		/* Issue USB D3 command to the device here */
2692 		rval = usb_set_device_pwrlvl3(usbprnp->usbprn_dip);
2693 		ASSERT(rval == USB_SUCCESS);
2694 
2695 		usbprnp->usbprn_dev_state = USB_DEV_PWRED_DOWN;
2696 		usbprnp->usbprn_pm->usbprn_current_power =
2697 		    USB_DEV_OS_PWR_OFF;
2698 		/* FALLTHRU */
2699 	case USB_DEV_DISCONNECTED:
2700 	case USB_DEV_SUSPENDED:
2701 		/* allow a disconnect/cpr'ed device to go to lower power */
2702 
2703 		return (USB_SUCCESS);
2704 	case USB_DEV_PWRED_DOWN:
2705 	default:
2706 		USB_DPRINTF_L2(PRINT_MASK_PM, usbprnp->usbprn_log_handle,
2707 		    "usbprn_pwrlvl0: illegal dev state");
2708 
2709 		return (USB_FAILURE);
2710 	}
2711 }
2712 
2713 
2714 /*
2715  * usbprn_pwrlvl1:
2716  *	Functions to handle power transition to OS levels -> 2
2717  */
2718 static int
2719 usbprn_pwrlvl1(usbprn_state_t *usbprnp)
2720 {
2721 	int	rval;
2722 
2723 	USB_DPRINTF_L4(PRINT_MASK_PM, usbprnp->usbprn_log_handle,
2724 	    "usbprn_pwrlvl1:");
2725 
2726 	/* Issue USB D2 command to the device here */
2727 	rval = usb_set_device_pwrlvl2(usbprnp->usbprn_dip);
2728 	ASSERT(rval == USB_SUCCESS);
2729 
2730 	return (USB_FAILURE);
2731 }
2732 
2733 
2734 /*
2735  * usbprn_pwrlvl2:
2736  *	Functions to handle power transition to OS levels -> 1
2737  */
2738 static int
2739 usbprn_pwrlvl2(usbprn_state_t *usbprnp)
2740 {
2741 	int	rval;
2742 
2743 	USB_DPRINTF_L4(PRINT_MASK_PM, usbprnp->usbprn_log_handle,
2744 	    "usbprn_pwrlvl2:");
2745 
2746 	/* Issue USB D1 command to the device here */
2747 	rval = usb_set_device_pwrlvl1(usbprnp->usbprn_dip);
2748 	ASSERT(rval == USB_SUCCESS);
2749 
2750 	return (USB_FAILURE);
2751 }
2752 
2753 
2754 /*
2755  * usbprn_pwrlvl3:
2756  *	Functions to handle power transition to OS level -> 0
2757  */
2758 static int
2759 usbprn_pwrlvl3(usbprn_state_t *usbprnp)
2760 {
2761 	USB_DPRINTF_L4(PRINT_MASK_PM, usbprnp->usbprn_log_handle,
2762 	    "usbprn_pwrlvl3:");
2763 
2764 	switch (usbprnp->usbprn_dev_state) {
2765 	case USB_DEV_PWRED_DOWN:
2766 		/* Issue USB D0 command to the device here */
2767 		(void) usb_set_device_pwrlvl0(usbprnp->usbprn_dip);
2768 
2769 		usbprnp->usbprn_dev_state = USB_DEV_ONLINE;
2770 		usbprnp->usbprn_pm->usbprn_current_power =
2771 		    USB_DEV_OS_FULL_PWR;
2772 
2773 		/* FALLTHRU */
2774 	case USB_DEV_ONLINE:
2775 		/* we are already in full power */
2776 		/* FALLTHRU */
2777 	case USB_DEV_DISCONNECTED:
2778 	case USB_DEV_SUSPENDED:
2779 		/*
2780 		 * PM framework tries to put us in full power
2781 		 * during system shutdown. If we are disconnected/cpr'ed
2782 		 * return success anyways
2783 		 */
2784 
2785 		return (USB_SUCCESS);
2786 	default:
2787 		USB_DPRINTF_L4(PRINT_MASK_PM, usbprnp->usbprn_log_handle,
2788 		    "usbprn_pwrlvl3:");
2789 
2790 
2791 		return (USB_FAILURE);
2792 	}
2793 }
2794 
2795 
2796 /*
2797  * usbprn_power :
2798  *	Power entry point
2799  */
2800 /* ARGSUSED */
2801 static int
2802 usbprn_power(dev_info_t *dip, int comp, int level)
2803 {
2804 	usbprn_state_t	*usbprnp;
2805 	usbprn_power_t	*pm;
2806 	int		rval = USB_FAILURE;
2807 
2808 	usbprnp = (usbprn_state_t *)ddi_get_soft_state(usbprn_statep,
2809 	    ddi_get_instance(dip));
2810 
2811 	USB_DPRINTF_L3(PRINT_MASK_PM, usbprnp->usbprn_log_handle,
2812 	    "usbprn_power: Begin: level=%d", level);
2813 
2814 	(void) usb_serialize_access(usbprnp->usbprn_ser_acc, USB_WAIT, 0);
2815 
2816 	mutex_enter(&usbprnp->usbprn_mutex);
2817 	pm = usbprnp->usbprn_pm;
2818 	ASSERT(pm != NULL);
2819 
2820 	/* Check if we are transitioning to a legal power level */
2821 	if (USB_DEV_PWRSTATE_OK(pm->usbprn_pwr_states, level)) {
2822 		USB_DPRINTF_L2(PRINT_MASK_PM, usbprnp->usbprn_log_handle,
2823 		    "usbprn_power: illegal power level=%d "
2824 		    "pwr_states=0x%x", level, pm->usbprn_pwr_states);
2825 
2826 		goto done;
2827 	}
2828 
2829 	switch (level) {
2830 	case USB_DEV_OS_PWR_OFF :
2831 		rval = usbprn_pwrlvl0(usbprnp);
2832 
2833 		break;
2834 	case USB_DEV_OS_PWR_1 :
2835 		rval = usbprn_pwrlvl1(usbprnp);
2836 
2837 		break;
2838 	case USB_DEV_OS_PWR_2 :
2839 		rval = usbprn_pwrlvl2(usbprnp);
2840 
2841 		break;
2842 	case USB_DEV_OS_FULL_PWR :
2843 		rval = usbprn_pwrlvl3(usbprnp);
2844 
2845 		break;
2846 	}
2847 
2848 done:
2849 	mutex_exit(&usbprnp->usbprn_mutex);
2850 
2851 	usb_release_access(usbprnp->usbprn_ser_acc);
2852 
2853 	return ((rval == USB_SUCCESS) ? DDI_SUCCESS : DDI_FAILURE);
2854 }
2855 
2856 
2857 /*
2858  * usbprn_print_long:
2859  *	Breakup a string which is > USBPRN_PRINT_MAXLINE and print it
2860  */
2861 static void
2862 usbprn_print_long(usbprn_state_t *usbprnp, char *str, int len)
2863 {
2864 	char *tmp = str;
2865 	char pbuf[USBPRN_PRINT_MAXLINE];
2866 
2867 	for (;;) {
2868 		if (len <= USBPRN_PRINT_MAXLINE) {
2869 			USB_DPRINTF_L4(PRINT_MASK_ATTA,
2870 			    usbprnp->usbprn_log_handle, "%s", tmp);
2871 
2872 			break;
2873 		} else {
2874 			bcopy(tmp, pbuf, USBPRN_PRINT_MAXLINE);
2875 			USB_DPRINTF_L4(PRINT_MASK_ATTA,
2876 			    usbprnp->usbprn_log_handle, "%s", pbuf);
2877 			tmp += USBPRN_PRINT_MAXLINE;
2878 			len -= USBPRN_PRINT_MAXLINE;
2879 		}
2880 	}
2881 }
2882 
2883 
2884 static void
2885 usbprn_pm_busy_component(usbprn_state_t *usbprn_statep)
2886 {
2887 	ASSERT(!mutex_owned(&usbprn_statep->usbprn_mutex));
2888 	if (usbprn_statep->usbprn_pm != NULL) {
2889 		mutex_enter(&usbprn_statep->usbprn_mutex);
2890 		usbprn_statep->usbprn_pm->usbprn_pm_busy++;
2891 
2892 		USB_DPRINTF_L4(PRINT_MASK_PM, usbprn_statep->usbprn_log_handle,
2893 		    "usbprn_pm_busy_component: %d",
2894 		    usbprn_statep->usbprn_pm->usbprn_pm_busy);
2895 
2896 		mutex_exit(&usbprn_statep->usbprn_mutex);
2897 
2898 		if (pm_busy_component(usbprn_statep->usbprn_dip, 0) !=
2899 		    DDI_SUCCESS) {
2900 			mutex_enter(&usbprn_statep->usbprn_mutex);
2901 			usbprn_statep->usbprn_pm->usbprn_pm_busy--;
2902 
2903 			USB_DPRINTF_L2(PRINT_MASK_PM,
2904 			    usbprn_statep->usbprn_log_handle,
2905 			    "usbprn_pm_busy_component: %d",
2906 			    usbprn_statep->usbprn_pm->usbprn_pm_busy);
2907 
2908 			mutex_exit(&usbprn_statep->usbprn_mutex);
2909 		}
2910 
2911 	}
2912 }
2913 
2914 
2915 static void
2916 usbprn_pm_idle_component(usbprn_state_t *usbprn_statep)
2917 {
2918 	ASSERT(!mutex_owned(&usbprn_statep->usbprn_mutex));
2919 	if (usbprn_statep->usbprn_pm != NULL) {
2920 		if (pm_idle_component(usbprn_statep->usbprn_dip, 0) ==
2921 		    DDI_SUCCESS) {
2922 			mutex_enter(&usbprn_statep->usbprn_mutex);
2923 			ASSERT(usbprn_statep->usbprn_pm->usbprn_pm_busy > 0);
2924 			usbprn_statep->usbprn_pm->usbprn_pm_busy--;
2925 
2926 			USB_DPRINTF_L4(PRINT_MASK_PM,
2927 			    usbprn_statep->usbprn_log_handle,
2928 			    "usbprn_pm_idle_component: %d",
2929 			    usbprn_statep->usbprn_pm->usbprn_pm_busy);
2930 
2931 			mutex_exit(&usbprn_statep->usbprn_mutex);
2932 		}
2933 
2934 	}
2935 }
2936