disk_access.c 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181
  1. /*
  2. * Copyright (c) 2018 Intel Corporation.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <string.h>
  7. #include <zephyr/types.h>
  8. #include <sys/__assert.h>
  9. #include <sys/util.h>
  10. #include <init.h>
  11. #include <storage/disk_access.h>
  12. #include <errno.h>
  13. #include <device.h>
  14. #define LOG_LEVEL CONFIG_DISK_LOG_LEVEL
  15. #include <logging/log.h>
  16. LOG_MODULE_REGISTER(disk);
  17. /* list of mounted file systems */
  18. static sys_dlist_t disk_access_list;
  19. /* lock to protect storage layer registration */
  20. static struct k_mutex mutex;
  21. struct disk_info *disk_access_get_di(const char *name)
  22. {
  23. struct disk_info *disk = NULL, *itr;
  24. size_t name_len = strlen(name);
  25. sys_dnode_t *node;
  26. k_mutex_lock(&mutex, K_FOREVER);
  27. SYS_DLIST_FOR_EACH_NODE(&disk_access_list, node) {
  28. itr = CONTAINER_OF(node, struct disk_info, node);
  29. /*
  30. * Move to next node if mount point length is
  31. * shorter than longest_match match or if path
  32. * name is shorter than the mount point name.
  33. */
  34. if (strlen(itr->name) != name_len) {
  35. continue;
  36. }
  37. /* Check for disk name match */
  38. if (strncmp(name, itr->name, name_len) == 0) {
  39. disk = itr;
  40. break;
  41. }
  42. }
  43. k_mutex_unlock(&mutex);
  44. return disk;
  45. }
  46. int disk_access_init(const char *pdrv)
  47. {
  48. struct disk_info *disk = disk_access_get_di(pdrv);
  49. int rc = -EINVAL;
  50. if ((disk != NULL) && (disk->ops != NULL) &&
  51. (disk->ops->init != NULL)) {
  52. rc = disk->ops->init(disk);
  53. }
  54. return rc;
  55. }
  56. int disk_access_status(const char *pdrv)
  57. {
  58. struct disk_info *disk = disk_access_get_di(pdrv);
  59. int rc = -EINVAL;
  60. if ((disk != NULL) && (disk->ops != NULL) &&
  61. (disk->ops->status != NULL)) {
  62. rc = disk->ops->status(disk);
  63. }
  64. return rc;
  65. }
  66. int disk_access_read(const char *pdrv, uint8_t *data_buf,
  67. uint32_t start_sector, uint32_t num_sector)
  68. {
  69. struct disk_info *disk = disk_access_get_di(pdrv);
  70. int rc = -EINVAL;
  71. if ((disk != NULL) && (disk->ops != NULL) &&
  72. (disk->ops->read != NULL)) {
  73. rc = disk->ops->read(disk, data_buf, start_sector, num_sector);
  74. }
  75. return rc;
  76. }
  77. int disk_access_write(const char *pdrv, const uint8_t *data_buf,
  78. uint32_t start_sector, uint32_t num_sector)
  79. {
  80. struct disk_info *disk = disk_access_get_di(pdrv);
  81. int rc = -EINVAL;
  82. if ((disk != NULL) && (disk->ops != NULL) &&
  83. (disk->ops->write != NULL)) {
  84. rc = disk->ops->write(disk, data_buf, start_sector, num_sector);
  85. }
  86. return rc;
  87. }
  88. int disk_access_ioctl(const char *pdrv, uint8_t cmd, void *buf)
  89. {
  90. struct disk_info *disk = disk_access_get_di(pdrv);
  91. int rc = -EINVAL;
  92. if ((disk != NULL) && (disk->ops != NULL) &&
  93. (disk->ops->ioctl != NULL)) {
  94. rc = disk->ops->ioctl(disk, cmd, buf);
  95. }
  96. return rc;
  97. }
  98. int disk_access_register(struct disk_info *disk)
  99. {
  100. int rc = 0;
  101. k_mutex_lock(&mutex, K_FOREVER);
  102. if ((disk == NULL) || (disk->name == NULL)) {
  103. LOG_ERR("invalid disk interface!!");
  104. rc = -EINVAL;
  105. goto reg_err;
  106. }
  107. if (disk_access_get_di(disk->name) != NULL) {
  108. LOG_ERR("disk interface already registered!!");
  109. rc = -EINVAL;
  110. goto reg_err;
  111. }
  112. /* append to the disk list */
  113. sys_dlist_append(&disk_access_list, &disk->node);
  114. LOG_DBG("disk interface(%s) registred", disk->name);
  115. reg_err:
  116. k_mutex_unlock(&mutex);
  117. return rc;
  118. }
  119. int disk_access_unregister(struct disk_info *disk)
  120. {
  121. int rc = 0;
  122. k_mutex_lock(&mutex, K_FOREVER);
  123. if ((disk == NULL) || (disk->name == NULL)) {
  124. LOG_ERR("invalid disk interface!!");
  125. rc = -EINVAL;
  126. goto unreg_err;
  127. }
  128. if (disk_access_get_di(disk->name) == NULL) {
  129. LOG_ERR("disk interface not registered!!");
  130. rc = -EINVAL;
  131. goto unreg_err;
  132. }
  133. /* remove disk node from the list */
  134. sys_dlist_remove(&disk->node);
  135. LOG_DBG("disk interface(%s) unregistred", disk->name);
  136. unreg_err:
  137. k_mutex_unlock(&mutex);
  138. return rc;
  139. }
  140. static int disk_init(const struct device *dev)
  141. {
  142. ARG_UNUSED(dev);
  143. k_mutex_init(&mutex);
  144. sys_dlist_init(&disk_access_list);
  145. return 0;
  146. }
  147. SYS_INIT(disk_init, POST_KERNEL, CONFIG_KERNEL_INIT_PRIORITY_DEFAULT);