hotplug_usb.c 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933
  1. /*
  2. * Copyright (c) 2019 Actions Semiconductor Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file hotplug usb interface
  8. */
  9. #ifdef SYS_LOG_DOMAIN
  10. #undef SYS_LOG_DOMAIN
  11. #endif
  12. #define SYS_LOG_DOMAIN "hotpug_manager"
  13. #include <os_common_api.h>
  14. #include <msg_manager.h>
  15. #include <mem_manager.h>
  16. #include <sys_manager.h>
  17. #include <string.h>
  18. #include <fs_manager.h>
  19. #include <hotplug_manager.h>
  20. #include <init.h>
  21. #include <stdio.h>
  22. #include <device.h>
  23. #include <stdio.h>
  24. #include <usb/usb_otg.h>
  25. #include <usb/usb_device.h>
  26. #ifdef CONFIG_USB_HOST
  27. #include <usb/usb_host.h>
  28. #endif
  29. #include <usb/class/usb_msc.h>
  30. #ifdef CONFIG_USB_DEVICE
  31. #ifdef CONFIG_PM_DEVICE
  32. #include <pm/device.h>
  33. #include <sys_wakelock.h>
  34. static bool usb_wake_unlock;
  35. #endif
  36. #endif
  37. #define USB_HOTPLUG_NONE 0
  38. /* host mode: peripheral device connected */
  39. #define USB_HOTPLUG_A_IN 1
  40. /* host mode: peripheral device disconnected */
  41. #define USB_HOTPLUG_A_OUT 2
  42. /* device mode: connected to host */
  43. #define USB_HOTPLUG_B_IN 3
  44. /* device mode: disconnected from host */
  45. #define USB_HOTPLUG_B_OUT 4
  46. /* device mode: connected to charger */
  47. #define USB_HOTPLUG_C_IN 5
  48. /* device mode: disconnected from charger */
  49. #define USB_HOTPLUG_C_OUT 6
  50. /* usb hotplug suspend: stop detection */
  51. #define USB_HOTPLUG_SUSPEND 7
  52. static uint8_t usb_hotplug_state;
  53. #if CONFIG_LOG
  54. static char *state_string[] = {
  55. "undefined",
  56. "b_idle",
  57. "b_srp_init",
  58. "b_peripheral",
  59. "b_wait_acon",
  60. "b_host",
  61. "a_idle",
  62. "a_wait_vrise",
  63. "a_wait_bcon",
  64. "a_host",
  65. "a_suspend",
  66. "a_peripheral",
  67. "a_wait_vfall",
  68. "a_vbus_err"
  69. };
  70. #endif
  71. static uint8_t otg_state;
  72. #ifdef CONFIG_USB_DEVICE
  73. /* distinguish pc/charger: nearly 10s */
  74. #define USB_CONNECT_COUNT_MAX (10000/ CONFIG_MONITOR_PERIOD)
  75. static uint16_t usb_connect_count;
  76. /* optimize: if no_plugin */
  77. #define KEEP_IN_B_IDLE_RETRY 10
  78. static uint8_t keep_in_b_idle;
  79. /* doing b_peripheral exit process */
  80. static uint8_t otg_b_peripheral_exiting;
  81. #endif /* CONFIG_USB_DEVICE */
  82. #ifdef CONFIG_USB_HOST
  83. /* timeout: nearly 1s */
  84. #define OTG_A_WAIT_BCON_MAX (K_SECONDS(1) / CONFIG_MONITOR_PERIOD)
  85. static uint8_t otg_a_wait_bcon_count;
  86. /* timeout: nearly 4s */
  87. #define OTG_A_IDLE_MAX (K_SECONDS(4) / CONFIG_MONITOR_PERIOD)
  88. static uint8_t otg_a_idle_count;
  89. /* doing a_host exit process */
  90. static uint8_t otg_a_host_exiting;
  91. #endif /* CONFIG_USB_HOST */
  92. #ifdef CONFIG_USB_HOTPLUG_THREAD_ENABLED
  93. #define STACK_SZ CONFIG_USB_HOTPLUG_STACKSIZE
  94. #define THREAD_PRIO CONFIG_USB_HOTPLUG_PRIORITY
  95. static uint8_t usb_hotplug_stack[CONFIG_USB_HOTPLUG_STACKSIZE];
  96. #ifdef CONFIG_USB_HOST
  97. static void host_scan_thread(void *p1, void *p2, void *p3)
  98. {
  99. ARG_UNUSED(p1);
  100. ARG_UNUSED(p2);
  101. ARG_UNUSED(p3);
  102. usbh_scan_device();
  103. }
  104. static inline int usb_thread_init_host_scan(void)
  105. {
  106. k_tid_t tid;
  107. usbh_prepare_scan();
  108. /* Start a thread to offload USB scan/enumeration */
  109. tid = os_thread_create(usb_hotplug_stack, STACK_SZ,
  110. host_scan_thread,
  111. NULL, NULL, NULL,
  112. THREAD_PRIO, 0, 0);
  113. os_thread_name_set(tid, "host_scan");
  114. return 0;
  115. }
  116. static inline int usb_thread_exit_host_scan(void)
  117. {
  118. return usbh_disconnect();
  119. }
  120. #endif /* CONFIG_USB_HOST */
  121. #ifdef CONFIG_USB_MASS_STORAGE_SHARE_THREAD
  122. int usb_mass_storage_start(void)
  123. {
  124. k_tid_t tid;
  125. SYS_LOG_INF("shared");
  126. tid = os_thread_create(usb_hotplug_stack, STACK_SZ,
  127. usb_mass_storage_thread,
  128. NULL, NULL, NULL,
  129. CONFIG_MASS_STORAGE_PRIORITY, 0, 0);
  130. os_thread_name_set(tid, "mass_storage");
  131. return 0;
  132. }
  133. #endif /* CONFIG_USB_MASS_STORAGE_SHARE_THREAD */
  134. #endif /* CONFIG_USB_HOTPLUG_THREAD_ENABLED */
  135. static int usb_hotplug_dpdm_switch(bool host_mode)
  136. {
  137. #ifdef BOARD_USB_SWITCH_GPIO_NAME
  138. static struct device *usb_switch_gpio_dev;
  139. static uint8_t usb_switch_gpio_value;
  140. uint32_t value;
  141. int ret;
  142. if (usb_switch_gpio_dev == NULL) {
  143. usb_switch_gpio_dev = device_get_binding(BOARD_USB_SWITCH_GPIO_NAME);
  144. gpio_pin_configure(usb_switch_gpio_dev, BOARD_USB_SWITCH_EN_GPIO,
  145. GPIO_DIR_OUT);
  146. }
  147. value = BOARD_USB_SWITCH_HOST_EN_GPIO_VALUE;
  148. if (!host_mode) {
  149. value = !value;
  150. }
  151. /* already */
  152. if (value == usb_switch_gpio_value) {
  153. return 0;
  154. }
  155. usb_switch_gpio_value = value;
  156. ret = gpio_pin_write(usb_switch_gpio_dev, BOARD_USB_SWITCH_EN_GPIO,
  157. value);
  158. if (ret) {
  159. return ret;
  160. }
  161. #endif
  162. return 0;
  163. }
  164. static void usb_hotplug_status_cb(enum usb_dc_status_code status, uint8_t *param)
  165. {
  166. /* Check the USB status and do needed action if required */
  167. switch (status) {
  168. case USB_DC_HIGHSPEED:
  169. SYS_LOG_DBG("USB HS detected");
  170. usb_disable();
  171. break;
  172. case USB_DC_SOF:
  173. SYS_LOG_DBG("USB SOF detected");
  174. usb_disable();
  175. break;
  176. default:
  177. break;
  178. }
  179. }
  180. u8_t usb_hotplug_get_otgstate(void)
  181. {
  182. return otg_state;
  183. }
  184. static const struct usb_cfg_data usb_hotplug_config = {
  185. .cb_usb_status = usb_hotplug_status_cb,
  186. };
  187. static inline int usb_hotplug_detect_device(void)
  188. {
  189. int ret;
  190. ret = usb_enable((struct usb_cfg_data *)&usb_hotplug_config);
  191. if (ret < 0) {
  192. SYS_LOG_ERR("enable");
  193. return ret;
  194. }
  195. return 0;
  196. }
  197. #ifdef CONFIG_USB_HOST
  198. static inline int usb_hotplug_host_plugin(void)
  199. {
  200. usb_thread_init_host_scan();
  201. return 0;
  202. }
  203. static inline int usb_hotplug_host_plugout(void)
  204. {
  205. usb_hotplug_state = USB_HOTPLUG_A_OUT;
  206. return usb_thread_exit_host_scan();
  207. }
  208. #ifdef CONFIG_USB_HOST_STORAGE
  209. extern void usb_stor_set_access(bool accessed);
  210. extern bool usb_host_storage_enabled(void);
  211. int usb_hotplug_disk_accessed(bool access)
  212. {
  213. usb_stor_set_access(access);
  214. #ifdef CONFIG_FS_MANAGER
  215. if (access) {
  216. fs_manager_disk_init("USB:");
  217. } else {
  218. fs_manager_disk_uninit("USB:");
  219. }
  220. #endif
  221. return 0;
  222. }
  223. static inline int usb_hotplug_host_storage_plugin(void)
  224. {
  225. usb_hotplug_state = USB_HOTPLUG_A_IN;
  226. return 0;
  227. }
  228. #endif /* CONFIG_USB_HOST_STORAGE */
  229. #endif /* CONFIG_USB_HOST */
  230. #ifdef CONFIG_USB_DEVICE
  231. static inline int usb_hotplug_device_plugin(void)
  232. {
  233. usb_hotplug_state = USB_HOTPLUG_B_IN;
  234. return 0;
  235. }
  236. static inline int usb_hotplug_device_plugout(void)
  237. {
  238. usb_hotplug_state = USB_HOTPLUG_B_OUT;
  239. return 0;
  240. }
  241. static inline int usb_hotplug_charger_plugin(void)
  242. {
  243. usb_hotplug_state = USB_HOTPLUG_C_IN;
  244. return 0;
  245. }
  246. static inline int usb_hotplug_charger_plugout(void)
  247. {
  248. usb_hotplug_state = USB_HOTPLUG_C_OUT;
  249. return 0;
  250. }
  251. #endif /* CONFIG_USB_DEVICE */
  252. static inline uint8_t usb_hotplug_get_vbus(void)
  253. {
  254. #ifdef CONFIG_USB_DEVICE
  255. return usb_phy_get_vbus();
  256. #else
  257. return USB_VBUS_LOW; /* Host-only mode: never enter b_idle */
  258. #endif
  259. }
  260. static inline void usb_hotplug_update_state(void)
  261. {
  262. uint8_t vbus = usb_hotplug_get_vbus();
  263. #ifdef CONFIG_USB_DEVICE
  264. bool dc_attached;
  265. uint8_t i;
  266. #endif
  267. #ifdef CONFIG_USB_DEVICE
  268. #ifdef CONFIG_PM_DEVICE
  269. if (system_is_ready()) {
  270. if (vbus == USB_VBUS_HIGH) {
  271. if (!usb_wake_unlock) {
  272. sys_wake_lock_ext(PARTIAL_WAKE_LOCK,USB_WAKE_LOCK_USER);
  273. usb_wake_unlock = true;
  274. }
  275. } else {
  276. if (usb_wake_unlock) {
  277. sys_wake_unlock_ext(PARTIAL_WAKE_LOCK,USB_WAKE_LOCK_USER);
  278. usb_wake_unlock = false;
  279. }
  280. }
  281. }
  282. #endif
  283. #endif
  284. switch (otg_state) {
  285. #ifdef CONFIG_USB_DEVICE
  286. case OTG_STATE_B_IDLE:
  287. if (vbus == USB_VBUS_LOW) {
  288. keep_in_b_idle = 0;
  289. otg_state = OTG_STATE_A_IDLE;
  290. #ifndef CONFIG_USB_HOST
  291. usb_phy_reset();
  292. #endif
  293. usb_hotplug_charger_plugout();
  294. break;
  295. }
  296. if (keep_in_b_idle >= KEEP_IN_B_IDLE_RETRY) {
  297. break;
  298. }
  299. usb_phy_reset();
  300. usb_phy_enter_b_idle();
  301. k_sleep(K_MSEC(3)); /* necessary */
  302. repeat:
  303. dc_attached = usb_phy_dc_attached();
  304. for (i = 0; i < 4; i++) {
  305. k_busy_wait(1000);
  306. if (dc_attached != usb_phy_dc_attached()) {
  307. SYS_LOG_INF("dc_attached: %d, i: %d",
  308. dc_attached, i);
  309. goto repeat;
  310. }
  311. }
  312. if (dc_attached) {
  313. keep_in_b_idle = 0;
  314. otg_state = OTG_STATE_B_WAIT_ACON;
  315. usb_hotplug_detect_device();
  316. } else {
  317. if (++keep_in_b_idle >= KEEP_IN_B_IDLE_RETRY) {
  318. SYS_LOG_INF("b_idle");
  319. }
  320. usb_phy_enter_a_idle();
  321. usb_hotplug_charger_plugin();
  322. }
  323. break;
  324. case OTG_STATE_B_WAIT_ACON:
  325. if (vbus == USB_VBUS_LOW) {
  326. usb_connect_count = 0;
  327. otg_state = OTG_STATE_A_IDLE;
  328. /* exit device mode */
  329. usb_disable();
  330. usb_hotplug_charger_plugout();
  331. break;
  332. }
  333. if (usb_phy_dc_connected()) {
  334. usb_connect_count = 0;
  335. otg_state = OTG_STATE_B_PERIPHERAL;
  336. /* enter device mode */
  337. usb_hotplug_device_plugin();
  338. break;
  339. }
  340. if (++usb_connect_count >= USB_CONNECT_COUNT_MAX) {
  341. SYS_LOG_DBG("connected to charger");
  342. keep_in_b_idle = KEEP_IN_B_IDLE_RETRY;
  343. usb_connect_count = 0;
  344. otg_state = OTG_STATE_B_IDLE;
  345. /* exit device mode */
  346. usb_disable();
  347. usb_hotplug_charger_plugin();
  348. break;
  349. }
  350. break;
  351. case OTG_STATE_B_PERIPHERAL:
  352. if (otg_b_peripheral_exiting) {
  353. /* wait for exit done */
  354. if (usb_phy_dc_detached()) {
  355. otg_state = OTG_STATE_A_IDLE;
  356. otg_b_peripheral_exiting = 0;
  357. }
  358. break;
  359. }
  360. if (vbus == USB_VBUS_LOW) {
  361. /* exit device mode */
  362. usb_hotplug_device_plugout();
  363. otg_b_peripheral_exiting = 1;
  364. /* wait for exit done */
  365. if (usb_phy_dc_detached()) {
  366. otg_state = OTG_STATE_A_IDLE;
  367. otg_b_peripheral_exiting = 0;
  368. }
  369. break;
  370. }
  371. if (usb_device_unconfigured()) {
  372. usb_hotplug_device_plugout();
  373. otg_b_peripheral_exiting = 1;
  374. SYS_LOG_INF("usb unconfigured");
  375. break;
  376. }
  377. #ifdef CONFIG_USB_MASS_STORAGE
  378. if (usb_mass_storage_ejected()) {
  379. usb_hotplug_device_plugout();
  380. otg_b_peripheral_exiting = 1;
  381. SYS_LOG_INF("usb msc ejected");
  382. break;
  383. }
  384. #endif
  385. break;
  386. #endif /* CONFIG_USB_DEVICE */
  387. #ifdef CONFIG_USB_HOST
  388. case OTG_STATE_A_IDLE:
  389. if (vbus == USB_VBUS_HIGH) {
  390. otg_a_idle_count = 0;
  391. otg_state = OTG_STATE_B_IDLE;
  392. usbh_vbus_set(false);
  393. break;
  394. }
  395. /* Enable Vbus */
  396. usbh_vbus_set(true);
  397. if (usb_phy_hc_attached()) {
  398. otg_a_idle_count = 0;
  399. otg_state = OTG_STATE_A_WAIT_BCON;
  400. usb_phy_enter_a_wait_bcon();
  401. break;
  402. }
  403. /* reset Vbus */
  404. if (++otg_a_idle_count >= OTG_A_IDLE_MAX) {
  405. otg_a_idle_count = 0;
  406. usbh_vbus_set(false);
  407. break;
  408. }
  409. break;
  410. case OTG_STATE_A_WAIT_BCON:
  411. if (vbus == USB_VBUS_HIGH) {
  412. otg_a_wait_bcon_count = 0;
  413. otg_state = OTG_STATE_B_IDLE;
  414. /* exit host mode */
  415. usbh_vbus_set(false);
  416. break;
  417. }
  418. if (usb_phy_hc_connected()) {
  419. otg_a_wait_bcon_count = 0;
  420. otg_state = OTG_STATE_A_HOST;
  421. /* enter host mode */
  422. usb_hotplug_host_plugin();
  423. break;
  424. }
  425. if (++otg_a_wait_bcon_count >= OTG_A_WAIT_BCON_MAX) {
  426. otg_a_wait_bcon_count = 0;
  427. otg_state = OTG_STATE_A_IDLE;
  428. usbh_vbus_set(false);
  429. break;
  430. }
  431. break;
  432. case OTG_STATE_A_HOST:
  433. if (vbus == USB_VBUS_HIGH) {
  434. if (otg_a_host_exiting) {
  435. if (!usb_hotplug_host_plugout()) {
  436. otg_a_host_exiting = 0;
  437. otg_state = OTG_STATE_B_IDLE;
  438. }
  439. break;
  440. }
  441. /* exit host mode */
  442. usbh_vbus_set(false);
  443. if (usb_hotplug_host_plugout()) {
  444. otg_a_host_exiting = 1;
  445. } else {
  446. otg_state = OTG_STATE_B_IDLE;
  447. }
  448. break;
  449. }
  450. /* handle disconnect and connnect quickly */
  451. if (otg_a_host_exiting) {
  452. if (!usb_hotplug_host_plugout()) {
  453. otg_a_host_exiting = 0;
  454. otg_state = OTG_STATE_A_IDLE;
  455. }
  456. break;
  457. }
  458. if (usb_phy_hc_disconnected()) {
  459. /* exit host mode */
  460. usbh_vbus_set(false);
  461. if (usb_hotplug_host_plugout()) {
  462. otg_a_host_exiting = 1;
  463. } else {
  464. otg_state = OTG_STATE_A_IDLE;
  465. }
  466. break;
  467. }
  468. #ifdef CONFIG_USB_HOST_STORAGE
  469. if (usb_host_storage_enabled()) {
  470. usb_hotplug_host_storage_plugin();
  471. break;
  472. }
  473. #endif
  474. break;
  475. #else /* CONFIG_USB_HOST */
  476. case OTG_STATE_A_IDLE:
  477. if (vbus == USB_VBUS_HIGH) {
  478. otg_state = OTG_STATE_B_IDLE;
  479. break;
  480. }
  481. break;
  482. #endif
  483. default:
  484. break;
  485. }
  486. }
  487. static int usb_hotplug_loop(void)
  488. {
  489. enum usb_otg_state old_state = otg_state;
  490. usb_hotplug_update_state();
  491. if (otg_state == old_state) {
  492. return 0;
  493. }
  494. SYS_LOG_INF("%s -> %s", state_string[old_state],
  495. state_string[otg_state]);
  496. switch (otg_state) {
  497. case OTG_STATE_B_IDLE:
  498. usb_hotplug_dpdm_switch(false);
  499. break;
  500. case OTG_STATE_A_IDLE:
  501. usb_hotplug_dpdm_switch(true);
  502. #ifdef CONFIG_USB_HOST
  503. usb_phy_reset();
  504. usb_phy_enter_a_idle();
  505. #endif
  506. break;
  507. default:
  508. break;
  509. }
  510. return 0;
  511. }
  512. static inline int usb_hotplug_init(void)
  513. {
  514. uint8_t vbus = usb_hotplug_get_vbus();
  515. #ifdef CONFIG_USB_DEVICE
  516. /* suspend comes before hotplug and keep_in_b_idle maybe not reset */
  517. keep_in_b_idle = 0;
  518. #endif
  519. if (vbus == USB_VBUS_HIGH) {
  520. usb_hotplug_dpdm_switch(false);
  521. #ifdef CONFIG_USB_HOST
  522. usbh_vbus_set(false);
  523. #endif
  524. otg_state = OTG_STATE_B_IDLE;
  525. } else {
  526. usb_hotplug_dpdm_switch(true);
  527. #ifdef CONFIG_USB_HOST
  528. usb_phy_enter_a_idle();
  529. #endif
  530. otg_state = OTG_STATE_A_IDLE;
  531. }
  532. SYS_LOG_INF("state %s", state_string[otg_state]);
  533. /* For a_idle, detect later */
  534. if (otg_state == OTG_STATE_B_IDLE) {
  535. usb_hotplug_loop();
  536. }
  537. return 0;
  538. }
  539. /*
  540. * Check if USB is attached to charger or host
  541. */
  542. bool usb_hotplug_device_mode(void)
  543. {
  544. if ((otg_state == OTG_STATE_B_WAIT_ACON) ||
  545. (otg_state == OTG_STATE_B_PERIPHERAL)) {
  546. return true;
  547. }
  548. return false;
  549. }
  550. int usb_hotplug_suspend(void)
  551. {
  552. usb_hotplug_state = USB_HOTPLUG_SUSPEND;
  553. #ifdef CONFIG_USB_HOST
  554. usbh_vbus_set(false);
  555. if (otg_state == OTG_STATE_A_HOST) {
  556. /* should be fast */
  557. while (usbh_disconnect()) {
  558. k_sleep(K_MSEC(10));
  559. SYS_LOG_INF("");
  560. }
  561. }
  562. #endif
  563. usb_phy_exit();
  564. return 0;
  565. }
  566. int usb_hotplug_resume(void)
  567. {
  568. usb_hotplug_state = HOTPLUG_NONE;
  569. #ifdef CONFIG_USB_HOST
  570. if (otg_state == OTG_STATE_A_HOST) {
  571. /* broadcast when resume */
  572. struct app_msg msg = {0};
  573. msg.type = MSG_HOTPLUG_EVENT;
  574. msg.cmd = HOTPLUG_USB_HOST;
  575. msg.value = HOTPLUG_OUT;
  576. SYS_LOG_INF("host");
  577. return send_async_msg("main", &msg);
  578. }
  579. #endif
  580. return usb_hotplug_init();
  581. }
  582. #ifdef CONFIG_USB_HOST
  583. /* check and enumerate USB device directly */
  584. int usb_hotplug_host_check(void)
  585. {
  586. int timeout = 100; /* 1s */
  587. if (otg_state != OTG_STATE_A_IDLE) {
  588. return -EINVAL;
  589. }
  590. do {
  591. usb_hotplug_loop();
  592. switch (otg_state) {
  593. case OTG_STATE_A_WAIT_BCON:
  594. goto attached;
  595. case OTG_STATE_A_IDLE:
  596. break;
  597. default:
  598. goto exit;
  599. }
  600. k_sleep(K_MSEC(10));
  601. } while (--timeout);
  602. /* timeout */
  603. if (timeout == 0) {
  604. return -ENODEV;
  605. }
  606. attached:
  607. timeout = 10; /* 1s */
  608. do {
  609. usb_hotplug_loop();
  610. switch (otg_state) {
  611. case OTG_STATE_A_HOST:
  612. usbh_prepare_scan();
  613. usbh_scan_device();
  614. /* fall through */
  615. default:
  616. goto exit;
  617. case OTG_STATE_A_WAIT_BCON:
  618. break;
  619. }
  620. /* should be same as CONFIG_MONITOR_PERIOD */
  621. k_sleep(K_MSEC(100));
  622. } while (--timeout);
  623. /* timeout */
  624. if (timeout == 0) {
  625. return -ENODEV;
  626. }
  627. exit:
  628. return 0;
  629. }
  630. #endif
  631. static int hotplug_usb_get_type(void)
  632. {
  633. switch (usb_hotplug_state) {
  634. case USB_HOTPLUG_A_IN:
  635. case USB_HOTPLUG_A_OUT:
  636. return HOTPLUG_USB_HOST;
  637. case USB_HOTPLUG_B_IN:
  638. case USB_HOTPLUG_B_OUT:
  639. return HOTPLUG_USB_DEVICE;
  640. }
  641. /* never */
  642. return 0;
  643. }
  644. static int hotplug_usb_detect(void)
  645. {
  646. uint8_t old_state = usb_hotplug_state;
  647. if (usb_hotplug_state == USB_HOTPLUG_SUSPEND) {
  648. return HOTPLUG_NONE;
  649. }
  650. usb_hotplug_loop();
  651. if (usb_hotplug_state == old_state) {
  652. return HOTPLUG_NONE;
  653. }
  654. switch (usb_hotplug_state) {
  655. case USB_HOTPLUG_A_IN:
  656. case USB_HOTPLUG_B_IN:
  657. return HOTPLUG_IN;
  658. case USB_HOTPLUG_A_OUT:
  659. case USB_HOTPLUG_B_OUT:
  660. return HOTPLUG_OUT;
  661. }
  662. return HOTPLUG_NONE;
  663. }
  664. static int hotplug_usb_process(int device_state)
  665. {
  666. int res = -1;
  667. SYS_LOG_INF("%d", usb_hotplug_state);
  668. switch (usb_hotplug_state) {
  669. case USB_HOTPLUG_A_IN:
  670. #ifdef CONFIG_FS_MANAGER
  671. res = fs_manager_disk_init("USB:");
  672. #endif
  673. break;
  674. case USB_HOTPLUG_A_OUT:
  675. #ifdef CONFIG_FS_MANAGER
  676. res = fs_manager_disk_uninit("USB:");
  677. #endif
  678. break;
  679. case USB_HOTPLUG_B_IN:
  680. case USB_HOTPLUG_B_OUT:
  681. res = 0;
  682. break;
  683. default:
  684. break;
  685. }
  686. return res;
  687. }
  688. #ifdef CONFIG_USB_DEVICE
  689. static int hotplug_usb_device_get_state(void)
  690. {
  691. if (otg_b_peripheral_exiting == 1) {
  692. return HOTPLUG_OUT;
  693. }
  694. if (otg_state == OTG_STATE_B_PERIPHERAL) {
  695. return HOTPLUG_IN;
  696. }
  697. return HOTPLUG_OUT;
  698. }
  699. static const struct hotplug_device_t hotplug_usb_device = {
  700. .type = HOTPLUG_USB_DEVICE,
  701. .get_state = hotplug_usb_device_get_state,
  702. .get_type = hotplug_usb_get_type,
  703. .hotplug_detect = hotplug_usb_detect,
  704. .fs_process = hotplug_usb_process,
  705. };
  706. #endif /* CONFIG_USB_DEVICE */
  707. #ifdef CONFIG_USB_HOST
  708. static int hotplug_usb_host_get_state(void)
  709. {
  710. if (otg_state == OTG_STATE_A_HOST) {
  711. return HOTPLUG_IN;
  712. }
  713. return HOTPLUG_OUT;
  714. }
  715. static int __unused hotplug_usb_host_detect(void)
  716. {
  717. return HOTPLUG_NONE;
  718. }
  719. static const struct hotplug_device_t hotplug_usb_host = {
  720. .type = HOTPLUG_USB_HOST,
  721. .get_state = hotplug_usb_host_get_state,
  722. #ifndef CONFIG_USB
  723. .get_type = hotplug_usb_get_type,
  724. .hotplug_detect = hotplug_usb_detect,
  725. .fs_process = hotplug_usb_process,
  726. #else
  727. .hotplug_detect = hotplug_usb_host_detect,
  728. #endif
  729. };
  730. #endif /* CONFIG_USB_HOST */
  731. int hotplug_usb_init(void)
  732. {
  733. #ifdef CONFIG_USB_DEVICE
  734. hotplug_device_register(&hotplug_usb_device);
  735. #endif
  736. #ifdef CONFIG_USB_HOST
  737. hotplug_device_register(&hotplug_usb_host);
  738. #endif
  739. return 0;
  740. }
  741. #ifdef CONFIG_USB_HOTPLUG
  742. static struct k_timer usb_hotplug_timer;
  743. static void usb_hotplug_expiry_fn(struct k_timer *timer)
  744. {
  745. switch (otg_state) {
  746. case OTG_STATE_B_WAIT_ACON:
  747. case OTG_STATE_B_PERIPHERAL:
  748. if (!usb_phy_get_vbus() && !usb_phy_dc_detached()) {
  749. usb_phy_dc_disconnect();
  750. }
  751. break;
  752. default:
  753. break;
  754. }
  755. }
  756. #endif /* CONFIG_USB_HOTPLUG */
  757. int usb_hotplug_pre_init(const struct device *dev)
  758. {
  759. ARG_UNUSED(dev);
  760. #ifdef CONFIG_USB_HOTPLUG
  761. k_timer_init(&usb_hotplug_timer, usb_hotplug_expiry_fn, NULL);
  762. k_timer_start(&usb_hotplug_timer, K_SECONDS(1), K_MSEC(400));
  763. #endif
  764. usb_phy_init();
  765. #ifdef CONFIG_USB_HOST
  766. /* turn off Vbus by default */
  767. usbh_vbus_set(false);
  768. #endif
  769. usb_hotplug_init();
  770. return 0;
  771. }
  772. #define USB_HOTPLUG_PRE_INIT_PRIORITY CONFIG_APPLICATION_INIT_PRIORITY
  773. SYS_INIT(usb_hotplug_pre_init, POST_KERNEL, USB_HOTPLUG_PRE_INIT_PRIORITY);