xref: /dragonfly/sys/bus/u4b/usb_dynamic.c (revision 2b3f93ea)
17810100dSMarkus Pfeiffer /* $FreeBSD: head/sys/dev/usb/usb_dynamic.c 250207 2013-05-03 11:10:04Z hselasky $ */
212bd3c8bSSascha Wildner /*-
312bd3c8bSSascha Wildner  * Copyright (c) 2008 Hans Petter Selasky. All rights reserved.
412bd3c8bSSascha Wildner  *
512bd3c8bSSascha Wildner  * Redistribution and use in source and binary forms, with or without
612bd3c8bSSascha Wildner  * modification, are permitted provided that the following conditions
712bd3c8bSSascha Wildner  * are met:
812bd3c8bSSascha Wildner  * 1. Redistributions of source code must retain the above copyright
912bd3c8bSSascha Wildner  *    notice, this list of conditions and the following disclaimer.
1012bd3c8bSSascha Wildner  * 2. Redistributions in binary form must reproduce the above copyright
1112bd3c8bSSascha Wildner  *    notice, this list of conditions and the following disclaimer in the
1212bd3c8bSSascha Wildner  *    documentation and/or other materials provided with the distribution.
1312bd3c8bSSascha Wildner  *
1412bd3c8bSSascha Wildner  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
1512bd3c8bSSascha Wildner  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
1612bd3c8bSSascha Wildner  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
1712bd3c8bSSascha Wildner  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
1812bd3c8bSSascha Wildner  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
1912bd3c8bSSascha Wildner  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
2012bd3c8bSSascha Wildner  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
2112bd3c8bSSascha Wildner  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
2212bd3c8bSSascha Wildner  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
2312bd3c8bSSascha Wildner  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
2412bd3c8bSSascha Wildner  * SUCH DAMAGE.
2512bd3c8bSSascha Wildner  */
2612bd3c8bSSascha Wildner 
2712bd3c8bSSascha Wildner #include <sys/stdint.h>
2812bd3c8bSSascha Wildner #include <sys/param.h>
2912bd3c8bSSascha Wildner #include <sys/queue.h>
3012bd3c8bSSascha Wildner #include <sys/types.h>
3112bd3c8bSSascha Wildner #include <sys/systm.h>
3212bd3c8bSSascha Wildner #include <sys/kernel.h>
3312bd3c8bSSascha Wildner #include <sys/bus.h>
3412bd3c8bSSascha Wildner #include <sys/module.h>
3512bd3c8bSSascha Wildner #include <sys/lock.h>
3612bd3c8bSSascha Wildner #include <sys/condvar.h>
3712bd3c8bSSascha Wildner #include <sys/sysctl.h>
3812bd3c8bSSascha Wildner #include <sys/unistd.h>
3912bd3c8bSSascha Wildner #include <sys/callout.h>
4012bd3c8bSSascha Wildner #include <sys/malloc.h>
41*2b3f93eaSMatthew Dillon #include <sys/caps.h>
4212bd3c8bSSascha Wildner 
43722d05c3SSascha Wildner #include <bus/u4b/usb.h>
44722d05c3SSascha Wildner #include <bus/u4b/usbdi.h>
4512bd3c8bSSascha Wildner 
46722d05c3SSascha Wildner #include <bus/u4b/usb_core.h>
47722d05c3SSascha Wildner #include <bus/u4b/usb_process.h>
48722d05c3SSascha Wildner #include <bus/u4b/usb_device.h>
49722d05c3SSascha Wildner #include <bus/u4b/usb_dynamic.h>
5057bed822SMarkus Pfeiffer #include <bus/u4b/usb_request.h>
5112bd3c8bSSascha Wildner 
5212bd3c8bSSascha Wildner /* function prototypes */
5312bd3c8bSSascha Wildner static usb_handle_req_t usb_temp_get_desc_w;
5412bd3c8bSSascha Wildner static usb_temp_setup_by_index_t usb_temp_setup_by_index_w;
5512bd3c8bSSascha Wildner static usb_temp_unsetup_t usb_temp_unsetup_w;
5612bd3c8bSSascha Wildner static usb_test_quirk_t usb_test_quirk_w;
5712bd3c8bSSascha Wildner static usb_quirk_ioctl_t usb_quirk_ioctl_w;
5812bd3c8bSSascha Wildner 
5912bd3c8bSSascha Wildner /* global variables */
6012bd3c8bSSascha Wildner usb_handle_req_t *usb_temp_get_desc_p = &usb_temp_get_desc_w;
6112bd3c8bSSascha Wildner usb_temp_setup_by_index_t *usb_temp_setup_by_index_p = &usb_temp_setup_by_index_w;
6212bd3c8bSSascha Wildner usb_temp_unsetup_t *usb_temp_unsetup_p = &usb_temp_unsetup_w;
6312bd3c8bSSascha Wildner usb_test_quirk_t *usb_test_quirk_p = &usb_test_quirk_w;
6412bd3c8bSSascha Wildner usb_quirk_ioctl_t *usb_quirk_ioctl_p = &usb_quirk_ioctl_w;
6512bd3c8bSSascha Wildner devclass_t usb_devclass_ptr = NULL;
6612bd3c8bSSascha Wildner 
6712bd3c8bSSascha Wildner static usb_error_t
usb_temp_setup_by_index_w(struct usb_device * udev,uint16_t index)6812bd3c8bSSascha Wildner usb_temp_setup_by_index_w(struct usb_device *udev, uint16_t index)
6912bd3c8bSSascha Wildner {
7012bd3c8bSSascha Wildner 	return (USB_ERR_INVAL);
7112bd3c8bSSascha Wildner }
7212bd3c8bSSascha Wildner 
7312bd3c8bSSascha Wildner static uint8_t
usb_test_quirk_w(const struct usbd_lookup_info * info,uint16_t quirk)7412bd3c8bSSascha Wildner usb_test_quirk_w(const struct usbd_lookup_info *info, uint16_t quirk)
7512bd3c8bSSascha Wildner {
7612bd3c8bSSascha Wildner 	return (0);			/* no match */
7712bd3c8bSSascha Wildner }
7812bd3c8bSSascha Wildner 
7912bd3c8bSSascha Wildner static int
usb_quirk_ioctl_w(unsigned long cmd,caddr_t data,int fflag,struct thread * td)8012bd3c8bSSascha Wildner usb_quirk_ioctl_w(unsigned long cmd, caddr_t data, int fflag, struct thread *td)
8112bd3c8bSSascha Wildner {
8212bd3c8bSSascha Wildner 	return (ENOIOCTL);
8312bd3c8bSSascha Wildner }
8412bd3c8bSSascha Wildner 
8512bd3c8bSSascha Wildner static usb_error_t
usb_temp_get_desc_w(struct usb_device * udev,struct usb_device_request * req,const void ** pPtr,uint16_t * pLength)8612bd3c8bSSascha Wildner usb_temp_get_desc_w(struct usb_device *udev, struct usb_device_request *req, const void **pPtr, uint16_t *pLength)
8712bd3c8bSSascha Wildner {
8812bd3c8bSSascha Wildner 	/* stall */
8912bd3c8bSSascha Wildner 	return (USB_ERR_STALLED);
9012bd3c8bSSascha Wildner }
9112bd3c8bSSascha Wildner 
9212bd3c8bSSascha Wildner static void
usb_temp_unsetup_w(struct usb_device * udev)9312bd3c8bSSascha Wildner usb_temp_unsetup_w(struct usb_device *udev)
9412bd3c8bSSascha Wildner {
9512bd3c8bSSascha Wildner 	if(udev->usb_template_ptr) {
9657bed822SMarkus Pfeiffer 		usbd_free_config_desc(udev, udev->usb_template_ptr);
9712bd3c8bSSascha Wildner 		udev->usb_template_ptr = NULL;
9812bd3c8bSSascha Wildner 	}
9912bd3c8bSSascha Wildner }
10012bd3c8bSSascha Wildner 
10112bd3c8bSSascha Wildner void
usb_quirk_unload(void * arg)10212bd3c8bSSascha Wildner usb_quirk_unload(void *arg)
10312bd3c8bSSascha Wildner {
10412bd3c8bSSascha Wildner 	/* reset function pointers */
10512bd3c8bSSascha Wildner 
10612bd3c8bSSascha Wildner 	usb_test_quirk_p = &usb_test_quirk_w;
10712bd3c8bSSascha Wildner 	usb_quirk_ioctl_p = &usb_quirk_ioctl_w;
10812bd3c8bSSascha Wildner 
10912bd3c8bSSascha Wildner 	/* wait for CPU to exit the loaded functions, if any */
11012bd3c8bSSascha Wildner 
11112bd3c8bSSascha Wildner 	/* XXX this is a tradeoff */
11263da4a34SSascha Wildner 
113722d05c3SSascha Wildner 	tsleep(usb_quirk_unload, 0, "WAIT", hz);
11412bd3c8bSSascha Wildner }
11512bd3c8bSSascha Wildner 
11612bd3c8bSSascha Wildner void
usb_temp_unload(void * arg)11712bd3c8bSSascha Wildner usb_temp_unload(void *arg)
11812bd3c8bSSascha Wildner {
11912bd3c8bSSascha Wildner 	/* reset function pointers */
12012bd3c8bSSascha Wildner 
12112bd3c8bSSascha Wildner 	usb_temp_get_desc_p = &usb_temp_get_desc_w;
12212bd3c8bSSascha Wildner 	usb_temp_setup_by_index_p = &usb_temp_setup_by_index_w;
12312bd3c8bSSascha Wildner 	usb_temp_unsetup_p = &usb_temp_unsetup_w;
12412bd3c8bSSascha Wildner 
12512bd3c8bSSascha Wildner 	/* wait for CPU to exit the loaded functions, if any */
12612bd3c8bSSascha Wildner 
12712bd3c8bSSascha Wildner 	/* XXX this is a tradeoff */
12812bd3c8bSSascha Wildner 
129722d05c3SSascha Wildner 	tsleep(usb_quirk_unload, 0, "WAIT", hz);
13012bd3c8bSSascha Wildner }
13112bd3c8bSSascha Wildner 
13212bd3c8bSSascha Wildner void
usb_bus_unload(void * arg)13312bd3c8bSSascha Wildner usb_bus_unload(void *arg)
13412bd3c8bSSascha Wildner {
13512bd3c8bSSascha Wildner 	/* reset function pointers */
13612bd3c8bSSascha Wildner 
13712bd3c8bSSascha Wildner 	usb_devclass_ptr = NULL;
13812bd3c8bSSascha Wildner 
13912bd3c8bSSascha Wildner 	/* wait for CPU to exit the loaded functions, if any */
14012bd3c8bSSascha Wildner 
14112bd3c8bSSascha Wildner 	/* XXX this is a tradeoff */
14212bd3c8bSSascha Wildner 
143722d05c3SSascha Wildner 	tsleep(usb_quirk_unload, 0, "WAIT", hz);
14412bd3c8bSSascha Wildner }
145