test_move.toml 67 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795
  1. [[case]] # move file
  2. code = '''
  3. lfs_format(&lfs, &cfg) => 0;
  4. lfs_mount(&lfs, &cfg) => 0;
  5. lfs_mkdir(&lfs, "a") => 0;
  6. lfs_mkdir(&lfs, "b") => 0;
  7. lfs_mkdir(&lfs, "c") => 0;
  8. lfs_mkdir(&lfs, "d") => 0;
  9. lfs_file_open(&lfs, &file, "a/hello", LFS_O_CREAT | LFS_O_WRONLY) => 0;
  10. lfs_file_write(&lfs, &file, "hola\n", 5) => 5;
  11. lfs_file_write(&lfs, &file, "bonjour\n", 8) => 8;
  12. lfs_file_write(&lfs, &file, "ohayo\n", 6) => 6;
  13. lfs_file_close(&lfs, &file) => 0;
  14. lfs_unmount(&lfs) => 0;
  15. lfs_mount(&lfs, &cfg) => 0;
  16. lfs_rename(&lfs, "a/hello", "c/hello") => 0;
  17. lfs_unmount(&lfs) => 0;
  18. lfs_mount(&lfs, &cfg) => 0;
  19. lfs_dir_open(&lfs, &dir, "a") => 0;
  20. lfs_dir_read(&lfs, &dir, &info) => 1;
  21. assert(strcmp(info.name, ".") == 0);
  22. assert(info.type == LFS_TYPE_DIR);
  23. lfs_dir_read(&lfs, &dir, &info) => 1;
  24. assert(strcmp(info.name, "..") == 0);
  25. assert(info.type == LFS_TYPE_DIR);
  26. lfs_dir_read(&lfs, &dir, &info) => 0;
  27. lfs_dir_close(&lfs, &dir) => 0;
  28. lfs_dir_open(&lfs, &dir, "c") => 0;
  29. lfs_dir_read(&lfs, &dir, &info) => 1;
  30. assert(strcmp(info.name, ".") == 0);
  31. assert(info.type == LFS_TYPE_DIR);
  32. lfs_dir_read(&lfs, &dir, &info) => 1;
  33. assert(strcmp(info.name, "..") == 0);
  34. assert(info.type == LFS_TYPE_DIR);
  35. lfs_dir_read(&lfs, &dir, &info) => 1;
  36. assert(strcmp(info.name, "hello") == 0);
  37. assert(info.type == LFS_TYPE_REG);
  38. assert(info.size == 5+8+6);
  39. lfs_dir_read(&lfs, &dir, &info) => 0;
  40. lfs_dir_close(&lfs, &dir) => 0;
  41. lfs_file_open(&lfs, &file, "a/hello", LFS_O_RDONLY) => LFS_ERR_NOENT;
  42. lfs_file_open(&lfs, &file, "b/hello", LFS_O_RDONLY) => LFS_ERR_NOENT;
  43. lfs_file_open(&lfs, &file, "c/hello", LFS_O_RDONLY) => 0;
  44. lfs_file_read(&lfs, &file, buffer, 5) => 5;
  45. memcmp(buffer, "hola\n", 5) => 0;
  46. lfs_file_read(&lfs, &file, buffer, 8) => 8;
  47. memcmp(buffer, "bonjour\n", 8) => 0;
  48. lfs_file_read(&lfs, &file, buffer, 6) => 6;
  49. memcmp(buffer, "ohayo\n", 6) => 0;
  50. lfs_file_close(&lfs, &file) => 0;
  51. lfs_file_open(&lfs, &file, "d/hello", LFS_O_RDONLY) => LFS_ERR_NOENT;
  52. lfs_unmount(&lfs) => 0;
  53. '''
  54. [[case]] # move file corrupt source
  55. in = "lfs.c"
  56. code = '''
  57. lfs_format(&lfs, &cfg) => 0;
  58. lfs_mount(&lfs, &cfg) => 0;
  59. lfs_mkdir(&lfs, "a") => 0;
  60. lfs_mkdir(&lfs, "b") => 0;
  61. lfs_mkdir(&lfs, "c") => 0;
  62. lfs_mkdir(&lfs, "d") => 0;
  63. lfs_file_open(&lfs, &file, "a/hello", LFS_O_CREAT | LFS_O_WRONLY) => 0;
  64. lfs_file_write(&lfs, &file, "hola\n", 5) => 5;
  65. lfs_file_write(&lfs, &file, "bonjour\n", 8) => 8;
  66. lfs_file_write(&lfs, &file, "ohayo\n", 6) => 6;
  67. lfs_file_close(&lfs, &file) => 0;
  68. lfs_unmount(&lfs) => 0;
  69. lfs_mount(&lfs, &cfg) => 0;
  70. lfs_rename(&lfs, "a/hello", "c/hello") => 0;
  71. lfs_unmount(&lfs) => 0;
  72. // corrupt the source
  73. lfs_mount(&lfs, &cfg) => 0;
  74. lfs_dir_open(&lfs, &dir, "a") => 0;
  75. lfs_block_t block = dir.m.pair[0];
  76. lfs_dir_close(&lfs, &dir) => 0;
  77. lfs_unmount(&lfs) => 0;
  78. uint8_t bbuffer[LFS_BLOCK_SIZE];
  79. cfg.read(&cfg, block, 0, bbuffer, LFS_BLOCK_SIZE) => 0;
  80. int off = LFS_BLOCK_SIZE-1;
  81. while (off >= 0 && bbuffer[off] == LFS_ERASE_VALUE) {
  82. off -= 1;
  83. }
  84. memset(&bbuffer[off-3], LFS_BLOCK_SIZE, 3);
  85. cfg.erase(&cfg, block) => 0;
  86. cfg.prog(&cfg, block, 0, bbuffer, LFS_BLOCK_SIZE) => 0;
  87. cfg.sync(&cfg) => 0;
  88. lfs_mount(&lfs, &cfg) => 0;
  89. lfs_dir_open(&lfs, &dir, "a") => 0;
  90. lfs_dir_read(&lfs, &dir, &info) => 1;
  91. assert(strcmp(info.name, ".") == 0);
  92. assert(info.type == LFS_TYPE_DIR);
  93. lfs_dir_read(&lfs, &dir, &info) => 1;
  94. assert(strcmp(info.name, "..") == 0);
  95. assert(info.type == LFS_TYPE_DIR);
  96. lfs_dir_read(&lfs, &dir, &info) => 0;
  97. lfs_dir_close(&lfs, &dir) => 0;
  98. lfs_dir_open(&lfs, &dir, "c") => 0;
  99. lfs_dir_read(&lfs, &dir, &info) => 1;
  100. assert(strcmp(info.name, ".") == 0);
  101. assert(info.type == LFS_TYPE_DIR);
  102. lfs_dir_read(&lfs, &dir, &info) => 1;
  103. assert(strcmp(info.name, "..") == 0);
  104. assert(info.type == LFS_TYPE_DIR);
  105. lfs_dir_read(&lfs, &dir, &info) => 1;
  106. assert(strcmp(info.name, "hello") == 0);
  107. assert(info.type == LFS_TYPE_REG);
  108. assert(info.size == 5+8+6);
  109. lfs_dir_read(&lfs, &dir, &info) => 0;
  110. lfs_dir_close(&lfs, &dir) => 0;
  111. lfs_file_open(&lfs, &file, "a/hello", LFS_O_RDONLY) => LFS_ERR_NOENT;
  112. lfs_file_open(&lfs, &file, "b/hello", LFS_O_RDONLY) => LFS_ERR_NOENT;
  113. lfs_file_open(&lfs, &file, "c/hello", LFS_O_RDONLY) => 0;
  114. lfs_file_read(&lfs, &file, buffer, 5) => 5;
  115. memcmp(buffer, "hola\n", 5) => 0;
  116. lfs_file_read(&lfs, &file, buffer, 8) => 8;
  117. memcmp(buffer, "bonjour\n", 8) => 0;
  118. lfs_file_read(&lfs, &file, buffer, 6) => 6;
  119. memcmp(buffer, "ohayo\n", 6) => 0;
  120. lfs_file_close(&lfs, &file) => 0;
  121. lfs_file_open(&lfs, &file, "d/hello", LFS_O_RDONLY) => LFS_ERR_NOENT;
  122. lfs_unmount(&lfs) => 0;
  123. '''
  124. [[case]] # move file corrupt source and dest
  125. in = "lfs.c"
  126. code = '''
  127. lfs_format(&lfs, &cfg) => 0;
  128. lfs_mount(&lfs, &cfg) => 0;
  129. lfs_mkdir(&lfs, "a") => 0;
  130. lfs_mkdir(&lfs, "b") => 0;
  131. lfs_mkdir(&lfs, "c") => 0;
  132. lfs_mkdir(&lfs, "d") => 0;
  133. lfs_file_open(&lfs, &file, "a/hello", LFS_O_CREAT | LFS_O_WRONLY) => 0;
  134. lfs_file_write(&lfs, &file, "hola\n", 5) => 5;
  135. lfs_file_write(&lfs, &file, "bonjour\n", 8) => 8;
  136. lfs_file_write(&lfs, &file, "ohayo\n", 6) => 6;
  137. lfs_file_close(&lfs, &file) => 0;
  138. lfs_unmount(&lfs) => 0;
  139. lfs_mount(&lfs, &cfg) => 0;
  140. lfs_rename(&lfs, "a/hello", "c/hello") => 0;
  141. lfs_unmount(&lfs) => 0;
  142. // corrupt the source
  143. lfs_mount(&lfs, &cfg) => 0;
  144. lfs_dir_open(&lfs, &dir, "a") => 0;
  145. lfs_block_t block = dir.m.pair[0];
  146. lfs_dir_close(&lfs, &dir) => 0;
  147. lfs_unmount(&lfs) => 0;
  148. uint8_t bbuffer[LFS_BLOCK_SIZE];
  149. cfg.read(&cfg, block, 0, bbuffer, LFS_BLOCK_SIZE) => 0;
  150. int off = LFS_BLOCK_SIZE-1;
  151. while (off >= 0 && bbuffer[off] == LFS_ERASE_VALUE) {
  152. off -= 1;
  153. }
  154. memset(&bbuffer[off-3], LFS_BLOCK_SIZE, 3);
  155. cfg.erase(&cfg, block) => 0;
  156. cfg.prog(&cfg, block, 0, bbuffer, LFS_BLOCK_SIZE) => 0;
  157. cfg.sync(&cfg) => 0;
  158. // corrupt the destination
  159. lfs_mount(&lfs, &cfg) => 0;
  160. lfs_dir_open(&lfs, &dir, "c") => 0;
  161. block = dir.m.pair[0];
  162. lfs_dir_close(&lfs, &dir) => 0;
  163. lfs_unmount(&lfs) => 0;
  164. cfg.read(&cfg, block, 0, bbuffer, LFS_BLOCK_SIZE) => 0;
  165. off = LFS_BLOCK_SIZE-1;
  166. while (off >= 0 && bbuffer[off] == LFS_ERASE_VALUE) {
  167. off -= 1;
  168. }
  169. memset(&bbuffer[off-3], LFS_BLOCK_SIZE, 3);
  170. cfg.erase(&cfg, block) => 0;
  171. cfg.prog(&cfg, block, 0, bbuffer, LFS_BLOCK_SIZE) => 0;
  172. cfg.sync(&cfg) => 0;
  173. lfs_mount(&lfs, &cfg) => 0;
  174. lfs_dir_open(&lfs, &dir, "a") => 0;
  175. lfs_dir_read(&lfs, &dir, &info) => 1;
  176. assert(strcmp(info.name, ".") == 0);
  177. assert(info.type == LFS_TYPE_DIR);
  178. lfs_dir_read(&lfs, &dir, &info) => 1;
  179. assert(strcmp(info.name, "..") == 0);
  180. assert(info.type == LFS_TYPE_DIR);
  181. lfs_dir_read(&lfs, &dir, &info) => 1;
  182. assert(strcmp(info.name, "hello") == 0);
  183. assert(info.type == LFS_TYPE_REG);
  184. assert(info.size == 5+8+6);
  185. lfs_dir_read(&lfs, &dir, &info) => 0;
  186. lfs_dir_close(&lfs, &dir) => 0;
  187. lfs_dir_open(&lfs, &dir, "c") => 0;
  188. lfs_dir_read(&lfs, &dir, &info) => 1;
  189. assert(strcmp(info.name, ".") == 0);
  190. assert(info.type == LFS_TYPE_DIR);
  191. lfs_dir_read(&lfs, &dir, &info) => 1;
  192. assert(strcmp(info.name, "..") == 0);
  193. assert(info.type == LFS_TYPE_DIR);
  194. lfs_dir_read(&lfs, &dir, &info) => 0;
  195. lfs_dir_close(&lfs, &dir) => 0;
  196. lfs_file_open(&lfs, &file, "a/hello", LFS_O_RDONLY) => 0;
  197. lfs_file_read(&lfs, &file, buffer, 5) => 5;
  198. memcmp(buffer, "hola\n", 5) => 0;
  199. lfs_file_read(&lfs, &file, buffer, 8) => 8;
  200. memcmp(buffer, "bonjour\n", 8) => 0;
  201. lfs_file_read(&lfs, &file, buffer, 6) => 6;
  202. memcmp(buffer, "ohayo\n", 6) => 0;
  203. lfs_file_close(&lfs, &file) => 0;
  204. lfs_file_open(&lfs, &file, "b/hello", LFS_O_RDONLY) => LFS_ERR_NOENT;
  205. lfs_file_open(&lfs, &file, "c/hello", LFS_O_RDONLY) => LFS_ERR_NOENT;
  206. lfs_file_open(&lfs, &file, "d/hello", LFS_O_RDONLY) => LFS_ERR_NOENT;
  207. lfs_unmount(&lfs) => 0;
  208. '''
  209. [[case]] # move file after corrupt
  210. in = "lfs.c"
  211. code = '''
  212. lfs_format(&lfs, &cfg) => 0;
  213. lfs_mount(&lfs, &cfg) => 0;
  214. lfs_mkdir(&lfs, "a") => 0;
  215. lfs_mkdir(&lfs, "b") => 0;
  216. lfs_mkdir(&lfs, "c") => 0;
  217. lfs_mkdir(&lfs, "d") => 0;
  218. lfs_file_open(&lfs, &file, "a/hello", LFS_O_CREAT | LFS_O_WRONLY) => 0;
  219. lfs_file_write(&lfs, &file, "hola\n", 5) => 5;
  220. lfs_file_write(&lfs, &file, "bonjour\n", 8) => 8;
  221. lfs_file_write(&lfs, &file, "ohayo\n", 6) => 6;
  222. lfs_file_close(&lfs, &file) => 0;
  223. lfs_unmount(&lfs) => 0;
  224. lfs_mount(&lfs, &cfg) => 0;
  225. lfs_rename(&lfs, "a/hello", "c/hello") => 0;
  226. lfs_unmount(&lfs) => 0;
  227. // corrupt the source
  228. lfs_mount(&lfs, &cfg) => 0;
  229. lfs_dir_open(&lfs, &dir, "a") => 0;
  230. lfs_block_t block = dir.m.pair[0];
  231. lfs_dir_close(&lfs, &dir) => 0;
  232. lfs_unmount(&lfs) => 0;
  233. uint8_t bbuffer[LFS_BLOCK_SIZE];
  234. cfg.read(&cfg, block, 0, bbuffer, LFS_BLOCK_SIZE) => 0;
  235. int off = LFS_BLOCK_SIZE-1;
  236. while (off >= 0 && bbuffer[off] == LFS_ERASE_VALUE) {
  237. off -= 1;
  238. }
  239. memset(&bbuffer[off-3], LFS_BLOCK_SIZE, 3);
  240. cfg.erase(&cfg, block) => 0;
  241. cfg.prog(&cfg, block, 0, bbuffer, LFS_BLOCK_SIZE) => 0;
  242. cfg.sync(&cfg) => 0;
  243. // corrupt the destination
  244. lfs_mount(&lfs, &cfg) => 0;
  245. lfs_dir_open(&lfs, &dir, "c") => 0;
  246. block = dir.m.pair[0];
  247. lfs_dir_close(&lfs, &dir) => 0;
  248. lfs_unmount(&lfs) => 0;
  249. cfg.read(&cfg, block, 0, bbuffer, LFS_BLOCK_SIZE) => 0;
  250. off = LFS_BLOCK_SIZE-1;
  251. while (off >= 0 && bbuffer[off] == LFS_ERASE_VALUE) {
  252. off -= 1;
  253. }
  254. memset(&bbuffer[off-3], LFS_BLOCK_SIZE, 3);
  255. cfg.erase(&cfg, block) => 0;
  256. cfg.prog(&cfg, block, 0, bbuffer, LFS_BLOCK_SIZE) => 0;
  257. cfg.sync(&cfg) => 0;
  258. // continue move
  259. lfs_mount(&lfs, &cfg) => 0;
  260. lfs_rename(&lfs, "a/hello", "c/hello") => 0;
  261. lfs_unmount(&lfs) => 0;
  262. lfs_mount(&lfs, &cfg) => 0;
  263. lfs_dir_open(&lfs, &dir, "a") => 0;
  264. lfs_dir_read(&lfs, &dir, &info) => 1;
  265. assert(strcmp(info.name, ".") == 0);
  266. assert(info.type == LFS_TYPE_DIR);
  267. lfs_dir_read(&lfs, &dir, &info) => 1;
  268. assert(strcmp(info.name, "..") == 0);
  269. assert(info.type == LFS_TYPE_DIR);
  270. lfs_dir_read(&lfs, &dir, &info) => 0;
  271. lfs_dir_close(&lfs, &dir) => 0;
  272. lfs_dir_open(&lfs, &dir, "c") => 0;
  273. lfs_dir_read(&lfs, &dir, &info) => 1;
  274. assert(strcmp(info.name, ".") == 0);
  275. assert(info.type == LFS_TYPE_DIR);
  276. lfs_dir_read(&lfs, &dir, &info) => 1;
  277. assert(strcmp(info.name, "..") == 0);
  278. assert(info.type == LFS_TYPE_DIR);
  279. lfs_dir_read(&lfs, &dir, &info) => 1;
  280. assert(strcmp(info.name, "hello") == 0);
  281. assert(info.type == LFS_TYPE_REG);
  282. assert(info.size == 5+8+6);
  283. lfs_dir_read(&lfs, &dir, &info) => 0;
  284. lfs_dir_close(&lfs, &dir) => 0;
  285. lfs_file_open(&lfs, &file, "a/hello", LFS_O_RDONLY) => LFS_ERR_NOENT;
  286. lfs_file_open(&lfs, &file, "b/hello", LFS_O_RDONLY) => LFS_ERR_NOENT;
  287. lfs_file_open(&lfs, &file, "c/hello", LFS_O_RDONLY) => 0;
  288. lfs_file_read(&lfs, &file, buffer, 5) => 5;
  289. memcmp(buffer, "hola\n", 5) => 0;
  290. lfs_file_read(&lfs, &file, buffer, 8) => 8;
  291. memcmp(buffer, "bonjour\n", 8) => 0;
  292. lfs_file_read(&lfs, &file, buffer, 6) => 6;
  293. memcmp(buffer, "ohayo\n", 6) => 0;
  294. lfs_file_close(&lfs, &file) => 0;
  295. lfs_file_open(&lfs, &file, "d/hello", LFS_O_RDONLY) => LFS_ERR_NOENT;
  296. lfs_unmount(&lfs) => 0;
  297. '''
  298. [[case]] # simple reentrant move file
  299. reentrant = true
  300. code = '''
  301. err = lfs_mount(&lfs, &cfg);
  302. if (err) {
  303. lfs_format(&lfs, &cfg) => 0;
  304. lfs_mount(&lfs, &cfg) => 0;
  305. }
  306. err = lfs_mkdir(&lfs, "a");
  307. assert(!err || err == LFS_ERR_EXIST);
  308. err = lfs_mkdir(&lfs, "b");
  309. assert(!err || err == LFS_ERR_EXIST);
  310. err = lfs_mkdir(&lfs, "c");
  311. assert(!err || err == LFS_ERR_EXIST);
  312. err = lfs_mkdir(&lfs, "d");
  313. assert(!err || err == LFS_ERR_EXIST);
  314. lfs_unmount(&lfs) => 0;
  315. while (true) {
  316. lfs_mount(&lfs, &cfg) => 0;
  317. // there should never exist _2_ hello files
  318. int count = 0;
  319. if (lfs_stat(&lfs, "a/hello", &info) == 0) {
  320. assert(strcmp(info.name, "hello") == 0);
  321. assert(info.type == LFS_TYPE_REG);
  322. assert(info.size == 5+8+6 || info.size == 0);
  323. count += 1;
  324. }
  325. if (lfs_stat(&lfs, "b/hello", &info) == 0) {
  326. assert(strcmp(info.name, "hello") == 0);
  327. assert(info.type == LFS_TYPE_REG);
  328. assert(info.size == 5+8+6);
  329. count += 1;
  330. }
  331. if (lfs_stat(&lfs, "c/hello", &info) == 0) {
  332. assert(strcmp(info.name, "hello") == 0);
  333. assert(info.type == LFS_TYPE_REG);
  334. assert(info.size == 5+8+6);
  335. count += 1;
  336. }
  337. if (lfs_stat(&lfs, "d/hello", &info) == 0) {
  338. assert(strcmp(info.name, "hello") == 0);
  339. assert(info.type == LFS_TYPE_REG);
  340. assert(info.size == 5+8+6);
  341. count += 1;
  342. }
  343. assert(count <= 1);
  344. lfs_unmount(&lfs) => 0;
  345. lfs_mount(&lfs, &cfg) => 0;
  346. if (lfs_stat(&lfs, "a/hello", &info) == 0 && info.size > 0) {
  347. lfs_rename(&lfs, "a/hello", "b/hello") => 0;
  348. } else if (lfs_stat(&lfs, "b/hello", &info) == 0) {
  349. lfs_rename(&lfs, "b/hello", "c/hello") => 0;
  350. } else if (lfs_stat(&lfs, "c/hello", &info) == 0) {
  351. lfs_rename(&lfs, "c/hello", "d/hello") => 0;
  352. } else if (lfs_stat(&lfs, "d/hello", &info) == 0) {
  353. // success
  354. lfs_unmount(&lfs) => 0;
  355. break;
  356. } else {
  357. // create file
  358. lfs_file_open(&lfs, &file, "a/hello",
  359. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  360. lfs_file_write(&lfs, &file, "hola\n", 5) => 5;
  361. lfs_file_write(&lfs, &file, "bonjour\n", 8) => 8;
  362. lfs_file_write(&lfs, &file, "ohayo\n", 6) => 6;
  363. lfs_file_close(&lfs, &file) => 0;
  364. }
  365. lfs_unmount(&lfs) => 0;
  366. }
  367. lfs_mount(&lfs, &cfg) => 0;
  368. lfs_dir_open(&lfs, &dir, "a") => 0;
  369. lfs_dir_read(&lfs, &dir, &info) => 1;
  370. assert(strcmp(info.name, ".") == 0);
  371. assert(info.type == LFS_TYPE_DIR);
  372. lfs_dir_read(&lfs, &dir, &info) => 1;
  373. assert(strcmp(info.name, "..") == 0);
  374. assert(info.type == LFS_TYPE_DIR);
  375. lfs_dir_read(&lfs, &dir, &info) => 0;
  376. lfs_dir_close(&lfs, &dir) => 0;
  377. lfs_dir_open(&lfs, &dir, "d") => 0;
  378. lfs_dir_read(&lfs, &dir, &info) => 1;
  379. assert(strcmp(info.name, ".") == 0);
  380. assert(info.type == LFS_TYPE_DIR);
  381. lfs_dir_read(&lfs, &dir, &info) => 1;
  382. assert(strcmp(info.name, "..") == 0);
  383. assert(info.type == LFS_TYPE_DIR);
  384. lfs_dir_read(&lfs, &dir, &info) => 1;
  385. assert(strcmp(info.name, "hello") == 0);
  386. assert(info.type == LFS_TYPE_REG);
  387. assert(info.size == 5+8+6);
  388. lfs_dir_read(&lfs, &dir, &info) => 0;
  389. lfs_dir_close(&lfs, &dir) => 0;
  390. lfs_file_open(&lfs, &file, "a/hello", LFS_O_RDONLY) => LFS_ERR_NOENT;
  391. lfs_file_open(&lfs, &file, "b/hello", LFS_O_RDONLY) => LFS_ERR_NOENT;
  392. lfs_file_open(&lfs, &file, "c/hello", LFS_O_RDONLY) => LFS_ERR_NOENT;
  393. lfs_file_open(&lfs, &file, "d/hello", LFS_O_RDONLY) => 0;
  394. lfs_file_read(&lfs, &file, buffer, 5) => 5;
  395. memcmp(buffer, "hola\n", 5) => 0;
  396. lfs_file_read(&lfs, &file, buffer, 8) => 8;
  397. memcmp(buffer, "bonjour\n", 8) => 0;
  398. lfs_file_read(&lfs, &file, buffer, 6) => 6;
  399. memcmp(buffer, "ohayo\n", 6) => 0;
  400. lfs_file_close(&lfs, &file) => 0;
  401. lfs_unmount(&lfs) => 0;
  402. '''
  403. [[case]] # move dir
  404. code = '''
  405. lfs_format(&lfs, &cfg) => 0;
  406. lfs_mount(&lfs, &cfg) => 0;
  407. lfs_mkdir(&lfs, "a") => 0;
  408. lfs_mkdir(&lfs, "b") => 0;
  409. lfs_mkdir(&lfs, "c") => 0;
  410. lfs_mkdir(&lfs, "d") => 0;
  411. lfs_mkdir(&lfs, "a/hi") => 0;
  412. lfs_mkdir(&lfs, "a/hi/hola") => 0;
  413. lfs_mkdir(&lfs, "a/hi/bonjour") => 0;
  414. lfs_mkdir(&lfs, "a/hi/ohayo") => 0;
  415. lfs_unmount(&lfs) => 0;
  416. lfs_mount(&lfs, &cfg) => 0;
  417. lfs_rename(&lfs, "a/hi", "c/hi") => 0;
  418. lfs_unmount(&lfs) => 0;
  419. lfs_mount(&lfs, &cfg) => 0;
  420. lfs_dir_open(&lfs, &dir, "a") => 0;
  421. lfs_dir_read(&lfs, &dir, &info) => 1;
  422. assert(strcmp(info.name, ".") == 0);
  423. assert(info.type == LFS_TYPE_DIR);
  424. lfs_dir_read(&lfs, &dir, &info) => 1;
  425. assert(strcmp(info.name, "..") == 0);
  426. assert(info.type == LFS_TYPE_DIR);
  427. lfs_dir_read(&lfs, &dir, &info) => 0;
  428. lfs_dir_close(&lfs, &dir) => 0;
  429. lfs_dir_open(&lfs, &dir, "c") => 0;
  430. lfs_dir_read(&lfs, &dir, &info) => 1;
  431. assert(strcmp(info.name, ".") == 0);
  432. assert(info.type == LFS_TYPE_DIR);
  433. lfs_dir_read(&lfs, &dir, &info) => 1;
  434. assert(strcmp(info.name, "..") == 0);
  435. assert(info.type == LFS_TYPE_DIR);
  436. lfs_dir_read(&lfs, &dir, &info) => 1;
  437. assert(strcmp(info.name, "hi") == 0);
  438. assert(info.type == LFS_TYPE_DIR);
  439. lfs_dir_read(&lfs, &dir, &info) => 0;
  440. lfs_dir_close(&lfs, &dir) => 0;
  441. lfs_dir_open(&lfs, &dir, "a/hi") => LFS_ERR_NOENT;
  442. lfs_dir_open(&lfs, &dir, "b/hi") => LFS_ERR_NOENT;
  443. lfs_dir_open(&lfs, &dir, "c/hi") => 0;
  444. lfs_dir_read(&lfs, &dir, &info) => 1;
  445. assert(strcmp(info.name, ".") == 0);
  446. assert(info.type == LFS_TYPE_DIR);
  447. lfs_dir_read(&lfs, &dir, &info) => 1;
  448. assert(strcmp(info.name, "..") == 0);
  449. assert(info.type == LFS_TYPE_DIR);
  450. lfs_dir_read(&lfs, &dir, &info) => 1;
  451. assert(strcmp(info.name, "bonjour") == 0);
  452. assert(info.type == LFS_TYPE_DIR);
  453. lfs_dir_read(&lfs, &dir, &info) => 1;
  454. assert(strcmp(info.name, "hola") == 0);
  455. assert(info.type == LFS_TYPE_DIR);
  456. lfs_dir_read(&lfs, &dir, &info) => 1;
  457. assert(strcmp(info.name, "ohayo") == 0);
  458. assert(info.type == LFS_TYPE_DIR);
  459. lfs_dir_read(&lfs, &dir, &info) => 0;
  460. lfs_dir_close(&lfs, &dir) => 0;
  461. lfs_dir_open(&lfs, &dir, "d/hi") => LFS_ERR_NOENT;
  462. lfs_unmount(&lfs) => 0;
  463. '''
  464. [[case]] # move dir corrupt source
  465. in = "lfs.c"
  466. code = '''
  467. lfs_format(&lfs, &cfg) => 0;
  468. lfs_mount(&lfs, &cfg) => 0;
  469. lfs_mkdir(&lfs, "a") => 0;
  470. lfs_mkdir(&lfs, "b") => 0;
  471. lfs_mkdir(&lfs, "c") => 0;
  472. lfs_mkdir(&lfs, "d") => 0;
  473. lfs_mkdir(&lfs, "a/hi") => 0;
  474. lfs_mkdir(&lfs, "a/hi/hola") => 0;
  475. lfs_mkdir(&lfs, "a/hi/bonjour") => 0;
  476. lfs_mkdir(&lfs, "a/hi/ohayo") => 0;
  477. lfs_unmount(&lfs) => 0;
  478. lfs_mount(&lfs, &cfg) => 0;
  479. lfs_rename(&lfs, "a/hi", "c/hi") => 0;
  480. lfs_unmount(&lfs) => 0;
  481. // corrupt the source
  482. lfs_mount(&lfs, &cfg) => 0;
  483. lfs_dir_open(&lfs, &dir, "a") => 0;
  484. lfs_block_t block = dir.m.pair[0];
  485. lfs_dir_close(&lfs, &dir) => 0;
  486. lfs_unmount(&lfs) => 0;
  487. uint8_t bbuffer[LFS_BLOCK_SIZE];
  488. cfg.read(&cfg, block, 0, bbuffer, LFS_BLOCK_SIZE) => 0;
  489. int off = LFS_BLOCK_SIZE-1;
  490. while (off >= 0 && bbuffer[off] == LFS_ERASE_VALUE) {
  491. off -= 1;
  492. }
  493. memset(&bbuffer[off-3], LFS_BLOCK_SIZE, 3);
  494. cfg.erase(&cfg, block) => 0;
  495. cfg.prog(&cfg, block, 0, bbuffer, LFS_BLOCK_SIZE) => 0;
  496. cfg.sync(&cfg) => 0;
  497. lfs_mount(&lfs, &cfg) => 0;
  498. lfs_dir_open(&lfs, &dir, "a") => 0;
  499. lfs_dir_read(&lfs, &dir, &info) => 1;
  500. assert(strcmp(info.name, ".") == 0);
  501. assert(info.type == LFS_TYPE_DIR);
  502. lfs_dir_read(&lfs, &dir, &info) => 1;
  503. assert(strcmp(info.name, "..") == 0);
  504. assert(info.type == LFS_TYPE_DIR);
  505. lfs_dir_read(&lfs, &dir, &info) => 0;
  506. lfs_dir_close(&lfs, &dir) => 0;
  507. lfs_dir_open(&lfs, &dir, "c") => 0;
  508. lfs_dir_read(&lfs, &dir, &info) => 1;
  509. assert(strcmp(info.name, ".") == 0);
  510. assert(info.type == LFS_TYPE_DIR);
  511. lfs_dir_read(&lfs, &dir, &info) => 1;
  512. assert(strcmp(info.name, "..") == 0);
  513. assert(info.type == LFS_TYPE_DIR);
  514. lfs_dir_read(&lfs, &dir, &info) => 1;
  515. assert(strcmp(info.name, "hi") == 0);
  516. assert(info.type == LFS_TYPE_DIR);
  517. lfs_dir_read(&lfs, &dir, &info) => 0;
  518. lfs_dir_close(&lfs, &dir) => 0;
  519. lfs_dir_open(&lfs, &dir, "a/hi") => LFS_ERR_NOENT;
  520. lfs_dir_open(&lfs, &dir, "b/hi") => LFS_ERR_NOENT;
  521. lfs_dir_open(&lfs, &dir, "c/hi") => 0;
  522. lfs_dir_read(&lfs, &dir, &info) => 1;
  523. assert(strcmp(info.name, ".") == 0);
  524. assert(info.type == LFS_TYPE_DIR);
  525. lfs_dir_read(&lfs, &dir, &info) => 1;
  526. assert(strcmp(info.name, "..") == 0);
  527. assert(info.type == LFS_TYPE_DIR);
  528. lfs_dir_read(&lfs, &dir, &info) => 1;
  529. assert(strcmp(info.name, "bonjour") == 0);
  530. assert(info.type == LFS_TYPE_DIR);
  531. lfs_dir_read(&lfs, &dir, &info) => 1;
  532. assert(strcmp(info.name, "hola") == 0);
  533. assert(info.type == LFS_TYPE_DIR);
  534. lfs_dir_read(&lfs, &dir, &info) => 1;
  535. assert(strcmp(info.name, "ohayo") == 0);
  536. assert(info.type == LFS_TYPE_DIR);
  537. lfs_dir_read(&lfs, &dir, &info) => 0;
  538. lfs_dir_close(&lfs, &dir) => 0;
  539. lfs_dir_open(&lfs, &dir, "d/hi") => LFS_ERR_NOENT;
  540. lfs_unmount(&lfs) => 0;
  541. '''
  542. [[case]] # move dir corrupt source and dest
  543. in = "lfs.c"
  544. code = '''
  545. lfs_format(&lfs, &cfg) => 0;
  546. lfs_mount(&lfs, &cfg) => 0;
  547. lfs_mkdir(&lfs, "a") => 0;
  548. lfs_mkdir(&lfs, "b") => 0;
  549. lfs_mkdir(&lfs, "c") => 0;
  550. lfs_mkdir(&lfs, "d") => 0;
  551. lfs_mkdir(&lfs, "a/hi") => 0;
  552. lfs_mkdir(&lfs, "a/hi/hola") => 0;
  553. lfs_mkdir(&lfs, "a/hi/bonjour") => 0;
  554. lfs_mkdir(&lfs, "a/hi/ohayo") => 0;
  555. lfs_unmount(&lfs) => 0;
  556. lfs_mount(&lfs, &cfg) => 0;
  557. lfs_rename(&lfs, "a/hi", "c/hi") => 0;
  558. lfs_unmount(&lfs) => 0;
  559. // corrupt the source
  560. lfs_mount(&lfs, &cfg) => 0;
  561. lfs_dir_open(&lfs, &dir, "a") => 0;
  562. lfs_block_t block = dir.m.pair[0];
  563. lfs_dir_close(&lfs, &dir) => 0;
  564. lfs_unmount(&lfs) => 0;
  565. uint8_t bbuffer[LFS_BLOCK_SIZE];
  566. cfg.read(&cfg, block, 0, bbuffer, LFS_BLOCK_SIZE) => 0;
  567. int off = LFS_BLOCK_SIZE-1;
  568. while (off >= 0 && bbuffer[off] == LFS_ERASE_VALUE) {
  569. off -= 1;
  570. }
  571. memset(&bbuffer[off-3], LFS_BLOCK_SIZE, 3);
  572. cfg.erase(&cfg, block) => 0;
  573. cfg.prog(&cfg, block, 0, bbuffer, LFS_BLOCK_SIZE) => 0;
  574. cfg.sync(&cfg) => 0;
  575. // corrupt the destination
  576. lfs_mount(&lfs, &cfg) => 0;
  577. lfs_dir_open(&lfs, &dir, "c") => 0;
  578. block = dir.m.pair[0];
  579. lfs_dir_close(&lfs, &dir) => 0;
  580. lfs_unmount(&lfs) => 0;
  581. cfg.read(&cfg, block, 0, bbuffer, LFS_BLOCK_SIZE) => 0;
  582. off = LFS_BLOCK_SIZE-1;
  583. while (off >= 0 && bbuffer[off] == LFS_ERASE_VALUE) {
  584. off -= 1;
  585. }
  586. memset(&bbuffer[off-3], LFS_BLOCK_SIZE, 3);
  587. cfg.erase(&cfg, block) => 0;
  588. cfg.prog(&cfg, block, 0, bbuffer, LFS_BLOCK_SIZE) => 0;
  589. cfg.sync(&cfg) => 0;
  590. lfs_mount(&lfs, &cfg) => 0;
  591. lfs_dir_open(&lfs, &dir, "a") => 0;
  592. lfs_dir_read(&lfs, &dir, &info) => 1;
  593. assert(strcmp(info.name, ".") == 0);
  594. assert(info.type == LFS_TYPE_DIR);
  595. lfs_dir_read(&lfs, &dir, &info) => 1;
  596. assert(strcmp(info.name, "..") == 0);
  597. assert(info.type == LFS_TYPE_DIR);
  598. lfs_dir_read(&lfs, &dir, &info) => 1;
  599. assert(strcmp(info.name, "hi") == 0);
  600. assert(info.type == LFS_TYPE_DIR);
  601. lfs_dir_read(&lfs, &dir, &info) => 0;
  602. lfs_dir_close(&lfs, &dir) => 0;
  603. lfs_dir_open(&lfs, &dir, "c") => 0;
  604. lfs_dir_read(&lfs, &dir, &info) => 1;
  605. assert(strcmp(info.name, ".") == 0);
  606. assert(info.type == LFS_TYPE_DIR);
  607. lfs_dir_read(&lfs, &dir, &info) => 1;
  608. assert(strcmp(info.name, "..") == 0);
  609. assert(info.type == LFS_TYPE_DIR);
  610. lfs_dir_read(&lfs, &dir, &info) => 0;
  611. lfs_dir_close(&lfs, &dir) => 0;
  612. lfs_dir_open(&lfs, &dir, "a/hi") => 0;
  613. lfs_dir_read(&lfs, &dir, &info) => 1;
  614. assert(strcmp(info.name, ".") == 0);
  615. assert(info.type == LFS_TYPE_DIR);
  616. lfs_dir_read(&lfs, &dir, &info) => 1;
  617. assert(strcmp(info.name, "..") == 0);
  618. assert(info.type == LFS_TYPE_DIR);
  619. lfs_dir_read(&lfs, &dir, &info) => 1;
  620. assert(strcmp(info.name, "bonjour") == 0);
  621. assert(info.type == LFS_TYPE_DIR);
  622. lfs_dir_read(&lfs, &dir, &info) => 1;
  623. assert(strcmp(info.name, "hola") == 0);
  624. assert(info.type == LFS_TYPE_DIR);
  625. lfs_dir_read(&lfs, &dir, &info) => 1;
  626. assert(strcmp(info.name, "ohayo") == 0);
  627. assert(info.type == LFS_TYPE_DIR);
  628. lfs_dir_read(&lfs, &dir, &info) => 0;
  629. lfs_dir_close(&lfs, &dir) => 0;
  630. lfs_dir_open(&lfs, &dir, "b/hi") => LFS_ERR_NOENT;
  631. lfs_dir_open(&lfs, &dir, "c/hi") => LFS_ERR_NOENT;
  632. lfs_dir_open(&lfs, &dir, "d/hi") => LFS_ERR_NOENT;
  633. lfs_unmount(&lfs) => 0;
  634. '''
  635. [[case]] # move dir after corrupt
  636. in = "lfs.c"
  637. code = '''
  638. lfs_format(&lfs, &cfg) => 0;
  639. lfs_mount(&lfs, &cfg) => 0;
  640. lfs_mkdir(&lfs, "a") => 0;
  641. lfs_mkdir(&lfs, "b") => 0;
  642. lfs_mkdir(&lfs, "c") => 0;
  643. lfs_mkdir(&lfs, "d") => 0;
  644. lfs_mkdir(&lfs, "a/hi") => 0;
  645. lfs_mkdir(&lfs, "a/hi/hola") => 0;
  646. lfs_mkdir(&lfs, "a/hi/bonjour") => 0;
  647. lfs_mkdir(&lfs, "a/hi/ohayo") => 0;
  648. lfs_unmount(&lfs) => 0;
  649. lfs_mount(&lfs, &cfg) => 0;
  650. lfs_rename(&lfs, "a/hi", "c/hi") => 0;
  651. lfs_unmount(&lfs) => 0;
  652. // corrupt the source
  653. lfs_mount(&lfs, &cfg) => 0;
  654. lfs_dir_open(&lfs, &dir, "a") => 0;
  655. lfs_block_t block = dir.m.pair[0];
  656. lfs_dir_close(&lfs, &dir) => 0;
  657. lfs_unmount(&lfs) => 0;
  658. uint8_t bbuffer[LFS_BLOCK_SIZE];
  659. cfg.read(&cfg, block, 0, bbuffer, LFS_BLOCK_SIZE) => 0;
  660. int off = LFS_BLOCK_SIZE-1;
  661. while (off >= 0 && bbuffer[off] == LFS_ERASE_VALUE) {
  662. off -= 1;
  663. }
  664. memset(&bbuffer[off-3], LFS_BLOCK_SIZE, 3);
  665. cfg.erase(&cfg, block) => 0;
  666. cfg.prog(&cfg, block, 0, bbuffer, LFS_BLOCK_SIZE) => 0;
  667. cfg.sync(&cfg) => 0;
  668. // corrupt the destination
  669. lfs_mount(&lfs, &cfg) => 0;
  670. lfs_dir_open(&lfs, &dir, "c") => 0;
  671. block = dir.m.pair[0];
  672. lfs_dir_close(&lfs, &dir) => 0;
  673. lfs_unmount(&lfs) => 0;
  674. cfg.read(&cfg, block, 0, bbuffer, LFS_BLOCK_SIZE) => 0;
  675. off = LFS_BLOCK_SIZE-1;
  676. while (off >= 0 && bbuffer[off] == LFS_ERASE_VALUE) {
  677. off -= 1;
  678. }
  679. memset(&bbuffer[off-3], LFS_BLOCK_SIZE, 3);
  680. cfg.erase(&cfg, block) => 0;
  681. cfg.prog(&cfg, block, 0, bbuffer, LFS_BLOCK_SIZE) => 0;
  682. cfg.sync(&cfg) => 0;
  683. // continue move
  684. lfs_mount(&lfs, &cfg) => 0;
  685. lfs_rename(&lfs, "a/hi", "c/hi") => 0;
  686. lfs_unmount(&lfs) => 0;
  687. lfs_mount(&lfs, &cfg) => 0;
  688. lfs_dir_open(&lfs, &dir, "a") => 0;
  689. lfs_dir_read(&lfs, &dir, &info) => 1;
  690. assert(strcmp(info.name, ".") == 0);
  691. assert(info.type == LFS_TYPE_DIR);
  692. lfs_dir_read(&lfs, &dir, &info) => 1;
  693. assert(strcmp(info.name, "..") == 0);
  694. assert(info.type == LFS_TYPE_DIR);
  695. lfs_dir_read(&lfs, &dir, &info) => 0;
  696. lfs_dir_close(&lfs, &dir) => 0;
  697. lfs_dir_open(&lfs, &dir, "c") => 0;
  698. lfs_dir_read(&lfs, &dir, &info) => 1;
  699. assert(strcmp(info.name, ".") == 0);
  700. assert(info.type == LFS_TYPE_DIR);
  701. lfs_dir_read(&lfs, &dir, &info) => 1;
  702. assert(strcmp(info.name, "..") == 0);
  703. assert(info.type == LFS_TYPE_DIR);
  704. lfs_dir_read(&lfs, &dir, &info) => 1;
  705. assert(strcmp(info.name, "hi") == 0);
  706. assert(info.type == LFS_TYPE_DIR);
  707. lfs_dir_read(&lfs, &dir, &info) => 0;
  708. lfs_dir_close(&lfs, &dir) => 0;
  709. lfs_dir_open(&lfs, &dir, "a/hi") => LFS_ERR_NOENT;
  710. lfs_dir_open(&lfs, &dir, "b/hi") => LFS_ERR_NOENT;
  711. lfs_dir_open(&lfs, &dir, "c/hi") => 0;
  712. lfs_dir_read(&lfs, &dir, &info) => 1;
  713. assert(strcmp(info.name, ".") == 0);
  714. assert(info.type == LFS_TYPE_DIR);
  715. lfs_dir_read(&lfs, &dir, &info) => 1;
  716. assert(strcmp(info.name, "..") == 0);
  717. assert(info.type == LFS_TYPE_DIR);
  718. lfs_dir_read(&lfs, &dir, &info) => 1;
  719. assert(strcmp(info.name, "bonjour") == 0);
  720. assert(info.type == LFS_TYPE_DIR);
  721. lfs_dir_read(&lfs, &dir, &info) => 1;
  722. assert(strcmp(info.name, "hola") == 0);
  723. assert(info.type == LFS_TYPE_DIR);
  724. lfs_dir_read(&lfs, &dir, &info) => 1;
  725. assert(strcmp(info.name, "ohayo") == 0);
  726. assert(info.type == LFS_TYPE_DIR);
  727. lfs_dir_read(&lfs, &dir, &info) => 0;
  728. lfs_dir_close(&lfs, &dir) => 0;
  729. lfs_dir_open(&lfs, &dir, "d/hi") => LFS_ERR_NOENT;
  730. lfs_unmount(&lfs) => 0;
  731. '''
  732. [[case]] # simple reentrant move dir
  733. reentrant = true
  734. code = '''
  735. err = lfs_mount(&lfs, &cfg);
  736. if (err) {
  737. lfs_format(&lfs, &cfg) => 0;
  738. lfs_mount(&lfs, &cfg) => 0;
  739. }
  740. err = lfs_mkdir(&lfs, "a");
  741. assert(!err || err == LFS_ERR_EXIST);
  742. err = lfs_mkdir(&lfs, "b");
  743. assert(!err || err == LFS_ERR_EXIST);
  744. err = lfs_mkdir(&lfs, "c");
  745. assert(!err || err == LFS_ERR_EXIST);
  746. err = lfs_mkdir(&lfs, "d");
  747. assert(!err || err == LFS_ERR_EXIST);
  748. lfs_unmount(&lfs) => 0;
  749. while (true) {
  750. lfs_mount(&lfs, &cfg) => 0;
  751. // there should never exist _2_ hi directories
  752. int count = 0;
  753. if (lfs_stat(&lfs, "a/hi", &info) == 0) {
  754. assert(strcmp(info.name, "hi") == 0);
  755. assert(info.type == LFS_TYPE_DIR);
  756. count += 1;
  757. }
  758. if (lfs_stat(&lfs, "b/hi", &info) == 0) {
  759. assert(strcmp(info.name, "hi") == 0);
  760. assert(info.type == LFS_TYPE_DIR);
  761. count += 1;
  762. }
  763. if (lfs_stat(&lfs, "c/hi", &info) == 0) {
  764. assert(strcmp(info.name, "hi") == 0);
  765. assert(info.type == LFS_TYPE_DIR);
  766. count += 1;
  767. }
  768. if (lfs_stat(&lfs, "d/hi", &info) == 0) {
  769. assert(strcmp(info.name, "hi") == 0);
  770. assert(info.type == LFS_TYPE_DIR);
  771. count += 1;
  772. }
  773. assert(count <= 1);
  774. lfs_unmount(&lfs) => 0;
  775. lfs_mount(&lfs, &cfg) => 0;
  776. if (lfs_stat(&lfs, "a/hi", &info) == 0) {
  777. lfs_rename(&lfs, "a/hi", "b/hi") => 0;
  778. } else if (lfs_stat(&lfs, "b/hi", &info) == 0) {
  779. lfs_rename(&lfs, "b/hi", "c/hi") => 0;
  780. } else if (lfs_stat(&lfs, "c/hi", &info) == 0) {
  781. lfs_rename(&lfs, "c/hi", "d/hi") => 0;
  782. } else if (lfs_stat(&lfs, "d/hi", &info) == 0) {
  783. lfs_unmount(&lfs) => 0;
  784. break; // success
  785. } else {
  786. // create dir and rename for atomicity
  787. err = lfs_mkdir(&lfs, "temp");
  788. assert(!err || err == LFS_ERR_EXIST);
  789. err = lfs_mkdir(&lfs, "temp/hola");
  790. assert(!err || err == LFS_ERR_EXIST);
  791. err = lfs_mkdir(&lfs, "temp/bonjour");
  792. assert(!err || err == LFS_ERR_EXIST);
  793. err = lfs_mkdir(&lfs, "temp/ohayo");
  794. assert(!err || err == LFS_ERR_EXIST);
  795. lfs_rename(&lfs, "temp", "a/hi") => 0;
  796. }
  797. lfs_unmount(&lfs) => 0;
  798. }
  799. lfs_mount(&lfs, &cfg) => 0;
  800. lfs_dir_open(&lfs, &dir, "a") => 0;
  801. lfs_dir_read(&lfs, &dir, &info) => 1;
  802. assert(strcmp(info.name, ".") == 0);
  803. assert(info.type == LFS_TYPE_DIR);
  804. lfs_dir_read(&lfs, &dir, &info) => 1;
  805. assert(strcmp(info.name, "..") == 0);
  806. assert(info.type == LFS_TYPE_DIR);
  807. lfs_dir_read(&lfs, &dir, &info) => 0;
  808. lfs_dir_close(&lfs, &dir) => 0;
  809. lfs_dir_open(&lfs, &dir, "d") => 0;
  810. lfs_dir_read(&lfs, &dir, &info) => 1;
  811. assert(strcmp(info.name, ".") == 0);
  812. assert(info.type == LFS_TYPE_DIR);
  813. lfs_dir_read(&lfs, &dir, &info) => 1;
  814. assert(strcmp(info.name, "..") == 0);
  815. assert(info.type == LFS_TYPE_DIR);
  816. lfs_dir_read(&lfs, &dir, &info) => 1;
  817. assert(strcmp(info.name, "hi") == 0);
  818. assert(info.type == LFS_TYPE_DIR);
  819. lfs_dir_read(&lfs, &dir, &info) => 0;
  820. lfs_dir_close(&lfs, &dir) => 0;
  821. lfs_dir_open(&lfs, &dir, "a/hi") => LFS_ERR_NOENT;
  822. lfs_dir_open(&lfs, &dir, "b/hi") => LFS_ERR_NOENT;
  823. lfs_dir_open(&lfs, &dir, "c/hi") => LFS_ERR_NOENT;
  824. lfs_dir_open(&lfs, &dir, "d/hi") => 0;
  825. lfs_dir_read(&lfs, &dir, &info) => 1;
  826. assert(strcmp(info.name, ".") == 0);
  827. assert(info.type == LFS_TYPE_DIR);
  828. lfs_dir_read(&lfs, &dir, &info) => 1;
  829. assert(strcmp(info.name, "..") == 0);
  830. assert(info.type == LFS_TYPE_DIR);
  831. lfs_dir_read(&lfs, &dir, &info) => 1;
  832. assert(strcmp(info.name, "bonjour") == 0);
  833. assert(info.type == LFS_TYPE_DIR);
  834. lfs_dir_read(&lfs, &dir, &info) => 1;
  835. assert(strcmp(info.name, "hola") == 0);
  836. assert(info.type == LFS_TYPE_DIR);
  837. lfs_dir_read(&lfs, &dir, &info) => 1;
  838. assert(strcmp(info.name, "ohayo") == 0);
  839. assert(info.type == LFS_TYPE_DIR);
  840. lfs_dir_read(&lfs, &dir, &info) => 0;
  841. lfs_dir_close(&lfs, &dir) => 0;
  842. lfs_unmount(&lfs) => 0;
  843. '''
  844. [[case]] # move state stealing
  845. code = '''
  846. lfs_format(&lfs, &cfg) => 0;
  847. lfs_mount(&lfs, &cfg) => 0;
  848. lfs_mkdir(&lfs, "a") => 0;
  849. lfs_mkdir(&lfs, "b") => 0;
  850. lfs_mkdir(&lfs, "c") => 0;
  851. lfs_mkdir(&lfs, "d") => 0;
  852. lfs_file_open(&lfs, &file, "a/hello", LFS_O_CREAT | LFS_O_WRONLY) => 0;
  853. lfs_file_write(&lfs, &file, "hola\n", 5) => 5;
  854. lfs_file_write(&lfs, &file, "bonjour\n", 8) => 8;
  855. lfs_file_write(&lfs, &file, "ohayo\n", 6) => 6;
  856. lfs_file_close(&lfs, &file) => 0;
  857. lfs_unmount(&lfs) => 0;
  858. lfs_mount(&lfs, &cfg) => 0;
  859. lfs_rename(&lfs, "a/hello", "b/hello") => 0;
  860. lfs_unmount(&lfs) => 0;
  861. lfs_mount(&lfs, &cfg) => 0;
  862. lfs_rename(&lfs, "b/hello", "c/hello") => 0;
  863. lfs_unmount(&lfs) => 0;
  864. lfs_mount(&lfs, &cfg) => 0;
  865. lfs_rename(&lfs, "c/hello", "d/hello") => 0;
  866. lfs_unmount(&lfs) => 0;
  867. lfs_mount(&lfs, &cfg) => 0;
  868. lfs_file_open(&lfs, &file, "a/hello", LFS_O_RDONLY) => LFS_ERR_NOENT;
  869. lfs_file_open(&lfs, &file, "b/hello", LFS_O_RDONLY) => LFS_ERR_NOENT;
  870. lfs_file_open(&lfs, &file, "c/hello", LFS_O_RDONLY) => LFS_ERR_NOENT;
  871. lfs_file_open(&lfs, &file, "d/hello", LFS_O_RDONLY) => 0;
  872. lfs_file_read(&lfs, &file, buffer, 5) => 5;
  873. memcmp(buffer, "hola\n", 5) => 0;
  874. lfs_file_read(&lfs, &file, buffer, 8) => 8;
  875. memcmp(buffer, "bonjour\n", 8) => 0;
  876. lfs_file_read(&lfs, &file, buffer, 6) => 6;
  877. memcmp(buffer, "ohayo\n", 6) => 0;
  878. lfs_file_close(&lfs, &file) => 0;
  879. lfs_unmount(&lfs) => 0;
  880. lfs_mount(&lfs, &cfg) => 0;
  881. lfs_remove(&lfs, "b") => 0;
  882. lfs_remove(&lfs, "c") => 0;
  883. lfs_unmount(&lfs) => 0;
  884. lfs_mount(&lfs, &cfg) => 0;
  885. lfs_stat(&lfs, "a", &info) => 0;
  886. lfs_stat(&lfs, "b", &info) => LFS_ERR_NOENT;
  887. lfs_stat(&lfs, "c", &info) => LFS_ERR_NOENT;
  888. lfs_stat(&lfs, "d", &info) => 0;
  889. lfs_file_open(&lfs, &file, "a/hello", LFS_O_RDONLY) => LFS_ERR_NOENT;
  890. lfs_file_open(&lfs, &file, "b/hello", LFS_O_RDONLY) => LFS_ERR_NOENT;
  891. lfs_file_open(&lfs, &file, "c/hello", LFS_O_RDONLY) => LFS_ERR_NOENT;
  892. lfs_file_open(&lfs, &file, "d/hello", LFS_O_RDONLY) => 0;
  893. lfs_file_read(&lfs, &file, buffer, 5) => 5;
  894. memcmp(buffer, "hola\n", 5) => 0;
  895. lfs_file_read(&lfs, &file, buffer, 8) => 8;
  896. memcmp(buffer, "bonjour\n", 8) => 0;
  897. lfs_file_read(&lfs, &file, buffer, 6) => 6;
  898. memcmp(buffer, "ohayo\n", 6) => 0;
  899. lfs_file_close(&lfs, &file) => 0;
  900. lfs_unmount(&lfs) => 0;
  901. '''
  902. # Other specific corner cases
  903. [[case]] # create + delete in same commit with neighbors
  904. code = '''
  905. lfs_format(&lfs, &cfg) => 0;
  906. lfs_mount(&lfs, &cfg) => 0;
  907. // littlefs keeps files sorted, so we know the order these will be in
  908. lfs_file_open(&lfs, &file, "/1.move_me",
  909. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  910. lfs_file_close(&lfs, &file) => 0;
  911. lfs_file_open(&lfs, &file, "/0.before",
  912. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  913. lfs_file_write(&lfs, &file, "test.1", 7) => 7;
  914. lfs_file_close(&lfs, &file) => 0;
  915. lfs_file_open(&lfs, &file, "/2.in_between",
  916. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  917. lfs_file_write(&lfs, &file, "test.2", 7) => 7;
  918. lfs_file_close(&lfs, &file) => 0;
  919. lfs_file_open(&lfs, &file, "/4.after",
  920. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  921. lfs_file_write(&lfs, &file, "test.3", 7) => 7;
  922. lfs_file_close(&lfs, &file) => 0;
  923. lfs_file_t files[3];
  924. lfs_file_open(&lfs, &files[0], "0.before",
  925. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  926. lfs_file_open(&lfs, &files[1], "2.in_between",
  927. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  928. lfs_file_open(&lfs, &files[2], "4.after",
  929. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  930. lfs_file_write(&lfs, &files[0], "test.4", 7) => 7;
  931. lfs_file_write(&lfs, &files[1], "test.5", 7) => 7;
  932. lfs_file_write(&lfs, &files[2], "test.6", 7) => 7;
  933. // rename file while everything is open, this triggers both
  934. // a create and delete simultaneously
  935. lfs_rename(&lfs, "/1.move_me", "/3.move_me") => 0;
  936. lfs_file_close(&lfs, &files[0]) => 0;
  937. lfs_file_close(&lfs, &files[1]) => 0;
  938. lfs_file_close(&lfs, &files[2]) => 0;
  939. // check that nothing was corrupted
  940. lfs_dir_open(&lfs, &dir, "/") => 0;
  941. lfs_dir_read(&lfs, &dir, &info) => 1;
  942. assert(strcmp(info.name, ".") == 0);
  943. assert(info.type == LFS_TYPE_DIR);
  944. lfs_dir_read(&lfs, &dir, &info) => 1;
  945. assert(strcmp(info.name, "..") == 0);
  946. assert(info.type == LFS_TYPE_DIR);
  947. lfs_dir_read(&lfs, &dir, &info) => 1;
  948. assert(strcmp(info.name, "0.before") == 0);
  949. assert(info.type == LFS_TYPE_REG);
  950. assert(info.size == 7);
  951. lfs_dir_read(&lfs, &dir, &info) => 1;
  952. assert(strcmp(info.name, "2.in_between") == 0);
  953. assert(info.type == LFS_TYPE_REG);
  954. assert(info.size == 7);
  955. lfs_dir_read(&lfs, &dir, &info) => 1;
  956. assert(strcmp(info.name, "3.move_me") == 0);
  957. assert(info.type == LFS_TYPE_REG);
  958. assert(info.size == 0);
  959. lfs_dir_read(&lfs, &dir, &info) => 1;
  960. assert(strcmp(info.name, "4.after") == 0);
  961. assert(info.type == LFS_TYPE_REG);
  962. assert(info.size == 7);
  963. lfs_dir_read(&lfs, &dir, &info) => 0;
  964. lfs_dir_close(&lfs, &dir) => 0;
  965. lfs_file_open(&lfs, &file, "/0.before", LFS_O_RDONLY) => 0;
  966. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  967. assert(strcmp((char*)buffer, "test.4") == 0);
  968. lfs_file_close(&lfs, &file) => 0;
  969. lfs_file_open(&lfs, &file, "/2.in_between", LFS_O_RDONLY) => 0;
  970. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  971. assert(strcmp((char*)buffer, "test.5") == 0);
  972. lfs_file_close(&lfs, &file) => 0;
  973. lfs_file_open(&lfs, &file, "/4.after", LFS_O_RDONLY) => 0;
  974. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  975. assert(strcmp((char*)buffer, "test.6") == 0);
  976. lfs_file_close(&lfs, &file) => 0;
  977. // now move back
  978. lfs_file_open(&lfs, &files[0], "0.before",
  979. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  980. lfs_file_open(&lfs, &files[1], "2.in_between",
  981. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  982. lfs_file_open(&lfs, &files[2], "4.after",
  983. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  984. lfs_file_write(&lfs, &files[0], "test.7", 7) => 7;
  985. lfs_file_write(&lfs, &files[1], "test.8", 7) => 7;
  986. lfs_file_write(&lfs, &files[2], "test.9", 7) => 7;
  987. // rename file while everything is open, this triggers both
  988. // a create and delete simultaneously
  989. lfs_rename(&lfs, "/3.move_me", "/1.move_me") => 0;
  990. lfs_file_close(&lfs, &files[0]) => 0;
  991. lfs_file_close(&lfs, &files[1]) => 0;
  992. lfs_file_close(&lfs, &files[2]) => 0;
  993. // and check that nothing was corrupted again
  994. lfs_dir_open(&lfs, &dir, "/") => 0;
  995. lfs_dir_read(&lfs, &dir, &info) => 1;
  996. assert(strcmp(info.name, ".") == 0);
  997. assert(info.type == LFS_TYPE_DIR);
  998. lfs_dir_read(&lfs, &dir, &info) => 1;
  999. assert(strcmp(info.name, "..") == 0);
  1000. assert(info.type == LFS_TYPE_DIR);
  1001. lfs_dir_read(&lfs, &dir, &info) => 1;
  1002. assert(strcmp(info.name, "0.before") == 0);
  1003. assert(info.type == LFS_TYPE_REG);
  1004. assert(info.size == 7);
  1005. lfs_dir_read(&lfs, &dir, &info) => 1;
  1006. assert(strcmp(info.name, "1.move_me") == 0);
  1007. assert(info.type == LFS_TYPE_REG);
  1008. assert(info.size == 0);
  1009. lfs_dir_read(&lfs, &dir, &info) => 1;
  1010. assert(strcmp(info.name, "2.in_between") == 0);
  1011. assert(info.type == LFS_TYPE_REG);
  1012. assert(info.size == 7);
  1013. lfs_dir_read(&lfs, &dir, &info) => 1;
  1014. assert(strcmp(info.name, "4.after") == 0);
  1015. assert(info.type == LFS_TYPE_REG);
  1016. assert(info.size == 7);
  1017. lfs_dir_read(&lfs, &dir, &info) => 0;
  1018. lfs_dir_close(&lfs, &dir) => 0;
  1019. lfs_file_open(&lfs, &file, "/0.before", LFS_O_RDONLY) => 0;
  1020. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1021. assert(strcmp((char*)buffer, "test.7") == 0);
  1022. lfs_file_close(&lfs, &file) => 0;
  1023. lfs_file_open(&lfs, &file, "/2.in_between", LFS_O_RDONLY) => 0;
  1024. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1025. assert(strcmp((char*)buffer, "test.8") == 0);
  1026. lfs_file_close(&lfs, &file) => 0;
  1027. lfs_file_open(&lfs, &file, "/4.after", LFS_O_RDONLY) => 0;
  1028. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1029. assert(strcmp((char*)buffer, "test.9") == 0);
  1030. lfs_file_close(&lfs, &file) => 0;
  1031. lfs_unmount(&lfs) => 0;
  1032. '''
  1033. # Other specific corner cases
  1034. [[case]] # create + delete + delete in same commit with neighbors
  1035. code = '''
  1036. lfs_format(&lfs, &cfg) => 0;
  1037. lfs_mount(&lfs, &cfg) => 0;
  1038. // littlefs keeps files sorted, so we know the order these will be in
  1039. lfs_file_open(&lfs, &file, "/1.move_me",
  1040. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  1041. lfs_file_close(&lfs, &file) => 0;
  1042. lfs_file_open(&lfs, &file, "/3.move_me",
  1043. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  1044. lfs_file_write(&lfs, &file, "remove me",
  1045. sizeof("remove me")) => sizeof("remove me");
  1046. lfs_file_close(&lfs, &file) => 0;
  1047. lfs_file_open(&lfs, &file, "/0.before",
  1048. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  1049. lfs_file_write(&lfs, &file, "test.1", 7) => 7;
  1050. lfs_file_close(&lfs, &file) => 0;
  1051. lfs_file_open(&lfs, &file, "/2.in_between",
  1052. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  1053. lfs_file_write(&lfs, &file, "test.2", 7) => 7;
  1054. lfs_file_close(&lfs, &file) => 0;
  1055. lfs_file_open(&lfs, &file, "/4.after",
  1056. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  1057. lfs_file_write(&lfs, &file, "test.3", 7) => 7;
  1058. lfs_file_close(&lfs, &file) => 0;
  1059. lfs_file_t files[3];
  1060. lfs_file_open(&lfs, &files[0], "0.before",
  1061. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  1062. lfs_file_open(&lfs, &files[1], "2.in_between",
  1063. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  1064. lfs_file_open(&lfs, &files[2], "4.after",
  1065. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  1066. lfs_file_write(&lfs, &files[0], "test.4", 7) => 7;
  1067. lfs_file_write(&lfs, &files[1], "test.5", 7) => 7;
  1068. lfs_file_write(&lfs, &files[2], "test.6", 7) => 7;
  1069. // rename file while everything is open, this triggers both
  1070. // a create and delete simultaneously
  1071. lfs_rename(&lfs, "/1.move_me", "/3.move_me") => 0;
  1072. lfs_file_close(&lfs, &files[0]) => 0;
  1073. lfs_file_close(&lfs, &files[1]) => 0;
  1074. lfs_file_close(&lfs, &files[2]) => 0;
  1075. // check that nothing was corrupted
  1076. lfs_dir_open(&lfs, &dir, "/") => 0;
  1077. lfs_dir_read(&lfs, &dir, &info) => 1;
  1078. assert(strcmp(info.name, ".") == 0);
  1079. assert(info.type == LFS_TYPE_DIR);
  1080. lfs_dir_read(&lfs, &dir, &info) => 1;
  1081. assert(strcmp(info.name, "..") == 0);
  1082. assert(info.type == LFS_TYPE_DIR);
  1083. lfs_dir_read(&lfs, &dir, &info) => 1;
  1084. assert(strcmp(info.name, "0.before") == 0);
  1085. assert(info.type == LFS_TYPE_REG);
  1086. assert(info.size == 7);
  1087. lfs_dir_read(&lfs, &dir, &info) => 1;
  1088. assert(strcmp(info.name, "2.in_between") == 0);
  1089. assert(info.type == LFS_TYPE_REG);
  1090. assert(info.size == 7);
  1091. lfs_dir_read(&lfs, &dir, &info) => 1;
  1092. assert(strcmp(info.name, "3.move_me") == 0);
  1093. assert(info.type == LFS_TYPE_REG);
  1094. assert(info.size == 0);
  1095. lfs_dir_read(&lfs, &dir, &info) => 1;
  1096. assert(strcmp(info.name, "4.after") == 0);
  1097. assert(info.type == LFS_TYPE_REG);
  1098. assert(info.size == 7);
  1099. lfs_dir_read(&lfs, &dir, &info) => 0;
  1100. lfs_dir_close(&lfs, &dir) => 0;
  1101. lfs_file_open(&lfs, &file, "/0.before", LFS_O_RDONLY) => 0;
  1102. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1103. assert(strcmp((char*)buffer, "test.4") == 0);
  1104. lfs_file_close(&lfs, &file) => 0;
  1105. lfs_file_open(&lfs, &file, "/2.in_between", LFS_O_RDONLY) => 0;
  1106. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1107. assert(strcmp((char*)buffer, "test.5") == 0);
  1108. lfs_file_close(&lfs, &file) => 0;
  1109. lfs_file_open(&lfs, &file, "/4.after", LFS_O_RDONLY) => 0;
  1110. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1111. assert(strcmp((char*)buffer, "test.6") == 0);
  1112. lfs_file_close(&lfs, &file) => 0;
  1113. // now move back
  1114. lfs_file_open(&lfs, &file, "/1.move_me",
  1115. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  1116. lfs_file_write(&lfs, &file, "remove me",
  1117. sizeof("remove me")) => sizeof("remove me");
  1118. lfs_file_close(&lfs, &file) => 0;
  1119. lfs_file_open(&lfs, &files[0], "0.before",
  1120. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  1121. lfs_file_open(&lfs, &files[1], "2.in_between",
  1122. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  1123. lfs_file_open(&lfs, &files[2], "4.after",
  1124. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  1125. lfs_file_write(&lfs, &files[0], "test.7", 7) => 7;
  1126. lfs_file_write(&lfs, &files[1], "test.8", 7) => 7;
  1127. lfs_file_write(&lfs, &files[2], "test.9", 7) => 7;
  1128. // rename file while everything is open, this triggers both
  1129. // a create and delete simultaneously
  1130. lfs_rename(&lfs, "/3.move_me", "/1.move_me") => 0;
  1131. lfs_file_close(&lfs, &files[0]) => 0;
  1132. lfs_file_close(&lfs, &files[1]) => 0;
  1133. lfs_file_close(&lfs, &files[2]) => 0;
  1134. // and check that nothing was corrupted again
  1135. lfs_dir_open(&lfs, &dir, "/") => 0;
  1136. lfs_dir_read(&lfs, &dir, &info) => 1;
  1137. assert(strcmp(info.name, ".") == 0);
  1138. assert(info.type == LFS_TYPE_DIR);
  1139. lfs_dir_read(&lfs, &dir, &info) => 1;
  1140. assert(strcmp(info.name, "..") == 0);
  1141. assert(info.type == LFS_TYPE_DIR);
  1142. lfs_dir_read(&lfs, &dir, &info) => 1;
  1143. assert(strcmp(info.name, "0.before") == 0);
  1144. assert(info.type == LFS_TYPE_REG);
  1145. assert(info.size == 7);
  1146. lfs_dir_read(&lfs, &dir, &info) => 1;
  1147. assert(strcmp(info.name, "1.move_me") == 0);
  1148. assert(info.type == LFS_TYPE_REG);
  1149. assert(info.size == 0);
  1150. lfs_dir_read(&lfs, &dir, &info) => 1;
  1151. assert(strcmp(info.name, "2.in_between") == 0);
  1152. assert(info.type == LFS_TYPE_REG);
  1153. assert(info.size == 7);
  1154. lfs_dir_read(&lfs, &dir, &info) => 1;
  1155. assert(strcmp(info.name, "4.after") == 0);
  1156. assert(info.type == LFS_TYPE_REG);
  1157. assert(info.size == 7);
  1158. lfs_dir_read(&lfs, &dir, &info) => 0;
  1159. lfs_dir_close(&lfs, &dir) => 0;
  1160. lfs_file_open(&lfs, &file, "/0.before", LFS_O_RDONLY) => 0;
  1161. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1162. assert(strcmp((char*)buffer, "test.7") == 0);
  1163. lfs_file_close(&lfs, &file) => 0;
  1164. lfs_file_open(&lfs, &file, "/2.in_between", LFS_O_RDONLY) => 0;
  1165. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1166. assert(strcmp((char*)buffer, "test.8") == 0);
  1167. lfs_file_close(&lfs, &file) => 0;
  1168. lfs_file_open(&lfs, &file, "/4.after", LFS_O_RDONLY) => 0;
  1169. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1170. assert(strcmp((char*)buffer, "test.9") == 0);
  1171. lfs_file_close(&lfs, &file) => 0;
  1172. lfs_unmount(&lfs) => 0;
  1173. '''
  1174. [[case]] # create + delete in different dirs with neighbors
  1175. code = '''
  1176. lfs_format(&lfs, &cfg) => 0;
  1177. lfs_mount(&lfs, &cfg) => 0;
  1178. // littlefs keeps files sorted, so we know the order these will be in
  1179. lfs_mkdir(&lfs, "/dir.1") => 0;
  1180. lfs_mkdir(&lfs, "/dir.2") => 0;
  1181. lfs_file_open(&lfs, &file, "/dir.1/1.move_me",
  1182. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  1183. lfs_file_close(&lfs, &file) => 0;
  1184. lfs_file_open(&lfs, &file, "/dir.2/1.move_me",
  1185. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  1186. lfs_file_write(&lfs, &file, "remove me",
  1187. sizeof("remove me")) => sizeof("remove me");
  1188. lfs_file_close(&lfs, &file) => 0;
  1189. lfs_file_open(&lfs, &file, "/dir.1/0.before",
  1190. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  1191. lfs_file_write(&lfs, &file, "test.1", 7) => 7;
  1192. lfs_file_close(&lfs, &file) => 0;
  1193. lfs_file_open(&lfs, &file, "/dir.1/2.after",
  1194. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  1195. lfs_file_write(&lfs, &file, "test.2", 7) => 7;
  1196. lfs_file_close(&lfs, &file) => 0;
  1197. lfs_file_open(&lfs, &file, "/dir.2/0.before",
  1198. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  1199. lfs_file_write(&lfs, &file, "test.3", 7) => 7;
  1200. lfs_file_close(&lfs, &file) => 0;
  1201. lfs_file_open(&lfs, &file, "/dir.2/2.after",
  1202. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  1203. lfs_file_write(&lfs, &file, "test.4", 7) => 7;
  1204. lfs_file_close(&lfs, &file) => 0;
  1205. lfs_file_t files[4];
  1206. lfs_file_open(&lfs, &files[0], "/dir.1/0.before",
  1207. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  1208. lfs_file_open(&lfs, &files[1], "/dir.1/2.after",
  1209. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  1210. lfs_file_open(&lfs, &files[2], "/dir.2/0.before",
  1211. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  1212. lfs_file_open(&lfs, &files[3], "/dir.2/2.after",
  1213. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  1214. lfs_file_write(&lfs, &files[0], "test.5", 7) => 7;
  1215. lfs_file_write(&lfs, &files[1], "test.6", 7) => 7;
  1216. lfs_file_write(&lfs, &files[2], "test.7", 7) => 7;
  1217. lfs_file_write(&lfs, &files[3], "test.8", 7) => 7;
  1218. // rename file while everything is open, this triggers both
  1219. // a create and delete as it overwrites the destination file
  1220. lfs_rename(&lfs, "/dir.1/1.move_me", "/dir.2/1.move_me") => 0;
  1221. lfs_file_close(&lfs, &files[0]) => 0;
  1222. lfs_file_close(&lfs, &files[1]) => 0;
  1223. lfs_file_close(&lfs, &files[2]) => 0;
  1224. lfs_file_close(&lfs, &files[3]) => 0;
  1225. // check that nothing was corrupted
  1226. lfs_dir_open(&lfs, &dir, "/") => 0;
  1227. lfs_dir_read(&lfs, &dir, &info) => 1;
  1228. assert(strcmp(info.name, ".") == 0);
  1229. assert(info.type == LFS_TYPE_DIR);
  1230. lfs_dir_read(&lfs, &dir, &info) => 1;
  1231. assert(strcmp(info.name, "..") == 0);
  1232. assert(info.type == LFS_TYPE_DIR);
  1233. lfs_dir_read(&lfs, &dir, &info) => 1;
  1234. assert(strcmp(info.name, "dir.1") == 0);
  1235. assert(info.type == LFS_TYPE_DIR);
  1236. lfs_dir_read(&lfs, &dir, &info) => 1;
  1237. assert(strcmp(info.name, "dir.2") == 0);
  1238. assert(info.type == LFS_TYPE_DIR);
  1239. lfs_dir_read(&lfs, &dir, &info) => 0;
  1240. lfs_dir_close(&lfs, &dir) => 0;
  1241. lfs_dir_open(&lfs, &dir, "/dir.1") => 0;
  1242. lfs_dir_read(&lfs, &dir, &info) => 1;
  1243. assert(strcmp(info.name, ".") == 0);
  1244. assert(info.type == LFS_TYPE_DIR);
  1245. lfs_dir_read(&lfs, &dir, &info) => 1;
  1246. assert(strcmp(info.name, "..") == 0);
  1247. assert(info.type == LFS_TYPE_DIR);
  1248. lfs_dir_read(&lfs, &dir, &info) => 1;
  1249. assert(strcmp(info.name, "0.before") == 0);
  1250. assert(info.type == LFS_TYPE_REG);
  1251. assert(info.size == 7);
  1252. lfs_dir_read(&lfs, &dir, &info) => 1;
  1253. assert(strcmp(info.name, "2.after") == 0);
  1254. assert(info.type == LFS_TYPE_REG);
  1255. assert(info.size == 7);
  1256. lfs_dir_read(&lfs, &dir, &info) => 0;
  1257. lfs_dir_close(&lfs, &dir) => 0;
  1258. lfs_dir_open(&lfs, &dir, "/dir.2") => 0;
  1259. lfs_dir_read(&lfs, &dir, &info) => 1;
  1260. assert(strcmp(info.name, ".") == 0);
  1261. assert(info.type == LFS_TYPE_DIR);
  1262. lfs_dir_read(&lfs, &dir, &info) => 1;
  1263. assert(strcmp(info.name, "..") == 0);
  1264. assert(info.type == LFS_TYPE_DIR);
  1265. lfs_dir_read(&lfs, &dir, &info) => 1;
  1266. assert(strcmp(info.name, "0.before") == 0);
  1267. assert(info.type == LFS_TYPE_REG);
  1268. assert(info.size == 7);
  1269. lfs_dir_read(&lfs, &dir, &info) => 1;
  1270. assert(strcmp(info.name, "1.move_me") == 0);
  1271. assert(info.type == LFS_TYPE_REG);
  1272. assert(info.size == 0);
  1273. lfs_dir_read(&lfs, &dir, &info) => 1;
  1274. assert(strcmp(info.name, "2.after") == 0);
  1275. assert(info.type == LFS_TYPE_REG);
  1276. assert(info.size == 7);
  1277. lfs_dir_read(&lfs, &dir, &info) => 0;
  1278. lfs_dir_close(&lfs, &dir) => 0;
  1279. lfs_file_open(&lfs, &file, "/dir.1/0.before", LFS_O_RDONLY) => 0;
  1280. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1281. assert(strcmp((char*)buffer, "test.5") == 0);
  1282. lfs_file_close(&lfs, &file) => 0;
  1283. lfs_file_open(&lfs, &file, "/dir.1/2.after", LFS_O_RDONLY) => 0;
  1284. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1285. assert(strcmp((char*)buffer, "test.6") == 0);
  1286. lfs_file_close(&lfs, &file) => 0;
  1287. lfs_file_open(&lfs, &file, "/dir.2/0.before", LFS_O_RDONLY) => 0;
  1288. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1289. assert(strcmp((char*)buffer, "test.7") == 0);
  1290. lfs_file_close(&lfs, &file) => 0;
  1291. lfs_file_open(&lfs, &file, "/dir.2/2.after", LFS_O_RDONLY) => 0;
  1292. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1293. assert(strcmp((char*)buffer, "test.8") == 0);
  1294. lfs_file_close(&lfs, &file) => 0;
  1295. // now move back
  1296. lfs_file_open(&lfs, &file, "/dir.1/1.move_me",
  1297. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  1298. lfs_file_write(&lfs, &file, "remove me",
  1299. sizeof("remove me")) => sizeof("remove me");
  1300. lfs_file_close(&lfs, &file) => 0;
  1301. lfs_file_open(&lfs, &files[0], "/dir.1/0.before",
  1302. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  1303. lfs_file_open(&lfs, &files[1], "/dir.1/2.after",
  1304. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  1305. lfs_file_open(&lfs, &files[2], "/dir.2/0.before",
  1306. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  1307. lfs_file_open(&lfs, &files[3], "/dir.2/2.after",
  1308. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  1309. lfs_file_write(&lfs, &files[0], "test.9", 7) => 7;
  1310. lfs_file_write(&lfs, &files[1], "test.a", 7) => 7;
  1311. lfs_file_write(&lfs, &files[2], "test.b", 7) => 7;
  1312. lfs_file_write(&lfs, &files[3], "test.c", 7) => 7;
  1313. // rename file while everything is open, this triggers both
  1314. // a create and delete simultaneously
  1315. lfs_rename(&lfs, "/dir.2/1.move_me", "/dir.1/1.move_me") => 0;
  1316. lfs_file_close(&lfs, &files[0]) => 0;
  1317. lfs_file_close(&lfs, &files[1]) => 0;
  1318. lfs_file_close(&lfs, &files[2]) => 0;
  1319. lfs_file_close(&lfs, &files[3]) => 0;
  1320. // and check that nothing was corrupted again
  1321. lfs_dir_open(&lfs, &dir, "/") => 0;
  1322. lfs_dir_read(&lfs, &dir, &info) => 1;
  1323. assert(strcmp(info.name, ".") == 0);
  1324. assert(info.type == LFS_TYPE_DIR);
  1325. lfs_dir_read(&lfs, &dir, &info) => 1;
  1326. assert(strcmp(info.name, "..") == 0);
  1327. assert(info.type == LFS_TYPE_DIR);
  1328. lfs_dir_read(&lfs, &dir, &info) => 1;
  1329. assert(strcmp(info.name, "dir.1") == 0);
  1330. assert(info.type == LFS_TYPE_DIR);
  1331. lfs_dir_read(&lfs, &dir, &info) => 1;
  1332. assert(strcmp(info.name, "dir.2") == 0);
  1333. assert(info.type == LFS_TYPE_DIR);
  1334. lfs_dir_read(&lfs, &dir, &info) => 0;
  1335. lfs_dir_close(&lfs, &dir) => 0;
  1336. lfs_dir_open(&lfs, &dir, "/dir.1") => 0;
  1337. lfs_dir_read(&lfs, &dir, &info) => 1;
  1338. assert(strcmp(info.name, ".") == 0);
  1339. assert(info.type == LFS_TYPE_DIR);
  1340. lfs_dir_read(&lfs, &dir, &info) => 1;
  1341. assert(strcmp(info.name, "..") == 0);
  1342. assert(info.type == LFS_TYPE_DIR);
  1343. lfs_dir_read(&lfs, &dir, &info) => 1;
  1344. assert(strcmp(info.name, "0.before") == 0);
  1345. assert(info.type == LFS_TYPE_REG);
  1346. assert(info.size == 7);
  1347. lfs_dir_read(&lfs, &dir, &info) => 1;
  1348. assert(strcmp(info.name, "1.move_me") == 0);
  1349. assert(info.type == LFS_TYPE_REG);
  1350. assert(info.size == 0);
  1351. lfs_dir_read(&lfs, &dir, &info) => 1;
  1352. assert(strcmp(info.name, "2.after") == 0);
  1353. assert(info.type == LFS_TYPE_REG);
  1354. assert(info.size == 7);
  1355. lfs_dir_read(&lfs, &dir, &info) => 0;
  1356. lfs_dir_close(&lfs, &dir) => 0;
  1357. lfs_dir_open(&lfs, &dir, "/dir.2") => 0;
  1358. lfs_dir_read(&lfs, &dir, &info) => 1;
  1359. assert(strcmp(info.name, ".") == 0);
  1360. assert(info.type == LFS_TYPE_DIR);
  1361. lfs_dir_read(&lfs, &dir, &info) => 1;
  1362. assert(strcmp(info.name, "..") == 0);
  1363. assert(info.type == LFS_TYPE_DIR);
  1364. lfs_dir_read(&lfs, &dir, &info) => 1;
  1365. assert(strcmp(info.name, "0.before") == 0);
  1366. assert(info.type == LFS_TYPE_REG);
  1367. assert(info.size == 7);
  1368. lfs_dir_read(&lfs, &dir, &info) => 1;
  1369. assert(strcmp(info.name, "2.after") == 0);
  1370. assert(info.type == LFS_TYPE_REG);
  1371. assert(info.size == 7);
  1372. lfs_dir_read(&lfs, &dir, &info) => 0;
  1373. lfs_dir_close(&lfs, &dir) => 0;
  1374. lfs_file_open(&lfs, &file, "/dir.1/0.before", LFS_O_RDONLY) => 0;
  1375. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1376. assert(strcmp((char*)buffer, "test.9") == 0);
  1377. lfs_file_close(&lfs, &file) => 0;
  1378. lfs_file_open(&lfs, &file, "/dir.1/2.after", LFS_O_RDONLY) => 0;
  1379. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1380. assert(strcmp((char*)buffer, "test.a") == 0);
  1381. lfs_file_close(&lfs, &file) => 0;
  1382. lfs_file_open(&lfs, &file, "/dir.2/0.before", LFS_O_RDONLY) => 0;
  1383. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1384. assert(strcmp((char*)buffer, "test.b") == 0);
  1385. lfs_file_close(&lfs, &file) => 0;
  1386. lfs_file_open(&lfs, &file, "/dir.2/2.after", LFS_O_RDONLY) => 0;
  1387. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1388. assert(strcmp((char*)buffer, "test.c") == 0);
  1389. lfs_file_close(&lfs, &file) => 0;
  1390. lfs_unmount(&lfs) => 0;
  1391. '''
  1392. [[case]] # move fix in relocation
  1393. in = "lfs.c"
  1394. define.RELOCATIONS = 'range(0x3+1)'
  1395. define.LFS_ERASE_CYCLES = 0xffffffff
  1396. code = '''
  1397. lfs_format(&lfs, &cfg) => 0;
  1398. lfs_mount(&lfs, &cfg) => 0;
  1399. lfs_mkdir(&lfs, "/parent") => 0;
  1400. lfs_mkdir(&lfs, "/parent/child") => 0;
  1401. lfs_file_open(&lfs, &file, "/parent/1.move_me",
  1402. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  1403. lfs_file_write(&lfs, &file, "move me",
  1404. sizeof("move me")) => sizeof("move me");
  1405. lfs_file_close(&lfs, &file) => 0;
  1406. lfs_file_open(&lfs, &file, "/parent/0.before",
  1407. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  1408. lfs_file_write(&lfs, &file, "test.1", 7) => 7;
  1409. lfs_file_close(&lfs, &file) => 0;
  1410. lfs_file_open(&lfs, &file, "/parent/2.after",
  1411. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  1412. lfs_file_write(&lfs, &file, "test.2", 7) => 7;
  1413. lfs_file_close(&lfs, &file) => 0;
  1414. lfs_file_open(&lfs, &file, "/parent/child/0.before",
  1415. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  1416. lfs_file_write(&lfs, &file, "test.3", 7) => 7;
  1417. lfs_file_close(&lfs, &file) => 0;
  1418. lfs_file_open(&lfs, &file, "/parent/child/2.after",
  1419. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  1420. lfs_file_write(&lfs, &file, "test.4", 7) => 7;
  1421. lfs_file_close(&lfs, &file) => 0;
  1422. lfs_file_t files[4];
  1423. lfs_file_open(&lfs, &files[0], "/parent/0.before",
  1424. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  1425. lfs_file_open(&lfs, &files[1], "/parent/2.after",
  1426. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  1427. lfs_file_open(&lfs, &files[2], "/parent/child/0.before",
  1428. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  1429. lfs_file_open(&lfs, &files[3], "/parent/child/2.after",
  1430. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  1431. lfs_file_write(&lfs, &files[0], "test.5", 7) => 7;
  1432. lfs_file_write(&lfs, &files[1], "test.6", 7) => 7;
  1433. lfs_file_write(&lfs, &files[2], "test.7", 7) => 7;
  1434. lfs_file_write(&lfs, &files[3], "test.8", 7) => 7;
  1435. // force specific directories to relocate
  1436. if (RELOCATIONS & 0x1) {
  1437. lfs_dir_open(&lfs, &dir, "/parent");
  1438. lfs_testbd_setwear(&cfg, dir.m.pair[0], 0xffffffff) => 0;
  1439. lfs_testbd_setwear(&cfg, dir.m.pair[1], 0xffffffff) => 0;
  1440. lfs_dir_close(&lfs, &dir) => 0;
  1441. }
  1442. if (RELOCATIONS & 0x2) {
  1443. lfs_dir_open(&lfs, &dir, "/parent/child");
  1444. lfs_testbd_setwear(&cfg, dir.m.pair[0], 0xffffffff) => 0;
  1445. lfs_testbd_setwear(&cfg, dir.m.pair[1], 0xffffffff) => 0;
  1446. lfs_dir_close(&lfs, &dir) => 0;
  1447. }
  1448. // ok, now we move the file, this creates a move that needs to be
  1449. // fixed, possibly in a metadata-pair that needs to be relocated
  1450. //
  1451. // the worst case is if we need to relocate and we need to implicit
  1452. // fix the move in our parent before it falls out of date
  1453. lfs_rename(&lfs, "/parent/1.move_me", "/parent/child/1.move_me") => 0;
  1454. lfs_file_close(&lfs, &files[0]) => 0;
  1455. lfs_file_close(&lfs, &files[1]) => 0;
  1456. lfs_file_close(&lfs, &files[2]) => 0;
  1457. lfs_file_close(&lfs, &files[3]) => 0;
  1458. // check that nothing was corrupted
  1459. lfs_dir_open(&lfs, &dir, "/parent") => 0;
  1460. lfs_dir_read(&lfs, &dir, &info) => 1;
  1461. assert(strcmp(info.name, ".") == 0);
  1462. assert(info.type == LFS_TYPE_DIR);
  1463. lfs_dir_read(&lfs, &dir, &info) => 1;
  1464. assert(strcmp(info.name, "..") == 0);
  1465. assert(info.type == LFS_TYPE_DIR);
  1466. lfs_dir_read(&lfs, &dir, &info) => 1;
  1467. assert(strcmp(info.name, "0.before") == 0);
  1468. assert(info.type == LFS_TYPE_REG);
  1469. assert(info.size == 7);
  1470. lfs_dir_read(&lfs, &dir, &info) => 1;
  1471. assert(strcmp(info.name, "2.after") == 0);
  1472. assert(info.type == LFS_TYPE_REG);
  1473. assert(info.size == 7);
  1474. lfs_dir_read(&lfs, &dir, &info) => 1;
  1475. assert(strcmp(info.name, "child") == 0);
  1476. assert(info.type == LFS_TYPE_DIR);
  1477. lfs_dir_read(&lfs, &dir, &info) => 0;
  1478. lfs_dir_close(&lfs, &dir) => 0;
  1479. lfs_dir_open(&lfs, &dir, "/parent/child") => 0;
  1480. lfs_dir_read(&lfs, &dir, &info) => 1;
  1481. assert(strcmp(info.name, ".") == 0);
  1482. assert(info.type == LFS_TYPE_DIR);
  1483. lfs_dir_read(&lfs, &dir, &info) => 1;
  1484. assert(strcmp(info.name, "..") == 0);
  1485. assert(info.type == LFS_TYPE_DIR);
  1486. lfs_dir_read(&lfs, &dir, &info) => 1;
  1487. assert(strcmp(info.name, "0.before") == 0);
  1488. assert(info.type == LFS_TYPE_REG);
  1489. assert(info.size == 7);
  1490. lfs_dir_read(&lfs, &dir, &info) => 1;
  1491. assert(strcmp(info.name, "1.move_me") == 0);
  1492. assert(info.type == LFS_TYPE_REG);
  1493. assert(info.size == sizeof("move me"));
  1494. lfs_dir_read(&lfs, &dir, &info) => 1;
  1495. assert(strcmp(info.name, "2.after") == 0);
  1496. assert(info.type == LFS_TYPE_REG);
  1497. assert(info.size == 7);
  1498. lfs_dir_read(&lfs, &dir, &info) => 0;
  1499. lfs_dir_close(&lfs, &dir) => 0;
  1500. lfs_file_open(&lfs, &file, "/parent/0.before", LFS_O_RDONLY) => 0;
  1501. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1502. assert(strcmp((char*)buffer, "test.5") == 0);
  1503. lfs_file_close(&lfs, &file) => 0;
  1504. lfs_file_open(&lfs, &file, "/parent/2.after", LFS_O_RDONLY) => 0;
  1505. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1506. assert(strcmp((char*)buffer, "test.6") == 0);
  1507. lfs_file_close(&lfs, &file) => 0;
  1508. lfs_file_open(&lfs, &file, "/parent/child/0.before", LFS_O_RDONLY) => 0;
  1509. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1510. assert(strcmp((char*)buffer, "test.7") == 0);
  1511. lfs_file_close(&lfs, &file) => 0;
  1512. lfs_file_open(&lfs, &file, "/parent/child/2.after", LFS_O_RDONLY) => 0;
  1513. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1514. assert(strcmp((char*)buffer, "test.8") == 0);
  1515. lfs_file_close(&lfs, &file) => 0;
  1516. lfs_unmount(&lfs) => 0;
  1517. '''
  1518. [[case]] # move fix in relocation with predecessor
  1519. in = "lfs.c"
  1520. define.RELOCATIONS = 'range(0x7+1)'
  1521. define.LFS_ERASE_CYCLES = 0xffffffff
  1522. code = '''
  1523. lfs_format(&lfs, &cfg) => 0;
  1524. lfs_mount(&lfs, &cfg) => 0;
  1525. lfs_mkdir(&lfs, "/parent") => 0;
  1526. lfs_mkdir(&lfs, "/parent/child") => 0;
  1527. lfs_mkdir(&lfs, "/parent/sibling") => 0;
  1528. lfs_file_open(&lfs, &file, "/parent/sibling/1.move_me",
  1529. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  1530. lfs_file_write(&lfs, &file, "move me",
  1531. sizeof("move me")) => sizeof("move me");
  1532. lfs_file_close(&lfs, &file) => 0;
  1533. lfs_file_open(&lfs, &file, "/parent/sibling/0.before",
  1534. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  1535. lfs_file_write(&lfs, &file, "test.1", 7) => 7;
  1536. lfs_file_close(&lfs, &file) => 0;
  1537. lfs_file_open(&lfs, &file, "/parent/sibling/2.after",
  1538. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  1539. lfs_file_write(&lfs, &file, "test.2", 7) => 7;
  1540. lfs_file_close(&lfs, &file) => 0;
  1541. lfs_file_open(&lfs, &file, "/parent/child/0.before",
  1542. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  1543. lfs_file_write(&lfs, &file, "test.3", 7) => 7;
  1544. lfs_file_close(&lfs, &file) => 0;
  1545. lfs_file_open(&lfs, &file, "/parent/child/2.after",
  1546. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  1547. lfs_file_write(&lfs, &file, "test.4", 7) => 7;
  1548. lfs_file_close(&lfs, &file) => 0;
  1549. lfs_file_t files[4];
  1550. lfs_file_open(&lfs, &files[0], "/parent/sibling/0.before",
  1551. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  1552. lfs_file_open(&lfs, &files[1], "/parent/sibling/2.after",
  1553. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  1554. lfs_file_open(&lfs, &files[2], "/parent/child/0.before",
  1555. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  1556. lfs_file_open(&lfs, &files[3], "/parent/child/2.after",
  1557. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  1558. lfs_file_write(&lfs, &files[0], "test.5", 7) => 7;
  1559. lfs_file_write(&lfs, &files[1], "test.6", 7) => 7;
  1560. lfs_file_write(&lfs, &files[2], "test.7", 7) => 7;
  1561. lfs_file_write(&lfs, &files[3], "test.8", 7) => 7;
  1562. // force specific directories to relocate
  1563. if (RELOCATIONS & 0x1) {
  1564. lfs_dir_open(&lfs, &dir, "/parent");
  1565. lfs_testbd_setwear(&cfg, dir.m.pair[0], 0xffffffff) => 0;
  1566. lfs_testbd_setwear(&cfg, dir.m.pair[1], 0xffffffff) => 0;
  1567. lfs_dir_close(&lfs, &dir) => 0;
  1568. }
  1569. if (RELOCATIONS & 0x2) {
  1570. lfs_dir_open(&lfs, &dir, "/parent/sibling");
  1571. lfs_testbd_setwear(&cfg, dir.m.pair[0], 0xffffffff) => 0;
  1572. lfs_testbd_setwear(&cfg, dir.m.pair[1], 0xffffffff) => 0;
  1573. lfs_dir_close(&lfs, &dir) => 0;
  1574. }
  1575. if (RELOCATIONS & 0x4) {
  1576. lfs_dir_open(&lfs, &dir, "/parent/child");
  1577. lfs_testbd_setwear(&cfg, dir.m.pair[0], 0xffffffff) => 0;
  1578. lfs_testbd_setwear(&cfg, dir.m.pair[1], 0xffffffff) => 0;
  1579. lfs_dir_close(&lfs, &dir) => 0;
  1580. }
  1581. // ok, now we move the file, this creates a move that needs to be
  1582. // fixed, possibly in a metadata-pair that needs to be relocated
  1583. //
  1584. // and now relocations can force us to need to fix our move in either
  1585. // the parent or child before things break
  1586. lfs_rename(&lfs,
  1587. "/parent/sibling/1.move_me",
  1588. "/parent/child/1.move_me") => 0;
  1589. lfs_file_close(&lfs, &files[0]) => 0;
  1590. lfs_file_close(&lfs, &files[1]) => 0;
  1591. lfs_file_close(&lfs, &files[2]) => 0;
  1592. lfs_file_close(&lfs, &files[3]) => 0;
  1593. // check that nothing was corrupted
  1594. lfs_dir_open(&lfs, &dir, "/parent") => 0;
  1595. lfs_dir_read(&lfs, &dir, &info) => 1;
  1596. assert(strcmp(info.name, ".") == 0);
  1597. assert(info.type == LFS_TYPE_DIR);
  1598. lfs_dir_read(&lfs, &dir, &info) => 1;
  1599. assert(strcmp(info.name, "..") == 0);
  1600. assert(info.type == LFS_TYPE_DIR);
  1601. lfs_dir_read(&lfs, &dir, &info) => 1;
  1602. assert(strcmp(info.name, "child") == 0);
  1603. assert(info.type == LFS_TYPE_DIR);
  1604. lfs_dir_read(&lfs, &dir, &info) => 1;
  1605. assert(strcmp(info.name, "sibling") == 0);
  1606. assert(info.type == LFS_TYPE_DIR);
  1607. lfs_dir_read(&lfs, &dir, &info) => 0;
  1608. lfs_dir_close(&lfs, &dir) => 0;
  1609. lfs_dir_open(&lfs, &dir, "/parent/sibling") => 0;
  1610. lfs_dir_read(&lfs, &dir, &info) => 1;
  1611. assert(strcmp(info.name, ".") == 0);
  1612. assert(info.type == LFS_TYPE_DIR);
  1613. lfs_dir_read(&lfs, &dir, &info) => 1;
  1614. assert(strcmp(info.name, "..") == 0);
  1615. assert(info.type == LFS_TYPE_DIR);
  1616. lfs_dir_read(&lfs, &dir, &info) => 1;
  1617. assert(strcmp(info.name, "0.before") == 0);
  1618. assert(info.type == LFS_TYPE_REG);
  1619. assert(info.size == 7);
  1620. lfs_dir_read(&lfs, &dir, &info) => 1;
  1621. assert(strcmp(info.name, "2.after") == 0);
  1622. assert(info.type == LFS_TYPE_REG);
  1623. assert(info.size == 7);
  1624. lfs_dir_read(&lfs, &dir, &info) => 0;
  1625. lfs_dir_close(&lfs, &dir) => 0;
  1626. lfs_dir_open(&lfs, &dir, "/parent/child") => 0;
  1627. lfs_dir_read(&lfs, &dir, &info) => 1;
  1628. assert(strcmp(info.name, ".") == 0);
  1629. assert(info.type == LFS_TYPE_DIR);
  1630. lfs_dir_read(&lfs, &dir, &info) => 1;
  1631. assert(strcmp(info.name, "..") == 0);
  1632. assert(info.type == LFS_TYPE_DIR);
  1633. lfs_dir_read(&lfs, &dir, &info) => 1;
  1634. assert(strcmp(info.name, "0.before") == 0);
  1635. assert(info.type == LFS_TYPE_REG);
  1636. assert(info.size == 7);
  1637. lfs_dir_read(&lfs, &dir, &info) => 1;
  1638. assert(strcmp(info.name, "1.move_me") == 0);
  1639. assert(info.type == LFS_TYPE_REG);
  1640. assert(info.size == sizeof("move me"));
  1641. lfs_dir_read(&lfs, &dir, &info) => 1;
  1642. assert(strcmp(info.name, "2.after") == 0);
  1643. assert(info.type == LFS_TYPE_REG);
  1644. assert(info.size == 7);
  1645. lfs_dir_read(&lfs, &dir, &info) => 0;
  1646. lfs_dir_close(&lfs, &dir) => 0;
  1647. lfs_file_open(&lfs, &file, "/parent/sibling/0.before", LFS_O_RDONLY) => 0;
  1648. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1649. assert(strcmp((char*)buffer, "test.5") == 0);
  1650. lfs_file_close(&lfs, &file) => 0;
  1651. lfs_file_open(&lfs, &file, "/parent/sibling/2.after", LFS_O_RDONLY) => 0;
  1652. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1653. assert(strcmp((char*)buffer, "test.6") == 0);
  1654. lfs_file_close(&lfs, &file) => 0;
  1655. lfs_file_open(&lfs, &file, "/parent/child/0.before", LFS_O_RDONLY) => 0;
  1656. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1657. assert(strcmp((char*)buffer, "test.7") == 0);
  1658. lfs_file_close(&lfs, &file) => 0;
  1659. lfs_file_open(&lfs, &file, "/parent/child/2.after", LFS_O_RDONLY) => 0;
  1660. lfs_file_read(&lfs, &file, buffer, 7) => 7;
  1661. assert(strcmp((char*)buffer, "test.8") == 0);
  1662. lfs_file_close(&lfs, &file) => 0;
  1663. lfs_unmount(&lfs) => 0;
  1664. '''