Date: Sat, 27 Jan 2007 15:01:02 GMT From: Hans Petter Selasky <hselasky@FreeBSD.org> To: Perforce Change Reviews <perforce@FreeBSD.org> Subject: PERFORCE change 113598 for review Message-ID: <200701271501.l0RF12C3083473@repoman.freebsd.org>
next in thread | raw e-mail | index | archive | help
http://perforce.freebsd.org/chv.cgi?CH=113598 Change 113598 by hselasky@hselasky_mini_itx on 2007/01/27 15:00:54 Make the isochronous end-of-transfer timestamp 16-bits in width. This simplifies the USB device drivers doing echo-cancelling for example. Affected files ... .. //depot/projects/usb/src/sys/dev/usb/ehci.c#25 edit .. //depot/projects/usb/src/sys/dev/usb/ohci.c#19 edit .. //depot/projects/usb/src/sys/dev/usb/uhci.c#21 edit .. //depot/projects/usb/src/sys/dev/usb/usb_subr.c#27 edit .. //depot/projects/usb/src/sys/dev/usb/usb_subr.h#32 edit Differences ... ==== //depot/projects/usb/src/sys/dev/usb/ehci.c#25 (text+ko) ==== @@ -2458,11 +2458,18 @@ DPRINTFN(5,("xfer=%p next=%d nframes=%d\n", xfer, xfer->pipe->isoc_next, xfer->nframes)); + /* get the current frame index */ + nframes = EOREAD4(sc, EHCI_FRINDEX) / 8; - if(((nframes - xfer->pipe->isoc_next) & - (EHCI_VIRTUAL_FRAMELIST_COUNT-1)) < xfer->nframes) - { + /* check if the frame index is within + * the window where the frames will be + * inserted + */ + buf_offset = (nframes - xfer->pipe->isoc_next) & + (EHCI_VIRTUAL_FRAMELIST_COUNT-1); + + if (buf_offset < xfer->nframes) { /* not in use yet, schedule it a few frames ahead */ /* data underflow */ xfer->pipe->isoc_next = (nframes + 3) & @@ -2470,7 +2477,21 @@ DPRINTFN(2,("start next=%d\n", xfer->pipe->isoc_next)); } - xfer->isoc_complete_time = (xfer->pipe->isoc_next + xfer->nframes) % USBD_ISOC_TIME_MAX; + /* compute how many milliseconds the + * insertion is ahead of the current + * frame position: + */ + buf_offset = (xfer->pipe->isoc_next - nframes) & + (EHCI_VIRTUAL_FRAMELIST_COUNT-1); + + /* pre-compute when the isochronous transfer + * will be finished: + */ + xfer->isoc_time_complete = + usbd_isoc_time_expand(&(sc->sc_bus), nframes) + buf_offset + + xfer->nframes; + + /* get the real number of frames */ nframes = xfer->nframes; @@ -2693,11 +2714,18 @@ DPRINTFN(5,("xfer=%p next=%d nframes=%d\n", xfer, xfer->pipe->isoc_next, xfer->nframes)); + /* get the current frame index */ + nframes = EOREAD4(sc, EHCI_FRINDEX) / 8; - if(((nframes - xfer->pipe->isoc_next) & - (EHCI_VIRTUAL_FRAMELIST_COUNT-1)) < xfer->nframes) - { + /* check if the frame index is within + * the window where the frames will be + * inserted + */ + buf_offset = (nframes - xfer->pipe->isoc_next) & + (EHCI_VIRTUAL_FRAMELIST_COUNT-1); + + if (buf_offset < xfer->nframes) { /* not in use yet, schedule it a few frames ahead */ /* data underflow */ xfer->pipe->isoc_next = (nframes + 3) & @@ -2705,7 +2733,21 @@ DPRINTFN(2,("start next=%d\n", xfer->pipe->isoc_next)); } - xfer->isoc_complete_time = (xfer->pipe->isoc_next + ((xfer->nframes+7)/8)) % USBD_ISOC_TIME_MAX; + /* compute how many milliseconds the + * insertion is ahead of the current + * frame position: + */ + buf_offset = (xfer->pipe->isoc_next - nframes) & + (EHCI_VIRTUAL_FRAMELIST_COUNT-1); + + /* pre-compute when the isochronous transfer + * will be finished: + */ + xfer->isoc_time_complete = + usbd_isoc_time_expand(&(sc->sc_bus), nframes) + buf_offset + + ((xfer->nframes+7)/8); + + /* get the real number of frames */ nframes = xfer->nframes; ==== //depot/projects/usb/src/sys/dev/usb/ohci.c#19 (text+ko) ==== @@ -1896,16 +1896,34 @@ nframes = le32toh(hw_ptr->hcca.hcca_frame_number); usbd_page_dma_enter(&(sc->sc_hw_page)); - if((((nframes - xfer->pipe->isoc_next) & ((1<<16)-1)) < xfer->nframes) || - (((xfer->pipe->isoc_next - nframes) & ((1<<16)-1)) >= 256)) + /* check if the frame index is within + * the window where the frames will be + * inserted and if the delay until start + * is too long + */ + if ((((nframes - xfer->pipe->isoc_next) & 0xFFFF) < xfer->nframes) || + (((xfer->pipe->isoc_next - nframes) & 0xFFFF) >= 128)) { /* not in use yet, schedule it a few frames ahead */ /* data underflow */ - xfer->pipe->isoc_next = (nframes + 5) & ((1<<16)-1); + xfer->pipe->isoc_next = (nframes + 3) & 0xFFFF; DPRINTFN(2,("start next=%d\n", xfer->pipe->isoc_next)); } - xfer->isoc_complete_time = (xfer->pipe->isoc_next + xfer->nframes) % USBD_ISOC_TIME_MAX; + /* compute how many milliseconds the + * insertion is ahead of the current + * frame position: + */ + buf_offset = ((xfer->pipe->isoc_next - nframes) & 0xFFFF); + + /* pre-compute when the isochronous transfer + * will be finished: + */ + xfer->isoc_time_complete = + (usbd_isoc_time_expand(&(sc->sc_bus), nframes) + buf_offset + + xfer->nframes); + + /* get the real number of frames */ nframes = xfer->nframes; ==== //depot/projects/usb/src/sys/dev/usb/uhci.c#21 (text+ko) ==== @@ -1998,16 +1998,31 @@ nframes = UREAD2(sc, UHCI_FRNUM); - if(((nframes - xfer->pipe->isoc_next) & - (UHCI_VFRAMELIST_COUNT-1)) < xfer->nframes) - { + buf_offset = (nframes - xfer->pipe->isoc_next) & + (UHCI_VFRAMELIST_COUNT-1); + + if (buf_offset < xfer->nframes) { /* not in use yet, schedule it a few frames ahead */ /* data underflow */ xfer->pipe->isoc_next = (nframes + 3) & (UHCI_VFRAMELIST_COUNT-1); DPRINTFN(2,("start next=%d\n", xfer->pipe->isoc_next)); } - xfer->isoc_complete_time = (xfer->pipe->isoc_next + xfer->nframes) % USBD_ISOC_TIME_MAX; + /* compute how many milliseconds the + * insertion is ahead of the current + * frame position: + */ + buf_offset = (xfer->pipe->isoc_next - nframes) & + (UHCI_VFRAMELIST_COUNT-1); + + /* pre-compute when the isochronous transfer + * will be finished: + */ + xfer->isoc_time_complete = + usbd_isoc_time_expand(&(sc->sc_bus), nframes) + buf_offset + + xfer->nframes; + + /* get the real number of frames */ nframes = xfer->nframes; ==== //depot/projects/usb/src/sys/dev/usb/usb_subr.c#27 (text+ko) ==== @@ -2774,3 +2774,30 @@ USETW(edesc->wMaxPacketSize, n); return; } + +/*---------------------------------------------------------------------------* + * usbd_isoc_time_expand - expand time counter from 7-bit to 16-bit + *---------------------------------------------------------------------------*/ +uint16_t +usbd_isoc_time_expand(struct usbd_bus *bus, uint16_t isoc_time_curr) +{ + uint16_t rem; + + mtx_assert(&(bus->mtx), MA_OWNED); + + rem = bus->isoc_time_last & (USBD_ISOC_TIME_MAX-1); + + isoc_time_curr &= (USBD_ISOC_TIME_MAX-1); + + if (isoc_time_curr < rem) { + /* the time counter wrapped around */ + bus->isoc_time_last += USBD_ISOC_TIME_MAX; + } + + /* update the remainder */ + + bus->isoc_time_last &= ~(USBD_ISOC_TIME_MAX-1); + bus->isoc_time_last |= isoc_time_curr; + + return bus->isoc_time_last; +} ==== //depot/projects/usb/src/sys/dev/usb/usb_subr.h#32 (text+ko) ==== @@ -178,6 +178,9 @@ uint32_t no_intrs; + uint16_t isoc_time_last; /* ms */ +#define USBD_ISOC_TIME_MAX 128 /* ms */ + uint8_t is_exploring; uint8_t wait_explore; uint8_t needs_explore;/* Set if a hub signalled a change. @@ -361,10 +364,8 @@ uint16_t max_packet_size; uint16_t max_frame_size; uint16_t qh_pos; + uint16_t isoc_time_complete; /* in ms */ -#define USBD_ISOC_TIME_MAX 128 /* ms */ - - uint8_t isoc_complete_time; /* in ms */ uint8_t address; uint8_t endpoint; uint8_t interval; /* milliseconds */ @@ -602,6 +603,7 @@ uint16_t usbd_get_max_packet_count(usb_endpoint_descriptor_t *edesc); uint16_t usbd_get_max_frame_size(usb_endpoint_descriptor_t *edesc); void usbd_set_max_packet_size_count(usb_endpoint_descriptor_t *edesc, uint16_t size, uint16_t count); +uint16_t usbd_isoc_time_expand(struct usbd_bus *bus, uint16_t isoc_time_curr); /* prototypes from usb.c */
Want to link to this message? Use this URL: <https://mail-archive.FreeBSD.org/cgi/mid.cgi?200701271501.l0RF12C3083473>