usb_hc_musb.c 37 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139
  1. /*
  2. * Copyright (c) 2022, sakumisu
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include "usbh_core.h"
  7. #include "usbh_hub.h"
  8. #include "usb_musb_reg.h"
  9. #define HWREG(x) \
  10. (*((volatile uint32_t *)(x)))
  11. #define HWREGH(x) \
  12. (*((volatile uint16_t *)(x)))
  13. #define HWREGB(x) \
  14. (*((volatile uint8_t *)(x)))
  15. #define USB_BASE (bus->hcd.reg_base)
  16. #if defined(CONFIG_USB_MUSB_SUNXI)
  17. #define MUSB_FADDR_OFFSET 0x98
  18. #define MUSB_POWER_OFFSET 0x40
  19. #define MUSB_TXIS_OFFSET 0x44
  20. #define MUSB_RXIS_OFFSET 0x46
  21. #define MUSB_TXIE_OFFSET 0x48
  22. #define MUSB_RXIE_OFFSET 0x4A
  23. #define MUSB_IS_OFFSET 0x4C
  24. #define MUSB_IE_OFFSET 0x50
  25. #define MUSB_EPIDX_OFFSET 0x42
  26. #define MUSB_IND_TXMAP_OFFSET 0x80
  27. #define MUSB_IND_TXCSRL_OFFSET 0x82
  28. #define MUSB_IND_TXCSRH_OFFSET 0x83
  29. #define MUSB_IND_RXMAP_OFFSET 0x84
  30. #define MUSB_IND_RXCSRL_OFFSET 0x86
  31. #define MUSB_IND_RXCSRH_OFFSET 0x87
  32. #define MUSB_IND_RXCOUNT_OFFSET 0x88
  33. #define MUSB_IND_TXTYPE_OFFSET 0x8C
  34. #define MUSB_IND_TXINTERVAL_OFFSET 0x8D
  35. #define MUSB_IND_RXTYPE_OFFSET 0x8E
  36. #define MUSB_IND_RXINTERVAL_OFFSET 0x8F
  37. #define MUSB_FIFO_OFFSET 0x00
  38. #define MUSB_DEVCTL_OFFSET 0x41
  39. #define MUSB_TXFIFOSZ_OFFSET 0x90
  40. #define MUSB_RXFIFOSZ_OFFSET 0x94
  41. #define MUSB_TXFIFOADD_OFFSET 0x92
  42. #define MUSB_RXFIFOADD_OFFSET 0x96
  43. #define MUSB_TXFUNCADDR0_OFFSET 0x98
  44. #define MUSB_TXHUBADDR0_OFFSET 0x9A
  45. #define MUSB_TXHUBPORT0_OFFSET 0x9B
  46. #define MUSB_TXFUNCADDRx_OFFSET 0x98
  47. #define MUSB_TXHUBADDRx_OFFSET 0x9A
  48. #define MUSB_TXHUBPORTx_OFFSET 0x9B
  49. #define MUSB_RXFUNCADDRx_OFFSET 0x9C
  50. #define MUSB_RXHUBADDRx_OFFSET 0x9E
  51. #define MUSB_RXHUBPORTx_OFFSET 0x9F
  52. #define USB_TXMAP_BASE(ep_idx) (USB_BASE + MUSB_IND_TXMAP_OFFSET)
  53. #define USB_TXCSRL_BASE(ep_idx) (USB_BASE + MUSB_IND_TXCSRL_OFFSET)
  54. #define USB_TXCSRH_BASE(ep_idx) (USB_BASE + MUSB_IND_TXCSRH_OFFSET)
  55. #define USB_RXMAP_BASE(ep_idx) (USB_BASE + MUSB_IND_RXMAP_OFFSET)
  56. #define USB_RXCSRL_BASE(ep_idx) (USB_BASE + MUSB_IND_RXCSRL_OFFSET)
  57. #define USB_RXCSRH_BASE(ep_idx) (USB_BASE + MUSB_IND_RXCSRH_OFFSET)
  58. #define USB_RXCOUNT_BASE(ep_idx) (USB_BASE + MUSB_IND_RXCOUNT_OFFSET)
  59. #define USB_TXTYPE_BASE(ep_idx) (USB_BASE + MUSB_IND_TXTYPE_OFFSET)
  60. #define USB_TXINTERVAL_BASE(ep_idx) (USB_BASE + MUSB_IND_TXINTERVAL_OFFSET)
  61. #define USB_RXTYPE_BASE(ep_idx) (USB_BASE + MUSB_IND_RXTYPE_OFFSET)
  62. #define USB_RXINTERVAL_BASE(ep_idx) (USB_BASE + MUSB_IND_RXINTERVAL_OFFSET)
  63. #define USB_TXADDR_BASE(ep_idx) (USB_BASE + MUSB_TXFUNCADDRx_OFFSET)
  64. #define USB_TXHUBADDR_BASE(ep_idx) (USB_BASE + MUSB_TXHUBADDRx_OFFSET)
  65. #define USB_TXHUBPORT_BASE(ep_idx) (USB_BASE + MUSB_TXHUBPORTx_OFFSET)
  66. #define USB_RXADDR_BASE(ep_idx) (USB_BASE + MUSB_RXFUNCADDRx_OFFSET)
  67. #define USB_RXHUBADDR_BASE(ep_idx) (USB_BASE + MUSB_RXHUBADDRx_OFFSET)
  68. #define USB_RXHUBPORT_BASE(ep_idx) (USB_BASE + MUSB_RXHUBPORTx_OFFSET)
  69. #elif defined(CONFIG_USB_MUSB_CUSTOM)
  70. #include "musb_custom.h"
  71. #else
  72. #define MUSB_FADDR_OFFSET 0x00
  73. #define MUSB_POWER_OFFSET 0x01
  74. #define MUSB_TXIS_OFFSET 0x02
  75. #define MUSB_RXIS_OFFSET 0x04
  76. #define MUSB_TXIE_OFFSET 0x06
  77. #define MUSB_RXIE_OFFSET 0x08
  78. #define MUSB_IS_OFFSET 0x0A
  79. #define MUSB_IE_OFFSET 0x0B
  80. #define MUSB_EPIDX_OFFSET 0x0E
  81. #define MUSB_IND_TXMAP_OFFSET 0x10
  82. #define MUSB_IND_TXCSRL_OFFSET 0x12
  83. #define MUSB_IND_TXCSRH_OFFSET 0x13
  84. #define MUSB_IND_RXMAP_OFFSET 0x14
  85. #define MUSB_IND_RXCSRL_OFFSET 0x16
  86. #define MUSB_IND_RXCSRH_OFFSET 0x17
  87. #define MUSB_IND_RXCOUNT_OFFSET 0x18
  88. #define MUSB_IND_TXTYPE_OFFSET 0x1A
  89. #define MUSB_IND_TXINTERVAL_OFFSET 0x1B
  90. #define MUSB_IND_RXTYPE_OFFSET 0x1C
  91. #define MUSB_IND_RXINTERVAL_OFFSET 0x1D
  92. #define MUSB_FIFO_OFFSET 0x20
  93. #define MUSB_DEVCTL_OFFSET 0x60
  94. #define MUSB_TXFIFOSZ_OFFSET 0x62
  95. #define MUSB_RXFIFOSZ_OFFSET 0x63
  96. #define MUSB_TXFIFOADD_OFFSET 0x64
  97. #define MUSB_RXFIFOADD_OFFSET 0x66
  98. #define MUSB_TXFUNCADDR0_OFFSET 0x80
  99. #define MUSB_TXHUBADDR0_OFFSET 0x82
  100. #define MUSB_TXHUBPORT0_OFFSET 0x83
  101. #define MUSB_TXFUNCADDRx_OFFSET 0x88
  102. #define MUSB_TXHUBADDRx_OFFSET 0x8A
  103. #define MUSB_TXHUBPORTx_OFFSET 0x8B
  104. #define MUSB_RXFUNCADDRx_OFFSET 0x8C
  105. #define MUSB_RXHUBADDRx_OFFSET 0x8E
  106. #define MUSB_RXHUBPORTx_OFFSET 0x8F
  107. #define MUSB_TXMAP0_OFFSET 0x100
  108. // do not use EPIDX
  109. #define USB_TXMAP_BASE(ep_idx) (USB_BASE + MUSB_TXMAP0_OFFSET + 0x10 * ep_idx)
  110. #define USB_TXCSRL_BASE(ep_idx) (USB_BASE + MUSB_TXMAP0_OFFSET + 0x10 * ep_idx + 2)
  111. #define USB_TXCSRH_BASE(ep_idx) (USB_BASE + MUSB_TXMAP0_OFFSET + 0x10 * ep_idx + 3)
  112. #define USB_RXMAP_BASE(ep_idx) (USB_BASE + MUSB_TXMAP0_OFFSET + 0x10 * ep_idx + 4)
  113. #define USB_RXCSRL_BASE(ep_idx) (USB_BASE + MUSB_TXMAP0_OFFSET + 0x10 * ep_idx + 6)
  114. #define USB_RXCSRH_BASE(ep_idx) (USB_BASE + MUSB_TXMAP0_OFFSET + 0x10 * ep_idx + 7)
  115. #define USB_RXCOUNT_BASE(ep_idx) (USB_BASE + MUSB_TXMAP0_OFFSET + 0x10 * ep_idx + 8)
  116. #define USB_TXTYPE_BASE(ep_idx) (USB_BASE + MUSB_TXMAP0_OFFSET + 0x10 * ep_idx + 0x0A)
  117. #define USB_TXINTERVAL_BASE(ep_idx) (USB_BASE + MUSB_TXMAP0_OFFSET + 0x10 * ep_idx + 0x0B)
  118. #define USB_RXTYPE_BASE(ep_idx) (USB_BASE + MUSB_TXMAP0_OFFSET + 0x10 * ep_idx + 0x0C)
  119. #define USB_RXINTERVAL_BASE(ep_idx) (USB_BASE + MUSB_TXMAP0_OFFSET + 0x10 * ep_idx + 0x0D)
  120. #define USB_TXADDR_BASE(ep_idx) (USB_BASE + MUSB_TXFUNCADDR0_OFFSET + 0x8 * ep_idx)
  121. #define USB_TXHUBADDR_BASE(ep_idx) (USB_BASE + MUSB_TXFUNCADDR0_OFFSET + 0x8 * ep_idx + 2)
  122. #define USB_TXHUBPORT_BASE(ep_idx) (USB_BASE + MUSB_TXFUNCADDR0_OFFSET + 0x8 * ep_idx + 3)
  123. #define USB_RXADDR_BASE(ep_idx) (USB_BASE + MUSB_TXFUNCADDR0_OFFSET + 0x8 * ep_idx + 4)
  124. #define USB_RXHUBADDR_BASE(ep_idx) (USB_BASE + MUSB_TXFUNCADDR0_OFFSET + 0x8 * ep_idx + 6)
  125. #define USB_RXHUBPORT_BASE(ep_idx) (USB_BASE + MUSB_TXFUNCADDR0_OFFSET + 0x8 * ep_idx + 7)
  126. #endif
  127. #define USB_FIFO_BASE(ep_idx) (USB_BASE + MUSB_FIFO_OFFSET + 0x4 * ep_idx)
  128. typedef enum {
  129. USB_EP0_STATE_SETUP = 0x0, /**< SETUP DATA */
  130. USB_EP0_STATE_IN_DATA, /**< IN DATA */
  131. USB_EP0_STATE_IN_STATUS, /**< IN status*/
  132. USB_EP0_STATE_OUT_DATA, /**< OUT DATA */
  133. USB_EP0_STATE_OUT_STATUS, /**< OUT status */
  134. } ep0_state_t;
  135. struct musb_pipe {
  136. uint8_t chidx;
  137. bool inuse;
  138. uint32_t xfrd;
  139. volatile uint8_t ep0_state;
  140. usb_osal_sem_t waitsem;
  141. struct usbh_urb *urb;
  142. };
  143. struct musb_hcd {
  144. volatile bool port_csc;
  145. volatile bool port_pec;
  146. volatile bool port_pe;
  147. struct musb_pipe pipe_pool[CONFIG_USB_MUSB_PIPE_NUM];
  148. } g_musb_hcd[CONFIG_USBHOST_MAX_BUS];
  149. /* get current active ep */
  150. static uint8_t musb_get_active_ep(struct usbh_bus *bus)
  151. {
  152. return HWREGB(USB_BASE + MUSB_EPIDX_OFFSET);
  153. }
  154. /* set the active ep */
  155. static void musb_set_active_ep(struct usbh_bus *bus, uint8_t ep_index)
  156. {
  157. HWREGB(USB_BASE + MUSB_EPIDX_OFFSET) = ep_index;
  158. }
  159. static void musb_fifo_flush(struct usbh_bus *bus, uint8_t ep)
  160. {
  161. uint8_t ep_idx = ep & 0x7f;
  162. if (ep_idx == 0) {
  163. if ((HWREGB(USB_TXCSRL_BASE(ep_idx)) & (USB_CSRL0_RXRDY | USB_CSRL0_TXRDY)) != 0)
  164. HWREGB(USB_RXCSRL_BASE(ep_idx)) |= USB_CSRH0_FLUSH;
  165. } else {
  166. if (ep & 0x80) {
  167. if (HWREGB(USB_TXCSRL_BASE(ep_idx)) & USB_TXCSRL1_TXRDY)
  168. HWREGB(USB_TXCSRL_BASE(ep_idx)) |= USB_TXCSRL1_FLUSH;
  169. } else {
  170. if (HWREGB(USB_RXCSRL_BASE(ep_idx)) & USB_RXCSRL1_RXRDY)
  171. HWREGB(USB_RXCSRL_BASE(ep_idx)) |= USB_RXCSRL1_FLUSH;
  172. }
  173. }
  174. }
  175. static void musb_write_packet(struct usbh_bus *bus, uint8_t ep_idx, uint8_t *buffer, uint16_t len)
  176. {
  177. uint32_t *buf32;
  178. uint8_t *buf8;
  179. uint32_t count32;
  180. uint32_t count8;
  181. int i;
  182. if ((uint32_t)buffer & 0x03) {
  183. buf8 = buffer;
  184. for (i = 0; i < len; i++) {
  185. HWREGB(USB_FIFO_BASE(ep_idx)) = *buf8++;
  186. }
  187. } else {
  188. count32 = len >> 2;
  189. count8 = len & 0x03;
  190. buf32 = (uint32_t *)buffer;
  191. while (count32--) {
  192. HWREG(USB_FIFO_BASE(ep_idx)) = *buf32++;
  193. }
  194. buf8 = (uint8_t *)buf32;
  195. while (count8--) {
  196. HWREGB(USB_FIFO_BASE(ep_idx)) = *buf8++;
  197. }
  198. }
  199. }
  200. static void musb_read_packet(struct usbh_bus *bus, uint8_t ep_idx, uint8_t *buffer, uint16_t len)
  201. {
  202. uint32_t *buf32;
  203. uint8_t *buf8;
  204. uint32_t count32;
  205. uint32_t count8;
  206. int i;
  207. if ((uint32_t)buffer & 0x03) {
  208. buf8 = buffer;
  209. for (i = 0; i < len; i++) {
  210. *buf8++ = HWREGB(USB_FIFO_BASE(ep_idx));
  211. }
  212. } else {
  213. count32 = len >> 2;
  214. count8 = len & 0x03;
  215. buf32 = (uint32_t *)buffer;
  216. while (count32--) {
  217. *buf32++ = HWREG(USB_FIFO_BASE(ep_idx));
  218. }
  219. buf8 = (uint8_t *)buf32;
  220. while (count8--) {
  221. *buf8++ = HWREGB(USB_FIFO_BASE(ep_idx));
  222. }
  223. }
  224. }
  225. static uint32_t musb_get_fifo_size(uint16_t mps, uint16_t *used)
  226. {
  227. uint32_t size;
  228. for (uint8_t i = USB_TXFIFOSZ_SIZE_8; i <= USB_TXFIFOSZ_SIZE_2048; i++) {
  229. size = (8 << i);
  230. if (mps <= size) {
  231. *used = size;
  232. return i;
  233. }
  234. }
  235. *used = 0;
  236. return USB_TXFIFOSZ_SIZE_8;
  237. }
  238. static uint32_t usbh_musb_fifo_config(struct usbh_bus *bus, struct musb_fifo_cfg *cfg, uint32_t offset)
  239. {
  240. uint16_t fifo_used;
  241. uint8_t c_size;
  242. uint16_t c_off;
  243. c_off = offset >> 3;
  244. c_size = musb_get_fifo_size(cfg->maxpacket, &fifo_used);
  245. musb_set_active_ep(bus, cfg->ep_num);
  246. switch (cfg->style) {
  247. case FIFO_TX:
  248. HWREGB(USB_BASE + MUSB_TXFIFOSZ_OFFSET) = c_size & 0x0f;
  249. HWREGH(USB_BASE + MUSB_TXFIFOADD_OFFSET) = c_off;
  250. break;
  251. case FIFO_RX:
  252. HWREGB(USB_BASE + MUSB_RXFIFOSZ_OFFSET) = c_size & 0x0f;
  253. HWREGH(USB_BASE + MUSB_RXFIFOADD_OFFSET) = c_off;
  254. break;
  255. case FIFO_TXRX:
  256. HWREGB(USB_BASE + MUSB_TXFIFOSZ_OFFSET) = c_size & 0x0f;
  257. HWREGH(USB_BASE + MUSB_TXFIFOADD_OFFSET) = c_off;
  258. HWREGB(USB_BASE + MUSB_RXFIFOSZ_OFFSET) = c_size & 0x0f;
  259. HWREGH(USB_BASE + MUSB_RXFIFOADD_OFFSET) = c_off;
  260. break;
  261. default:
  262. break;
  263. }
  264. return (offset + fifo_used);
  265. }
  266. void musb_control_urb_init(struct usbh_bus *bus, uint8_t chidx, struct usbh_urb *urb, struct usb_setup_packet *setup, uint8_t *buffer, uint32_t buflen)
  267. {
  268. uint8_t old_ep_index;
  269. uint8_t speed = USB_TXTYPE1_SPEED_FULL;
  270. old_ep_index = musb_get_active_ep(bus);
  271. musb_set_active_ep(bus, chidx);
  272. if (urb->hport->speed == USB_SPEED_HIGH) {
  273. speed = USB_TYPE0_SPEED_HIGH;
  274. } else if (urb->hport->speed == USB_SPEED_FULL) {
  275. speed = USB_TYPE0_SPEED_FULL;
  276. } else if (urb->hport->speed == USB_SPEED_LOW) {
  277. speed = USB_TYPE0_SPEED_LOW;
  278. }
  279. HWREGB(USB_TXADDR_BASE(chidx)) = urb->hport->dev_addr;
  280. HWREGB(USB_TXTYPE_BASE(chidx)) = speed;
  281. HWREGB(USB_TXHUBADDR_BASE(chidx)) = 0;
  282. HWREGB(USB_TXHUBPORT_BASE(chidx)) = 0;
  283. musb_write_packet(bus, chidx, (uint8_t *)setup, 8);
  284. HWREGB(USB_TXCSRL_BASE(chidx)) = USB_CSRL0_TXRDY | USB_CSRL0_SETUP;
  285. musb_set_active_ep(bus, old_ep_index);
  286. }
  287. int musb_bulk_urb_init(struct usbh_bus *bus, uint8_t chidx, struct usbh_urb *urb, uint8_t *buffer, uint32_t buflen)
  288. {
  289. uint8_t old_ep_index;
  290. uint8_t speed = USB_TXTYPE1_SPEED_FULL;
  291. old_ep_index = musb_get_active_ep(bus);
  292. musb_set_active_ep(bus, chidx);
  293. if (urb->hport->speed == USB_SPEED_HIGH) {
  294. speed = USB_TXTYPE1_SPEED_HIGH;
  295. } else if (urb->hport->speed == USB_SPEED_FULL) {
  296. speed = USB_TXTYPE1_SPEED_FULL;
  297. } else if (urb->hport->speed == USB_SPEED_LOW) {
  298. speed = USB_TXTYPE1_SPEED_LOW;
  299. }
  300. if (urb->ep->bEndpointAddress & 0x80) {
  301. if ((8 << HWREGB(USB_BASE + MUSB_RXFIFOSZ_OFFSET)) < USB_GET_MAXPACKETSIZE(urb->ep->wMaxPacketSize)) {
  302. USB_LOG_ERR("Ep %02x fifo is overflow\r\n", urb->ep->bEndpointAddress);
  303. return -USB_ERR_RANGE;
  304. }
  305. HWREGB(USB_RXADDR_BASE(chidx)) = urb->hport->dev_addr;
  306. HWREGB(USB_RXTYPE_BASE(chidx)) = (urb->ep->bEndpointAddress & 0x0f) | speed | USB_TXTYPE1_PROTO_BULK;
  307. HWREGH(USB_RXMAP_BASE(chidx)) = USB_GET_MAXPACKETSIZE(urb->ep->wMaxPacketSize);
  308. HWREGB(USB_RXINTERVAL_BASE(chidx)) = 0;
  309. HWREGB(USB_RXHUBADDR_BASE(chidx)) = 0;
  310. HWREGB(USB_RXHUBPORT_BASE(chidx)) = 0;
  311. HWREGB(USB_TXCSRH_BASE(chidx)) &= ~USB_TXCSRH1_MODE;
  312. HWREGB(USB_RXCSRL_BASE(chidx)) = USB_RXCSRL1_REQPKT;
  313. HWREGH(USB_BASE + MUSB_RXIE_OFFSET) |= (1 << chidx);
  314. } else {
  315. if ((8 << HWREGB(USB_BASE + MUSB_TXFIFOSZ_OFFSET)) < USB_GET_MAXPACKETSIZE(urb->ep->wMaxPacketSize)) {
  316. USB_LOG_ERR("Ep %02x fifo is overflow\r\n", urb->ep->bEndpointAddress);
  317. return -USB_ERR_RANGE;
  318. }
  319. HWREGB(USB_TXADDR_BASE(chidx)) = urb->hport->dev_addr;
  320. HWREGB(USB_TXTYPE_BASE(chidx)) = (urb->ep->bEndpointAddress & 0x0f) | speed | USB_TXTYPE1_PROTO_BULK;
  321. HWREGH(USB_TXMAP_BASE(chidx)) = USB_GET_MAXPACKETSIZE(urb->ep->wMaxPacketSize);
  322. HWREGB(USB_TXINTERVAL_BASE(chidx)) = 0;
  323. HWREGB(USB_TXHUBADDR_BASE(chidx)) = 0;
  324. HWREGB(USB_TXHUBPORT_BASE(chidx)) = 0;
  325. if (buflen > USB_GET_MAXPACKETSIZE(urb->ep->wMaxPacketSize)) {
  326. buflen = USB_GET_MAXPACKETSIZE(urb->ep->wMaxPacketSize);
  327. }
  328. musb_write_packet(bus, chidx, buffer, buflen);
  329. HWREGB(USB_TXCSRH_BASE(chidx)) |= USB_TXCSRH1_MODE;
  330. HWREGB(USB_TXCSRL_BASE(chidx)) = USB_TXCSRL1_TXRDY;
  331. HWREGH(USB_BASE + MUSB_TXIE_OFFSET) |= (1 << chidx);
  332. }
  333. musb_set_active_ep(bus, old_ep_index);
  334. return 0;
  335. }
  336. int musb_intr_urb_init(struct usbh_bus *bus, uint8_t chidx, struct usbh_urb *urb, uint8_t *buffer, uint32_t buflen)
  337. {
  338. uint8_t old_ep_index;
  339. uint8_t speed = USB_TXTYPE1_SPEED_FULL;
  340. old_ep_index = musb_get_active_ep(bus);
  341. musb_set_active_ep(bus, chidx);
  342. if (urb->hport->speed == USB_SPEED_HIGH) {
  343. speed = USB_TXTYPE1_SPEED_HIGH;
  344. } else if (urb->hport->speed == USB_SPEED_FULL) {
  345. speed = USB_TXTYPE1_SPEED_FULL;
  346. } else if (urb->hport->speed == USB_SPEED_LOW) {
  347. speed = USB_TXTYPE1_SPEED_LOW;
  348. }
  349. if (urb->ep->bEndpointAddress & 0x80) {
  350. if ((8 << HWREGB(USB_BASE + MUSB_RXFIFOSZ_OFFSET)) < USB_GET_MAXPACKETSIZE(urb->ep->wMaxPacketSize)) {
  351. USB_LOG_ERR("Ep %02x fifo is overflow\r\n", urb->ep->bEndpointAddress);
  352. return -USB_ERR_RANGE;
  353. }
  354. HWREGB(USB_RXADDR_BASE(chidx)) = urb->hport->dev_addr;
  355. HWREGB(USB_RXTYPE_BASE(chidx)) = (urb->ep->bEndpointAddress & 0x0f) | speed | USB_TXTYPE1_PROTO_INT;
  356. HWREGH(USB_RXMAP_BASE(chidx)) = USB_GET_MAXPACKETSIZE(urb->ep->wMaxPacketSize);
  357. HWREGB(USB_RXINTERVAL_BASE(chidx)) = urb->ep->bInterval;
  358. HWREGB(USB_RXHUBADDR_BASE(chidx)) = 0;
  359. HWREGB(USB_RXHUBPORT_BASE(chidx)) = 0;
  360. HWREGB(USB_TXCSRH_BASE(chidx)) &= ~USB_TXCSRH1_MODE;
  361. HWREGB(USB_RXCSRL_BASE(chidx)) = USB_RXCSRL1_REQPKT;
  362. HWREGH(USB_BASE + MUSB_RXIE_OFFSET) |= (1 << chidx);
  363. } else {
  364. if ((8 << HWREGB(USB_BASE + MUSB_TXFIFOSZ_OFFSET)) < USB_GET_MAXPACKETSIZE(urb->ep->wMaxPacketSize)) {
  365. USB_LOG_ERR("Ep %02x fifo is overflow\r\n", urb->ep->bEndpointAddress);
  366. return -USB_ERR_RANGE;
  367. }
  368. HWREGB(USB_TXADDR_BASE(chidx)) = urb->hport->dev_addr;
  369. HWREGB(USB_TXTYPE_BASE(chidx)) = (urb->ep->bEndpointAddress & 0x0f) | speed | USB_TXTYPE1_PROTO_INT;
  370. HWREGH(USB_TXMAP_BASE(chidx)) = USB_GET_MAXPACKETSIZE(urb->ep->wMaxPacketSize);
  371. HWREGB(USB_TXINTERVAL_BASE(chidx)) = urb->ep->bInterval;
  372. HWREGB(USB_TXHUBADDR_BASE(chidx)) = 0;
  373. HWREGB(USB_TXHUBPORT_BASE(chidx)) = 0;
  374. if (buflen > USB_GET_MAXPACKETSIZE(urb->ep->wMaxPacketSize)) {
  375. buflen = USB_GET_MAXPACKETSIZE(urb->ep->wMaxPacketSize);
  376. }
  377. musb_write_packet(bus, chidx, buffer, buflen);
  378. HWREGB(USB_TXCSRH_BASE(chidx)) |= USB_TXCSRH1_MODE;
  379. HWREGB(USB_TXCSRL_BASE(chidx)) = USB_TXCSRL1_TXRDY;
  380. HWREGH(USB_BASE + MUSB_TXIE_OFFSET) |= (1 << chidx);
  381. }
  382. musb_set_active_ep(bus, old_ep_index);
  383. return 0;
  384. }
  385. static int usbh_reset_port(struct usbh_bus *bus, const uint8_t port)
  386. {
  387. g_musb_hcd[bus->hcd.hcd_id].port_pe = 0;
  388. HWREGB(USB_BASE + MUSB_POWER_OFFSET) |= USB_POWER_RESET;
  389. #ifdef CONFIG_USB_MUSB_SIFLI
  390. extern void musb_reset_prev(void);
  391. musb_reset_prev();
  392. #endif
  393. usb_osal_msleep(20);
  394. HWREGB(USB_BASE + MUSB_POWER_OFFSET) &= ~(USB_POWER_RESET);
  395. usb_osal_msleep(20);
  396. #ifdef CONFIG_USB_MUSB_SIFLI
  397. extern void musb_reset_post(void);
  398. musb_reset_post();
  399. #endif
  400. g_musb_hcd[bus->hcd.hcd_id].port_pe = 1;
  401. return 0;
  402. }
  403. static uint8_t usbh_get_port_speed(struct usbh_bus *bus, const uint8_t port)
  404. {
  405. uint8_t speed = USB_SPEED_UNKNOWN;
  406. if (HWREGB(USB_BASE + MUSB_POWER_OFFSET) & USB_POWER_HSMODE)
  407. speed = USB_SPEED_HIGH;
  408. else if (HWREGB(USB_BASE + MUSB_DEVCTL_OFFSET) & USB_DEVCTL_FSDEV)
  409. speed = USB_SPEED_FULL;
  410. else if (HWREGB(USB_BASE + MUSB_DEVCTL_OFFSET) & USB_DEVCTL_LSDEV)
  411. speed = USB_SPEED_LOW;
  412. return speed;
  413. }
  414. #if 0
  415. static int musb_pipe_alloc(void)
  416. {
  417. int chidx;
  418. for (chidx = 1; chidx < CONFIG_USB_MUSB_PIPE_NUM; chidx++) {
  419. if (!g_musb_hcd[bus->hcd.hcd_id].pipe_pool[chidx].inuse) {
  420. g_musb_hcd[bus->hcd.hcd_id].pipe_pool[chidx].inuse = true;
  421. return chidx;
  422. }
  423. }
  424. return -1;
  425. }
  426. #endif
  427. static void musb_pipe_free(struct musb_pipe *pipe)
  428. {
  429. if (pipe->urb) {
  430. pipe->urb->hcpriv = NULL;
  431. pipe->urb = NULL;
  432. }
  433. #if 0
  434. pipe->inuse = false;
  435. #endif
  436. }
  437. __WEAK void usb_hc_low_level_init(struct usbh_bus *bus)
  438. {
  439. (void)bus;
  440. }
  441. __WEAK void usb_hc_low_level_deinit(struct usbh_bus *bus)
  442. {
  443. (void)bus;
  444. }
  445. int usb_hc_init(struct usbh_bus *bus)
  446. {
  447. uint8_t regval;
  448. uint16_t offset = 0;
  449. uint8_t cfg_num;
  450. struct musb_fifo_cfg *cfg;
  451. usb_hc_low_level_init(bus);
  452. memset(&g_musb_hcd[bus->hcd.hcd_id], 0, sizeof(struct musb_hcd));
  453. for (uint8_t i = 0; i < CONFIG_USB_MUSB_PIPE_NUM; i++) {
  454. g_musb_hcd[bus->hcd.hcd_id].pipe_pool[i].waitsem = usb_osal_sem_create(0);
  455. }
  456. cfg_num = usbh_get_musb_fifo_cfg(&cfg);
  457. for (uint8_t i = 0; i < cfg_num; i++) {
  458. offset = usbh_musb_fifo_config(bus, &cfg[i], offset);
  459. }
  460. USB_ASSERT_MSG(offset <= usb_get_musb_ram_size(), "Your fifo config is overflow, please check");
  461. /* Enable USB interrupts */
  462. regval = USB_IE_RESET | USB_IE_CONN | USB_IE_DISCON |
  463. USB_IE_RESUME | USB_IE_SUSPND |
  464. USB_IE_BABBLE | USB_IE_SESREQ | USB_IE_VBUSERR;
  465. HWREGB(USB_BASE + MUSB_IE_OFFSET) = regval;
  466. HWREGH(USB_BASE + MUSB_TXIE_OFFSET) = USB_TXIE_EP0;
  467. HWREGH(USB_BASE + MUSB_RXIE_OFFSET) = 0;
  468. HWREGB(USB_BASE + MUSB_POWER_OFFSET) |= USB_POWER_HSENAB;
  469. HWREGB(USB_BASE + MUSB_DEVCTL_OFFSET) |= USB_DEVCTL_SESSION;
  470. #ifdef CONFIG_USB_MUSB_SUNXI
  471. musb_set_active_ep(bus, 0);
  472. HWREGB(USB_TXCSRL_BASE(0)) = USB_CSRL0_TXRDY;
  473. #endif
  474. return 0;
  475. }
  476. int usb_hc_deinit(struct usbh_bus *bus)
  477. {
  478. HWREGB(USB_BASE + MUSB_IE_OFFSET) = 0;
  479. HWREGH(USB_BASE + MUSB_TXIE_OFFSET) = 0;
  480. HWREGH(USB_BASE + MUSB_RXIE_OFFSET) = 0;
  481. HWREGB(USB_BASE + MUSB_POWER_OFFSET) &= ~USB_POWER_HSENAB;
  482. HWREGB(USB_BASE + MUSB_DEVCTL_OFFSET) &= ~USB_DEVCTL_SESSION;
  483. for (uint8_t i = 0; i < CONFIG_USB_MUSB_PIPE_NUM; i++) {
  484. usb_osal_sem_delete(g_musb_hcd[bus->hcd.hcd_id].pipe_pool[i].waitsem);
  485. }
  486. usb_hc_low_level_deinit(bus);
  487. return 0;
  488. }
  489. int usbh_roothub_control(struct usbh_bus *bus, struct usb_setup_packet *setup, uint8_t *buf)
  490. {
  491. uint8_t nports;
  492. uint8_t port;
  493. uint32_t status;
  494. nports = CONFIG_USBHOST_MAX_RHPORTS;
  495. port = setup->wIndex;
  496. if (setup->bmRequestType & USB_REQUEST_RECIPIENT_DEVICE) {
  497. switch (setup->bRequest) {
  498. case HUB_REQUEST_CLEAR_FEATURE:
  499. switch (setup->wValue) {
  500. case HUB_FEATURE_HUB_C_LOCALPOWER:
  501. break;
  502. case HUB_FEATURE_HUB_C_OVERCURRENT:
  503. break;
  504. default:
  505. return -USB_ERR_INVAL;
  506. }
  507. break;
  508. case HUB_REQUEST_SET_FEATURE:
  509. switch (setup->wValue) {
  510. case HUB_FEATURE_HUB_C_LOCALPOWER:
  511. break;
  512. case HUB_FEATURE_HUB_C_OVERCURRENT:
  513. break;
  514. default:
  515. return -USB_ERR_INVAL;
  516. }
  517. break;
  518. case HUB_REQUEST_GET_DESCRIPTOR:
  519. break;
  520. case HUB_REQUEST_GET_STATUS:
  521. memset(buf, 0, 4);
  522. break;
  523. default:
  524. break;
  525. }
  526. } else if (setup->bmRequestType & USB_REQUEST_RECIPIENT_OTHER) {
  527. switch (setup->bRequest) {
  528. case HUB_REQUEST_CLEAR_FEATURE:
  529. if (!port || port > nports) {
  530. return -USB_ERR_INVAL;
  531. }
  532. switch (setup->wValue) {
  533. case HUB_PORT_FEATURE_ENABLE:
  534. break;
  535. case HUB_PORT_FEATURE_SUSPEND:
  536. case HUB_PORT_FEATURE_C_SUSPEND:
  537. break;
  538. case HUB_PORT_FEATURE_POWER:
  539. break;
  540. case HUB_PORT_FEATURE_C_CONNECTION:
  541. g_musb_hcd[bus->hcd.hcd_id].port_csc = 0;
  542. break;
  543. case HUB_PORT_FEATURE_C_ENABLE:
  544. g_musb_hcd[bus->hcd.hcd_id].port_pec = 0;
  545. break;
  546. case HUB_PORT_FEATURE_C_OVER_CURREN:
  547. break;
  548. case HUB_PORT_FEATURE_C_RESET:
  549. break;
  550. default:
  551. return -USB_ERR_INVAL;
  552. }
  553. break;
  554. case HUB_REQUEST_SET_FEATURE:
  555. if (!port || port > nports) {
  556. return -USB_ERR_INVAL;
  557. }
  558. switch (setup->wValue) {
  559. case HUB_PORT_FEATURE_SUSPEND:
  560. break;
  561. case HUB_PORT_FEATURE_POWER:
  562. break;
  563. case HUB_PORT_FEATURE_RESET:
  564. usbh_reset_port(bus, port);
  565. break;
  566. default:
  567. return -USB_ERR_INVAL;
  568. }
  569. break;
  570. case HUB_REQUEST_GET_STATUS:
  571. if (!port || port > nports) {
  572. return -USB_ERR_INVAL;
  573. }
  574. status = 0;
  575. if (g_musb_hcd[bus->hcd.hcd_id].port_csc) {
  576. status |= (1 << HUB_PORT_FEATURE_C_CONNECTION);
  577. }
  578. if (g_musb_hcd[bus->hcd.hcd_id].port_pec) {
  579. status |= (1 << HUB_PORT_FEATURE_C_ENABLE);
  580. }
  581. if (g_musb_hcd[bus->hcd.hcd_id].port_pe) {
  582. status |= (1 << HUB_PORT_FEATURE_CONNECTION);
  583. status |= (1 << HUB_PORT_FEATURE_ENABLE);
  584. if (usbh_get_port_speed(bus, port) == USB_SPEED_LOW) {
  585. status |= (1 << HUB_PORT_FEATURE_LOWSPEED);
  586. } else if (usbh_get_port_speed(bus, port) == USB_SPEED_HIGH) {
  587. status |= (1 << HUB_PORT_FEATURE_HIGHSPEED);
  588. }
  589. }
  590. status |= (1 << HUB_PORT_FEATURE_POWER);
  591. memcpy(buf, &status, 4);
  592. break;
  593. default:
  594. break;
  595. }
  596. }
  597. return 0;
  598. }
  599. int usbh_submit_urb(struct usbh_urb *urb)
  600. {
  601. struct musb_pipe *pipe;
  602. struct usbh_bus *bus;
  603. int chidx;
  604. size_t flags;
  605. int ret = 0;
  606. if (!urb || !urb->hport || !urb->ep || !urb->hport->bus) {
  607. return -USB_ERR_INVAL;
  608. }
  609. if (!urb->hport->connected) {
  610. return -USB_ERR_NOTCONN;
  611. }
  612. if (urb->errorcode == -USB_ERR_BUSY) {
  613. return -USB_ERR_BUSY;
  614. }
  615. bus = urb->hport->bus;
  616. if (USB_GET_ENDPOINT_TYPE(urb->ep->bmAttributes) == USB_ENDPOINT_TYPE_CONTROL) {
  617. chidx = 0;
  618. } else {
  619. chidx = (urb->ep->bEndpointAddress & 0x0f);
  620. if (chidx > (CONFIG_USB_MUSB_PIPE_NUM - 1)) {
  621. return -USB_ERR_RANGE;
  622. }
  623. }
  624. flags = usb_osal_enter_critical_section();
  625. pipe = &g_musb_hcd[bus->hcd.hcd_id].pipe_pool[chidx];
  626. pipe->chidx = chidx;
  627. pipe->urb = urb;
  628. urb->hcpriv = pipe;
  629. urb->errorcode = -USB_ERR_BUSY;
  630. urb->actual_length = 0;
  631. switch (USB_GET_ENDPOINT_TYPE(urb->ep->bmAttributes)) {
  632. case USB_ENDPOINT_TYPE_CONTROL:
  633. pipe->ep0_state = USB_EP0_STATE_SETUP;
  634. musb_control_urb_init(bus, 0, urb, urb->setup, urb->transfer_buffer, urb->transfer_buffer_length);
  635. break;
  636. case USB_ENDPOINT_TYPE_BULK:
  637. ret = musb_bulk_urb_init(bus, chidx, urb, urb->transfer_buffer, urb->transfer_buffer_length);
  638. if (ret < 0) {
  639. usb_osal_leave_critical_section(flags);
  640. return ret;
  641. }
  642. break;
  643. case USB_ENDPOINT_TYPE_INTERRUPT:
  644. ret = musb_intr_urb_init(bus, chidx, urb, urb->transfer_buffer, urb->transfer_buffer_length);
  645. if (ret < 0) {
  646. usb_osal_leave_critical_section(flags);
  647. return ret;
  648. }
  649. break;
  650. case USB_ENDPOINT_TYPE_ISOCHRONOUS:
  651. return -USB_ERR_NOTSUPP;
  652. default:
  653. break;
  654. }
  655. usb_osal_leave_critical_section(flags);
  656. if (urb->timeout > 0) {
  657. /* wait until timeout or sem give */
  658. ret = usb_osal_sem_take(pipe->waitsem, urb->timeout);
  659. if (ret < 0) {
  660. goto errout_timeout;
  661. }
  662. urb->timeout = 0;
  663. ret = urb->errorcode;
  664. /* we can free pipe when waitsem is done */
  665. musb_pipe_free(pipe);
  666. }
  667. return ret;
  668. errout_timeout:
  669. urb->timeout = 0;
  670. usbh_kill_urb(urb);
  671. return ret;
  672. }
  673. int usbh_kill_urb(struct usbh_urb *urb)
  674. {
  675. struct musb_pipe *pipe;
  676. struct usbh_bus *bus;
  677. size_t flags;
  678. if (!urb || !urb->hcpriv || !urb->hport->bus) {
  679. return -USB_ERR_INVAL;
  680. }
  681. bus = urb->hport->bus;
  682. ARG_UNUSED(bus);
  683. flags = usb_osal_enter_critical_section();
  684. pipe = (struct musb_pipe *)urb->hcpriv;
  685. urb->errorcode = -USB_ERR_SHUTDOWN;
  686. if (urb->ep->bEndpointAddress & 0x80) {
  687. HWREGH(USB_BASE + MUSB_RXIE_OFFSET) &= ~(1 << (urb->ep->bEndpointAddress & 0x0f));
  688. HWREGH(USB_BASE + MUSB_RXIS_OFFSET) = (1 << (urb->ep->bEndpointAddress & 0x0f));
  689. } else {
  690. HWREGH(USB_BASE + MUSB_TXIE_OFFSET) &= ~(1 << (urb->ep->bEndpointAddress & 0x0f));
  691. HWREGH(USB_BASE + MUSB_TXIS_OFFSET) = (1 << (urb->ep->bEndpointAddress & 0x0f));
  692. }
  693. musb_fifo_flush(bus, urb->ep->bEndpointAddress);
  694. if (urb->timeout) {
  695. usb_osal_sem_give(pipe->waitsem);
  696. } else {
  697. musb_pipe_free(pipe);
  698. }
  699. if (urb->complete) {
  700. urb->complete(urb->arg, urb->errorcode);
  701. }
  702. usb_osal_leave_critical_section(flags);
  703. return 0;
  704. }
  705. static void musb_urb_waitup(struct usbh_urb *urb)
  706. {
  707. struct musb_pipe *pipe;
  708. pipe = (struct musb_pipe *)urb->hcpriv;
  709. pipe->urb = NULL;
  710. urb->hcpriv = NULL;
  711. if (urb->timeout) {
  712. usb_osal_sem_give(pipe->waitsem);
  713. } else {
  714. musb_pipe_free(pipe);
  715. }
  716. if (urb->complete) {
  717. if (urb->errorcode < 0) {
  718. urb->complete(urb->arg, urb->errorcode);
  719. } else {
  720. urb->complete(urb->arg, urb->actual_length);
  721. }
  722. }
  723. }
  724. void handle_ep0(struct usbh_bus *bus)
  725. {
  726. uint8_t ep_idx = 0;
  727. uint8_t ep0_status;
  728. struct musb_pipe *pipe;
  729. struct usbh_urb *urb;
  730. uint32_t size;
  731. pipe = (struct musb_pipe *)&g_musb_hcd[bus->hcd.hcd_id].pipe_pool[0];
  732. urb = pipe->urb;
  733. if (urb == NULL) {
  734. return;
  735. }
  736. musb_set_active_ep(bus, 0);
  737. ep0_status = HWREGB(USB_TXCSRL_BASE(ep_idx));
  738. if (ep0_status & USB_CSRL0_STALLED) {
  739. HWREGB(USB_TXCSRL_BASE(ep_idx)) &= ~USB_CSRL0_STALLED;
  740. pipe->ep0_state = USB_EP0_STATE_SETUP;
  741. urb->errorcode = -USB_ERR_STALL;
  742. musb_urb_waitup(urb);
  743. return;
  744. }
  745. if (ep0_status & USB_CSRL0_ERROR) {
  746. HWREGB(USB_TXCSRL_BASE(ep_idx)) &= ~USB_CSRL0_ERROR;
  747. musb_fifo_flush(bus, 0);
  748. pipe->ep0_state = USB_EP0_STATE_SETUP;
  749. urb->errorcode = -USB_ERR_IO;
  750. musb_urb_waitup(urb);
  751. return;
  752. }
  753. if (ep0_status & USB_CSRL0_STALL) {
  754. HWREGB(USB_TXCSRL_BASE(ep_idx)) &= ~USB_CSRL0_STALL;
  755. pipe->ep0_state = USB_EP0_STATE_SETUP;
  756. urb->errorcode = -USB_ERR_STALL;
  757. musb_urb_waitup(urb);
  758. return;
  759. }
  760. switch (pipe->ep0_state) {
  761. case USB_EP0_STATE_SETUP:
  762. urb->actual_length += 8;
  763. if (urb->transfer_buffer_length) {
  764. if (urb->setup->bmRequestType & 0x80) {
  765. pipe->ep0_state = USB_EP0_STATE_IN_DATA;
  766. HWREGB(USB_TXCSRL_BASE(ep_idx)) = USB_CSRL0_REQPKT;
  767. } else {
  768. pipe->ep0_state = USB_EP0_STATE_OUT_DATA;
  769. size = urb->transfer_buffer_length;
  770. if (size > USB_GET_MAXPACKETSIZE(urb->ep->wMaxPacketSize)) {
  771. size = USB_GET_MAXPACKETSIZE(urb->ep->wMaxPacketSize);
  772. }
  773. musb_write_packet(bus, 0, urb->transfer_buffer, size);
  774. HWREGB(USB_TXCSRL_BASE(ep_idx)) = USB_CSRL0_TXRDY;
  775. urb->transfer_buffer += size;
  776. urb->transfer_buffer_length -= size;
  777. urb->actual_length += size;
  778. }
  779. } else {
  780. pipe->ep0_state = USB_EP0_STATE_IN_STATUS;
  781. HWREGB(USB_TXCSRL_BASE(ep_idx)) = (USB_CSRL0_REQPKT | USB_CSRL0_STATUS);
  782. }
  783. break;
  784. case USB_EP0_STATE_IN_DATA:
  785. if (ep0_status & USB_CSRL0_RXRDY) {
  786. size = HWREGH(USB_RXCOUNT_BASE(ep_idx));
  787. musb_read_packet(bus, 0, urb->transfer_buffer, size);
  788. HWREGB(USB_TXCSRL_BASE(ep_idx)) &= ~USB_CSRL0_RXRDY;
  789. urb->transfer_buffer += size;
  790. urb->transfer_buffer_length -= size;
  791. urb->actual_length += size;
  792. if ((size < USB_GET_MAXPACKETSIZE(urb->ep->wMaxPacketSize)) || (urb->transfer_buffer_length == 0)) {
  793. pipe->ep0_state = USB_EP0_STATE_OUT_STATUS;
  794. HWREGB(USB_TXCSRL_BASE(ep_idx)) = (USB_CSRL0_TXRDY | USB_CSRL0_STATUS);
  795. } else {
  796. HWREGB(USB_TXCSRL_BASE(ep_idx)) = USB_CSRL0_REQPKT;
  797. }
  798. }
  799. break;
  800. case USB_EP0_STATE_OUT_DATA:
  801. if (urb->transfer_buffer_length > 0) {
  802. size = urb->transfer_buffer_length;
  803. if (size > USB_GET_MAXPACKETSIZE(urb->ep->wMaxPacketSize)) {
  804. size = USB_GET_MAXPACKETSIZE(urb->ep->wMaxPacketSize);
  805. }
  806. musb_write_packet(bus, 0, urb->transfer_buffer, size);
  807. HWREGB(USB_TXCSRL_BASE(ep_idx)) = USB_CSRL0_TXRDY;
  808. urb->transfer_buffer += size;
  809. urb->transfer_buffer_length -= size;
  810. urb->actual_length += size;
  811. } else {
  812. pipe->ep0_state = USB_EP0_STATE_IN_STATUS;
  813. HWREGB(USB_TXCSRL_BASE(ep_idx)) = (USB_CSRL0_REQPKT | USB_CSRL0_STATUS);
  814. }
  815. break;
  816. case USB_EP0_STATE_OUT_STATUS:
  817. urb->errorcode = 0;
  818. musb_urb_waitup(urb);
  819. break;
  820. case USB_EP0_STATE_IN_STATUS:
  821. if (ep0_status & (USB_CSRL0_RXRDY | USB_CSRL0_STATUS)) {
  822. HWREGB(USB_TXCSRL_BASE(ep_idx)) &= ~(USB_CSRL0_RXRDY | USB_CSRL0_STATUS);
  823. urb->errorcode = 0;
  824. musb_urb_waitup(urb);
  825. }
  826. break;
  827. }
  828. }
  829. void USBH_IRQHandler(uint8_t busid)
  830. {
  831. uint32_t is;
  832. uint32_t txis;
  833. uint32_t rxis;
  834. uint8_t ep_csrl_status;
  835. // uint8_t ep_csrh_status;
  836. struct musb_pipe *pipe;
  837. struct usbh_urb *urb;
  838. uint8_t ep_idx;
  839. uint8_t old_ep_idx;
  840. struct usbh_bus *bus;
  841. uint32_t size;
  842. bus = &g_usbhost_bus[busid];
  843. if (!(HWREGB(USB_BASE + MUSB_DEVCTL_OFFSET) & USB_DEVCTL_HOST)) {
  844. return;
  845. }
  846. is = HWREGB(USB_BASE + MUSB_IS_OFFSET);
  847. txis = HWREGH(USB_BASE + MUSB_TXIS_OFFSET);
  848. rxis = HWREGH(USB_BASE + MUSB_RXIS_OFFSET);
  849. HWREGB(USB_BASE + MUSB_IS_OFFSET) = is;
  850. old_ep_idx = musb_get_active_ep(bus);
  851. if (is & USB_IS_CONN) {
  852. g_musb_hcd[bus->hcd.hcd_id].port_csc = 1;
  853. g_musb_hcd[bus->hcd.hcd_id].port_pec = 1;
  854. g_musb_hcd[bus->hcd.hcd_id].port_pe = 1;
  855. bus->hcd.roothub.int_buffer[0] = (1 << 1);
  856. usbh_hub_thread_wakeup(&bus->hcd.roothub);
  857. }
  858. if (is & USB_IS_DISCON) {
  859. g_musb_hcd[bus->hcd.hcd_id].port_csc = 1;
  860. g_musb_hcd[bus->hcd.hcd_id].port_pec = 1;
  861. g_musb_hcd[bus->hcd.hcd_id].port_pe = 0;
  862. bus->hcd.roothub.int_buffer[0] = (1 << 1);
  863. usbh_hub_thread_wakeup(&bus->hcd.roothub);
  864. }
  865. if (is & USB_IS_SOF) {
  866. }
  867. if (is & USB_IS_RESUME) {
  868. }
  869. if (is & USB_IS_SUSPEND) {
  870. }
  871. if (is & USB_IS_VBUSERR) {
  872. }
  873. if (is & USB_IS_SESREQ) {
  874. }
  875. if (is & USB_IS_BABBLE) {
  876. }
  877. txis &= HWREGH(USB_BASE + MUSB_TXIE_OFFSET);
  878. /* Handle EP0 interrupt */
  879. if (txis & USB_TXIE_EP0) {
  880. txis &= ~USB_TXIE_EP0;
  881. HWREGH(USB_BASE + MUSB_TXIS_OFFSET) = USB_TXIE_EP0;
  882. handle_ep0(bus);
  883. }
  884. for (ep_idx = 1; ep_idx < CONFIG_USB_MUSB_PIPE_NUM; ep_idx++) {
  885. if (txis & (1 << ep_idx)) {
  886. HWREGH(USB_BASE + MUSB_TXIS_OFFSET) = (1 << ep_idx);
  887. pipe = &g_musb_hcd[bus->hcd.hcd_id].pipe_pool[ep_idx];
  888. urb = pipe->urb;
  889. musb_set_active_ep(bus, ep_idx);
  890. ep_csrl_status = HWREGB(USB_TXCSRL_BASE(ep_idx));
  891. if (ep_csrl_status & USB_TXCSRL1_ERROR) {
  892. HWREGB(USB_TXCSRL_BASE(ep_idx)) &= ~USB_TXCSRL1_ERROR;
  893. urb->errorcode = -USB_ERR_IO;
  894. musb_urb_waitup(urb);
  895. } else if (ep_csrl_status & USB_TXCSRL1_NAKTO) {
  896. HWREGB(USB_TXCSRL_BASE(ep_idx)) &= ~USB_TXCSRL1_NAKTO;
  897. urb->errorcode = -USB_ERR_NAK;
  898. musb_urb_waitup(urb);
  899. } else if (ep_csrl_status & USB_TXCSRL1_STALL) {
  900. HWREGB(USB_TXCSRL_BASE(ep_idx)) &= ~USB_TXCSRL1_STALL;
  901. urb->errorcode = -USB_ERR_STALL;
  902. musb_urb_waitup(urb);
  903. } else {
  904. if (USB_GET_ENDPOINT_TYPE(urb->ep->bmAttributes) != USB_ENDPOINT_TYPE_ISOCHRONOUS) {
  905. uint32_t size = urb->transfer_buffer_length;
  906. if (size > USB_GET_MAXPACKETSIZE(urb->ep->wMaxPacketSize)) {
  907. size = USB_GET_MAXPACKETSIZE(urb->ep->wMaxPacketSize);
  908. }
  909. urb->transfer_buffer += size;
  910. urb->transfer_buffer_length -= size;
  911. urb->actual_length += size;
  912. if (urb->transfer_buffer_length == 0) {
  913. //HWREGH(USB_BASE + MUSB_TXIE_OFFSET) &= ~(1 << ep_idx);
  914. urb->errorcode = 0;
  915. musb_urb_waitup(urb);
  916. } else {
  917. musb_write_packet(bus, ep_idx, urb->transfer_buffer, MIN(urb->transfer_buffer_length, USB_GET_MAXPACKETSIZE(urb->ep->wMaxPacketSize)));
  918. HWREGB(USB_TXCSRL_BASE(ep_idx)) = USB_TXCSRL1_TXRDY;
  919. }
  920. }
  921. }
  922. }
  923. }
  924. rxis &= HWREGH(USB_BASE + MUSB_RXIE_OFFSET);
  925. for (ep_idx = 1; ep_idx < CONFIG_USB_MUSB_PIPE_NUM; ep_idx++) {
  926. if (rxis & (1 << ep_idx)) {
  927. HWREGH(USB_BASE + MUSB_RXIS_OFFSET) = (1 << ep_idx); // clear isr flag
  928. pipe = &g_musb_hcd[bus->hcd.hcd_id].pipe_pool[ep_idx];
  929. urb = pipe->urb;
  930. musb_set_active_ep(bus, ep_idx);
  931. ep_csrl_status = HWREGB(USB_RXCSRL_BASE(ep_idx));
  932. //ep_csrh_status = HWREGB(USB_BASE + USB_RXCSRH_BASE(ep_idx)); // todo:for iso transfer
  933. if (ep_csrl_status & USB_RXCSRL1_ERROR) {
  934. HWREGB(USB_RXCSRL_BASE(ep_idx)) &= ~USB_RXCSRL1_ERROR;
  935. urb->errorcode = -USB_ERR_IO;
  936. musb_urb_waitup(urb);
  937. } else if (ep_csrl_status & USB_RXCSRL1_NAKTO) {
  938. HWREGB(USB_RXCSRL_BASE(ep_idx)) &= ~USB_RXCSRL1_NAKTO;
  939. urb->errorcode = -USB_ERR_NAK;
  940. musb_urb_waitup(urb);
  941. } else if (ep_csrl_status & USB_RXCSRL1_STALL) {
  942. HWREGB(USB_RXCSRL_BASE(ep_idx)) &= ~USB_RXCSRL1_STALL;
  943. urb->errorcode = -USB_ERR_STALL;
  944. musb_urb_waitup(urb);
  945. } else if (ep_csrl_status & USB_RXCSRL1_RXRDY) {
  946. if (USB_GET_ENDPOINT_TYPE(urb->ep->bmAttributes) != USB_ENDPOINT_TYPE_ISOCHRONOUS) {
  947. size = HWREGH(USB_RXCOUNT_BASE(ep_idx));
  948. musb_read_packet(bus, ep_idx, urb->transfer_buffer, size);
  949. HWREGB(USB_RXCSRL_BASE(ep_idx)) &= ~USB_RXCSRL1_RXRDY;
  950. urb->transfer_buffer += size;
  951. urb->transfer_buffer_length -= size;
  952. urb->actual_length += size;
  953. if ((size < USB_GET_MAXPACKETSIZE(urb->ep->wMaxPacketSize)) || (urb->transfer_buffer_length == 0)) {
  954. //HWREGH(USB_BASE + MUSB_RXIE_OFFSET) &= ~(1 << ep_idx);
  955. urb->errorcode = 0;
  956. musb_urb_waitup(urb);
  957. } else {
  958. HWREGB(USB_RXCSRL_BASE(ep_idx)) = USB_RXCSRL1_REQPKT;
  959. }
  960. }
  961. }
  962. }
  963. }
  964. musb_set_active_ep(bus, old_ep_idx);
  965. }