g_dnl.c 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * g_dnl.c -- USB Downloader Gadget
  4. *
  5. * Copyright (C) 2012 Samsung Electronics
  6. * Lukasz Majewski <l.majewski@samsung.com>
  7. */
  8. #include <common.h>
  9. #include <malloc.h>
  10. #include <mmc.h>
  11. #include <part.h>
  12. #include <usb.h>
  13. #include <g_dnl.h>
  14. #include <usb_mass_storage.h>
  15. #include <dfu.h>
  16. #include <thor.h>
  17. #include <env_callback.h>
  18. #include "gadget_chips.h"
  19. #include "composite.c"
  20. /*
  21. * One needs to define the following:
  22. * CONFIG_USB_GADGET_VENDOR_NUM
  23. * CONFIG_USB_GADGET_PRODUCT_NUM
  24. * CONFIG_USB_GADGET_MANUFACTURER
  25. * at e.g. ./configs/<board>_defconfig
  26. */
  27. #define STRING_MANUFACTURER 25
  28. #define STRING_PRODUCT 2
  29. /* Index of String Descriptor describing this configuration */
  30. #define STRING_USBDOWN 2
  31. /* Index of String serial */
  32. #define STRING_SERIAL 3
  33. #define MAX_STRING_SERIAL 256
  34. /* Number of supported configurations */
  35. #define CONFIGURATION_NUMBER 1
  36. #define DRIVER_VERSION "usb_dnl 2.0"
  37. static const char product[] = "USB download gadget";
  38. static char g_dnl_serial[MAX_STRING_SERIAL];
  39. static const char manufacturer[] = CONFIG_USB_GADGET_MANUFACTURER;
  40. void g_dnl_set_serialnumber(char *s)
  41. {
  42. memset(g_dnl_serial, 0, MAX_STRING_SERIAL);
  43. strncpy(g_dnl_serial, s, MAX_STRING_SERIAL - 1);
  44. }
  45. static struct usb_device_descriptor device_desc = {
  46. .bLength = sizeof device_desc,
  47. .bDescriptorType = USB_DT_DEVICE,
  48. .bcdUSB = __constant_cpu_to_le16(0x0200),
  49. .bDeviceClass = USB_CLASS_PER_INTERFACE,
  50. .bDeviceSubClass = 0, /*0x02:CDC-modem , 0x00:CDC-serial*/
  51. .idVendor = __constant_cpu_to_le16(CONFIG_USB_GADGET_VENDOR_NUM),
  52. .idProduct = __constant_cpu_to_le16(CONFIG_USB_GADGET_PRODUCT_NUM),
  53. /* .iProduct = DYNAMIC */
  54. /* .iSerialNumber = DYNAMIC */
  55. .bNumConfigurations = 1,
  56. };
  57. /*
  58. * static strings, in UTF-8
  59. * IDs for those strings are assigned dynamically at g_dnl_bind()
  60. */
  61. static struct usb_string g_dnl_string_defs[] = {
  62. {.s = manufacturer},
  63. {.s = product},
  64. {.s = g_dnl_serial},
  65. { } /* end of list */
  66. };
  67. static struct usb_gadget_strings g_dnl_string_tab = {
  68. .language = 0x0409, /* en-us */
  69. .strings = g_dnl_string_defs,
  70. };
  71. static struct usb_gadget_strings *g_dnl_composite_strings[] = {
  72. &g_dnl_string_tab,
  73. NULL,
  74. };
  75. static int g_dnl_unbind(struct usb_composite_dev *cdev)
  76. {
  77. struct usb_gadget *gadget = cdev->gadget;
  78. debug("%s: calling usb_gadget_disconnect for "
  79. "controller '%s'\n", __func__, gadget->name);
  80. usb_gadget_disconnect(gadget);
  81. return 0;
  82. }
  83. static inline struct g_dnl_bind_callback *g_dnl_bind_callback_first(void)
  84. {
  85. return ll_entry_start(struct g_dnl_bind_callback,
  86. g_dnl_bind_callbacks);
  87. }
  88. static inline struct g_dnl_bind_callback *g_dnl_bind_callback_end(void)
  89. {
  90. return ll_entry_end(struct g_dnl_bind_callback,
  91. g_dnl_bind_callbacks);
  92. }
  93. static int g_dnl_do_config(struct usb_configuration *c)
  94. {
  95. const char *s = c->cdev->driver->name;
  96. struct g_dnl_bind_callback *callback = g_dnl_bind_callback_first();
  97. debug("%s: configuration: 0x%p composite dev: 0x%p\n",
  98. __func__, c, c->cdev);
  99. for (; callback != g_dnl_bind_callback_end(); callback++)
  100. if (!strcmp(s, callback->usb_function_name))
  101. return callback->fptr(c);
  102. return -ENODEV;
  103. }
  104. static int g_dnl_config_register(struct usb_composite_dev *cdev)
  105. {
  106. struct usb_configuration *config;
  107. const char *name = "usb_dnload";
  108. config = memalign(CONFIG_SYS_CACHELINE_SIZE, sizeof(*config));
  109. if (!config)
  110. return -ENOMEM;
  111. memset(config, 0, sizeof(*config));
  112. config->label = name;
  113. config->bmAttributes = USB_CONFIG_ATT_ONE | USB_CONFIG_ATT_SELFPOWER;
  114. config->bConfigurationValue = CONFIGURATION_NUMBER;
  115. config->iConfiguration = STRING_USBDOWN;
  116. config->bind = g_dnl_do_config;
  117. return usb_add_config(cdev, config);
  118. }
  119. __weak
  120. int board_usb_init(int index, enum usb_init_type init)
  121. {
  122. return 0;
  123. }
  124. __weak
  125. int board_usb_cleanup(int index, enum usb_init_type init)
  126. {
  127. return 0;
  128. }
  129. __weak
  130. int g_dnl_bind_fixup(struct usb_device_descriptor *dev, const char *name)
  131. {
  132. return 0;
  133. }
  134. __weak int g_dnl_get_board_bcd_device_number(int gcnum)
  135. {
  136. return gcnum;
  137. }
  138. __weak int g_dnl_board_usb_cable_connected(void)
  139. {
  140. return -EOPNOTSUPP;
  141. }
  142. static bool g_dnl_detach_request;
  143. bool g_dnl_detach(void)
  144. {
  145. return g_dnl_detach_request;
  146. }
  147. void g_dnl_trigger_detach(void)
  148. {
  149. g_dnl_detach_request = true;
  150. }
  151. void g_dnl_clear_detach(void)
  152. {
  153. g_dnl_detach_request = false;
  154. }
  155. static int g_dnl_get_bcd_device_number(struct usb_composite_dev *cdev)
  156. {
  157. struct usb_gadget *gadget = cdev->gadget;
  158. int gcnum;
  159. gcnum = usb_gadget_controller_number(gadget);
  160. if (gcnum > 0)
  161. gcnum += 0x200;
  162. return g_dnl_get_board_bcd_device_number(gcnum);
  163. }
  164. /**
  165. * Update internal serial number variable when the "serial#" env var changes.
  166. *
  167. * Handle all cases, even when flags == H_PROGRAMMATIC or op == env_op_delete.
  168. */
  169. static int on_serialno(const char *name, const char *value, enum env_op op,
  170. int flags)
  171. {
  172. g_dnl_set_serialnumber((char *)value);
  173. return 0;
  174. }
  175. U_BOOT_ENV_CALLBACK(serialno, on_serialno);
  176. static int g_dnl_bind(struct usb_composite_dev *cdev)
  177. {
  178. struct usb_gadget *gadget = cdev->gadget;
  179. int id, ret;
  180. int gcnum;
  181. debug("%s: gadget: 0x%p cdev: 0x%p\n", __func__, gadget, cdev);
  182. id = usb_string_id(cdev);
  183. if (id < 0)
  184. return id;
  185. g_dnl_string_defs[0].id = id;
  186. device_desc.iManufacturer = id;
  187. id = usb_string_id(cdev);
  188. if (id < 0)
  189. return id;
  190. g_dnl_string_defs[1].id = id;
  191. device_desc.iProduct = id;
  192. g_dnl_bind_fixup(&device_desc, cdev->driver->name);
  193. if (strlen(g_dnl_serial)) {
  194. id = usb_string_id(cdev);
  195. if (id < 0)
  196. return id;
  197. g_dnl_string_defs[2].id = id;
  198. device_desc.iSerialNumber = id;
  199. }
  200. ret = g_dnl_config_register(cdev);
  201. if (ret)
  202. goto error;
  203. gcnum = g_dnl_get_bcd_device_number(cdev);
  204. if (gcnum >= 0)
  205. device_desc.bcdDevice = cpu_to_le16(gcnum);
  206. else {
  207. debug("%s: controller '%s' not recognized\n",
  208. __func__, gadget->name);
  209. device_desc.bcdDevice = __constant_cpu_to_le16(0x9999);
  210. }
  211. debug("%s: calling usb_gadget_connect for "
  212. "controller '%s'\n", __func__, gadget->name);
  213. usb_gadget_connect(gadget);
  214. return 0;
  215. error:
  216. g_dnl_unbind(cdev);
  217. return -ENOMEM;
  218. }
  219. static struct usb_composite_driver g_dnl_driver = {
  220. .name = NULL,
  221. .dev = &device_desc,
  222. .strings = g_dnl_composite_strings,
  223. .bind = g_dnl_bind,
  224. .unbind = g_dnl_unbind,
  225. };
  226. /*
  227. * NOTICE:
  228. * Registering via USB function name won't be necessary after rewriting
  229. * g_dnl to support multiple USB functions.
  230. */
  231. int g_dnl_register(const char *name)
  232. {
  233. int ret;
  234. debug("%s: g_dnl_driver.name = %s\n", __func__, name);
  235. g_dnl_driver.name = name;
  236. ret = usb_composite_register(&g_dnl_driver);
  237. if (ret) {
  238. printf("%s: failed!, error: %d\n", __func__, ret);
  239. return ret;
  240. }
  241. return 0;
  242. }
  243. void g_dnl_unregister(void)
  244. {
  245. usb_composite_unregister(&g_dnl_driver);
  246. }