test_truncate.toml 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461
  1. [[case]] # simple truncate
  2. define.MEDIUMSIZE = [31, 32, 33, 511, 512, 513, 2047, 2048, 2049]
  3. define.LARGESIZE = [32, 33, 512, 513, 2048, 2049, 8192, 8193]
  4. if = 'MEDIUMSIZE < LARGESIZE'
  5. code = '''
  6. lfs_format(&lfs, &cfg) => 0;
  7. lfs_mount(&lfs, &cfg) => 0;
  8. lfs_file_open(&lfs, &file, "baldynoop",
  9. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  10. strcpy((char*)buffer, "hair");
  11. size = strlen((char*)buffer);
  12. for (lfs_off_t j = 0; j < LARGESIZE; j += size) {
  13. lfs_file_write(&lfs, &file, buffer, lfs_min(size, LARGESIZE-j))
  14. => lfs_min(size, LARGESIZE-j);
  15. }
  16. lfs_file_size(&lfs, &file) => LARGESIZE;
  17. lfs_file_close(&lfs, &file) => 0;
  18. lfs_unmount(&lfs) => 0;
  19. lfs_mount(&lfs, &cfg) => 0;
  20. lfs_file_open(&lfs, &file, "baldynoop", LFS_O_RDWR) => 0;
  21. lfs_file_size(&lfs, &file) => LARGESIZE;
  22. lfs_file_truncate(&lfs, &file, MEDIUMSIZE) => 0;
  23. lfs_file_size(&lfs, &file) => MEDIUMSIZE;
  24. lfs_file_close(&lfs, &file) => 0;
  25. lfs_unmount(&lfs) => 0;
  26. lfs_mount(&lfs, &cfg) => 0;
  27. lfs_file_open(&lfs, &file, "baldynoop", LFS_O_RDONLY) => 0;
  28. lfs_file_size(&lfs, &file) => MEDIUMSIZE;
  29. size = strlen("hair");
  30. for (lfs_off_t j = 0; j < MEDIUMSIZE; j += size) {
  31. lfs_file_read(&lfs, &file, buffer, lfs_min(size, MEDIUMSIZE-j))
  32. => lfs_min(size, MEDIUMSIZE-j);
  33. memcmp(buffer, "hair", lfs_min(size, MEDIUMSIZE-j)) => 0;
  34. }
  35. lfs_file_read(&lfs, &file, buffer, size) => 0;
  36. lfs_file_close(&lfs, &file) => 0;
  37. lfs_unmount(&lfs) => 0;
  38. '''
  39. [[case]] # truncate and read
  40. define.MEDIUMSIZE = [31, 32, 33, 511, 512, 513, 2047, 2048, 2049]
  41. define.LARGESIZE = [32, 33, 512, 513, 2048, 2049, 8192, 8193]
  42. if = 'MEDIUMSIZE < LARGESIZE'
  43. code = '''
  44. lfs_format(&lfs, &cfg) => 0;
  45. lfs_mount(&lfs, &cfg) => 0;
  46. lfs_file_open(&lfs, &file, "baldyread",
  47. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  48. strcpy((char*)buffer, "hair");
  49. size = strlen((char*)buffer);
  50. for (lfs_off_t j = 0; j < LARGESIZE; j += size) {
  51. lfs_file_write(&lfs, &file, buffer, lfs_min(size, LARGESIZE-j))
  52. => lfs_min(size, LARGESIZE-j);
  53. }
  54. lfs_file_size(&lfs, &file) => LARGESIZE;
  55. lfs_file_close(&lfs, &file) => 0;
  56. lfs_unmount(&lfs) => 0;
  57. lfs_mount(&lfs, &cfg) => 0;
  58. lfs_file_open(&lfs, &file, "baldyread", LFS_O_RDWR) => 0;
  59. lfs_file_size(&lfs, &file) => LARGESIZE;
  60. lfs_file_truncate(&lfs, &file, MEDIUMSIZE) => 0;
  61. lfs_file_size(&lfs, &file) => MEDIUMSIZE;
  62. size = strlen("hair");
  63. for (lfs_off_t j = 0; j < MEDIUMSIZE; j += size) {
  64. lfs_file_read(&lfs, &file, buffer, lfs_min(size, MEDIUMSIZE-j))
  65. => lfs_min(size, MEDIUMSIZE-j);
  66. memcmp(buffer, "hair", lfs_min(size, MEDIUMSIZE-j)) => 0;
  67. }
  68. lfs_file_read(&lfs, &file, buffer, size) => 0;
  69. lfs_file_close(&lfs, &file) => 0;
  70. lfs_unmount(&lfs) => 0;
  71. lfs_mount(&lfs, &cfg) => 0;
  72. lfs_file_open(&lfs, &file, "baldyread", LFS_O_RDONLY) => 0;
  73. lfs_file_size(&lfs, &file) => MEDIUMSIZE;
  74. size = strlen("hair");
  75. for (lfs_off_t j = 0; j < MEDIUMSIZE; j += size) {
  76. lfs_file_read(&lfs, &file, buffer, lfs_min(size, MEDIUMSIZE-j))
  77. => lfs_min(size, MEDIUMSIZE-j);
  78. memcmp(buffer, "hair", lfs_min(size, MEDIUMSIZE-j)) => 0;
  79. }
  80. lfs_file_read(&lfs, &file, buffer, size) => 0;
  81. lfs_file_close(&lfs, &file) => 0;
  82. lfs_unmount(&lfs) => 0;
  83. '''
  84. [[case]] # write, truncate, and read
  85. code = '''
  86. lfs_format(&lfs, &cfg) => 0;
  87. lfs_mount(&lfs, &cfg) => 0;
  88. lfs_file_open(&lfs, &file, "sequence",
  89. LFS_O_RDWR | LFS_O_CREAT | LFS_O_TRUNC) => 0;
  90. size = lfs_min(lfs.cfg->cache_size, sizeof(buffer)/2);
  91. lfs_size_t qsize = size / 4;
  92. uint8_t *wb = buffer;
  93. uint8_t *rb = buffer + size;
  94. for (lfs_off_t j = 0; j < size; ++j) {
  95. wb[j] = j;
  96. }
  97. /* Spread sequence over size */
  98. lfs_file_write(&lfs, &file, wb, size) => size;
  99. lfs_file_size(&lfs, &file) => size;
  100. lfs_file_tell(&lfs, &file) => size;
  101. lfs_file_seek(&lfs, &file, 0, LFS_SEEK_SET) => 0;
  102. lfs_file_tell(&lfs, &file) => 0;
  103. /* Chop off the last quarter */
  104. lfs_size_t trunc = size - qsize;
  105. lfs_file_truncate(&lfs, &file, trunc) => 0;
  106. lfs_file_tell(&lfs, &file) => 0;
  107. lfs_file_size(&lfs, &file) => trunc;
  108. /* Read should produce first 3/4 */
  109. lfs_file_read(&lfs, &file, rb, size) => trunc;
  110. memcmp(rb, wb, trunc) => 0;
  111. /* Move to 1/4 */
  112. lfs_file_size(&lfs, &file) => trunc;
  113. lfs_file_seek(&lfs, &file, qsize, LFS_SEEK_SET) => qsize;
  114. lfs_file_tell(&lfs, &file) => qsize;
  115. /* Chop to 1/2 */
  116. trunc -= qsize;
  117. lfs_file_truncate(&lfs, &file, trunc) => 0;
  118. lfs_file_tell(&lfs, &file) => qsize;
  119. lfs_file_size(&lfs, &file) => trunc;
  120. /* Read should produce second quarter */
  121. lfs_file_read(&lfs, &file, rb, size) => trunc - qsize;
  122. memcmp(rb, wb + qsize, trunc - qsize) => 0;
  123. lfs_file_close(&lfs, &file) => 0;
  124. lfs_unmount(&lfs) => 0;
  125. '''
  126. [[case]] # truncate and write
  127. define.MEDIUMSIZE = [31, 32, 33, 511, 512, 513, 2047, 2048, 2049]
  128. define.LARGESIZE = [32, 33, 512, 513, 2048, 2049, 8192, 8193]
  129. if = 'MEDIUMSIZE < LARGESIZE'
  130. code = '''
  131. lfs_format(&lfs, &cfg) => 0;
  132. lfs_mount(&lfs, &cfg) => 0;
  133. lfs_file_open(&lfs, &file, "baldywrite",
  134. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  135. strcpy((char*)buffer, "hair");
  136. size = strlen((char*)buffer);
  137. for (lfs_off_t j = 0; j < LARGESIZE; j += size) {
  138. lfs_file_write(&lfs, &file, buffer, lfs_min(size, LARGESIZE-j))
  139. => lfs_min(size, LARGESIZE-j);
  140. }
  141. lfs_file_size(&lfs, &file) => LARGESIZE;
  142. lfs_file_close(&lfs, &file) => 0;
  143. lfs_unmount(&lfs) => 0;
  144. lfs_mount(&lfs, &cfg) => 0;
  145. lfs_file_open(&lfs, &file, "baldywrite", LFS_O_RDWR) => 0;
  146. lfs_file_size(&lfs, &file) => LARGESIZE;
  147. /* truncate */
  148. lfs_file_truncate(&lfs, &file, MEDIUMSIZE) => 0;
  149. lfs_file_size(&lfs, &file) => MEDIUMSIZE;
  150. /* and write */
  151. strcpy((char*)buffer, "bald");
  152. size = strlen((char*)buffer);
  153. for (lfs_off_t j = 0; j < MEDIUMSIZE; j += size) {
  154. lfs_file_write(&lfs, &file, buffer, lfs_min(size, MEDIUMSIZE-j))
  155. => lfs_min(size, MEDIUMSIZE-j);
  156. }
  157. lfs_file_size(&lfs, &file) => MEDIUMSIZE;
  158. lfs_file_close(&lfs, &file) => 0;
  159. lfs_unmount(&lfs) => 0;
  160. lfs_mount(&lfs, &cfg) => 0;
  161. lfs_file_open(&lfs, &file, "baldywrite", LFS_O_RDONLY) => 0;
  162. lfs_file_size(&lfs, &file) => MEDIUMSIZE;
  163. size = strlen("bald");
  164. for (lfs_off_t j = 0; j < MEDIUMSIZE; j += size) {
  165. lfs_file_read(&lfs, &file, buffer, lfs_min(size, MEDIUMSIZE-j))
  166. => lfs_min(size, MEDIUMSIZE-j);
  167. memcmp(buffer, "bald", lfs_min(size, MEDIUMSIZE-j)) => 0;
  168. }
  169. lfs_file_read(&lfs, &file, buffer, size) => 0;
  170. lfs_file_close(&lfs, &file) => 0;
  171. lfs_unmount(&lfs) => 0;
  172. '''
  173. [[case]] # truncate write under powerloss
  174. define.SMALLSIZE = [4, 512]
  175. define.MEDIUMSIZE = [0, 3, 4, 5, 31, 32, 33, 511, 512, 513, 1023, 1024, 1025]
  176. define.LARGESIZE = 2048
  177. reentrant = true
  178. code = '''
  179. err = lfs_mount(&lfs, &cfg);
  180. if (err) {
  181. lfs_format(&lfs, &cfg) => 0;
  182. lfs_mount(&lfs, &cfg) => 0;
  183. }
  184. err = lfs_file_open(&lfs, &file, "baldy", LFS_O_RDONLY);
  185. assert(!err || err == LFS_ERR_NOENT);
  186. if (!err) {
  187. size = lfs_file_size(&lfs, &file);
  188. assert(size == 0 ||
  189. size == LARGESIZE ||
  190. size == MEDIUMSIZE ||
  191. size == SMALLSIZE);
  192. for (lfs_off_t j = 0; j < size; j += 4) {
  193. lfs_file_read(&lfs, &file, buffer, lfs_min(4, size-j))
  194. => lfs_min(4, size-j);
  195. assert(memcmp(buffer, "hair", lfs_min(4, size-j)) == 0 ||
  196. memcmp(buffer, "bald", lfs_min(4, size-j)) == 0 ||
  197. memcmp(buffer, "comb", lfs_min(4, size-j)) == 0);
  198. }
  199. lfs_file_close(&lfs, &file) => 0;
  200. }
  201. lfs_file_open(&lfs, &file, "baldy",
  202. LFS_O_WRONLY | LFS_O_CREAT | LFS_O_TRUNC) => 0;
  203. lfs_file_size(&lfs, &file) => 0;
  204. strcpy((char*)buffer, "hair");
  205. size = strlen((char*)buffer);
  206. for (lfs_off_t j = 0; j < LARGESIZE; j += size) {
  207. lfs_file_write(&lfs, &file, buffer, lfs_min(size, LARGESIZE-j))
  208. => lfs_min(size, LARGESIZE-j);
  209. }
  210. lfs_file_size(&lfs, &file) => LARGESIZE;
  211. lfs_file_close(&lfs, &file) => 0;
  212. lfs_file_open(&lfs, &file, "baldy", LFS_O_RDWR) => 0;
  213. lfs_file_size(&lfs, &file) => LARGESIZE;
  214. /* truncate */
  215. lfs_file_truncate(&lfs, &file, MEDIUMSIZE) => 0;
  216. lfs_file_size(&lfs, &file) => MEDIUMSIZE;
  217. /* and write */
  218. strcpy((char*)buffer, "bald");
  219. size = strlen((char*)buffer);
  220. for (lfs_off_t j = 0; j < MEDIUMSIZE; j += size) {
  221. lfs_file_write(&lfs, &file, buffer, lfs_min(size, MEDIUMSIZE-j))
  222. => lfs_min(size, MEDIUMSIZE-j);
  223. }
  224. lfs_file_size(&lfs, &file) => MEDIUMSIZE;
  225. lfs_file_close(&lfs, &file) => 0;
  226. lfs_file_open(&lfs, &file, "baldy", LFS_O_RDWR) => 0;
  227. lfs_file_size(&lfs, &file) => MEDIUMSIZE;
  228. lfs_file_truncate(&lfs, &file, SMALLSIZE) => 0;
  229. lfs_file_size(&lfs, &file) => SMALLSIZE;
  230. strcpy((char*)buffer, "comb");
  231. size = strlen((char*)buffer);
  232. for (lfs_off_t j = 0; j < SMALLSIZE; j += size) {
  233. lfs_file_write(&lfs, &file, buffer, lfs_min(size, SMALLSIZE-j))
  234. => lfs_min(size, SMALLSIZE-j);
  235. }
  236. lfs_file_size(&lfs, &file) => SMALLSIZE;
  237. lfs_file_close(&lfs, &file) => 0;
  238. lfs_unmount(&lfs) => 0;
  239. '''
  240. [[case]] # more aggressive general truncation tests
  241. define.CONFIG = 'range(6)'
  242. define.SMALLSIZE = 32
  243. define.MEDIUMSIZE = 2048
  244. define.LARGESIZE = 8192
  245. code = '''
  246. #define COUNT 5
  247. const struct {
  248. lfs_off_t startsizes[COUNT];
  249. lfs_off_t startseeks[COUNT];
  250. lfs_off_t hotsizes[COUNT];
  251. lfs_off_t coldsizes[COUNT];
  252. } configs[] = {
  253. // cold shrinking
  254. {{2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE},
  255. {2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE},
  256. {2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE},
  257. { 0, SMALLSIZE, MEDIUMSIZE, LARGESIZE, 2*LARGESIZE}},
  258. // cold expanding
  259. {{2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE},
  260. {2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE},
  261. {2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE},
  262. { 0, SMALLSIZE, MEDIUMSIZE, LARGESIZE, 2*LARGESIZE}},
  263. // warm shrinking truncate
  264. {{2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE},
  265. {2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE},
  266. { 0, SMALLSIZE, MEDIUMSIZE, LARGESIZE, 2*LARGESIZE},
  267. { 0, 0, 0, 0, 0}},
  268. // warm expanding truncate
  269. {{ 0, SMALLSIZE, MEDIUMSIZE, LARGESIZE, 2*LARGESIZE},
  270. { 0, SMALLSIZE, MEDIUMSIZE, LARGESIZE, 2*LARGESIZE},
  271. {2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE},
  272. {2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE}},
  273. // mid-file shrinking truncate
  274. {{2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE},
  275. { LARGESIZE, LARGESIZE, LARGESIZE, LARGESIZE, LARGESIZE},
  276. { 0, SMALLSIZE, MEDIUMSIZE, LARGESIZE, 2*LARGESIZE},
  277. { 0, 0, 0, 0, 0}},
  278. // mid-file expanding truncate
  279. {{ 0, SMALLSIZE, MEDIUMSIZE, LARGESIZE, 2*LARGESIZE},
  280. { 0, 0, SMALLSIZE, MEDIUMSIZE, LARGESIZE},
  281. {2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE},
  282. {2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE, 2*LARGESIZE}},
  283. };
  284. const lfs_off_t *startsizes = configs[CONFIG].startsizes;
  285. const lfs_off_t *startseeks = configs[CONFIG].startseeks;
  286. const lfs_off_t *hotsizes = configs[CONFIG].hotsizes;
  287. const lfs_off_t *coldsizes = configs[CONFIG].coldsizes;
  288. lfs_format(&lfs, &cfg) => 0;
  289. lfs_mount(&lfs, &cfg) => 0;
  290. for (unsigned i = 0; i < COUNT; i++) {
  291. sprintf(path, "hairyhead%d", i);
  292. lfs_file_open(&lfs, &file, path,
  293. LFS_O_WRONLY | LFS_O_CREAT | LFS_O_TRUNC) => 0;
  294. strcpy((char*)buffer, "hair");
  295. size = strlen((char*)buffer);
  296. for (lfs_off_t j = 0; j < startsizes[i]; j += size) {
  297. lfs_file_write(&lfs, &file, buffer, size) => size;
  298. }
  299. lfs_file_size(&lfs, &file) => startsizes[i];
  300. if (startseeks[i] != startsizes[i]) {
  301. lfs_file_seek(&lfs, &file,
  302. startseeks[i], LFS_SEEK_SET) => startseeks[i];
  303. }
  304. lfs_file_truncate(&lfs, &file, hotsizes[i]) => 0;
  305. lfs_file_size(&lfs, &file) => hotsizes[i];
  306. lfs_file_close(&lfs, &file) => 0;
  307. }
  308. lfs_unmount(&lfs) => 0;
  309. lfs_mount(&lfs, &cfg) => 0;
  310. for (unsigned i = 0; i < COUNT; i++) {
  311. sprintf(path, "hairyhead%d", i);
  312. lfs_file_open(&lfs, &file, path, LFS_O_RDWR) => 0;
  313. lfs_file_size(&lfs, &file) => hotsizes[i];
  314. size = strlen("hair");
  315. lfs_off_t j = 0;
  316. for (; j < startsizes[i] && j < hotsizes[i]; j += size) {
  317. lfs_file_read(&lfs, &file, buffer, size) => size;
  318. memcmp(buffer, "hair", size) => 0;
  319. }
  320. for (; j < hotsizes[i]; j += size) {
  321. lfs_file_read(&lfs, &file, buffer, size) => size;
  322. memcmp(buffer, "\0\0\0\0", size) => 0;
  323. }
  324. lfs_file_truncate(&lfs, &file, coldsizes[i]) => 0;
  325. lfs_file_size(&lfs, &file) => coldsizes[i];
  326. lfs_file_close(&lfs, &file) => 0;
  327. }
  328. lfs_unmount(&lfs) => 0;
  329. lfs_mount(&lfs, &cfg) => 0;
  330. for (unsigned i = 0; i < COUNT; i++) {
  331. sprintf(path, "hairyhead%d", i);
  332. lfs_file_open(&lfs, &file, path, LFS_O_RDONLY) => 0;
  333. lfs_file_size(&lfs, &file) => coldsizes[i];
  334. size = strlen("hair");
  335. lfs_off_t j = 0;
  336. for (; j < startsizes[i] && j < hotsizes[i] && j < coldsizes[i];
  337. j += size) {
  338. lfs_file_read(&lfs, &file, buffer, size) => size;
  339. memcmp(buffer, "hair", size) => 0;
  340. }
  341. for (; j < coldsizes[i]; j += size) {
  342. lfs_file_read(&lfs, &file, buffer, size) => size;
  343. memcmp(buffer, "\0\0\0\0", size) => 0;
  344. }
  345. lfs_file_close(&lfs, &file) => 0;
  346. }
  347. lfs_unmount(&lfs) => 0;
  348. '''
  349. [[case]] # noop truncate
  350. define.MEDIUMSIZE = [32, 33, 512, 513, 2048, 2049, 8192, 8193]
  351. code = '''
  352. lfs_format(&lfs, &cfg) => 0;
  353. lfs_mount(&lfs, &cfg) => 0;
  354. lfs_file_open(&lfs, &file, "baldynoop",
  355. LFS_O_RDWR | LFS_O_CREAT) => 0;
  356. strcpy((char*)buffer, "hair");
  357. size = strlen((char*)buffer);
  358. for (lfs_off_t j = 0; j < MEDIUMSIZE; j += size) {
  359. lfs_file_write(&lfs, &file, buffer, lfs_min(size, MEDIUMSIZE-j))
  360. => lfs_min(size, MEDIUMSIZE-j);
  361. // this truncate should do nothing
  362. lfs_file_truncate(&lfs, &file, j+lfs_min(size, MEDIUMSIZE-j)) => 0;
  363. }
  364. lfs_file_size(&lfs, &file) => MEDIUMSIZE;
  365. lfs_file_seek(&lfs, &file, 0, LFS_SEEK_SET) => 0;
  366. // should do nothing again
  367. lfs_file_truncate(&lfs, &file, MEDIUMSIZE) => 0;
  368. lfs_file_size(&lfs, &file) => MEDIUMSIZE;
  369. for (lfs_off_t j = 0; j < MEDIUMSIZE; j += size) {
  370. lfs_file_read(&lfs, &file, buffer, lfs_min(size, MEDIUMSIZE-j))
  371. => lfs_min(size, MEDIUMSIZE-j);
  372. memcmp(buffer, "hair", lfs_min(size, MEDIUMSIZE-j)) => 0;
  373. }
  374. lfs_file_read(&lfs, &file, buffer, size) => 0;
  375. lfs_file_close(&lfs, &file) => 0;
  376. lfs_unmount(&lfs) => 0;
  377. // still there after reboot?
  378. lfs_mount(&lfs, &cfg) => 0;
  379. lfs_file_open(&lfs, &file, "baldynoop", LFS_O_RDWR) => 0;
  380. lfs_file_size(&lfs, &file) => MEDIUMSIZE;
  381. for (lfs_off_t j = 0; j < MEDIUMSIZE; j += size) {
  382. lfs_file_read(&lfs, &file, buffer, lfs_min(size, MEDIUMSIZE-j))
  383. => lfs_min(size, MEDIUMSIZE-j);
  384. memcmp(buffer, "hair", lfs_min(size, MEDIUMSIZE-j)) => 0;
  385. }
  386. lfs_file_read(&lfs, &file, buffer, size) => 0;
  387. lfs_file_close(&lfs, &file) => 0;
  388. lfs_unmount(&lfs) => 0;
  389. '''