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