X-Git-Url: https://git.ozlabs.org/?p=ccan;a=blobdiff_plain;f=ccan%2Ftdb2%2Ftdb.c;h=d3d12250cea8b21f4bf76431957f660796d83dad;hp=a1ec332a09b1af790cff0934fac37b084dfae914;hb=a97da100b00206544c7a68593b64a49f2b854f7e;hpb=74b0109ebd2a30d57d19ae9a56f16c0f3b3637eb diff --git a/ccan/tdb2/tdb.c b/ccan/tdb2/tdb.c index a1ec332a..d3d12250 100644 --- a/ccan/tdb2/tdb.c +++ b/ccan/tdb2/tdb.c @@ -243,6 +243,21 @@ enum TDB_ERROR tdb_fetch(struct tdb_context *tdb, struct tdb_data key, return ecode; } +bool tdb_exists(struct tdb_context *tdb, TDB_DATA key) +{ + tdb_off_t off; + struct tdb_used_record rec; + struct hash_info h; + + off = find_and_lock(tdb, key, F_RDLCK, &h, &rec, NULL); + if (TDB_OFF_IS_ERR(off)) { + return false; + } + tdb_unlock_hashes(tdb, h.hlock_start, h.hlock_range, F_RDLCK); + + return off ? true : false; +} + enum TDB_ERROR tdb_delete(struct tdb_context *tdb, struct tdb_data key) { tdb_off_t off; @@ -377,3 +392,42 @@ enum TDB_ERROR COLD tdb_logerr(struct tdb_context *tdb, errno = saved_errno; return ecode; } + +enum TDB_ERROR tdb_parse_record_(struct tdb_context *tdb, + TDB_DATA key, + enum TDB_ERROR (*parse)(TDB_DATA key, + TDB_DATA data, + void *p), + void *p) +{ + tdb_off_t off; + struct tdb_used_record rec; + struct hash_info h; + TDB_DATA data; + enum TDB_ERROR ecode; + + off = find_and_lock(tdb, key, F_RDLCK, &h, &rec, NULL); + if (TDB_OFF_IS_ERR(off)) { + return off; + } + + if (!off) { + ecode = TDB_ERR_NOEXIST; + } else { + data.dsize = rec_data_length(&rec); + data.dptr = (void *)tdb_access_read(tdb, + off + sizeof(rec) + + key.dsize, + data.dsize, false); + if (TDB_PTR_IS_ERR(data.dptr)) { + ecode = TDB_PTR_ERR(data.dptr); + } else { + ecode = parse(key, data, p); + tdb_access_release(tdb, data.dptr); + } + } + + tdb_unlock_hashes(tdb, h.hlock_start, h.hlock_range, F_RDLCK); + return ecode; +} +