mirror of
				git://git.openwrt.org/openwrt/openwrt.git
				synced 2025-10-31 05:54:26 -04:00 
			
		
		
		
	it was changed in DTS by commit oxnas-target: fix typo in OX820 device-tree Signed-off-by: Daniel Golle <daniel@makrotopia.org> SVN-Revision: 43474
		
			
				
	
	
		
			317 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			317 lines
		
	
	
		
			7.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * drivers/usb/host/ehci-oxnas.c
 | |
|  *
 | |
|  * Tzachi Perelstein <tzachi@marvell.com>
 | |
|  *
 | |
|  * This file is licensed under  the terms of the GNU General Public
 | |
|  * License version 2. This program is licensed "as is" without any
 | |
|  * warranty of any kind, whether express or implied.
 | |
|  */
 | |
| 
 | |
| #include <linux/kernel.h>
 | |
| #include <linux/module.h>
 | |
| #include <linux/platform_device.h>
 | |
| #include <linux/of.h>
 | |
| #include <linux/of_address.h>
 | |
| #include <linux/of_irq.h>
 | |
| #include <linux/usb.h>
 | |
| #include <linux/usb/hcd.h>
 | |
| #include <linux/dma-mapping.h>
 | |
| #include <linux/clk.h>
 | |
| #include <linux/reset.h>
 | |
| #include <mach/hardware.h>
 | |
| #include <mach/utils.h>
 | |
| 
 | |
| #include "ehci.h"
 | |
| 
 | |
| struct oxnas_hcd {
 | |
| 	struct clk *clk;
 | |
| 	struct clk *refsrc;
 | |
| 	struct clk *phyref;
 | |
| 	int use_pllb;
 | |
| 	int use_phya;
 | |
| 	struct reset_control *rst_host;
 | |
| 	struct reset_control *rst_phya;
 | |
| 	struct reset_control *rst_phyb;
 | |
| };
 | |
| 
 | |
| #define DRIVER_DESC "Oxnas On-Chip EHCI Host Controller"
 | |
| 
 | |
| static struct hc_driver __read_mostly oxnas_hc_driver;
 | |
| 
 | |
| static void start_oxnas_usb_ehci(struct oxnas_hcd *oxnas)
 | |
| {
 | |
| 	u32 reg;
 | |
| 
 | |
| 	if (oxnas->use_pllb) {
 | |
| 		/* enable pllb */
 | |
| 		clk_prepare_enable(oxnas->refsrc);
 | |
| 		/* enable ref600 */
 | |
| 		clk_prepare_enable(oxnas->phyref);
 | |
| 		/* 600MHz pllb divider for 12MHz */
 | |
| 		writel(PLLB_DIV_INT(50) | PLLB_DIV_FRAC(0),
 | |
| 			SEC_CTRL_PLLB_DIV_CTRL);
 | |
| 
 | |
| 	} else {
 | |
| 		/* ref 300 divider for 12MHz */
 | |
| 		writel(REF300_DIV_INT(25) | REF300_DIV_FRAC(0),
 | |
| 			SYS_CTRL_REF300_DIV);
 | |
| 	}
 | |
| 
 | |
| 	/* Ensure the USB block is properly reset */
 | |
| 	reset_control_reset(oxnas->rst_host);
 | |
| 	reset_control_reset(oxnas->rst_phya);
 | |
| 	reset_control_reset(oxnas->rst_phyb);
 | |
| 
 | |
| 	/* Force the high speed clock to be generated all the time, via serial
 | |
| 	 programming of the USB HS PHY */
 | |
| 	writel((2UL << USBHSPHY_TEST_ADD) |
 | |
| 		   (0xe0UL << USBHSPHY_TEST_DIN), SYS_CTRL_USBHSPHY_CTRL);
 | |
| 
 | |
| 	writel((1UL << USBHSPHY_TEST_CLK) |
 | |
| 		   (2UL << USBHSPHY_TEST_ADD) |
 | |
| 		   (0xe0UL << USBHSPHY_TEST_DIN), SYS_CTRL_USBHSPHY_CTRL);
 | |
| 
 | |
| 	writel((0xfUL << USBHSPHY_TEST_ADD) |
 | |
| 		   (0xaaUL << USBHSPHY_TEST_DIN), SYS_CTRL_USBHSPHY_CTRL);
 | |
| 
 | |
| 	writel((1UL << USBHSPHY_TEST_CLK) |
 | |
| 		   (0xfUL << USBHSPHY_TEST_ADD) |
 | |
| 		   (0xaaUL << USBHSPHY_TEST_DIN), SYS_CTRL_USBHSPHY_CTRL);
 | |
| 
 | |
| 	if (oxnas->use_pllb) /* use pllb clock */
 | |
| 		writel(USB_CLK_INTERNAL | USB_INT_CLK_PLLB, SYS_CTRL_USB_CTRL);
 | |
| 	else /* use ref300 derived clock */
 | |
| 		writel(USB_CLK_INTERNAL | USB_INT_CLK_REF300,
 | |
| 			SYS_CTRL_USB_CTRL);
 | |
| 
 | |
| 	if (oxnas->use_phya) {
 | |
| 		/* Configure USB PHYA as a host */
 | |
| 		reg = readl(SYS_CTRL_USB_CTRL);
 | |
| 		reg &= ~USBAMUX_DEVICE;
 | |
| 		writel(reg, SYS_CTRL_USB_CTRL);
 | |
| 	}
 | |
| 
 | |
| 	/* Enable the clock to the USB block */
 | |
| 	clk_prepare_enable(oxnas->clk);
 | |
| }
 | |
| 
 | |
| static void stop_oxnas_usb_ehci(struct oxnas_hcd *oxnas)
 | |
| {
 | |
| 	reset_control_assert(oxnas->rst_host);
 | |
| 	reset_control_assert(oxnas->rst_phya);
 | |
| 	reset_control_assert(oxnas->rst_phyb);
 | |
| 
 | |
| 	if (oxnas->use_pllb) {
 | |
| 		clk_disable_unprepare(oxnas->phyref);
 | |
| 		clk_disable_unprepare(oxnas->refsrc);
 | |
| 	}
 | |
| 	clk_disable_unprepare(oxnas->clk);
 | |
| }
 | |
| 
 | |
| static int ehci_oxnas_reset(struct usb_hcd *hcd)
 | |
| {
 | |
| 	#define  txttfill_tuning	reserved2[0]
 | |
| 
 | |
| 	struct ehci_hcd	*ehci;
 | |
| 	u32 tmp;
 | |
| 	int retval = ehci_setup(hcd);
 | |
| 	if (retval)
 | |
| 		return retval;
 | |
| 
 | |
| 	ehci = hcd_to_ehci(hcd);
 | |
| 	tmp = ehci_readl(ehci, &ehci->regs->txfill_tuning);
 | |
| 	tmp &= ~0x00ff0000;
 | |
| 	tmp |= 0x003f0000; /* set burst pre load count to 0x40 (63 * 4 bytes)  */
 | |
| 	tmp |= 0x16; /* set sheduler overhead to 22 * 1.267us (HS) or 22 * 6.33us (FS/LS)*/
 | |
| 	ehci_writel(ehci, tmp,  &ehci->regs->txfill_tuning);
 | |
| 
 | |
| 	tmp = ehci_readl(ehci, &ehci->regs->txttfill_tuning);
 | |
| 	tmp |= 0x2; /* set sheduler overhead to 2 * 6.333us */
 | |
| 	ehci_writel(ehci, tmp,  &ehci->regs->txttfill_tuning);
 | |
| 
 | |
| 	return retval;
 | |
| }
 | |
| 
 | |
| static int ehci_oxnas_drv_probe(struct platform_device *ofdev)
 | |
| {
 | |
| 	struct device_node *np = ofdev->dev.of_node;
 | |
| 	struct usb_hcd *hcd;
 | |
| 	struct ehci_hcd *ehci;
 | |
| 	struct resource res;
 | |
| 	struct oxnas_hcd *oxnas;
 | |
| 	int irq, err;
 | |
| 	struct reset_control *rstc;
 | |
| 
 | |
| 	if (usb_disabled())
 | |
| 		return -ENODEV;
 | |
| 
 | |
| 	if (!ofdev->dev.dma_mask)
 | |
| 		ofdev->dev.dma_mask = &ofdev->dev.coherent_dma_mask;
 | |
| 	if (!ofdev->dev.coherent_dma_mask)
 | |
| 		ofdev->dev.coherent_dma_mask = DMA_BIT_MASK(32);
 | |
| 
 | |
| 	hcd = usb_create_hcd(&oxnas_hc_driver,	&ofdev->dev,
 | |
| 					dev_name(&ofdev->dev));
 | |
| 	if (!hcd)
 | |
| 		return -ENOMEM;
 | |
| 
 | |
| 	err = of_address_to_resource(np, 0, &res);
 | |
| 	if (err)
 | |
| 		goto err_res;
 | |
| 
 | |
| 	hcd->rsrc_start = res.start;
 | |
| 	hcd->rsrc_len = resource_size(&res);
 | |
| 
 | |
| 	hcd->regs = devm_ioremap_resource(&ofdev->dev, &res);
 | |
| 	if (IS_ERR(hcd->regs)) {
 | |
| 		dev_err(&ofdev->dev, "devm_ioremap_resource failed\n");
 | |
| 		err = PTR_ERR(hcd->regs);
 | |
| 		goto err_ioremap;
 | |
| 	}
 | |
| 
 | |
| 	oxnas = (struct oxnas_hcd *)hcd_to_ehci(hcd)->priv;
 | |
| 
 | |
| 	oxnas->use_pllb = of_property_read_bool(np, "plxtech,ehci_use_pllb");
 | |
| 	oxnas->use_phya = of_property_read_bool(np, "plxtech,ehci_use_phya");
 | |
| 
 | |
| 	oxnas->clk = of_clk_get_by_name(np, "usb");
 | |
| 	if (IS_ERR(oxnas->clk)) {
 | |
| 		err = PTR_ERR(oxnas->clk);
 | |
| 		goto err_clk;
 | |
| 	}
 | |
| 
 | |
| 	if (oxnas->use_pllb) {
 | |
| 		oxnas->refsrc = of_clk_get_by_name(np, "refsrc");
 | |
| 		if (IS_ERR(oxnas->refsrc)) {
 | |
| 			err = PTR_ERR(oxnas->refsrc);
 | |
| 			goto err_refsrc;
 | |
| 		}
 | |
| 		oxnas->phyref = of_clk_get_by_name(np, "phyref");
 | |
| 		if (IS_ERR(oxnas->refsrc)) {
 | |
| 			err = PTR_ERR(oxnas->refsrc);
 | |
| 			goto err_phyref;
 | |
| 		}
 | |
| 
 | |
| 	} else {
 | |
| 		oxnas->refsrc = NULL;
 | |
| 		oxnas->phyref = NULL;
 | |
| 	}
 | |
| 
 | |
| 	rstc = devm_reset_control_get(&ofdev->dev, "host");
 | |
| 	if (IS_ERR(rstc)) {
 | |
| 		err = PTR_ERR(rstc);
 | |
| 		goto err_rst;
 | |
| 	}
 | |
| 	oxnas->rst_host = rstc;
 | |
| 
 | |
| 	rstc = devm_reset_control_get(&ofdev->dev, "phya");
 | |
| 	if (IS_ERR(rstc)) {
 | |
| 		err = PTR_ERR(rstc);
 | |
| 		goto err_rst;
 | |
| 	}
 | |
| 	oxnas->rst_phya = rstc;
 | |
| 
 | |
| 	rstc = devm_reset_control_get(&ofdev->dev, "phyb");
 | |
| 	if (IS_ERR(rstc)) {
 | |
| 		err = PTR_ERR(rstc);
 | |
| 		goto err_rst;
 | |
| 	}
 | |
| 	oxnas->rst_phyb = rstc;
 | |
| 
 | |
| 	irq = irq_of_parse_and_map(np, 0);
 | |
| 	if (!irq) {
 | |
| 		dev_err(&ofdev->dev, "irq_of_parse_and_map failed\n");
 | |
| 		err = -EBUSY;
 | |
| 		goto err_irq;
 | |
| 	}
 | |
| 
 | |
| 	hcd->has_tt = 1;
 | |
| 	ehci = hcd_to_ehci(hcd);
 | |
| 	ehci->caps = hcd->regs;
 | |
| 
 | |
| 	start_oxnas_usb_ehci(oxnas);
 | |
| 
 | |
| 	err = usb_add_hcd(hcd, irq, IRQF_SHARED | IRQF_DISABLED);
 | |
| 	if (err)
 | |
| 		goto err_hcd;
 | |
| 
 | |
| 	return 0;
 | |
| 
 | |
| err_hcd:
 | |
| 	stop_oxnas_usb_ehci(oxnas);
 | |
| err_irq:
 | |
| err_rst:
 | |
| 	if (oxnas->phyref)
 | |
| 		clk_put(oxnas->phyref);
 | |
| err_phyref:
 | |
| 	if (oxnas->refsrc)
 | |
| 		clk_put(oxnas->refsrc);
 | |
| err_refsrc:
 | |
| 	clk_put(oxnas->clk);
 | |
| err_clk:
 | |
| err_ioremap:
 | |
| err_res:
 | |
| 	usb_put_hcd(hcd);
 | |
| 
 | |
| 	return err;
 | |
| }
 | |
| 
 | |
| static int ehci_oxnas_drv_remove(struct platform_device *pdev)
 | |
| {
 | |
| 	struct usb_hcd *hcd = platform_get_drvdata(pdev);
 | |
| 	struct oxnas_hcd *oxnas = (struct oxnas_hcd *)hcd_to_ehci(hcd)->priv;
 | |
| 
 | |
| 	usb_remove_hcd(hcd);
 | |
| 	if (oxnas->use_pllb) {
 | |
| 		clk_disable_unprepare(oxnas->phyref);
 | |
| 		clk_put(oxnas->phyref);
 | |
| 		clk_disable_unprepare(oxnas->refsrc);
 | |
| 		clk_put(oxnas->refsrc);
 | |
| 	}
 | |
| 	clk_disable_unprepare(oxnas->clk);
 | |
| 	usb_put_hcd(hcd);
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static const struct of_device_id oxnas_ehci_dt_ids[] = {
 | |
| 	{ .compatible = "plxtech,nas782x-ehci" },
 | |
| 	{ /* sentinel */ }
 | |
| };
 | |
| 
 | |
| MODULE_DEVICE_TABLE(of, oxnas_ehci_dt_ids);
 | |
| 
 | |
| static struct platform_driver ehci_oxnas_driver = {
 | |
| 	.probe		= ehci_oxnas_drv_probe,
 | |
| 	.remove		= ehci_oxnas_drv_remove,
 | |
| 	.shutdown	= usb_hcd_platform_shutdown,
 | |
| 	.driver.name	= "oxnas-ehci",
 | |
| 	.driver.of_match_table	= oxnas_ehci_dt_ids,
 | |
| };
 | |
| 
 | |
| static const struct ehci_driver_overrides oxnas_overrides __initconst = {
 | |
| 	.reset = ehci_oxnas_reset,
 | |
| 	.extra_priv_size = sizeof(struct oxnas_hcd),
 | |
| };
 | |
| 
 | |
| static int __init ehci_oxnas_init(void)
 | |
| {
 | |
| 	if (usb_disabled())
 | |
| 		return -ENODEV;
 | |
| 
 | |
| 	ehci_init_driver(&oxnas_hc_driver, &oxnas_overrides);
 | |
| 	return platform_driver_register(&ehci_oxnas_driver);
 | |
| }
 | |
| module_init(ehci_oxnas_init);
 | |
| 
 | |
| static void __exit ehci_oxnas_cleanup(void)
 | |
| {
 | |
| 	platform_driver_unregister(&ehci_oxnas_driver);
 | |
| }
 | |
| module_exit(ehci_oxnas_cleanup);
 | |
| 
 | |
| MODULE_DESCRIPTION(DRIVER_DESC);
 | |
| MODULE_ALIAS("platform:oxnas-ehci");
 | |
| MODULE_LICENSE("GPL");
 |