sf_mtd.c 2.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Copyright (C) 2012-2014 Daniel Schwierzeck, daniel.schwierzeck@gmail.com
  4. */
  5. #include <common.h>
  6. #include <malloc.h>
  7. #include <linux/errno.h>
  8. #include <linux/mtd/mtd.h>
  9. #include <spi_flash.h>
  10. static struct mtd_info sf_mtd_info;
  11. static char sf_mtd_name[8];
  12. static int spi_flash_mtd_erase(struct mtd_info *mtd, struct erase_info *instr)
  13. {
  14. struct spi_flash *flash = mtd->priv;
  15. int err;
  16. instr->state = MTD_ERASING;
  17. err = spi_flash_erase(flash, instr->addr, instr->len);
  18. if (err) {
  19. instr->state = MTD_ERASE_FAILED;
  20. instr->fail_addr = MTD_FAIL_ADDR_UNKNOWN;
  21. return -EIO;
  22. }
  23. instr->state = MTD_ERASE_DONE;
  24. mtd_erase_callback(instr);
  25. return 0;
  26. }
  27. static int spi_flash_mtd_read(struct mtd_info *mtd, loff_t from, size_t len,
  28. size_t *retlen, u_char *buf)
  29. {
  30. struct spi_flash *flash = mtd->priv;
  31. int err;
  32. err = spi_flash_read(flash, from, len, buf);
  33. if (!err)
  34. *retlen = len;
  35. return err;
  36. }
  37. static int spi_flash_mtd_write(struct mtd_info *mtd, loff_t to, size_t len,
  38. size_t *retlen, const u_char *buf)
  39. {
  40. struct spi_flash *flash = mtd->priv;
  41. int err;
  42. err = spi_flash_write(flash, to, len, buf);
  43. if (!err)
  44. *retlen = len;
  45. return err;
  46. }
  47. static void spi_flash_mtd_sync(struct mtd_info *mtd)
  48. {
  49. }
  50. static int spi_flash_mtd_number(void)
  51. {
  52. #ifdef CONFIG_SYS_MAX_FLASH_BANKS
  53. return CONFIG_SYS_MAX_FLASH_BANKS;
  54. #else
  55. return 0;
  56. #endif
  57. }
  58. int spi_flash_mtd_register(struct spi_flash *flash)
  59. {
  60. memset(&sf_mtd_info, 0, sizeof(sf_mtd_info));
  61. sprintf(sf_mtd_name, "nor%d", spi_flash_mtd_number());
  62. sf_mtd_info.name = sf_mtd_name;
  63. sf_mtd_info.type = MTD_NORFLASH;
  64. sf_mtd_info.flags = MTD_CAP_NORFLASH;
  65. sf_mtd_info.writesize = 1;
  66. sf_mtd_info.writebufsize = flash->page_size;
  67. sf_mtd_info._erase = spi_flash_mtd_erase;
  68. sf_mtd_info._read = spi_flash_mtd_read;
  69. sf_mtd_info._write = spi_flash_mtd_write;
  70. sf_mtd_info._sync = spi_flash_mtd_sync;
  71. sf_mtd_info.size = flash->size;
  72. sf_mtd_info.priv = flash;
  73. /* Only uniform flash devices for now */
  74. sf_mtd_info.numeraseregions = 0;
  75. sf_mtd_info.erasesize = flash->sector_size;
  76. return add_mtd_device(&sf_mtd_info);
  77. }
  78. void spi_flash_mtd_unregister(void)
  79. {
  80. del_mtd_device(&sf_mtd_info);
  81. }