test_alloc.sh 14 KB

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