usb_dc_dwc2.c 42 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191
  1. /*
  2. * Copyright (c) 2022, sakumisu
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include "usbd_core.h"
  7. #include "usb_dwc2_reg.h"
  8. // clang-format off
  9. #if defined ( __CC_ARM )
  10. #ifndef __UNALIGNED_UINT32_WRITE
  11. #define __UNALIGNED_UINT32_WRITE(addr, val) ((*((__packed uint32_t *)(addr))) = (val))
  12. #endif
  13. #ifndef __UNALIGNED_UINT32_READ
  14. #define __UNALIGNED_UINT32_READ(addr) (*((const __packed uint32_t *)(addr)))
  15. #endif
  16. #elif defined (__ARMCC_VERSION) && (__ARMCC_VERSION >= 6010050)
  17. #ifndef __UNALIGNED_UINT32_WRITE
  18. #pragma clang diagnostic push
  19. #pragma clang diagnostic ignored "-Wpacked"
  20. /*lint -esym(9058, T_UINT32_WRITE)*/ /* disable MISRA 2012 Rule 2.4 for T_UINT32_WRITE */
  21. __PACKED_STRUCT T_UINT32_WRITE { uint32_t v; };
  22. #pragma clang diagnostic pop
  23. #define __UNALIGNED_UINT32_WRITE(addr, val) (void)((((struct T_UINT32_WRITE *)(void *)(addr))->v) = (val))
  24. #endif
  25. #ifndef __UNALIGNED_UINT32_READ
  26. #pragma clang diagnostic push
  27. #pragma clang diagnostic ignored "-Wpacked"
  28. /*lint -esym(9058, T_UINT32_READ)*/ /* disable MISRA 2012 Rule 2.4 for T_UINT32_READ */
  29. __PACKED_STRUCT T_UINT32_READ { uint32_t v; };
  30. #pragma clang diagnostic pop
  31. #define __UNALIGNED_UINT32_READ(addr) (((const struct T_UINT32_READ *)(const void *)(addr))->v)
  32. #endif
  33. #elif defined ( __GNUC__ )
  34. #ifndef __UNALIGNED_UINT32_WRITE
  35. #pragma GCC diagnostic push
  36. #pragma GCC diagnostic ignored "-Wpacked"
  37. #pragma GCC diagnostic ignored "-Wattributes"
  38. __PACKED_STRUCT T_UINT32_WRITE { uint32_t v; };
  39. #pragma GCC diagnostic pop
  40. #define __UNALIGNED_UINT32_WRITE(addr, val) (void)((((struct T_UINT32_WRITE *)(void *)(addr))->v) = (val))
  41. #endif
  42. #ifndef __UNALIGNED_UINT32_READ
  43. #pragma GCC diagnostic push
  44. #pragma GCC diagnostic ignored "-Wpacked"
  45. #pragma GCC diagnostic ignored "-Wattributes"
  46. __PACKED_STRUCT T_UINT32_READ { uint32_t v; };
  47. #pragma GCC diagnostic pop
  48. #define __UNALIGNED_UINT32_READ(addr) (((const struct T_UINT32_READ *)(const void *)(addr))->v)
  49. #endif
  50. #endif
  51. // clang-format on
  52. //#define CONFIG_USB_DWC2_DMA_ENABLE
  53. #ifndef CONFIG_USB_DWC2_RXALL_FIFO_SIZE
  54. #define CONFIG_USB_DWC2_RXALL_FIFO_SIZE (1024 / 4)
  55. #endif
  56. #ifndef CONFIG_USB_DWC2_TX0_FIFO_SIZE
  57. #define CONFIG_USB_DWC2_TX0_FIFO_SIZE (64 / 4)
  58. #endif
  59. #ifndef CONFIG_USB_DWC2_TX1_FIFO_SIZE
  60. #define CONFIG_USB_DWC2_TX1_FIFO_SIZE (512 / 4)
  61. #endif
  62. #ifndef CONFIG_USB_DWC2_TX2_FIFO_SIZE
  63. #define CONFIG_USB_DWC2_TX2_FIFO_SIZE (64 / 4)
  64. #endif
  65. #ifndef CONFIG_USB_DWC2_TX3_FIFO_SIZE
  66. #define CONFIG_USB_DWC2_TX3_FIFO_SIZE (64 / 4)
  67. #endif
  68. #ifndef CONFIG_USB_DWC2_TX4_FIFO_SIZE
  69. #define CONFIG_USB_DWC2_TX4_FIFO_SIZE (64 / 4)
  70. #endif
  71. #ifndef CONFIG_USB_DWC2_TX5_FIFO_SIZE
  72. #define CONFIG_USB_DWC2_TX5_FIFO_SIZE (64 / 4)
  73. #endif
  74. #ifndef CONFIG_USB_DWC2_TX6_FIFO_SIZE
  75. #define CONFIG_USB_DWC2_TX6_FIFO_SIZE (0 / 4)
  76. #endif
  77. #ifndef CONFIG_USB_DWC2_TX7_FIFO_SIZE
  78. #define CONFIG_USB_DWC2_TX7_FIFO_SIZE (0 / 4)
  79. #endif
  80. #ifndef CONFIG_USB_DWC2_TX8_FIFO_SIZE
  81. #define CONFIG_USB_DWC2_TX8_FIFO_SIZE (0 / 4)
  82. #endif
  83. #define USBD_BASE (g_usbdev_bus[0].reg_base)
  84. #define USB_OTG_GLB ((DWC2_GlobalTypeDef *)(USBD_BASE))
  85. #define USB_OTG_DEV ((DWC2_DeviceTypeDef *)(USBD_BASE + USB_OTG_DEVICE_BASE))
  86. #define USB_OTG_PCGCCTL *(__IO uint32_t *)((uint32_t)USBD_BASE + USB_OTG_PCGCCTL_BASE)
  87. #define USB_OTG_INEP(i) ((DWC2_INEndpointTypeDef *)(USBD_BASE + USB_OTG_IN_ENDPOINT_BASE + ((i)*USB_OTG_EP_REG_SIZE)))
  88. #define USB_OTG_OUTEP(i) ((DWC2_OUTEndpointTypeDef *)(USBD_BASE + USB_OTG_OUT_ENDPOINT_BASE + ((i)*USB_OTG_EP_REG_SIZE)))
  89. #define USB_OTG_FIFO(i) *(__IO uint32_t *)(USBD_BASE + USB_OTG_FIFO_BASE + ((i)*USB_OTG_FIFO_SIZE))
  90. extern uint32_t SystemCoreClock;
  91. /* Endpoint state */
  92. struct dwc2_ep_state {
  93. uint16_t ep_mps; /* Endpoint max packet size */
  94. uint8_t ep_type; /* Endpoint type */
  95. uint8_t ep_stalled; /* Endpoint stall flag */
  96. uint8_t *xfer_buf;
  97. uint32_t xfer_len;
  98. uint32_t actual_xfer_len;
  99. };
  100. /* Driver state */
  101. USB_NOCACHE_RAM_SECTION struct dwc2_udc {
  102. __attribute__((aligned(32))) struct usb_setup_packet setup;
  103. struct dwc2_ep_state in_ep[CONFIG_USBDEV_EP_NUM]; /*!< IN endpoint parameters*/
  104. struct dwc2_ep_state out_ep[CONFIG_USBDEV_EP_NUM]; /*!< OUT endpoint parameters */
  105. } g_dwc2_udc;
  106. static inline int dwc2_reset(void)
  107. {
  108. volatile uint32_t count = 0U;
  109. /* Wait for AHB master IDLE state. */
  110. do {
  111. if (++count > 200000U) {
  112. return -1;
  113. }
  114. } while ((USB_OTG_GLB->GRSTCTL & USB_OTG_GRSTCTL_AHBIDL) == 0U);
  115. /* Core Soft Reset */
  116. count = 0U;
  117. USB_OTG_GLB->GRSTCTL |= USB_OTG_GRSTCTL_CSRST;
  118. do {
  119. if (++count > 200000U) {
  120. return -1;
  121. }
  122. } while ((USB_OTG_GLB->GRSTCTL & USB_OTG_GRSTCTL_CSRST) == USB_OTG_GRSTCTL_CSRST);
  123. return 0;
  124. }
  125. static inline int dwc2_core_init(void)
  126. {
  127. int ret;
  128. #if defined(CONFIG_USB_HS)
  129. /* Init The ULPI Interface */
  130. USB_OTG_GLB->GUSBCFG &= ~(USB_OTG_GUSBCFG_TSDPS | USB_OTG_GUSBCFG_ULPIFSLS | USB_OTG_GUSBCFG_PHYSEL);
  131. /* Select vbus source */
  132. USB_OTG_GLB->GUSBCFG &= ~(USB_OTG_GUSBCFG_ULPIEVBUSD | USB_OTG_GUSBCFG_ULPIEVBUSI);
  133. /* Reset after a PHY select */
  134. ret = dwc2_reset();
  135. #else
  136. /* Select FS Embedded PHY */
  137. USB_OTG_GLB->GUSBCFG |= USB_OTG_GUSBCFG_PHYSEL;
  138. /* Reset after a PHY select */
  139. ret = dwc2_reset();
  140. #endif
  141. return ret;
  142. }
  143. static inline void dwc2_set_mode(uint8_t mode)
  144. {
  145. USB_OTG_GLB->GUSBCFG &= ~(USB_OTG_GUSBCFG_FHMOD | USB_OTG_GUSBCFG_FDMOD);
  146. if (mode == USB_OTG_MODE_HOST) {
  147. USB_OTG_GLB->GUSBCFG |= USB_OTG_GUSBCFG_FHMOD;
  148. } else if (mode == USB_OTG_MODE_DEVICE) {
  149. USB_OTG_GLB->GUSBCFG |= USB_OTG_GUSBCFG_FDMOD;
  150. }
  151. }
  152. static inline int dwc2_flush_rxfifo(void)
  153. {
  154. volatile uint32_t count = 0;
  155. USB_OTG_GLB->GRSTCTL = USB_OTG_GRSTCTL_RXFFLSH;
  156. do {
  157. if (++count > 200000U) {
  158. return -1;
  159. }
  160. } while ((USB_OTG_GLB->GRSTCTL & USB_OTG_GRSTCTL_RXFFLSH) == USB_OTG_GRSTCTL_RXFFLSH);
  161. return 0;
  162. }
  163. static inline int dwc2_flush_txfifo(uint32_t num)
  164. {
  165. volatile uint32_t count = 0U;
  166. USB_OTG_GLB->GRSTCTL = (USB_OTG_GRSTCTL_TXFFLSH | (num << 6));
  167. do {
  168. if (++count > 200000U) {
  169. return -1;
  170. }
  171. } while ((USB_OTG_GLB->GRSTCTL & USB_OTG_GRSTCTL_TXFFLSH) == USB_OTG_GRSTCTL_TXFFLSH);
  172. return 0;
  173. }
  174. static void dwc2_set_turnaroundtime(uint32_t hclk, uint8_t speed)
  175. {
  176. uint32_t UsbTrd;
  177. /* The USBTRD is configured according to the tables below, depending on AHB frequency
  178. used by application. In the low AHB frequency range it is used to stretch enough the USB response
  179. time to IN tokens, the USB turnaround time, so to compensate for the longer AHB read access
  180. latency to the Data FIFO */
  181. if (speed == USB_OTG_SPEED_FULL) {
  182. if ((hclk >= 14200000U) && (hclk < 15000000U)) {
  183. /* hclk Clock Range between 14.2-15 MHz */
  184. UsbTrd = 0xFU;
  185. } else if ((hclk >= 15000000U) && (hclk < 16000000U)) {
  186. /* hclk Clock Range between 15-16 MHz */
  187. UsbTrd = 0xEU;
  188. } else if ((hclk >= 16000000U) && (hclk < 17200000U)) {
  189. /* hclk Clock Range between 16-17.2 MHz */
  190. UsbTrd = 0xDU;
  191. } else if ((hclk >= 17200000U) && (hclk < 18500000U)) {
  192. /* hclk Clock Range between 17.2-18.5 MHz */
  193. UsbTrd = 0xCU;
  194. } else if ((hclk >= 18500000U) && (hclk < 20000000U)) {
  195. /* hclk Clock Range between 18.5-20 MHz */
  196. UsbTrd = 0xBU;
  197. } else if ((hclk >= 20000000U) && (hclk < 21800000U)) {
  198. /* hclk Clock Range between 20-21.8 MHz */
  199. UsbTrd = 0xAU;
  200. } else if ((hclk >= 21800000U) && (hclk < 24000000U)) {
  201. /* hclk Clock Range between 21.8-24 MHz */
  202. UsbTrd = 0x9U;
  203. } else if ((hclk >= 24000000U) && (hclk < 27700000U)) {
  204. /* hclk Clock Range between 24-27.7 MHz */
  205. UsbTrd = 0x8U;
  206. } else if ((hclk >= 27700000U) && (hclk < 32000000U)) {
  207. /* hclk Clock Range between 27.7-32 MHz */
  208. UsbTrd = 0x7U;
  209. } else /* if(hclk >= 32000000) */
  210. {
  211. /* hclk Clock Range between 32-200 MHz */
  212. UsbTrd = 0x6U;
  213. }
  214. } else if (speed == USB_OTG_SPEED_HIGH) {
  215. UsbTrd = USBD_HS_TRDT_VALUE;
  216. } else {
  217. UsbTrd = USBD_DEFAULT_TRDT_VALUE;
  218. }
  219. USB_OTG_GLB->GUSBCFG |= USB_OTG_GUSBCFG_TOCAL;
  220. USB_OTG_GLB->GUSBCFG &= ~USB_OTG_GUSBCFG_TRDT;
  221. USB_OTG_GLB->GUSBCFG |= (uint32_t)((UsbTrd << USB_OTG_GUSBCFG_TRDT_Pos) & USB_OTG_GUSBCFG_TRDT);
  222. }
  223. static void dwc2_set_txfifo(uint8_t fifo, uint16_t size)
  224. {
  225. uint8_t i;
  226. uint32_t tx_offset;
  227. /* TXn min size = 16 words. (n : Transmit FIFO index)
  228. When a TxFIFO is not used, the Configuration should be as follows:
  229. case 1 : n > m and Txn is not used (n,m : Transmit FIFO indexes)
  230. --> Txm can use the space allocated for Txn.
  231. case2 : n < m and Txn is not used (n,m : Transmit FIFO indexes)
  232. --> Txn should be configured with the minimum space of 16 words
  233. The FIFO is used optimally when used TxFIFOs are allocated in the top
  234. of the FIFO.Ex: use EP1 and EP2 as IN instead of EP1 and EP3 as IN ones.
  235. When DMA is used 3n * FIFO locations should be reserved for internal DMA registers */
  236. tx_offset = USB_OTG_GLB->GRXFSIZ;
  237. if (fifo == 0U) {
  238. USB_OTG_GLB->DIEPTXF0_HNPTXFSIZ = ((uint32_t)size << 16) | tx_offset;
  239. } else {
  240. tx_offset += (USB_OTG_GLB->DIEPTXF0_HNPTXFSIZ) >> 16;
  241. for (i = 0U; i < (fifo - 1U); i++) {
  242. tx_offset += (USB_OTG_GLB->DIEPTXF[i] >> 16);
  243. }
  244. /* Multiply Tx_Size by 2 to get higher performance */
  245. USB_OTG_GLB->DIEPTXF[fifo - 1U] = ((uint32_t)size << 16) | tx_offset;
  246. }
  247. USB_LOG_INFO("fifo%d size:%04x, offset:%04x\r\n", fifo, size, tx_offset);
  248. }
  249. static uint8_t dwc2_get_devspeed(void)
  250. {
  251. uint8_t speed;
  252. uint32_t DevEnumSpeed = USB_OTG_DEV->DSTS & USB_OTG_DSTS_ENUMSPD;
  253. if (DevEnumSpeed == DSTS_ENUMSPD_HS_PHY_30MHZ_OR_60MHZ) {
  254. speed = USB_OTG_SPEED_HIGH;
  255. } else if ((DevEnumSpeed == DSTS_ENUMSPD_FS_PHY_30MHZ_OR_60MHZ) ||
  256. (DevEnumSpeed == DSTS_ENUMSPD_FS_PHY_48MHZ)) {
  257. speed = USB_OTG_SPEED_FULL;
  258. } else {
  259. speed = 0xFU;
  260. }
  261. return speed;
  262. }
  263. static void dwc2_ep0_start_read_setup(uint8_t *psetup)
  264. {
  265. USB_OTG_OUTEP(0U)->DOEPTSIZ = 0U;
  266. USB_OTG_OUTEP(0U)->DOEPTSIZ |= (USB_OTG_DOEPTSIZ_PKTCNT & (1U << 19));
  267. USB_OTG_OUTEP(0U)->DOEPTSIZ |= (3U * 8U);
  268. USB_OTG_OUTEP(0U)->DOEPTSIZ |= USB_OTG_DOEPTSIZ_STUPCNT;
  269. #ifdef CONFIG_USB_DWC2_DMA_ENABLE
  270. USB_OTG_OUTEP(0U)->DOEPDMA = (uint32_t)psetup;
  271. /* EP enable */
  272. USB_OTG_OUTEP(0U)->DOEPCTL |= USB_OTG_DOEPCTL_EPENA | USB_OTG_DOEPCTL_USBAEP;
  273. #endif
  274. }
  275. void dwc2_ep_write(uint8_t ep_idx, uint8_t *src, uint16_t len)
  276. {
  277. uint32_t *pSrc = (uint32_t *)src;
  278. uint32_t count32b, i;
  279. count32b = ((uint32_t)len + 3U) / 4U;
  280. for (i = 0U; i < count32b; i++) {
  281. USB_OTG_FIFO((uint32_t)ep_idx) = __UNALIGNED_UINT32_READ(pSrc);
  282. pSrc++;
  283. }
  284. }
  285. void dwc2_ep_read(uint8_t *dest, uint16_t len)
  286. {
  287. uint32_t *pDest = (uint32_t *)dest;
  288. uint32_t i;
  289. uint32_t count32b = ((uint32_t)len + 3U) / 4U;
  290. for (i = 0U; i < count32b; i++) {
  291. __UNALIGNED_UINT32_WRITE(pDest, USB_OTG_FIFO(0U));
  292. pDest++;
  293. }
  294. }
  295. static void dwc2_tx_fifo_empty_procecss(uint8_t ep_idx)
  296. {
  297. uint32_t len;
  298. uint32_t len32b;
  299. uint32_t fifoemptymsk;
  300. len = g_dwc2_udc.in_ep[ep_idx].xfer_len - g_dwc2_udc.in_ep[ep_idx].actual_xfer_len;
  301. if (len > g_dwc2_udc.in_ep[ep_idx].ep_mps) {
  302. len = g_dwc2_udc.in_ep[ep_idx].ep_mps;
  303. }
  304. len32b = (len + 3U) / 4U;
  305. while (((USB_OTG_INEP(ep_idx)->DTXFSTS & USB_OTG_DTXFSTS_INEPTFSAV) >= len32b) &&
  306. (g_dwc2_udc.in_ep[ep_idx].actual_xfer_len < g_dwc2_udc.in_ep[ep_idx].xfer_len) && (g_dwc2_udc.in_ep[ep_idx].xfer_len != 0U)) {
  307. /* Write the FIFO */
  308. len = g_dwc2_udc.in_ep[ep_idx].xfer_len - g_dwc2_udc.in_ep[ep_idx].actual_xfer_len;
  309. if (len > g_dwc2_udc.in_ep[ep_idx].ep_mps) {
  310. len = g_dwc2_udc.in_ep[ep_idx].ep_mps;
  311. }
  312. dwc2_ep_write(ep_idx, g_dwc2_udc.in_ep[ep_idx].xfer_buf, len);
  313. g_dwc2_udc.in_ep[ep_idx].xfer_buf += len;
  314. g_dwc2_udc.in_ep[ep_idx].actual_xfer_len += len;
  315. }
  316. if (g_dwc2_udc.in_ep[ep_idx].xfer_len <= g_dwc2_udc.in_ep[ep_idx].actual_xfer_len) {
  317. fifoemptymsk = (uint32_t)(0x1UL << (ep_idx & 0x0f));
  318. USB_OTG_DEV->DIEPEMPMSK &= ~fifoemptymsk;
  319. }
  320. }
  321. /**
  322. * @brief dwc2_get_glb_intstatus: return the global USB interrupt status
  323. * @retval status
  324. */
  325. static inline uint32_t dwc2_get_glb_intstatus(void)
  326. {
  327. uint32_t tmpreg;
  328. tmpreg = USB_OTG_GLB->GINTSTS;
  329. tmpreg &= USB_OTG_GLB->GINTMSK;
  330. return tmpreg;
  331. }
  332. /**
  333. * @brief dwc2_get_outeps_intstatus: return the USB device OUT endpoints interrupt status
  334. * @retval status
  335. */
  336. static inline uint32_t dwc2_get_outeps_intstatus(void)
  337. {
  338. uint32_t tmpreg;
  339. tmpreg = USB_OTG_DEV->DAINT;
  340. tmpreg &= USB_OTG_DEV->DAINTMSK;
  341. return ((tmpreg & 0xffff0000U) >> 16);
  342. }
  343. /**
  344. * @brief dwc2_get_ineps_intstatus: return the USB device IN endpoints interrupt status
  345. * @retval status
  346. */
  347. static inline uint32_t dwc2_get_ineps_intstatus(void)
  348. {
  349. uint32_t tmpreg;
  350. tmpreg = USB_OTG_DEV->DAINT;
  351. tmpreg &= USB_OTG_DEV->DAINTMSK;
  352. return ((tmpreg & 0xFFFFU));
  353. }
  354. /**
  355. * @brief Returns Device OUT EP Interrupt register
  356. * @param epnum endpoint number
  357. * This parameter can be a value from 0 to 15
  358. * @retval Device OUT EP Interrupt register
  359. */
  360. static inline uint32_t dwc2_get_outep_intstatus(uint8_t epnum)
  361. {
  362. uint32_t tmpreg;
  363. tmpreg = USB_OTG_OUTEP((uint32_t)epnum)->DOEPINT;
  364. tmpreg &= USB_OTG_DEV->DOEPMSK;
  365. return tmpreg;
  366. }
  367. /**
  368. * @brief Returns Device IN EP Interrupt register
  369. * @param epnum endpoint number
  370. * This parameter can be a value from 0 to 15
  371. * @retval Device IN EP Interrupt register
  372. */
  373. static inline uint32_t dwc2_get_inep_intstatus(uint8_t epnum)
  374. {
  375. uint32_t tmpreg, msk, emp;
  376. msk = USB_OTG_DEV->DIEPMSK;
  377. emp = USB_OTG_DEV->DIEPEMPMSK;
  378. msk |= ((emp >> (epnum & 0x07)) & 0x1U) << 7;
  379. tmpreg = USB_OTG_INEP((uint32_t)epnum)->DIEPINT & msk;
  380. return tmpreg;
  381. }
  382. __WEAK void usb_dc_low_level_init(void)
  383. {
  384. }
  385. __WEAK void usb_dc_low_level_deinit(void)
  386. {
  387. }
  388. int usb_dc_init(uint8_t busid)
  389. {
  390. int ret;
  391. uint8_t fsphy_type;
  392. uint8_t hsphy_type;
  393. uint8_t dma_support;
  394. uint8_t endpoints;
  395. uint32_t fifo_num;
  396. memset(&g_dwc2_udc, 0, sizeof(struct dwc2_udc));
  397. usb_dc_low_level_init();
  398. /*
  399. Full-Speed PHY Interface Type (FSPhyType)
  400. 2'b00: Full-speed interface not supported
  401. 2'b01: Dedicated full-speed interface
  402. 2'b10: FS pins shared with UTMI+ pins
  403. 2'b11: FS pins shared with ULPI pins
  404. High-Speed PHY Interface Type (HSPhyType)
  405. 2'b00: High-Speed interface not supported
  406. 2'b01: UTMI+
  407. 2'b10: ULPI
  408. 2'b11: UTMI+ and ULPI
  409. Architecture (OtgArch)
  410. 2'b00: Slave-Only
  411. 2'b01: External DMA
  412. 2'b10: Internal DMA
  413. Others: Reserved
  414. */
  415. fsphy_type = ((USB_OTG_GLB->GHWCFG2 & (0x03 << 8)) >> 8);
  416. hsphy_type = ((USB_OTG_GLB->GHWCFG2 & (0x03 << 6)) >> 6);
  417. dma_support = ((USB_OTG_GLB->GHWCFG2 & (0x03 << 3)) >> 3);
  418. endpoints = ((USB_OTG_GLB->GHWCFG2 & (0x0f << 10)) >> 10) + 1;
  419. USB_LOG_INFO("========== dwc2 udc params ==========\r\n");
  420. USB_LOG_INFO("CID:%08x\r\n", USB_OTG_GLB->CID);
  421. USB_LOG_INFO("GSNPSID:%08x\r\n", USB_OTG_GLB->GSNPSID);
  422. USB_LOG_INFO("GHWCFG1:%08x\r\n", USB_OTG_GLB->GHWCFG1);
  423. USB_LOG_INFO("GHWCFG2:%08x\r\n", USB_OTG_GLB->GHWCFG2);
  424. USB_LOG_INFO("GHWCFG3:%08x\r\n", USB_OTG_GLB->GHWCFG3);
  425. USB_LOG_INFO("GHWCFG4:%08x\r\n", USB_OTG_GLB->GHWCFG4);
  426. USB_LOG_INFO("dwc2 fsphy type:%d, hsphy type:%d, dma support:%d\r\n", fsphy_type, hsphy_type, dma_support);
  427. USB_LOG_INFO("dwc2 has %d endpoints and dfifo depth(32-bit words) is %d, default config: %d endpoints\r\n", endpoints, (USB_OTG_GLB->GHWCFG3 >> 16), CONFIG_USBDEV_EP_NUM);
  428. USB_LOG_INFO("=================================\r\n");
  429. if (endpoints < CONFIG_USBDEV_EP_NUM) {
  430. USB_LOG_ERR("dwc2 has less endpoints than config, please check\r\n");
  431. while (1) {
  432. }
  433. }
  434. USB_OTG_DEV->DCTL |= USB_OTG_DCTL_SDIS;
  435. USB_OTG_GLB->GAHBCFG &= ~USB_OTG_GAHBCFG_GINT;
  436. /* This is vendor register */
  437. USB_OTG_GLB->GCCFG = usbd_get_dwc2_gccfg_conf(USBD_BASE);
  438. ret = dwc2_core_init();
  439. /* Force Device Mode*/
  440. dwc2_set_mode(USB_OTG_MODE_DEVICE);
  441. /* B-peripheral session valid override enable */
  442. // USB_OTG_GLB->GOTGCTL |= USB_OTG_GOTGCTL_BVALOEN;
  443. // USB_OTG_GLB->GOTGCTL |= USB_OTG_GOTGCTL_BVALOVAL;
  444. for (uint8_t i = 0U; i < 15U; i++) {
  445. USB_OTG_GLB->DIEPTXF[i] = 0U;
  446. }
  447. /* Restart the Phy Clock */
  448. USB_OTG_PCGCCTL = 0U;
  449. /* Device speed configuration */
  450. USB_OTG_DEV->DCFG &= ~USB_OTG_DCFG_DSPD;
  451. #if defined(CONFIG_USB_HS)
  452. USB_OTG_DEV->DCFG |= USB_OTG_SPEED_HIGH;
  453. #else
  454. if (hsphy_type == 0) {
  455. USB_OTG_DEV->DCFG |= USB_OTG_SPEED_FULL;
  456. } else {
  457. USB_OTG_DEV->DCFG |= USB_OTG_SPEED_HIGH_IN_FULL;
  458. }
  459. #endif
  460. /* Clear all pending Device Interrupts */
  461. USB_OTG_DEV->DIEPMSK = 0U;
  462. USB_OTG_DEV->DOEPMSK = 0U;
  463. USB_OTG_DEV->DAINTMSK = 0U;
  464. /* Disable all interrupts. */
  465. USB_OTG_GLB->GINTMSK = 0U;
  466. /* Clear any pending interrupts */
  467. USB_OTG_GLB->GINTSTS = 0xBFFFFFFFU;
  468. /* Enable interrupts matching to the Device mode ONLY */
  469. USB_OTG_GLB->GINTMSK = USB_OTG_GINTMSK_USBRST | USB_OTG_GINTMSK_ENUMDNEM |
  470. USB_OTG_GINTMSK_OEPINT | USB_OTG_GINTMSK_IEPINT |
  471. USB_OTG_GINTMSK_IISOIXFRM | USB_OTG_GINTMSK_PXFRM_IISOOXFRM;
  472. #ifdef CONFIG_USB_DWC2_DMA_ENABLE
  473. if (((USB_OTG_GLB->GHWCFG2 & (0x3U << 3)) >> 3) != 2) {
  474. USB_LOG_ERR("This dwc2 version does not support dma mode, so stop working\r\n");
  475. while (1) {
  476. }
  477. }
  478. USB_OTG_DEV->DCFG &= ~USB_OTG_DCFG_DESCDMA;
  479. USB_OTG_GLB->GAHBCFG |= (USB_OTG_GAHBCFG_DMAEN | USB_OTG_GAHBCFG_HBSTLEN_4);
  480. #else
  481. USB_OTG_GLB->GINTMSK |= USB_OTG_GINTMSK_RXFLVLM;
  482. #endif
  483. #if CONFIG_DWC2_VBUS_SENSING
  484. USB_OTG_GLB->GINTMSK |= (USB_OTG_GINTMSK_OTGINT | USB_OTG_GINTMSK_SRQIM);
  485. #endif
  486. #if 0
  487. USB_OTG_GLB->GINTMSK |= USB_OTG_GINTMSK_SOFM;
  488. #endif
  489. USB_OTG_GLB->GRXFSIZ = (CONFIG_USB_DWC2_RXALL_FIFO_SIZE);
  490. dwc2_set_txfifo(0, CONFIG_USB_DWC2_TX0_FIFO_SIZE);
  491. dwc2_set_txfifo(1, CONFIG_USB_DWC2_TX1_FIFO_SIZE);
  492. dwc2_set_txfifo(2, CONFIG_USB_DWC2_TX2_FIFO_SIZE);
  493. dwc2_set_txfifo(3, CONFIG_USB_DWC2_TX3_FIFO_SIZE);
  494. fifo_num = CONFIG_USB_DWC2_RXALL_FIFO_SIZE;
  495. fifo_num += CONFIG_USB_DWC2_TX0_FIFO_SIZE;
  496. fifo_num += CONFIG_USB_DWC2_TX1_FIFO_SIZE;
  497. fifo_num += CONFIG_USB_DWC2_TX2_FIFO_SIZE;
  498. fifo_num += CONFIG_USB_DWC2_TX3_FIFO_SIZE;
  499. #if CONFIG_USBDEV_EP_NUM > 4
  500. dwc2_set_txfifo(4, CONFIG_USB_DWC2_TX4_FIFO_SIZE);
  501. fifo_num += CONFIG_USB_DWC2_TX4_FIFO_SIZE;
  502. #endif
  503. #if CONFIG_USBDEV_EP_NUM > 5
  504. dwc2_set_txfifo(5, CONFIG_USB_DWC2_TX5_FIFO_SIZE);
  505. fifo_num += CONFIG_USB_DWC2_TX5_FIFO_SIZE;
  506. #endif
  507. #if CONFIG_USBDEV_EP_NUM > 6
  508. dwc2_set_txfifo(6, CONFIG_USB_DWC2_TX6_FIFO_SIZE);
  509. fifo_num += CONFIG_USB_DWC2_TX6_FIFO_SIZE;
  510. #endif
  511. #if CONFIG_USBDEV_EP_NUM > 7
  512. dwc2_set_txfifo(7, CONFIG_USB_DWC2_TX7_FIFO_SIZE);
  513. fifo_num += CONFIG_USB_DWC2_TX7_FIFO_SIZE;
  514. #endif
  515. #if CONFIG_USBDEV_EP_NUM > 8
  516. dwc2_set_txfifo(8, CONFIG_USB_DWC2_TX8_FIFO_SIZE);
  517. fifo_num += CONFIG_USB_DWC2_TX8_FIFO_SIZE;
  518. #endif
  519. if (fifo_num > (USB_OTG_GLB->GHWCFG3 >> 16)) {
  520. USB_LOG_ERR("Your fifo config is overflow, please check\r\n");
  521. while (1) {
  522. }
  523. }
  524. /* xxx32 chips do not follow (USB_OTG_GLB->GHWCFG3 >> 16) if hsphy_type is zero, they use 1.25KB(320 DWORD) */
  525. if ((hsphy_type == 0) && (fifo_num > 320)) {
  526. USB_LOG_ERR("Your fifo config is larger than 320 , please check\r\n");
  527. while (1) {
  528. }
  529. }
  530. ret = dwc2_flush_txfifo(0x10U);
  531. ret = dwc2_flush_rxfifo();
  532. USB_OTG_GLB->GAHBCFG |= USB_OTG_GAHBCFG_GINT;
  533. USB_OTG_DEV->DCTL &= ~USB_OTG_DCTL_SDIS;
  534. return ret;
  535. }
  536. int usb_dc_deinit(uint8_t busid)
  537. {
  538. USB_OTG_GLB->GAHBCFG |= USB_OTG_GAHBCFG_GINT;
  539. USB_OTG_DEV->DCTL |= USB_OTG_DCTL_SDIS;
  540. /* Clear Pending interrupt */
  541. for (uint8_t i = 0U; i < 15U; i++) {
  542. USB_OTG_INEP(i)->DIEPINT = 0xFB7FU;
  543. USB_OTG_OUTEP(i)->DOEPINT = 0xFB7FU;
  544. }
  545. /* Clear interrupt masks */
  546. USB_OTG_DEV->DIEPMSK = 0U;
  547. USB_OTG_DEV->DOEPMSK = 0U;
  548. USB_OTG_DEV->DAINTMSK = 0U;
  549. /* Flush the FIFO */
  550. dwc2_flush_txfifo(0x10U);
  551. dwc2_flush_rxfifo();
  552. usb_dc_low_level_deinit();
  553. return 0;
  554. }
  555. int usbd_set_address(uint8_t busid, const uint8_t addr)
  556. {
  557. USB_OTG_DEV->DCFG &= ~(USB_OTG_DCFG_DAD);
  558. USB_OTG_DEV->DCFG |= ((uint32_t)addr << 4) & USB_OTG_DCFG_DAD;
  559. return 0;
  560. }
  561. uint8_t usbd_get_port_speed(uint8_t busid)
  562. {
  563. uint8_t speed;
  564. uint32_t DevEnumSpeed = USB_OTG_DEV->DSTS & USB_OTG_DSTS_ENUMSPD;
  565. if (DevEnumSpeed == DSTS_ENUMSPD_HS_PHY_30MHZ_OR_60MHZ) {
  566. speed = USB_SPEED_HIGH;
  567. } else if ((DevEnumSpeed == DSTS_ENUMSPD_FS_PHY_30MHZ_OR_60MHZ) ||
  568. (DevEnumSpeed == DSTS_ENUMSPD_FS_PHY_48MHZ)) {
  569. speed = USB_SPEED_FULL;
  570. } else {
  571. speed = USB_SPEED_FULL;
  572. }
  573. return speed;
  574. }
  575. int usbd_ep_open(uint8_t busid, const struct usb_endpoint_descriptor *ep)
  576. {
  577. uint8_t ep_idx = USB_EP_GET_IDX(ep->bEndpointAddress);
  578. if (ep_idx > (CONFIG_USBDEV_EP_NUM - 1)) {
  579. USB_LOG_ERR("Ep addr %02x overflow\r\n", ep->bEndpointAddress);
  580. return -1;
  581. }
  582. if (USB_EP_DIR_IS_OUT(ep->bEndpointAddress)) {
  583. g_dwc2_udc.out_ep[ep_idx].ep_mps = USB_GET_MAXPACKETSIZE(ep->wMaxPacketSize);
  584. g_dwc2_udc.out_ep[ep_idx].ep_type = USB_GET_ENDPOINT_TYPE(ep->bmAttributes);
  585. USB_OTG_DEV->DAINTMSK |= USB_OTG_DAINTMSK_OEPM & (uint32_t)(1UL << (16 + ep_idx));
  586. if ((USB_OTG_OUTEP(ep_idx)->DOEPCTL & USB_OTG_DOEPCTL_USBAEP) == 0) {
  587. USB_OTG_OUTEP(ep_idx)->DOEPCTL |= (USB_GET_MAXPACKETSIZE(ep->wMaxPacketSize) & USB_OTG_DOEPCTL_MPSIZ) |
  588. ((uint32_t)USB_GET_ENDPOINT_TYPE(ep->bmAttributes) << 18) |
  589. USB_OTG_DIEPCTL_SD0PID_SEVNFRM |
  590. USB_OTG_DOEPCTL_USBAEP;
  591. }
  592. } else {
  593. uint16_t fifo_size;
  594. if (ep_idx == 0) {
  595. fifo_size = (USB_OTG_GLB->DIEPTXF0_HNPTXFSIZ >> 16);
  596. } else {
  597. fifo_size = (USB_OTG_GLB->DIEPTXF[ep_idx - 1U] >> 16);
  598. }
  599. if ((fifo_size * 4) < USB_GET_MAXPACKETSIZE(ep->wMaxPacketSize)) {
  600. USB_LOG_ERR("Ep addr %02x fifo overflow\r\n", ep->bEndpointAddress);
  601. return -2;
  602. }
  603. g_dwc2_udc.in_ep[ep_idx].ep_mps = USB_GET_MAXPACKETSIZE(ep->wMaxPacketSize);
  604. g_dwc2_udc.in_ep[ep_idx].ep_type = USB_GET_ENDPOINT_TYPE(ep->bmAttributes);
  605. USB_OTG_DEV->DAINTMSK |= USB_OTG_DAINTMSK_IEPM & (uint32_t)(1UL << ep_idx);
  606. if ((USB_OTG_INEP(ep_idx)->DIEPCTL & USB_OTG_DIEPCTL_USBAEP) == 0) {
  607. USB_OTG_INEP(ep_idx)->DIEPCTL |= (USB_GET_MAXPACKETSIZE(ep->wMaxPacketSize) & USB_OTG_DIEPCTL_MPSIZ) |
  608. ((uint32_t)USB_GET_ENDPOINT_TYPE(ep->bmAttributes) << 18) | (ep_idx << 22) |
  609. USB_OTG_DIEPCTL_SD0PID_SEVNFRM |
  610. USB_OTG_DIEPCTL_USBAEP;
  611. }
  612. dwc2_flush_txfifo(ep_idx);
  613. }
  614. return 0;
  615. }
  616. int usbd_ep_close(uint8_t busid, const uint8_t ep)
  617. {
  618. uint8_t ep_idx = USB_EP_GET_IDX(ep);
  619. volatile uint32_t count = 0U;
  620. if (USB_EP_DIR_IS_OUT(ep)) {
  621. if (USB_OTG_OUTEP(ep_idx)->DOEPCTL & USB_OTG_DOEPCTL_EPENA) {
  622. USB_OTG_OUTEP(ep_idx)->DOEPCTL |= USB_OTG_DOEPCTL_SNAK;
  623. USB_OTG_OUTEP(ep_idx)->DOEPCTL |= USB_OTG_DOEPCTL_EPDIS;
  624. /* Wait for endpoint disabled interrupt */
  625. count = 0;
  626. do {
  627. if (++count > 50000) {
  628. break;
  629. }
  630. } while ((USB_OTG_OUTEP(ep_idx)->DOEPINT & USB_OTG_DOEPINT_EPDISD) != USB_OTG_DOEPINT_EPDISD);
  631. /* Clear and unmask endpoint disabled interrupt */
  632. USB_OTG_OUTEP(ep_idx)->DOEPINT |= USB_OTG_DOEPINT_EPDISD;
  633. }
  634. USB_OTG_DEV->DEACHMSK &= ~(USB_OTG_DAINTMSK_OEPM & ((uint32_t)(1UL << (ep_idx & 0x07)) << 16));
  635. USB_OTG_DEV->DAINTMSK &= ~(USB_OTG_DAINTMSK_OEPM & ((uint32_t)(1UL << (ep_idx & 0x07)) << 16));
  636. USB_OTG_OUTEP(ep_idx)->DOEPCTL = 0;
  637. } else {
  638. if (USB_OTG_INEP(ep_idx)->DIEPCTL & USB_OTG_DIEPCTL_EPENA) {
  639. USB_OTG_INEP(ep_idx)->DIEPCTL |= USB_OTG_DIEPCTL_SNAK;
  640. USB_OTG_INEP(ep_idx)->DIEPCTL |= USB_OTG_DIEPCTL_EPDIS;
  641. /* Wait for endpoint disabled interrupt */
  642. count = 0;
  643. do {
  644. if (++count > 50000) {
  645. break;
  646. }
  647. } while ((USB_OTG_INEP(ep_idx)->DIEPINT & USB_OTG_DIEPINT_EPDISD) != USB_OTG_DIEPINT_EPDISD);
  648. /* Clear and unmask endpoint disabled interrupt */
  649. USB_OTG_INEP(ep_idx)->DIEPINT |= USB_OTG_DIEPINT_EPDISD;
  650. }
  651. USB_OTG_DEV->DEACHMSK &= ~(USB_OTG_DAINTMSK_IEPM & (uint32_t)(1UL << (ep_idx & 0x07)));
  652. USB_OTG_DEV->DAINTMSK &= ~(USB_OTG_DAINTMSK_IEPM & (uint32_t)(1UL << (ep_idx & 0x07)));
  653. USB_OTG_INEP(ep_idx)->DIEPCTL = 0;
  654. }
  655. return 0;
  656. }
  657. int usbd_ep_set_stall(uint8_t busid, const uint8_t ep)
  658. {
  659. uint8_t ep_idx = USB_EP_GET_IDX(ep);
  660. if (USB_EP_DIR_IS_OUT(ep)) {
  661. if (((USB_OTG_OUTEP(ep_idx)->DOEPCTL & USB_OTG_DOEPCTL_EPENA) == 0U) && (ep_idx != 0U)) {
  662. USB_OTG_OUTEP(ep_idx)->DOEPCTL &= ~(USB_OTG_DOEPCTL_EPDIS);
  663. }
  664. USB_OTG_OUTEP(ep_idx)->DOEPCTL |= USB_OTG_DOEPCTL_STALL;
  665. } else {
  666. if (((USB_OTG_INEP(ep_idx)->DIEPCTL & USB_OTG_DIEPCTL_EPENA) == 0U) && (ep_idx != 0U)) {
  667. USB_OTG_INEP(ep_idx)->DIEPCTL &= ~(USB_OTG_DIEPCTL_EPDIS);
  668. }
  669. USB_OTG_INEP(ep_idx)->DIEPCTL |= USB_OTG_DIEPCTL_STALL;
  670. }
  671. #ifdef CONFIG_USB_DWC2_DMA_ENABLE
  672. if (ep_idx == 0) {
  673. dwc2_ep0_start_read_setup((uint8_t *)&g_dwc2_udc.setup);
  674. }
  675. #endif
  676. return 0;
  677. }
  678. int usbd_ep_clear_stall(uint8_t busid, const uint8_t ep)
  679. {
  680. uint8_t ep_idx = USB_EP_GET_IDX(ep);
  681. if (USB_EP_DIR_IS_OUT(ep)) {
  682. USB_OTG_OUTEP(ep_idx)->DOEPCTL &= ~USB_OTG_DOEPCTL_STALL;
  683. if ((g_dwc2_udc.out_ep[ep_idx].ep_type == USB_ENDPOINT_TYPE_INTERRUPT) ||
  684. (g_dwc2_udc.out_ep[ep_idx].ep_type == USB_ENDPOINT_TYPE_BULK)) {
  685. USB_OTG_OUTEP(ep_idx)->DOEPCTL |= USB_OTG_DOEPCTL_SD0PID_SEVNFRM; /* DATA0 */
  686. }
  687. } else {
  688. USB_OTG_INEP(ep_idx)->DIEPCTL &= ~USB_OTG_DIEPCTL_STALL;
  689. if ((g_dwc2_udc.in_ep[ep_idx].ep_type == USB_ENDPOINT_TYPE_INTERRUPT) ||
  690. (g_dwc2_udc.in_ep[ep_idx].ep_type == USB_ENDPOINT_TYPE_BULK)) {
  691. USB_OTG_INEP(ep_idx)->DIEPCTL |= USB_OTG_DIEPCTL_SD0PID_SEVNFRM; /* DATA0 */
  692. }
  693. }
  694. return 0;
  695. }
  696. int usbd_ep_is_stalled(uint8_t busid, const uint8_t ep, uint8_t *stalled)
  697. {
  698. if (USB_EP_DIR_IS_OUT(ep)) {
  699. } else {
  700. }
  701. return 0;
  702. }
  703. int usbd_ep_start_write(uint8_t busid, const uint8_t ep, const uint8_t *data, uint32_t data_len)
  704. {
  705. uint8_t ep_idx = USB_EP_GET_IDX(ep);
  706. uint32_t pktcnt = 0;
  707. if (!data && data_len) {
  708. return -1;
  709. }
  710. #if 0 /* some chips have confused with this, so disable as default */
  711. if (USB_OTG_INEP(ep_idx)->DIEPCTL & USB_OTG_DIEPCTL_EPENA) {
  712. return -2;
  713. }
  714. #endif
  715. if (ep_idx && !(USB_OTG_INEP(ep_idx)->DIEPCTL & USB_OTG_DIEPCTL_MPSIZ)) {
  716. return -3;
  717. }
  718. if ((uint32_t)data & 0x03) {
  719. return -4;
  720. }
  721. g_dwc2_udc.in_ep[ep_idx].xfer_buf = (uint8_t *)data;
  722. g_dwc2_udc.in_ep[ep_idx].xfer_len = data_len;
  723. g_dwc2_udc.in_ep[ep_idx].actual_xfer_len = 0;
  724. USB_OTG_INEP(ep_idx)->DIEPTSIZ &= ~(USB_OTG_DIEPTSIZ_PKTCNT);
  725. USB_OTG_INEP(ep_idx)->DIEPTSIZ &= ~(USB_OTG_DIEPTSIZ_XFRSIZ);
  726. if (data_len == 0) {
  727. USB_OTG_INEP(ep_idx)->DIEPTSIZ |= (USB_OTG_DIEPTSIZ_PKTCNT & (1U << 19));
  728. USB_OTG_INEP(ep_idx)->DIEPCTL |= (USB_OTG_DIEPCTL_CNAK | USB_OTG_DIEPCTL_EPENA);
  729. return 0;
  730. }
  731. if (ep_idx == 0) {
  732. if (data_len > g_dwc2_udc.in_ep[ep_idx].ep_mps) {
  733. data_len = g_dwc2_udc.in_ep[ep_idx].ep_mps;
  734. }
  735. g_dwc2_udc.in_ep[ep_idx].xfer_len = data_len;
  736. USB_OTG_INEP(ep_idx)->DIEPTSIZ |= (USB_OTG_DIEPTSIZ_PKTCNT & (1U << 19));
  737. USB_OTG_INEP(ep_idx)->DIEPTSIZ |= (USB_OTG_DIEPTSIZ_XFRSIZ & data_len);
  738. } else {
  739. pktcnt = (uint16_t)((data_len + g_dwc2_udc.in_ep[ep_idx].ep_mps - 1U) / g_dwc2_udc.in_ep[ep_idx].ep_mps);
  740. USB_OTG_INEP(ep_idx)->DIEPTSIZ |= (USB_OTG_DIEPTSIZ_PKTCNT & (pktcnt << 19));
  741. USB_OTG_INEP(ep_idx)->DIEPTSIZ |= (USB_OTG_DIEPTSIZ_XFRSIZ & data_len);
  742. }
  743. if (g_dwc2_udc.in_ep[ep_idx].ep_type == USB_ENDPOINT_TYPE_ISOCHRONOUS) {
  744. if ((USB_OTG_DEV->DSTS & (1U << 8)) == 0U) {
  745. USB_OTG_INEP(ep_idx)->DIEPCTL &= ~USB_OTG_DIEPCTL_SD0PID_SEVNFRM;
  746. USB_OTG_INEP(ep_idx)->DIEPCTL |= USB_OTG_DIEPCTL_SODDFRM;
  747. } else {
  748. USB_OTG_INEP(ep_idx)->DIEPCTL &= ~USB_OTG_DIEPCTL_SODDFRM;
  749. USB_OTG_INEP(ep_idx)->DIEPCTL |= USB_OTG_DIEPCTL_SD0PID_SEVNFRM;
  750. }
  751. USB_OTG_INEP(ep_idx)->DIEPTSIZ &= ~(USB_OTG_DIEPTSIZ_MULCNT);
  752. USB_OTG_INEP(ep_idx)->DIEPTSIZ |= (USB_OTG_DIEPTSIZ_MULCNT & (1U << 29));
  753. }
  754. #ifdef CONFIG_USB_DWC2_DMA_ENABLE
  755. USB_OTG_INEP(ep_idx)->DIEPDMA = (uint32_t)data;
  756. USB_OTG_INEP(ep_idx)->DIEPCTL |= (USB_OTG_DIEPCTL_CNAK | USB_OTG_DIEPCTL_EPENA);
  757. #else
  758. USB_OTG_INEP(ep_idx)->DIEPCTL |= (USB_OTG_DIEPCTL_CNAK | USB_OTG_DIEPCTL_EPENA);
  759. /* Enable the Tx FIFO Empty Interrupt for this EP */
  760. if (data_len > 0U) {
  761. USB_OTG_DEV->DIEPEMPMSK |= 1UL << (ep_idx & 0x0f);
  762. }
  763. #endif
  764. return 0;
  765. }
  766. int usbd_ep_start_read(uint8_t busid, const uint8_t ep, uint8_t *data, uint32_t data_len)
  767. {
  768. uint8_t ep_idx = USB_EP_GET_IDX(ep);
  769. uint32_t pktcnt = 0;
  770. if (!data && data_len) {
  771. return -1;
  772. }
  773. #if 0 /* some chips have confused with this, so disable as default */
  774. if (USB_OTG_OUTEP(ep_idx)->DOEPCTL & USB_OTG_DOEPCTL_EPENA) {
  775. return -2;
  776. }
  777. #endif
  778. if (ep_idx && !(USB_OTG_OUTEP(ep_idx)->DOEPCTL & USB_OTG_DOEPCTL_MPSIZ)) {
  779. return -3;
  780. }
  781. if (((uint32_t)data) & 0x03) {
  782. return -4;
  783. }
  784. g_dwc2_udc.out_ep[ep_idx].xfer_buf = (uint8_t *)data;
  785. g_dwc2_udc.out_ep[ep_idx].xfer_len = data_len;
  786. g_dwc2_udc.out_ep[ep_idx].actual_xfer_len = 0;
  787. USB_OTG_OUTEP(ep_idx)->DOEPTSIZ &= ~(USB_OTG_DOEPTSIZ_PKTCNT);
  788. USB_OTG_OUTEP(ep_idx)->DOEPTSIZ &= ~(USB_OTG_DOEPTSIZ_XFRSIZ);
  789. if (data_len == 0) {
  790. USB_OTG_OUTEP(ep_idx)->DOEPTSIZ |= (USB_OTG_DOEPTSIZ_PKTCNT & (1 << 19));
  791. USB_OTG_OUTEP(ep_idx)->DOEPTSIZ |= (USB_OTG_DOEPTSIZ_XFRSIZ & g_dwc2_udc.out_ep[ep_idx].ep_mps);
  792. USB_OTG_OUTEP(ep_idx)->DOEPCTL |= (USB_OTG_DOEPCTL_CNAK | USB_OTG_DOEPCTL_EPENA);
  793. return 0;
  794. }
  795. if (ep_idx == 0) {
  796. if (data_len > g_dwc2_udc.out_ep[ep_idx].ep_mps) {
  797. data_len = g_dwc2_udc.out_ep[ep_idx].ep_mps;
  798. }
  799. g_dwc2_udc.out_ep[ep_idx].xfer_len = data_len;
  800. USB_OTG_OUTEP(ep_idx)->DOEPTSIZ |= (USB_OTG_DOEPTSIZ_PKTCNT & (1U << 19));
  801. USB_OTG_OUTEP(ep_idx)->DOEPTSIZ |= (USB_OTG_DOEPTSIZ_XFRSIZ & data_len);
  802. } else {
  803. pktcnt = (uint16_t)((data_len + g_dwc2_udc.out_ep[ep_idx].ep_mps - 1U) / g_dwc2_udc.out_ep[ep_idx].ep_mps);
  804. USB_OTG_OUTEP(ep_idx)->DOEPTSIZ |= (USB_OTG_DOEPTSIZ_PKTCNT & (pktcnt << 19));
  805. USB_OTG_OUTEP(ep_idx)->DOEPTSIZ |= (USB_OTG_DOEPTSIZ_XFRSIZ & data_len);
  806. }
  807. #ifdef CONFIG_USB_DWC2_DMA_ENABLE
  808. USB_OTG_OUTEP(ep_idx)->DOEPDMA = (uint32_t)data;
  809. #endif
  810. if (g_dwc2_udc.out_ep[ep_idx].ep_type == USB_ENDPOINT_TYPE_ISOCHRONOUS) {
  811. if ((USB_OTG_DEV->DSTS & (1U << 8)) == 0U) {
  812. USB_OTG_OUTEP(ep_idx)->DOEPCTL &= ~USB_OTG_DOEPCTL_SD0PID_SEVNFRM;
  813. USB_OTG_OUTEP(ep_idx)->DOEPCTL |= USB_OTG_DOEPCTL_SODDFRM;
  814. } else {
  815. USB_OTG_OUTEP(ep_idx)->DOEPCTL &= ~USB_OTG_DOEPCTL_SODDFRM;
  816. USB_OTG_OUTEP(ep_idx)->DOEPCTL |= USB_OTG_DOEPCTL_SD0PID_SEVNFRM;
  817. }
  818. }
  819. USB_OTG_OUTEP(ep_idx)->DOEPCTL |= (USB_OTG_DOEPCTL_CNAK | USB_OTG_DOEPCTL_EPENA);
  820. return 0;
  821. }
  822. void USBD_IRQHandler(uint8_t busid)
  823. {
  824. uint32_t gint_status, temp, ep_idx, ep_intr, epint, read_count, daintmask;
  825. gint_status = dwc2_get_glb_intstatus();
  826. if ((USB_OTG_GLB->GINTSTS & 0x1U) == USB_OTG_MODE_DEVICE) {
  827. /* Avoid spurious interrupt */
  828. if (gint_status == 0) {
  829. return;
  830. }
  831. #ifndef CONFIG_USB_DWC2_DMA_ENABLE
  832. /* Handle RxQLevel Interrupt */
  833. if (gint_status & USB_OTG_GINTSTS_RXFLVL) {
  834. USB_MASK_INTERRUPT(USB_OTG_GLB, USB_OTG_GINTSTS_RXFLVL);
  835. temp = USB_OTG_GLB->GRXSTSP;
  836. ep_idx = temp & USB_OTG_GRXSTSP_EPNUM;
  837. if (((temp & USB_OTG_GRXSTSP_PKTSTS) >> USB_OTG_GRXSTSP_PKTSTS_Pos) == STS_DATA_UPDT) {
  838. read_count = (temp & USB_OTG_GRXSTSP_BCNT) >> 4;
  839. if (read_count != 0) {
  840. dwc2_ep_read(g_dwc2_udc.out_ep[ep_idx].xfer_buf, read_count);
  841. g_dwc2_udc.out_ep[ep_idx].xfer_buf += read_count;
  842. }
  843. } else if (((temp & USB_OTG_GRXSTSP_PKTSTS) >> USB_OTG_GRXSTSP_PKTSTS_Pos) == STS_SETUP_UPDT) {
  844. read_count = (temp & USB_OTG_GRXSTSP_BCNT) >> 4;
  845. dwc2_ep_read((uint8_t *)&g_dwc2_udc.setup, read_count);
  846. } else {
  847. /* ... */
  848. }
  849. USB_UNMASK_INTERRUPT(USB_OTG_GLB, USB_OTG_GINTSTS_RXFLVL);
  850. }
  851. #endif
  852. if (gint_status & USB_OTG_GINTSTS_OEPINT) {
  853. ep_idx = 0;
  854. ep_intr = dwc2_get_outeps_intstatus();
  855. while (ep_intr != 0U) {
  856. if ((ep_intr & 0x1U) != 0U) {
  857. epint = dwc2_get_outep_intstatus(ep_idx);
  858. uint32_t DoepintReg = USB_OTG_OUTEP(ep_idx)->DOEPINT;
  859. USB_OTG_OUTEP(ep_idx)->DOEPINT = DoepintReg;
  860. if ((epint & USB_OTG_DOEPINT_XFRC) == USB_OTG_DOEPINT_XFRC) {
  861. if (ep_idx == 0) {
  862. if (g_dwc2_udc.out_ep[ep_idx].xfer_len == 0) {
  863. /* Out status, start reading setup */
  864. dwc2_ep0_start_read_setup((uint8_t *)&g_dwc2_udc.setup);
  865. } else {
  866. g_dwc2_udc.out_ep[ep_idx].actual_xfer_len = g_dwc2_udc.out_ep[ep_idx].xfer_len - ((USB_OTG_OUTEP(ep_idx)->DOEPTSIZ) & USB_OTG_DOEPTSIZ_XFRSIZ);
  867. g_dwc2_udc.out_ep[ep_idx].xfer_len = 0;
  868. usbd_event_ep_out_complete_handler(0, 0x00, g_dwc2_udc.out_ep[ep_idx].actual_xfer_len);
  869. }
  870. } else {
  871. g_dwc2_udc.out_ep[ep_idx].actual_xfer_len = g_dwc2_udc.out_ep[ep_idx].xfer_len - ((USB_OTG_OUTEP(ep_idx)->DOEPTSIZ) & USB_OTG_DOEPTSIZ_XFRSIZ);
  872. g_dwc2_udc.out_ep[ep_idx].xfer_len = 0;
  873. usbd_event_ep_out_complete_handler(0, ep_idx, g_dwc2_udc.out_ep[ep_idx].actual_xfer_len);
  874. }
  875. }
  876. if ((epint & USB_OTG_DOEPINT_STUP) == USB_OTG_DOEPINT_STUP) {
  877. usbd_event_ep0_setup_complete_handler(0, (uint8_t *)&g_dwc2_udc.setup);
  878. }
  879. }
  880. ep_intr >>= 1U;
  881. ep_idx++;
  882. }
  883. }
  884. if (gint_status & USB_OTG_GINTSTS_IEPINT) {
  885. ep_idx = 0U;
  886. ep_intr = dwc2_get_ineps_intstatus();
  887. while (ep_intr != 0U) {
  888. if ((ep_intr & 0x1U) != 0U) {
  889. epint = dwc2_get_inep_intstatus(ep_idx);
  890. uint32_t DiepintReg = USB_OTG_INEP(ep_idx)->DIEPINT;
  891. USB_OTG_INEP(ep_idx)->DIEPINT = DiepintReg;
  892. if ((epint & USB_OTG_DIEPINT_XFRC) == USB_OTG_DIEPINT_XFRC) {
  893. if (ep_idx == 0) {
  894. g_dwc2_udc.in_ep[ep_idx].actual_xfer_len = g_dwc2_udc.in_ep[ep_idx].xfer_len - ((USB_OTG_INEP(ep_idx)->DIEPTSIZ) & USB_OTG_DIEPTSIZ_XFRSIZ);
  895. g_dwc2_udc.in_ep[ep_idx].xfer_len = 0;
  896. usbd_event_ep_in_complete_handler(0, 0x80, g_dwc2_udc.in_ep[ep_idx].actual_xfer_len);
  897. if (g_dwc2_udc.setup.wLength && ((g_dwc2_udc.setup.bmRequestType & USB_REQUEST_DIR_MASK) == USB_REQUEST_DIR_OUT)) {
  898. /* In status, start reading setup */
  899. dwc2_ep0_start_read_setup((uint8_t *)&g_dwc2_udc.setup);
  900. } else if (g_dwc2_udc.setup.wLength == 0) {
  901. /* In status, start reading setup */
  902. dwc2_ep0_start_read_setup((uint8_t *)&g_dwc2_udc.setup);
  903. }
  904. } else {
  905. g_dwc2_udc.in_ep[ep_idx].actual_xfer_len = g_dwc2_udc.in_ep[ep_idx].xfer_len - ((USB_OTG_INEP(ep_idx)->DIEPTSIZ) & USB_OTG_DIEPTSIZ_XFRSIZ);
  906. g_dwc2_udc.in_ep[ep_idx].xfer_len = 0;
  907. usbd_event_ep_in_complete_handler(0, ep_idx | 0x80, g_dwc2_udc.in_ep[ep_idx].actual_xfer_len);
  908. }
  909. }
  910. if ((epint & USB_OTG_DIEPINT_TXFE) == USB_OTG_DIEPINT_TXFE) {
  911. dwc2_tx_fifo_empty_procecss(ep_idx);
  912. }
  913. }
  914. ep_intr >>= 1U;
  915. ep_idx++;
  916. }
  917. }
  918. if (gint_status & USB_OTG_GINTSTS_USBRST) {
  919. USB_OTG_GLB->GINTSTS |= USB_OTG_GINTSTS_USBRST;
  920. USB_OTG_DEV->DCTL &= ~USB_OTG_DCTL_RWUSIG;
  921. dwc2_flush_txfifo(0x10U);
  922. dwc2_flush_rxfifo();
  923. for (uint8_t i = 0U; i < CONFIG_USBDEV_EP_NUM; i++) {
  924. if (i == 0U) {
  925. USB_OTG_INEP(i)->DIEPCTL = USB_OTG_DIEPCTL_SNAK;
  926. USB_OTG_OUTEP(i)->DOEPCTL = USB_OTG_DOEPCTL_SNAK;
  927. } else {
  928. if (USB_OTG_INEP(i)->DIEPCTL & USB_OTG_DIEPCTL_EPENA) {
  929. USB_OTG_INEP(i)->DIEPCTL = (USB_OTG_DIEPCTL_EPDIS | USB_OTG_DIEPCTL_SNAK);
  930. } else {
  931. USB_OTG_INEP(i)->DIEPCTL = 0;
  932. }
  933. if (USB_OTG_OUTEP(i)->DOEPCTL & USB_OTG_DOEPCTL_EPENA) {
  934. USB_OTG_OUTEP(i)->DOEPCTL = (USB_OTG_DOEPCTL_EPDIS | USB_OTG_DOEPCTL_SNAK);
  935. } else {
  936. USB_OTG_OUTEP(i)->DOEPCTL = 0;
  937. }
  938. }
  939. USB_OTG_INEP(i)->DIEPTSIZ = 0U;
  940. USB_OTG_INEP(i)->DIEPINT = 0xFBFFU;
  941. USB_OTG_OUTEP(i)->DOEPTSIZ = 0U;
  942. USB_OTG_OUTEP(i)->DOEPINT = 0xFBFFU;
  943. }
  944. USB_OTG_DEV->DAINTMSK |= 0x10001U;
  945. USB_OTG_DEV->DOEPMSK = USB_OTG_DOEPMSK_STUPM |
  946. USB_OTG_DOEPMSK_XFRCM;
  947. USB_OTG_DEV->DIEPMSK = USB_OTG_DIEPMSK_XFRCM;
  948. memset(&g_dwc2_udc, 0, sizeof(struct dwc2_udc));
  949. usbd_event_reset_handler(0);
  950. /* Start reading setup */
  951. dwc2_ep0_start_read_setup((uint8_t *)&g_dwc2_udc.setup);
  952. }
  953. if (gint_status & USB_OTG_GINTSTS_ENUMDNE) {
  954. USB_OTG_GLB->GINTSTS |= USB_OTG_GINTSTS_ENUMDNE;
  955. dwc2_set_turnaroundtime(SystemCoreClock, dwc2_get_devspeed());
  956. USB_OTG_DEV->DCTL |= USB_OTG_DCTL_CGINAK;
  957. }
  958. if (gint_status & USB_OTG_GINTSTS_PXFR_INCOMPISOOUT) {
  959. daintmask = USB_OTG_DEV->DAINTMSK;
  960. daintmask >>= 16;
  961. for (ep_idx = 1; ep_idx < CONFIG_USBDEV_EP_NUM; ep_idx++) {
  962. if ((BIT(ep_idx) & ~daintmask) || (g_dwc2_udc.out_ep[ep_idx].ep_type != USB_ENDPOINT_TYPE_ISOCHRONOUS))
  963. continue;
  964. if (!(USB_OTG_OUTEP(ep_idx)->DOEPCTL & USB_OTG_DOEPCTL_USBAEP))
  965. continue;
  966. if ((USB_OTG_DEV->DSTS & (1U << 8)) != 0U) {
  967. USB_OTG_OUTEP(ep_idx)->DOEPCTL |= USB_OTG_DOEPCTL_SD0PID_SEVNFRM;
  968. USB_OTG_OUTEP(ep_idx)->DOEPCTL &= ~USB_OTG_DOEPCTL_SODDFRM;
  969. } else {
  970. USB_OTG_OUTEP(ep_idx)->DOEPCTL &= ~USB_OTG_DOEPCTL_SD0PID_SEVNFRM;
  971. USB_OTG_OUTEP(ep_idx)->DOEPCTL |= USB_OTG_DOEPCTL_SODDFRM;
  972. }
  973. }
  974. USB_OTG_GLB->GINTSTS |= USB_OTG_GINTSTS_PXFR_INCOMPISOOUT;
  975. }
  976. if (gint_status & USB_OTG_GINTSTS_IISOIXFR) {
  977. daintmask = USB_OTG_DEV->DAINTMSK;
  978. daintmask >>= 16;
  979. for (ep_idx = 1; ep_idx < CONFIG_USBDEV_EP_NUM; ep_idx++) {
  980. if (((BIT(ep_idx) & ~daintmask)) || (g_dwc2_udc.in_ep[ep_idx].ep_type != USB_ENDPOINT_TYPE_ISOCHRONOUS))
  981. continue;
  982. if (!(USB_OTG_INEP(ep_idx)->DIEPCTL & USB_OTG_DIEPCTL_USBAEP))
  983. continue;
  984. if ((USB_OTG_DEV->DSTS & (1U << 8)) != 0U) {
  985. USB_OTG_INEP(ep_idx)->DIEPCTL |= USB_OTG_DIEPCTL_SD0PID_SEVNFRM;
  986. USB_OTG_INEP(ep_idx)->DIEPCTL &= ~USB_OTG_DIEPCTL_SODDFRM;
  987. } else {
  988. USB_OTG_INEP(ep_idx)->DIEPCTL &= ~USB_OTG_DIEPCTL_SD0PID_SEVNFRM;
  989. USB_OTG_INEP(ep_idx)->DIEPCTL |= USB_OTG_DIEPCTL_SODDFRM;
  990. }
  991. }
  992. USB_OTG_GLB->GINTSTS |= USB_OTG_GINTSTS_IISOIXFR;
  993. }
  994. if (gint_status & USB_OTG_GINTSTS_SOF) {
  995. USB_OTG_GLB->GINTSTS |= USB_OTG_GINTSTS_SOF;
  996. }
  997. if (gint_status & USB_OTG_GINTSTS_USBSUSP) {
  998. USB_OTG_GLB->GINTSTS |= USB_OTG_GINTSTS_USBSUSP;
  999. }
  1000. if (gint_status & USB_OTG_GINTSTS_WKUINT) {
  1001. USB_OTG_GLB->GINTSTS |= USB_OTG_GINTSTS_WKUINT;
  1002. }
  1003. if (gint_status & USB_OTG_GINTSTS_OTGINT) {
  1004. temp = USB_OTG_GLB->GOTGINT;
  1005. if ((temp & USB_OTG_GOTGINT_SEDET) == USB_OTG_GOTGINT_SEDET) {
  1006. } else {
  1007. }
  1008. USB_OTG_GLB->GOTGINT |= temp;
  1009. }
  1010. }
  1011. }