free-space-tree-tests.c 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Copyright (C) 2015 Facebook. All rights reserved.
  4. */
  5. #include <linux/types.h>
  6. #include "btrfs-tests.h"
  7. #include "../ctree.h"
  8. #include "../disk-io.h"
  9. #include "../free-space-tree.h"
  10. #include "../transaction.h"
  11. struct free_space_extent {
  12. u64 start;
  13. u64 length;
  14. };
  15. static int __check_free_space_extents(struct btrfs_trans_handle *trans,
  16. struct btrfs_fs_info *fs_info,
  17. struct btrfs_block_group_cache *cache,
  18. struct btrfs_path *path,
  19. const struct free_space_extent * const extents,
  20. unsigned int num_extents)
  21. {
  22. struct btrfs_free_space_info *info;
  23. struct btrfs_key key;
  24. int prev_bit = 0, bit;
  25. u64 extent_start = 0, offset, end;
  26. u32 flags, extent_count;
  27. unsigned int i;
  28. int ret;
  29. info = search_free_space_info(trans, fs_info, cache, path, 0);
  30. if (IS_ERR(info)) {
  31. test_err("could not find free space info");
  32. ret = PTR_ERR(info);
  33. goto out;
  34. }
  35. flags = btrfs_free_space_flags(path->nodes[0], info);
  36. extent_count = btrfs_free_space_extent_count(path->nodes[0], info);
  37. if (extent_count != num_extents) {
  38. test_err("extent count is wrong");
  39. ret = -EINVAL;
  40. goto out;
  41. }
  42. if (flags & BTRFS_FREE_SPACE_USING_BITMAPS) {
  43. if (path->slots[0] != 0)
  44. goto invalid;
  45. end = cache->key.objectid + cache->key.offset;
  46. i = 0;
  47. while (++path->slots[0] < btrfs_header_nritems(path->nodes[0])) {
  48. btrfs_item_key_to_cpu(path->nodes[0], &key, path->slots[0]);
  49. if (key.type != BTRFS_FREE_SPACE_BITMAP_KEY)
  50. goto invalid;
  51. offset = key.objectid;
  52. while (offset < key.objectid + key.offset) {
  53. bit = free_space_test_bit(cache, path, offset);
  54. if (prev_bit == 0 && bit == 1) {
  55. extent_start = offset;
  56. } else if (prev_bit == 1 && bit == 0) {
  57. if (i >= num_extents)
  58. goto invalid;
  59. if (i >= num_extents ||
  60. extent_start != extents[i].start ||
  61. offset - extent_start != extents[i].length)
  62. goto invalid;
  63. i++;
  64. }
  65. prev_bit = bit;
  66. offset += fs_info->sectorsize;
  67. }
  68. }
  69. if (prev_bit == 1) {
  70. if (i >= num_extents ||
  71. extent_start != extents[i].start ||
  72. end - extent_start != extents[i].length)
  73. goto invalid;
  74. i++;
  75. }
  76. if (i != num_extents)
  77. goto invalid;
  78. } else {
  79. if (btrfs_header_nritems(path->nodes[0]) != num_extents + 1 ||
  80. path->slots[0] != 0)
  81. goto invalid;
  82. for (i = 0; i < num_extents; i++) {
  83. path->slots[0]++;
  84. btrfs_item_key_to_cpu(path->nodes[0], &key, path->slots[0]);
  85. if (key.type != BTRFS_FREE_SPACE_EXTENT_KEY ||
  86. key.objectid != extents[i].start ||
  87. key.offset != extents[i].length)
  88. goto invalid;
  89. }
  90. }
  91. ret = 0;
  92. out:
  93. btrfs_release_path(path);
  94. return ret;
  95. invalid:
  96. test_err("free space tree is invalid");
  97. ret = -EINVAL;
  98. goto out;
  99. }
  100. static int check_free_space_extents(struct btrfs_trans_handle *trans,
  101. struct btrfs_fs_info *fs_info,
  102. struct btrfs_block_group_cache *cache,
  103. struct btrfs_path *path,
  104. const struct free_space_extent * const extents,
  105. unsigned int num_extents)
  106. {
  107. struct btrfs_free_space_info *info;
  108. u32 flags;
  109. int ret;
  110. info = search_free_space_info(trans, fs_info, cache, path, 0);
  111. if (IS_ERR(info)) {
  112. test_err("could not find free space info");
  113. btrfs_release_path(path);
  114. return PTR_ERR(info);
  115. }
  116. flags = btrfs_free_space_flags(path->nodes[0], info);
  117. btrfs_release_path(path);
  118. ret = __check_free_space_extents(trans, fs_info, cache, path, extents,
  119. num_extents);
  120. if (ret)
  121. return ret;
  122. /* Flip it to the other format and check that for good measure. */
  123. if (flags & BTRFS_FREE_SPACE_USING_BITMAPS) {
  124. ret = convert_free_space_to_extents(trans, cache, path);
  125. if (ret) {
  126. test_err("could not convert to extents");
  127. return ret;
  128. }
  129. } else {
  130. ret = convert_free_space_to_bitmaps(trans, cache, path);
  131. if (ret) {
  132. test_err("could not convert to bitmaps");
  133. return ret;
  134. }
  135. }
  136. return __check_free_space_extents(trans, fs_info, cache, path, extents,
  137. num_extents);
  138. }
  139. static int test_empty_block_group(struct btrfs_trans_handle *trans,
  140. struct btrfs_fs_info *fs_info,
  141. struct btrfs_block_group_cache *cache,
  142. struct btrfs_path *path,
  143. u32 alignment)
  144. {
  145. const struct free_space_extent extents[] = {
  146. {cache->key.objectid, cache->key.offset},
  147. };
  148. return check_free_space_extents(trans, fs_info, cache, path,
  149. extents, ARRAY_SIZE(extents));
  150. }
  151. static int test_remove_all(struct btrfs_trans_handle *trans,
  152. struct btrfs_fs_info *fs_info,
  153. struct btrfs_block_group_cache *cache,
  154. struct btrfs_path *path,
  155. u32 alignment)
  156. {
  157. const struct free_space_extent extents[] = {};
  158. int ret;
  159. ret = __remove_from_free_space_tree(trans, cache, path,
  160. cache->key.objectid,
  161. cache->key.offset);
  162. if (ret) {
  163. test_err("could not remove free space");
  164. return ret;
  165. }
  166. return check_free_space_extents(trans, fs_info, cache, path,
  167. extents, ARRAY_SIZE(extents));
  168. }
  169. static int test_remove_beginning(struct btrfs_trans_handle *trans,
  170. struct btrfs_fs_info *fs_info,
  171. struct btrfs_block_group_cache *cache,
  172. struct btrfs_path *path,
  173. u32 alignment)
  174. {
  175. const struct free_space_extent extents[] = {
  176. {cache->key.objectid + alignment,
  177. cache->key.offset - alignment},
  178. };
  179. int ret;
  180. ret = __remove_from_free_space_tree(trans, cache, path,
  181. cache->key.objectid, alignment);
  182. if (ret) {
  183. test_err("could not remove free space");
  184. return ret;
  185. }
  186. return check_free_space_extents(trans, fs_info, cache, path,
  187. extents, ARRAY_SIZE(extents));
  188. }
  189. static int test_remove_end(struct btrfs_trans_handle *trans,
  190. struct btrfs_fs_info *fs_info,
  191. struct btrfs_block_group_cache *cache,
  192. struct btrfs_path *path,
  193. u32 alignment)
  194. {
  195. const struct free_space_extent extents[] = {
  196. {cache->key.objectid, cache->key.offset - alignment},
  197. };
  198. int ret;
  199. ret = __remove_from_free_space_tree(trans, cache, path,
  200. cache->key.objectid +
  201. cache->key.offset - alignment,
  202. alignment);
  203. if (ret) {
  204. test_err("could not remove free space");
  205. return ret;
  206. }
  207. return check_free_space_extents(trans, fs_info, cache, path,
  208. extents, ARRAY_SIZE(extents));
  209. }
  210. static int test_remove_middle(struct btrfs_trans_handle *trans,
  211. struct btrfs_fs_info *fs_info,
  212. struct btrfs_block_group_cache *cache,
  213. struct btrfs_path *path,
  214. u32 alignment)
  215. {
  216. const struct free_space_extent extents[] = {
  217. {cache->key.objectid, alignment},
  218. {cache->key.objectid + 2 * alignment,
  219. cache->key.offset - 2 * alignment},
  220. };
  221. int ret;
  222. ret = __remove_from_free_space_tree(trans, cache, path,
  223. cache->key.objectid + alignment,
  224. alignment);
  225. if (ret) {
  226. test_err("could not remove free space");
  227. return ret;
  228. }
  229. return check_free_space_extents(trans, fs_info, cache, path,
  230. extents, ARRAY_SIZE(extents));
  231. }
  232. static int test_merge_left(struct btrfs_trans_handle *trans,
  233. struct btrfs_fs_info *fs_info,
  234. struct btrfs_block_group_cache *cache,
  235. struct btrfs_path *path,
  236. u32 alignment)
  237. {
  238. const struct free_space_extent extents[] = {
  239. {cache->key.objectid, 2 * alignment},
  240. };
  241. int ret;
  242. ret = __remove_from_free_space_tree(trans, cache, path,
  243. cache->key.objectid,
  244. cache->key.offset);
  245. if (ret) {
  246. test_err("could not remove free space");
  247. return ret;
  248. }
  249. ret = __add_to_free_space_tree(trans, cache, path, cache->key.objectid,
  250. alignment);
  251. if (ret) {
  252. test_err("could not add free space");
  253. return ret;
  254. }
  255. ret = __add_to_free_space_tree(trans, cache, path,
  256. cache->key.objectid + alignment,
  257. alignment);
  258. if (ret) {
  259. test_err("could not add free space");
  260. return ret;
  261. }
  262. return check_free_space_extents(trans, fs_info, cache, path,
  263. extents, ARRAY_SIZE(extents));
  264. }
  265. static int test_merge_right(struct btrfs_trans_handle *trans,
  266. struct btrfs_fs_info *fs_info,
  267. struct btrfs_block_group_cache *cache,
  268. struct btrfs_path *path,
  269. u32 alignment)
  270. {
  271. const struct free_space_extent extents[] = {
  272. {cache->key.objectid + alignment, 2 * alignment},
  273. };
  274. int ret;
  275. ret = __remove_from_free_space_tree(trans, cache, path,
  276. cache->key.objectid,
  277. cache->key.offset);
  278. if (ret) {
  279. test_err("could not remove free space");
  280. return ret;
  281. }
  282. ret = __add_to_free_space_tree(trans, cache, path,
  283. cache->key.objectid + 2 * alignment,
  284. alignment);
  285. if (ret) {
  286. test_err("could not add free space");
  287. return ret;
  288. }
  289. ret = __add_to_free_space_tree(trans, cache, path,
  290. cache->key.objectid + alignment,
  291. alignment);
  292. if (ret) {
  293. test_err("could not add free space");
  294. return ret;
  295. }
  296. return check_free_space_extents(trans, fs_info, cache, path,
  297. extents, ARRAY_SIZE(extents));
  298. }
  299. static int test_merge_both(struct btrfs_trans_handle *trans,
  300. struct btrfs_fs_info *fs_info,
  301. struct btrfs_block_group_cache *cache,
  302. struct btrfs_path *path,
  303. u32 alignment)
  304. {
  305. const struct free_space_extent extents[] = {
  306. {cache->key.objectid, 3 * alignment},
  307. };
  308. int ret;
  309. ret = __remove_from_free_space_tree(trans, cache, path,
  310. cache->key.objectid,
  311. cache->key.offset);
  312. if (ret) {
  313. test_err("could not remove free space");
  314. return ret;
  315. }
  316. ret = __add_to_free_space_tree(trans, cache, path, cache->key.objectid,
  317. alignment);
  318. if (ret) {
  319. test_err("could not add free space");
  320. return ret;
  321. }
  322. ret = __add_to_free_space_tree(trans, cache, path,
  323. cache->key.objectid + 2 * alignment,
  324. alignment);
  325. if (ret) {
  326. test_err("could not add free space");
  327. return ret;
  328. }
  329. ret = __add_to_free_space_tree(trans, cache, path,
  330. cache->key.objectid + alignment,
  331. alignment);
  332. if (ret) {
  333. test_err("could not add free space");
  334. return ret;
  335. }
  336. return check_free_space_extents(trans, fs_info, cache, path,
  337. extents, ARRAY_SIZE(extents));
  338. }
  339. static int test_merge_none(struct btrfs_trans_handle *trans,
  340. struct btrfs_fs_info *fs_info,
  341. struct btrfs_block_group_cache *cache,
  342. struct btrfs_path *path,
  343. u32 alignment)
  344. {
  345. const struct free_space_extent extents[] = {
  346. {cache->key.objectid, alignment},
  347. {cache->key.objectid + 2 * alignment, alignment},
  348. {cache->key.objectid + 4 * alignment, alignment},
  349. };
  350. int ret;
  351. ret = __remove_from_free_space_tree(trans, cache, path,
  352. cache->key.objectid,
  353. cache->key.offset);
  354. if (ret) {
  355. test_err("could not remove free space");
  356. return ret;
  357. }
  358. ret = __add_to_free_space_tree(trans, cache, path, cache->key.objectid,
  359. alignment);
  360. if (ret) {
  361. test_err("could not add free space");
  362. return ret;
  363. }
  364. ret = __add_to_free_space_tree(trans, cache, path,
  365. cache->key.objectid + 4 * alignment,
  366. alignment);
  367. if (ret) {
  368. test_err("could not add free space");
  369. return ret;
  370. }
  371. ret = __add_to_free_space_tree(trans, cache, path,
  372. cache->key.objectid + 2 * alignment,
  373. alignment);
  374. if (ret) {
  375. test_err("could not add free space");
  376. return ret;
  377. }
  378. return check_free_space_extents(trans, fs_info, cache, path,
  379. extents, ARRAY_SIZE(extents));
  380. }
  381. typedef int (*test_func_t)(struct btrfs_trans_handle *,
  382. struct btrfs_fs_info *,
  383. struct btrfs_block_group_cache *,
  384. struct btrfs_path *,
  385. u32 alignment);
  386. static int run_test(test_func_t test_func, int bitmaps, u32 sectorsize,
  387. u32 nodesize, u32 alignment)
  388. {
  389. struct btrfs_fs_info *fs_info;
  390. struct btrfs_root *root = NULL;
  391. struct btrfs_block_group_cache *cache = NULL;
  392. struct btrfs_trans_handle trans;
  393. struct btrfs_path *path = NULL;
  394. int ret;
  395. fs_info = btrfs_alloc_dummy_fs_info(nodesize, sectorsize);
  396. if (!fs_info) {
  397. test_err("couldn't allocate dummy fs info");
  398. ret = -ENOMEM;
  399. goto out;
  400. }
  401. root = btrfs_alloc_dummy_root(fs_info);
  402. if (IS_ERR(root)) {
  403. test_err("couldn't allocate dummy root");
  404. ret = PTR_ERR(root);
  405. goto out;
  406. }
  407. btrfs_set_super_compat_ro_flags(root->fs_info->super_copy,
  408. BTRFS_FEATURE_COMPAT_RO_FREE_SPACE_TREE);
  409. root->fs_info->free_space_root = root;
  410. root->fs_info->tree_root = root;
  411. root->node = alloc_test_extent_buffer(root->fs_info, nodesize);
  412. if (IS_ERR(root->node)) {
  413. test_err("couldn't allocate dummy buffer");
  414. ret = PTR_ERR(root->node);
  415. goto out;
  416. }
  417. btrfs_set_header_level(root->node, 0);
  418. btrfs_set_header_nritems(root->node, 0);
  419. root->alloc_bytenr += 2 * nodesize;
  420. cache = btrfs_alloc_dummy_block_group(fs_info, 8 * alignment);
  421. if (!cache) {
  422. test_err("couldn't allocate dummy block group cache");
  423. ret = -ENOMEM;
  424. goto out;
  425. }
  426. cache->bitmap_low_thresh = 0;
  427. cache->bitmap_high_thresh = (u32)-1;
  428. cache->needs_free_space = 1;
  429. cache->fs_info = root->fs_info;
  430. btrfs_init_dummy_trans(&trans, root->fs_info);
  431. path = btrfs_alloc_path();
  432. if (!path) {
  433. test_err("couldn't allocate path");
  434. ret = -ENOMEM;
  435. goto out;
  436. }
  437. ret = add_block_group_free_space(&trans, cache);
  438. if (ret) {
  439. test_err("could not add block group free space");
  440. goto out;
  441. }
  442. if (bitmaps) {
  443. ret = convert_free_space_to_bitmaps(&trans, cache, path);
  444. if (ret) {
  445. test_err("could not convert block group to bitmaps");
  446. goto out;
  447. }
  448. }
  449. ret = test_func(&trans, root->fs_info, cache, path, alignment);
  450. if (ret)
  451. goto out;
  452. ret = remove_block_group_free_space(&trans, cache);
  453. if (ret) {
  454. test_err("could not remove block group free space");
  455. goto out;
  456. }
  457. if (btrfs_header_nritems(root->node) != 0) {
  458. test_err("free space tree has leftover items");
  459. ret = -EINVAL;
  460. goto out;
  461. }
  462. ret = 0;
  463. out:
  464. btrfs_free_path(path);
  465. btrfs_free_dummy_block_group(cache);
  466. btrfs_free_dummy_root(root);
  467. btrfs_free_dummy_fs_info(fs_info);
  468. return ret;
  469. }
  470. static int run_test_both_formats(test_func_t test_func, u32 sectorsize,
  471. u32 nodesize, u32 alignment)
  472. {
  473. int test_ret = 0;
  474. int ret;
  475. ret = run_test(test_func, 0, sectorsize, nodesize, alignment);
  476. if (ret) {
  477. test_err(
  478. "%pf failed with extents, sectorsize=%u, nodesize=%u, alignment=%u",
  479. test_func, sectorsize, nodesize, alignment);
  480. test_ret = ret;
  481. }
  482. ret = run_test(test_func, 1, sectorsize, nodesize, alignment);
  483. if (ret) {
  484. test_err(
  485. "%pf failed with bitmaps, sectorsize=%u, nodesize=%u, alignment=%u",
  486. test_func, sectorsize, nodesize, alignment);
  487. test_ret = ret;
  488. }
  489. return test_ret;
  490. }
  491. int btrfs_test_free_space_tree(u32 sectorsize, u32 nodesize)
  492. {
  493. test_func_t tests[] = {
  494. test_empty_block_group,
  495. test_remove_all,
  496. test_remove_beginning,
  497. test_remove_end,
  498. test_remove_middle,
  499. test_merge_left,
  500. test_merge_right,
  501. test_merge_both,
  502. test_merge_none,
  503. };
  504. u32 bitmap_alignment;
  505. int test_ret = 0;
  506. int i;
  507. /*
  508. * Align some operations to a page to flush out bugs in the extent
  509. * buffer bitmap handling of highmem.
  510. */
  511. bitmap_alignment = BTRFS_FREE_SPACE_BITMAP_BITS * PAGE_SIZE;
  512. test_msg("running free space tree tests");
  513. for (i = 0; i < ARRAY_SIZE(tests); i++) {
  514. int ret;
  515. ret = run_test_both_formats(tests[i], sectorsize, nodesize,
  516. sectorsize);
  517. if (ret)
  518. test_ret = ret;
  519. ret = run_test_both_formats(tests[i], sectorsize, nodesize,
  520. bitmap_alignment);
  521. if (ret)
  522. test_ret = ret;
  523. }
  524. return test_ret;
  525. }