Skip to content

Commit

Permalink
nvdimm: Use namespace index data to reduce number of label reads needed
Browse files Browse the repository at this point in the history
This patch adds logic that is meant to make use of the namespace index data
to reduce the number of reads that are needed to initialize a given
namespace. The general idea is that once we have enough data to validate
the namespace index we do so and then proceed to fetch only those labels
that are not listed as being "free". By doing this I am seeing a total time
reduction from about 4-5 seconds to 2-3 seconds for 24 NVDIMM modules each
with 128K of label config area.

Reviewed-by: Toshi Kani <toshi.kani@hpe.com>
Signed-off-by: Alexander Duyck <alexander.h.duyck@linux.intel.com>
Signed-off-by: Dan Williams <dan.j.williams@intel.com>
  • Loading branch information
Alexander Duyck authored and djbw committed Oct 12, 2018
1 parent 2d657d1 commit 7d47aad
Show file tree
Hide file tree
Showing 3 changed files with 88 additions and 12 deletions.
4 changes: 0 additions & 4 deletions drivers/nvdimm/dimm.c
Original file line number Diff line number Diff line change
Expand Up @@ -84,10 +84,6 @@ static int nvdimm_probe(struct device *dev)
dev_dbg(dev, "config data size: %d\n", ndd->nsarea.config_size);

nvdimm_bus_lock(dev);
ndd->ns_current = nd_label_validate(ndd);
ndd->ns_next = nd_label_next_nsindex(ndd->ns_current);
nd_label_copy(ndd, to_next_namespace_index(ndd),
to_current_namespace_index(ndd));
if (ndd->ns_current >= 0) {
rc = nd_label_reserve_dpa(ndd);
if (rc == 0)
Expand Down
93 changes: 88 additions & 5 deletions drivers/nvdimm/label.c
Original file line number Diff line number Diff line change
Expand Up @@ -235,7 +235,7 @@ static int __nd_label_validate(struct nvdimm_drvdata *ndd)
return -1;
}

int nd_label_validate(struct nvdimm_drvdata *ndd)
static int nd_label_validate(struct nvdimm_drvdata *ndd)
{
/*
* In order to probe for and validate namespace index blocks we
Expand All @@ -258,8 +258,9 @@ int nd_label_validate(struct nvdimm_drvdata *ndd)
return -1;
}

void nd_label_copy(struct nvdimm_drvdata *ndd, struct nd_namespace_index *dst,
struct nd_namespace_index *src)
static void nd_label_copy(struct nvdimm_drvdata *ndd,
struct nd_namespace_index *dst,
struct nd_namespace_index *src)
{
/* just exit if either destination or source is NULL */
if (!dst || !src)
Expand Down Expand Up @@ -419,7 +420,9 @@ int nd_label_reserve_dpa(struct nvdimm_drvdata *ndd)

int nd_label_data_init(struct nvdimm_drvdata *ndd)
{
size_t config_size, read_size;
size_t config_size, read_size, max_xfer, offset;
struct nd_namespace_index *nsindex;
unsigned int i;
int rc = 0;

if (ndd->data)
Expand Down Expand Up @@ -452,7 +455,87 @@ int nd_label_data_init(struct nvdimm_drvdata *ndd)
if (!ndd->data)
return -ENOMEM;

return nvdimm_get_config_data(ndd, ndd->data, 0, config_size);
/*
* We want to guarantee as few reads as possible while conserving
* memory. To do that we figure out how much unused space will be left
* in the last read, divide that by the total number of reads it is
* going to take given our maximum transfer size, and then reduce our
* maximum transfer size based on that result.
*/
max_xfer = min_t(size_t, ndd->nsarea.max_xfer, config_size);
if (read_size < max_xfer) {
/* trim waste */
max_xfer -= ((max_xfer - 1) - (config_size - 1) % max_xfer) /
DIV_ROUND_UP(config_size, max_xfer);
/* make certain we read indexes in exactly 1 read */
if (max_xfer < read_size)
max_xfer = read_size;
}

/* Make our initial read size a multiple of max_xfer size */
read_size = min(DIV_ROUND_UP(read_size, max_xfer) * max_xfer,
config_size);

/* Read the index data */
rc = nvdimm_get_config_data(ndd, ndd->data, 0, read_size);
if (rc)
goto out_err;

/* Validate index data, if not valid assume all labels are invalid */
ndd->ns_current = nd_label_validate(ndd);
if (ndd->ns_current < 0)
return 0;

/* Record our index values */
ndd->ns_next = nd_label_next_nsindex(ndd->ns_current);

/* Copy "current" index on top of the "next" index */
nsindex = to_current_namespace_index(ndd);
nd_label_copy(ndd, to_next_namespace_index(ndd), nsindex);

/* Determine starting offset for label data */
offset = __le64_to_cpu(nsindex->labeloff);

/* Loop through the free list pulling in any active labels */
for (i = 0; i < nsindex->nslot; i++, offset += ndd->nslabel_size) {
size_t label_read_size;

/* zero out the unused labels */
if (test_bit_le(i, nsindex->free)) {
memset(ndd->data + offset, 0, ndd->nslabel_size);
continue;
}

/* if we already read past here then just continue */
if (offset + ndd->nslabel_size <= read_size)
continue;

/* if we haven't read in a while reset our read_size offset */
if (read_size < offset)
read_size = offset;

/* determine how much more will be read after this next call. */
label_read_size = offset + ndd->nslabel_size - read_size;
label_read_size = DIV_ROUND_UP(label_read_size, max_xfer) *
max_xfer;

/* truncate last read if needed */
if (read_size + label_read_size > config_size)
label_read_size = config_size - read_size;

/* Read the label data */
rc = nvdimm_get_config_data(ndd, ndd->data + read_size,
read_size, label_read_size);
if (rc)
goto out_err;

/* push read_size to next read offset */
read_size += label_read_size;
}

dev_dbg(ndd->dev, "len: %zu rc: %d\n", offset, rc);
out_err:
return rc;
}

int nd_label_active_count(struct nvdimm_drvdata *ndd)
Expand Down
3 changes: 0 additions & 3 deletions drivers/nvdimm/label.h
Original file line number Diff line number Diff line change
Expand Up @@ -138,9 +138,6 @@ static inline int nd_label_next_nsindex(int index)
}

struct nvdimm_drvdata;
int nd_label_validate(struct nvdimm_drvdata *ndd);
void nd_label_copy(struct nvdimm_drvdata *ndd, struct nd_namespace_index *dst,
struct nd_namespace_index *src);
int nd_label_data_init(struct nvdimm_drvdata *ndd);
size_t sizeof_namespace_index(struct nvdimm_drvdata *ndd);
int nd_label_active_count(struct nvdimm_drvdata *ndd);
Expand Down

0 comments on commit 7d47aad

Please sign in to comment.