sf_mtd.c 2.2 KB

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