Commit 63e3809c authored by Javier González's avatar Javier González Committed by Jens Axboe

lightnvm: pblk: set metadata list for all I/Os

Set a dma area for all I/Os in order to read/write from/to the metadata
stored on the per-sector out-of-bound area.
Signed-off-by: default avatarJavier González <javier@cnexlabs.com>
Signed-off-by: default avatarMatias Bjørling <matias@cnexlabs.com>
Signed-off-by: default avatarJens Axboe <axboe@kernel.dk>
parent d45ebd47
...@@ -555,10 +555,10 @@ static int pblk_line_submit_emeta_io(struct pblk *pblk, struct pblk_line *line, ...@@ -555,10 +555,10 @@ static int pblk_line_submit_emeta_io(struct pblk *pblk, struct pblk_line *line,
struct nvm_tgt_dev *dev = pblk->dev; struct nvm_tgt_dev *dev = pblk->dev;
struct nvm_geo *geo = &dev->geo; struct nvm_geo *geo = &dev->geo;
struct pblk_line_meta *lm = &pblk->lm; struct pblk_line_meta *lm = &pblk->lm;
void *ppa_list, *meta_list;
struct bio *bio; struct bio *bio;
struct nvm_rq rqd; struct nvm_rq rqd;
struct ppa_addr *ppa_list; dma_addr_t dma_ppa_list, dma_meta_list;
dma_addr_t dma_ppa_list;
int min = pblk->min_write_pgs; int min = pblk->min_write_pgs;
int left_ppas = lm->emeta_sec[0]; int left_ppas = lm->emeta_sec[0];
int id = line->id; int id = line->id;
...@@ -577,10 +577,14 @@ static int pblk_line_submit_emeta_io(struct pblk *pblk, struct pblk_line *line, ...@@ -577,10 +577,14 @@ static int pblk_line_submit_emeta_io(struct pblk *pblk, struct pblk_line *line,
} else } else
return -EINVAL; return -EINVAL;
ppa_list = nvm_dev_dma_alloc(dev->parent, GFP_KERNEL, &dma_ppa_list); meta_list = nvm_dev_dma_alloc(dev->parent, GFP_KERNEL,
if (!ppa_list) &dma_meta_list);
if (!meta_list)
return -ENOMEM; return -ENOMEM;
ppa_list = meta_list + pblk_dma_meta_size;
dma_ppa_list = dma_meta_list + pblk_dma_meta_size;
next_rq: next_rq:
memset(&rqd, 0, sizeof(struct nvm_rq)); memset(&rqd, 0, sizeof(struct nvm_rq));
...@@ -597,23 +601,29 @@ static int pblk_line_submit_emeta_io(struct pblk *pblk, struct pblk_line *line, ...@@ -597,23 +601,29 @@ static int pblk_line_submit_emeta_io(struct pblk *pblk, struct pblk_line *line,
bio_set_op_attrs(bio, bio_op, 0); bio_set_op_attrs(bio, bio_op, 0);
rqd.bio = bio; rqd.bio = bio;
rqd.opcode = cmd_op; rqd.meta_list = meta_list;
rqd.nr_ppas = rq_ppas;
rqd.ppa_list = ppa_list; rqd.ppa_list = ppa_list;
rqd.dma_meta_list = dma_meta_list;
rqd.dma_ppa_list = dma_ppa_list; rqd.dma_ppa_list = dma_ppa_list;
rqd.opcode = cmd_op;
rqd.nr_ppas = rq_ppas;
rqd.end_io = pblk_end_io_sync; rqd.end_io = pblk_end_io_sync;
rqd.private = &wait; rqd.private = &wait;
if (dir == WRITE) { if (dir == WRITE) {
struct pblk_sec_meta *meta_list = rqd.meta_list;
rqd.flags = pblk_set_progr_mode(pblk, WRITE); rqd.flags = pblk_set_progr_mode(pblk, WRITE);
for (i = 0; i < rqd.nr_ppas; ) { for (i = 0; i < rqd.nr_ppas; ) {
spin_lock(&line->lock); spin_lock(&line->lock);
paddr = __pblk_alloc_page(pblk, line, min); paddr = __pblk_alloc_page(pblk, line, min);
spin_unlock(&line->lock); spin_unlock(&line->lock);
for (j = 0; j < min; j++, i++, paddr++) for (j = 0; j < min; j++, i++, paddr++) {
meta_list[i].lba = cpu_to_le64(ADDR_EMPTY);
rqd.ppa_list[i] = rqd.ppa_list[i] =
addr_to_gen_ppa(pblk, paddr, id); addr_to_gen_ppa(pblk, paddr, id);
} }
}
} else { } else {
for (i = 0; i < rqd.nr_ppas; ) { for (i = 0; i < rqd.nr_ppas; ) {
struct ppa_addr ppa = addr_to_gen_ppa(pblk, paddr, id); struct ppa_addr ppa = addr_to_gen_ppa(pblk, paddr, id);
...@@ -680,7 +690,7 @@ static int pblk_line_submit_emeta_io(struct pblk *pblk, struct pblk_line *line, ...@@ -680,7 +690,7 @@ static int pblk_line_submit_emeta_io(struct pblk *pblk, struct pblk_line *line,
if (left_ppas) if (left_ppas)
goto next_rq; goto next_rq;
free_rqd_dma: free_rqd_dma:
nvm_dev_dma_free(dev->parent, ppa_list, dma_ppa_list); nvm_dev_dma_free(dev->parent, rqd.meta_list, rqd.dma_meta_list);
return ret; return ret;
} }
...@@ -726,11 +736,14 @@ static int pblk_line_submit_smeta_io(struct pblk *pblk, struct pblk_line *line, ...@@ -726,11 +736,14 @@ static int pblk_line_submit_smeta_io(struct pblk *pblk, struct pblk_line *line,
memset(&rqd, 0, sizeof(struct nvm_rq)); memset(&rqd, 0, sizeof(struct nvm_rq));
rqd.ppa_list = nvm_dev_dma_alloc(dev->parent, GFP_KERNEL, rqd.meta_list = nvm_dev_dma_alloc(dev->parent, GFP_KERNEL,
&rqd.dma_ppa_list); &rqd.dma_meta_list);
if (!rqd.ppa_list) if (!rqd.meta_list)
return -ENOMEM; return -ENOMEM;
rqd.ppa_list = rqd.meta_list + pblk_dma_meta_size;
rqd.dma_ppa_list = rqd.dma_meta_list + pblk_dma_meta_size;
bio = bio_map_kern(dev->q, line->smeta, lm->smeta_len, GFP_KERNEL); bio = bio_map_kern(dev->q, line->smeta, lm->smeta_len, GFP_KERNEL);
if (IS_ERR(bio)) { if (IS_ERR(bio)) {
ret = PTR_ERR(bio); ret = PTR_ERR(bio);
...@@ -748,9 +761,15 @@ static int pblk_line_submit_smeta_io(struct pblk *pblk, struct pblk_line *line, ...@@ -748,9 +761,15 @@ static int pblk_line_submit_smeta_io(struct pblk *pblk, struct pblk_line *line,
rqd.private = &wait; rqd.private = &wait;
for (i = 0; i < lm->smeta_sec; i++, paddr++) { for (i = 0; i < lm->smeta_sec; i++, paddr++) {
struct pblk_sec_meta *meta_list = rqd.meta_list;
rqd.ppa_list[i] = addr_to_gen_ppa(pblk, paddr, line->id); rqd.ppa_list[i] = addr_to_gen_ppa(pblk, paddr, line->id);
if (dir == WRITE)
lba_list[paddr] = cpu_to_le64(ADDR_EMPTY); if (dir == WRITE) {
u64 addr_empty = cpu_to_le64(ADDR_EMPTY);
meta_list[i].lba = lba_list[paddr] = addr_empty;
}
} }
/* /*
...@@ -778,7 +797,7 @@ static int pblk_line_submit_smeta_io(struct pblk *pblk, struct pblk_line *line, ...@@ -778,7 +797,7 @@ static int pblk_line_submit_smeta_io(struct pblk *pblk, struct pblk_line *line,
} }
free_ppa_list: free_ppa_list:
nvm_dev_dma_free(dev->parent, rqd.ppa_list, rqd.dma_ppa_list); nvm_dev_dma_free(dev->parent, rqd.meta_list, rqd.dma_meta_list);
return ret; return ret;
} }
......
...@@ -123,8 +123,7 @@ static void pblk_end_io_read(struct nvm_rq *rqd) ...@@ -123,8 +123,7 @@ static void pblk_end_io_read(struct nvm_rq *rqd)
WARN_ONCE(bio->bi_status, "pblk: corrupted read error\n"); WARN_ONCE(bio->bi_status, "pblk: corrupted read error\n");
#endif #endif
if (rqd->nr_ppas > 1) nvm_dev_dma_free(dev->parent, rqd->meta_list, rqd->dma_meta_list);
nvm_dev_dma_free(dev->parent, rqd->ppa_list, rqd->dma_ppa_list);
bio_put(bio); bio_put(bio);
if (r_ctx->private) { if (r_ctx->private) {
...@@ -329,14 +328,17 @@ int pblk_submit_read(struct pblk *pblk, struct bio *bio) ...@@ -329,14 +328,17 @@ int pblk_submit_read(struct pblk *pblk, struct bio *bio)
*/ */
bio_init_idx = pblk_get_bi_idx(bio); bio_init_idx = pblk_get_bi_idx(bio);
if (nr_secs > 1) { rqd->meta_list = nvm_dev_dma_alloc(dev->parent, GFP_KERNEL,
rqd->ppa_list = nvm_dev_dma_alloc(dev->parent, GFP_KERNEL, &rqd->dma_meta_list);
&rqd->dma_ppa_list); if (!rqd->meta_list) {
if (!rqd->ppa_list) {
pr_err("pblk: not able to allocate ppa list\n"); pr_err("pblk: not able to allocate ppa list\n");
goto fail_rqd_free; goto fail_rqd_free;
} }
if (nr_secs > 1) {
rqd->ppa_list = rqd->meta_list + pblk_dma_meta_size;
rqd->dma_ppa_list = rqd->dma_meta_list + pblk_dma_meta_size;
pblk_read_ppalist_rq(pblk, rqd, &read_bitmap); pblk_read_ppalist_rq(pblk, rqd, &read_bitmap);
} else { } else {
pblk_read_rq(pblk, rqd, &read_bitmap); pblk_read_rq(pblk, rqd, &read_bitmap);
...@@ -466,22 +468,19 @@ int pblk_submit_read_gc(struct pblk *pblk, u64 *lba_list, void *data, ...@@ -466,22 +468,19 @@ int pblk_submit_read_gc(struct pblk *pblk, u64 *lba_list, void *data,
memset(&rqd, 0, sizeof(struct nvm_rq)); memset(&rqd, 0, sizeof(struct nvm_rq));
if (nr_secs > 1) { rqd.meta_list = nvm_dev_dma_alloc(dev->parent, GFP_KERNEL,
rqd.ppa_list = nvm_dev_dma_alloc(dev->parent, GFP_KERNEL, &rqd.dma_meta_list);
&rqd.dma_ppa_list); if (!rqd.meta_list)
if (!rqd.ppa_list)
return NVM_IO_ERR; return NVM_IO_ERR;
if (nr_secs > 1) {
rqd.ppa_list = rqd.meta_list + pblk_dma_meta_size;
rqd.dma_ppa_list = rqd.dma_meta_list + pblk_dma_meta_size;
*secs_to_gc = read_ppalist_rq_gc(pblk, &rqd, line, lba_list, *secs_to_gc = read_ppalist_rq_gc(pblk, &rqd, line, lba_list,
nr_secs); nr_secs);
if (*secs_to_gc == 1) { if (*secs_to_gc == 1)
struct ppa_addr ppa; rqd.ppa_addr = rqd.ppa_list[0];
ppa = rqd.ppa_list[0];
nvm_dev_dma_free(dev->parent, rqd.ppa_list,
rqd.dma_ppa_list);
rqd.ppa_addr = ppa;
}
} else { } else {
*secs_to_gc = read_rq_gc(pblk, &rqd, line, lba_list[0]); *secs_to_gc = read_rq_gc(pblk, &rqd, line, lba_list[0]);
} }
...@@ -532,12 +531,10 @@ int pblk_submit_read_gc(struct pblk *pblk, u64 *lba_list, void *data, ...@@ -532,12 +531,10 @@ int pblk_submit_read_gc(struct pblk *pblk, u64 *lba_list, void *data,
#endif #endif
out: out:
if (rqd.nr_ppas > 1) nvm_dev_dma_free(dev->parent, rqd.meta_list, rqd.dma_meta_list);
nvm_dev_dma_free(dev->parent, rqd.ppa_list, rqd.dma_ppa_list);
return NVM_IO_OK; return NVM_IO_OK;
err_free_dma: err_free_dma:
if (rqd.nr_ppas > 1) nvm_dev_dma_free(dev->parent, rqd.meta_list, rqd.dma_meta_list);
nvm_dev_dma_free(dev->parent, rqd.ppa_list, rqd.dma_ppa_list);
return NVM_IO_ERR; return NVM_IO_ERR;
} }
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment