BitfieldManTest.cc 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787
  1. #include "BitfieldMan.h"
  2. #include <cstring>
  3. #include <vector>
  4. #include <cppunit/extensions/HelperMacros.h>
  5. #include "bitfield.h"
  6. #include "array_fun.h"
  7. namespace aria2 {
  8. class BitfieldManTest:public CppUnit::TestFixture {
  9. CPPUNIT_TEST_SUITE(BitfieldManTest);
  10. CPPUNIT_TEST(testGetBlockSize);
  11. CPPUNIT_TEST(testGetFirstMissingUnusedIndex);
  12. CPPUNIT_TEST(testGetFirstMissingIndex);
  13. CPPUNIT_TEST(testIsAllBitSet);
  14. CPPUNIT_TEST(testFilter);
  15. CPPUNIT_TEST(testAddFilter_zeroLength);
  16. CPPUNIT_TEST(testAddNotFilter);
  17. CPPUNIT_TEST(testAddNotFilter_zeroLength);
  18. CPPUNIT_TEST(testAddNotFilter_overflow);
  19. CPPUNIT_TEST(testGetSparseMissingUnusedIndex);
  20. CPPUNIT_TEST(testGetSparseMissingUnusedIndex_setBit);
  21. CPPUNIT_TEST(testGetSparseMissingUnusedIndex_withMinSplitSize);
  22. CPPUNIT_TEST(testIsBitSetOffsetRange);
  23. CPPUNIT_TEST(testGetOffsetCompletedLength);
  24. CPPUNIT_TEST(testGetMissingUnusedLength);
  25. CPPUNIT_TEST(testSetBitRange);
  26. CPPUNIT_TEST(testGetAllMissingIndexes);
  27. CPPUNIT_TEST(testGetAllMissingIndexes_noarg);
  28. CPPUNIT_TEST(testGetAllMissingIndexes_checkLastByte);
  29. CPPUNIT_TEST(testGetAllMissingUnusedIndexes);
  30. CPPUNIT_TEST(testCountFilteredBlock);
  31. CPPUNIT_TEST(testCountMissingBlock);
  32. CPPUNIT_TEST(testZeroLengthFilter);
  33. CPPUNIT_TEST(testGetFirstNMissingUnusedIndex);
  34. CPPUNIT_TEST(testGetInorderMissingUnusedIndex);
  35. CPPUNIT_TEST(testGetGeomMissingUnusedIndex);
  36. CPPUNIT_TEST_SUITE_END();
  37. public:
  38. void testGetBlockSize();
  39. void testGetFirstMissingUnusedIndex();
  40. void testGetFirstMissingIndex();
  41. void testGetAllMissingIndexes();
  42. void testGetAllMissingIndexes_noarg();
  43. void testGetAllMissingIndexes_checkLastByte();
  44. void testGetAllMissingUnusedIndexes();
  45. void testIsAllBitSet();
  46. void testFilter();
  47. void testAddFilter_zeroLength();
  48. void testAddNotFilter();
  49. void testAddNotFilter_zeroLength();
  50. void testAddNotFilter_overflow();
  51. void testGetSparseMissingUnusedIndex();
  52. void testGetSparseMissingUnusedIndex_setBit();
  53. void testGetSparseMissingUnusedIndex_withMinSplitSize();
  54. void testIsBitSetOffsetRange();
  55. void testGetOffsetCompletedLength();
  56. void testGetMissingUnusedLength();
  57. void testSetBitRange();
  58. void testCountFilteredBlock();
  59. void testCountMissingBlock();
  60. void testZeroLengthFilter();
  61. void testGetFirstNMissingUnusedIndex();
  62. void testGetInorderMissingUnusedIndex();
  63. void testGetGeomMissingUnusedIndex();
  64. };
  65. CPPUNIT_TEST_SUITE_REGISTRATION( BitfieldManTest );
  66. void BitfieldManTest::testGetBlockSize() {
  67. BitfieldMan bt1(1024, 1024*10);
  68. CPPUNIT_ASSERT_EQUAL(1024, bt1.getBlockLength(9));
  69. BitfieldMan bt2(1024, 1024*10+1);
  70. CPPUNIT_ASSERT_EQUAL(1024, bt2.getBlockLength(9));
  71. CPPUNIT_ASSERT_EQUAL(1, bt2.getBlockLength(10));
  72. CPPUNIT_ASSERT_EQUAL(0, bt2.getBlockLength(11));
  73. }
  74. void BitfieldManTest::testGetFirstMissingUnusedIndex()
  75. {
  76. {
  77. BitfieldMan bt1(1024, 1024*10);
  78. {
  79. size_t index;
  80. CPPUNIT_ASSERT(bt1.getFirstMissingUnusedIndex(index));
  81. CPPUNIT_ASSERT_EQUAL((size_t)0, index);
  82. }
  83. bt1.setUseBit(0);
  84. {
  85. size_t index;
  86. CPPUNIT_ASSERT(bt1.getFirstMissingUnusedIndex(index));
  87. CPPUNIT_ASSERT_EQUAL((size_t)1, index);
  88. }
  89. bt1.unsetUseBit(0);
  90. bt1.setBit(0);
  91. {
  92. size_t index;
  93. CPPUNIT_ASSERT(bt1.getFirstMissingUnusedIndex(index));
  94. CPPUNIT_ASSERT_EQUAL((size_t)1, index);
  95. }
  96. bt1.setAllBit();
  97. {
  98. size_t index;
  99. CPPUNIT_ASSERT(!bt1.getFirstMissingUnusedIndex(index));
  100. }
  101. }
  102. {
  103. BitfieldMan bt1(1024, 1024*10);
  104. bt1.addFilter(1024, 1024*10);
  105. bt1.enableFilter();
  106. {
  107. size_t index;
  108. CPPUNIT_ASSERT(bt1.getFirstMissingUnusedIndex(index));
  109. CPPUNIT_ASSERT_EQUAL((size_t)1, index);
  110. }
  111. bt1.setUseBit(1);
  112. {
  113. size_t index;
  114. CPPUNIT_ASSERT(bt1.getFirstMissingUnusedIndex(index));
  115. CPPUNIT_ASSERT_EQUAL((size_t)2, index);
  116. }
  117. bt1.setBit(2);
  118. {
  119. size_t index;
  120. CPPUNIT_ASSERT(bt1.getFirstMissingUnusedIndex(index));
  121. CPPUNIT_ASSERT_EQUAL((size_t)3, index);
  122. }
  123. }
  124. }
  125. void BitfieldManTest::testGetFirstMissingIndex()
  126. {
  127. {
  128. BitfieldMan bt1(1024, 1024*10);
  129. {
  130. size_t index;
  131. CPPUNIT_ASSERT(bt1.getFirstMissingIndex(index));
  132. CPPUNIT_ASSERT_EQUAL((size_t)0, index);
  133. }
  134. bt1.setUseBit(0);
  135. {
  136. size_t index;
  137. CPPUNIT_ASSERT(bt1.getFirstMissingIndex(index));
  138. CPPUNIT_ASSERT_EQUAL((size_t)0, index);
  139. }
  140. bt1.unsetUseBit(0);
  141. bt1.setBit(0);
  142. {
  143. size_t index;
  144. CPPUNIT_ASSERT(bt1.getFirstMissingIndex(index));
  145. CPPUNIT_ASSERT_EQUAL((size_t)1, index);
  146. }
  147. bt1.setAllBit();
  148. {
  149. size_t index;
  150. CPPUNIT_ASSERT(!bt1.getFirstMissingIndex(index));
  151. }
  152. }
  153. {
  154. BitfieldMan bt1(1024, 1024*10);
  155. bt1.addFilter(1024, 1024*10);
  156. bt1.enableFilter();
  157. {
  158. size_t index;
  159. CPPUNIT_ASSERT(bt1.getFirstMissingIndex(index));
  160. CPPUNIT_ASSERT_EQUAL((size_t)1, index);
  161. }
  162. bt1.setUseBit(1);
  163. {
  164. size_t index;
  165. CPPUNIT_ASSERT(bt1.getFirstMissingIndex(index));
  166. CPPUNIT_ASSERT_EQUAL((size_t)1, index);
  167. }
  168. bt1.setBit(1);
  169. {
  170. size_t index;
  171. CPPUNIT_ASSERT(bt1.getFirstMissingIndex(index));
  172. CPPUNIT_ASSERT_EQUAL((size_t)2, index);
  173. }
  174. }
  175. }
  176. void BitfieldManTest::testIsAllBitSet() {
  177. BitfieldMan bt1(1024, 1024*10);
  178. CPPUNIT_ASSERT(!bt1.isAllBitSet());
  179. bt1.setBit(1);
  180. CPPUNIT_ASSERT(!bt1.isAllBitSet());
  181. for(size_t i = 0; i < 8; i++) {
  182. CPPUNIT_ASSERT(bt1.setBit(i));
  183. }
  184. CPPUNIT_ASSERT(!bt1.isAllBitSet());
  185. for(size_t i = 0; i < bt1.countBlock(); i++) {
  186. CPPUNIT_ASSERT(bt1.setBit(i));
  187. }
  188. CPPUNIT_ASSERT(bt1.isAllBitSet());
  189. BitfieldMan btzero(1024, 0);
  190. CPPUNIT_ASSERT(btzero.isAllBitSet());
  191. }
  192. void BitfieldManTest::testFilter()
  193. {
  194. BitfieldMan btman(2, 32);
  195. // test offset=4, length=12
  196. btman.addFilter(4, 12);
  197. btman.enableFilter();
  198. std::vector<size_t> out;
  199. CPPUNIT_ASSERT_EQUAL((size_t)6, btman.getFirstNMissingUnusedIndex(out, 32));
  200. const size_t ans[] = { 2, 3, 4, 5, 6, 7 };
  201. for(size_t i = 0; i < A2_ARRAY_LEN(ans); ++i) {
  202. CPPUNIT_ASSERT_EQUAL(ans[i], out[i]);
  203. }
  204. CPPUNIT_ASSERT_EQUAL((off_t)12ULL, btman.getFilteredTotalLength());
  205. // test offset=5, length=2
  206. out.clear();
  207. btman.clearAllBit();
  208. btman.clearAllUseBit();
  209. btman.clearFilter();
  210. btman.addFilter(5, 2);
  211. btman.enableFilter();
  212. CPPUNIT_ASSERT_EQUAL((size_t)2, btman.getFirstNMissingUnusedIndex(out, 32));
  213. CPPUNIT_ASSERT_EQUAL((size_t)2, out[0]);
  214. CPPUNIT_ASSERT_EQUAL((size_t)3, out[1]);
  215. btman.setBit(2);
  216. btman.setBit(3);
  217. CPPUNIT_ASSERT_EQUAL((off_t)4ULL, btman.getFilteredTotalLength());
  218. CPPUNIT_ASSERT(btman.isFilteredAllBitSet());
  219. BitfieldMan btman2(2, 31);
  220. btman2.addFilter(0, 31);
  221. btman2.enableFilter();
  222. CPPUNIT_ASSERT_EQUAL((off_t)31ULL, btman2.getFilteredTotalLength());
  223. }
  224. void BitfieldManTest::testAddFilter_zeroLength()
  225. {
  226. BitfieldMan bits(1024, 1024*1024);
  227. bits.addFilter(2048, 0);
  228. bits.enableFilter();
  229. CPPUNIT_ASSERT_EQUAL((size_t)0, bits.countMissingBlock());
  230. CPPUNIT_ASSERT(bits.isFilteredAllBitSet());
  231. }
  232. void BitfieldManTest::testAddNotFilter() {
  233. BitfieldMan btman(2, 32);
  234. btman.addNotFilter(3, 6);
  235. CPPUNIT_ASSERT(bitfield::test(btman.getFilterBitfield(), 16, 0));
  236. for(size_t i = 1; i < 5; ++i) {
  237. CPPUNIT_ASSERT(!bitfield::test(btman.getFilterBitfield(), 16, i));
  238. }
  239. for(size_t i = 5; i < 16; ++i) {
  240. CPPUNIT_ASSERT(bitfield::test(btman.getFilterBitfield(), 16, i));
  241. }
  242. }
  243. void BitfieldManTest::testAddNotFilter_zeroLength() {
  244. BitfieldMan btman(2, 6);
  245. btman.addNotFilter(2, 0);
  246. CPPUNIT_ASSERT(!bitfield::test(btman.getFilterBitfield(), 3, 0));
  247. CPPUNIT_ASSERT(!bitfield::test(btman.getFilterBitfield(), 3, 1));
  248. CPPUNIT_ASSERT(!bitfield::test(btman.getFilterBitfield(), 3, 2));
  249. }
  250. void BitfieldManTest::testAddNotFilter_overflow() {
  251. BitfieldMan btman(2, 6);
  252. btman.addNotFilter(6, 100);
  253. CPPUNIT_ASSERT(bitfield::test(btman.getFilterBitfield(), 3, 0));
  254. CPPUNIT_ASSERT(bitfield::test(btman.getFilterBitfield(), 3, 1));
  255. CPPUNIT_ASSERT(bitfield::test(btman.getFilterBitfield(), 3, 2));
  256. }
  257. // TODO1.5 add test using ignoreBitfield
  258. void BitfieldManTest::testGetSparseMissingUnusedIndex() {
  259. BitfieldMan bitfield(1024*1024, 10*1024*1024);
  260. const size_t length = 2;
  261. unsigned char ignoreBitfield[length];
  262. memset(ignoreBitfield, 0, sizeof(ignoreBitfield));
  263. size_t minSplitSize = 1024*1024;
  264. size_t index;
  265. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  266. ignoreBitfield, length));
  267. CPPUNIT_ASSERT_EQUAL((size_t)0, index);
  268. bitfield.setUseBit(0);
  269. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  270. ignoreBitfield, length));
  271. CPPUNIT_ASSERT_EQUAL((size_t)5, index);
  272. bitfield.setUseBit(5);
  273. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  274. ignoreBitfield, length));
  275. CPPUNIT_ASSERT_EQUAL((size_t)3, index);
  276. bitfield.setUseBit(3);
  277. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  278. ignoreBitfield, length));
  279. CPPUNIT_ASSERT_EQUAL((size_t)8, index);
  280. bitfield.setUseBit(8);
  281. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  282. ignoreBitfield, length));
  283. CPPUNIT_ASSERT_EQUAL((size_t)2, index);
  284. bitfield.setUseBit(2);
  285. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  286. ignoreBitfield, length));
  287. CPPUNIT_ASSERT_EQUAL((size_t)1, index);
  288. bitfield.setUseBit(1);
  289. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  290. ignoreBitfield, length));
  291. CPPUNIT_ASSERT_EQUAL((size_t)4, index);
  292. bitfield.setUseBit(4);
  293. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  294. ignoreBitfield, length));
  295. CPPUNIT_ASSERT_EQUAL((size_t)7, index);
  296. bitfield.setUseBit(7);
  297. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  298. ignoreBitfield, length));
  299. CPPUNIT_ASSERT_EQUAL((size_t)6, index);
  300. bitfield.setUseBit(6);
  301. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  302. ignoreBitfield, length));
  303. CPPUNIT_ASSERT_EQUAL((size_t)9, index);
  304. bitfield.setUseBit(9);
  305. CPPUNIT_ASSERT(!bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  306. ignoreBitfield, length));
  307. }
  308. void BitfieldManTest::testGetSparseMissingUnusedIndex_setBit() {
  309. BitfieldMan bitfield(1024*1024, 10*1024*1024);
  310. const size_t length = 2;
  311. unsigned char ignoreBitfield[length];
  312. memset(ignoreBitfield, 0, sizeof(ignoreBitfield));
  313. size_t minSplitSize = 1024*1024;
  314. size_t index;
  315. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  316. ignoreBitfield, length));
  317. CPPUNIT_ASSERT_EQUAL((size_t)0, index);
  318. bitfield.setBit(0);
  319. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  320. ignoreBitfield, length));
  321. CPPUNIT_ASSERT_EQUAL((size_t)1, index);
  322. bitfield.setBit(1);
  323. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  324. ignoreBitfield, length));
  325. CPPUNIT_ASSERT_EQUAL((size_t)2, index);
  326. bitfield.setBit(2);
  327. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  328. ignoreBitfield, length));
  329. CPPUNIT_ASSERT_EQUAL((size_t)3, index);
  330. bitfield.setBit(3);
  331. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  332. ignoreBitfield, length));
  333. CPPUNIT_ASSERT_EQUAL((size_t)4, index);
  334. bitfield.setBit(4);
  335. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  336. ignoreBitfield, length));
  337. CPPUNIT_ASSERT_EQUAL((size_t)5, index);
  338. bitfield.setBit(5);
  339. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  340. ignoreBitfield, length));
  341. CPPUNIT_ASSERT_EQUAL((size_t)6, index);
  342. bitfield.setBit(6);
  343. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  344. ignoreBitfield, length));
  345. CPPUNIT_ASSERT_EQUAL((size_t)7, index);
  346. bitfield.setBit(7);
  347. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  348. ignoreBitfield, length));
  349. CPPUNIT_ASSERT_EQUAL((size_t)8, index);
  350. bitfield.setBit(8);
  351. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  352. ignoreBitfield, length));
  353. CPPUNIT_ASSERT_EQUAL((size_t)9, index);
  354. bitfield.setBit(9);
  355. CPPUNIT_ASSERT(!bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  356. ignoreBitfield, length));
  357. }
  358. void BitfieldManTest::testGetSparseMissingUnusedIndex_withMinSplitSize()
  359. {
  360. BitfieldMan bitfield(1024*1024, 10*1024*1024);
  361. const size_t length = 2;
  362. unsigned char ignoreBitfield[length];
  363. memset(ignoreBitfield, 0, sizeof(ignoreBitfield));
  364. size_t minSplitSize = 2*1024*1024;
  365. size_t index;
  366. bitfield.setUseBit(1);
  367. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  368. ignoreBitfield, length));
  369. CPPUNIT_ASSERT_EQUAL((size_t)6, index);
  370. bitfield.setBit(6);
  371. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  372. ignoreBitfield, length));
  373. CPPUNIT_ASSERT_EQUAL((size_t)7, index);
  374. bitfield.setUseBit(7);
  375. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  376. ignoreBitfield, length));
  377. CPPUNIT_ASSERT_EQUAL((size_t)4, index);
  378. bitfield.setBit(4);
  379. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  380. ignoreBitfield, length));
  381. CPPUNIT_ASSERT_EQUAL((size_t)0, index);
  382. bitfield.setBit(0);
  383. CPPUNIT_ASSERT(bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  384. ignoreBitfield, length));
  385. CPPUNIT_ASSERT_EQUAL((size_t)5, index);
  386. bitfield.setBit(5);
  387. CPPUNIT_ASSERT(!bitfield.getSparseMissingUnusedIndex(index, minSplitSize,
  388. ignoreBitfield, length));
  389. }
  390. void BitfieldManTest::testIsBitSetOffsetRange()
  391. {
  392. int64_t totalLength = 4ULL*1024*1024*1024;
  393. int32_t pieceLength = 4*1024*1024;
  394. BitfieldMan bitfield(pieceLength, totalLength);
  395. bitfield.setAllBit();
  396. CPPUNIT_ASSERT(!bitfield.isBitSetOffsetRange(0, 0));
  397. CPPUNIT_ASSERT(!bitfield.isBitSetOffsetRange(totalLength, 100));
  398. CPPUNIT_ASSERT(!bitfield.isBitSetOffsetRange(totalLength+1, 100));
  399. CPPUNIT_ASSERT(bitfield.isBitSetOffsetRange(0, totalLength));
  400. CPPUNIT_ASSERT(bitfield.isBitSetOffsetRange(0, totalLength+1));
  401. bitfield.clearAllBit();
  402. bitfield.setBit(100);
  403. bitfield.setBit(101);
  404. CPPUNIT_ASSERT(bitfield.isBitSetOffsetRange(pieceLength*100, pieceLength*2));
  405. CPPUNIT_ASSERT(!bitfield.isBitSetOffsetRange(pieceLength*100-10, pieceLength*2));
  406. CPPUNIT_ASSERT(!bitfield.isBitSetOffsetRange(pieceLength*100, pieceLength*2+1));
  407. bitfield.clearAllBit();
  408. bitfield.setBit(100);
  409. bitfield.setBit(102);
  410. CPPUNIT_ASSERT(!bitfield.isBitSetOffsetRange(pieceLength*100, pieceLength*3));
  411. }
  412. void BitfieldManTest::testGetOffsetCompletedLength()
  413. {
  414. BitfieldMan bt(1024, 1024*20);
  415. // 00000|00000|00000|00000
  416. CPPUNIT_ASSERT_EQUAL((off_t)0, bt.getOffsetCompletedLength(0, 1024));
  417. CPPUNIT_ASSERT_EQUAL((off_t)0, bt.getOffsetCompletedLength(0, 0));
  418. for(size_t i = 2; i <= 4; ++i) {
  419. bt.setBit(i);
  420. }
  421. // 00111|00000|00000|00000
  422. CPPUNIT_ASSERT_EQUAL((off_t)3072, bt.getOffsetCompletedLength(2048, 3072));
  423. CPPUNIT_ASSERT_EQUAL((off_t)3071, bt.getOffsetCompletedLength(2047, 3072));
  424. CPPUNIT_ASSERT_EQUAL((off_t)3071, bt.getOffsetCompletedLength(2049, 3072));
  425. CPPUNIT_ASSERT_EQUAL((off_t)0, bt.getOffsetCompletedLength(2048, 0));
  426. CPPUNIT_ASSERT_EQUAL((off_t)1, bt.getOffsetCompletedLength(2048, 1));
  427. CPPUNIT_ASSERT_EQUAL((off_t)0, bt.getOffsetCompletedLength(2047, 1));
  428. CPPUNIT_ASSERT_EQUAL((off_t)3072, bt.getOffsetCompletedLength(0, 1024*20));
  429. CPPUNIT_ASSERT_EQUAL((off_t)3072,
  430. bt.getOffsetCompletedLength(0, 1024*20+10));
  431. CPPUNIT_ASSERT_EQUAL((off_t)0, bt.getOffsetCompletedLength(1024*20, 1));
  432. }
  433. void BitfieldManTest::testGetMissingUnusedLength()
  434. {
  435. off_t totalLength = 1024*10+10;
  436. size_t blockLength = 1024;
  437. BitfieldMan bf(blockLength, totalLength);
  438. // from index 0 and all blocks are unused and not acquired.
  439. CPPUNIT_ASSERT_EQUAL(totalLength, bf.getMissingUnusedLength(0));
  440. // from index 10 and all blocks are unused and not acquired.
  441. CPPUNIT_ASSERT_EQUAL((off_t)10ULL, bf.getMissingUnusedLength(10));
  442. // from index 11
  443. CPPUNIT_ASSERT_EQUAL((off_t)0ULL, bf.getMissingUnusedLength(11));
  444. // from index 12
  445. CPPUNIT_ASSERT_EQUAL((off_t)0ULL, bf.getMissingUnusedLength(12));
  446. // from index 0 and 5th block is used.
  447. bf.setUseBit(5);
  448. CPPUNIT_ASSERT_EQUAL((off_t)(5LL*blockLength), bf.getMissingUnusedLength(0));
  449. // from index 0 and 4th block is acquired.
  450. bf.setBit(4);
  451. CPPUNIT_ASSERT_EQUAL((off_t)(4LL*blockLength), bf.getMissingUnusedLength(0));
  452. // from index 1
  453. CPPUNIT_ASSERT_EQUAL((off_t)(3LL*blockLength), bf.getMissingUnusedLength(1));
  454. }
  455. void BitfieldManTest::testSetBitRange()
  456. {
  457. size_t blockLength = 1024*1024;
  458. off_t totalLength = 10*blockLength;
  459. BitfieldMan bf(blockLength, totalLength);
  460. bf.setBitRange(0, 4);
  461. for(size_t i = 0; i < 5; ++i) {
  462. CPPUNIT_ASSERT(bf.isBitSet(i));
  463. }
  464. for(size_t i = 5; i < 10; ++i) {
  465. CPPUNIT_ASSERT(!bf.isBitSet(i));
  466. }
  467. CPPUNIT_ASSERT_EQUAL((off_t)(5LL*blockLength), bf.getCompletedLength());
  468. }
  469. void BitfieldManTest::testGetAllMissingIndexes_noarg()
  470. {
  471. size_t blockLength = 16*1024;
  472. off_t totalLength = 1024*1024;
  473. size_t nbits = (totalLength+blockLength-1)/blockLength;
  474. BitfieldMan bf(blockLength, totalLength);
  475. unsigned char misbitfield[8];
  476. CPPUNIT_ASSERT(bf.getAllMissingIndexes(misbitfield, sizeof(misbitfield)));
  477. CPPUNIT_ASSERT_EQUAL((size_t)64, bitfield::countSetBit(misbitfield, nbits));
  478. for(size_t i = 0; i < 63; ++i) {
  479. bf.setBit(i);
  480. }
  481. CPPUNIT_ASSERT(bf.getAllMissingIndexes(misbitfield, sizeof(misbitfield)));
  482. CPPUNIT_ASSERT_EQUAL((size_t)1, bitfield::countSetBit(misbitfield, nbits));
  483. CPPUNIT_ASSERT(bitfield::test(misbitfield, nbits, 63));
  484. }
  485. // See garbage bits of last byte are 0
  486. void BitfieldManTest::testGetAllMissingIndexes_checkLastByte()
  487. {
  488. size_t blockLength = 16*1024;
  489. off_t totalLength = blockLength*2;
  490. size_t nbits = (totalLength+blockLength-1)/blockLength;
  491. BitfieldMan bf(blockLength, totalLength);
  492. unsigned char misbitfield[1];
  493. CPPUNIT_ASSERT(bf.getAllMissingIndexes(misbitfield, sizeof(misbitfield)));
  494. CPPUNIT_ASSERT_EQUAL((size_t)2, bitfield::countSetBit(misbitfield, nbits));
  495. CPPUNIT_ASSERT(bitfield::test(misbitfield, nbits, 0));
  496. CPPUNIT_ASSERT(bitfield::test(misbitfield, nbits, 1));
  497. }
  498. void BitfieldManTest::testGetAllMissingIndexes()
  499. {
  500. size_t blockLength = 16*1024;
  501. off_t totalLength = 1024*1024;
  502. size_t nbits = (totalLength+blockLength-1)/blockLength;
  503. BitfieldMan bf(blockLength, totalLength);
  504. BitfieldMan peerBf(blockLength, totalLength);
  505. peerBf.setAllBit();
  506. unsigned char misbitfield[8];
  507. CPPUNIT_ASSERT(bf.getAllMissingIndexes(misbitfield, sizeof(misbitfield),
  508. peerBf.getBitfield(),
  509. peerBf.getBitfieldLength()));
  510. CPPUNIT_ASSERT_EQUAL((size_t)64, bitfield::countSetBit(misbitfield, nbits));
  511. for(size_t i = 0; i < 62; ++i) {
  512. bf.setBit(i);
  513. }
  514. peerBf.unsetBit(62);
  515. CPPUNIT_ASSERT(bf.getAllMissingIndexes(misbitfield, sizeof(misbitfield),
  516. peerBf.getBitfield(),
  517. peerBf.getBitfieldLength()));
  518. CPPUNIT_ASSERT_EQUAL((size_t)1, bitfield::countSetBit(misbitfield, nbits));
  519. CPPUNIT_ASSERT(bitfield::test(misbitfield, nbits, 63));
  520. }
  521. void BitfieldManTest::testGetAllMissingUnusedIndexes()
  522. {
  523. size_t blockLength = 16*1024;
  524. off_t totalLength = 1024*1024;
  525. size_t nbits = (totalLength+blockLength-1)/blockLength;
  526. BitfieldMan bf(blockLength, totalLength);
  527. BitfieldMan peerBf(blockLength, totalLength);
  528. peerBf.setAllBit();
  529. unsigned char misbitfield[8];
  530. CPPUNIT_ASSERT(bf.getAllMissingUnusedIndexes(misbitfield,
  531. sizeof(misbitfield),
  532. peerBf.getBitfield(),
  533. peerBf.getBitfieldLength()));
  534. CPPUNIT_ASSERT_EQUAL((size_t)64, bitfield::countSetBit(misbitfield, nbits));
  535. for(size_t i = 0; i < 61; ++i) {
  536. bf.setBit(i);
  537. }
  538. bf.setUseBit(61);
  539. peerBf.unsetBit(62);
  540. CPPUNIT_ASSERT(bf.getAllMissingUnusedIndexes(misbitfield,
  541. sizeof(misbitfield),
  542. peerBf.getBitfield(),
  543. peerBf.getBitfieldLength()));
  544. CPPUNIT_ASSERT_EQUAL((size_t)1, bitfield::countSetBit(misbitfield, nbits));
  545. CPPUNIT_ASSERT(bitfield::test(misbitfield, nbits, 63));
  546. }
  547. void BitfieldManTest::testCountFilteredBlock()
  548. {
  549. BitfieldMan bt(1024, 1024*256);
  550. CPPUNIT_ASSERT_EQUAL((size_t)256, bt.countBlock());
  551. CPPUNIT_ASSERT_EQUAL((size_t)0, bt.countFilteredBlock());
  552. bt.addFilter(1024, 1024*256);
  553. bt.enableFilter();
  554. CPPUNIT_ASSERT_EQUAL((size_t)256, bt.countBlock());
  555. CPPUNIT_ASSERT_EQUAL((size_t)255, bt.countFilteredBlock());
  556. bt.disableFilter();
  557. CPPUNIT_ASSERT_EQUAL((size_t)256, bt.countBlock());
  558. CPPUNIT_ASSERT_EQUAL((size_t)0, bt.countFilteredBlock());
  559. }
  560. void BitfieldManTest::testCountMissingBlock()
  561. {
  562. BitfieldMan bt(1024, 1024*10);
  563. CPPUNIT_ASSERT_EQUAL((size_t)10, bt.countMissingBlock());
  564. bt.setBit(1);
  565. CPPUNIT_ASSERT_EQUAL((size_t)9, bt.countMissingBlock());
  566. bt.setAllBit();
  567. CPPUNIT_ASSERT_EQUAL((size_t)0, bt.countMissingBlock());
  568. }
  569. void BitfieldManTest::testZeroLengthFilter()
  570. {
  571. BitfieldMan bt(1024, 1024*10);
  572. bt.enableFilter();
  573. CPPUNIT_ASSERT_EQUAL((size_t)0, bt.countMissingBlock());
  574. }
  575. void BitfieldManTest::testGetFirstNMissingUnusedIndex()
  576. {
  577. BitfieldMan bt(1024, 1024*10);
  578. bt.setUseBit(1);
  579. bt.setBit(5);
  580. std::vector<size_t> out;
  581. CPPUNIT_ASSERT_EQUAL((size_t)8, bt.getFirstNMissingUnusedIndex(out, 256));
  582. CPPUNIT_ASSERT_EQUAL((size_t)8, out.size());
  583. const size_t ans[] = {0, 2, 3, 4, 6, 7, 8, 9};
  584. for(size_t i = 0; i < out.size(); ++i) {
  585. CPPUNIT_ASSERT_EQUAL(ans[i], out[i]);
  586. }
  587. out.clear();
  588. CPPUNIT_ASSERT_EQUAL((size_t)3, bt.getFirstNMissingUnusedIndex(out, 3));
  589. CPPUNIT_ASSERT_EQUAL((size_t)3, out.size());
  590. for(size_t i = 0; i < out.size(); ++i) {
  591. CPPUNIT_ASSERT_EQUAL(ans[i], out[i]);
  592. }
  593. CPPUNIT_ASSERT_EQUAL((size_t)0, bt.getFirstNMissingUnusedIndex(out, 0));
  594. bt.setAllBit();
  595. CPPUNIT_ASSERT_EQUAL((size_t)0, bt.getFirstNMissingUnusedIndex(out, 10));
  596. bt.clearAllBit();
  597. out.clear();
  598. bt.addFilter(1024*9, 1024);
  599. bt.enableFilter();
  600. CPPUNIT_ASSERT_EQUAL((size_t)1, bt.getFirstNMissingUnusedIndex(out, 256));
  601. CPPUNIT_ASSERT_EQUAL((size_t)1, out.size());
  602. CPPUNIT_ASSERT_EQUAL((size_t)9, out[0]);
  603. }
  604. void BitfieldManTest::testGetInorderMissingUnusedIndex()
  605. {
  606. BitfieldMan bt(1024, 1024*20);
  607. const size_t length = 3;
  608. unsigned char ignoreBitfield[length];
  609. memset(ignoreBitfield, 0, sizeof(ignoreBitfield));
  610. size_t minSplitSize = 1024;
  611. size_t index;
  612. // 00000|00000|00000|00000
  613. CPPUNIT_ASSERT
  614. (bt.getInorderMissingUnusedIndex
  615. (index, minSplitSize, ignoreBitfield, length));
  616. CPPUNIT_ASSERT_EQUAL((size_t)0, index);
  617. bt.setUseBit(0);
  618. // 10000|00000|00000|00000
  619. CPPUNIT_ASSERT
  620. (bt.getInorderMissingUnusedIndex
  621. (index, minSplitSize, ignoreBitfield, length));
  622. CPPUNIT_ASSERT_EQUAL((size_t)1, index);
  623. minSplitSize = 1024*2;
  624. CPPUNIT_ASSERT
  625. (bt.getInorderMissingUnusedIndex
  626. (index, minSplitSize, ignoreBitfield, length));
  627. CPPUNIT_ASSERT_EQUAL((size_t)2, index);
  628. bt.unsetUseBit(0);
  629. bt.setBit(0);
  630. CPPUNIT_ASSERT
  631. (bt.getInorderMissingUnusedIndex
  632. (index, minSplitSize, ignoreBitfield, length));
  633. CPPUNIT_ASSERT_EQUAL((size_t)1, index);
  634. bt.setAllBit();
  635. bt.unsetBit(10);
  636. // 11111|11111|01111|11111
  637. CPPUNIT_ASSERT
  638. (bt.getInorderMissingUnusedIndex
  639. (index, minSplitSize, ignoreBitfield, length));
  640. CPPUNIT_ASSERT_EQUAL((size_t)10, index);
  641. bt.setUseBit(10);
  642. CPPUNIT_ASSERT
  643. (!bt.getInorderMissingUnusedIndex
  644. (index, minSplitSize, ignoreBitfield, length));
  645. bt.unsetUseBit(10);
  646. bt.setAllBit();
  647. // 11111|11111|11111|11111
  648. CPPUNIT_ASSERT
  649. (!bt.getInorderMissingUnusedIndex
  650. (index, minSplitSize, ignoreBitfield, length));
  651. bt.clearAllBit();
  652. // 00000|00000|00000|00000
  653. for(int i = 0; i <= 1; ++i) {
  654. bitfield::flipBit(ignoreBitfield, length, i);
  655. }
  656. CPPUNIT_ASSERT
  657. (bt.getInorderMissingUnusedIndex
  658. (index, minSplitSize, ignoreBitfield, length));
  659. CPPUNIT_ASSERT_EQUAL((size_t)2, index);
  660. bt.addFilter(1024*3, 1024*3);
  661. bt.enableFilter();
  662. CPPUNIT_ASSERT
  663. (bt.getInorderMissingUnusedIndex
  664. (index, minSplitSize, ignoreBitfield, length));
  665. CPPUNIT_ASSERT_EQUAL((size_t)3, index);
  666. }
  667. void BitfieldManTest::testGetGeomMissingUnusedIndex()
  668. {
  669. BitfieldMan bt(1024, 1024*20);
  670. const size_t length = 3;
  671. unsigned char ignoreBitfield[length];
  672. memset(ignoreBitfield, 0, sizeof(ignoreBitfield));
  673. size_t minSplitSize = 1024;
  674. size_t index;
  675. // 00000|00000|00000|00000
  676. CPPUNIT_ASSERT
  677. (bt.getGeomMissingUnusedIndex
  678. (index, minSplitSize, ignoreBitfield, length, 2, 0));
  679. CPPUNIT_ASSERT_EQUAL((size_t)0, index);
  680. bt.setUseBit(0);
  681. // 10000|00000|00000|00000
  682. CPPUNIT_ASSERT
  683. (bt.getGeomMissingUnusedIndex
  684. (index, minSplitSize, ignoreBitfield, length, 2, 0));
  685. CPPUNIT_ASSERT_EQUAL((size_t)1, index);
  686. bt.setUseBit(1);
  687. // 11000|00000|00000|00000
  688. CPPUNIT_ASSERT
  689. (bt.getGeomMissingUnusedIndex
  690. (index, minSplitSize, ignoreBitfield, length, 2, 0));
  691. CPPUNIT_ASSERT_EQUAL((size_t)2, index);
  692. bt.setUseBit(2);
  693. // 11100|00000|00000|00000
  694. CPPUNIT_ASSERT
  695. (bt.getGeomMissingUnusedIndex
  696. (index, minSplitSize, ignoreBitfield, length, 2, 0));
  697. CPPUNIT_ASSERT_EQUAL((size_t)4, index);
  698. bt.setUseBit(4);
  699. // 11110|00000|00000|00000
  700. CPPUNIT_ASSERT
  701. (bt.getGeomMissingUnusedIndex
  702. (index, minSplitSize, ignoreBitfield, length, 2, 0));
  703. CPPUNIT_ASSERT_EQUAL((size_t)8, index);
  704. bt.setUseBit(8);
  705. // 11110|00010|00000|00000
  706. CPPUNIT_ASSERT
  707. (bt.getGeomMissingUnusedIndex
  708. (index, minSplitSize, ignoreBitfield, length, 2, 0));
  709. CPPUNIT_ASSERT_EQUAL((size_t)16, index);
  710. bt.setUseBit(16);
  711. // 11110|00010|00000|01000
  712. CPPUNIT_ASSERT
  713. (bt.getGeomMissingUnusedIndex
  714. (index, minSplitSize, ignoreBitfield, length, 2, 0));
  715. CPPUNIT_ASSERT_EQUAL((size_t)12, index);
  716. bt.setUseBit(12);
  717. }
  718. } // namespace aria2