]> git.ipfire.org Git - thirdparty/u-boot.git/commitdiff
zynq: ehci: Added USB host driver support
authorSiva Durga Prasad Paladugu <siva.durga.paladugu@xilinx.com>
Thu, 20 Feb 2014 04:58:26 +0000 (10:28 +0530)
committerMichal Simek <michal.simek@xilinx.com>
Thu, 20 Feb 2014 07:37:30 +0000 (08:37 +0100)
Added USB host driver for zynq.

Signed-off-by: Siva Durga Prasad Paladugu <sivadur@xilinx.com>
Signed-off-by: Michal Simek <michal.simek@xilinx.com>
arch/arm/include/asm/arch-zynq/hardware.h
drivers/usb/host/Makefile
drivers/usb/host/ehci-zynq.c [new file with mode: 0644]

index f12617930ad4fed3144a49cdf211857393e0ec7b..2dc9ce59551040537d70df96a65050db5314cc03 100644 (file)
@@ -26,6 +26,8 @@
 #define ZYNQ_NAND_BASEADDR             0xE1000000
 #define ZYNQ_DDRC_BASEADDR             0xF8006000
 #define ZYNQ_EFUSE_BASEADDR            0xF800D000
+#define ZYNQ_USB_BASEADDR0             0xE0002000
+#define ZYNQ_USB_BASEADDR1             0xE0003000
 
 /* Bootmode setting values */
 #define ZYNQ_BM_MASK           0x0F
index 14170282dbb11b4be360b12195fa857ae0844e8a..6629fef66adb9361a3524501578bc942110900c4 100644 (file)
@@ -37,6 +37,7 @@ obj-$(CONFIG_USB_EHCI_PCI) += ehci-pci.o
 obj-$(CONFIG_USB_EHCI_SPEAR) += ehci-spear.o
 obj-$(CONFIG_USB_EHCI_TEGRA) += ehci-tegra.o
 obj-$(CONFIG_USB_EHCI_VCT) += ehci-vct.o
+obj-$(CONFIG_USB_EHCI_ZYNQ) += ehci-zynq.o
 
 # xhci
 obj-$(CONFIG_USB_XHCI) += xhci.o xhci-mem.o xhci-ring.o
diff --git a/drivers/usb/host/ehci-zynq.c b/drivers/usb/host/ehci-zynq.c
new file mode 100644 (file)
index 0000000..7770d05
--- /dev/null
@@ -0,0 +1,104 @@
+/*
+ * (C) Copyright 2014, Xilinx, Inc
+ *
+ * USB Low level initialization(Specific to zynq)
+ *
+ * SPDX-License-Identifier:    GPL-2.0+
+ */
+
+#include <common.h>
+#include <asm/arch/hardware.h>
+#include <asm/arch/sys_proto.h>
+#include <asm/io.h>
+#include <usb.h>
+#include <usb/ehci-fsl.h>
+#include <usb/ulpi.h>
+
+#include "ehci.h"
+
+#define ZYNQ_USB_USBCMD_RST                    0x0000002
+#define ZYNQ_USB_USBCMD_STOP                   0x0000000
+#define ZYNQ_USB_NUM_MIO                       12
+
+/*
+ * Create the appropriate control structures to manage
+ * a new EHCI host controller.
+ */
+int ehci_hcd_init(int index,  enum usb_init_type init, struct ehci_hccr **hccr,
+                 struct ehci_hcor **hcor)
+{
+       struct usb_ehci *ehci;
+       struct ulpi_viewport ulpi_vp;
+       int ret, mio_usb;
+       /* Used for writing the ULPI data address */
+       struct ulpi_regs *ulpi = (struct ulpi_regs *)0;
+
+       if (!index) {
+               mio_usb = zynq_slcr_get_mio_pin_status("usb0");
+               if (mio_usb != ZYNQ_USB_NUM_MIO) {
+                       printf("usb0 wrong num MIO: %d, Index %d\n", mio_usb,
+                              index);
+                       return -1;
+               }
+               ehci = (struct usb_ehci *)ZYNQ_USB_BASEADDR0;
+       } else {
+               mio_usb = zynq_slcr_get_mio_pin_status("usb1");
+               if (mio_usb != ZYNQ_USB_NUM_MIO) {
+                       printf("usb1 wrong num MIO: %d, Index %d\n", mio_usb,
+                              index);
+                       return -1;
+               }
+               ehci = (struct usb_ehci *)ZYNQ_USB_BASEADDR1;
+       }
+
+       *hccr = (struct ehci_hccr *)((uint32_t)&ehci->caplength);
+       *hcor = (struct ehci_hcor *)((uint32_t) *hccr +
+                       HC_LENGTH(ehci_readl(&(*hccr)->cr_capbase)));
+
+       ulpi_vp.viewport_addr = (u32)&ehci->ulpi_viewpoint;
+       ulpi_vp.port_num = 0;
+
+       ret = ulpi_init(&ulpi_vp);
+       if (ret) {
+               puts("zynq ULPI viewport init failed\n");
+               return -1;
+       }
+
+       /* ULPI set flags */
+       ulpi_write(&ulpi_vp, &ulpi->otg_ctrl,
+                  ULPI_OTG_DP_PULLDOWN | ULPI_OTG_DM_PULLDOWN |
+                  ULPI_OTG_EXTVBUSIND);
+       ulpi_write(&ulpi_vp, &ulpi->function_ctrl,
+                  ULPI_FC_FULL_SPEED | ULPI_FC_OPMODE_NORMAL |
+                  ULPI_FC_SUSPENDM);
+       ulpi_write(&ulpi_vp, &ulpi->iface_ctrl, 0);
+
+       /* Set VBus */
+       ulpi_write(&ulpi_vp, &ulpi->otg_ctrl_set,
+                  ULPI_OTG_DRVVBUS | ULPI_OTG_DRVVBUS_EXT);
+
+       return 0;
+}
+
+/*
+ * Destroy the appropriate control structures corresponding
+ * the the EHCI host controller.
+ */
+int ehci_hcd_stop(int index)
+{
+       struct usb_ehci *ehci;
+
+       if (!index)
+               ehci = (struct usb_ehci *)ZYNQ_USB_BASEADDR0;
+       else
+               ehci = (struct usb_ehci *)ZYNQ_USB_BASEADDR1;
+
+       /* Stop controller */
+       writel(ZYNQ_USB_USBCMD_STOP, &ehci->usbcmd);
+       udelay(1000);
+
+       /* Initiate controller reset */
+       writel(ZYNQ_USB_USBCMD_RST, &ehci->usbcmd);
+
+       return 0;
+}