lfs_emubd.c 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291
  1. /*
  2. * Block device emulated on standard files
  3. *
  4. * Copyright (c) 2017 Christopher Haster
  5. * Distributed under the MIT license
  6. */
  7. #include "emubd/lfs_emubd.h"
  8. #include <errno.h>
  9. #include <string.h>
  10. #include <stdlib.h>
  11. #include <stdio.h>
  12. #include <limits.h>
  13. #include <dirent.h>
  14. #include <sys/stat.h>
  15. #include <unistd.h>
  16. // Block device emulated on existing filesystem
  17. int lfs_emubd_create(lfs_emubd_t *emu, const char *path) {
  18. memset(&emu->info, 0, sizeof(emu->info));
  19. memset(&emu->stats, 0, sizeof(emu->stats));
  20. // Allocate buffer for creating children files
  21. size_t pathlen = strlen(path);
  22. emu->path = malloc(pathlen + 1 + LFS_NAME_MAX + 1);
  23. if (!emu->path) {
  24. return -ENOMEM;
  25. }
  26. strcpy(emu->path, path);
  27. emu->path[pathlen] = '/';
  28. emu->child = &emu->path[pathlen+1];
  29. memset(emu->child, '\0', LFS_NAME_MAX+1);
  30. // Create directory if it doesn't exist
  31. int err = mkdir(path, 0777);
  32. if (err && errno != EEXIST) {
  33. return -errno;
  34. }
  35. // Setup info based on configuration
  36. emu->info.read_size = LFS_EMUBD_READ_SIZE;
  37. emu->info.prog_size = LFS_EMUBD_PROG_SIZE;
  38. emu->info.erase_size = LFS_EMUBD_ERASE_SIZE;
  39. emu->info.total_size = LFS_EMUBD_TOTAL_SIZE;
  40. // Load stats to continue incrementing
  41. snprintf(emu->child, LFS_NAME_MAX, "stats");
  42. FILE *f = fopen(emu->path, "r");
  43. if (!f) {
  44. return -errno;
  45. }
  46. size_t res = fread(&emu->stats, sizeof(emu->stats), 1, f);
  47. if (res < 1) {
  48. return -errno;
  49. }
  50. err = fclose(f);
  51. if (err) {
  52. return -errno;
  53. }
  54. return 0;
  55. }
  56. void lfs_emubd_destroy(lfs_emubd_t *emu) {
  57. lfs_emubd_sync(emu);
  58. free(emu->path);
  59. }
  60. int lfs_emubd_read(lfs_emubd_t *emu, lfs_block_t block,
  61. lfs_off_t off, lfs_size_t size, void *buffer) {
  62. uint8_t *data = buffer;
  63. // Check if read is valid
  64. if (!(off % emu->info.read_size == 0 &&
  65. size % emu->info.read_size == 0 &&
  66. ((uint64_t)block*emu->info.erase_size + off + size
  67. < emu->info.total_size))) {
  68. return -EINVAL;
  69. }
  70. // Zero out buffer for debugging
  71. memset(data, 0, size);
  72. // Iterate over blocks until enough data is read
  73. while (size > 0) {
  74. snprintf(emu->child, LFS_NAME_MAX, "%d", block);
  75. size_t count = lfs_min(emu->info.erase_size - off, size);
  76. FILE *f = fopen(emu->path, "rb");
  77. if (!f && errno != ENOENT) {
  78. return -errno;
  79. }
  80. if (f) {
  81. int err = fseek(f, off, SEEK_SET);
  82. if (err) {
  83. return -errno;
  84. }
  85. size_t res = fread(data, 1, count, f);
  86. if (res < count && !feof(f)) {
  87. return -errno;
  88. }
  89. err = fclose(f);
  90. if (err) {
  91. return -errno;
  92. }
  93. }
  94. size -= count;
  95. data += count;
  96. block += 1;
  97. off = 0;
  98. }
  99. emu->stats.read_count += 1;
  100. return 0;
  101. }
  102. int lfs_emubd_prog(lfs_emubd_t *emu, lfs_block_t block,
  103. lfs_off_t off, lfs_size_t size, const void *buffer) {
  104. const uint8_t *data = buffer;
  105. // Check if write is valid
  106. if (!(off % emu->info.prog_size == 0 &&
  107. size % emu->info.prog_size == 0 &&
  108. ((uint64_t)block*emu->info.erase_size + off + size
  109. < emu->info.total_size))) {
  110. return -EINVAL;
  111. }
  112. // Iterate over blocks until enough data is read
  113. while (size > 0) {
  114. snprintf(emu->child, LFS_NAME_MAX, "%d", block);
  115. size_t count = lfs_min(emu->info.erase_size - off, size);
  116. FILE *f = fopen(emu->path, "r+b");
  117. if (!f && errno == ENOENT) {
  118. f = fopen(emu->path, "w+b");
  119. if (!f) {
  120. return -errno;
  121. }
  122. }
  123. int err = fseek(f, off, SEEK_SET);
  124. if (err) {
  125. return -errno;
  126. }
  127. size_t res = fwrite(data, 1, count, f);
  128. if (res < count) {
  129. return -errno;
  130. }
  131. err = fclose(f);
  132. if (err) {
  133. return -errno;
  134. }
  135. size -= count;
  136. data += count;
  137. block += 1;
  138. off = 0;
  139. }
  140. emu->stats.prog_count += 1;
  141. return 0;
  142. }
  143. int lfs_emubd_erase(lfs_emubd_t *emu, lfs_block_t block,
  144. lfs_off_t off, lfs_size_t size) {
  145. // Check if erase is valid
  146. if (!(off % emu->info.erase_size == 0 &&
  147. size % emu->info.erase_size == 0 &&
  148. ((uint64_t)block*emu->info.erase_size + off + size
  149. < emu->info.total_size))) {
  150. return -EINVAL;
  151. }
  152. // Iterate and erase blocks
  153. while (size > 0) {
  154. snprintf(emu->child, LFS_NAME_MAX, "%d", block);
  155. struct stat st;
  156. int err = stat(emu->path, &st);
  157. if (err && errno != ENOENT) {
  158. return -errno;
  159. }
  160. if (!err && S_ISREG(st.st_mode)) {
  161. int err = unlink(emu->path);
  162. if (err) {
  163. return -errno;
  164. }
  165. }
  166. size -= emu->info.erase_size;
  167. block += 1;
  168. off = 0;
  169. }
  170. emu->stats.erase_count += 1;
  171. return 0;
  172. }
  173. int lfs_emubd_sync(lfs_emubd_t *emu) {
  174. // Just write out info/stats for later lookup
  175. snprintf(emu->child, LFS_NAME_MAX, "info");
  176. FILE *f = fopen(emu->path, "w");
  177. if (!f) {
  178. return -errno;
  179. }
  180. size_t res = fwrite(&emu->info, sizeof(emu->info), 1, f);
  181. if (res < 1) {
  182. return -errno;
  183. }
  184. int err = fclose(f);
  185. if (err) {
  186. return -errno;
  187. }
  188. snprintf(emu->child, LFS_NAME_MAX, "stats");
  189. f = fopen(emu->path, "w");
  190. if (!f) {
  191. return -errno;
  192. }
  193. res = fwrite(&emu->stats, sizeof(emu->stats), 1, f);
  194. if (res < 1) {
  195. return -errno;
  196. }
  197. err = fclose(f);
  198. if (err) {
  199. return -errno;
  200. }
  201. return 0;
  202. }
  203. int lfs_emubd_info(lfs_emubd_t *emu, struct lfs_bd_info *info) {
  204. *info = emu->info;
  205. return 0;
  206. }
  207. int lfs_emubd_stats(lfs_emubd_t *emu, struct lfs_bd_stats *stats) {
  208. *stats = emu->stats;
  209. return 0;
  210. }
  211. // Wrappers for void*s
  212. static int lfs_emubd_bd_read(void *bd, lfs_block_t block,
  213. lfs_off_t off, lfs_size_t size, void *buffer) {
  214. return lfs_emubd_read((lfs_emubd_t*)bd, block, off, size, buffer);
  215. }
  216. static int lfs_emubd_bd_prog(void *bd, lfs_block_t block,
  217. lfs_off_t off, lfs_size_t size, const void *buffer) {
  218. return lfs_emubd_prog((lfs_emubd_t*)bd, block, off, size, buffer);
  219. }
  220. static int lfs_emubd_bd_erase(void *bd, lfs_block_t block,
  221. lfs_off_t off, lfs_size_t size) {
  222. return lfs_emubd_erase((lfs_emubd_t*)bd, block, off, size);
  223. }
  224. static int lfs_emubd_bd_sync(void *bd) {
  225. return lfs_emubd_sync((lfs_emubd_t*)bd);
  226. }
  227. static int lfs_emubd_bd_info(void *bd, struct lfs_bd_info *info) {
  228. return lfs_emubd_info((lfs_emubd_t*)bd, info);
  229. }
  230. const struct lfs_bd_ops lfs_emubd_ops = {
  231. .read = lfs_emubd_bd_read,
  232. .prog = lfs_emubd_bd_prog,
  233. .erase = lfs_emubd_bd_erase,
  234. .sync = lfs_emubd_bd_sync,
  235. .info = lfs_emubd_bd_info,
  236. };