102ac6454SAndrew Thompson /* $FreeBSD$ */ 202ac6454SAndrew Thompson /*- 302ac6454SAndrew Thompson * Copyright (c) 2008 Hans Petter Selasky. All rights reserved. 402ac6454SAndrew Thompson * 502ac6454SAndrew Thompson * Redistribution and use in source and binary forms, with or without 602ac6454SAndrew Thompson * modification, are permitted provided that the following conditions 702ac6454SAndrew Thompson * are met: 802ac6454SAndrew Thompson * 1. Redistributions of source code must retain the above copyright 902ac6454SAndrew Thompson * notice, this list of conditions and the following disclaimer. 1002ac6454SAndrew Thompson * 2. Redistributions in binary form must reproduce the above copyright 1102ac6454SAndrew Thompson * notice, this list of conditions and the following disclaimer in the 1202ac6454SAndrew Thompson * documentation and/or other materials provided with the distribution. 1302ac6454SAndrew Thompson * 1402ac6454SAndrew Thompson * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 1502ac6454SAndrew Thompson * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 1602ac6454SAndrew Thompson * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 1702ac6454SAndrew Thompson * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 1802ac6454SAndrew Thompson * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 1902ac6454SAndrew Thompson * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 2002ac6454SAndrew Thompson * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 2102ac6454SAndrew Thompson * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 2202ac6454SAndrew Thompson * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 2302ac6454SAndrew Thompson * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 2402ac6454SAndrew Thompson * SUCH DAMAGE. 2502ac6454SAndrew Thompson */ 2602ac6454SAndrew Thompson 27*d2b99310SHans Petter Selasky #ifdef USB_GLOBAL_INCLUDE_FILE 28*d2b99310SHans Petter Selasky #include USB_GLOBAL_INCLUDE_FILE 29*d2b99310SHans Petter Selasky #else 30ed6d949aSAndrew Thompson #include <sys/stdint.h> 31ed6d949aSAndrew Thompson #include <sys/stddef.h> 32ed6d949aSAndrew Thompson #include <sys/param.h> 33ed6d949aSAndrew Thompson #include <sys/queue.h> 34ed6d949aSAndrew Thompson #include <sys/types.h> 35ed6d949aSAndrew Thompson #include <sys/systm.h> 36ed6d949aSAndrew Thompson #include <sys/kernel.h> 37ed6d949aSAndrew Thompson #include <sys/bus.h> 38ed6d949aSAndrew Thompson #include <sys/module.h> 39ed6d949aSAndrew Thompson #include <sys/lock.h> 40ed6d949aSAndrew Thompson #include <sys/mutex.h> 41ed6d949aSAndrew Thompson #include <sys/condvar.h> 42ed6d949aSAndrew Thompson #include <sys/sysctl.h> 43ed6d949aSAndrew Thompson #include <sys/sx.h> 44ed6d949aSAndrew Thompson #include <sys/unistd.h> 45ed6d949aSAndrew Thompson #include <sys/callout.h> 46ed6d949aSAndrew Thompson #include <sys/malloc.h> 47ed6d949aSAndrew Thompson #include <sys/priv.h> 48ed6d949aSAndrew Thompson 4902ac6454SAndrew Thompson #include <dev/usb/usb.h> 50ed6d949aSAndrew Thompson #include <dev/usb/usbdi.h> 5102ac6454SAndrew Thompson 5202ac6454SAndrew Thompson #include <dev/usb/usb_core.h> 5302ac6454SAndrew Thompson #include <dev/usb/usb_debug.h> 5402ac6454SAndrew Thompson #include <dev/usb/usb_process.h> 5502ac6454SAndrew Thompson #include <dev/usb/usb_device.h> 5602ac6454SAndrew Thompson #include <dev/usb/usb_busdma.h> 5702ac6454SAndrew Thompson #include <dev/usb/usb_transfer.h> 5802ac6454SAndrew Thompson 59ed6d949aSAndrew Thompson #include <ddb/ddb.h> 60ed6d949aSAndrew Thompson #include <ddb/db_sym.h> 61*d2b99310SHans Petter Selasky #endif /* USB_GLOBAL_INCLUDE_FILE */ 62ed6d949aSAndrew Thompson 6302ac6454SAndrew Thompson /* 6402ac6454SAndrew Thompson * Define this unconditionally in case a kernel module is loaded that 6502ac6454SAndrew Thompson * has been compiled with debugging options. 6602ac6454SAndrew Thompson */ 67a593f6b8SAndrew Thompson int usb_debug = 0; 6802ac6454SAndrew Thompson 699360ae40SAndrew Thompson SYSCTL_NODE(_hw, OID_AUTO, usb, CTLFLAG_RW, 0, "USB debugging"); 7083cadd7dSHans Petter Selasky SYSCTL_INT(_hw_usb, OID_AUTO, debug, CTLFLAG_RW | CTLFLAG_TUN, 71a593f6b8SAndrew Thompson &usb_debug, 0, "Debug level"); 72c13fd8d4SAndrew Thompson TUNABLE_INT("hw.usb.debug", &usb_debug); 73c13fd8d4SAndrew Thompson 7437506412SHans Petter Selasky #ifdef USB_DEBUG 7537506412SHans Petter Selasky /* 7637506412SHans Petter Selasky * Sysctls to modify timings/delays 7737506412SHans Petter Selasky */ 7837506412SHans Petter Selasky static SYSCTL_NODE(_hw_usb, OID_AUTO, timings, CTLFLAG_RW, 0, "Timings"); 7937506412SHans Petter Selasky static int usb_timings_sysctl_handler(SYSCTL_HANDLER_ARGS); 8037506412SHans Petter Selasky 8137506412SHans Petter Selasky TUNABLE_INT("hw.usb.timings.port_reset_delay", (int *)&usb_port_reset_delay); 8283cadd7dSHans Petter Selasky SYSCTL_PROC(_hw_usb_timings, OID_AUTO, port_reset_delay, CTLTYPE_UINT | CTLFLAG_RW | CTLFLAG_TUN, 8337506412SHans Petter Selasky &usb_port_reset_delay, sizeof(usb_port_reset_delay), 8437506412SHans Petter Selasky usb_timings_sysctl_handler, "IU", "Port Reset Delay"); 8537506412SHans Petter Selasky TUNABLE_INT("hw.usb.timings.port_root_reset_delay", (int *)&usb_port_root_reset_delay); 8683cadd7dSHans Petter Selasky SYSCTL_PROC(_hw_usb_timings, OID_AUTO, port_root_reset_delay, CTLTYPE_UINT | CTLFLAG_RW | CTLFLAG_TUN, 8737506412SHans Petter Selasky &usb_port_root_reset_delay, sizeof(usb_port_root_reset_delay), 8837506412SHans Petter Selasky usb_timings_sysctl_handler, "IU", "Root Port Reset Delay"); 8937506412SHans Petter Selasky TUNABLE_INT("hw.usb.timings.port_reset_recovery", (int *)&usb_port_reset_recovery); 9083cadd7dSHans Petter Selasky SYSCTL_PROC(_hw_usb_timings, OID_AUTO, port_reset_recovery, CTLTYPE_UINT | CTLFLAG_RW | CTLFLAG_TUN, 9137506412SHans Petter Selasky &usb_port_reset_recovery, sizeof(usb_port_reset_recovery), 9237506412SHans Petter Selasky usb_timings_sysctl_handler, "IU", "Port Reset Recovery"); 9337506412SHans Petter Selasky TUNABLE_INT("hw.usb.timings.port_powerup_delay", (int *)&usb_port_powerup_delay); 9483cadd7dSHans Petter Selasky SYSCTL_PROC(_hw_usb_timings, OID_AUTO, port_powerup_delay, CTLTYPE_UINT | CTLFLAG_RW | CTLFLAG_TUN, 9537506412SHans Petter Selasky &usb_port_powerup_delay, sizeof(usb_port_powerup_delay), 9637506412SHans Petter Selasky usb_timings_sysctl_handler, "IU", "Port PowerUp Delay"); 9737506412SHans Petter Selasky TUNABLE_INT("hw.usb.timings.port_resume_delay", (int *)&usb_port_resume_delay); 9883cadd7dSHans Petter Selasky SYSCTL_PROC(_hw_usb_timings, OID_AUTO, port_resume_delay, CTLTYPE_UINT | CTLFLAG_RW | CTLFLAG_TUN, 9937506412SHans Petter Selasky &usb_port_resume_delay, sizeof(usb_port_resume_delay), 10037506412SHans Petter Selasky usb_timings_sysctl_handler, "IU", "Port Resume Delay"); 10137506412SHans Petter Selasky TUNABLE_INT("hw.usb.timings.set_address_settle", (int *)&usb_set_address_settle); 10283cadd7dSHans Petter Selasky SYSCTL_PROC(_hw_usb_timings, OID_AUTO, set_address_settle, CTLTYPE_UINT | CTLFLAG_RW | CTLFLAG_TUN, 10337506412SHans Petter Selasky &usb_set_address_settle, sizeof(usb_set_address_settle), 10437506412SHans Petter Selasky usb_timings_sysctl_handler, "IU", "Set Address Settle"); 10537506412SHans Petter Selasky TUNABLE_INT("hw.usb.timings.resume_delay", (int *)&usb_resume_delay); 10683cadd7dSHans Petter Selasky SYSCTL_PROC(_hw_usb_timings, OID_AUTO, resume_delay, CTLTYPE_UINT | CTLFLAG_RW | CTLFLAG_TUN, 10737506412SHans Petter Selasky &usb_resume_delay, sizeof(usb_resume_delay), 10837506412SHans Petter Selasky usb_timings_sysctl_handler, "IU", "Resume Delay"); 10937506412SHans Petter Selasky TUNABLE_INT("hw.usb.timings.resume_wait", (int *)&usb_resume_wait); 11083cadd7dSHans Petter Selasky SYSCTL_PROC(_hw_usb_timings, OID_AUTO, resume_wait, CTLTYPE_UINT | CTLFLAG_RW | CTLFLAG_TUN, 11137506412SHans Petter Selasky &usb_resume_wait, sizeof(usb_resume_wait), 11237506412SHans Petter Selasky usb_timings_sysctl_handler, "IU", "Resume Wait"); 11337506412SHans Petter Selasky TUNABLE_INT("hw.usb.timings.resume_recovery", (int *)&usb_resume_recovery); 11483cadd7dSHans Petter Selasky SYSCTL_PROC(_hw_usb_timings, OID_AUTO, resume_recovery, CTLTYPE_UINT | CTLFLAG_RW | CTLFLAG_TUN, 11537506412SHans Petter Selasky &usb_resume_recovery, sizeof(usb_resume_recovery), 11637506412SHans Petter Selasky usb_timings_sysctl_handler, "IU", "Resume Recovery"); 11737506412SHans Petter Selasky TUNABLE_INT("hw.usb.timings.extra_power_up_time", (int *)&usb_extra_power_up_time); 11883cadd7dSHans Petter Selasky SYSCTL_PROC(_hw_usb_timings, OID_AUTO, extra_power_up_time, CTLTYPE_UINT | CTLFLAG_RW | CTLFLAG_TUN, 11937506412SHans Petter Selasky &usb_extra_power_up_time, sizeof(usb_extra_power_up_time), 12037506412SHans Petter Selasky usb_timings_sysctl_handler, "IU", "Extra PowerUp Time"); 12137506412SHans Petter Selasky #endif 12237506412SHans Petter Selasky 12302ac6454SAndrew Thompson /*------------------------------------------------------------------------* 124a593f6b8SAndrew Thompson * usb_dump_iface 12502ac6454SAndrew Thompson * 12602ac6454SAndrew Thompson * This function dumps information about an USB interface. 12702ac6454SAndrew Thompson *------------------------------------------------------------------------*/ 12802ac6454SAndrew Thompson void 129a593f6b8SAndrew Thompson usb_dump_iface(struct usb_interface *iface) 13002ac6454SAndrew Thompson { 131a593f6b8SAndrew Thompson printf("usb_dump_iface: iface=%p\n", iface); 13202ac6454SAndrew Thompson if (iface == NULL) { 13302ac6454SAndrew Thompson return; 13402ac6454SAndrew Thompson } 13502ac6454SAndrew Thompson printf(" iface=%p idesc=%p altindex=%d\n", 13602ac6454SAndrew Thompson iface, iface->idesc, iface->alt_index); 13702ac6454SAndrew Thompson } 13802ac6454SAndrew Thompson 13902ac6454SAndrew Thompson /*------------------------------------------------------------------------* 140a593f6b8SAndrew Thompson * usb_dump_device 14102ac6454SAndrew Thompson * 14202ac6454SAndrew Thompson * This function dumps information about an USB device. 14302ac6454SAndrew Thompson *------------------------------------------------------------------------*/ 14402ac6454SAndrew Thompson void 145a593f6b8SAndrew Thompson usb_dump_device(struct usb_device *udev) 14602ac6454SAndrew Thompson { 147a593f6b8SAndrew Thompson printf("usb_dump_device: dev=%p\n", udev); 14802ac6454SAndrew Thompson if (udev == NULL) { 14902ac6454SAndrew Thompson return; 15002ac6454SAndrew Thompson } 15102ac6454SAndrew Thompson printf(" bus=%p \n" 15202ac6454SAndrew Thompson " address=%d config=%d depth=%d speed=%d self_powered=%d\n" 15302ac6454SAndrew Thompson " power=%d langid=%d\n", 15402ac6454SAndrew Thompson udev->bus, 15502ac6454SAndrew Thompson udev->address, udev->curr_config_no, udev->depth, udev->speed, 15602ac6454SAndrew Thompson udev->flags.self_powered, udev->power, udev->langid); 15702ac6454SAndrew Thompson } 15802ac6454SAndrew Thompson 15902ac6454SAndrew Thompson /*------------------------------------------------------------------------* 160a593f6b8SAndrew Thompson * usb_dump_queue 16102ac6454SAndrew Thompson * 162ae60fdfbSAndrew Thompson * This function dumps the USB transfer that are queued up on an USB endpoint. 16302ac6454SAndrew Thompson *------------------------------------------------------------------------*/ 16402ac6454SAndrew Thompson void 165a593f6b8SAndrew Thompson usb_dump_queue(struct usb_endpoint *ep) 16602ac6454SAndrew Thompson { 167760bc48eSAndrew Thompson struct usb_xfer *xfer; 168a5cf1aaaSHans Petter Selasky usb_stream_t x; 16902ac6454SAndrew Thompson 170a593f6b8SAndrew Thompson printf("usb_dump_queue: endpoint=%p xfer: ", ep); 171a5cf1aaaSHans Petter Selasky for (x = 0; x != USB_MAX_EP_STREAMS; x++) { 172a5cf1aaaSHans Petter Selasky TAILQ_FOREACH(xfer, &ep->endpoint_q[x].head, wait_entry) 17302ac6454SAndrew Thompson printf(" %p", xfer); 17402ac6454SAndrew Thompson } 17502ac6454SAndrew Thompson printf("\n"); 17602ac6454SAndrew Thompson } 17702ac6454SAndrew Thompson 17802ac6454SAndrew Thompson /*------------------------------------------------------------------------* 179a593f6b8SAndrew Thompson * usb_dump_endpoint 18002ac6454SAndrew Thompson * 181ae60fdfbSAndrew Thompson * This function dumps information about an USB endpoint. 18202ac6454SAndrew Thompson *------------------------------------------------------------------------*/ 18302ac6454SAndrew Thompson void 184a593f6b8SAndrew Thompson usb_dump_endpoint(struct usb_endpoint *ep) 18502ac6454SAndrew Thompson { 186ae60fdfbSAndrew Thompson if (ep) { 187a593f6b8SAndrew Thompson printf("usb_dump_endpoint: endpoint=%p", ep); 18802ac6454SAndrew Thompson 18902ac6454SAndrew Thompson printf(" edesc=%p isoc_next=%d toggle_next=%d", 190ae60fdfbSAndrew Thompson ep->edesc, ep->isoc_next, ep->toggle_next); 19102ac6454SAndrew Thompson 192ae60fdfbSAndrew Thompson if (ep->edesc) { 19302ac6454SAndrew Thompson printf(" bEndpointAddress=0x%02x", 194ae60fdfbSAndrew Thompson ep->edesc->bEndpointAddress); 19502ac6454SAndrew Thompson } 19602ac6454SAndrew Thompson printf("\n"); 197a593f6b8SAndrew Thompson usb_dump_queue(ep); 19802ac6454SAndrew Thompson } else { 199a593f6b8SAndrew Thompson printf("usb_dump_endpoint: endpoint=NULL\n"); 20002ac6454SAndrew Thompson } 20102ac6454SAndrew Thompson } 20202ac6454SAndrew Thompson 20302ac6454SAndrew Thompson /*------------------------------------------------------------------------* 204a593f6b8SAndrew Thompson * usb_dump_xfer 20502ac6454SAndrew Thompson * 20602ac6454SAndrew Thompson * This function dumps information about an USB transfer. 20702ac6454SAndrew Thompson *------------------------------------------------------------------------*/ 20802ac6454SAndrew Thompson void 209a593f6b8SAndrew Thompson usb_dump_xfer(struct usb_xfer *xfer) 21002ac6454SAndrew Thompson { 211760bc48eSAndrew Thompson struct usb_device *udev; 212a593f6b8SAndrew Thompson printf("usb_dump_xfer: xfer=%p\n", xfer); 21302ac6454SAndrew Thompson if (xfer == NULL) { 21402ac6454SAndrew Thompson return; 21502ac6454SAndrew Thompson } 216ae60fdfbSAndrew Thompson if (xfer->endpoint == NULL) { 217ae60fdfbSAndrew Thompson printf("xfer %p: endpoint=NULL\n", 21802ac6454SAndrew Thompson xfer); 21902ac6454SAndrew Thompson return; 22002ac6454SAndrew Thompson } 22102ac6454SAndrew Thompson udev = xfer->xroot->udev; 22202ac6454SAndrew Thompson printf("xfer %p: udev=%p vid=0x%04x pid=0x%04x addr=%d " 223ae60fdfbSAndrew Thompson "endpoint=%p ep=0x%02x attr=0x%02x\n", 22402ac6454SAndrew Thompson xfer, udev, 22502ac6454SAndrew Thompson UGETW(udev->ddesc.idVendor), 22602ac6454SAndrew Thompson UGETW(udev->ddesc.idProduct), 227ae60fdfbSAndrew Thompson udev->address, xfer->endpoint, 228ae60fdfbSAndrew Thompson xfer->endpoint->edesc->bEndpointAddress, 229ae60fdfbSAndrew Thompson xfer->endpoint->edesc->bmAttributes); 23002ac6454SAndrew Thompson } 23137506412SHans Petter Selasky 23237506412SHans Petter Selasky #ifdef USB_DEBUG 23337506412SHans Petter Selasky unsigned int usb_port_reset_delay = USB_PORT_RESET_DELAY; 23437506412SHans Petter Selasky unsigned int usb_port_root_reset_delay = USB_PORT_ROOT_RESET_DELAY; 23537506412SHans Petter Selasky unsigned int usb_port_reset_recovery = USB_PORT_RESET_RECOVERY; 23637506412SHans Petter Selasky unsigned int usb_port_powerup_delay = USB_PORT_POWERUP_DELAY; 23737506412SHans Petter Selasky unsigned int usb_port_resume_delay = USB_PORT_RESUME_DELAY; 23837506412SHans Petter Selasky unsigned int usb_set_address_settle = USB_SET_ADDRESS_SETTLE; 23937506412SHans Petter Selasky unsigned int usb_resume_delay = USB_RESUME_DELAY; 24037506412SHans Petter Selasky unsigned int usb_resume_wait = USB_RESUME_WAIT; 24137506412SHans Petter Selasky unsigned int usb_resume_recovery = USB_RESUME_RECOVERY; 24237506412SHans Petter Selasky unsigned int usb_extra_power_up_time = USB_EXTRA_POWER_UP_TIME; 24337506412SHans Petter Selasky 24437506412SHans Petter Selasky /*------------------------------------------------------------------------* 24537506412SHans Petter Selasky * usb_timings_sysctl_handler 24637506412SHans Petter Selasky * 24737506412SHans Petter Selasky * This function updates timings variables, adjusting them where necessary. 24837506412SHans Petter Selasky *------------------------------------------------------------------------*/ 24937506412SHans Petter Selasky static int usb_timings_sysctl_handler(SYSCTL_HANDLER_ARGS) 25037506412SHans Petter Selasky { 25137506412SHans Petter Selasky int error = 0; 25237506412SHans Petter Selasky unsigned int val; 25337506412SHans Petter Selasky 25437506412SHans Petter Selasky /* 25537506412SHans Petter Selasky * Attempt to get a coherent snapshot by making a copy of the data. 25637506412SHans Petter Selasky */ 25737506412SHans Petter Selasky if (arg1) 25837506412SHans Petter Selasky val = *(unsigned int *)arg1; 25937506412SHans Petter Selasky else 26037506412SHans Petter Selasky val = arg2; 26137506412SHans Petter Selasky error = SYSCTL_OUT(req, &val, sizeof(int)); 26237506412SHans Petter Selasky if (error || !req->newptr) 26337506412SHans Petter Selasky return (error); 26437506412SHans Petter Selasky 26537506412SHans Petter Selasky if (!arg1) 26637506412SHans Petter Selasky return EPERM; 26737506412SHans Petter Selasky 26837506412SHans Petter Selasky error = SYSCTL_IN(req, &val, sizeof(unsigned int)); 26937506412SHans Petter Selasky if (error) 27037506412SHans Petter Selasky return (error); 27137506412SHans Petter Selasky 27237506412SHans Petter Selasky /* 27337506412SHans Petter Selasky * Now make sure the values are decent, and certainly no lower than 27437506412SHans Petter Selasky * what the USB spec prescribes. 27537506412SHans Petter Selasky */ 27637506412SHans Petter Selasky unsigned int *p = (unsigned int *)arg1; 27737506412SHans Petter Selasky if (p == &usb_port_reset_delay) { 27837506412SHans Petter Selasky if (val < USB_PORT_RESET_DELAY_SPEC) 27937506412SHans Petter Selasky return (EINVAL); 28037506412SHans Petter Selasky } else if (p == &usb_port_root_reset_delay) { 28137506412SHans Petter Selasky if (val < USB_PORT_ROOT_RESET_DELAY_SPEC) 28237506412SHans Petter Selasky return (EINVAL); 28337506412SHans Petter Selasky } else if (p == &usb_port_reset_recovery) { 28437506412SHans Petter Selasky if (val < USB_PORT_RESET_RECOVERY_SPEC) 28537506412SHans Petter Selasky return (EINVAL); 28637506412SHans Petter Selasky } else if (p == &usb_port_powerup_delay) { 28737506412SHans Petter Selasky if (val < USB_PORT_POWERUP_DELAY_SPEC) 28837506412SHans Petter Selasky return (EINVAL); 28937506412SHans Petter Selasky } else if (p == &usb_port_resume_delay) { 29037506412SHans Petter Selasky if (val < USB_PORT_RESUME_DELAY_SPEC) 29137506412SHans Petter Selasky return (EINVAL); 29237506412SHans Petter Selasky } else if (p == &usb_set_address_settle) { 29337506412SHans Petter Selasky if (val < USB_SET_ADDRESS_SETTLE_SPEC) 29437506412SHans Petter Selasky return (EINVAL); 29537506412SHans Petter Selasky } else if (p == &usb_resume_delay) { 29637506412SHans Petter Selasky if (val < USB_RESUME_DELAY_SPEC) 29737506412SHans Petter Selasky return (EINVAL); 29837506412SHans Petter Selasky } else if (p == &usb_resume_wait) { 29937506412SHans Petter Selasky if (val < USB_RESUME_WAIT_SPEC) 30037506412SHans Petter Selasky return (EINVAL); 30137506412SHans Petter Selasky } else if (p == &usb_resume_recovery) { 30237506412SHans Petter Selasky if (val < USB_RESUME_RECOVERY_SPEC) 30337506412SHans Petter Selasky return (EINVAL); 30437506412SHans Petter Selasky } else if (p == &usb_extra_power_up_time) { 30537506412SHans Petter Selasky if (val < USB_EXTRA_POWER_UP_TIME_SPEC) 30637506412SHans Petter Selasky return (EINVAL); 30737506412SHans Petter Selasky } else { 30837506412SHans Petter Selasky /* noop */ 30937506412SHans Petter Selasky } 31037506412SHans Petter Selasky 31137506412SHans Petter Selasky *p = val; 31237506412SHans Petter Selasky return 0; 31337506412SHans Petter Selasky } 31437506412SHans Petter Selasky #endif 315