aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMatias Bjørling <m@bjorling.me>2016-05-06 14:02:58 -0400
committerJens Axboe <axboe@fb.com>2016-05-06 14:51:10 -0400
commit22e8c9766a669d49cf3749d397082a5cd93374a9 (patch)
treec24facba8321c5359bf6f111f2ca892cf894b62a
parent4891d120b9cd419f4350b11e1231083745dcdc8b (diff)
lightnvm: move block fold outside of get_bb_tbl()
The get block table command returns a list of blocks and planes with their associated state. Users, such as gennvm and sysblk, manages all planes as a single virtual block. It was therefore natural to fold the bad block list before it is returned. However, to allow users, which manages on a per-plane block level, to also use the interface, the get_bb_tbl interface is changed to not fold by default and instead let the caller fold if necessary. Reviewed by: Johannes Thumshirn <jthumshirn@suse.de> Signed-off-by: Matias Bjørling <m@bjorling.me> Signed-off-by: Jens Axboe <axboe@fb.com>
-rw-r--r--drivers/lightnvm/core.c35
-rw-r--r--drivers/lightnvm/gennvm.c14
-rw-r--r--drivers/lightnvm/sysblk.c29
-rw-r--r--drivers/nvme/host/lightnvm.c47
-rw-r--r--include/linux/lightnvm.h6
5 files changed, 73 insertions, 58 deletions
diff --git a/drivers/lightnvm/core.c b/drivers/lightnvm/core.c
index 652b8c7673ee..4cadbe0cd537 100644
--- a/drivers/lightnvm/core.c
+++ b/drivers/lightnvm/core.c
@@ -420,6 +420,41 @@ int nvm_submit_ppa(struct nvm_dev *dev, struct ppa_addr *ppa, int nr_ppas,
420} 420}
421EXPORT_SYMBOL(nvm_submit_ppa); 421EXPORT_SYMBOL(nvm_submit_ppa);
422 422
423/*
424 * folds a bad block list from its plane representation to its virtual
425 * block representation. The fold is done in place and reduced size is
426 * returned.
427 *
428 * If any of the planes status are bad or grown bad block, the virtual block
429 * is marked bad. If not bad, the first plane state acts as the block state.
430 */
431int nvm_bb_tbl_fold(struct nvm_dev *dev, u8 *blks, int nr_blks)
432{
433 int blk, offset, pl, blktype;
434
435 if (nr_blks != dev->blks_per_lun * dev->plane_mode)
436 return -EINVAL;
437
438 for (blk = 0; blk < dev->blks_per_lun; blk++) {
439 offset = blk * dev->plane_mode;
440 blktype = blks[offset];
441
442 /* Bad blocks on any planes take precedence over other types */
443 for (pl = 0; pl < dev->plane_mode; pl++) {
444 if (blks[offset + pl] &
445 (NVM_BLK_T_BAD|NVM_BLK_T_GRWN_BAD)) {
446 blktype = blks[offset + pl];
447 break;
448 }
449 }
450
451 blks[blk] = blktype;
452 }
453
454 return dev->blks_per_lun;
455}
456EXPORT_SYMBOL(nvm_bb_tbl_fold);
457
423static int nvm_init_slc_tbl(struct nvm_dev *dev, struct nvm_id_group *grp) 458static int nvm_init_slc_tbl(struct nvm_dev *dev, struct nvm_id_group *grp)
424{ 459{
425 int i; 460 int i;
diff --git a/drivers/lightnvm/gennvm.c b/drivers/lightnvm/gennvm.c
index 72e124a3927d..6096077c5dde 100644
--- a/drivers/lightnvm/gennvm.c
+++ b/drivers/lightnvm/gennvm.c
@@ -129,18 +129,21 @@ static int gennvm_luns_init(struct nvm_dev *dev, struct gen_nvm *gn)
129 return 0; 129 return 0;
130} 130}
131 131
132static int gennvm_block_bb(struct ppa_addr ppa, int nr_blocks, u8 *blks, 132static int gennvm_block_bb(struct nvm_dev *dev, struct ppa_addr ppa,
133 void *private) 133 u8 *blks, int nr_blks, void *private)
134{ 134{
135 struct gen_nvm *gn = private; 135 struct gen_nvm *gn = private;
136 struct nvm_dev *dev = gn->dev;
137 struct gen_lun *lun; 136 struct gen_lun *lun;
138 struct nvm_block *blk; 137 struct nvm_block *blk;
139 int i; 138 int i;
140 139
140 nr_blks = nvm_bb_tbl_fold(dev, blks, nr_blks);
141 if (nr_blks < 0)
142 return nr_blks;
143
141 lun = &gn->luns[(dev->luns_per_chnl * ppa.g.ch) + ppa.g.lun]; 144 lun = &gn->luns[(dev->luns_per_chnl * ppa.g.ch) + ppa.g.lun];
142 145
143 for (i = 0; i < nr_blocks; i++) { 146 for (i = 0; i < nr_blks; i++) {
144 if (blks[i] == 0) 147 if (blks[i] == 0)
145 continue; 148 continue;
146 149
@@ -250,8 +253,7 @@ static int gennvm_blocks_init(struct nvm_dev *dev, struct gen_nvm *gn)
250 ppa = generic_to_dev_addr(dev, ppa); 253 ppa = generic_to_dev_addr(dev, ppa);
251 254
252 ret = dev->ops->get_bb_tbl(dev, ppa, 255 ret = dev->ops->get_bb_tbl(dev, ppa,
253 dev->blks_per_lun, 256 gennvm_block_bb, gn);
254 gennvm_block_bb, gn);
255 if (ret) 257 if (ret)
256 pr_err("gennvm: could not read BB table\n"); 258 pr_err("gennvm: could not read BB table\n");
257 } 259 }
diff --git a/drivers/lightnvm/sysblk.c b/drivers/lightnvm/sysblk.c
index b6971f8ae3c0..7fce58833a07 100644
--- a/drivers/lightnvm/sysblk.c
+++ b/drivers/lightnvm/sysblk.c
@@ -93,12 +93,16 @@ void nvm_setup_sysblk_scan(struct nvm_dev *dev, struct sysblk_scan *s,
93 s->nr_rows = nvm_setup_sysblks(dev, sysblk_ppas); 93 s->nr_rows = nvm_setup_sysblks(dev, sysblk_ppas);
94} 94}
95 95
96static int sysblk_get_host_blks(struct ppa_addr ppa, int nr_blks, u8 *blks, 96static int sysblk_get_host_blks(struct nvm_dev *dev, struct ppa_addr ppa,
97 void *private) 97 u8 *blks, int nr_blks, void *private)
98{ 98{
99 struct sysblk_scan *s = private; 99 struct sysblk_scan *s = private;
100 int i, nr_sysblk = 0; 100 int i, nr_sysblk = 0;
101 101
102 nr_blks = nvm_bb_tbl_fold(dev, blks, nr_blks);
103 if (nr_blks < 0)
104 return nr_blks;
105
102 for (i = 0; i < nr_blks; i++) { 106 for (i = 0; i < nr_blks; i++) {
103 if (blks[i] != NVM_BLK_T_HOST) 107 if (blks[i] != NVM_BLK_T_HOST)
104 continue; 108 continue;
@@ -130,7 +134,7 @@ static int nvm_get_all_sysblks(struct nvm_dev *dev, struct sysblk_scan *s,
130 dppa = generic_to_dev_addr(dev, ppas[i]); 134 dppa = generic_to_dev_addr(dev, ppas[i]);
131 s->row = i; 135 s->row = i;
132 136
133 ret = dev->ops->get_bb_tbl(dev, dppa, dev->blks_per_lun, fn, s); 137 ret = dev->ops->get_bb_tbl(dev, dppa, fn, s);
134 if (ret) { 138 if (ret) {
135 pr_err("nvm: failed bb tbl for ppa (%u %u)\n", 139 pr_err("nvm: failed bb tbl for ppa (%u %u)\n",
136 ppas[i].g.ch, 140 ppas[i].g.ch,
@@ -235,13 +239,17 @@ static int nvm_set_bb_tbl(struct nvm_dev *dev, struct sysblk_scan *s, int type)
235 return 0; 239 return 0;
236} 240}
237 241
238static int sysblk_get_free_blks(struct ppa_addr ppa, int nr_blks, u8 *blks, 242static int sysblk_get_free_blks(struct nvm_dev *dev, struct ppa_addr ppa,
239 void *private) 243 u8 *blks, int nr_blks, void *private)
240{ 244{
241 struct sysblk_scan *s = private; 245 struct sysblk_scan *s = private;
242 struct ppa_addr *sppa; 246 struct ppa_addr *sppa;
243 int i, blkid = 0; 247 int i, blkid = 0;
244 248
249 nr_blks = nvm_bb_tbl_fold(dev, blks, nr_blks);
250 if (nr_blks < 0)
251 return nr_blks;
252
245 for (i = 0; i < nr_blks; i++) { 253 for (i = 0; i < nr_blks; i++) {
246 if (blks[i] == NVM_BLK_T_HOST) 254 if (blks[i] == NVM_BLK_T_HOST)
247 return -EEXIST; 255 return -EEXIST;
@@ -578,13 +586,16 @@ static unsigned int factory_blk_offset(struct nvm_dev *dev, int ch, int lun)
578 BITS_PER_LONG; 586 BITS_PER_LONG;
579} 587}
580 588
581static int nvm_factory_blks(struct ppa_addr ppa, int nr_blks, u8 *blks, 589static int nvm_factory_blks(struct nvm_dev *dev, struct ppa_addr ppa,
582 void *private) 590 u8 *blks, int nr_blks, void *private)
583{ 591{
584 struct factory_blks *f = private; 592 struct factory_blks *f = private;
585 struct nvm_dev *dev = f->dev;
586 int i, lunoff; 593 int i, lunoff;
587 594
595 nr_blks = nvm_bb_tbl_fold(dev, blks, nr_blks);
596 if (nr_blks < 0)
597 return nr_blks;
598
588 lunoff = factory_blk_offset(dev, ppa.g.ch, ppa.g.lun); 599 lunoff = factory_blk_offset(dev, ppa.g.ch, ppa.g.lun);
589 600
590 /* non-set bits correspond to the block must be erased */ 601 /* non-set bits correspond to the block must be erased */
@@ -661,7 +672,7 @@ static int nvm_fact_get_bb_tbl(struct nvm_dev *dev, struct ppa_addr ppa,
661 672
662 dev_ppa = generic_to_dev_addr(dev, ppa); 673 dev_ppa = generic_to_dev_addr(dev, ppa);
663 674
664 ret = dev->ops->get_bb_tbl(dev, dev_ppa, dev->blks_per_lun, fn, priv); 675 ret = dev->ops->get_bb_tbl(dev, dev_ppa, fn, priv);
665 if (ret) 676 if (ret)
666 pr_err("nvm: failed bb tbl for ch%u lun%u\n", 677 pr_err("nvm: failed bb tbl for ch%u lun%u\n",
667 ppa.g.ch, ppa.g.blk); 678 ppa.g.ch, ppa.g.blk);
diff --git a/drivers/nvme/host/lightnvm.c b/drivers/nvme/host/lightnvm.c
index 9461dd639acd..d289980d2bc8 100644
--- a/drivers/nvme/host/lightnvm.c
+++ b/drivers/nvme/host/lightnvm.c
@@ -387,41 +387,16 @@ out:
387 return ret; 387 return ret;
388} 388}
389 389
390static void nvme_nvm_bb_tbl_fold(struct nvm_dev *nvmdev,
391 int nr_dst_blks, u8 *dst_blks,
392 int nr_src_blks, u8 *src_blks)
393{
394 int blk, offset, pl, blktype;
395
396 for (blk = 0; blk < nr_dst_blks; blk++) {
397 offset = blk * nvmdev->plane_mode;
398 blktype = src_blks[offset];
399
400 /* Bad blocks on any planes take precedence over other types */
401 for (pl = 0; pl < nvmdev->plane_mode; pl++) {
402 if (src_blks[offset + pl] &
403 (NVM_BLK_T_BAD|NVM_BLK_T_GRWN_BAD)) {
404 blktype = src_blks[offset + pl];
405 break;
406 }
407 }
408
409 dst_blks[blk] = blktype;
410 }
411}
412
413static int nvme_nvm_get_bb_tbl(struct nvm_dev *nvmdev, struct ppa_addr ppa, 390static int nvme_nvm_get_bb_tbl(struct nvm_dev *nvmdev, struct ppa_addr ppa,
414 int nr_dst_blks, nvm_bb_update_fn *update_bbtbl, 391 nvm_bb_update_fn *update_bbtbl, void *priv)
415 void *priv)
416{ 392{
417 struct request_queue *q = nvmdev->q; 393 struct request_queue *q = nvmdev->q;
418 struct nvme_ns *ns = q->queuedata; 394 struct nvme_ns *ns = q->queuedata;
419 struct nvme_ctrl *ctrl = ns->ctrl; 395 struct nvme_ctrl *ctrl = ns->ctrl;
420 struct nvme_nvm_command c = {}; 396 struct nvme_nvm_command c = {};
421 struct nvme_nvm_bb_tbl *bb_tbl; 397 struct nvme_nvm_bb_tbl *bb_tbl;
422 u8 *dst_blks = NULL; 398 int nr_blks = nvmdev->blks_per_lun * nvmdev->plane_mode;
423 int nr_src_blks = nr_dst_blks * nvmdev->plane_mode; 399 int tblsz = sizeof(struct nvme_nvm_bb_tbl) + nr_blks;
424 int tblsz = sizeof(struct nvme_nvm_bb_tbl) + nr_src_blks;
425 int ret = 0; 400 int ret = 0;
426 401
427 c.get_bb.opcode = nvme_nvm_admin_get_bb_tbl; 402 c.get_bb.opcode = nvme_nvm_admin_get_bb_tbl;
@@ -432,12 +407,6 @@ static int nvme_nvm_get_bb_tbl(struct nvm_dev *nvmdev, struct ppa_addr ppa,
432 if (!bb_tbl) 407 if (!bb_tbl)
433 return -ENOMEM; 408 return -ENOMEM;
434 409
435 dst_blks = kzalloc(nr_dst_blks, GFP_KERNEL);
436 if (!dst_blks) {
437 ret = -ENOMEM;
438 goto out;
439 }
440
441 ret = nvme_submit_sync_cmd(ctrl->admin_q, (struct nvme_command *)&c, 410 ret = nvme_submit_sync_cmd(ctrl->admin_q, (struct nvme_command *)&c,
442 bb_tbl, tblsz); 411 bb_tbl, tblsz);
443 if (ret) { 412 if (ret) {
@@ -459,21 +428,17 @@ static int nvme_nvm_get_bb_tbl(struct nvm_dev *nvmdev, struct ppa_addr ppa,
459 goto out; 428 goto out;
460 } 429 }
461 430
462 if (le32_to_cpu(bb_tbl->tblks) != nr_src_blks) { 431 if (le32_to_cpu(bb_tbl->tblks) != nr_blks) {
463 ret = -EINVAL; 432 ret = -EINVAL;
464 dev_err(ctrl->dev, "bbt unsuspected blocks returned (%u!=%u)", 433 dev_err(ctrl->dev, "bbt unsuspected blocks returned (%u!=%u)",
465 le32_to_cpu(bb_tbl->tblks), nr_src_blks); 434 le32_to_cpu(bb_tbl->tblks), nr_blks);
466 goto out; 435 goto out;
467 } 436 }
468 437
469 nvme_nvm_bb_tbl_fold(nvmdev, nr_dst_blks, dst_blks,
470 nr_src_blks, bb_tbl->blk);
471
472 ppa = dev_to_generic_addr(nvmdev, ppa); 438 ppa = dev_to_generic_addr(nvmdev, ppa);
473 ret = update_bbtbl(ppa, nr_dst_blks, dst_blks, priv); 439 ret = update_bbtbl(nvmdev, ppa, bb_tbl->blk, nr_blks, priv);
474 440
475out: 441out:
476 kfree(dst_blks);
477 kfree(bb_tbl); 442 kfree(bb_tbl);
478 return ret; 443 return ret;
479} 444}
diff --git a/include/linux/lightnvm.h b/include/linux/lightnvm.h
index f7c607f96858..dacaa2850428 100644
--- a/include/linux/lightnvm.h
+++ b/include/linux/lightnvm.h
@@ -41,11 +41,12 @@ struct nvm_id;
41struct nvm_dev; 41struct nvm_dev;
42 42
43typedef int (nvm_l2p_update_fn)(u64, u32, __le64 *, void *); 43typedef int (nvm_l2p_update_fn)(u64, u32, __le64 *, void *);
44typedef int (nvm_bb_update_fn)(struct ppa_addr, int, u8 *, void *); 44typedef int (nvm_bb_update_fn)(struct nvm_dev *, struct ppa_addr, u8 *, int,
45 void *);
45typedef int (nvm_id_fn)(struct nvm_dev *, struct nvm_id *); 46typedef int (nvm_id_fn)(struct nvm_dev *, struct nvm_id *);
46typedef int (nvm_get_l2p_tbl_fn)(struct nvm_dev *, u64, u32, 47typedef int (nvm_get_l2p_tbl_fn)(struct nvm_dev *, u64, u32,
47 nvm_l2p_update_fn *, void *); 48 nvm_l2p_update_fn *, void *);
48typedef int (nvm_op_bb_tbl_fn)(struct nvm_dev *, struct ppa_addr, int, 49typedef int (nvm_op_bb_tbl_fn)(struct nvm_dev *, struct ppa_addr,
49 nvm_bb_update_fn *, void *); 50 nvm_bb_update_fn *, void *);
50typedef int (nvm_op_set_bb_fn)(struct nvm_dev *, struct nvm_rq *, int); 51typedef int (nvm_op_set_bb_fn)(struct nvm_dev *, struct nvm_rq *, int);
51typedef int (nvm_submit_io_fn)(struct nvm_dev *, struct nvm_rq *); 52typedef int (nvm_submit_io_fn)(struct nvm_dev *, struct nvm_rq *);
@@ -538,6 +539,7 @@ extern int nvm_submit_ppa(struct nvm_dev *, struct ppa_addr *, int, int, int,
538 void *, int); 539 void *, int);
539extern int nvm_submit_ppa_list(struct nvm_dev *, struct ppa_addr *, int, int, 540extern int nvm_submit_ppa_list(struct nvm_dev *, struct ppa_addr *, int, int,
540 int, void *, int); 541 int, void *, int);
542extern int nvm_bb_tbl_fold(struct nvm_dev *, u8 *, int);
541 543
542/* sysblk.c */ 544/* sysblk.c */
543#define NVM_SYSBLK_MAGIC 0x4E564D53 /* "NVMS" */ 545#define NVM_SYSBLK_MAGIC 0x4E564D53 /* "NVMS" */