usbh_dfs.c 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241
  1. /*
  2. * Copyright (c) 2024, sakumisu
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include "usbh_core.h"
  7. #include "usbh_msc.h"
  8. #include "rtthread.h"
  9. #include <dfs_fs.h>
  10. #define DEV_FORMAT "/sd%c"
  11. #ifndef CONFIG_USB_DFS_MOUNT_POINT
  12. #define CONFIG_USB_DFS_MOUNT_POINT "/"
  13. #endif
  14. #if defined(SOC_SERIES_STM32H7) || defined(SOC_SERIES_STM32F7) || \
  15. defined(SOC_HPM5000) || defined(SOC_HPM6000) || defined(SOC_HPM6E00) || defined(BSP_USING_BL61X)
  16. #ifndef RT_USING_CACHE
  17. #error usbh msc must enable RT_USING_CACHE in this chip
  18. #endif
  19. #if RT_ALIGN_SIZE != 32 && RT_ALIGN_SIZE != 64
  20. #error usbh msc must set cache line to 32 or 64
  21. #endif
  22. #endif
  23. #if defined(BSP_USING_BL61X)
  24. #include "bflb_l1c.h"
  25. void rt_hw_cpu_dcache_ops(int ops, void *addr, int size)
  26. {
  27. if (ops == RT_HW_CACHE_FLUSH) {
  28. bflb_l1c_dcache_clean_range(addr, size);
  29. } else {
  30. bflb_l1c_dcache_invalidate_range(addr, size);
  31. }
  32. }
  33. #elif defined(SOC_HPM5000) || defined(SOC_HPM6000)
  34. #include "hpm_l1c_drv.h"
  35. void rt_hw_cpu_dcache_ops(int ops, void *addr, int size)
  36. {
  37. if (ops == RT_HW_CACHE_FLUSH) {
  38. l1c_dc_flush((uint32_t)addr, size);
  39. } else {
  40. l1c_dc_invalidate((uint32_t)addr, size);
  41. }
  42. }
  43. #endif
  44. USB_NOCACHE_RAM_SECTION USB_MEM_ALIGNX uint8_t msc_sector[512];
  45. static rt_err_t rt_udisk_init(rt_device_t dev)
  46. {
  47. return RT_EOK;
  48. }
  49. static ssize_t rt_udisk_read(rt_device_t dev, rt_off_t pos, void *buffer,
  50. rt_size_t size)
  51. {
  52. struct usbh_msc *msc_class = (struct usbh_msc *)dev->user_data;
  53. int ret;
  54. #ifdef RT_USING_CACHE
  55. rt_uint32_t *align_buf;
  56. if ((uint32_t)buffer & (RT_ALIGN_SIZE - 1)) {
  57. align_buf = rt_malloc_align(size * msc_class->blocksize, RT_ALIGN_SIZE);
  58. if (!align_buf) {
  59. rt_kprintf("msc get align buf failed\n");
  60. return 0;
  61. }
  62. } else {
  63. align_buf = (rt_uint32_t *)buffer;
  64. }
  65. ret = usbh_msc_scsi_read10(msc_class, pos, (uint8_t *)align_buf, size);
  66. if (ret < 0) {
  67. rt_kprintf("usb mass_storage read failed\n");
  68. return 0;
  69. }
  70. rt_hw_cpu_dcache_ops(RT_HW_CACHE_INVALIDATE, align_buf, size * msc_class->blocksize);
  71. if ((uint32_t)buffer & (RT_ALIGN_SIZE - 1)) {
  72. rt_memcpy(buffer, align_buf, size * msc_class->blocksize);
  73. rt_free_align(align_buf);
  74. }
  75. #else
  76. ret = usbh_msc_scsi_read10(msc_class, pos, buffer, size);
  77. if (ret < 0) {
  78. rt_kprintf("usb mass_storage read failed\n");
  79. return 0;
  80. }
  81. #endif
  82. return size;
  83. }
  84. static ssize_t rt_udisk_write(rt_device_t dev, rt_off_t pos, const void *buffer,
  85. rt_size_t size)
  86. {
  87. struct usbh_msc *msc_class = (struct usbh_msc *)dev->user_data;
  88. int ret;
  89. #ifdef RT_USING_CACHE
  90. rt_uint32_t *align_buf;
  91. if ((uint32_t)buffer & (RT_ALIGN_SIZE - 1)) {
  92. align_buf = rt_malloc_align(size * msc_class->blocksize, RT_ALIGN_SIZE);
  93. if (!align_buf) {
  94. rt_kprintf("msc get align buf failed\n");
  95. return 0;
  96. }
  97. rt_memcpy(align_buf, buffer, size * msc_class->blocksize);
  98. } else {
  99. align_buf = (rt_uint32_t *)buffer;
  100. }
  101. rt_hw_cpu_dcache_ops(RT_HW_CACHE_FLUSH, align_buf, size * msc_class->blocksize);
  102. ret = usbh_msc_scsi_write10(msc_class, pos, (uint8_t *)align_buf, size);
  103. if (ret < 0) {
  104. rt_kprintf("usb mass_storage write failed\n");
  105. return 0;
  106. }
  107. if ((uint32_t)buffer & (RT_ALIGN_SIZE - 1)) {
  108. rt_free_align(align_buf);
  109. }
  110. #else
  111. ret = usbh_msc_scsi_write10(msc_class, pos, buffer, size);
  112. if (ret < 0) {
  113. rt_kprintf("usb mass_storage write failed\n");
  114. return 0;
  115. }
  116. #endif
  117. return size;
  118. }
  119. static rt_err_t rt_udisk_control(rt_device_t dev, int cmd, void *args)
  120. {
  121. /* check parameter */
  122. RT_ASSERT(dev != RT_NULL);
  123. struct usbh_msc *msc_class = (struct usbh_msc *)dev->user_data;
  124. if (cmd == RT_DEVICE_CTRL_BLK_GETGEOME) {
  125. struct rt_device_blk_geometry *geometry;
  126. geometry = (struct rt_device_blk_geometry *)args;
  127. if (geometry == RT_NULL)
  128. return -RT_ERROR;
  129. geometry->bytes_per_sector = msc_class->blocksize;
  130. geometry->block_size = msc_class->blocksize;
  131. geometry->sector_count = msc_class->blocknum;
  132. }
  133. return RT_EOK;
  134. }
  135. #ifdef RT_USING_DEVICE_OPS
  136. const static struct rt_device_ops udisk_device_ops = {
  137. rt_udisk_init,
  138. RT_NULL,
  139. RT_NULL,
  140. rt_udisk_read,
  141. rt_udisk_write,
  142. rt_udisk_control
  143. };
  144. #endif
  145. int udisk_init(struct usbh_msc *msc_class)
  146. {
  147. rt_err_t ret = 0;
  148. rt_uint8_t i;
  149. struct dfs_partition part0;
  150. struct rt_device *dev;
  151. char name[CONFIG_USBHOST_DEV_NAMELEN];
  152. char mount_point[CONFIG_USBHOST_DEV_NAMELEN];
  153. dev = rt_malloc(sizeof(struct rt_device));
  154. memset(dev, 0, sizeof(struct rt_device));
  155. snprintf(name, CONFIG_USBHOST_DEV_NAMELEN, DEV_FORMAT, msc_class->sdchar);
  156. snprintf(mount_point, CONFIG_USBHOST_DEV_NAMELEN, CONFIG_USB_DFS_MOUNT_POINT, msc_class->sdchar);
  157. ret = usbh_msc_scsi_read10(msc_class, 0, msc_sector, 1);
  158. if (ret != RT_EOK) {
  159. rt_kprintf("usb mass_storage read failed\n");
  160. return ret;
  161. }
  162. for (i = 0; i < 16; i++) {
  163. /* Get the first partition */
  164. ret = dfs_filesystem_get_partition(&part0, msc_sector, i);
  165. if (ret == RT_EOK) {
  166. rt_kprintf("Found partition %d: type = %d, offet=0x%x, size=0x%x\n",
  167. i, part0.type, part0.offset, part0.size);
  168. } else {
  169. break;
  170. }
  171. }
  172. dev->type = RT_Device_Class_Block;
  173. #ifdef RT_USING_DEVICE_OPS
  174. dev->ops = &udisk_device_ops;
  175. #else
  176. dev->init = rt_udisk_init;
  177. dev->read = rt_udisk_read;
  178. dev->write = rt_udisk_write;
  179. dev->control = rt_udisk_control;
  180. #endif
  181. dev->user_data = msc_class;
  182. rt_device_register(dev, name, RT_DEVICE_FLAG_RDWR | RT_DEVICE_FLAG_REMOVABLE | RT_DEVICE_FLAG_STANDALONE);
  183. ret = dfs_mount(name, mount_point, "elm", 0, 0);
  184. if (ret == 0) {
  185. rt_kprintf("udisk: %s mount successfully\n", name);
  186. } else {
  187. rt_kprintf("udisk: %s mount failed, ret = %d\n", name, ret);
  188. }
  189. return ret;
  190. }
  191. void usbh_msc_run(struct usbh_msc *msc_class)
  192. {
  193. udisk_init(msc_class);
  194. }
  195. void usbh_msc_stop(struct usbh_msc *msc_class)
  196. {
  197. char name[CONFIG_USBHOST_DEV_NAMELEN];
  198. char mount_point[CONFIG_USBHOST_DEV_NAMELEN];
  199. snprintf(name, CONFIG_USBHOST_DEV_NAMELEN, DEV_FORMAT, msc_class->sdchar);
  200. snprintf(mount_point, CONFIG_USBHOST_DEV_NAMELEN, CONFIG_USB_DFS_MOUNT_POINT, msc_class->sdchar);
  201. dfs_unmount(mount_point);
  202. rt_device_unregister(rt_device_find(name));
  203. }