lib_flashiap.c 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355
  1. /***************************************************************
  2. *Copyright (C), 2017, Shanghai Eastsoft Microelectronics Co., Ltd
  3. *文件名: lib_flashiap.c
  4. *作 者: Liut
  5. *版 本: V1.00
  6. *日 期: 2017/07/14
  7. *描 述: flash读写库函数
  8. *备 注: 适用于 ES8P508x芯片
  9. 本软件仅供学习和演示使用,对用户直接引用代码所带来的风险或后果不承担任何法律责任。
  10. ***************************************************************/
  11. #include "lib_flashiap.h"
  12. /***************************************************************
  13. 函数名:FlashIap_Close_WPROT
  14. 描 述:IAP关闭写保护
  15. 输入值:Page:0-63,每page对应2K字节,64 为INFO区
  16. 输出值:无
  17. 返回值:成功、失败
  18. ***************************************************************/
  19. ErrorStatus FlashIap_Close_WPROT(uint8_t Page)
  20. {
  21. if(Page > 64)
  22. return ERROR;
  23. if(Page == 64)
  24. {
  25. IAP->WPROT2.Word = 0x00000000;
  26. return SUCCESS;
  27. }
  28. if(Page < 32)
  29. {
  30. IAP->WPROT0.Word &=~ ((uint32_t)0x1 << Page);
  31. }
  32. else
  33. {
  34. Page -= 32;
  35. IAP->WPROT1.Word &=~ ((uint32_t)0x1 << Page);
  36. }
  37. return SUCCESS;
  38. }
  39. /***************************************************************
  40. 函数名:FlashIap_Open_WPROT
  41. 描 述:IAP开启写保护
  42. 输入值:Page:0-63,每page对应2K字节,,64 为INFO区
  43. 输出值:无
  44. 返回值:成功、失败
  45. ***************************************************************/
  46. ErrorStatus FlashIap_Open_WPROT(uint8_t Page)
  47. {
  48. if(Page > 64)
  49. return ERROR;
  50. if(Page == 64)
  51. {
  52. IAP->WPROT2.Word = 0x00000001;
  53. return SUCCESS;
  54. }
  55. if(Page < 32)
  56. {
  57. IAP->WPROT0.Word &=~ ((uint32_t)0x1 << Page);
  58. IAP->WPROT0.Word |= ((uint32_t)0x1 << Page);
  59. }
  60. else
  61. {
  62. Page -= 32;
  63. IAP->WPROT1.Word &=~ ((uint32_t)0x1 << Page);
  64. IAP->WPROT1.Word |= ((uint32_t)0x1 << Page);
  65. }
  66. return SUCCESS;
  67. }
  68. /***************************************************************
  69. 函数名:FlashIap_CloseAll_WPROT
  70. 描 述:IAP关闭所有写保护
  71. 输入值:Page:0-63,每page对应2K字节
  72. 输出值:无
  73. 返回值:成功、失败
  74. ***************************************************************/
  75. ErrorStatus FlashIap_CloseAll_WPROT(void)
  76. {
  77. IAP->WPROT0.Word = 0x00000000;
  78. IAP->WPROT1.Word = 0x00000000;
  79. IAP->WPROT2.Word = 0x00000000;
  80. return SUCCESS;
  81. }
  82. /***************************************************************
  83. 函数名:FlashIap_OpenAll_WPROT
  84. 描 述:IAP打开所有写保护
  85. 输入值:
  86. 输出值:无
  87. 返回值:成功、失败
  88. ***************************************************************/
  89. ErrorStatus FlashIap_OpenAll_WPROT(void)
  90. {
  91. IAP->WPROT0.Word = 0xFFFFFFFF;
  92. IAP->WPROT1.Word = 0xFFFFFFFF;
  93. IAP->WPROT2.Word = 0xFFFFFFFF;
  94. return SUCCESS;
  95. }
  96. /***************************************************************
  97. 函数名:FlashIap_Unlock
  98. 描 述:IAP解锁与访问请求
  99. 输入值:无
  100. 输出值:无
  101. 返回值:成功、失败
  102. ***************************************************************/
  103. ErrorStatus FlashIap_Unlock(void)
  104. {
  105. uint16_t Temp16;
  106. FlashIAP_RegUnLock(); //解锁 IAP
  107. FlashIAP_Enable(); //使能IAP
  108. FlashIAP_REQ(); //访问请求
  109. for(Temp16 = 0; Temp16 < 0xFFFF; Temp16++) //等待FLASH应答信号
  110. {
  111. if(IAP->CON.FLASH_ACK != 0)
  112. break;
  113. }
  114. if(Temp16 == 0xFFFF)
  115. return ERROR;
  116. else
  117. return SUCCESS;
  118. }
  119. /***************************************************************
  120. 函数名:FlashIap_WriteEnd
  121. 描 述:IAP写结束
  122. 输入值:无
  123. 输出值:无
  124. 返回值:成功、失败
  125. ***************************************************************/
  126. ErrorStatus FlashIap_WriteEnd(void)
  127. {
  128. uint32_t Temp32;
  129. FlashIAP_RegUnLock(); //IAP解锁
  130. IAP->CON.Word &= 0xFFFFFFEE; //IAP访问FLASH请求(结束)
  131. for(Temp32 = 0; Temp32 < 0xFFFF; Temp32++) //等待FLASH应答信号(结束)
  132. {
  133. if(IAP->CON.FLASH_ACK == 0)
  134. break;
  135. }
  136. if(Temp32 == 0xFFFF)
  137. return ERROR;
  138. else
  139. return SUCCESS;
  140. }
  141. /***************************************************************
  142. 函数名:FlashIap_ErasePage
  143. 描 述:IAP页擦除
  144. 输入值:Page_Addr:页地址
  145. 输出值:无
  146. 返回值:成功、失败
  147. ***************************************************************/
  148. ErrorStatus FlashIap_ErasePage(uint8_t Page_Addr)
  149. {
  150. uint16_t Temp16;
  151. uint32_t temp;
  152. temp = __get_PRIMASK(); //获取PRIMASK寄存器当前状态
  153. __disable_irq(); //屏蔽所有中断
  154. if(FlashIap_Unlock() == ERROR)
  155. {
  156. __set_PRIMASK(temp); //恢复PRIMASK寄存器状态
  157. return ERROR;
  158. }
  159. if(FlashIap_CloseAll_WPROT() == ERROR)
  160. {
  161. __set_PRIMASK(temp);
  162. return ERROR;
  163. }
  164. IAP->ADDR.IAPPA = Page_Addr; //输入页地址
  165. IAP->TRIG.TRIG = 0x00005EA1; //输入编程命令
  166. for(Temp16 = 0; Temp16 < 0xFFFF; Temp16++)
  167. { //判断IAP工作状态
  168. if((IAP->STA.Word & (uint32_t)0x01) == (uint32_t)0x00)
  169. break;
  170. }
  171. if(Temp16 == 0xFFFF)
  172. {
  173. __set_PRIMASK(temp); //恢复PRIMASK寄存器状态
  174. return ERROR;
  175. }
  176. for(Temp16 = 0; Temp16 < 0xFFFF; Temp16++)
  177. {
  178. if((IAP->STA.Word & (uint32_t)0x02) == (uint32_t)0x02) //判断IAP页擦除标志
  179. break;
  180. }
  181. if(Temp16 == 0xFFFF)
  182. {
  183. __set_PRIMASK(temp); //恢复PRIMASK寄存器状态
  184. return ERROR;
  185. }
  186. if(FlashIap_WriteEnd() == ERROR)
  187. {
  188. __set_PRIMASK(temp); //恢复PRIMASK寄存器状态
  189. return ERROR;
  190. }
  191. if(FlashIap_OpenAll_WPROT() == ERROR)
  192. {
  193. __set_PRIMASK(temp);
  194. return ERROR;
  195. }
  196. __set_PRIMASK(temp); //恢复PRIMASK寄存器状态
  197. return SUCCESS;
  198. }
  199. /***************************************************************
  200. 函数名:FlashIap_WriteCont
  201. 描 述:IAP连续写
  202. 输入值:Unit_addr:单元地址 、 Page_addr:页地址 、 Data32:数据
  203. 输出值:无
  204. 返回值:成功、失败
  205. ***************************************************************/
  206. ErrorStatus FlashIap_WriteCont(uint8_t Unit_addr, uint8_t Page_addr, uint32_t Data32)
  207. {
  208. uint16_t temp16;
  209. IAP->ADDR.IAPPA = Page_addr; //输入地址
  210. IAP->ADDR.IAPCA = Unit_addr;
  211. IAP->DATA.DATA = Data32; //输入数据
  212. IAP->TRIG.TRIG = 0x00005DA2; //输入编程命令
  213. for(temp16 = 0; temp16 < 0xFFFF; temp16++)
  214. {
  215. if((IAP->STA.Word & (uint32_t)0x01) == (uint32_t)0x00) //判断IAP工作状态
  216. break;
  217. }
  218. if(temp16 == 0xFFFF)
  219. return ERROR;
  220. for(temp16 = 0; temp16 < 0xFFFF; temp16++)
  221. {
  222. if((IAP->STA.Word & 0x04)==0x04) //判断IAP编程结束标志
  223. break;
  224. }
  225. if(temp16 == 0xFFFF)
  226. return ERROR;
  227. return SUCCESS;
  228. }
  229. /***************************************************************
  230. 函数名:FlashIap_WriteWord
  231. 描 述:IAP写一个字
  232. 输入值:Unit_addr:单元地址 、 Page_addr:页地址 、 Data32:数据
  233. 输出值:无
  234. 返回值:成功、失败
  235. ***************************************************************/
  236. ErrorStatus FlashIap_WriteWord(uint8_t Unit_addr, uint8_t Page_addr, uint32_t Data32)
  237. {
  238. uint32_t temp;
  239. temp = __get_PRIMASK(); //获取PRIMASK寄存器当前状态
  240. __disable_irq(); //屏蔽所有中断
  241. if(FlashIap_Unlock() == ERROR)
  242. {
  243. __set_PRIMASK(temp); //恢复PRIMASK寄存器状态
  244. return ERROR;
  245. }
  246. if(FlashIap_CloseAll_WPROT() == ERROR)
  247. {
  248. __set_PRIMASK(temp); //关闭所有写保护状态
  249. return ERROR;
  250. }
  251. if(FlashIap_WriteCont(Unit_addr, Page_addr, Data32) == ERROR)
  252. {
  253. __set_PRIMASK(temp); //恢复PRIMASK寄存器状态
  254. return ERROR;
  255. }
  256. if(FlashIap_WriteEnd() == ERROR)
  257. {
  258. __set_PRIMASK(temp); //恢复PRIMASK寄存器状态
  259. return ERROR;
  260. }
  261. if(FlashIap_OpenAll_WPROT() == ERROR)
  262. {
  263. __set_PRIMASK(temp); //打开所有写保护状态
  264. return ERROR;
  265. }
  266. __set_PRIMASK(temp); //恢复PRIMASK寄存器状态
  267. return SUCCESS;
  268. }
  269. /***************************************************************
  270. 函数名:Flash_Read
  271. 描 述:Flash读数据
  272. 输入值:Ram_Addr:读出数据的存放地址 、 Flash_Addr:Flash地址(0x00000000 ~ 0x0001FFFF)、 Len:读取的字长度
  273. 输出值:读出的数据
  274. 返回值:成功、失败
  275. ***************************************************************/
  276. ErrorStatus Flash_Read(uint32_t * Ram_Addr, uint32_t Flash_Addr, uint8_t Len)
  277. {
  278. uint8_t i;
  279. uint32_t *ram_addr32;
  280. const uint32_t *flash_addr32;
  281. ram_addr32 = (uint32_t *)Ram_Addr;
  282. flash_addr32 = (const uint32_t *)Flash_Addr;
  283. if((Len == 0) & (Len>(0x20000 - Flash_Addr) / 4)) //判断读取长度是否合法
  284. {
  285. return ERROR;
  286. }
  287. for(i=0; i<Len; i++)
  288. {
  289. *ram_addr32 = *flash_addr32;
  290. ram_addr32++;
  291. flash_addr32++;
  292. }
  293. return SUCCESS;
  294. }
  295. /*************************END OF FILE**********************/