xref: /dflybsd-src/sys/bus/u4b/usb_pf.c (revision bb54c3a2a4a6207be408250820bde132dab4af2d)
17df2ba56SMarkus Pfeiffer /* $FreeBSD: head/sys/dev/usb/usb_pf.c 265779 2014-05-09 14:28:11Z hselasky $ */
212bd3c8bSSascha Wildner /*-
312bd3c8bSSascha Wildner  * Copyright (c) 1990, 1991, 1993
412bd3c8bSSascha Wildner  *	The Regents of the University of California.  All rights reserved.
512bd3c8bSSascha Wildner  *
612bd3c8bSSascha Wildner  * This code is derived from the Stanford/CMU enet packet filter,
712bd3c8bSSascha Wildner  * (net/enet.c) distributed as part of 4.3BSD, and code contributed
812bd3c8bSSascha Wildner  * to Berkeley by Steven McCanne and Van Jacobson both of Lawrence
912bd3c8bSSascha Wildner  * Berkeley Laboratory.
1012bd3c8bSSascha Wildner  *
1112bd3c8bSSascha Wildner  * Redistribution and use in source and binary forms, with or without
1212bd3c8bSSascha Wildner  * modification, are permitted provided that the following conditions
1312bd3c8bSSascha Wildner  * are met:
1412bd3c8bSSascha Wildner  * 1. Redistributions of source code must retain the above copyright
1512bd3c8bSSascha Wildner  *    notice, this list of conditions and the following disclaimer.
1612bd3c8bSSascha Wildner  * 2. Redistributions in binary form must reproduce the above copyright
1712bd3c8bSSascha Wildner  *    notice, this list of conditions and the following disclaimer in the
1812bd3c8bSSascha Wildner  *    documentation and/or other materials provided with the distribution.
19dc71b7abSJustin C. Sherrill  * 3. Neither the name of the University nor the names of its contributors
2012bd3c8bSSascha Wildner  *    may be used to endorse or promote products derived from this software
2112bd3c8bSSascha Wildner  *    without specific prior written permission.
2212bd3c8bSSascha Wildner  *
2312bd3c8bSSascha Wildner  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
2412bd3c8bSSascha Wildner  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
2512bd3c8bSSascha Wildner  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
2612bd3c8bSSascha Wildner  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
2712bd3c8bSSascha Wildner  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
2812bd3c8bSSascha Wildner  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
2912bd3c8bSSascha Wildner  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
3012bd3c8bSSascha Wildner  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
3112bd3c8bSSascha Wildner  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
3212bd3c8bSSascha Wildner  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
3312bd3c8bSSascha Wildner  * SUCH DAMAGE.
3412bd3c8bSSascha Wildner  */
3512bd3c8bSSascha Wildner 
3612bd3c8bSSascha Wildner #include <sys/param.h>
3712bd3c8bSSascha Wildner #include <sys/kernel.h>
3812bd3c8bSSascha Wildner #include <sys/bus.h>
3912bd3c8bSSascha Wildner #include <sys/fcntl.h>
4012bd3c8bSSascha Wildner #include <sys/malloc.h>
4112bd3c8bSSascha Wildner #include <sys/proc.h>
4212bd3c8bSSascha Wildner #include <sys/socket.h>
4312bd3c8bSSascha Wildner #include <sys/sockio.h>
4412bd3c8bSSascha Wildner #include <net/if.h>
45bff82488SAaron LI #include <net/if_var.h>
46f5199fbfSMarkus Pfeiffer #include <net/if_clone.h>
4712bd3c8bSSascha Wildner #include <net/if_types.h>
48f5199fbfSMarkus Pfeiffer #include <net/ifq_var.h>
4912bd3c8bSSascha Wildner #include <net/bpf.h>
50f5199fbfSMarkus Pfeiffer #include <net/route.h>
5112bd3c8bSSascha Wildner #include <sys/sysctl.h>
52722d05c3SSascha Wildner #include <sys/condvar.h>
5312bd3c8bSSascha Wildner 
54722d05c3SSascha Wildner #include <bus/u4b/usb.h>
55722d05c3SSascha Wildner #include <bus/u4b/usbdi.h>
56722d05c3SSascha Wildner #include <bus/u4b/usb_busdma.h>
578922de18SMarkus Pfeiffer 
58722d05c3SSascha Wildner #include <bus/u4b/usb_controller.h>
59722d05c3SSascha Wildner #include <bus/u4b/usb_core.h>
60722d05c3SSascha Wildner #include <bus/u4b/usb_process.h>
61722d05c3SSascha Wildner #include <bus/u4b/usb_device.h>
62722d05c3SSascha Wildner #include <bus/u4b/usb_bus.h>
63722d05c3SSascha Wildner #include <bus/u4b/usb_pf.h>
64722d05c3SSascha Wildner #include <bus/u4b/usb_transfer.h>
65f5199fbfSMarkus Pfeiffer 
668922de18SMarkus Pfeiffer static void usbpf_init(void *);
678922de18SMarkus Pfeiffer static void usbpf_uninit(void *);
68f5199fbfSMarkus Pfeiffer static int usbpf_ioctl(struct ifnet *, u_long, caddr_t, struct ucred *);
69*bb54c3a2SAaron LI static int usbpf_clone_create(struct if_clone *, int, caddr_t, caddr_t);
70f5199fbfSMarkus Pfeiffer static int usbpf_clone_destroy(struct ifnet *);
71f5199fbfSMarkus Pfeiffer static struct usb_bus *usbpf_ifname2ubus(int unit);
728922de18SMarkus Pfeiffer static uint32_t usbpf_aggregate_xferflags(struct usb_xfer_flags *);
738922de18SMarkus Pfeiffer static uint32_t usbpf_aggregate_status(struct usb_xfer_flags_int *);
748922de18SMarkus Pfeiffer static int usbpf_xfer_frame_is_read(struct usb_xfer *, uint32_t);
758922de18SMarkus Pfeiffer static uint32_t usbpf_xfer_precompute_size(struct usb_xfer *, int);
7612bd3c8bSSascha Wildner 
7712bd3c8bSSascha Wildner 
788922de18SMarkus Pfeiffer SYSINIT(usbpf_init, SI_SUB_PSEUDO, SI_ORDER_MIDDLE, usbpf_init, NULL);
798922de18SMarkus Pfeiffer SYSUNINIT(usbpf_uninit, SI_SUB_PSEUDO, SI_ORDER_MIDDLE, usbpf_uninit, NULL);
8012bd3c8bSSascha Wildner 
81f5199fbfSMarkus Pfeiffer static const char * usbusname = "usbus";
82f5199fbfSMarkus Pfeiffer struct if_clone usbpf_cloner = IF_CLONE_INITIALIZER("usbus",
83f5199fbfSMarkus Pfeiffer 				 usbpf_clone_create,
84f5199fbfSMarkus Pfeiffer 				 usbpf_clone_destroy,
85f5199fbfSMarkus Pfeiffer 				 0, IF_MAXUNIT);
86f5199fbfSMarkus Pfeiffer 
87f5199fbfSMarkus Pfeiffer static void
usbpf_init(void * arg)88f5199fbfSMarkus Pfeiffer usbpf_init(void *arg)
89f5199fbfSMarkus Pfeiffer {
90f5199fbfSMarkus Pfeiffer 	if_clone_attach(&usbpf_cloner);
91f5199fbfSMarkus Pfeiffer 	if (bootverbose)
92f5199fbfSMarkus Pfeiffer 		kprintf("usbpf: Initialized\n");
93f5199fbfSMarkus Pfeiffer 	return;
94f5199fbfSMarkus Pfeiffer }
95f5199fbfSMarkus Pfeiffer 
968922de18SMarkus Pfeiffer static void
usbpf_uninit(void * arg)978922de18SMarkus Pfeiffer usbpf_uninit(void *arg)
9812bd3c8bSSascha Wildner {
998922de18SMarkus Pfeiffer 	int devlcnt;
1008922de18SMarkus Pfeiffer 	device_t *devlp;
1018922de18SMarkus Pfeiffer 	devclass_t dc;
1028922de18SMarkus Pfeiffer 	struct usb_bus *ubus;
1038922de18SMarkus Pfeiffer 	int error;
1048922de18SMarkus Pfeiffer 	int i;
10512bd3c8bSSascha Wildner 
106f5199fbfSMarkus Pfeiffer 	if_clone_detach(&usbpf_cloner);
1078922de18SMarkus Pfeiffer 
1088922de18SMarkus Pfeiffer 	dc = devclass_find(usbusname);
1098922de18SMarkus Pfeiffer 	if (dc == NULL)
11012bd3c8bSSascha Wildner 		return;
1118922de18SMarkus Pfeiffer 	error = devclass_get_devices(dc, &devlp, &devlcnt);
1128922de18SMarkus Pfeiffer 	if (error)
1138922de18SMarkus Pfeiffer 		return;
1148922de18SMarkus Pfeiffer 	for (i = 0; i < devlcnt; i++) {
1158922de18SMarkus Pfeiffer 		ubus = device_get_softc(devlp[i]);
1168922de18SMarkus Pfeiffer 		if (ubus != NULL && ubus->ifp != NULL)
117f5199fbfSMarkus Pfeiffer 			usbpf_clone_destroy(ubus->ifp);
1188922de18SMarkus Pfeiffer 	}
119f5199fbfSMarkus Pfeiffer 	kfree(devlp, M_TEMP);
12012bd3c8bSSascha Wildner }
12112bd3c8bSSascha Wildner 
1228922de18SMarkus Pfeiffer static int
usbpf_ioctl(struct ifnet * ifp,u_long cmd,caddr_t data,struct ucred * cr)123f5199fbfSMarkus Pfeiffer usbpf_ioctl(struct ifnet *ifp, u_long cmd, caddr_t data, struct ucred *cr)
1248922de18SMarkus Pfeiffer {
1258922de18SMarkus Pfeiffer 	/* No configuration allowed. */
1268922de18SMarkus Pfeiffer 	return (EINVAL);
1278922de18SMarkus Pfeiffer }
1288922de18SMarkus Pfeiffer 
1298922de18SMarkus Pfeiffer static struct usb_bus *
usbpf_ifname2ubus(int unit)130f5199fbfSMarkus Pfeiffer usbpf_ifname2ubus(int unit)
1318922de18SMarkus Pfeiffer {
1328922de18SMarkus Pfeiffer 	device_t dev;
1338922de18SMarkus Pfeiffer 	devclass_t dc;
1348922de18SMarkus Pfeiffer 
1358922de18SMarkus Pfeiffer 	dc = devclass_find(usbusname);
1368922de18SMarkus Pfeiffer 	if (dc == NULL)
1378922de18SMarkus Pfeiffer 		return (NULL);
1388922de18SMarkus Pfeiffer 	dev = devclass_get_device(dc, unit);
1398922de18SMarkus Pfeiffer 	if (dev == NULL)
1408922de18SMarkus Pfeiffer 		return (NULL);
1418922de18SMarkus Pfeiffer 
1428922de18SMarkus Pfeiffer 	return (device_get_softc(dev));
1438922de18SMarkus Pfeiffer }
1448922de18SMarkus Pfeiffer 
1458922de18SMarkus Pfeiffer static int
usbpf_clone_create(struct if_clone * ifc,int unit,caddr_t params,caddr_t data __unused)146*bb54c3a2SAaron LI usbpf_clone_create(struct if_clone *ifc, int unit, caddr_t params,
147*bb54c3a2SAaron LI 		   caddr_t data __unused)
1488922de18SMarkus Pfeiffer {
1498922de18SMarkus Pfeiffer 	struct ifnet *ifp;
1508922de18SMarkus Pfeiffer 	struct usb_bus *ubus;
1518922de18SMarkus Pfeiffer 
152f5199fbfSMarkus Pfeiffer 	ubus = usbpf_ifname2ubus(unit);
1538922de18SMarkus Pfeiffer 	if (ubus == NULL)
154f5199fbfSMarkus Pfeiffer 		return (EINVAL);
1558922de18SMarkus Pfeiffer 	if (ubus->ifp != NULL)
156f5199fbfSMarkus Pfeiffer 		return (EINVAL);
1578922de18SMarkus Pfeiffer 	ifp = ubus->ifp = if_alloc(IFT_USB);
1588922de18SMarkus Pfeiffer 	if (ifp == NULL) {
1598922de18SMarkus Pfeiffer 		device_printf(ubus->parent, "usbpf: Could not allocate "
1608922de18SMarkus Pfeiffer 		    "instance\n");
1618922de18SMarkus Pfeiffer 		return (ENOSPC);
1628922de18SMarkus Pfeiffer 	}
163f5199fbfSMarkus Pfeiffer 	ksnprintf(ifp->if_xname, sizeof(ifp->if_xname), "%s%d", usbusname, unit);
1648922de18SMarkus Pfeiffer 	ifp->if_softc = ubus;
1658922de18SMarkus Pfeiffer 	ifp->if_dname = usbusname;
1668922de18SMarkus Pfeiffer 	ifp->if_dunit = unit;
1678922de18SMarkus Pfeiffer 	ifp->if_ioctl = usbpf_ioctl;
168f5199fbfSMarkus Pfeiffer 	ifq_set_maxlen(&ifp->if_snd, ifqmaxlen);
169f5199fbfSMarkus Pfeiffer 	if_attach(ifp, NULL);
1708922de18SMarkus Pfeiffer 	ifp->if_flags |= IFF_UP;
1718922de18SMarkus Pfeiffer 	rt_ifmsg(ifp);
17212bd3c8bSSascha Wildner 	/*
17312bd3c8bSSascha Wildner 	 * XXX According to the specification of DLT_USB, it indicates
17412bd3c8bSSascha Wildner 	 * packets beginning with USB setup header. But not sure all
17512bd3c8bSSascha Wildner 	 * packets would be.
17612bd3c8bSSascha Wildner 	 */
17712bd3c8bSSascha Wildner 	bpfattach(ifp, DLT_USB, USBPF_HDR_LEN);
17812bd3c8bSSascha Wildner 
1798922de18SMarkus Pfeiffer 	return (0);
1808922de18SMarkus Pfeiffer }
1818922de18SMarkus Pfeiffer 
1828922de18SMarkus Pfeiffer static int
usbpf_clone_destroy(struct ifnet * ifp)183f5199fbfSMarkus Pfeiffer usbpf_clone_destroy(struct ifnet *ifp)
1848922de18SMarkus Pfeiffer {
1858922de18SMarkus Pfeiffer 	struct usb_bus *ubus;
1868922de18SMarkus Pfeiffer 	int unit;
1878922de18SMarkus Pfeiffer 
1888922de18SMarkus Pfeiffer 	ubus = ifp->if_softc;
1898922de18SMarkus Pfeiffer 	unit = ifp->if_dunit;
1908922de18SMarkus Pfeiffer 
1918922de18SMarkus Pfeiffer 	ubus->ifp = NULL;
1928922de18SMarkus Pfeiffer 	bpfdetach(ifp);
1938922de18SMarkus Pfeiffer 	if_detach(ifp);
1948922de18SMarkus Pfeiffer 	if_free(ifp);
1958922de18SMarkus Pfeiffer 
1968922de18SMarkus Pfeiffer 	return (0);
1978922de18SMarkus Pfeiffer }
1988922de18SMarkus Pfeiffer 
1998922de18SMarkus Pfeiffer void
usbpf_attach(struct usb_bus * ubus)2008922de18SMarkus Pfeiffer usbpf_attach(struct usb_bus *ubus)
2018922de18SMarkus Pfeiffer {
20212bd3c8bSSascha Wildner 	if (bootverbose)
20312bd3c8bSSascha Wildner 		device_printf(ubus->parent, "usbpf: Attached\n");
20412bd3c8bSSascha Wildner }
20512bd3c8bSSascha Wildner 
20612bd3c8bSSascha Wildner void
usbpf_detach(struct usb_bus * ubus)20712bd3c8bSSascha Wildner usbpf_detach(struct usb_bus *ubus)
20812bd3c8bSSascha Wildner {
2098922de18SMarkus Pfeiffer 	if (ubus->ifp != NULL)
210f5199fbfSMarkus Pfeiffer 		usbpf_clone_destroy(ubus->ifp);
2118922de18SMarkus Pfeiffer 	if (bootverbose)
2128922de18SMarkus Pfeiffer 		device_printf(ubus->parent, "usbpf: Detached\n");
21312bd3c8bSSascha Wildner }
21412bd3c8bSSascha Wildner 
21512bd3c8bSSascha Wildner static uint32_t
usbpf_aggregate_xferflags(struct usb_xfer_flags * flags)21612bd3c8bSSascha Wildner usbpf_aggregate_xferflags(struct usb_xfer_flags *flags)
21712bd3c8bSSascha Wildner {
21812bd3c8bSSascha Wildner 	uint32_t val = 0;
21912bd3c8bSSascha Wildner 
22012bd3c8bSSascha Wildner 	if (flags->force_short_xfer == 1)
22112bd3c8bSSascha Wildner 		val |= USBPF_FLAG_FORCE_SHORT_XFER;
22212bd3c8bSSascha Wildner 	if (flags->short_xfer_ok == 1)
22312bd3c8bSSascha Wildner 		val |= USBPF_FLAG_SHORT_XFER_OK;
22412bd3c8bSSascha Wildner 	if (flags->short_frames_ok == 1)
22512bd3c8bSSascha Wildner 		val |= USBPF_FLAG_SHORT_FRAMES_OK;
22612bd3c8bSSascha Wildner 	if (flags->pipe_bof == 1)
22712bd3c8bSSascha Wildner 		val |= USBPF_FLAG_PIPE_BOF;
22812bd3c8bSSascha Wildner 	if (flags->proxy_buffer == 1)
22912bd3c8bSSascha Wildner 		val |= USBPF_FLAG_PROXY_BUFFER;
23012bd3c8bSSascha Wildner 	if (flags->ext_buffer == 1)
23112bd3c8bSSascha Wildner 		val |= USBPF_FLAG_EXT_BUFFER;
23212bd3c8bSSascha Wildner 	if (flags->manual_status == 1)
23312bd3c8bSSascha Wildner 		val |= USBPF_FLAG_MANUAL_STATUS;
23412bd3c8bSSascha Wildner 	if (flags->no_pipe_ok == 1)
23512bd3c8bSSascha Wildner 		val |= USBPF_FLAG_NO_PIPE_OK;
23612bd3c8bSSascha Wildner 	if (flags->stall_pipe == 1)
23712bd3c8bSSascha Wildner 		val |= USBPF_FLAG_STALL_PIPE;
23812bd3c8bSSascha Wildner 	return (val);
23912bd3c8bSSascha Wildner }
24012bd3c8bSSascha Wildner 
24112bd3c8bSSascha Wildner static uint32_t
usbpf_aggregate_status(struct usb_xfer_flags_int * flags)24212bd3c8bSSascha Wildner usbpf_aggregate_status(struct usb_xfer_flags_int *flags)
24312bd3c8bSSascha Wildner {
24412bd3c8bSSascha Wildner 	uint32_t val = 0;
24512bd3c8bSSascha Wildner 
24612bd3c8bSSascha Wildner 	if (flags->open == 1)
24712bd3c8bSSascha Wildner 		val |= USBPF_STATUS_OPEN;
24812bd3c8bSSascha Wildner 	if (flags->transferring == 1)
24912bd3c8bSSascha Wildner 		val |= USBPF_STATUS_TRANSFERRING;
25012bd3c8bSSascha Wildner 	if (flags->did_dma_delay == 1)
25112bd3c8bSSascha Wildner 		val |= USBPF_STATUS_DID_DMA_DELAY;
25212bd3c8bSSascha Wildner 	if (flags->did_close == 1)
25312bd3c8bSSascha Wildner 		val |= USBPF_STATUS_DID_CLOSE;
25412bd3c8bSSascha Wildner 	if (flags->draining == 1)
25512bd3c8bSSascha Wildner 		val |= USBPF_STATUS_DRAINING;
25612bd3c8bSSascha Wildner 	if (flags->started == 1)
25712bd3c8bSSascha Wildner 		val |= USBPF_STATUS_STARTED;
25812bd3c8bSSascha Wildner 	if (flags->bandwidth_reclaimed == 1)
25912bd3c8bSSascha Wildner 		val |= USBPF_STATUS_BW_RECLAIMED;
26012bd3c8bSSascha Wildner 	if (flags->control_xfr == 1)
26112bd3c8bSSascha Wildner 		val |= USBPF_STATUS_CONTROL_XFR;
26212bd3c8bSSascha Wildner 	if (flags->control_hdr == 1)
26312bd3c8bSSascha Wildner 		val |= USBPF_STATUS_CONTROL_HDR;
26412bd3c8bSSascha Wildner 	if (flags->control_act == 1)
26512bd3c8bSSascha Wildner 		val |= USBPF_STATUS_CONTROL_ACT;
26612bd3c8bSSascha Wildner 	if (flags->control_stall == 1)
26712bd3c8bSSascha Wildner 		val |= USBPF_STATUS_CONTROL_STALL;
26812bd3c8bSSascha Wildner 	if (flags->short_frames_ok == 1)
26912bd3c8bSSascha Wildner 		val |= USBPF_STATUS_SHORT_FRAMES_OK;
27012bd3c8bSSascha Wildner 	if (flags->short_xfer_ok == 1)
27112bd3c8bSSascha Wildner 		val |= USBPF_STATUS_SHORT_XFER_OK;
27212bd3c8bSSascha Wildner #if USB_HAVE_BUSDMA
27312bd3c8bSSascha Wildner 	if (flags->bdma_enable == 1)
27412bd3c8bSSascha Wildner 		val |= USBPF_STATUS_BDMA_ENABLE;
27512bd3c8bSSascha Wildner 	if (flags->bdma_no_post_sync == 1)
27612bd3c8bSSascha Wildner 		val |= USBPF_STATUS_BDMA_NO_POST_SYNC;
27712bd3c8bSSascha Wildner 	if (flags->bdma_setup == 1)
27812bd3c8bSSascha Wildner 		val |= USBPF_STATUS_BDMA_SETUP;
27912bd3c8bSSascha Wildner #endif
28012bd3c8bSSascha Wildner 	if (flags->isochronous_xfr == 1)
28112bd3c8bSSascha Wildner 		val |= USBPF_STATUS_ISOCHRONOUS_XFR;
28212bd3c8bSSascha Wildner 	if (flags->curr_dma_set == 1)
28312bd3c8bSSascha Wildner 		val |= USBPF_STATUS_CURR_DMA_SET;
28412bd3c8bSSascha Wildner 	if (flags->can_cancel_immed == 1)
28512bd3c8bSSascha Wildner 		val |= USBPF_STATUS_CAN_CANCEL_IMMED;
28612bd3c8bSSascha Wildner 	if (flags->doing_callback == 1)
28712bd3c8bSSascha Wildner 		val |= USBPF_STATUS_DOING_CALLBACK;
28812bd3c8bSSascha Wildner 
28912bd3c8bSSascha Wildner 	return (val);
29012bd3c8bSSascha Wildner }
29112bd3c8bSSascha Wildner 
29212bd3c8bSSascha Wildner static int
usbpf_xfer_frame_is_read(struct usb_xfer * xfer,uint32_t frame)29312bd3c8bSSascha Wildner usbpf_xfer_frame_is_read(struct usb_xfer *xfer, uint32_t frame)
29412bd3c8bSSascha Wildner {
29512bd3c8bSSascha Wildner 	int isread;
29612bd3c8bSSascha Wildner 
29712bd3c8bSSascha Wildner 	if ((frame == 0) && (xfer->flags_int.control_xfr != 0) &&
29812bd3c8bSSascha Wildner 	    (xfer->flags_int.control_hdr != 0)) {
29912bd3c8bSSascha Wildner 		/* special case */
30012bd3c8bSSascha Wildner 		if (xfer->flags_int.usb_mode == USB_MODE_DEVICE) {
30112bd3c8bSSascha Wildner 			/* The device controller writes to memory */
30212bd3c8bSSascha Wildner 			isread = 1;
30312bd3c8bSSascha Wildner 		} else {
30412bd3c8bSSascha Wildner 			/* The host controller reads from memory */
30512bd3c8bSSascha Wildner 			isread = 0;
30612bd3c8bSSascha Wildner 		}
30712bd3c8bSSascha Wildner 	} else {
30812bd3c8bSSascha Wildner 		isread = USB_GET_DATA_ISREAD(xfer);
30912bd3c8bSSascha Wildner 	}
31012bd3c8bSSascha Wildner 	return (isread);
31112bd3c8bSSascha Wildner }
31212bd3c8bSSascha Wildner 
31312bd3c8bSSascha Wildner static uint32_t
usbpf_xfer_precompute_size(struct usb_xfer * xfer,int type)31412bd3c8bSSascha Wildner usbpf_xfer_precompute_size(struct usb_xfer *xfer, int type)
31512bd3c8bSSascha Wildner {
31612bd3c8bSSascha Wildner 	uint32_t totlen;
31712bd3c8bSSascha Wildner 	uint32_t x;
31812bd3c8bSSascha Wildner 	uint32_t nframes;
31912bd3c8bSSascha Wildner 
32012bd3c8bSSascha Wildner 	if (type == USBPF_XFERTAP_SUBMIT)
32112bd3c8bSSascha Wildner 		nframes = xfer->nframes;
32212bd3c8bSSascha Wildner 	else
32312bd3c8bSSascha Wildner 		nframes = xfer->aframes;
32412bd3c8bSSascha Wildner 
32512bd3c8bSSascha Wildner 	totlen = USBPF_HDR_LEN + (USBPF_FRAME_HDR_LEN * nframes);
32612bd3c8bSSascha Wildner 
32712bd3c8bSSascha Wildner 	/* precompute all trace lengths */
32812bd3c8bSSascha Wildner 	for (x = 0; x != nframes; x++) {
32912bd3c8bSSascha Wildner 		if (usbpf_xfer_frame_is_read(xfer, x)) {
33012bd3c8bSSascha Wildner 			if (type != USBPF_XFERTAP_SUBMIT) {
33112bd3c8bSSascha Wildner 				totlen += USBPF_FRAME_ALIGN(
33212bd3c8bSSascha Wildner 				    xfer->frlengths[x]);
33312bd3c8bSSascha Wildner 			}
33412bd3c8bSSascha Wildner 		} else {
33512bd3c8bSSascha Wildner 			if (type == USBPF_XFERTAP_SUBMIT) {
33612bd3c8bSSascha Wildner 				totlen += USBPF_FRAME_ALIGN(
33712bd3c8bSSascha Wildner 				    xfer->frlengths[x]);
33812bd3c8bSSascha Wildner 			}
33912bd3c8bSSascha Wildner 		}
34012bd3c8bSSascha Wildner 	}
34112bd3c8bSSascha Wildner 	return (totlen);
34212bd3c8bSSascha Wildner }
34312bd3c8bSSascha Wildner 
34412bd3c8bSSascha Wildner void
usbpf_xfertap(struct usb_xfer * xfer,int type)34512bd3c8bSSascha Wildner usbpf_xfertap(struct usb_xfer *xfer, int type)
34612bd3c8bSSascha Wildner {
34712bd3c8bSSascha Wildner 	struct usb_bus *bus;
34812bd3c8bSSascha Wildner 	struct usbpf_pkthdr *up;
34912bd3c8bSSascha Wildner 	struct usbpf_framehdr *uf;
35012bd3c8bSSascha Wildner 	usb_frlength_t offset;
35112bd3c8bSSascha Wildner 	uint32_t totlen;
35212bd3c8bSSascha Wildner 	uint32_t frame;
35312bd3c8bSSascha Wildner 	uint32_t temp;
35412bd3c8bSSascha Wildner 	uint32_t nframes;
35512bd3c8bSSascha Wildner 	uint32_t x;
35612bd3c8bSSascha Wildner 	uint8_t *buf;
35712bd3c8bSSascha Wildner 	uint8_t *ptr;
35812bd3c8bSSascha Wildner 
35912bd3c8bSSascha Wildner 	bus = xfer->xroot->bus;
36012bd3c8bSSascha Wildner 
36112bd3c8bSSascha Wildner 	/* sanity checks */
36212bd3c8bSSascha Wildner 	if (bus->ifp == NULL)
36312bd3c8bSSascha Wildner 		return;
36415130067Szrj #if 0 /* XXX this is not needed on dragonfly */
36512bd3c8bSSascha Wildner 	if (!bpf_peers_present(bus->ifp->if_bpf))
36612bd3c8bSSascha Wildner 		return;
36715130067Szrj #endif
36812bd3c8bSSascha Wildner 	totlen = usbpf_xfer_precompute_size(xfer, type);
36912bd3c8bSSascha Wildner 
37012bd3c8bSSascha Wildner 	if (type == USBPF_XFERTAP_SUBMIT)
37112bd3c8bSSascha Wildner 		nframes = xfer->nframes;
37212bd3c8bSSascha Wildner 	else
37312bd3c8bSSascha Wildner 		nframes = xfer->aframes;
37412bd3c8bSSascha Wildner 
37512bd3c8bSSascha Wildner 	/*
37612bd3c8bSSascha Wildner 	 * XXX TODO XXX
37712bd3c8bSSascha Wildner 	 *
37812bd3c8bSSascha Wildner 	 * When BPF supports it we could pass a fragmented array of
37912bd3c8bSSascha Wildner 	 * buffers avoiding the data copy operation here.
38012bd3c8bSSascha Wildner 	 */
381f5199fbfSMarkus Pfeiffer 	buf = ptr = kmalloc(totlen, M_TEMP, M_NOWAIT);
38212bd3c8bSSascha Wildner 	if (buf == NULL) {
38312bd3c8bSSascha Wildner 		device_printf(bus->parent, "usbpf: Out of memory\n");
38412bd3c8bSSascha Wildner 		return;
38512bd3c8bSSascha Wildner 	}
38612bd3c8bSSascha Wildner 
38712bd3c8bSSascha Wildner 	up = (struct usbpf_pkthdr *)ptr;
38812bd3c8bSSascha Wildner 	ptr += USBPF_HDR_LEN;
38912bd3c8bSSascha Wildner 
39012bd3c8bSSascha Wildner 	/* fill out header */
39112bd3c8bSSascha Wildner 	temp = device_get_unit(bus->bdev);
39212bd3c8bSSascha Wildner 	up->up_totlen = htole32(totlen);
39312bd3c8bSSascha Wildner 	up->up_busunit = htole32(temp);
39412bd3c8bSSascha Wildner 	up->up_address = xfer->xroot->udev->device_index;
39512bd3c8bSSascha Wildner 	if (xfer->flags_int.usb_mode == USB_MODE_DEVICE)
39612bd3c8bSSascha Wildner 		up->up_mode = USBPF_MODE_DEVICE;
39712bd3c8bSSascha Wildner 	else
39812bd3c8bSSascha Wildner 		up->up_mode = USBPF_MODE_HOST;
39912bd3c8bSSascha Wildner 	up->up_type = type;
40012bd3c8bSSascha Wildner 	up->up_xfertype = xfer->endpoint->edesc->bmAttributes & UE_XFERTYPE;
40112bd3c8bSSascha Wildner 	temp = usbpf_aggregate_xferflags(&xfer->flags);
40212bd3c8bSSascha Wildner 	up->up_flags = htole32(temp);
40312bd3c8bSSascha Wildner 	temp = usbpf_aggregate_status(&xfer->flags_int);
40412bd3c8bSSascha Wildner 	up->up_status = htole32(temp);
40512bd3c8bSSascha Wildner 	temp = xfer->error;
40612bd3c8bSSascha Wildner 	up->up_error = htole32(temp);
40712bd3c8bSSascha Wildner 	temp = xfer->interval;
40812bd3c8bSSascha Wildner 	up->up_interval = htole32(temp);
40912bd3c8bSSascha Wildner 	up->up_frames = htole32(nframes);
41012bd3c8bSSascha Wildner 	temp = xfer->max_packet_size;
41112bd3c8bSSascha Wildner 	up->up_packet_size = htole32(temp);
41212bd3c8bSSascha Wildner 	temp = xfer->max_packet_count;
41312bd3c8bSSascha Wildner 	up->up_packet_count = htole32(temp);
41412bd3c8bSSascha Wildner 	temp = xfer->endpointno;
41512bd3c8bSSascha Wildner 	up->up_endpoint = htole32(temp);
41612bd3c8bSSascha Wildner 	up->up_speed = xfer->xroot->udev->speed;
41712bd3c8bSSascha Wildner 
41812bd3c8bSSascha Wildner 	/* clear reserved area */
41912bd3c8bSSascha Wildner 	memset(up->up_reserved, 0, sizeof(up->up_reserved));
42012bd3c8bSSascha Wildner 
42112bd3c8bSSascha Wildner 	/* init offset and frame */
42212bd3c8bSSascha Wildner 	offset = 0;
42312bd3c8bSSascha Wildner 	frame = 0;
42412bd3c8bSSascha Wildner 
42512bd3c8bSSascha Wildner 	/* iterate all the USB frames and copy data, if any */
42612bd3c8bSSascha Wildner 	for (x = 0; x != nframes; x++) {
42712bd3c8bSSascha Wildner 		uint32_t length;
42812bd3c8bSSascha Wildner 		int isread;
42912bd3c8bSSascha Wildner 
43012bd3c8bSSascha Wildner 		/* get length */
43112bd3c8bSSascha Wildner 		length = xfer->frlengths[x];
43212bd3c8bSSascha Wildner 
43312bd3c8bSSascha Wildner 		/* get frame header pointer */
43412bd3c8bSSascha Wildner 		uf = (struct usbpf_framehdr *)ptr;
43512bd3c8bSSascha Wildner 		ptr += USBPF_FRAME_HDR_LEN;
43612bd3c8bSSascha Wildner 
43712bd3c8bSSascha Wildner 		/* fill out packet header */
43812bd3c8bSSascha Wildner 		uf->length = htole32(length);
43912bd3c8bSSascha Wildner 		uf->flags = 0;
44012bd3c8bSSascha Wildner 
44112bd3c8bSSascha Wildner 		/* get information about data read/write */
44212bd3c8bSSascha Wildner 		isread = usbpf_xfer_frame_is_read(xfer, x);
44312bd3c8bSSascha Wildner 
44412bd3c8bSSascha Wildner 		/* check if we need to copy any data */
44512bd3c8bSSascha Wildner 		if (isread) {
44612bd3c8bSSascha Wildner 			if (type == USBPF_XFERTAP_SUBMIT)
44712bd3c8bSSascha Wildner 				length = 0;
44812bd3c8bSSascha Wildner 			else {
44912bd3c8bSSascha Wildner 				uf->flags |= htole32(
45012bd3c8bSSascha Wildner 				    USBPF_FRAMEFLAG_DATA_FOLLOWS);
45112bd3c8bSSascha Wildner 			}
45212bd3c8bSSascha Wildner 		} else {
45312bd3c8bSSascha Wildner 			if (type != USBPF_XFERTAP_SUBMIT)
45412bd3c8bSSascha Wildner 				length = 0;
45512bd3c8bSSascha Wildner 			else {
45612bd3c8bSSascha Wildner 				uf->flags |= htole32(
45712bd3c8bSSascha Wildner 				    USBPF_FRAMEFLAG_DATA_FOLLOWS);
45812bd3c8bSSascha Wildner 			}
45912bd3c8bSSascha Wildner 		}
46012bd3c8bSSascha Wildner 
46112bd3c8bSSascha Wildner 		/* check if data is read direction */
46212bd3c8bSSascha Wildner 		if (isread)
46312bd3c8bSSascha Wildner 			uf->flags |= htole32(USBPF_FRAMEFLAG_READ);
46412bd3c8bSSascha Wildner 
46512bd3c8bSSascha Wildner 		/* copy USB data, if any */
46612bd3c8bSSascha Wildner 		if (length != 0) {
46712bd3c8bSSascha Wildner 			/* copy data */
46812bd3c8bSSascha Wildner 			usbd_copy_out(&xfer->frbuffers[frame],
46912bd3c8bSSascha Wildner 			    offset, ptr, length);
47012bd3c8bSSascha Wildner 
47112bd3c8bSSascha Wildner 			/* align length */
47212bd3c8bSSascha Wildner 			temp = USBPF_FRAME_ALIGN(length);
47312bd3c8bSSascha Wildner 
47412bd3c8bSSascha Wildner 			/* zero pad */
47512bd3c8bSSascha Wildner 			if (temp != length)
47612bd3c8bSSascha Wildner 				memset(ptr + length, 0, temp - length);
47712bd3c8bSSascha Wildner 
47812bd3c8bSSascha Wildner 			ptr += temp;
47912bd3c8bSSascha Wildner 		}
48012bd3c8bSSascha Wildner 
48112bd3c8bSSascha Wildner 		if (xfer->flags_int.isochronous_xfr) {
48212bd3c8bSSascha Wildner 			offset += usbd_xfer_old_frame_length(xfer, x);
48312bd3c8bSSascha Wildner 		} else {
48412bd3c8bSSascha Wildner 			frame ++;
48512bd3c8bSSascha Wildner 		}
48612bd3c8bSSascha Wildner 	}
48712bd3c8bSSascha Wildner 
48812bd3c8bSSascha Wildner 	bpf_tap(bus->ifp->if_bpf, buf, totlen);
48912bd3c8bSSascha Wildner 
490f5199fbfSMarkus Pfeiffer 	kfree(buf, M_TEMP);
49112bd3c8bSSascha Wildner }
492