2007-01-16 12:11:47 +08:00
|
|
|
/*
|
|
|
|
* PS3 EHCI Host Controller driver
|
|
|
|
*
|
|
|
|
* Copyright (C) 2006 Sony Computer Entertainment Inc.
|
|
|
|
* Copyright 2006 Sony Corp.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; version 2 of the License.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*/
|
|
|
|
|
2007-06-06 11:04:35 +08:00
|
|
|
#include <asm/firmware.h>
|
2007-01-16 12:11:47 +08:00
|
|
|
#include <asm/ps3.h>
|
|
|
|
|
2011-11-23 10:33:45 +08:00
|
|
|
static void ps3_ehci_setup_insnreg(struct ehci_hcd *ehci)
|
|
|
|
{
|
|
|
|
/* PS3 HC internal setup register offsets. */
|
|
|
|
|
|
|
|
enum ps3_ehci_hc_insnreg {
|
|
|
|
ps3_ehci_hc_insnreg01 = 0x084,
|
|
|
|
ps3_ehci_hc_insnreg02 = 0x088,
|
|
|
|
ps3_ehci_hc_insnreg03 = 0x08c,
|
|
|
|
};
|
|
|
|
|
|
|
|
/* PS3 EHCI HC errata fix 316 - The PS3 EHCI HC will reset its
|
|
|
|
* internal INSNREGXX setup regs back to the chip default values
|
|
|
|
* on Host Controller Reset (CMD_RESET) or Light Host Controller
|
|
|
|
* Reset (CMD_LRESET). The work-around for this is for the HC
|
|
|
|
* driver to re-initialise these regs when ever the HC is reset.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Set burst transfer counts to 256 out, 32 in. */
|
|
|
|
|
|
|
|
writel_be(0x01000020, (void __iomem *)ehci->regs +
|
|
|
|
ps3_ehci_hc_insnreg01);
|
|
|
|
|
|
|
|
/* Enable burst transfer counts. */
|
|
|
|
|
|
|
|
writel_be(0x00000001, (void __iomem *)ehci->regs +
|
|
|
|
ps3_ehci_hc_insnreg03);
|
|
|
|
}
|
|
|
|
|
2007-01-16 12:11:47 +08:00
|
|
|
static int ps3_ehci_hc_reset(struct usb_hcd *hcd)
|
|
|
|
{
|
|
|
|
int result;
|
|
|
|
struct ehci_hcd *ehci = hcd_to_ehci(hcd);
|
|
|
|
|
|
|
|
ehci->big_endian_mmio = 1;
|
|
|
|
ehci->caps = hcd->regs;
|
|
|
|
|
2012-07-10 03:55:14 +08:00
|
|
|
result = ehci_setup(hcd);
|
2007-01-16 12:11:47 +08:00
|
|
|
if (result)
|
|
|
|
return result;
|
|
|
|
|
2011-11-23 10:33:45 +08:00
|
|
|
ps3_ehci_setup_insnreg(ehci);
|
|
|
|
|
2007-01-16 12:11:47 +08:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct hc_driver ps3_ehci_hc_driver = {
|
|
|
|
.description = hcd_name,
|
|
|
|
.product_desc = "PS3 EHCI Host Controller",
|
|
|
|
.hcd_priv_size = sizeof(struct ehci_hcd),
|
|
|
|
.irq = ehci_irq,
|
USB: EHCI: support running URB giveback in tasklet context
All 4 transfer types can work well on EHCI HCD after switching to run
URB giveback in tasklet context, so mark all HCD drivers to support
it.
Also we don't need to release ehci->lock during URB giveback any more.
>From below test results on 3 machines(2 ARM and one x86), time
consumed by EHCI interrupt handler droped much without performance
loss.
1 test description
1.1 mass storage performance test:
- run below command 10 times and compute the average performance
dd if=/dev/sdN iflag=direct of=/dev/null bs=200M count=1
- two usb mass storage device:
A: sandisk extreme USB 3.0 16G(used in test case 1 & case 2)
B: kingston DataTraveler G2 4GB(only used in test case 2)
1.2 uvc function test:
- run one simple capture program in the below link
http://kernel.ubuntu.com/~ming/up/capture.c
- capture format 640*480 and results in High Bandwidth mode on the
uvc device: Z-Star 0x0ac8/0x3450
- on T410(x86) laptop, also use guvcview to watch video capture/playback
1.3 about test2 and test4
- both two devices involved are tested concurrently by above test items
1.4 how to compute irq time(the time consumed by ehci_irq)
- use trace points of irq:irq_handler_entry and irq:irq_handler_exit
1.5 kernel
3.10.0-rc3-next-20130528
1.6 test machines
Pandaboard A1: ARM CortexA9 dural core
Arndale board: ARM CortexA15 dural core
T410: i5 CPU 2.67GHz quad core
2 test result
2.1 test case1: single mass storage device performance test
--------------------------------------------------------------------
upstream | patched
perf(MB/s)+irq time(us) | perf(MB/s)+irq time(us)
--------------------------------------------------------------------
Pandaboard A1: 25.280(avg:145,max:772) | 25.540(avg:14, max:75)
Arndale board: 29.700(avg:33, max:129) | 29.700(avg:10, max:50)
T410: 34.430(avg:17, max:154*)| 34.660(avg:12, max:155)
---------------------------------------------------------------------
2.2 test case2: two mass storage devices' performance test
--------------------------------------------------------------------
upstream | patched
perf(MB/s)+irq time(us) | perf(MB/s)+irq time(us)
--------------------------------------------------------------------
Pandaboard A1: 15.840/15.580(avg:158,max:1216) | 16.500/16.160(avg:15,max:139)
Arndale board: 17.370/16.220(avg:33 max:234) | 17.480/16.200(avg:11, max:91)
T410: 21.180/19.820(avg:18 max:160) | 21.220/19.880(avg:11, max:149)
---------------------------------------------------------------------
2.3 test case3: one uvc streaming test
- uvc device works well(on x86, luvcview can be used too and has
same result with uvc capture)
--------------------------------------------------------------------
upstream | patched
irq time(us) | irq time(us)
--------------------------------------------------------------------
Pandaboard A1: (avg:445, max:873) | (avg:33, max:44)
Arndale board: (avg:316, max:630) | (avg:20, max:27)
T410: (avg:39, max:107) | (avg:10, max:65)
---------------------------------------------------------------------
2.4 test case4: one uvc streaming plus one mass storage device test
--------------------------------------------------------------------
upstream | patched
perf(MB/s)+irq time(us) | perf(MB/s)+irq time(us)
--------------------------------------------------------------------
Pandaboard A1: 20.340(avg:259,max:1704)| 20.390(avg:24, max:101)
Arndale board: 23.460(avg:124,max:726) | 23.370(avg:15, max:52)
T410: 28.520(avg:27, max:169) | 28.630(avg:13, max:160)
---------------------------------------------------------------------
2.5 test case5: read single mass storage device with small transfer
- run below command 10 times and compute the average speed
dd if=/dev/sdN iflag=direct of=/dev/null bs=4K count=4000
1), test device A:
--------------------------------------------------------------------
upstream | patched
perf(MB/s)+irq time(us) | perf(MB/s)+irq time(us)
--------------------------------------------------------------------
Pandaboard A1: 6.5(avg:21, max:64) | 6.5(avg:10, max:24)
Arndale board: 8.13(avg:12, max:23) | 8.06(avg:7, max:17)
T410: 6.66(avg:13, max:131) | 6.84(avg:11, max:149)
---------------------------------------------------------------------
2), test device B:
--------------------------------------------------------------------
upstream | patched
perf(MB/s)+irq time(us) | perf(MB/s)+irq time(us)
--------------------------------------------------------------------
Pandaboard A1: 5.5(avg:21,max:43) | 5.49(avg:10, max:24)
Arndale board: 5.9(avg:12, max:22) | 5.9(avg:7, max:17)
T410: 5.48(avg:13, max:155) | 5.48(avg:7, max:140)
---------------------------------------------------------------------
* On T410, sometimes read ehci status register in ehci_irq takes more
than 100us, and the problem has been reported on the link:
http://marc.info/?t=137065867300001&r=1&w=2
Acked-by: Alan Stern <stern@rowland.harvard.edu>
Signed-off-by: Ming Lei <ming.lei@canonical.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
2013-07-03 22:53:11 +08:00
|
|
|
.flags = HCD_MEMORY | HCD_USB2 | HCD_BH,
|
2007-01-16 12:11:47 +08:00
|
|
|
.reset = ps3_ehci_hc_reset,
|
|
|
|
.start = ehci_run,
|
|
|
|
.stop = ehci_stop,
|
|
|
|
.shutdown = ehci_shutdown,
|
|
|
|
.urb_enqueue = ehci_urb_enqueue,
|
|
|
|
.urb_dequeue = ehci_urb_dequeue,
|
|
|
|
.endpoint_disable = ehci_endpoint_disable,
|
2009-05-28 06:21:56 +08:00
|
|
|
.endpoint_reset = ehci_endpoint_reset,
|
2007-01-16 12:11:47 +08:00
|
|
|
.get_frame_number = ehci_get_frame,
|
|
|
|
.hub_status_data = ehci_hub_status_data,
|
|
|
|
.hub_control = ehci_hub_control,
|
|
|
|
#if defined(CONFIG_PM)
|
|
|
|
.bus_suspend = ehci_bus_suspend,
|
|
|
|
.bus_resume = ehci_bus_resume,
|
|
|
|
#endif
|
2007-11-22 04:28:14 +08:00
|
|
|
.relinquish_port = ehci_relinquish_port,
|
2008-05-21 04:58:29 +08:00
|
|
|
.port_handed_over = ehci_port_handed_over,
|
2009-06-29 22:47:30 +08:00
|
|
|
|
|
|
|
.clear_tt_buffer_complete = ehci_clear_tt_buffer_complete,
|
2007-01-16 12:11:47 +08:00
|
|
|
};
|
|
|
|
|
2012-11-20 02:21:48 +08:00
|
|
|
static int ps3_ehci_probe(struct ps3_system_bus_device *dev)
|
2007-01-16 12:11:47 +08:00
|
|
|
{
|
|
|
|
int result;
|
|
|
|
struct usb_hcd *hcd;
|
|
|
|
unsigned int virq;
|
2009-04-07 10:01:15 +08:00
|
|
|
static u64 dummy_mask = DMA_BIT_MASK(32);
|
2007-01-16 12:11:47 +08:00
|
|
|
|
|
|
|
if (usb_disabled()) {
|
|
|
|
result = -ENODEV;
|
|
|
|
goto fail_start;
|
|
|
|
}
|
|
|
|
|
2007-06-06 11:04:35 +08:00
|
|
|
result = ps3_open_hv_device(dev);
|
|
|
|
|
|
|
|
if (result) {
|
|
|
|
dev_dbg(&dev->core, "%s:%d: ps3_open_hv_device failed\n",
|
|
|
|
__func__, __LINE__);
|
|
|
|
goto fail_open;
|
|
|
|
}
|
|
|
|
|
|
|
|
result = ps3_dma_region_create(dev->d_region);
|
|
|
|
|
|
|
|
if (result) {
|
|
|
|
dev_dbg(&dev->core, "%s:%d: ps3_dma_region_create failed: "
|
|
|
|
"(%d)\n", __func__, __LINE__, result);
|
|
|
|
BUG_ON("check region type");
|
|
|
|
goto fail_dma_region;
|
|
|
|
}
|
|
|
|
|
2007-01-16 12:11:47 +08:00
|
|
|
result = ps3_mmio_region_create(dev->m_region);
|
|
|
|
|
|
|
|
if (result) {
|
|
|
|
dev_dbg(&dev->core, "%s:%d: ps3_map_mmio_region failed\n",
|
|
|
|
__func__, __LINE__);
|
|
|
|
result = -EPERM;
|
2007-06-06 11:04:35 +08:00
|
|
|
goto fail_mmio_region;
|
2007-01-16 12:11:47 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
dev_dbg(&dev->core, "%s:%d: mmio mapped_addr %lxh\n", __func__,
|
|
|
|
__LINE__, dev->m_region->lpar_addr);
|
|
|
|
|
2007-05-01 05:01:01 +08:00
|
|
|
result = ps3_io_irq_setup(PS3_BINDING_CPU_ANY, dev->interrupt_id, &virq);
|
2007-01-16 12:11:47 +08:00
|
|
|
|
|
|
|
if (result) {
|
|
|
|
dev_dbg(&dev->core, "%s:%d: ps3_construct_io_irq(%d) failed.\n",
|
|
|
|
__func__, __LINE__, virq);
|
|
|
|
result = -EPERM;
|
|
|
|
goto fail_irq;
|
|
|
|
}
|
|
|
|
|
|
|
|
dev->core.dma_mask = &dummy_mask; /* FIXME: for improper usb code */
|
|
|
|
|
2008-05-02 12:02:41 +08:00
|
|
|
hcd = usb_create_hcd(&ps3_ehci_hc_driver, &dev->core, dev_name(&dev->core));
|
2007-01-16 12:11:47 +08:00
|
|
|
|
|
|
|
if (!hcd) {
|
|
|
|
dev_dbg(&dev->core, "%s:%d: usb_create_hcd failed\n", __func__,
|
|
|
|
__LINE__);
|
|
|
|
result = -ENOMEM;
|
|
|
|
goto fail_create_hcd;
|
|
|
|
}
|
|
|
|
|
|
|
|
hcd->rsrc_start = dev->m_region->lpar_addr;
|
|
|
|
hcd->rsrc_len = dev->m_region->len;
|
2007-06-06 11:04:35 +08:00
|
|
|
|
|
|
|
if (!request_mem_region(hcd->rsrc_start, hcd->rsrc_len, hcd_name))
|
|
|
|
dev_dbg(&dev->core, "%s:%d: request_mem_region failed\n",
|
|
|
|
__func__, __LINE__);
|
|
|
|
|
2007-01-16 12:11:47 +08:00
|
|
|
hcd->regs = ioremap(dev->m_region->lpar_addr, dev->m_region->len);
|
|
|
|
|
|
|
|
if (!hcd->regs) {
|
|
|
|
dev_dbg(&dev->core, "%s:%d: ioremap failed\n", __func__,
|
|
|
|
__LINE__);
|
|
|
|
result = -EPERM;
|
|
|
|
goto fail_ioremap;
|
|
|
|
}
|
|
|
|
|
|
|
|
dev_dbg(&dev->core, "%s:%d: hcd->rsrc_start %lxh\n", __func__, __LINE__,
|
|
|
|
(unsigned long)hcd->rsrc_start);
|
|
|
|
dev_dbg(&dev->core, "%s:%d: hcd->rsrc_len %lxh\n", __func__, __LINE__,
|
|
|
|
(unsigned long)hcd->rsrc_len);
|
|
|
|
dev_dbg(&dev->core, "%s:%d: hcd->regs %lxh\n", __func__, __LINE__,
|
|
|
|
(unsigned long)hcd->regs);
|
|
|
|
dev_dbg(&dev->core, "%s:%d: virq %lu\n", __func__, __LINE__,
|
|
|
|
(unsigned long)virq);
|
|
|
|
|
2009-06-10 12:38:54 +08:00
|
|
|
ps3_system_bus_set_drvdata(dev, hcd);
|
2007-01-16 12:11:47 +08:00
|
|
|
|
2011-09-07 16:10:52 +08:00
|
|
|
result = usb_add_hcd(hcd, virq, 0);
|
2007-01-16 12:11:47 +08:00
|
|
|
|
|
|
|
if (result) {
|
|
|
|
dev_dbg(&dev->core, "%s:%d: usb_add_hcd failed (%d)\n",
|
|
|
|
__func__, __LINE__, result);
|
|
|
|
goto fail_add_hcd;
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
|
|
|
fail_add_hcd:
|
|
|
|
iounmap(hcd->regs);
|
|
|
|
fail_ioremap:
|
2007-06-06 11:04:35 +08:00
|
|
|
release_mem_region(hcd->rsrc_start, hcd->rsrc_len);
|
2007-01-16 12:11:47 +08:00
|
|
|
usb_put_hcd(hcd);
|
|
|
|
fail_create_hcd:
|
2007-05-01 05:01:01 +08:00
|
|
|
ps3_io_irq_destroy(virq);
|
2007-01-16 12:11:47 +08:00
|
|
|
fail_irq:
|
|
|
|
ps3_free_mmio_region(dev->m_region);
|
2007-06-06 11:04:35 +08:00
|
|
|
fail_mmio_region:
|
|
|
|
ps3_dma_region_free(dev->d_region);
|
|
|
|
fail_dma_region:
|
|
|
|
ps3_close_hv_device(dev);
|
|
|
|
fail_open:
|
2007-01-16 12:11:47 +08:00
|
|
|
fail_start:
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2007-06-06 11:04:35 +08:00
|
|
|
static int ps3_ehci_remove(struct ps3_system_bus_device *dev)
|
2007-01-16 12:11:47 +08:00
|
|
|
{
|
2007-06-06 11:04:35 +08:00
|
|
|
unsigned int tmp;
|
2009-06-10 12:38:54 +08:00
|
|
|
struct usb_hcd *hcd = ps3_system_bus_get_drvdata(dev);
|
2007-01-16 12:11:47 +08:00
|
|
|
|
2007-06-06 11:04:35 +08:00
|
|
|
BUG_ON(!hcd);
|
|
|
|
|
|
|
|
dev_dbg(&dev->core, "%s:%d: regs %p\n", __func__, __LINE__, hcd->regs);
|
|
|
|
dev_dbg(&dev->core, "%s:%d: irq %u\n", __func__, __LINE__, hcd->irq);
|
|
|
|
|
|
|
|
tmp = hcd->irq;
|
|
|
|
|
|
|
|
usb_remove_hcd(hcd);
|
|
|
|
|
2009-06-10 12:38:54 +08:00
|
|
|
ps3_system_bus_set_drvdata(dev, NULL);
|
2007-01-16 12:11:47 +08:00
|
|
|
|
2007-06-06 11:04:35 +08:00
|
|
|
BUG_ON(!hcd->regs);
|
|
|
|
iounmap(hcd->regs);
|
|
|
|
|
|
|
|
release_mem_region(hcd->rsrc_start, hcd->rsrc_len);
|
|
|
|
usb_put_hcd(hcd);
|
|
|
|
|
|
|
|
ps3_io_irq_destroy(tmp);
|
|
|
|
ps3_free_mmio_region(dev->m_region);
|
|
|
|
|
|
|
|
ps3_dma_region_free(dev->d_region);
|
|
|
|
ps3_close_hv_device(dev);
|
|
|
|
|
2007-01-16 12:11:47 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2009-06-10 12:38:59 +08:00
|
|
|
static int __init ps3_ehci_driver_register(struct ps3_system_bus_driver *drv)
|
2007-06-06 11:04:35 +08:00
|
|
|
{
|
|
|
|
return firmware_has_feature(FW_FEATURE_PS3_LV1)
|
|
|
|
? ps3_system_bus_driver_register(drv)
|
|
|
|
: 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void ps3_ehci_driver_unregister(struct ps3_system_bus_driver *drv)
|
|
|
|
{
|
|
|
|
if (firmware_has_feature(FW_FEATURE_PS3_LV1))
|
|
|
|
ps3_system_bus_driver_unregister(drv);
|
|
|
|
}
|
|
|
|
|
|
|
|
MODULE_ALIAS(PS3_MODULE_ALIAS_EHCI);
|
2007-01-16 12:11:47 +08:00
|
|
|
|
2007-06-06 11:04:35 +08:00
|
|
|
static struct ps3_system_bus_driver ps3_ehci_driver = {
|
|
|
|
.core.name = "ps3-ehci-driver",
|
|
|
|
.core.owner = THIS_MODULE,
|
2007-01-16 12:11:47 +08:00
|
|
|
.match_id = PS3_MATCH_ID_EHCI,
|
2007-06-06 11:04:35 +08:00
|
|
|
.probe = ps3_ehci_probe,
|
|
|
|
.remove = ps3_ehci_remove,
|
|
|
|
.shutdown = ps3_ehci_remove,
|
2007-01-16 12:11:47 +08:00
|
|
|
};
|