lfs_emubd.c 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287
  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. #include <assert.h>
  17. #include <stdbool.h>
  18. // Block device emulated on existing filesystem
  19. int lfs_emubd_create(lfs_emubd_t *emu, const char *path) {
  20. memset(&emu->info, 0, sizeof(emu->info));
  21. memset(&emu->stats, 0, sizeof(emu->stats));
  22. // Allocate buffer for creating children files
  23. size_t pathlen = strlen(path);
  24. emu->path = malloc(pathlen + 1 + LFS_NAME_MAX + 1);
  25. if (!emu->path) {
  26. return -ENOMEM;
  27. }
  28. strcpy(emu->path, path);
  29. emu->path[pathlen] = '/';
  30. emu->child = &emu->path[pathlen+1];
  31. memset(emu->child, '\0', LFS_NAME_MAX+1);
  32. // Create directory if it doesn't exist
  33. int err = mkdir(path, 0777);
  34. if (err && errno != EEXIST) {
  35. return -errno;
  36. }
  37. // Setup info based on configuration
  38. emu->info.read_size = LFS_EMUBD_READ_SIZE;
  39. emu->info.prog_size = LFS_EMUBD_PROG_SIZE;
  40. emu->info.erase_size = LFS_EMUBD_ERASE_SIZE;
  41. emu->info.total_size = LFS_EMUBD_TOTAL_SIZE;
  42. // Load stats to continue incrementing
  43. snprintf(emu->child, LFS_NAME_MAX, "stats");
  44. FILE *f = fopen(emu->path, "r");
  45. if (!f) {
  46. return -errno;
  47. }
  48. size_t res = fread(&emu->stats, sizeof(emu->stats), 1, f);
  49. if (res < 1) {
  50. return -errno;
  51. }
  52. err = fclose(f);
  53. if (err) {
  54. return -errno;
  55. }
  56. return 0;
  57. }
  58. void lfs_emubd_destroy(lfs_emubd_t *emu) {
  59. lfs_emubd_sync(emu);
  60. free(emu->path);
  61. }
  62. int lfs_emubd_read(lfs_emubd_t *emu, lfs_block_t block,
  63. lfs_off_t off, lfs_size_t size, void *buffer) {
  64. uint8_t *data = buffer;
  65. // Check if read is valid
  66. assert(off % emu->info.read_size == 0);
  67. assert(size % emu->info.read_size == 0);
  68. assert((uint64_t)block*emu->info.erase_size + off + size
  69. < emu->info.total_size);
  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, "%x", 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. assert(off % emu->info.prog_size == 0);
  107. assert(size % emu->info.prog_size == 0);
  108. assert((uint64_t)block*emu->info.erase_size + off + size
  109. < emu->info.total_size);
  110. // Iterate over blocks until enough data is read
  111. while (size > 0) {
  112. snprintf(emu->child, LFS_NAME_MAX, "%x", block);
  113. size_t count = lfs_min(emu->info.erase_size - off, size);
  114. FILE *f = fopen(emu->path, "r+b");
  115. if (!f && errno == ENOENT) {
  116. f = fopen(emu->path, "w+b");
  117. if (!f) {
  118. return -errno;
  119. }
  120. }
  121. int err = fseek(f, off, SEEK_SET);
  122. if (err) {
  123. return -errno;
  124. }
  125. size_t res = fwrite(data, 1, count, f);
  126. if (res < count) {
  127. return -errno;
  128. }
  129. err = fclose(f);
  130. if (err) {
  131. return -errno;
  132. }
  133. size -= count;
  134. data += count;
  135. block += 1;
  136. off = 0;
  137. }
  138. emu->stats.prog_count += 1;
  139. return 0;
  140. }
  141. int lfs_emubd_erase(lfs_emubd_t *emu, lfs_block_t block,
  142. lfs_off_t off, lfs_size_t size) {
  143. // Check if erase is valid
  144. assert(off % emu->info.erase_size == 0);
  145. assert(size % emu->info.erase_size == 0);
  146. assert((uint64_t)block*emu->info.erase_size + off + size
  147. < emu->info.total_size);
  148. // Iterate and erase blocks
  149. while (size > 0) {
  150. snprintf(emu->child, LFS_NAME_MAX, "%x", block);
  151. struct stat st;
  152. int err = stat(emu->path, &st);
  153. if (err && errno != ENOENT) {
  154. return -errno;
  155. }
  156. if (!err && S_ISREG(st.st_mode)) {
  157. int err = unlink(emu->path);
  158. if (err) {
  159. return -errno;
  160. }
  161. }
  162. size -= emu->info.erase_size;
  163. block += 1;
  164. off = 0;
  165. }
  166. emu->stats.erase_count += 1;
  167. return 0;
  168. }
  169. int lfs_emubd_sync(lfs_emubd_t *emu) {
  170. // Just write out info/stats for later lookup
  171. snprintf(emu->child, LFS_NAME_MAX, "info");
  172. FILE *f = fopen(emu->path, "w");
  173. if (!f) {
  174. return -errno;
  175. }
  176. size_t res = fwrite(&emu->info, sizeof(emu->info), 1, f);
  177. if (res < 1) {
  178. return -errno;
  179. }
  180. int err = fclose(f);
  181. if (err) {
  182. return -errno;
  183. }
  184. snprintf(emu->child, LFS_NAME_MAX, "stats");
  185. f = fopen(emu->path, "w");
  186. if (!f) {
  187. return -errno;
  188. }
  189. res = fwrite(&emu->stats, sizeof(emu->stats), 1, f);
  190. if (res < 1) {
  191. return -errno;
  192. }
  193. err = fclose(f);
  194. if (err) {
  195. return -errno;
  196. }
  197. return 0;
  198. }
  199. int lfs_emubd_info(lfs_emubd_t *emu, struct lfs_bd_info *info) {
  200. *info = emu->info;
  201. return 0;
  202. }
  203. int lfs_emubd_stats(lfs_emubd_t *emu, struct lfs_bd_stats *stats) {
  204. *stats = emu->stats;
  205. return 0;
  206. }
  207. // Wrappers for void*s
  208. static int lfs_emubd_bd_read(void *bd, lfs_block_t block,
  209. lfs_off_t off, lfs_size_t size, void *buffer) {
  210. return lfs_emubd_read((lfs_emubd_t*)bd, block, off, size, buffer);
  211. }
  212. static int lfs_emubd_bd_prog(void *bd, lfs_block_t block,
  213. lfs_off_t off, lfs_size_t size, const void *buffer) {
  214. return lfs_emubd_prog((lfs_emubd_t*)bd, block, off, size, buffer);
  215. }
  216. static int lfs_emubd_bd_erase(void *bd, lfs_block_t block,
  217. lfs_off_t off, lfs_size_t size) {
  218. return lfs_emubd_erase((lfs_emubd_t*)bd, block, off, size);
  219. }
  220. static int lfs_emubd_bd_sync(void *bd) {
  221. return lfs_emubd_sync((lfs_emubd_t*)bd);
  222. }
  223. static int lfs_emubd_bd_info(void *bd, struct lfs_bd_info *info) {
  224. return lfs_emubd_info((lfs_emubd_t*)bd, info);
  225. }
  226. const struct lfs_bd_ops lfs_emubd_ops = {
  227. .read = lfs_emubd_bd_read,
  228. .prog = lfs_emubd_bd_prog,
  229. .erase = lfs_emubd_bd_erase,
  230. .sync = lfs_emubd_bd_sync,
  231. .info = lfs_emubd_bd_info,
  232. };