xhci-zynqmp.c 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Copyright 2015 Xilinx, Inc.
  4. *
  5. * Zynq USB HOST xHCI Controller
  6. *
  7. * Author: Siva Durga Prasad Paladugu<sivadur@xilinx.com>
  8. *
  9. * This file was reused from Freescale USB xHCI
  10. */
  11. #include <common.h>
  12. #include <dm.h>
  13. #include <usb.h>
  14. #include <linux/errno.h>
  15. #include <asm/arch-zynqmp/hardware.h>
  16. #include <linux/compat.h>
  17. #include <linux/usb/dwc3.h>
  18. #include "xhci.h"
  19. /* Declare global data pointer */
  20. /* Default to the ZYNQMP XHCI defines */
  21. #define USB3_PWRCTL_CLK_CMD_MASK 0x3FE000
  22. #define USB3_PWRCTL_CLK_FREQ_MASK 0xFFC
  23. #define USB3_PHY_PARTIAL_RX_POWERON BIT(6)
  24. #define USB3_PHY_RX_POWERON BIT(14)
  25. #define USB3_PHY_TX_POWERON BIT(15)
  26. #define USB3_PHY_TX_RX_POWERON (USB3_PHY_RX_POWERON | USB3_PHY_TX_POWERON)
  27. #define USB3_PWRCTL_CLK_CMD_SHIFT 14
  28. #define USB3_PWRCTL_CLK_FREQ_SHIFT 22
  29. /* USBOTGSS_WRAPPER definitions */
  30. #define USBOTGSS_WRAPRESET BIT(17)
  31. #define USBOTGSS_DMADISABLE BIT(16)
  32. #define USBOTGSS_STANDBYMODE_NO_STANDBY BIT(4)
  33. #define USBOTGSS_STANDBYMODE_SMRT BIT(5)
  34. #define USBOTGSS_STANDBYMODE_SMRT_WKUP (0x3 << 4)
  35. #define USBOTGSS_IDLEMODE_NOIDLE BIT(2)
  36. #define USBOTGSS_IDLEMODE_SMRT BIT(3)
  37. #define USBOTGSS_IDLEMODE_SMRT_WKUP (0x3 << 2)
  38. /* USBOTGSS_IRQENABLE_SET_0 bit */
  39. #define USBOTGSS_COREIRQ_EN BIT(1)
  40. /* USBOTGSS_IRQENABLE_SET_1 bits */
  41. #define USBOTGSS_IRQ_SET_1_IDPULLUP_FALL_EN BIT(1)
  42. #define USBOTGSS_IRQ_SET_1_DISCHRGVBUS_FALL_EN BIT(3)
  43. #define USBOTGSS_IRQ_SET_1_CHRGVBUS_FALL_EN BIT(4)
  44. #define USBOTGSS_IRQ_SET_1_DRVVBUS_FALL_EN BIT(5)
  45. #define USBOTGSS_IRQ_SET_1_IDPULLUP_RISE_EN BIT(8)
  46. #define USBOTGSS_IRQ_SET_1_DISCHRGVBUS_RISE_EN BIT(11)
  47. #define USBOTGSS_IRQ_SET_1_CHRGVBUS_RISE_EN BIT(12)
  48. #define USBOTGSS_IRQ_SET_1_DRVVBUS_RISE_EN BIT(13)
  49. #define USBOTGSS_IRQ_SET_1_OEVT_EN BIT(16)
  50. #define USBOTGSS_IRQ_SET_1_DMADISABLECLR_EN BIT(17)
  51. struct zynqmp_xhci {
  52. #ifdef CONFIG_DM_USB
  53. struct usb_platdata usb_plat;
  54. #endif
  55. struct xhci_ctrl ctrl;
  56. struct xhci_hccr *hcd;
  57. struct dwc3 *dwc3_reg;
  58. };
  59. #ifdef CONFIG_DM_USB
  60. struct zynqmp_xhci_platdata {
  61. fdt_addr_t hcd_base;
  62. };
  63. #else
  64. static struct zynqmp_xhci zynqmp_xhci;
  65. unsigned long ctr_addr[] = CONFIG_ZYNQMP_XHCI_LIST;
  66. #endif
  67. static int zynqmp_xhci_core_init(struct zynqmp_xhci *zynqmp_xhci)
  68. {
  69. int ret = 0;
  70. ret = dwc3_core_init(zynqmp_xhci->dwc3_reg);
  71. if (ret) {
  72. debug("%s:failed to initialize core\n", __func__);
  73. return ret;
  74. }
  75. /* We are hard-coding DWC3 core to Host Mode */
  76. dwc3_set_mode(zynqmp_xhci->dwc3_reg, DWC3_GCTL_PRTCAP_HOST);
  77. return ret;
  78. }
  79. #ifndef CONFIG_DM_USB
  80. int xhci_hcd_init(int index, struct xhci_hccr **hccr, struct xhci_hcor **hcor)
  81. {
  82. struct zynqmp_xhci *ctx = &zynqmp_xhci;
  83. int ret = 0;
  84. uint32_t hclen;
  85. if (index < 0 || index >= ARRAY_SIZE(ctr_addr))
  86. return -EINVAL;
  87. ctx->hcd = (struct xhci_hccr *)ctr_addr[index];
  88. ctx->dwc3_reg = (struct dwc3 *)((void *)ctx->hcd + DWC3_REG_OFFSET);
  89. ret = board_usb_init(index, USB_INIT_HOST);
  90. if (ret != 0) {
  91. puts("Failed to initialize board for USB\n");
  92. return ret;
  93. }
  94. ret = zynqmp_xhci_core_init(ctx);
  95. if (ret < 0) {
  96. puts("Failed to initialize xhci\n");
  97. return ret;
  98. }
  99. *hccr = (struct xhci_hccr *)ctx->hcd;
  100. hclen = HC_LENGTH(xhci_readl(&(*hccr)->cr_capbase));
  101. *hcor = (struct xhci_hcor *)((uintptr_t) *hccr + hclen);
  102. debug("zynqmp-xhci: init hccr %p and hcor %p hc_length %d\n",
  103. *hccr, *hcor, hclen);
  104. return ret;
  105. }
  106. #endif
  107. void xhci_hcd_stop(int index)
  108. {
  109. /*
  110. * Currently zynqmp socs do not support PHY shutdown from
  111. * sw. But this support may be added in future socs.
  112. */
  113. return;
  114. }
  115. #ifdef CONFIG_DM_USB
  116. static int xhci_usb_probe(struct udevice *dev)
  117. {
  118. struct zynqmp_xhci_platdata *plat = dev_get_platdata(dev);
  119. struct zynqmp_xhci *ctx = dev_get_priv(dev);
  120. struct xhci_hcor *hcor;
  121. int ret;
  122. ctx->hcd = (struct xhci_hccr *)plat->hcd_base;
  123. ctx->dwc3_reg = (struct dwc3 *)((char *)(ctx->hcd) + DWC3_REG_OFFSET);
  124. ret = zynqmp_xhci_core_init(ctx);
  125. if (ret) {
  126. puts("XHCI: failed to initialize controller\n");
  127. return -EINVAL;
  128. }
  129. hcor = (struct xhci_hcor *)((ulong)ctx->hcd +
  130. HC_LENGTH(xhci_readl(&ctx->hcd->cr_capbase)));
  131. return xhci_register(dev, ctx->hcd, hcor);
  132. }
  133. static int xhci_usb_remove(struct udevice *dev)
  134. {
  135. return xhci_deregister(dev);
  136. }
  137. static int xhci_usb_ofdata_to_platdata(struct udevice *dev)
  138. {
  139. struct zynqmp_xhci_platdata *plat = dev_get_platdata(dev);
  140. const void *blob = gd->fdt_blob;
  141. /* Get the base address for XHCI controller from the device node */
  142. plat->hcd_base = fdtdec_get_addr(blob, dev_of_offset(dev), "reg");
  143. if (plat->hcd_base == FDT_ADDR_T_NONE) {
  144. debug("Can't get the XHCI register base address\n");
  145. return -ENXIO;
  146. }
  147. return 0;
  148. }
  149. U_BOOT_DRIVER(dwc3_generic_host) = {
  150. .name = "dwc3-generic-host",
  151. .id = UCLASS_USB,
  152. .ofdata_to_platdata = xhci_usb_ofdata_to_platdata,
  153. .probe = xhci_usb_probe,
  154. .remove = xhci_usb_remove,
  155. .ops = &xhci_usb_ops,
  156. .platdata_auto_alloc_size = sizeof(struct zynqmp_xhci_platdata),
  157. .priv_auto_alloc_size = sizeof(struct zynqmp_xhci),
  158. .flags = DM_FLAG_ALLOC_PRIV_DMA,
  159. };
  160. #endif