flash_write_protection.c 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. #include <drivers/flash.h>
  2. #include <drivers/spi.h>
  3. #include <logging/log.h>
  4. #include <soc.h>
  5. #include "spi_flash.h"
  6. #define NOR_STATUS1_MASK (0x1f<<2) /*bp4-bp0 status1 bit6-bit2*/
  7. #define NOR_STATUS2_MASK (0x1<<6) /*cmp status2 bit6*/
  8. /**********ATS3085 NOR GD25Q16E 2MB ****************/
  9. #define ATS3085_NOR_CHIPID 0x1560c8 /*GD25Q16E*/
  10. #define ATS3085_CMP0_VAL 0
  11. #define ATS3085_CMP0_PROTECT_16KB 0x1b /*0-16KB*/
  12. #define ATS3085_CMP0_PROTECT_64KB 0x9 /*0-64KB*/
  13. #define ATS3085_CMP0_PROTECT_128KB 0xa /*0-128KB*/
  14. #define ATS3085_CMP0_PROTECT_256KB 0xb /*0-256KB*/
  15. #define ATS3085_CMP0_PROTECT_512KB 0xc /*0-512KB*/
  16. #define ATS3085_CMP0_PROTECT_1MB 0xd /*0-1MB*/
  17. #define ATS3085_CMP0_PROTECT_2MB 0xe /*0-2MB*/
  18. #define ATS3085_CMP1_VAL 1
  19. #define ATS3085_CMP1_PROTECT_1984KB 0x1 /*0- (2MB-64kb)*/
  20. #define ATS3085_CMP1_PROTECT_1920KB 0x2 /*0- (2MB-128kb)*/
  21. #define ATS3085_CMP1_PROTECT_1792KB 0x3 /*0- (2MB-256kb)*/
  22. #define ATS3085_CMP1_PROTECT_1536KB 0x4 /*0- (2MB-512kb)*/
  23. #define ATS3085_NOR_BP_STATUS1 (ATS3085_CMP0_PROTECT_16KB<<2)
  24. #define ATS3085_NOR_CMP_STATUS2 (ATS3085_CMP0_VAL<<6)
  25. /**********GD25Q256E(3085c) 32MB****************/
  26. #define GD25Q256E_NOR_CHIPID 0x1940c8 /*GD25Q256E*/
  27. #define GD25Q256E_PROTECT_64KB 0x11 /*0-64KB*/
  28. #define GD25Q256E_PROTECT_128KB 0x12 /*0-128KB*/
  29. #define GD25Q256E_PROTECT_256KB 0x13 /*0-256KB*/
  30. #define GD25Q256E_PROTECT_512KB 0x14 /*0-512KB*/
  31. #define GD25Q256E_PROTECT_1MB 0x15 /*0-1MB*/
  32. #define GD25Q256E_PROTECT_2MB 0x16 /*0-2MB*/
  33. #define GD25Q256E_PROTECT_4MB 0x17 /*0-4MB*/
  34. #define GD25Q256E_PROTECT_8MB 0x18 /*0-8MB*/
  35. #define GD25Q256E_PROTECT_16MB 0x19 /*0-16MB*/
  36. #define GD25Q256E_PROTECT_32MB 0x1A /*0-32MB*/
  37. #define GD25Q256E_STATUS1_MASK (NOR_STATUS1_MASK|(1<<7))
  38. #define GD25Q256E_NOR_BP_STATUS1 (GD25Q256E_PROTECT_64KB<<2)
  39. #define GD25Q256E_NOR_STATUS2 (0<<6)
  40. /**********GD25LF32E(3089c) 4MB****************/
  41. #define GD25LF32E_NOR_CHIPID 0x1663c8 /*GD25LF32E*/
  42. #define GD25LF32E_CMP0_VAL 0
  43. #define GD25LF32E_CMP0_PROTECT_4KB 0x19 /*0-4KB*/
  44. #define GD25LF32E_CMP0_PROTECT_8KB 0x1a /*0-8KB*/
  45. #define GD25LF32E_CMP0_PROTECT_16KB 0x1b /*0-16KB*/
  46. #define GD25LF32E_CMP0_PROTECT_32KB 0x1c /*0-32KB*/
  47. #define GD25LF32E_CMP0_PROTECT_64KB 0x9 /*0-64KB*/
  48. #define GD25LF32E_CMP0_PROTECT_128KB 0xa /*0-128KB*/
  49. #define GD25LF32E_CMP0_PROTECT_256KB 0xb /*0-256KB*/
  50. #define GD25LF32E_CMP0_PROTECT_512KB 0xc /*0-512KB*/
  51. #define GD25LF32E_CMP0_PROTECT_1MB 0xd /*0-1MB*/
  52. #define GD25LF32E_CMP0_PROTECT_2MB 0xe /*0-2MB*/
  53. #define GD25LF32E_CMP0_PROTECT_4MB 0xf /*0-4MB*/
  54. #define GD25LF32E_NOR_BP_STATUS1 (GD25LF32E_CMP0_PROTECT_16KB<<2)
  55. #define GD25LF32E_NOR_STATUS2 (GD25LF32E_CMP0_VAL<<6)
  56. /**********GD25LE64E(3089) 8MB****************/
  57. #define GD25LE64E_NOR_CHIPID 0x1760c8 /*GD25LF32E*/
  58. #define GD25LE64E_CMP0_VAL 0
  59. #define GD25LE64E_CMP0_PROTECT_4KB 0x19 /*0-4KB*/
  60. #define GD25LE64E_CMP0_PROTECT_8KB 0x1a /*0-8KB*/
  61. #define GD25LE64E_CMP0_PROTECT_16KB 0x1b /*0-16KB*/
  62. #define GD25LE64E_CMP0_PROTECT_32KB 0x1c /*0-32KB*/
  63. #define GD25LE64E_CMP0_PROTECT_128KB 0x9 /*0-128KB*/
  64. #define GD25LE64E_CMP0_PROTECT_256KB 0xa /*0-256KB*/
  65. #define GD25LE64E_CMP0_PROTECT_512KB 0xb /*0-512KB*/
  66. #define GD25LE64E_CMP0_PROTECT_1MB 0xc /*0-1MB*/
  67. #define GD25LE64E_CMP0_PROTECT_2MB 0xd /*0-2MB*/
  68. #define GD25LE64E_CMP0_PROTECT_4MB 0xe /*0-4MB*/
  69. #define GD25LE64E_CMP0_PROTECT_8MB 0xf /*0-8MB*/
  70. #define GD25LE64E_NOR_BP_STATUS1 (GD25LE64E_CMP0_PROTECT_16KB<<2)
  71. #define GD25LE64E_NOR_STATUS2 (GD25LE64E_CMP0_VAL<<6)
  72. struct nor_wp_info {
  73. unsigned int chipid;
  74. unsigned char wp_status1_val;
  75. unsigned char wp_status1_mask;
  76. unsigned char wp_status2_val;
  77. unsigned char wp_status2_mask;
  78. };
  79. const struct nor_wp_info g_nor_wp_info[] = {
  80. {ATS3085_NOR_CHIPID, ATS3085_NOR_BP_STATUS1, NOR_STATUS1_MASK, ATS3085_NOR_CMP_STATUS2, NOR_STATUS2_MASK},
  81. {GD25Q256E_NOR_CHIPID, GD25Q256E_NOR_BP_STATUS1, GD25Q256E_STATUS1_MASK, GD25Q256E_NOR_STATUS2, NOR_STATUS2_MASK},
  82. {GD25LF32E_NOR_CHIPID, GD25LF32E_NOR_BP_STATUS1, NOR_STATUS1_MASK, GD25LF32E_NOR_STATUS2, NOR_STATUS2_MASK},
  83. {GD25LE64E_NOR_CHIPID, GD25LE64E_NOR_BP_STATUS1, NOR_STATUS1_MASK, GD25LE64E_NOR_STATUS2, NOR_STATUS2_MASK},
  84. };
  85. static __ramfunc void xspi_nor_write_status(struct spinor_info *sni, u8_t status1, u8_t status2)
  86. {
  87. u8_t status[2];
  88. unsigned int flag;
  89. status[0] = status1;
  90. status[1] = status2;
  91. flag = sni->spi.flag;
  92. sni->spi.flag &= ~SPI_FLAG_NO_IRQ_LOCK; //wait ready
  93. p_spinor_api->write_status(sni, XSPI_NOR_CMD_WRITE_STATUS2, &status2, 1);
  94. p_spinor_api->write_status(sni, XSPI_NOR_CMD_WRITE_STATUS, &status1, 1);
  95. p_spinor_api->write_status(sni, XSPI_NOR_CMD_WRITE_STATUS, status, 2);
  96. sni->spi.flag = flag;
  97. }
  98. static __ramfunc void xspi_nor_read_status(struct spinor_info *sni, u8_t *status1, u8_t *status2)
  99. {
  100. *status1 = p_spinor_api->read_status(sni, XSPI_NOR_CMD_READ_STATUS);
  101. *status2 = p_spinor_api->read_status(sni, XSPI_NOR_CMD_READ_STATUS2);
  102. }
  103. static __ramfunc void xspi_nor_protect_handle(struct spinor_info *sni, const struct nor_wp_info *wp, bool benable)
  104. {
  105. u32_t flags;
  106. u8_t status1, status2;
  107. u8_t val1;
  108. flags = irq_lock();
  109. xspi_nor_read_status(sni, &status1, &status2);
  110. //#ifdef CONFIG_XSPI_NOR_ACTS_DUMP_INFO
  111. printk("status1-2=0x%x,0x%x\n", status1, status2);
  112. //#endif
  113. val1 = wp->wp_status1_mask & status1;
  114. if(benable){
  115. if(val1 != wp->wp_status1_val){// enable protect
  116. status1 = ((~wp->wp_status1_mask) & status1) | wp->wp_status1_val;
  117. status2 = ((~wp->wp_status2_mask) & status2) | wp->wp_status2_val;
  118. printk("enable status1-2=0x%x,0x%x\n", status1, status2);
  119. xspi_nor_write_status(sni, status1, status2);
  120. }
  121. }else{
  122. if(val1 != 0){// diable protect
  123. status1 = (~wp->wp_status1_mask) & status1;
  124. status2 = (~wp->wp_status2_mask) & status2;
  125. printk("disable status1-2=0x%x,0x%x\n", status1, status2);
  126. xspi_nor_write_status(sni, status1, status2);
  127. }
  128. }
  129. //#ifdef CONFIG_XSPI_NOR_ACTS_DUMP_INFO
  130. xspi_nor_read_status(sni, &status1, &status2);
  131. printk("read again status1-2=0x%x,0x%x\n", status1, status2);
  132. //#endif
  133. irq_unlock(flags);
  134. }
  135. int nor_write_protection(const struct device *dev, bool enable)
  136. {
  137. struct spinor_info *sni = (struct spinor_info *)(dev)->data;
  138. const struct nor_wp_info *wp;
  139. int i;
  140. for(i = 0; i < ARRAY_SIZE(g_nor_wp_info); i++){
  141. wp = &g_nor_wp_info[i];
  142. if(wp->chipid == sni->chipid)
  143. xspi_nor_protect_handle(sni, wp, enable);
  144. }
  145. return 0;
  146. }