bbt.c 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Copyright (c) 2017 Free Electrons
  4. *
  5. * Authors:
  6. * Boris Brezillon <boris.brezillon@free-electrons.com>
  7. * Peter Pan <peterpandong@micron.com>
  8. */
  9. #define pr_fmt(fmt) "nand-bbt: " fmt
  10. #include <linux/mtd/nand.h>
  11. #include <linux/slab.h>
  12. /**
  13. * nanddev_bbt_init() - Initialize the BBT (Bad Block Table)
  14. * @nand: NAND device
  15. *
  16. * Initialize the in-memory BBT.
  17. *
  18. * Return: 0 in case of success, a negative error code otherwise.
  19. */
  20. int nanddev_bbt_init(struct nand_device *nand)
  21. {
  22. unsigned int bits_per_block = fls(NAND_BBT_BLOCK_NUM_STATUS);
  23. unsigned int nblocks = nanddev_neraseblocks(nand);
  24. unsigned int nwords = DIV_ROUND_UP(nblocks * bits_per_block,
  25. BITS_PER_LONG);
  26. nand->bbt.cache = kcalloc(nwords, sizeof(*nand->bbt.cache),
  27. GFP_KERNEL);
  28. if (!nand->bbt.cache)
  29. return -ENOMEM;
  30. return 0;
  31. }
  32. EXPORT_SYMBOL_GPL(nanddev_bbt_init);
  33. /**
  34. * nanddev_bbt_cleanup() - Cleanup the BBT (Bad Block Table)
  35. * @nand: NAND device
  36. *
  37. * Undoes what has been done in nanddev_bbt_init()
  38. */
  39. void nanddev_bbt_cleanup(struct nand_device *nand)
  40. {
  41. kfree(nand->bbt.cache);
  42. }
  43. EXPORT_SYMBOL_GPL(nanddev_bbt_cleanup);
  44. /**
  45. * nanddev_bbt_update() - Update a BBT
  46. * @nand: nand device
  47. *
  48. * Update the BBT. Currently a NOP function since on-flash bbt is not yet
  49. * supported.
  50. *
  51. * Return: 0 in case of success, a negative error code otherwise.
  52. */
  53. int nanddev_bbt_update(struct nand_device *nand)
  54. {
  55. return 0;
  56. }
  57. EXPORT_SYMBOL_GPL(nanddev_bbt_update);
  58. /**
  59. * nanddev_bbt_get_block_status() - Return the status of an eraseblock
  60. * @nand: nand device
  61. * @entry: the BBT entry
  62. *
  63. * Return: a positive number nand_bbt_block_status status or -%ERANGE if @entry
  64. * is bigger than the BBT size.
  65. */
  66. int nanddev_bbt_get_block_status(const struct nand_device *nand,
  67. unsigned int entry)
  68. {
  69. unsigned int bits_per_block = fls(NAND_BBT_BLOCK_NUM_STATUS);
  70. unsigned long *pos = nand->bbt.cache +
  71. ((entry * bits_per_block) / BITS_PER_LONG);
  72. unsigned int offs = (entry * bits_per_block) % BITS_PER_LONG;
  73. unsigned long status;
  74. if (entry >= nanddev_neraseblocks(nand))
  75. return -ERANGE;
  76. status = pos[0] >> offs;
  77. if (bits_per_block + offs > BITS_PER_LONG)
  78. status |= pos[1] << (BITS_PER_LONG - offs);
  79. return status & GENMASK(bits_per_block - 1, 0);
  80. }
  81. EXPORT_SYMBOL_GPL(nanddev_bbt_get_block_status);
  82. /**
  83. * nanddev_bbt_set_block_status() - Update the status of an eraseblock in the
  84. * in-memory BBT
  85. * @nand: nand device
  86. * @entry: the BBT entry to update
  87. * @status: the new status
  88. *
  89. * Update an entry of the in-memory BBT. If you want to push the updated BBT
  90. * the NAND you should call nanddev_bbt_update().
  91. *
  92. * Return: 0 in case of success or -%ERANGE if @entry is bigger than the BBT
  93. * size.
  94. */
  95. int nanddev_bbt_set_block_status(struct nand_device *nand, unsigned int entry,
  96. enum nand_bbt_block_status status)
  97. {
  98. unsigned int bits_per_block = fls(NAND_BBT_BLOCK_NUM_STATUS);
  99. unsigned long *pos = nand->bbt.cache +
  100. ((entry * bits_per_block) / BITS_PER_LONG);
  101. unsigned int offs = (entry * bits_per_block) % BITS_PER_LONG;
  102. unsigned long val = status & GENMASK(bits_per_block - 1, 0);
  103. if (entry >= nanddev_neraseblocks(nand))
  104. return -ERANGE;
  105. pos[0] &= ~GENMASK(offs + bits_per_block - 1, offs);
  106. pos[0] |= val << offs;
  107. if (bits_per_block + offs > BITS_PER_LONG) {
  108. unsigned int rbits = bits_per_block + offs - BITS_PER_LONG;
  109. pos[1] &= ~GENMASK(rbits - 1, 0);
  110. pos[1] |= val >> rbits;
  111. }
  112. return 0;
  113. }
  114. EXPORT_SYMBOL_GPL(nanddev_bbt_set_block_status);