dwc3-omap.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451
  1. /**
  2. * dwc3-omap.c - OMAP Specific Glue layer
  3. *
  4. * Copyright (C) 2015 Texas Instruments Incorporated - http://www.ti.com
  5. *
  6. * Authors: Felipe Balbi <balbi@ti.com>,
  7. * Sebastian Andrzej Siewior <bigeasy@linutronix.de>
  8. *
  9. * Taken from Linux Kernel v3.19-rc1 (drivers/usb/dwc3/dwc3-omap.c) and ported
  10. * to uboot.
  11. *
  12. * commit 7ee2566ff5 : usb: dwc3: dwc3-omap: get rid of ->prepare()/->complete()
  13. *
  14. * SPDX-License-Identifier: GPL-2.0
  15. */
  16. #include <common.h>
  17. #include <malloc.h>
  18. #include <asm/io.h>
  19. #include <dwc3-omap-uboot.h>
  20. #include <linux/usb/dwc3-omap.h>
  21. #include <linux/ioport.h>
  22. #include <linux/usb/otg.h>
  23. #include <linux/compat.h>
  24. #include "linux-compat.h"
  25. /*
  26. * All these registers belong to OMAP's Wrapper around the
  27. * DesignWare USB3 Core.
  28. */
  29. #define USBOTGSS_REVISION 0x0000
  30. #define USBOTGSS_SYSCONFIG 0x0010
  31. #define USBOTGSS_IRQ_EOI 0x0020
  32. #define USBOTGSS_EOI_OFFSET 0x0008
  33. #define USBOTGSS_IRQSTATUS_RAW_0 0x0024
  34. #define USBOTGSS_IRQSTATUS_0 0x0028
  35. #define USBOTGSS_IRQENABLE_SET_0 0x002c
  36. #define USBOTGSS_IRQENABLE_CLR_0 0x0030
  37. #define USBOTGSS_IRQ0_OFFSET 0x0004
  38. #define USBOTGSS_IRQSTATUS_RAW_1 0x0030
  39. #define USBOTGSS_IRQSTATUS_1 0x0034
  40. #define USBOTGSS_IRQENABLE_SET_1 0x0038
  41. #define USBOTGSS_IRQENABLE_CLR_1 0x003c
  42. #define USBOTGSS_IRQSTATUS_RAW_2 0x0040
  43. #define USBOTGSS_IRQSTATUS_2 0x0044
  44. #define USBOTGSS_IRQENABLE_SET_2 0x0048
  45. #define USBOTGSS_IRQENABLE_CLR_2 0x004c
  46. #define USBOTGSS_IRQSTATUS_RAW_3 0x0050
  47. #define USBOTGSS_IRQSTATUS_3 0x0054
  48. #define USBOTGSS_IRQENABLE_SET_3 0x0058
  49. #define USBOTGSS_IRQENABLE_CLR_3 0x005c
  50. #define USBOTGSS_IRQSTATUS_EOI_MISC 0x0030
  51. #define USBOTGSS_IRQSTATUS_RAW_MISC 0x0034
  52. #define USBOTGSS_IRQSTATUS_MISC 0x0038
  53. #define USBOTGSS_IRQENABLE_SET_MISC 0x003c
  54. #define USBOTGSS_IRQENABLE_CLR_MISC 0x0040
  55. #define USBOTGSS_IRQMISC_OFFSET 0x03fc
  56. #define USBOTGSS_UTMI_OTG_CTRL 0x0080
  57. #define USBOTGSS_UTMI_OTG_STATUS 0x0084
  58. #define USBOTGSS_UTMI_OTG_OFFSET 0x0480
  59. #define USBOTGSS_TXFIFO_DEPTH 0x0508
  60. #define USBOTGSS_RXFIFO_DEPTH 0x050c
  61. #define USBOTGSS_MMRAM_OFFSET 0x0100
  62. #define USBOTGSS_FLADJ 0x0104
  63. #define USBOTGSS_DEBUG_CFG 0x0108
  64. #define USBOTGSS_DEBUG_DATA 0x010c
  65. #define USBOTGSS_DEV_EBC_EN 0x0110
  66. #define USBOTGSS_DEBUG_OFFSET 0x0600
  67. /* SYSCONFIG REGISTER */
  68. #define USBOTGSS_SYSCONFIG_DMADISABLE (1 << 16)
  69. /* IRQ_EOI REGISTER */
  70. #define USBOTGSS_IRQ_EOI_LINE_NUMBER (1 << 0)
  71. /* IRQS0 BITS */
  72. #define USBOTGSS_IRQO_COREIRQ_ST (1 << 0)
  73. /* IRQMISC BITS */
  74. #define USBOTGSS_IRQMISC_DMADISABLECLR (1 << 17)
  75. #define USBOTGSS_IRQMISC_OEVT (1 << 16)
  76. #define USBOTGSS_IRQMISC_DRVVBUS_RISE (1 << 13)
  77. #define USBOTGSS_IRQMISC_CHRGVBUS_RISE (1 << 12)
  78. #define USBOTGSS_IRQMISC_DISCHRGVBUS_RISE (1 << 11)
  79. #define USBOTGSS_IRQMISC_IDPULLUP_RISE (1 << 8)
  80. #define USBOTGSS_IRQMISC_DRVVBUS_FALL (1 << 5)
  81. #define USBOTGSS_IRQMISC_CHRGVBUS_FALL (1 << 4)
  82. #define USBOTGSS_IRQMISC_DISCHRGVBUS_FALL (1 << 3)
  83. #define USBOTGSS_IRQMISC_IDPULLUP_FALL (1 << 0)
  84. #define USBOTGSS_INTERRUPTS (USBOTGSS_IRQMISC_OEVT | \
  85. USBOTGSS_IRQMISC_DRVVBUS_RISE | \
  86. USBOTGSS_IRQMISC_CHRGVBUS_RISE | \
  87. USBOTGSS_IRQMISC_DISCHRGVBUS_RISE | \
  88. USBOTGSS_IRQMISC_IDPULLUP_RISE | \
  89. USBOTGSS_IRQMISC_DRVVBUS_FALL | \
  90. USBOTGSS_IRQMISC_CHRGVBUS_FALL | \
  91. USBOTGSS_IRQMISC_DISCHRGVBUS_FALL | \
  92. USBOTGSS_IRQMISC_IDPULLUP_FALL)
  93. /* UTMI_OTG_CTRL REGISTER */
  94. #define USBOTGSS_UTMI_OTG_CTRL_DRVVBUS (1 << 5)
  95. #define USBOTGSS_UTMI_OTG_CTRL_CHRGVBUS (1 << 4)
  96. #define USBOTGSS_UTMI_OTG_CTRL_DISCHRGVBUS (1 << 3)
  97. #define USBOTGSS_UTMI_OTG_CTRL_IDPULLUP (1 << 0)
  98. /* UTMI_OTG_STATUS REGISTER */
  99. #define USBOTGSS_UTMI_OTG_STATUS_SW_MODE (1 << 31)
  100. #define USBOTGSS_UTMI_OTG_STATUS_POWERPRESENT (1 << 9)
  101. #define USBOTGSS_UTMI_OTG_STATUS_TXBITSTUFFENABLE (1 << 8)
  102. #define USBOTGSS_UTMI_OTG_STATUS_IDDIG (1 << 4)
  103. #define USBOTGSS_UTMI_OTG_STATUS_SESSEND (1 << 3)
  104. #define USBOTGSS_UTMI_OTG_STATUS_SESSVALID (1 << 2)
  105. #define USBOTGSS_UTMI_OTG_STATUS_VBUSVALID (1 << 1)
  106. struct dwc3_omap {
  107. struct device *dev;
  108. void __iomem *base;
  109. u32 utmi_otg_status;
  110. u32 utmi_otg_offset;
  111. u32 irqmisc_offset;
  112. u32 irq_eoi_offset;
  113. u32 debug_offset;
  114. u32 irq0_offset;
  115. u32 dma_status:1;
  116. struct list_head list;
  117. u32 index;
  118. };
  119. static LIST_HEAD(dwc3_omap_list);
  120. static inline u32 dwc3_omap_readl(void __iomem *base, u32 offset)
  121. {
  122. return readl(base + offset);
  123. }
  124. static inline void dwc3_omap_writel(void __iomem *base, u32 offset, u32 value)
  125. {
  126. writel(value, base + offset);
  127. }
  128. static u32 dwc3_omap_read_utmi_status(struct dwc3_omap *omap)
  129. {
  130. return dwc3_omap_readl(omap->base, USBOTGSS_UTMI_OTG_STATUS +
  131. omap->utmi_otg_offset);
  132. }
  133. static void dwc3_omap_write_utmi_status(struct dwc3_omap *omap, u32 value)
  134. {
  135. dwc3_omap_writel(omap->base, USBOTGSS_UTMI_OTG_STATUS +
  136. omap->utmi_otg_offset, value);
  137. }
  138. static u32 dwc3_omap_read_irq0_status(struct dwc3_omap *omap)
  139. {
  140. return dwc3_omap_readl(omap->base, USBOTGSS_IRQSTATUS_0 -
  141. omap->irq0_offset);
  142. }
  143. static void dwc3_omap_write_irq0_status(struct dwc3_omap *omap, u32 value)
  144. {
  145. dwc3_omap_writel(omap->base, USBOTGSS_IRQSTATUS_0 -
  146. omap->irq0_offset, value);
  147. }
  148. static u32 dwc3_omap_read_irqmisc_status(struct dwc3_omap *omap)
  149. {
  150. return dwc3_omap_readl(omap->base, USBOTGSS_IRQSTATUS_MISC +
  151. omap->irqmisc_offset);
  152. }
  153. static void dwc3_omap_write_irqmisc_status(struct dwc3_omap *omap, u32 value)
  154. {
  155. dwc3_omap_writel(omap->base, USBOTGSS_IRQSTATUS_MISC +
  156. omap->irqmisc_offset, value);
  157. }
  158. static void dwc3_omap_write_irqmisc_set(struct dwc3_omap *omap, u32 value)
  159. {
  160. dwc3_omap_writel(omap->base, USBOTGSS_IRQENABLE_SET_MISC +
  161. omap->irqmisc_offset, value);
  162. }
  163. static void dwc3_omap_write_irq0_set(struct dwc3_omap *omap, u32 value)
  164. {
  165. dwc3_omap_writel(omap->base, USBOTGSS_IRQENABLE_SET_0 -
  166. omap->irq0_offset, value);
  167. }
  168. static void dwc3_omap_write_irqmisc_clr(struct dwc3_omap *omap, u32 value)
  169. {
  170. dwc3_omap_writel(omap->base, USBOTGSS_IRQENABLE_CLR_MISC +
  171. omap->irqmisc_offset, value);
  172. }
  173. static void dwc3_omap_write_irq0_clr(struct dwc3_omap *omap, u32 value)
  174. {
  175. dwc3_omap_writel(omap->base, USBOTGSS_IRQENABLE_CLR_0 -
  176. omap->irq0_offset, value);
  177. }
  178. static void dwc3_omap_set_mailbox(struct dwc3_omap *omap,
  179. enum omap_dwc3_vbus_id_status status)
  180. {
  181. u32 val;
  182. switch (status) {
  183. case OMAP_DWC3_ID_GROUND:
  184. dev_dbg(omap->dev, "ID GND\n");
  185. val = dwc3_omap_read_utmi_status(omap);
  186. val &= ~(USBOTGSS_UTMI_OTG_STATUS_IDDIG
  187. | USBOTGSS_UTMI_OTG_STATUS_VBUSVALID
  188. | USBOTGSS_UTMI_OTG_STATUS_SESSEND);
  189. val |= USBOTGSS_UTMI_OTG_STATUS_SESSVALID
  190. | USBOTGSS_UTMI_OTG_STATUS_POWERPRESENT;
  191. dwc3_omap_write_utmi_status(omap, val);
  192. break;
  193. case OMAP_DWC3_VBUS_VALID:
  194. dev_dbg(omap->dev, "VBUS Connect\n");
  195. val = dwc3_omap_read_utmi_status(omap);
  196. val &= ~USBOTGSS_UTMI_OTG_STATUS_SESSEND;
  197. val |= USBOTGSS_UTMI_OTG_STATUS_IDDIG
  198. | USBOTGSS_UTMI_OTG_STATUS_VBUSVALID
  199. | USBOTGSS_UTMI_OTG_STATUS_SESSVALID
  200. | USBOTGSS_UTMI_OTG_STATUS_POWERPRESENT;
  201. dwc3_omap_write_utmi_status(omap, val);
  202. break;
  203. case OMAP_DWC3_ID_FLOAT:
  204. case OMAP_DWC3_VBUS_OFF:
  205. dev_dbg(omap->dev, "VBUS Disconnect\n");
  206. val = dwc3_omap_read_utmi_status(omap);
  207. val &= ~(USBOTGSS_UTMI_OTG_STATUS_SESSVALID
  208. | USBOTGSS_UTMI_OTG_STATUS_VBUSVALID
  209. | USBOTGSS_UTMI_OTG_STATUS_POWERPRESENT);
  210. val |= USBOTGSS_UTMI_OTG_STATUS_SESSEND
  211. | USBOTGSS_UTMI_OTG_STATUS_IDDIG;
  212. dwc3_omap_write_utmi_status(omap, val);
  213. break;
  214. default:
  215. dev_dbg(omap->dev, "invalid state\n");
  216. }
  217. }
  218. static irqreturn_t dwc3_omap_interrupt(int irq, void *_omap)
  219. {
  220. struct dwc3_omap *omap = _omap;
  221. u32 reg;
  222. reg = dwc3_omap_read_irqmisc_status(omap);
  223. if (reg & USBOTGSS_IRQMISC_DMADISABLECLR) {
  224. dev_dbg(omap->dev, "DMA Disable was Cleared\n");
  225. omap->dma_status = false;
  226. }
  227. if (reg & USBOTGSS_IRQMISC_OEVT)
  228. dev_dbg(omap->dev, "OTG Event\n");
  229. if (reg & USBOTGSS_IRQMISC_DRVVBUS_RISE)
  230. dev_dbg(omap->dev, "DRVVBUS Rise\n");
  231. if (reg & USBOTGSS_IRQMISC_CHRGVBUS_RISE)
  232. dev_dbg(omap->dev, "CHRGVBUS Rise\n");
  233. if (reg & USBOTGSS_IRQMISC_DISCHRGVBUS_RISE)
  234. dev_dbg(omap->dev, "DISCHRGVBUS Rise\n");
  235. if (reg & USBOTGSS_IRQMISC_IDPULLUP_RISE)
  236. dev_dbg(omap->dev, "IDPULLUP Rise\n");
  237. if (reg & USBOTGSS_IRQMISC_DRVVBUS_FALL)
  238. dev_dbg(omap->dev, "DRVVBUS Fall\n");
  239. if (reg & USBOTGSS_IRQMISC_CHRGVBUS_FALL)
  240. dev_dbg(omap->dev, "CHRGVBUS Fall\n");
  241. if (reg & USBOTGSS_IRQMISC_DISCHRGVBUS_FALL)
  242. dev_dbg(omap->dev, "DISCHRGVBUS Fall\n");
  243. if (reg & USBOTGSS_IRQMISC_IDPULLUP_FALL)
  244. dev_dbg(omap->dev, "IDPULLUP Fall\n");
  245. dwc3_omap_write_irqmisc_status(omap, reg);
  246. reg = dwc3_omap_read_irq0_status(omap);
  247. dwc3_omap_write_irq0_status(omap, reg);
  248. return IRQ_HANDLED;
  249. }
  250. static void dwc3_omap_enable_irqs(struct dwc3_omap *omap)
  251. {
  252. /* enable all IRQs */
  253. dwc3_omap_write_irq0_set(omap, USBOTGSS_IRQO_COREIRQ_ST);
  254. dwc3_omap_write_irqmisc_set(omap, USBOTGSS_INTERRUPTS);
  255. }
  256. static void dwc3_omap_disable_irqs(struct dwc3_omap *omap)
  257. {
  258. /* disable all IRQs */
  259. dwc3_omap_write_irq0_clr(omap, USBOTGSS_IRQO_COREIRQ_ST);
  260. dwc3_omap_write_irqmisc_clr(omap, USBOTGSS_INTERRUPTS);
  261. }
  262. static void dwc3_omap_map_offset(struct dwc3_omap *omap)
  263. {
  264. /*
  265. * Differentiate between OMAP5 and AM437x.
  266. *
  267. * For OMAP5(ES2.0) and AM437x wrapper revision is same, even
  268. * though there are changes in wrapper register offsets.
  269. *
  270. * Using dt compatible to differentiate AM437x.
  271. */
  272. #ifdef CONFIG_AM43XX
  273. omap->irq_eoi_offset = USBOTGSS_EOI_OFFSET;
  274. omap->irq0_offset = USBOTGSS_IRQ0_OFFSET;
  275. omap->irqmisc_offset = USBOTGSS_IRQMISC_OFFSET;
  276. omap->utmi_otg_offset = USBOTGSS_UTMI_OTG_OFFSET;
  277. omap->debug_offset = USBOTGSS_DEBUG_OFFSET;
  278. #endif
  279. }
  280. static void dwc3_omap_set_utmi_mode(struct dwc3_omap *omap, int utmi_mode)
  281. {
  282. u32 reg;
  283. reg = dwc3_omap_read_utmi_status(omap);
  284. switch (utmi_mode) {
  285. case DWC3_OMAP_UTMI_MODE_SW:
  286. reg |= USBOTGSS_UTMI_OTG_STATUS_SW_MODE;
  287. break;
  288. case DWC3_OMAP_UTMI_MODE_HW:
  289. reg &= ~USBOTGSS_UTMI_OTG_STATUS_SW_MODE;
  290. break;
  291. default:
  292. dev_dbg(omap->dev, "UNKNOWN utmi mode %d\n", utmi_mode);
  293. }
  294. dwc3_omap_write_utmi_status(omap, reg);
  295. }
  296. /**
  297. * dwc3_omap_uboot_init - dwc3 omap uboot initialization code
  298. * @dev: struct dwc3_omap_device containing initialization data
  299. *
  300. * Entry point for dwc3 omap driver (equivalent to dwc3_omap_probe in linux
  301. * kernel driver). Pointer to dwc3_omap_device should be passed containing
  302. * base address and other initialization data. Returns '0' on success and
  303. * a negative value on failure.
  304. *
  305. * Generally called from board_usb_init() implemented in board file.
  306. */
  307. int dwc3_omap_uboot_init(struct dwc3_omap_device *omap_dev)
  308. {
  309. u32 reg;
  310. struct device *dev = NULL;
  311. struct dwc3_omap *omap;
  312. omap = devm_kzalloc(dev, sizeof(*omap), GFP_KERNEL);
  313. if (!omap)
  314. return -ENOMEM;
  315. omap->base = omap_dev->base;
  316. omap->index = omap_dev->index;
  317. dwc3_omap_map_offset(omap);
  318. dwc3_omap_set_utmi_mode(omap, omap_dev->utmi_mode);
  319. /* check the DMA Status */
  320. reg = dwc3_omap_readl(omap->base, USBOTGSS_SYSCONFIG);
  321. omap->dma_status = !!(reg & USBOTGSS_SYSCONFIG_DMADISABLE);
  322. dwc3_omap_set_mailbox(omap, omap_dev->vbus_id_status);
  323. dwc3_omap_enable_irqs(omap);
  324. list_add_tail(&omap->list, &dwc3_omap_list);
  325. return 0;
  326. }
  327. /**
  328. * dwc3_omap_uboot_exit - dwc3 omap uboot cleanup code
  329. * @index: index of this controller
  330. *
  331. * Performs cleanup of memory allocated in dwc3_omap_uboot_init
  332. * (equivalent to dwc3_omap_remove in linux). index of _this_ controller
  333. * should be passed and should match with the index passed in
  334. * dwc3_omap_device during init.
  335. *
  336. * Generally called from board file.
  337. */
  338. void dwc3_omap_uboot_exit(int index)
  339. {
  340. struct dwc3_omap *omap = NULL;
  341. list_for_each_entry(omap, &dwc3_omap_list, list) {
  342. if (omap->index != index)
  343. continue;
  344. dwc3_omap_disable_irqs(omap);
  345. list_del(&omap->list);
  346. kfree(omap);
  347. break;
  348. }
  349. }
  350. /**
  351. * dwc3_omap_uboot_interrupt_status - check the status of interrupt
  352. * @index: index of this controller
  353. *
  354. * Checks the status of interrupts and returns true if an interrupt
  355. * is detected or false otherwise.
  356. *
  357. * Generally called from board file.
  358. */
  359. int dwc3_omap_uboot_interrupt_status(int index)
  360. {
  361. struct dwc3_omap *omap = NULL;
  362. list_for_each_entry(omap, &dwc3_omap_list, list)
  363. if (omap->index == index)
  364. return dwc3_omap_interrupt(-1, omap);
  365. return 0;
  366. }
  367. MODULE_ALIAS("platform:omap-dwc3");
  368. MODULE_AUTHOR("Felipe Balbi <balbi@ti.com>");
  369. MODULE_LICENSE("GPL v2");
  370. MODULE_DESCRIPTION("DesignWare USB3 OMAP Glue Layer");