Skip to content

Commit

Permalink
ZFS traverse_visitbp optimization to limit prefetch.
Browse files Browse the repository at this point in the history
Traversal code, traverse_visitbp() does visit blocks recursively.
Indirect (Non L0) Block of size 128k could contain, 1024 block pointers
of 128 bytes. In case of full traverse OR incremental traverse, where
all blocks were modified, it could traverse large number of blocks
pointed by indirect. Traversal code does issue prefetch of blocks
traversed below indirect. This could result into large number of
async reads queued on vdev queue. So, account for prefetch issued for
blocks pointed by indirect and limit max prefetch in one go.

Module Param:
zfs_traverse_indirect_prefetch_limit: Limit of prefetch while traversing
an indirect block

Local counters:
prefetched: Local counter to account for number prefetch done.
prefetchidx: Index for which next prefetch to be issued.
prefetchtriggeridx: Index at which next prefetch to be triggered.

Keeps prefetchtriggeridx somewhere in the middle of blocks prefetched,
so that prefetching blocks gets a enough time window before their demand
read is issued.

Signed-off-by: Jitendra Patidar <jitendra.patidar@nutanix.com>
Closes #11802
  • Loading branch information
jsai20 committed Mar 30, 2021
1 parent 3522f57 commit 2167ff7
Showing 1 changed file with 60 additions and 15 deletions.
75 changes: 60 additions & 15 deletions module/zfs/dmu_traverse.c
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,7 @@

int32_t zfs_pd_bytes_max = 50 * 1024 * 1024; /* 50MB */
int32_t send_holes_without_birth_time = 1;
int32_t zfs_traverse_indirect_prefetch_limit = 10;

typedef struct prefetch_data {
kmutex_t pd_mtx;
Expand Down Expand Up @@ -176,33 +177,37 @@ resume_skip_check(traverse_data_t *td, const dnode_phys_t *dnp,
return (RESUME_SKIP_NONE);
}

static void
/*
* Returns B_TRUE, if prefetch read is issued, otherwise B_FALSE.
*/
static boolean_t
traverse_prefetch_metadata(traverse_data_t *td,
const blkptr_t *bp, const zbookmark_phys_t *zb)
{
arc_flags_t flags = ARC_FLAG_NOWAIT | ARC_FLAG_PREFETCH;
int zio_flags = ZIO_FLAG_CANFAIL | ZIO_FLAG_SPECULATIVE;

if (!(td->td_flags & TRAVERSE_PREFETCH_METADATA))
return;
return (B_FALSE);
/*
* If we are in the process of resuming, don't prefetch, because
* some children will not be needed (and in fact may have already
* been freed).
*/
if (td->td_resume != NULL && !ZB_IS_ZERO(td->td_resume))
return;
return (B_FALSE);
if (BP_IS_HOLE(bp) || bp->blk_birth <= td->td_min_txg)
return;
return (B_FALSE);
if (BP_GET_LEVEL(bp) == 0 && BP_GET_TYPE(bp) != DMU_OT_DNODE)
return;
return (B_FALSE);
ASSERT(!BP_IS_REDACTED(bp));

if ((td->td_flags & TRAVERSE_NO_DECRYPT) && BP_IS_PROTECTED(bp))
zio_flags |= ZIO_FLAG_RAW;

(void) arc_read(NULL, td->td_spa, bp, NULL, NULL,
ZIO_PRIORITY_ASYNC_READ, zio_flags, &flags, zb);
return (B_TRUE);
}

static boolean_t
Expand Down Expand Up @@ -295,7 +300,8 @@ traverse_visitbp(traverse_data_t *td, const dnode_phys_t *dnp,

if (BP_GET_LEVEL(bp) > 0) {
uint32_t flags = ARC_FLAG_WAIT;
int32_t i;
int32_t i, j, prefetchtriggeridx;
int32_t prefetchidx, prefetchlimit, prefetched;
int32_t epb = BP_GET_LSIZE(bp) >> SPA_BLKPTRSHIFT;
zbookmark_phys_t *czb;

Expand All @@ -308,16 +314,51 @@ traverse_visitbp(traverse_data_t *td, const dnode_phys_t *dnp,

czb = kmem_alloc(sizeof (zbookmark_phys_t), KM_SLEEP);

/*
* recursively visitbp() blocks below this.
* Indirect (Non L0) Block of size 128k could contain, 1024
* block pointers of 128 bytes. In case of full traverse OR
* incremental traverse, where all blocks were modified, it
* could traverse large number of blocks pointed by indirect.
* Prefetching all blocks in go, could result into large number
* of async reads queued on vdev queue. So, account for prefetch
* issued for blocks pointed by indirect and limit max prefetch
* in one go to zfs_traverse_indirect_prefetch_limit.
*/
prefetchtriggeridx = 0;
prefetchidx = 0;
prefetched = 0;
prefetchlimit = MIN(epb, zfs_traverse_indirect_prefetch_limit);
ASSERT(prefetchtriggeridx <= prefetchidx);
for (i = 0; i < epb; i++) {
SET_BOOKMARK(czb, zb->zb_objset, zb->zb_object,
zb->zb_level - 1,
zb->zb_blkid * epb + i);
traverse_prefetch_metadata(td,
&((blkptr_t *)buf->b_data)[i], czb);
}

/* recursively visitbp() blocks below this */
for (i = 0; i < epb; i++) {
if (prefetchlimit && i == prefetchtriggeridx &&
prefetchidx < epb) {
ASSERT(prefetchtriggeridx <= prefetchidx);
prefetched = 0;
for (j = prefetchidx; j < epb &&
prefetched < prefetchlimit; j++) {
SET_BOOKMARK(czb, zb->zb_objset,
zb->zb_object, zb->zb_level - 1,
zb->zb_blkid * epb + j);
if (traverse_prefetch_metadata(td,
&((blkptr_t *)buf->b_data)[j], czb)
== B_TRUE) {
prefetched++;
if (prefetched ==
MAX((prefetchlimit >> 1),
1))
prefetchtriggeridx = j;
}
}
prefetchidx = j;
#ifdef DEBUG
dprintf("(objset,object,lvl):(%d,%d,%d), "
"(prefetched,prefetchtriggeridx,"
"prefetchidx):(%d,%d,%d)\n", zb->zb_objset,
zb->zb_object, zb->zb_level, prefetched,
prefetchtriggeridx, prefetchidx);
#endif
}
SET_BOOKMARK(czb, zb->zb_objset, zb->zb_object,
zb->zb_level - 1,
zb->zb_blkid * epb + i);
Expand Down Expand Up @@ -778,6 +819,10 @@ ZFS_MODULE_PARAM(zfs, zfs_, pd_bytes_max, INT, ZMOD_RW,
"Max number of bytes to prefetch");

#if defined(_KERNEL)
module_param(zfs_traverse_indirect_prefetch_limit, int, 0644);
MODULE_PARM_DESC(zfs_traverse_indirect_prefetch_limit,
"Traverse prefetch number of blocks pointed by indirect block");

module_param_named(ignore_hole_birth, send_holes_without_birth_time, int, 0644);
MODULE_PARM_DESC(ignore_hole_birth,
"Alias for send_holes_without_birth_time");
Expand Down

0 comments on commit 2167ff7

Please sign in to comment.