test_alloc.sh 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485
  1. #!/bin/bash
  2. set -eu
  3. echo "=== Allocator tests ==="
  4. rm -rf blocks
  5. tests/test.py << TEST
  6. lfs_format(&lfs, &cfg) => 0;
  7. TEST
  8. SIZE=15000
  9. lfs_mkdir() {
  10. tests/test.py << TEST
  11. lfs_mount(&lfs, &cfg) => 0;
  12. lfs_mkdir(&lfs, "$1") => 0;
  13. lfs_unmount(&lfs) => 0;
  14. TEST
  15. }
  16. lfs_remove() {
  17. tests/test.py << TEST
  18. lfs_mount(&lfs, &cfg) => 0;
  19. lfs_remove(&lfs, "$1/eggs") => 0;
  20. lfs_remove(&lfs, "$1/bacon") => 0;
  21. lfs_remove(&lfs, "$1/pancakes") => 0;
  22. lfs_remove(&lfs, "$1") => 0;
  23. lfs_unmount(&lfs) => 0;
  24. TEST
  25. }
  26. lfs_alloc_singleproc() {
  27. tests/test.py << TEST
  28. const char *names[] = {"bacon", "eggs", "pancakes"};
  29. lfs_mount(&lfs, &cfg) => 0;
  30. for (unsigned n = 0; n < sizeof(names)/sizeof(names[0]); n++) {
  31. sprintf((char*)buffer, "$1/%s", names[n]);
  32. lfs_file_open(&lfs, &file[n], (char*)buffer,
  33. LFS_O_WRONLY | LFS_O_CREAT | LFS_O_APPEND) => 0;
  34. }
  35. for (unsigned n = 0; n < sizeof(names)/sizeof(names[0]); n++) {
  36. size = strlen(names[n]);
  37. for (int i = 0; i < $SIZE; i++) {
  38. lfs_file_write(&lfs, &file[n], names[n], size) => size;
  39. }
  40. }
  41. for (unsigned n = 0; n < sizeof(names)/sizeof(names[0]); n++) {
  42. lfs_file_close(&lfs, &file[n]) => 0;
  43. }
  44. lfs_unmount(&lfs) => 0;
  45. TEST
  46. }
  47. lfs_alloc_multiproc() {
  48. for name in bacon eggs pancakes
  49. do
  50. tests/test.py << TEST
  51. lfs_mount(&lfs, &cfg) => 0;
  52. lfs_file_open(&lfs, &file[0], "$1/$name",
  53. LFS_O_WRONLY | LFS_O_CREAT | LFS_O_APPEND) => 0;
  54. size = strlen("$name");
  55. memcpy(buffer, "$name", size);
  56. for (int i = 0; i < $SIZE; i++) {
  57. lfs_file_write(&lfs, &file[0], buffer, size) => size;
  58. }
  59. lfs_file_close(&lfs, &file[0]) => 0;
  60. lfs_unmount(&lfs) => 0;
  61. TEST
  62. done
  63. }
  64. lfs_verify() {
  65. for name in bacon eggs pancakes
  66. do
  67. tests/test.py << TEST
  68. lfs_mount(&lfs, &cfg) => 0;
  69. lfs_file_open(&lfs, &file[0], "$1/$name", LFS_O_RDONLY) => 0;
  70. size = strlen("$name");
  71. for (int i = 0; i < $SIZE; i++) {
  72. lfs_file_read(&lfs, &file[0], buffer, size) => size;
  73. memcmp(buffer, "$name", size) => 0;
  74. }
  75. lfs_file_close(&lfs, &file[0]) => 0;
  76. lfs_unmount(&lfs) => 0;
  77. TEST
  78. done
  79. }
  80. echo "--- Single-process allocation test ---"
  81. lfs_mkdir singleproc
  82. lfs_alloc_singleproc singleproc
  83. lfs_verify singleproc
  84. echo "--- Multi-process allocation test ---"
  85. lfs_mkdir multiproc
  86. lfs_alloc_multiproc multiproc
  87. lfs_verify multiproc
  88. lfs_verify singleproc
  89. echo "--- Single-process reuse test ---"
  90. lfs_remove singleproc
  91. lfs_mkdir singleprocreuse
  92. lfs_alloc_singleproc singleprocreuse
  93. lfs_verify singleprocreuse
  94. lfs_verify multiproc
  95. echo "--- Multi-process reuse test ---"
  96. lfs_remove multiproc
  97. lfs_mkdir multiprocreuse
  98. lfs_alloc_singleproc multiprocreuse
  99. lfs_verify multiprocreuse
  100. lfs_verify singleprocreuse
  101. echo "--- Cleanup ---"
  102. lfs_remove multiprocreuse
  103. lfs_remove singleprocreuse
  104. echo "--- Exhaustion test ---"
  105. tests/test.py << TEST
  106. lfs_mount(&lfs, &cfg) => 0;
  107. lfs_file_open(&lfs, &file[0], "exhaustion", LFS_O_WRONLY | LFS_O_CREAT);
  108. size = strlen("exhaustion");
  109. memcpy(buffer, "exhaustion", size);
  110. lfs_file_write(&lfs, &file[0], buffer, size) => size;
  111. lfs_file_sync(&lfs, &file[0]) => 0;
  112. size = strlen("blahblahblahblah");
  113. memcpy(buffer, "blahblahblahblah", size);
  114. lfs_ssize_t res;
  115. while (true) {
  116. res = lfs_file_write(&lfs, &file[0], buffer, size);
  117. if (res < 0) {
  118. break;
  119. }
  120. res => size;
  121. }
  122. res => LFS_ERR_NOSPC;
  123. lfs_file_close(&lfs, &file[0]) => 0;
  124. lfs_unmount(&lfs) => 0;
  125. TEST
  126. tests/test.py << TEST
  127. lfs_mount(&lfs, &cfg) => 0;
  128. lfs_file_open(&lfs, &file[0], "exhaustion", LFS_O_RDONLY);
  129. size = strlen("exhaustion");
  130. lfs_file_size(&lfs, &file[0]) => size;
  131. lfs_file_read(&lfs, &file[0], buffer, size) => size;
  132. memcmp(buffer, "exhaustion", size) => 0;
  133. lfs_file_close(&lfs, &file[0]) => 0;
  134. lfs_unmount(&lfs) => 0;
  135. TEST
  136. echo "--- Exhaustion wraparound test ---"
  137. tests/test.py << TEST
  138. lfs_mount(&lfs, &cfg) => 0;
  139. lfs_remove(&lfs, "exhaustion") => 0;
  140. lfs_file_open(&lfs, &file[0], "padding", LFS_O_WRONLY | LFS_O_CREAT);
  141. size = strlen("buffering");
  142. memcpy(buffer, "buffering", size);
  143. for (int i = 0; i < $SIZE; i++) {
  144. lfs_file_write(&lfs, &file[0], buffer, size) => size;
  145. }
  146. lfs_file_close(&lfs, &file[0]) => 0;
  147. lfs_remove(&lfs, "padding") => 0;
  148. lfs_file_open(&lfs, &file[0], "exhaustion", LFS_O_WRONLY | LFS_O_CREAT);
  149. size = strlen("exhaustion");
  150. memcpy(buffer, "exhaustion", size);
  151. lfs_file_write(&lfs, &file[0], buffer, size) => size;
  152. lfs_file_sync(&lfs, &file[0]) => 0;
  153. size = strlen("blahblahblahblah");
  154. memcpy(buffer, "blahblahblahblah", size);
  155. lfs_ssize_t res;
  156. while (true) {
  157. res = lfs_file_write(&lfs, &file[0], buffer, size);
  158. if (res < 0) {
  159. break;
  160. }
  161. res => size;
  162. }
  163. res => LFS_ERR_NOSPC;
  164. lfs_file_close(&lfs, &file[0]) => 0;
  165. lfs_unmount(&lfs) => 0;
  166. TEST
  167. tests/test.py << TEST
  168. lfs_mount(&lfs, &cfg) => 0;
  169. lfs_file_open(&lfs, &file[0], "exhaustion", LFS_O_RDONLY);
  170. size = strlen("exhaustion");
  171. lfs_file_size(&lfs, &file[0]) => size;
  172. lfs_file_read(&lfs, &file[0], buffer, size) => size;
  173. memcmp(buffer, "exhaustion", size) => 0;
  174. lfs_file_close(&lfs, &file[0]) => 0;
  175. lfs_remove(&lfs, "exhaustion") => 0;
  176. lfs_unmount(&lfs) => 0;
  177. TEST
  178. echo "--- Dir exhaustion test ---"
  179. tests/test.py << TEST
  180. lfs_mount(&lfs, &cfg) => 0;
  181. // find out max file size
  182. lfs_mkdir(&lfs, "exhaustiondir") => 0;
  183. size = strlen("blahblahblahblah");
  184. memcpy(buffer, "blahblahblahblah", size);
  185. lfs_file_open(&lfs, &file[0], "exhaustion", LFS_O_WRONLY | LFS_O_CREAT);
  186. int count = 0;
  187. int err;
  188. while (true) {
  189. err = lfs_file_write(&lfs, &file[0], buffer, size);
  190. if (err < 0) {
  191. break;
  192. }
  193. count += 1;
  194. }
  195. err => LFS_ERR_NOSPC;
  196. lfs_file_close(&lfs, &file[0]) => 0;
  197. lfs_remove(&lfs, "exhaustion") => 0;
  198. lfs_remove(&lfs, "exhaustiondir") => 0;
  199. // see if dir fits with max file size
  200. lfs_file_open(&lfs, &file[0], "exhaustion", LFS_O_WRONLY | LFS_O_CREAT);
  201. for (int i = 0; i < count; i++) {
  202. lfs_file_write(&lfs, &file[0], buffer, size) => size;
  203. }
  204. lfs_file_close(&lfs, &file[0]) => 0;
  205. lfs_mkdir(&lfs, "exhaustiondir") => 0;
  206. lfs_remove(&lfs, "exhaustiondir") => 0;
  207. lfs_remove(&lfs, "exhaustion") => 0;
  208. // see if dir fits with > max file size
  209. lfs_file_open(&lfs, &file[0], "exhaustion", LFS_O_WRONLY | LFS_O_CREAT);
  210. for (int i = 0; i < count+1; i++) {
  211. lfs_file_write(&lfs, &file[0], buffer, size) => size;
  212. }
  213. lfs_file_close(&lfs, &file[0]) => 0;
  214. lfs_mkdir(&lfs, "exhaustiondir") => LFS_ERR_NOSPC;
  215. lfs_remove(&lfs, "exhaustion") => 0;
  216. lfs_unmount(&lfs) => 0;
  217. TEST
  218. echo "--- Chained dir exhaustion test ---"
  219. tests/test.py << TEST
  220. lfs_mount(&lfs, &cfg) => 0;
  221. // find out max file size
  222. lfs_mkdir(&lfs, "exhaustiondir") => 0;
  223. for (int i = 0; i < 9; i++) {
  224. sprintf((char*)buffer, "dirwithanexhaustivelylongnameforpadding%d", i);
  225. lfs_mkdir(&lfs, (char*)buffer) => 0;
  226. }
  227. size = strlen("blahblahblahblah");
  228. memcpy(buffer, "blahblahblahblah", size);
  229. lfs_file_open(&lfs, &file[0], "exhaustion", LFS_O_WRONLY | LFS_O_CREAT);
  230. int count = 0;
  231. int err;
  232. while (true) {
  233. err = lfs_file_write(&lfs, &file[0], buffer, size);
  234. if (err < 0) {
  235. break;
  236. }
  237. count += 1;
  238. }
  239. err => LFS_ERR_NOSPC;
  240. lfs_file_close(&lfs, &file[0]) => 0;
  241. lfs_remove(&lfs, "exhaustion") => 0;
  242. lfs_remove(&lfs, "exhaustiondir") => 0;
  243. for (int i = 0; i < 9; i++) {
  244. sprintf((char*)buffer, "dirwithanexhaustivelylongnameforpadding%d", i);
  245. lfs_remove(&lfs, (char*)buffer) => 0;
  246. }
  247. // see that chained dir fails
  248. lfs_file_open(&lfs, &file[0], "exhaustion", LFS_O_WRONLY | LFS_O_CREAT);
  249. for (int i = 0; i < count+1; i++) {
  250. lfs_file_write(&lfs, &file[0], buffer, size) => size;
  251. }
  252. lfs_file_sync(&lfs, &file[0]) => 0;
  253. for (int i = 0; i < 9; i++) {
  254. sprintf((char*)buffer, "dirwithanexhaustivelylongnameforpadding%d", i);
  255. lfs_mkdir(&lfs, (char*)buffer) => 0;
  256. }
  257. lfs_mkdir(&lfs, "exhaustiondir") => LFS_ERR_NOSPC;
  258. // shorten file to try a second chained dir
  259. while (true) {
  260. err = lfs_mkdir(&lfs, "exhaustiondir");
  261. if (err != LFS_ERR_NOSPC) {
  262. break;
  263. }
  264. lfs_ssize_t filesize = lfs_file_size(&lfs, &file[0]);
  265. filesize > 0 => true;
  266. lfs_file_truncate(&lfs, &file[0], filesize - size) => 0;
  267. lfs_file_sync(&lfs, &file[0]) => 0;
  268. }
  269. err => 0;
  270. lfs_mkdir(&lfs, "exhaustiondir2") => LFS_ERR_NOSPC;
  271. lfs_file_close(&lfs, &file[0]) => 0;
  272. lfs_unmount(&lfs) => 0;
  273. TEST
  274. echo "--- Split dir test ---"
  275. rm -rf blocks
  276. tests/test.py << TEST
  277. lfs_format(&lfs, &cfg) => 0;
  278. TEST
  279. tests/test.py << TEST
  280. lfs_mount(&lfs, &cfg) => 0;
  281. // create one block hole for half a directory
  282. lfs_file_open(&lfs, &file[0], "bump", LFS_O_WRONLY | LFS_O_CREAT) => 0;
  283. for (lfs_size_t i = 0; i < cfg.block_size; i += 2) {
  284. memcpy(&buffer[i], "hi", 2);
  285. }
  286. lfs_file_write(&lfs, &file[0], buffer, cfg.block_size) => cfg.block_size;
  287. lfs_file_close(&lfs, &file[0]) => 0;
  288. lfs_file_open(&lfs, &file[0], "exhaustion", LFS_O_WRONLY | LFS_O_CREAT);
  289. size = strlen("blahblahblahblah");
  290. memcpy(buffer, "blahblahblahblah", size);
  291. for (lfs_size_t i = 0;
  292. i < (cfg.block_count-4)*(cfg.block_size-8);
  293. i += size) {
  294. lfs_file_write(&lfs, &file[0], buffer, size) => size;
  295. }
  296. lfs_file_close(&lfs, &file[0]) => 0;
  297. // remount to force reset of lookahead
  298. lfs_unmount(&lfs) => 0;
  299. lfs_mount(&lfs, &cfg) => 0;
  300. // open hole
  301. lfs_remove(&lfs, "bump") => 0;
  302. lfs_mkdir(&lfs, "splitdir") => 0;
  303. lfs_file_open(&lfs, &file[0], "splitdir/bump",
  304. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  305. for (lfs_size_t i = 0; i < cfg.block_size; i += 2) {
  306. memcpy(&buffer[i], "hi", 2);
  307. }
  308. lfs_file_write(&lfs, &file[0], buffer, 2*cfg.block_size) => LFS_ERR_NOSPC;
  309. lfs_file_close(&lfs, &file[0]) => 0;
  310. lfs_unmount(&lfs) => 0;
  311. TEST
  312. echo "--- Outdated lookahead test ---"
  313. rm -rf blocks
  314. tests/test.py << TEST
  315. lfs_format(&lfs, &cfg) => 0;
  316. lfs_mount(&lfs, &cfg) => 0;
  317. // fill completely with two files
  318. lfs_file_open(&lfs, &file[0], "exhaustion1",
  319. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  320. size = strlen("blahblahblahblah");
  321. memcpy(buffer, "blahblahblahblah", size);
  322. for (lfs_size_t i = 0;
  323. i < ((cfg.block_count-2)/2)*(cfg.block_size-8);
  324. i += size) {
  325. lfs_file_write(&lfs, &file[0], buffer, size) => size;
  326. }
  327. lfs_file_close(&lfs, &file[0]) => 0;
  328. lfs_file_open(&lfs, &file[0], "exhaustion2",
  329. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  330. size = strlen("blahblahblahblah");
  331. memcpy(buffer, "blahblahblahblah", size);
  332. for (lfs_size_t i = 0;
  333. i < ((cfg.block_count-2+1)/2)*(cfg.block_size-8);
  334. i += size) {
  335. lfs_file_write(&lfs, &file[0], buffer, size) => size;
  336. }
  337. lfs_file_close(&lfs, &file[0]) => 0;
  338. // remount to force reset of lookahead
  339. lfs_unmount(&lfs) => 0;
  340. lfs_mount(&lfs, &cfg) => 0;
  341. // rewrite one file
  342. lfs_file_open(&lfs, &file[0], "exhaustion1",
  343. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  344. lfs_file_sync(&lfs, &file[0]) => 0;
  345. size = strlen("blahblahblahblah");
  346. memcpy(buffer, "blahblahblahblah", size);
  347. for (lfs_size_t i = 0;
  348. i < ((cfg.block_count-2)/2)*(cfg.block_size-8);
  349. i += size) {
  350. lfs_file_write(&lfs, &file[0], buffer, size) => size;
  351. }
  352. lfs_file_close(&lfs, &file[0]) => 0;
  353. // rewrite second file, this requires lookahead does not
  354. // use old population
  355. lfs_file_open(&lfs, &file[0], "exhaustion2",
  356. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  357. lfs_file_sync(&lfs, &file[0]) => 0;
  358. size = strlen("blahblahblahblah");
  359. memcpy(buffer, "blahblahblahblah", size);
  360. for (lfs_size_t i = 0;
  361. i < ((cfg.block_count-2+1)/2)*(cfg.block_size-8);
  362. i += size) {
  363. lfs_file_write(&lfs, &file[0], buffer, size) => size;
  364. }
  365. lfs_file_close(&lfs, &file[0]) => 0;
  366. TEST
  367. echo "--- Outdated lookahead and split dir test ---"
  368. rm -rf blocks
  369. tests/test.py << TEST
  370. lfs_format(&lfs, &cfg) => 0;
  371. lfs_mount(&lfs, &cfg) => 0;
  372. // fill completely with two files
  373. lfs_file_open(&lfs, &file[0], "exhaustion1",
  374. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  375. size = strlen("blahblahblahblah");
  376. memcpy(buffer, "blahblahblahblah", size);
  377. for (lfs_size_t i = 0;
  378. i < ((cfg.block_count-2)/2)*(cfg.block_size-8);
  379. i += size) {
  380. lfs_file_write(&lfs, &file[0], buffer, size) => size;
  381. }
  382. lfs_file_close(&lfs, &file[0]) => 0;
  383. lfs_file_open(&lfs, &file[0], "exhaustion2",
  384. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  385. size = strlen("blahblahblahblah");
  386. memcpy(buffer, "blahblahblahblah", size);
  387. for (lfs_size_t i = 0;
  388. i < ((cfg.block_count-2+1)/2)*(cfg.block_size-8);
  389. i += size) {
  390. lfs_file_write(&lfs, &file[0], buffer, size) => size;
  391. }
  392. lfs_file_close(&lfs, &file[0]) => 0;
  393. // remount to force reset of lookahead
  394. lfs_unmount(&lfs) => 0;
  395. lfs_mount(&lfs, &cfg) => 0;
  396. // rewrite one file with a hole of one block
  397. lfs_file_open(&lfs, &file[0], "exhaustion1",
  398. LFS_O_WRONLY | LFS_O_TRUNC) => 0;
  399. lfs_file_sync(&lfs, &file[0]) => 0;
  400. size = strlen("blahblahblahblah");
  401. memcpy(buffer, "blahblahblahblah", size);
  402. for (lfs_size_t i = 0;
  403. i < ((cfg.block_count-2)/2 - 1)*(cfg.block_size-8);
  404. i += size) {
  405. lfs_file_write(&lfs, &file[0], buffer, size) => size;
  406. }
  407. lfs_file_close(&lfs, &file[0]) => 0;
  408. // try to allocate a directory, should fail!
  409. lfs_mkdir(&lfs, "split") => LFS_ERR_NOSPC;
  410. // file should not fail
  411. lfs_file_open(&lfs, &file[0], "notasplit",
  412. LFS_O_WRONLY | LFS_O_CREAT) => 0;
  413. lfs_file_write(&lfs, &file[0], "hi", 2) => 2;
  414. lfs_file_close(&lfs, &file[0]) => 0;
  415. lfs_unmount(&lfs) => 0;
  416. TEST
  417. echo "--- Results ---"
  418. tests/stats.py