mirror of
https://git.proxmox.com/git/mirror_zfs.git
synced 2026-05-22 02:27:36 +03:00
ddt: introduce lightweight entry
The idea here is that sometimes you need the contents of an entry with no intent to modify it, and/or from a place where its difficult to get hold of its originating ddt_t to know how to interpret it. A lightweight entry contains everything you might need to "read" an entry - its key, type and phys contents - but none of the extras for modifying it or using it in a larger context. It also has the full complement of phys slots, so it can represent any kind of dedup entry without having to know the specific configuration of the table it came from. Reviewed-by: Alexander Motin <mav@FreeBSD.org> Reviewed-by: Brian Behlendorf <behlendorf1@llnl.gov> Signed-off-by: Rob Norris <rob.norris@klarasystems.com> Sponsored-by: Klara, Inc. Sponsored-by: iXsystems, Inc. Closes #15893
This commit is contained in:
committed by
Brian Behlendorf
parent
d17ab631a9
commit
4d686c3da5
+16
-15
@@ -401,13 +401,20 @@ ddt_object_remove(ddt_t *ddt, ddt_type_t type, ddt_class_t class,
|
||||
|
||||
int
|
||||
ddt_object_walk(ddt_t *ddt, ddt_type_t type, ddt_class_t class,
|
||||
uint64_t *walk, ddt_entry_t *dde)
|
||||
uint64_t *walk, ddt_lightweight_entry_t *ddlwe)
|
||||
{
|
||||
ASSERT(ddt_object_exists(ddt, type, class));
|
||||
|
||||
return (ddt_ops[type]->ddt_op_walk(ddt->ddt_os,
|
||||
ddt->ddt_object[type][class], walk, &dde->dde_key,
|
||||
dde->dde_phys, sizeof (dde->dde_phys)));
|
||||
int error = ddt_ops[type]->ddt_op_walk(ddt->ddt_os,
|
||||
ddt->ddt_object[type][class], walk, &ddlwe->ddlwe_key,
|
||||
ddlwe->ddlwe_phys, sizeof (ddlwe->ddlwe_phys));
|
||||
if (error == 0) {
|
||||
ddlwe->ddlwe_type = type;
|
||||
ddlwe->ddlwe_class = class;
|
||||
ddlwe->ddlwe_nphys = DDT_NPHYS(ddt);
|
||||
return (0);
|
||||
}
|
||||
return (error);
|
||||
}
|
||||
|
||||
int
|
||||
@@ -572,12 +579,6 @@ ddt_select(spa_t *spa, const blkptr_t *bp)
|
||||
return (spa->spa_ddt[BP_GET_CHECKSUM(bp)]);
|
||||
}
|
||||
|
||||
ddt_t *
|
||||
ddt_select_checksum(spa_t *spa, enum zio_checksum checksum)
|
||||
{
|
||||
return (spa->spa_ddt[checksum]);
|
||||
}
|
||||
|
||||
void
|
||||
ddt_enter(ddt_t *ddt)
|
||||
{
|
||||
@@ -1347,8 +1348,10 @@ ddt_sync_entry(ddt_t *ddt, ddt_entry_t *dde, dmu_tx_t *tx, uint64_t txg)
|
||||
* traversing.)
|
||||
*/
|
||||
if (nclass < oclass) {
|
||||
ddt_lightweight_entry_t ddlwe;
|
||||
DDT_ENTRY_TO_LIGHTWEIGHT(ddt, dde, &ddlwe);
|
||||
dsl_scan_ddt_entry(dp->dp_scan,
|
||||
ddt->ddt_checksum, dde, tx);
|
||||
ddt->ddt_checksum, &ddlwe, tx);
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -1455,7 +1458,7 @@ ddt_sync(spa_t *spa, uint64_t txg)
|
||||
}
|
||||
|
||||
int
|
||||
ddt_walk(spa_t *spa, ddt_bookmark_t *ddb, ddt_entry_t *dde)
|
||||
ddt_walk(spa_t *spa, ddt_bookmark_t *ddb, ddt_lightweight_entry_t *ddlwe)
|
||||
{
|
||||
do {
|
||||
do {
|
||||
@@ -1468,10 +1471,8 @@ ddt_walk(spa_t *spa, ddt_bookmark_t *ddb, ddt_entry_t *dde)
|
||||
ddb->ddb_class)) {
|
||||
error = ddt_object_walk(ddt,
|
||||
ddb->ddb_type, ddb->ddb_class,
|
||||
&ddb->ddb_cursor, dde);
|
||||
&ddb->ddb_cursor, ddlwe);
|
||||
}
|
||||
dde->dde_type = ddb->ddb_type;
|
||||
dde->dde_class = ddb->ddb_class;
|
||||
if (error == 0)
|
||||
return (0);
|
||||
if (error != ENOENT)
|
||||
|
||||
@@ -2929,10 +2929,10 @@ enqueue_cb(dsl_pool_t *dp, dsl_dataset_t *hds, void *arg)
|
||||
|
||||
void
|
||||
dsl_scan_ddt_entry(dsl_scan_t *scn, enum zio_checksum checksum,
|
||||
ddt_entry_t *dde, dmu_tx_t *tx)
|
||||
ddt_lightweight_entry_t *ddlwe, dmu_tx_t *tx)
|
||||
{
|
||||
(void) tx;
|
||||
const ddt_key_t *ddk = &dde->dde_key;
|
||||
const ddt_key_t *ddk = &ddlwe->ddlwe_key;
|
||||
blkptr_t bp;
|
||||
zbookmark_phys_t zb = { 0 };
|
||||
|
||||
@@ -2953,9 +2953,8 @@ dsl_scan_ddt_entry(dsl_scan_t *scn, enum zio_checksum checksum,
|
||||
if (scn->scn_done_txg != 0)
|
||||
return;
|
||||
|
||||
ddt_t *ddt = ddt_select_checksum(tx->tx_pool->dp_spa, checksum);
|
||||
for (int p = 0; p < DDT_NPHYS(ddt); p++) {
|
||||
ddt_phys_t *ddp = &dde->dde_phys[p];
|
||||
for (int p = 0; p < ddlwe->ddlwe_nphys; p++) {
|
||||
ddt_phys_t *ddp = &ddlwe->ddlwe_phys[p];
|
||||
|
||||
if (ddp->ddp_phys_birth == 0 ||
|
||||
ddp->ddp_phys_birth > scn->scn_phys.scn_max_txg)
|
||||
@@ -3004,11 +3003,11 @@ static void
|
||||
dsl_scan_ddt(dsl_scan_t *scn, dmu_tx_t *tx)
|
||||
{
|
||||
ddt_bookmark_t *ddb = &scn->scn_phys.scn_ddt_bookmark;
|
||||
ddt_entry_t dde = {{{{0}}}};
|
||||
ddt_lightweight_entry_t ddlwe = {0};
|
||||
int error;
|
||||
uint64_t n = 0;
|
||||
|
||||
while ((error = ddt_walk(scn->scn_dp->dp_spa, ddb, &dde)) == 0) {
|
||||
while ((error = ddt_walk(scn->scn_dp->dp_spa, ddb, &ddlwe)) == 0) {
|
||||
ddt_t *ddt;
|
||||
|
||||
if (ddb->ddb_class > scn->scn_phys.scn_ddt_class_max)
|
||||
@@ -3023,7 +3022,7 @@ dsl_scan_ddt(dsl_scan_t *scn, dmu_tx_t *tx)
|
||||
ddt = scn->scn_dp->dp_spa->spa_ddt[ddb->ddb_checksum];
|
||||
ASSERT(avl_first(&ddt->ddt_tree) == NULL);
|
||||
|
||||
dsl_scan_ddt_entry(scn, ddb->ddb_checksum, &dde, tx);
|
||||
dsl_scan_ddt_entry(scn, ddb->ddb_checksum, &ddlwe, tx);
|
||||
n++;
|
||||
|
||||
if (dsl_scan_check_suspend(scn, NULL))
|
||||
|
||||
Reference in New Issue
Block a user