xhci-zynqmp.c 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146
  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/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. struct usb_platdata usb_plat;
  53. struct xhci_ctrl ctrl;
  54. struct xhci_hccr *hcd;
  55. struct dwc3 *dwc3_reg;
  56. };
  57. struct zynqmp_xhci_platdata {
  58. fdt_addr_t hcd_base;
  59. };
  60. static int zynqmp_xhci_core_init(struct zynqmp_xhci *zynqmp_xhci)
  61. {
  62. int ret = 0;
  63. ret = dwc3_core_init(zynqmp_xhci->dwc3_reg);
  64. if (ret) {
  65. debug("%s:failed to initialize core\n", __func__);
  66. return ret;
  67. }
  68. /* We are hard-coding DWC3 core to Host Mode */
  69. dwc3_set_mode(zynqmp_xhci->dwc3_reg, DWC3_GCTL_PRTCAP_HOST);
  70. return ret;
  71. }
  72. void xhci_hcd_stop(int index)
  73. {
  74. /*
  75. * Currently zynqmp socs do not support PHY shutdown from
  76. * sw. But this support may be added in future socs.
  77. */
  78. return;
  79. }
  80. static int xhci_usb_probe(struct udevice *dev)
  81. {
  82. struct zynqmp_xhci_platdata *plat = dev_get_platdata(dev);
  83. struct zynqmp_xhci *ctx = dev_get_priv(dev);
  84. struct xhci_hcor *hcor;
  85. int ret;
  86. ctx->hcd = (struct xhci_hccr *)plat->hcd_base;
  87. ctx->dwc3_reg = (struct dwc3 *)((char *)(ctx->hcd) + DWC3_REG_OFFSET);
  88. ret = zynqmp_xhci_core_init(ctx);
  89. if (ret) {
  90. puts("XHCI: failed to initialize controller\n");
  91. return -EINVAL;
  92. }
  93. hcor = (struct xhci_hcor *)((ulong)ctx->hcd +
  94. HC_LENGTH(xhci_readl(&ctx->hcd->cr_capbase)));
  95. return xhci_register(dev, ctx->hcd, hcor);
  96. }
  97. static int xhci_usb_remove(struct udevice *dev)
  98. {
  99. return xhci_deregister(dev);
  100. }
  101. static int xhci_usb_ofdata_to_platdata(struct udevice *dev)
  102. {
  103. struct zynqmp_xhci_platdata *plat = dev_get_platdata(dev);
  104. const void *blob = gd->fdt_blob;
  105. /* Get the base address for XHCI controller from the device node */
  106. plat->hcd_base = fdtdec_get_addr(blob, dev_of_offset(dev), "reg");
  107. if (plat->hcd_base == FDT_ADDR_T_NONE) {
  108. debug("Can't get the XHCI register base address\n");
  109. return -ENXIO;
  110. }
  111. return 0;
  112. }
  113. U_BOOT_DRIVER(dwc3_generic_host) = {
  114. .name = "dwc3-generic-host",
  115. .id = UCLASS_USB,
  116. .ofdata_to_platdata = xhci_usb_ofdata_to_platdata,
  117. .probe = xhci_usb_probe,
  118. .remove = xhci_usb_remove,
  119. .ops = &xhci_usb_ops,
  120. .platdata_auto_alloc_size = sizeof(struct zynqmp_xhci_platdata),
  121. .priv_auto_alloc_size = sizeof(struct zynqmp_xhci),
  122. .flags = DM_FLAG_ALLOC_PRIV_DMA,
  123. };