diff options
Diffstat (limited to 'packages/linux/linux-rp-2.6.19/tmio-ohci-r6.patch')
-rw-r--r-- | packages/linux/linux-rp-2.6.19/tmio-ohci-r6.patch | 929 |
1 files changed, 0 insertions, 929 deletions
diff --git a/packages/linux/linux-rp-2.6.19/tmio-ohci-r6.patch b/packages/linux/linux-rp-2.6.19/tmio-ohci-r6.patch deleted file mode 100644 index 9fdd2962c9..0000000000 --- a/packages/linux/linux-rp-2.6.19/tmio-ohci-r6.patch +++ /dev/null @@ -1,929 +0,0 @@ - - drivers/usb/host/Kconfig | 1 - drivers/usb/host/ohci-hcd.c | 1 - drivers/usb/host/ohci-tmio.c | 894 +++++++++++++++++++++++++++++++++++++++++++ - 3 files changed, 896 insertions(+) - -Index: git/drivers/usb/host/ohci-tmio.c -=================================================================== ---- /dev/null 1970-01-01 00:00:00.000000000 +0000 -+++ git/drivers/usb/host/ohci-tmio.c 2006-11-07 21:48:33.000000000 +0000 -@@ -0,0 +1,894 @@ -+/* -+ * OHCI HCD (Host Controller Driver) for USB. -+ * -+ * (C) Copyright 1999 Roman Weissgaerber <weissg@vienna.at> -+ * (C) Copyright 2000-2002 David Brownell <dbrownell@users.sourceforge.net> -+ * (C) Copyright 2002 Hewlett-Packard Company -+ * -+ * Bus glue for Toshiba Mobile IO (TMIO) Controller's OHCI core -+ * (C) Copyright 2005 Chris Humbert <mahadri-usb@drigon.com> -+ * -+ * This is known to work with the following variants: -+ * TC6393XB revision 3 (32kB SRAM) -+ * -+ * The TMIO's OHCI core DMAs through a small internal buffer that -+ * is directly addressable by the CPU. dma_declare_coherent_memory -+ * and DMA bounce buffers allow the higher-level OHCI host driver to -+ * work. However, the dma API doesn't handle dma mapping failures -+ * well (dma_sg_map() is a prime example), so it is unusable. -+ * -+ * This HC pretends be a PIO-ish controller and uses the kernel's -+ * generic allocator for the entire SRAM. Using the USB core's -+ * usb_operations, we provide hcd_buffer_alloc/free. Using the OHCI's -+ * ohci_ops, we provide memory management for OHCI's TDs and EDs. We -+ * internally queue a URB's TDs until enough dma memory is available -+ * to enqueue them with the HC. -+ * -+ * Written from sparse documentation from Toshiba and Sharp's driver -+ * for the 2.4 kernel, -+ * usb-ohci-tc6393.c (C) Copyright 2004 Lineo Solutions, Inc. -+ * -+ * This program is free software; you can redistribute it and/or modify -+ * it under the terms of the GNU General Public License version 2 as -+ * published by the Free Software Foundation. -+ */ -+ -+#include <linux/fs.h> -+#include <linux/mount.h> -+#include <linux/pagemap.h> -+#include <linux/init.h> -+#include <linux/namei.h> -+#include <linux/sched.h> -+ -+#include <linux/genalloc.h> -+#include <asm/dma-mapping.h> /* for consistent_sync() */ -+#include <asm/hardware/tmio.h> -+ -+/*-------------------------------------------------------------------------*/ -+ -+/* -+ * USB Host Controller Configuration Register -+ */ -+struct tmio_uhccr { -+ u8 x00[8]; -+ u8 revid; /* 0x08 Revision ID */ -+ u8 x01[7]; -+ u16 basel; /* 0x10 USB Control Register Base Address Low */ -+ u16 baseh; /* 0x12 USB Control Register Base Address High */ -+ u8 x02[0x2c]; -+ u8 ilme; /* 0x40 Internal Local Memory Enable */ -+ u8 x03[0x0b]; -+ u16 pm; /* 0x4c Power Management */ -+ u8 x04[2]; -+ u8 intc; /* 0x50 INT Control */ -+ u8 x05[3]; -+ u16 lmw1l; /* 0x54 Local Memory Window 1 LMADRS Low */ -+ u16 lmw1h; /* 0x56 Local Memory Window 1 LMADRS High */ -+ u16 lmw1bl; /* 0x58 Local Memory Window 1 Base Address Low */ -+ u16 lmw1bh; /* 0x5A Local Memory Window 1 Base Address High */ -+ u16 lmw2l; /* 0x5C Local Memory Window 2 LMADRS Low */ -+ u16 lmw2h; /* 0x5E Local Memory Window 2 LMADRS High */ -+ u16 lmw2bl; /* 0x60 Local Memory Window 2 Base Address Low */ -+ u16 lmw2bh; /* 0x62 Local Memory Window 2 Base Address High */ -+ u8 x06[0x98]; -+ u8 misc; /* 0xFC MISC */ -+ u8 x07[3]; -+} __attribute__ ((packed)); -+ -+union tmio_uhccr_pm { -+ u16 raw; -+struct { -+ unsigned gcken:1; /* D0 */ -+ unsigned ckrnen:1; /* D1 */ -+ unsigned uspw1:1; /* D2 USB Port 1 Power Disable */ -+ unsigned uspw2:1; /* D3 USB Port 2 Power Disable */ -+ unsigned x00:4; -+ unsigned pmee:1; /* D8 */ -+ unsigned x01:6; -+ unsigned pmes:1; /* D15 */ -+} __attribute__ ((packed)); -+} __attribute__ ((packed)); -+ -+/*-------------------------------------------------------------------------*/ -+ -+struct tmio_dma_pool { -+ struct device* dev; -+ unsigned int size; -+}; -+ -+struct tmio_hcd { -+ struct gen_pool* poolp; -+ struct usb_operations ops; -+ struct tmio_dma_pool td_pool; -+ struct tmio_dma_pool ed_pool; -+ -+ struct tmio_uhccr __iomem *ccr; -+ void __iomem * sram; -+ size_t sram_len; -+}; -+ -+#define hcd_to_tmio(hcd) ((struct tmio_hcd*)(hcd_to_ohci (hcd) + 1)) -+ -+struct tmio_td { -+ void* data; /* td's data buffer */ -+ void __iomem * bounce; /* dma bounce buffer */ -+ dma_addr_t dma; /* bounce buffer dma address */ -+ size_t len; /* bounce buffer length */ -+ u32 info; /* parameter for td_fill */ -+}; -+ -+struct tmio_urb { -+ int td_add; /* next index to be added */ -+ int td_queue; /* next index to be HC enqueued */ -+ -+ struct tmio_td td [0]; /* private td data */ -+}; -+ -+static inline struct tmio_urb *urb_to_turb (struct urb *urb) -+{ -+ urb_priv_t* urb_priv = urb->hcpriv; -+ return (struct tmio_urb*)(urb_priv->td + urb_priv->length); -+} -+ -+/*-------------------------------------------------------------------------*/ -+ -+/* gen_pool_alloc page allocator callback */ -+static unsigned long tmio_pool_callback(struct gen_pool *poolp) -+{ -+ return 0; -+} -+ -+static inline void tmio_pool_destroy(struct tmio_hcd *tmio) -+{ -+ struct gen_pool *poolp = tmio->poolp; -+ -+ if (!poolp) -+ return; -+ if (poolp->h) -+ kfree(poolp->h); -+ kfree(poolp); -+ tmio->poolp = NULL; -+} -+ -+/*-------------------------------------------------------------------------*/ -+ -+#define BOUNDED_XYL(x,y,ylen) (((y) <= (x)) && ((x) < ((y)+(ylen)))) -+#define BOUNDED_XYY(x,y1,y2) (((y1) <= (x)) && ((x) < (y2))) -+ -+static inline dma_addr_t tmio_virt_to_dma (struct usb_hcd *hcd, void *vaddr) -+{ -+ struct resource* sram = tmio_resource_mem (hcd->self.controller); -+ struct tmio_hcd* tmio = hcd_to_tmio (hcd); -+ -+ return BOUNDED_XYL (vaddr, tmio->sram, tmio->sram_len) -+ ? sram->start + (vaddr - tmio->sram) -+ : ~0; -+} -+ -+static inline void* tmio_dma_to_virt (struct usb_hcd *hcd, dma_addr_t handle) -+{ -+ struct resource* sram = tmio_resource_mem (hcd->self.controller); -+ struct tmio_hcd* tmio = hcd_to_tmio (hcd); -+ -+ return BOUNDED_XYY (handle, sram->start, sram->end + 1) -+ ? tmio->sram + handle - sram->start -+ : NULL; -+} -+ -+/* -+ * allocate dma-able memory in the device's internal sram -+ * -+ * The generic pool allocator's minimum chunk size is 32 bytes, -+ * which is the cache line size on the PXA255, so we don't need -+ * to do anything special for smaller requests. -+ */ -+static inline void *tmio_dma_alloc (struct device *dev, size_t size, -+ dma_addr_t *handle, gfp_t mem_flags) -+{ -+ struct usb_hcd* hcd = dev_get_drvdata (dev); -+ struct tmio_hcd* tmio = hcd_to_tmio (hcd); -+ void* virt = (void*) gen_pool_alloc (tmio->poolp, size); -+ -+ return (*handle = tmio_virt_to_dma (hcd, virt)) == ~0 ? NULL : virt; -+} -+ -+static inline void tmio_dma_free (struct device *dev, size_t size, -+ void *cpu_addr, dma_addr_t handle) -+{ -+ struct usb_hcd* hcd = dev_get_drvdata (dev); -+ struct tmio_hcd* tmio = hcd_to_tmio (hcd); -+ dma_addr_t dma = tmio_virt_to_dma (hcd, cpu_addr); -+ -+ if (unlikely (dma == ~0)) { -+ dev_err (dev, "trying to free bad address 0x%p\n", cpu_addr); -+ return; -+ } -+ -+ if (unlikely (handle != dma)) -+ dev_err (dev, "dma address mismatch for 0x%p: %08x != %08x\n", -+ cpu_addr, handle, dma); -+ -+ gen_pool_free (tmio->poolp, (unsigned long) cpu_addr, size); -+} -+ -+/*-------------------------------------------------------------------------*/ -+ -+static void *tmio_dma_pool_alloc(struct dma_pool *pool, gfp_t mem_flags, -+ dma_addr_t *handle) -+{ -+ struct tmio_dma_pool *tdp = (struct tmio_dma_pool*) pool; -+ return tmio_dma_alloc (tdp->dev, tdp->size, handle, mem_flags); -+} -+ -+static void -+tmio_dma_pool_free (struct dma_pool *pool, void *vaddr, dma_addr_t addr) -+{ -+ struct tmio_dma_pool *tdp = (struct tmio_dma_pool*) pool; -+ return tmio_dma_free (tdp->dev, tdp->size, vaddr, addr); -+} -+ -+static void *tmio_buffer_alloc (struct usb_bus *bus, size_t size, -+ gfp_t mem_flags, dma_addr_t *dma) -+{ -+ return tmio_dma_alloc (bus->controller, size, dma, mem_flags); -+} -+ -+static void tmio_buffer_free (struct usb_bus *bus, size_t size, -+ void *addr, dma_addr_t dma) -+{ -+ tmio_dma_free (bus->controller, size, addr, dma); -+} -+ -+/*-------------------------------------------------------------------------*/ -+ -+static void tmio_hc_stop (struct usb_hcd *hcd) -+{ -+ struct device* dev = hcd->self.controller; -+ struct tmio_device* tdev = dev_to_tdev (dev); -+ struct tmio_hcd* tmio = hcd_to_tmio (hcd); -+ struct tmio_uhccr __iomem* ccr = tmio->ccr; -+ union tmio_uhccr_pm pm = {0}; -+ -+ pm.gcken = 1; -+ pm.ckrnen = 1; -+ pm.uspw1 = 1; -+ pm.uspw2 = 1; -+ -+ iowrite8 (0, &ccr->intc); -+ iowrite8 (0, &ccr->ilme); -+ iowrite16(0, &ccr->basel); -+ iowrite16(0, &ccr->baseh); -+ iowrite16(pm.raw, &ccr->pm); -+ -+ tdev->ops->function (dev, 0); -+ tdev->ops->clock (dev, 0); -+} -+ -+static void tmio_hc_start (struct usb_hcd *hcd) -+{ -+ struct device* dev = hcd->self.controller; -+ struct tmio_device* tdev = dev_to_tdev (dev); -+ struct tmio_hcd* tmio = hcd_to_tmio (hcd); -+ struct tmio_uhccr __iomem* ccr = tmio->ccr; -+ union tmio_uhccr_pm pm = {0}; -+ -+ pm.pmes = 1; -+ pm.pmee = 1; -+ pm.ckrnen = 1; -+ pm.gcken = 1; -+ -+ tdev->ops->clock (dev, 1); -+ tdev->ops->function (dev, 1); -+ -+ iowrite16(pm.raw, &ccr->pm); -+ iowrite16(hcd->rsrc_start, &ccr->basel); -+ iowrite16(hcd->rsrc_start >> 16, &ccr->baseh); -+ iowrite8 (1, &ccr->ilme); -+ iowrite8 (2, &ccr->intc); -+ -+ consistent_sync (tmio->sram, tmio->sram_len, DMA_BIDIRECTIONAL); -+ -+ dev_info (dev, "revision %d @ 0x%08llx, irq %d\n", -+ ioread8 (&ccr->revid), hcd->rsrc_start, hcd->irq); -+} -+ -+static void tmio_stop (struct usb_hcd *hcd) -+{ -+ struct ohci_hcd* ohci = hcd_to_ohci (hcd); -+ -+ /* NULL these so ohci_stop() doesn't try to free them */ -+ ohci->hcca = NULL; -+ ohci->td_cache = NULL; -+ ohci->ed_cache = NULL; -+ -+ ohci_stop (hcd); -+ tmio_hc_stop (hcd); -+ tmio_pool_destroy (hcd_to_tmio (hcd)); -+ -+ /* We don't free the hcca because tmio_hc_stop() turns off -+ * the sram and the memory allocation data is destroyed. */ -+} -+ -+static int tmio_start (struct usb_hcd *hcd) -+{ -+ struct device* dev = hcd->self.controller; -+ struct ohci_hcd* ohci = hcd_to_ohci (hcd); -+ struct tmio_hcd* tmio = hcd_to_tmio (hcd); -+ int retval; -+ -+ tmio_hc_start (hcd); -+ -+ tmio->poolp = gen_pool_create(0, fls(tmio->sram_len) - 1, -+ tmio_pool_callback, 0); -+ if (!tmio->poolp) { -+ retval = -ENOMEM; -+ goto err_gen_pool_create; -+ } -+ -+ gen_pool_free (tmio->poolp, (unsigned long)(tmio->sram), -+ tmio->sram_len); -+ -+ ohci->hcca = tmio_dma_alloc (dev, sizeof *ohci->hcca, -+ &ohci->hcca_dma, GFP_KERNEL); -+ if (!ohci->hcca) { -+ retval = -ENOMEM; -+ goto err_hcca_alloc; -+ } -+ -+ /* for our dma_pool_alloc/free hooks */ -+ ohci->td_cache = (struct dma_pool*) &tmio->td_pool; -+ ohci->ed_cache = (struct dma_pool*) &tmio->ed_pool; -+ -+ if ((retval = ohci_init (ohci)) < 0) -+ goto err_ohci_init; -+ -+ if ((retval = ohci_run (ohci)) < 0) -+ goto err_ohci_run; -+ -+ return 0; -+ -+err_ohci_run: -+ err ("can't start %s", hcd->self.bus_name); -+err_ohci_init: -+err_hcca_alloc: -+err_gen_pool_create: -+ tmio_stop (hcd); -+ return retval; -+} -+ -+/*-------------------------------------------------------------------------*/ -+ -+static inline void *tmio_urb_dma_to_virt(struct urb *urb, dma_addr_t dma) -+{ -+ if (BOUNDED_XYL(dma, urb->transfer_dma, urb->transfer_buffer_length)) -+ return urb->transfer_buffer + dma - urb->transfer_dma; -+ -+ if (BOUNDED_XYL(dma, urb->setup_dma, sizeof (struct usb_ctrlrequest))) -+ return urb->setup_packet + dma - urb->setup_dma; -+ -+ return NULL; -+} -+ -+static struct tmio_td* tmio_td_find (struct td *td, int *index) -+{ -+ struct urb* urb = td->urb; -+ urb_priv_t* urb_priv = urb->hcpriv; -+ struct tmio_urb* turb = urb_to_turb (urb); -+ int i; -+ -+ for (i=0; i < urb_priv->length; i++) -+ if (urb_priv->td[i] == td) { -+ *index = i; -+ return turb->td + i; -+ } -+ -+ return NULL; -+} -+ -+/* -+ * map the td's data to dma-able memory -+ * -+ * if this td transfers data, -+ * sets tmtd->data to the urb's data buffer -+ * sets tmtd->dma to dma-able memory -+ * sets tmtd->bounce to non-NULL if a bounce buffer is allocated -+ * copies the urb's data buffer to the bounce buffer if necessary -+ */ -+static int tmio_td_dma_map (struct ohci_hcd *ohci, struct urb *urb, -+ struct tmio_td *tmtd, int idx) -+{ -+ struct usb_hcd* hcd = ohci_to_hcd (ohci); -+ struct device* dev = hcd->self.controller; -+ dma_addr_t dma; -+ -+ if (!tmtd->len) -+ return 0; -+ -+ if (tmio_dma_to_virt (hcd, tmtd->dma)) -+ return 0; -+ -+ tmtd->data = tmio_urb_dma_to_virt (urb, tmtd->dma); -+ if (unlikely (!tmtd->data)) { -+ dev_err (dev, "TD has bad dma address 0x%08x\n", tmtd->dma); -+ return 0; -+ } -+ -+ tmtd->bounce = tmio_dma_alloc (dev, tmtd->len, &dma, GFP_ATOMIC); -+ if (!tmtd->bounce) -+ return -ENOMEM; -+ -+ if ((usb_pipecontrol (urb->pipe) && !idx) || usb_pipeout (urb->pipe)) { -+ consistent_sync (tmtd->bounce, tmtd->len, DMA_TO_DEVICE); -+ memcpy (tmtd->bounce, tmtd->data, tmtd->len); -+ } else -+ consistent_sync (tmtd->bounce, tmtd->len, DMA_FROM_DEVICE); -+ -+ tmtd->dma = dma; -+ return 0; -+} -+ -+/* -+ * unmaps the td's data from dma-able memory -+ * -+ * if a bounce buffer has been allocated, -+ * copy the bounce buffer to the urb's data buffer if necessary -+ * free the bounce buffer -+ */ -+static void tmio_td_dma_unmap (struct ohci_hcd *ohci, struct td *td) -+{ -+ struct device* dev = (ohci_to_hcd (ohci))->self.controller; -+ struct urb* urb = td->urb; -+ int idx; -+ struct tmio_td* tmtd = tmio_td_find (td, &idx); -+ -+ if (!tmtd->bounce) -+ return; -+ -+ if (usb_pipein (urb->pipe) && (usb_pipecontrol (urb->pipe) ? idx : 1)) { -+ memcpy (tmtd->data, tmtd->bounce, tmtd->len); -+ consistent_sync (tmtd->data, tmtd->len, DMA_TO_DEVICE); -+ } -+ -+ tmio_dma_free (dev, tmtd->len, tmtd->bounce, tmtd->dma); -+} -+ -+static int tmio_urb_runqueue (struct ohci_hcd *ohci, struct urb *urb) -+{ -+ struct tmio_urb* turb = urb_to_turb (urb); -+ urb_priv_t* urb_priv= urb->hcpriv; -+ int start = turb->td_queue; -+ int retval = 0; -+ int i; -+ -+ for (i = start; i < turb->td_add; i = ++turb->td_queue) { -+ struct tmio_td *tmtd = turb->td + i; -+ -+ if ((retval = tmio_td_dma_map (ohci, urb, tmtd, i))) -+ break; -+ -+ td_fill (ohci, tmtd->info, tmtd->dma, tmtd->len, urb, i); -+ } -+ -+ if (i <= start) -+ return retval; -+ -+ /* kickstart the appropriate list */ -+ wmb (); -+ switch (urb_priv->ed->type) { -+ case PIPE_BULK: -+ ohci_writel (ohci, OHCI_BLF, &ohci->regs->cmdstatus); -+ break; -+ case PIPE_CONTROL: -+ ohci_writel (ohci, OHCI_CLF, &ohci->regs->cmdstatus); -+ break; -+ } -+ -+ return retval; -+} -+ -+/* -+ * This needs to be called with ohci->lock held so the pending urb list -+ * isn't modified. -+ */ -+static int tmio_ohci_runqueue (struct ohci_hcd *ohci) -+{ -+ urb_priv_t* priv; -+ int retval = 0; -+ -+ list_for_each_entry_reverse (priv, &ohci->pending, pending) -+ if ((retval = tmio_urb_runqueue (ohci, priv->td[0]->urb))) -+ return retval; -+ -+ return retval; -+} -+ -+static void tmio_td_fill (struct ohci_hcd *ohci, u32 info, -+ dma_addr_t data, int len, struct urb *urb, int index) -+{ -+ struct tmio_urb* turb = urb_to_turb (urb); -+ struct tmio_td* tmtd = turb->td + index; -+ -+ tmtd->data = NULL; -+ tmtd->bounce = NULL; -+ tmtd->dma = data; -+ tmtd->len = len; -+ tmtd->info = info; -+ turb->td_add = index + 1; -+} -+ -+static void -+tmio_td_done (struct ohci_hcd *ohci, struct urb *urb, struct td *td) -+{ -+ tmio_td_dma_unmap (ohci, td); -+ td_done (ohci, urb, td); -+} -+ -+const static struct ohci_ops tmio_ops = { -+ .dma_pool_alloc = tmio_dma_pool_alloc, -+ .dma_pool_free = tmio_dma_pool_free, -+ .td_fill = tmio_td_fill, -+ .td_done = tmio_td_done, -+}; -+ -+/*-------------------------------------------------------------------------*/ -+ -+static irqreturn_t tmio_irq (struct usb_hcd *hcd, struct pt_regs *ptregs) -+{ -+ irqreturn_t retval = ohci_irq (hcd, ptregs); -+ -+ if (retval == IRQ_HANDLED) { -+ struct ohci_hcd *ohci = hcd_to_ohci (hcd); -+ unsigned long flags; -+ -+ spin_lock_irqsave(&ohci->lock, flags); -+ tmio_ohci_runqueue (ohci); -+ spin_unlock_irqrestore (&ohci->lock, flags); -+ } -+ -+ return retval; -+} -+ -+/* -+ * This is ohci_urb_enqueue with: -+ * dma address sanitization for tmio_urb_dma_to_virt() -+ * allocate extra space in urb_priv for our private data -+ * initialize urb_priv->td[0]->urb for tmio_ohci_runqueue() -+ * call tmio_ohci_runqueue() after submitting TDs -+ */ -+static int tmio_urb_enqueue ( -+ struct usb_hcd *hcd, -+ struct usb_host_endpoint *ep, -+ struct urb *urb, -+ gfp_t mem_flags -+) { -+ struct ohci_hcd *ohci = hcd_to_ohci (hcd); -+ struct ed *ed; -+ urb_priv_t *urb_priv; -+ unsigned int pipe = urb->pipe; -+ int i, size = 0; -+ unsigned long flags; -+ int retval = 0; -+ -+#ifdef OHCI_VERBOSE_DEBUG -+ urb_print (urb, "SUB", usb_pipein (pipe)); -+#endif -+ -+ /* make sure we can convert dma offsets back to virtual addresses */ -+ if (!tmio_dma_to_virt (hcd, urb->setup_dma)) -+ urb->setup_dma = 0; -+ -+ if (!tmio_dma_to_virt (hcd, urb->transfer_dma)) -+ urb->transfer_dma = sizeof (struct usb_ctrlrequest); -+ -+ /* every endpoint has a ed, locate and maybe (re)initialize it */ -+ if (! (ed = ed_get (ohci, ep, urb->dev, pipe, urb->interval))) -+ return -ENOMEM; -+ -+ /* for the private part of the URB we need the number of TDs (size) */ -+ switch (ed->type) { -+ case PIPE_CONTROL: -+ /* td_submit_urb() doesn't yet handle these */ -+ if (urb->transfer_buffer_length > 4096) -+ return -EMSGSIZE; -+ -+ /* 1 TD for setup, 1 for ACK, plus ... */ -+ size = 2; -+ /* FALLTHROUGH */ -+ // case PIPE_INTERRUPT: -+ // case PIPE_BULK: -+ default: -+ /* one TD for every 4096 Bytes (can be upto 8K) */ -+ size += urb->transfer_buffer_length / 4096; -+ /* ... and for any remaining bytes ... */ -+ if ((urb->transfer_buffer_length % 4096) != 0) -+ size++; -+ /* ... and maybe a zero length packet to wrap it up */ -+ if (size == 0) -+ size++; -+ else if ((urb->transfer_flags & URB_ZERO_PACKET) != 0 -+ && (urb->transfer_buffer_length -+ % usb_maxpacket (urb->dev, pipe, -+ usb_pipeout (pipe))) == 0) -+ size++; -+ break; -+ case PIPE_ISOCHRONOUS: /* number of packets from URB */ -+ size = urb->number_of_packets; -+ break; -+ } -+ -+ /* allocate the private part of the URB */ -+ urb_priv = kzalloc (sizeof (urb_priv_t) -+ + size * sizeof (struct td*) -+ + sizeof (struct tmio_urb) -+ + size * sizeof (struct tmio_td), -+ mem_flags); -+ if (!urb_priv) -+ return -ENOMEM; -+ INIT_LIST_HEAD (&urb_priv->pending); -+ urb_priv->length = size; -+ urb_priv->ed = ed; -+ -+ /* allocate the TDs (deferring hash chain updates) */ -+ for (i = 0; i < size; i++) { -+ urb_priv->td [i] = td_alloc (ohci, mem_flags); -+ if (!urb_priv->td [i]) { -+ urb_priv->length = i; -+ urb_free_priv (ohci, urb_priv); -+ return -ENOMEM; -+ } -+ urb_priv->td [i]->urb = urb; -+ } -+ -+ spin_lock_irqsave (&ohci->lock, flags); -+ -+ /* don't submit to a dead HC */ -+ if (!HC_IS_RUNNING(hcd->state)) { -+ retval = -ENODEV; -+ goto fail; -+ } -+ -+ /* in case of unlink-during-submit */ -+ spin_lock (&urb->lock); -+ if (urb->status != -EINPROGRESS) { -+ spin_unlock (&urb->lock); -+ urb->hcpriv = urb_priv; -+ finish_urb (ohci, urb, NULL); -+ retval = 0; -+ goto fail; -+ } -+ -+ /* schedule the ed if needed */ -+ if (ed->state == ED_IDLE) { -+ retval = ed_schedule (ohci, ed); -+ if (retval < 0) -+ goto fail0; -+ if (ed->type == PIPE_ISOCHRONOUS) { -+ u16 frame = ohci_frame_no(ohci); -+ -+ /* delay a few frames before the first TD */ -+ frame += max_t (u16, 8, ed->interval); -+ frame &= ~(ed->interval - 1); -+ frame |= ed->branch; -+ urb->start_frame = frame; -+ -+ /* yes, only URB_ISO_ASAP is supported, and -+ * urb->start_frame is never used as input. -+ */ -+ } -+ } else if (ed->type == PIPE_ISOCHRONOUS) -+ urb->start_frame = ed->last_iso + ed->interval; -+ -+ /* fill the TDs and link them to the ed; and -+ * enable that part of the schedule, if needed -+ * and update count of queued periodic urbs -+ */ -+ urb->hcpriv = urb_priv; -+ td_submit_urb (ohci, urb); -+ tmio_ohci_runqueue (ohci); -+ -+fail0: -+ spin_unlock (&urb->lock); -+fail: -+ if (retval) -+ urb_free_priv (ohci, urb_priv); -+ spin_unlock_irqrestore (&ohci->lock, flags); -+ return retval; -+} -+ -+static const struct hc_driver tmio_hc_driver = { -+ .description = hcd_name, -+ .product_desc = "TMIO OHCI USB Host Controller", -+ .hcd_priv_size = sizeof (struct ohci_hcd) -+ + sizeof (struct tmio_hcd), -+ -+ /* generic hardware linkage */ -+ .irq = tmio_irq, -+ .flags = HCD_USB11 | HCD_MEMORY, -+ -+ /* basic lifecycle operations */ -+ .start = tmio_start, -+ .stop = tmio_stop, -+ -+ /* managing i/o requests and associated device resources */ -+ .urb_enqueue = tmio_urb_enqueue, -+ .urb_dequeue = ohci_urb_dequeue, -+ .endpoint_disable = ohci_endpoint_disable, -+ -+ /* scheduling support */ -+ .get_frame_number = ohci_get_frame, -+ -+ /* root hub support */ -+ .hub_status_data = ohci_hub_status_data, -+ .hub_control = ohci_hub_control, -+#ifdef CONFIG_PM -+ .bus_suspend = ohci_bus_suspend, -+ .bus_resume = ohci_bus_resume, -+#endif -+ .start_port_reset = ohci_start_port_reset, -+}; -+ -+/*-------------------------------------------------------------------------*/ -+ -+/* configure so an HC device and id are always provided */ -+/* always called with process context; sleeping is OK */ -+ -+/** -+ * tmio_probe - initialize TMIO-based HCDs -+ * Context: !in_interrupt() -+ * -+ * Allocates basic resources for this USB host controller, and -+ * then invokes the start() method for the HCD associated with it -+ * through the hotplug entry's driver_data. -+ */ -+static int tmio_probe (struct device *dev) -+{ -+ struct tmio_device* tdev = dev_to_tdev (dev); -+ struct resource* config = tmio_resource_config (dev); -+ struct resource* regs = tmio_resource_control (dev); -+ struct resource* sram = tmio_resource_mem (dev); -+ struct resource* irq = tmio_resource_irq (dev); -+ struct usb_operations* ops; -+ struct tmio_hcd* tmio; -+ struct ohci_hcd* ohci; -+ struct usb_hcd* hcd; -+ int retval; -+ -+ if (usb_disabled ()) -+ return -ENODEV; -+ -+ if (dev->dma_mask || dev->coherent_dma_mask) { -+ dev_err (dev, "DMA not supported\n"); -+ return -ENODEV; -+ } -+ -+ hcd = usb_create_hcd (&tmio_hc_driver, dev, dev->bus_id); -+ if (!hcd) { -+ retval = -ENOMEM; -+ goto err_create_hcd; -+ } -+ -+ tmio = hcd_to_tmio (hcd); -+ tmio->td_pool.dev = dev; -+ tmio->ed_pool.dev = dev; -+ tmio->td_pool.size = sizeof (struct td); -+ tmio->ed_pool.size = sizeof (struct ed); -+ ohci = hcd_to_ohci (hcd); -+ ohci_hcd_init (ohci); -+ ohci->ops = &tmio_ops; -+ -+ retval = request_resource (tdev->iomem, config); -+ if (retval) -+ goto err_request_config_resource; -+ -+ retval = request_resource (tdev->iomem, regs); -+ if (retval) -+ goto err_request_regs_resource; -+ -+ retval = request_resource (tdev->iomem, sram); -+ if (retval) -+ goto err_request_sram_resource; -+ -+ hcd->rsrc_start = regs->start; -+ hcd->rsrc_len = regs->end - regs->start + 1; -+ tmio->sram_len = sram->end - sram->start + 1; -+ -+ tmio->ccr = ioremap (config->start, config->end - config->start + 1); -+ if (!tmio->ccr) { -+ retval = -ENOMEM; -+ goto err_ioremap_ccr; -+ } -+ -+ hcd->regs = ioremap (hcd->rsrc_start, hcd->rsrc_len); -+ if (!hcd->regs) { -+ retval = -ENOMEM; -+ goto err_ioremap_regs; -+ } -+ -+ tmio->sram = ioremap (sram->start, tmio->sram_len); -+ if (!tmio->sram) { -+ retval = -ENOMEM; -+ goto err_ioremap_sram; -+ } -+ -+ /* drivers should use our coherent buffer allocator */ -+ ops = &tmio->ops; -+ memcpy (ops, hcd->self.op, sizeof *ops); -+ ops->buffer_alloc = tmio_buffer_alloc; -+ ops->buffer_free = tmio_buffer_free; -+ hcd->self.op = ops; -+ -+ retval = usb_add_hcd (hcd, irq->start, SA_INTERRUPT); -+ if (retval) -+ goto err_usb_add_hcd; -+ -+ return 0; -+ -+err_usb_add_hcd: -+ iounmap (tmio->sram); -+err_ioremap_sram: -+ iounmap (hcd->regs); -+err_ioremap_regs: -+ iounmap (tmio->ccr); -+err_ioremap_ccr: -+ release_resource (sram); -+err_request_sram_resource: -+ release_resource (regs); -+err_request_regs_resource: -+ release_resource (config); -+err_request_config_resource: -+ usb_put_hcd (hcd); -+err_create_hcd: -+ return retval; -+} -+ -+/* may be called without controller electrically present */ -+/* may be called with controller, bus, and devices active */ -+ -+/** -+ * tmio_remove - shutdown processing for TMIO-based HCDs -+ * @dev: USB Host Controller being removed -+ * Context: !in_interrupt() -+ * -+ * Reverses the effect of tmio_probe(), first invoking -+ * the HCD's stop() method. It is always called from a thread -+ * context, normally "rmmod", "apmd", or something similar. -+ */ -+static int tmio_remove (struct device *dev) -+{ -+ struct usb_hcd* hcd = dev_get_drvdata (dev); -+ struct tmio_hcd* tmio = hcd_to_tmio (hcd); -+ -+ usb_remove_hcd (hcd); -+ iounmap (tmio->sram); -+ iounmap (hcd->regs); -+ iounmap (tmio->ccr); -+ release_resource (tmio_resource_mem (dev)); -+ release_resource (tmio_resource_control (dev)); -+ release_resource (tmio_resource_config (dev)); -+ usb_put_hcd (hcd); -+ return 0; -+} -+ -+static struct device_driver tmio_ohci = { -+ .name = TMIO_NAME_OHCI, -+ .bus = &tmio_bus_type, -+ .probe = tmio_probe, -+ .remove = tmio_remove, -+}; -+ -+static int __init tmio_init (void) -+{ -+ dbg (DRIVER_INFO " (%s)", TMIO_SOC_NAME); -+ dbg ("block sizes: ed %d td %d", -+ sizeof (struct ed), sizeof (struct td)); -+ -+ return driver_register (&tmio_ohci); -+} -+ -+static void __exit tmio_exit (void) -+{ -+ driver_unregister (&tmio_ohci); -+} -+ -+module_init (tmio_init); -+module_exit (tmio_exit); -Index: git/drivers/usb/host/Kconfig -=================================================================== ---- git.orig/drivers/usb/host/Kconfig 2006-11-07 21:46:32.000000000 +0000 -+++ git/drivers/usb/host/Kconfig 2006-11-07 21:48:38.000000000 +0000 -@@ -84,6 +84,7 @@ config USB_OHCI_HCD - depends on USB && USB_ARCH_HAS_OHCI - select ISP1301_OMAP if MACH_OMAP_H2 || MACH_OMAP_H3 - select I2C if ARCH_PNX4008 -+ select GENERIC_ALLOCATOR if TOSHIBA_TC6393XB - ---help--- - The Open Host Controller Interface (OHCI) is a standard for accessing - USB 1.1 host controller hardware. It does more in hardware than Intel's -Index: git/drivers/usb/host/ohci-hcd.c -=================================================================== ---- git.orig/drivers/usb/host/ohci-hcd.c 2006-11-07 21:46:32.000000000 +0000 -+++ git/drivers/usb/host/ohci-hcd.c 2006-11-07 21:48:33.000000000 +0000 -@@ -944,6 +944,7 @@ MODULE_LICENSE ("GPL"); - || defined(CONFIG_ARCH_OMAP) \ - || defined (CONFIG_ARCH_LH7A404) \ - || defined (CONFIG_PXA27x) \ -+ || defined (CONFIG_TOSHIBA_TC6393XB) \ - || defined (CONFIG_ARCH_EP93XX) \ - || defined (CONFIG_SOC_AU1X00) \ - || defined (CONFIG_USB_OHCI_HCD_PPC_SOC) \ |