check.c 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779
  1. /*
  2. Trivial Database 2: free list/block handling
  3. Copyright (C) Rusty Russell 2010
  4. This library is free software; you can redistribute it and/or
  5. modify it under the terms of the GNU Lesser General Public
  6. License as published by the Free Software Foundation; either
  7. version 3 of the License, or (at your option) any later version.
  8. This library is distributed in the hope that it will be useful,
  9. but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  11. Lesser General Public License for more details.
  12. You should have received a copy of the GNU Lesser General Public
  13. License along with this library; if not, see <http://www.gnu.org/licenses/>.
  14. */
  15. #include "private.h"
  16. #include <ccan/likely/likely.h>
  17. #include <ccan/asearch/asearch.h>
  18. /* We keep an ordered array of offsets. */
  19. static bool append(tdb_off_t **arr, size_t *num, tdb_off_t off)
  20. {
  21. tdb_off_t *new = realloc(*arr, (*num + 1) * sizeof(tdb_off_t));
  22. if (!new)
  23. return false;
  24. new[(*num)++] = off;
  25. *arr = new;
  26. return true;
  27. }
  28. static enum TDB_ERROR check_header(struct tdb_context *tdb, tdb_off_t *recovery)
  29. {
  30. uint64_t hash_test;
  31. struct tdb_header hdr;
  32. enum TDB_ERROR ecode;
  33. ecode = tdb_read_convert(tdb, 0, &hdr, sizeof(hdr));
  34. if (ecode != TDB_SUCCESS) {
  35. return ecode;
  36. }
  37. /* magic food should not be converted, so convert back. */
  38. tdb_convert(tdb, hdr.magic_food, sizeof(hdr.magic_food));
  39. hash_test = TDB_HASH_MAGIC;
  40. hash_test = tdb_hash(tdb, &hash_test, sizeof(hash_test));
  41. if (hdr.hash_test != hash_test) {
  42. return tdb_logerr(tdb, TDB_ERR_CORRUPT, TDB_LOG_ERROR,
  43. "check: hash test %llu should be %llu",
  44. (long long)hdr.hash_test,
  45. (long long)hash_test);
  46. }
  47. if (strcmp(hdr.magic_food, TDB_MAGIC_FOOD) != 0) {
  48. return tdb_logerr(tdb, TDB_ERR_CORRUPT, TDB_LOG_ERROR,
  49. "check: bad magic '%.*s'",
  50. (unsigned)sizeof(hdr.magic_food),
  51. hdr.magic_food);
  52. }
  53. *recovery = hdr.recovery;
  54. if (*recovery) {
  55. if (*recovery < sizeof(hdr) || *recovery > tdb->map_size) {
  56. return tdb_logerr(tdb, TDB_ERR_CORRUPT, TDB_LOG_ERROR,
  57. "tdb_check:"
  58. " invalid recovery offset %zu",
  59. (size_t)*recovery);
  60. }
  61. }
  62. /* Don't check reserved: they *can* be used later. */
  63. return TDB_SUCCESS;
  64. }
  65. static enum TDB_ERROR check_hash_tree(struct tdb_context *tdb,
  66. tdb_off_t off, unsigned int group_bits,
  67. uint64_t hprefix,
  68. unsigned hprefix_bits,
  69. tdb_off_t used[],
  70. size_t num_used,
  71. size_t *num_found,
  72. enum TDB_ERROR (*check)(TDB_DATA,
  73. TDB_DATA, void *),
  74. void *private_data);
  75. static enum TDB_ERROR check_hash_chain(struct tdb_context *tdb,
  76. tdb_off_t off,
  77. uint64_t hash,
  78. tdb_off_t used[],
  79. size_t num_used,
  80. size_t *num_found,
  81. enum TDB_ERROR (*check)(TDB_DATA,
  82. TDB_DATA,
  83. void *),
  84. void *private_data)
  85. {
  86. struct tdb_used_record rec;
  87. enum TDB_ERROR ecode;
  88. ecode = tdb_read_convert(tdb, off, &rec, sizeof(rec));
  89. if (ecode != TDB_SUCCESS) {
  90. return ecode;
  91. }
  92. if (rec_magic(&rec) != TDB_CHAIN_MAGIC) {
  93. return tdb_logerr(tdb, TDB_ERR_CORRUPT, TDB_LOG_ERROR,
  94. "tdb_check: Bad hash chain magic %llu",
  95. (long long)rec_magic(&rec));
  96. }
  97. if (rec_data_length(&rec) != sizeof(struct tdb_chain)) {
  98. return tdb_logerr(tdb, TDB_ERR_CORRUPT, TDB_LOG_ERROR,
  99. "tdb_check:"
  100. " Bad hash chain length %llu vs %zu",
  101. (long long)rec_data_length(&rec),
  102. sizeof(struct tdb_chain));
  103. }
  104. if (rec_key_length(&rec) != 0) {
  105. return tdb_logerr(tdb, TDB_ERR_CORRUPT, TDB_LOG_ERROR,
  106. "tdb_check: Bad hash chain key length %llu",
  107. (long long)rec_key_length(&rec));
  108. }
  109. if (rec_hash(&rec) != 0) {
  110. return tdb_logerr(tdb, TDB_ERR_CORRUPT, TDB_LOG_ERROR,
  111. "tdb_check: Bad hash chain hash value %llu",
  112. (long long)rec_hash(&rec));
  113. }
  114. off += sizeof(rec);
  115. ecode = check_hash_tree(tdb, off, 0, hash, 64,
  116. used, num_used, num_found, check, private_data);
  117. if (ecode != TDB_SUCCESS) {
  118. return false;
  119. }
  120. off = tdb_read_off(tdb, off + offsetof(struct tdb_chain, next));
  121. if (TDB_OFF_IS_ERR(off)) {
  122. return off;
  123. }
  124. if (off == 0)
  125. return TDB_SUCCESS;
  126. (*num_found)++;
  127. return check_hash_chain(tdb, off, hash, used, num_used, num_found,
  128. check, private_data);
  129. }
  130. static enum TDB_ERROR check_hash_record(struct tdb_context *tdb,
  131. tdb_off_t off,
  132. uint64_t hprefix,
  133. unsigned hprefix_bits,
  134. tdb_off_t used[],
  135. size_t num_used,
  136. size_t *num_found,
  137. enum TDB_ERROR (*check)(TDB_DATA,
  138. TDB_DATA,
  139. void *),
  140. void *private_data)
  141. {
  142. struct tdb_used_record rec;
  143. enum TDB_ERROR ecode;
  144. if (hprefix_bits >= 64)
  145. return check_hash_chain(tdb, off, hprefix, used, num_used,
  146. num_found, check, private_data);
  147. ecode = tdb_read_convert(tdb, off, &rec, sizeof(rec));
  148. if (ecode != TDB_SUCCESS) {
  149. return ecode;
  150. }
  151. if (rec_magic(&rec) != TDB_HTABLE_MAGIC) {
  152. return tdb_logerr(tdb, TDB_ERR_CORRUPT, TDB_LOG_ERROR,
  153. "tdb_check: Bad hash table magic %llu",
  154. (long long)rec_magic(&rec));
  155. }
  156. if (rec_data_length(&rec)
  157. != sizeof(tdb_off_t) << TDB_SUBLEVEL_HASH_BITS) {
  158. return tdb_logerr(tdb, TDB_ERR_CORRUPT, TDB_LOG_ERROR,
  159. "tdb_check:"
  160. " Bad hash table length %llu vs %llu",
  161. (long long)rec_data_length(&rec),
  162. (long long)sizeof(tdb_off_t)
  163. << TDB_SUBLEVEL_HASH_BITS);
  164. }
  165. if (rec_key_length(&rec) != 0) {
  166. return tdb_logerr(tdb, TDB_ERR_CORRUPT, TDB_LOG_ERROR,
  167. "tdb_check: Bad hash table key length %llu",
  168. (long long)rec_key_length(&rec));
  169. }
  170. if (rec_hash(&rec) != 0) {
  171. return tdb_logerr(tdb, TDB_ERR_CORRUPT, TDB_LOG_ERROR,
  172. "tdb_check: Bad hash table hash value %llu",
  173. (long long)rec_hash(&rec));
  174. }
  175. off += sizeof(rec);
  176. return check_hash_tree(tdb, off,
  177. TDB_SUBLEVEL_HASH_BITS-TDB_HASH_GROUP_BITS,
  178. hprefix, hprefix_bits,
  179. used, num_used, num_found, check, private_data);
  180. }
  181. static int off_cmp(const tdb_off_t *a, const tdb_off_t *b)
  182. {
  183. /* Can overflow an int. */
  184. return *a > *b ? 1
  185. : *a < *b ? -1
  186. : 0;
  187. }
  188. static uint64_t get_bits(uint64_t h, unsigned num, unsigned *used)
  189. {
  190. *used += num;
  191. return (h >> (64 - *used)) & ((1U << num) - 1);
  192. }
  193. static enum TDB_ERROR check_hash_tree(struct tdb_context *tdb,
  194. tdb_off_t off, unsigned int group_bits,
  195. uint64_t hprefix,
  196. unsigned hprefix_bits,
  197. tdb_off_t used[],
  198. size_t num_used,
  199. size_t *num_found,
  200. enum TDB_ERROR (*check)(TDB_DATA,
  201. TDB_DATA, void *),
  202. void *private_data)
  203. {
  204. unsigned int g, b;
  205. const tdb_off_t *hash;
  206. struct tdb_used_record rec;
  207. enum TDB_ERROR ecode;
  208. hash = tdb_access_read(tdb, off,
  209. sizeof(tdb_off_t)
  210. << (group_bits + TDB_HASH_GROUP_BITS),
  211. true);
  212. if (TDB_PTR_IS_ERR(hash)) {
  213. return TDB_PTR_ERR(hash);
  214. }
  215. for (g = 0; g < (1 << group_bits); g++) {
  216. const tdb_off_t *group = hash + (g << TDB_HASH_GROUP_BITS);
  217. for (b = 0; b < (1 << TDB_HASH_GROUP_BITS); b++) {
  218. unsigned int bucket, i, used_bits;
  219. uint64_t h;
  220. tdb_off_t *p;
  221. if (group[b] == 0)
  222. continue;
  223. off = group[b] & TDB_OFF_MASK;
  224. p = asearch(&off, used, num_used, off_cmp);
  225. if (!p) {
  226. ecode = tdb_logerr(tdb, TDB_ERR_CORRUPT,
  227. TDB_LOG_ERROR,
  228. "tdb_check: Invalid offset"
  229. " %llu in hash",
  230. (long long)off);
  231. goto fail;
  232. }
  233. /* Mark it invalid. */
  234. *p ^= 1;
  235. (*num_found)++;
  236. if (hprefix_bits == 64) {
  237. /* Chained entries are unordered. */
  238. if (is_subhash(group[b])) {
  239. ecode = TDB_ERR_CORRUPT;
  240. tdb_logerr(tdb, ecode,
  241. TDB_LOG_ERROR,
  242. "tdb_check: Invalid chain"
  243. " entry subhash");
  244. goto fail;
  245. }
  246. h = hash_record(tdb, off);
  247. if (h != hprefix) {
  248. ecode = TDB_ERR_CORRUPT;
  249. tdb_logerr(tdb, ecode,
  250. TDB_LOG_ERROR,
  251. "check: bad hash chain"
  252. " placement"
  253. " 0x%llx vs 0x%llx",
  254. (long long)h,
  255. (long long)hprefix);
  256. goto fail;
  257. }
  258. ecode = tdb_read_convert(tdb, off, &rec,
  259. sizeof(rec));
  260. if (ecode != TDB_SUCCESS) {
  261. goto fail;
  262. }
  263. goto check;
  264. }
  265. if (is_subhash(group[b])) {
  266. uint64_t subprefix;
  267. subprefix = (hprefix
  268. << (group_bits + TDB_HASH_GROUP_BITS))
  269. + g * (1 << TDB_HASH_GROUP_BITS) + b;
  270. ecode = check_hash_record(tdb,
  271. group[b] & TDB_OFF_MASK,
  272. subprefix,
  273. hprefix_bits
  274. + group_bits
  275. + TDB_HASH_GROUP_BITS,
  276. used, num_used, num_found,
  277. check, private_data);
  278. if (ecode != TDB_SUCCESS) {
  279. goto fail;
  280. }
  281. continue;
  282. }
  283. /* A normal entry */
  284. /* Does it belong here at all? */
  285. h = hash_record(tdb, off);
  286. used_bits = 0;
  287. if (get_bits(h, hprefix_bits, &used_bits) != hprefix
  288. && hprefix_bits) {
  289. ecode = tdb_logerr(tdb, TDB_ERR_CORRUPT,
  290. TDB_LOG_ERROR,
  291. "check: bad hash placement"
  292. " 0x%llx vs 0x%llx",
  293. (long long)h,
  294. (long long)hprefix);
  295. goto fail;
  296. }
  297. /* Does it belong in this group? */
  298. if (get_bits(h, group_bits, &used_bits) != g) {
  299. ecode = tdb_logerr(tdb, TDB_ERR_CORRUPT,
  300. TDB_LOG_ERROR,
  301. "check: bad group %llu"
  302. " vs %u",
  303. (long long)h, g);
  304. goto fail;
  305. }
  306. /* Are bucket bits correct? */
  307. bucket = group[b] & TDB_OFF_HASH_GROUP_MASK;
  308. if (get_bits(h, TDB_HASH_GROUP_BITS, &used_bits)
  309. != bucket) {
  310. used_bits -= TDB_HASH_GROUP_BITS;
  311. ecode = tdb_logerr(tdb, TDB_ERR_CORRUPT,
  312. TDB_LOG_ERROR,
  313. "check: bad bucket %u vs %u",
  314. (unsigned)get_bits(h,
  315. TDB_HASH_GROUP_BITS,
  316. &used_bits),
  317. bucket);
  318. goto fail;
  319. }
  320. /* There must not be any zero entries between
  321. * the bucket it belongs in and this one! */
  322. for (i = bucket;
  323. i != b;
  324. i = (i + 1) % (1 << TDB_HASH_GROUP_BITS)) {
  325. if (group[i] == 0) {
  326. ecode = TDB_ERR_CORRUPT;
  327. tdb_logerr(tdb, ecode,
  328. TDB_LOG_ERROR,
  329. "check: bad group placement"
  330. " %u vs %u",
  331. b, bucket);
  332. goto fail;
  333. }
  334. }
  335. ecode = tdb_read_convert(tdb, off, &rec, sizeof(rec));
  336. if (ecode != TDB_SUCCESS) {
  337. goto fail;
  338. }
  339. /* Bottom bits must match header. */
  340. if ((h & ((1 << 11)-1)) != rec_hash(&rec)) {
  341. ecode = tdb_logerr(tdb, TDB_ERR_CORRUPT,
  342. TDB_LOG_ERROR,
  343. "tdb_check: Bad hash magic"
  344. " at offset %llu"
  345. " (0x%llx vs 0x%llx)",
  346. (long long)off,
  347. (long long)h,
  348. (long long)rec_hash(&rec));
  349. goto fail;
  350. }
  351. check:
  352. if (check) {
  353. TDB_DATA key, data;
  354. key.dsize = rec_key_length(&rec);
  355. data.dsize = rec_data_length(&rec);
  356. key.dptr = (void *)tdb_access_read(tdb,
  357. off + sizeof(rec),
  358. key.dsize + data.dsize,
  359. false);
  360. if (TDB_PTR_IS_ERR(key.dptr)) {
  361. ecode = TDB_PTR_ERR(key.dptr);
  362. goto fail;
  363. }
  364. data.dptr = key.dptr + key.dsize;
  365. ecode = check(key, data, private_data);
  366. if (ecode != TDB_SUCCESS) {
  367. goto fail;
  368. }
  369. tdb_access_release(tdb, key.dptr);
  370. }
  371. }
  372. }
  373. tdb_access_release(tdb, hash);
  374. return TDB_SUCCESS;
  375. fail:
  376. tdb_access_release(tdb, hash);
  377. return ecode;
  378. }
  379. static enum TDB_ERROR check_hash(struct tdb_context *tdb,
  380. tdb_off_t used[],
  381. size_t num_used, size_t num_ftables,
  382. int (*check)(TDB_DATA, TDB_DATA, void *),
  383. void *private_data)
  384. {
  385. /* Free tables also show up as used. */
  386. size_t num_found = num_ftables;
  387. enum TDB_ERROR ecode;
  388. ecode = check_hash_tree(tdb, offsetof(struct tdb_header, hashtable),
  389. TDB_TOPLEVEL_HASH_BITS-TDB_HASH_GROUP_BITS,
  390. 0, 0, used, num_used, &num_found,
  391. check, private_data);
  392. if (ecode == TDB_SUCCESS) {
  393. if (num_found != num_used) {
  394. ecode = tdb_logerr(tdb, TDB_ERR_CORRUPT, TDB_LOG_ERROR,
  395. "tdb_check: Not all entries"
  396. " are in hash");
  397. }
  398. }
  399. return ecode;
  400. }
  401. static enum TDB_ERROR check_free(struct tdb_context *tdb,
  402. tdb_off_t off,
  403. const struct tdb_free_record *frec,
  404. tdb_off_t prev, unsigned int ftable,
  405. unsigned int bucket)
  406. {
  407. enum TDB_ERROR ecode;
  408. if (frec_magic(frec) != TDB_FREE_MAGIC) {
  409. return tdb_logerr(tdb, TDB_ERR_CORRUPT, TDB_LOG_ERROR,
  410. "tdb_check: offset %llu bad magic 0x%llx",
  411. (long long)off,
  412. (long long)frec->magic_and_prev);
  413. }
  414. if (frec_ftable(frec) != ftable) {
  415. return tdb_logerr(tdb, TDB_ERR_CORRUPT, TDB_LOG_ERROR,
  416. "tdb_check: offset %llu bad freetable %u",
  417. (long long)off, frec_ftable(frec));
  418. }
  419. ecode = tdb->methods->oob(tdb, off
  420. + frec_len(frec)
  421. + sizeof(struct tdb_used_record),
  422. false);
  423. if (ecode != TDB_SUCCESS) {
  424. return ecode;
  425. }
  426. if (size_to_bucket(frec_len(frec)) != bucket) {
  427. return tdb_logerr(tdb, TDB_ERR_CORRUPT, TDB_LOG_ERROR,
  428. "tdb_check: offset %llu in wrong bucket"
  429. " (%u vs %u)",
  430. (long long)off,
  431. bucket, size_to_bucket(frec_len(frec)));
  432. }
  433. if (prev != frec_prev(frec)) {
  434. return tdb_logerr(tdb, TDB_ERR_CORRUPT, TDB_LOG_ERROR,
  435. "tdb_check: offset %llu bad prev"
  436. " (%llu vs %llu)",
  437. (long long)off,
  438. (long long)prev, (long long)frec_len(frec));
  439. }
  440. return TDB_SUCCESS;
  441. }
  442. static enum TDB_ERROR check_free_table(struct tdb_context *tdb,
  443. tdb_off_t ftable_off,
  444. unsigned ftable_num,
  445. tdb_off_t fr[],
  446. size_t num_free,
  447. size_t *num_found)
  448. {
  449. struct tdb_freetable ft;
  450. tdb_off_t h;
  451. unsigned int i;
  452. enum TDB_ERROR ecode;
  453. ecode = tdb_read_convert(tdb, ftable_off, &ft, sizeof(ft));
  454. if (ecode != TDB_SUCCESS) {
  455. return ecode;
  456. }
  457. if (rec_magic(&ft.hdr) != TDB_FTABLE_MAGIC
  458. || rec_key_length(&ft.hdr) != 0
  459. || rec_data_length(&ft.hdr) != sizeof(ft) - sizeof(ft.hdr)
  460. || rec_hash(&ft.hdr) != 0) {
  461. return tdb_logerr(tdb, TDB_ERR_CORRUPT, TDB_LOG_ERROR,
  462. "tdb_check: Invalid header on free table");
  463. }
  464. for (i = 0; i < TDB_FREE_BUCKETS; i++) {
  465. tdb_off_t off, prev = 0, *p;
  466. struct tdb_free_record f;
  467. h = bucket_off(ftable_off, i);
  468. for (off = tdb_read_off(tdb, h); off; off = f.next) {
  469. if (TDB_OFF_IS_ERR(off)) {
  470. return off;
  471. }
  472. ecode = tdb_read_convert(tdb, off, &f, sizeof(f));
  473. if (ecode != TDB_SUCCESS) {
  474. return ecode;
  475. }
  476. ecode = check_free(tdb, off, &f, prev, ftable_num, i);
  477. if (ecode != TDB_SUCCESS) {
  478. return false;
  479. }
  480. /* FIXME: Check hash bits */
  481. p = asearch(&off, fr, num_free, off_cmp);
  482. if (!p) {
  483. return tdb_logerr(tdb, TDB_ERR_CORRUPT,
  484. TDB_LOG_ERROR,
  485. "tdb_check: Invalid offset"
  486. " %llu in free table",
  487. (long long)off);
  488. }
  489. /* Mark it invalid. */
  490. *p ^= 1;
  491. (*num_found)++;
  492. prev = off;
  493. }
  494. }
  495. return TDB_SUCCESS;
  496. }
  497. /* Slow, but should be very rare. */
  498. tdb_off_t dead_space(struct tdb_context *tdb, tdb_off_t off)
  499. {
  500. size_t len;
  501. enum TDB_ERROR ecode;
  502. for (len = 0; off + len < tdb->map_size; len++) {
  503. char c;
  504. ecode = tdb->methods->tread(tdb, off, &c, 1);
  505. if (ecode != TDB_SUCCESS) {
  506. return ecode;
  507. }
  508. if (c != 0 && c != 0x43)
  509. break;
  510. }
  511. return len;
  512. }
  513. static enum TDB_ERROR check_linear(struct tdb_context *tdb,
  514. tdb_off_t **used, size_t *num_used,
  515. tdb_off_t **fr, size_t *num_free,
  516. tdb_off_t recovery)
  517. {
  518. tdb_off_t off;
  519. tdb_len_t len;
  520. enum TDB_ERROR ecode;
  521. bool found_recovery = false;
  522. for (off = sizeof(struct tdb_header); off < tdb->map_size; off += len) {
  523. union {
  524. struct tdb_used_record u;
  525. struct tdb_free_record f;
  526. struct tdb_recovery_record r;
  527. } rec;
  528. /* r is larger: only get that if we need to. */
  529. ecode = tdb_read_convert(tdb, off, &rec, sizeof(rec.f));
  530. if (ecode != TDB_SUCCESS) {
  531. return ecode;
  532. }
  533. /* If we crash after ftruncate, we can get zeroes or fill. */
  534. if (rec.r.magic == TDB_RECOVERY_INVALID_MAGIC
  535. || rec.r.magic == 0x4343434343434343ULL) {
  536. ecode = tdb_read_convert(tdb, off, &rec, sizeof(rec.r));
  537. if (ecode != TDB_SUCCESS) {
  538. return ecode;
  539. }
  540. if (recovery == off) {
  541. found_recovery = true;
  542. len = sizeof(rec.r) + rec.r.max_len;
  543. } else {
  544. len = dead_space(tdb, off);
  545. if (TDB_OFF_IS_ERR(len)) {
  546. return len;
  547. }
  548. if (len < sizeof(rec.r)) {
  549. return tdb_logerr(tdb, TDB_ERR_CORRUPT,
  550. TDB_LOG_ERROR,
  551. "tdb_check: invalid"
  552. " dead space at %zu",
  553. (size_t)off);
  554. }
  555. tdb_logerr(tdb, TDB_SUCCESS, TDB_LOG_WARNING,
  556. "Dead space at %zu-%zu (of %zu)",
  557. (size_t)off, (size_t)(off + len),
  558. (size_t)tdb->map_size);
  559. }
  560. } else if (rec.r.magic == TDB_RECOVERY_MAGIC) {
  561. ecode = tdb_read_convert(tdb, off, &rec, sizeof(rec.r));
  562. if (ecode != TDB_SUCCESS) {
  563. return ecode;
  564. }
  565. if (recovery != off) {
  566. return tdb_logerr(tdb, TDB_ERR_CORRUPT,
  567. TDB_LOG_ERROR,
  568. "tdb_check: unexpected"
  569. " recovery record at offset"
  570. " %zu",
  571. (size_t)off);
  572. }
  573. if (rec.r.len > rec.r.max_len) {
  574. return tdb_logerr(tdb, TDB_ERR_CORRUPT,
  575. TDB_LOG_ERROR,
  576. "tdb_check: invalid recovery"
  577. " length %zu",
  578. (size_t)rec.r.len);
  579. }
  580. if (rec.r.eof > tdb->map_size) {
  581. return tdb_logerr(tdb, TDB_ERR_CORRUPT,
  582. TDB_LOG_ERROR,
  583. "tdb_check: invalid old EOF"
  584. " %zu", (size_t)rec.r.eof);
  585. }
  586. found_recovery = true;
  587. len = sizeof(rec.r) + rec.r.max_len;
  588. } else if (frec_magic(&rec.f) == TDB_FREE_MAGIC) {
  589. len = sizeof(rec.u) + frec_len(&rec.f);
  590. if (off + len > tdb->map_size) {
  591. return tdb_logerr(tdb, TDB_ERR_CORRUPT,
  592. TDB_LOG_ERROR,
  593. "tdb_check: free overlength"
  594. " %llu at offset %llu",
  595. (long long)len,
  596. (long long)off);
  597. }
  598. /* This record should be in free lists. */
  599. if (frec_ftable(&rec.f) != TDB_FTABLE_NONE
  600. && !append(fr, num_free, off)) {
  601. return tdb_logerr(tdb, TDB_ERR_OOM,
  602. TDB_LOG_ERROR,
  603. "tdb_check: tracking %zu'th"
  604. " free record.", *num_free);
  605. }
  606. } else if (rec_magic(&rec.u) == TDB_USED_MAGIC
  607. || rec_magic(&rec.u) == TDB_CHAIN_MAGIC
  608. || rec_magic(&rec.u) == TDB_HTABLE_MAGIC
  609. || rec_magic(&rec.u) == TDB_FTABLE_MAGIC) {
  610. uint64_t klen, dlen, extra;
  611. /* This record is used! */
  612. if (!append(used, num_used, off)) {
  613. return tdb_logerr(tdb, TDB_ERR_OOM,
  614. TDB_LOG_ERROR,
  615. "tdb_check: tracking %zu'th"
  616. " used record.", *num_used);
  617. }
  618. klen = rec_key_length(&rec.u);
  619. dlen = rec_data_length(&rec.u);
  620. extra = rec_extra_padding(&rec.u);
  621. len = sizeof(rec.u) + klen + dlen + extra;
  622. if (off + len > tdb->map_size) {
  623. return tdb_logerr(tdb, TDB_ERR_CORRUPT,
  624. TDB_LOG_ERROR,
  625. "tdb_check: used overlength"
  626. " %llu at offset %llu",
  627. (long long)len,
  628. (long long)off);
  629. }
  630. if (len < sizeof(rec.f)) {
  631. return tdb_logerr(tdb, TDB_ERR_CORRUPT,
  632. TDB_LOG_ERROR,
  633. "tdb_check: too short record"
  634. " %llu at %llu",
  635. (long long)len,
  636. (long long)off);
  637. }
  638. } else {
  639. return tdb_logerr(tdb, TDB_ERR_CORRUPT,
  640. TDB_LOG_ERROR,
  641. "tdb_check: Bad magic 0x%llx"
  642. " at offset %zu",
  643. (long long)rec_magic(&rec.u),
  644. (size_t)off);
  645. }
  646. }
  647. /* We must have found recovery area if there was one. */
  648. if (recovery != 0 && !found_recovery) {
  649. return tdb_logerr(tdb, TDB_ERR_CORRUPT, TDB_LOG_ERROR,
  650. "tdb_check: expected a recovery area at %zu",
  651. (size_t)recovery);
  652. }
  653. return TDB_SUCCESS;
  654. }
  655. enum TDB_ERROR tdb_check(struct tdb_context *tdb,
  656. enum TDB_ERROR (*check)(TDB_DATA key, TDB_DATA data,
  657. void *private_data),
  658. void *private_data)
  659. {
  660. tdb_off_t *fr = NULL, *used = NULL, ft, recovery;
  661. size_t num_free = 0, num_used = 0, num_found = 0, num_ftables = 0;
  662. enum TDB_ERROR ecode;
  663. ecode = tdb_allrecord_lock(tdb, F_RDLCK, TDB_LOCK_WAIT, false);
  664. if (ecode != TDB_SUCCESS) {
  665. return ecode;
  666. }
  667. ecode = tdb_lock_expand(tdb, F_RDLCK);
  668. if (ecode != TDB_SUCCESS) {
  669. tdb_allrecord_unlock(tdb, F_RDLCK);
  670. return ecode;
  671. }
  672. ecode = check_header(tdb, &recovery);
  673. if (ecode != TDB_SUCCESS)
  674. goto out;
  675. /* First we do a linear scan, checking all records. */
  676. ecode = check_linear(tdb, &used, &num_used, &fr, &num_free, recovery);
  677. if (ecode != TDB_SUCCESS)
  678. goto out;
  679. for (ft = first_ftable(tdb); ft; ft = next_ftable(tdb, ft)) {
  680. if (TDB_OFF_IS_ERR(ft)) {
  681. ecode = ft;
  682. goto out;
  683. }
  684. ecode = check_free_table(tdb, ft, num_ftables, fr, num_free,
  685. &num_found);
  686. if (ecode != TDB_SUCCESS)
  687. goto out;
  688. num_ftables++;
  689. }
  690. /* FIXME: Check key uniqueness? */
  691. ecode = check_hash(tdb, used, num_used, num_ftables, check,
  692. private_data);
  693. if (ecode != TDB_SUCCESS)
  694. goto out;
  695. if (num_found != num_free) {
  696. ecode = tdb_logerr(tdb, TDB_ERR_CORRUPT, TDB_LOG_ERROR,
  697. "tdb_check: Not all entries are in"
  698. " free table");
  699. }
  700. out:
  701. tdb_allrecord_unlock(tdb, F_RDLCK);
  702. tdb_unlock_expand(tdb, F_RDLCK);
  703. free(fr);
  704. free(used);
  705. return ecode;
  706. }