diff options
author | Matias Bjørling <m@bjorling.me> | 2016-05-06 14:02:58 -0400 |
---|---|---|
committer | Jens Axboe <axboe@fb.com> | 2016-05-06 14:51:10 -0400 |
commit | 22e8c9766a669d49cf3749d397082a5cd93374a9 (patch) | |
tree | c24facba8321c5359bf6f111f2ca892cf894b62a | |
parent | 4891d120b9cd419f4350b11e1231083745dcdc8b (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.c | 35 | ||||
-rw-r--r-- | drivers/lightnvm/gennvm.c | 14 | ||||
-rw-r--r-- | drivers/lightnvm/sysblk.c | 29 | ||||
-rw-r--r-- | drivers/nvme/host/lightnvm.c | 47 | ||||
-rw-r--r-- | include/linux/lightnvm.h | 6 |
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 | } |
421 | EXPORT_SYMBOL(nvm_submit_ppa); | 421 | EXPORT_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 | */ | ||
431 | int 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 | } | ||
456 | EXPORT_SYMBOL(nvm_bb_tbl_fold); | ||
457 | |||
423 | static int nvm_init_slc_tbl(struct nvm_dev *dev, struct nvm_id_group *grp) | 458 | static 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 | ||
132 | static int gennvm_block_bb(struct ppa_addr ppa, int nr_blocks, u8 *blks, | 132 | static 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 | ||
96 | static int sysblk_get_host_blks(struct ppa_addr ppa, int nr_blks, u8 *blks, | 96 | static 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 | ||
238 | static int sysblk_get_free_blks(struct ppa_addr ppa, int nr_blks, u8 *blks, | 242 | static 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 | ||
581 | static int nvm_factory_blks(struct ppa_addr ppa, int nr_blks, u8 *blks, | 589 | static 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 | ||
390 | static 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 | |||
413 | static int nvme_nvm_get_bb_tbl(struct nvm_dev *nvmdev, struct ppa_addr ppa, | 390 | static 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 | ||
475 | out: | 441 | out: |
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; | |||
41 | struct nvm_dev; | 41 | struct nvm_dev; |
42 | 42 | ||
43 | typedef int (nvm_l2p_update_fn)(u64, u32, __le64 *, void *); | 43 | typedef int (nvm_l2p_update_fn)(u64, u32, __le64 *, void *); |
44 | typedef int (nvm_bb_update_fn)(struct ppa_addr, int, u8 *, void *); | 44 | typedef int (nvm_bb_update_fn)(struct nvm_dev *, struct ppa_addr, u8 *, int, |
45 | void *); | ||
45 | typedef int (nvm_id_fn)(struct nvm_dev *, struct nvm_id *); | 46 | typedef int (nvm_id_fn)(struct nvm_dev *, struct nvm_id *); |
46 | typedef int (nvm_get_l2p_tbl_fn)(struct nvm_dev *, u64, u32, | 47 | typedef int (nvm_get_l2p_tbl_fn)(struct nvm_dev *, u64, u32, |
47 | nvm_l2p_update_fn *, void *); | 48 | nvm_l2p_update_fn *, void *); |
48 | typedef int (nvm_op_bb_tbl_fn)(struct nvm_dev *, struct ppa_addr, int, | 49 | typedef int (nvm_op_bb_tbl_fn)(struct nvm_dev *, struct ppa_addr, |
49 | nvm_bb_update_fn *, void *); | 50 | nvm_bb_update_fn *, void *); |
50 | typedef int (nvm_op_set_bb_fn)(struct nvm_dev *, struct nvm_rq *, int); | 51 | typedef int (nvm_op_set_bb_fn)(struct nvm_dev *, struct nvm_rq *, int); |
51 | typedef int (nvm_submit_io_fn)(struct nvm_dev *, struct nvm_rq *); | 52 | typedef 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); |
539 | extern int nvm_submit_ppa_list(struct nvm_dev *, struct ppa_addr *, int, int, | 540 | extern int nvm_submit_ppa_list(struct nvm_dev *, struct ppa_addr *, int, int, |
540 | int, void *, int); | 541 | int, void *, int); |
542 | extern 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" */ |