[技术问答] NUC029LAN内部EEPROM的使用问题。

[复制链接]
12387|21
 楼主| 18123784178 发表于 2017-10-17 15:07 | 显示全部楼层 |阅读模式
各位大牛,我是初次使用NUC029LAN这个芯片,要使用其内部的EEPROM进行断电保存数据,但是没有找到官方的例程,谁有的请帮忙发一个给我,不胜感激。
zhuomuniao110 发表于 2017-10-17 15:25 | 显示全部楼层
掉电后EEPROM的数据丢失了吗
zhuomuniao110 发表于 2017-10-17 15:25 | 显示全部楼层
掉电保护是一种特殊的保护数据的方式,是指在掉电时,所有的数据将一直保存在磁盘上,直到重构完成后才删除。掉电保护(Power Fail Safeguard),当此项设置为可用时,在重构过程中(非重建),这样如果在重构过程中发生掉电,将不会发生数据丢失的危险情况。
指在正常供电电源掉电时,迅速用备用直流电源供电,以保证在一段时间内信息不会丢失,当主电源恢复供电时,又自动切换为主电源供电。
zhuomuniao110 发表于 2017-10-17 15:26 | 显示全部楼层
请注意,你要有备用电源。
 楼主| 18123784178 发表于 2017-10-17 15:42 | 显示全部楼层
EEPROM读写函数,这个应该有个专门的c文件才对啊,我没有,谁有啊?上传一个给我咯
 楼主| 18123784178 发表于 2017-10-17 15:44 | 显示全部楼层
我还没有开始应用,我是准备用NUC029LAN开发一个项目,需要用到内部EEPROM,不知道没有找到相关的BSP
 楼主| 18123784178 发表于 2017-10-17 15:45 | 显示全部楼层
好像官方BSP里没有相关方面的内容呢,奇怪了
 楼主| 18123784178 发表于 2017-10-17 15:46 | 显示全部楼层
zhuomuniao110 发表于 2017-10-17 15:26
请注意,你要有备用电源。

有检测外部的220V交流电,有大电容续航,只要220V交流电没有了,就立即保存数据就ok了吧。
 楼主| 18123784178 发表于 2017-10-17 15:47 | 显示全部楼层
zhuomuniao110 发表于 2017-10-17 15:25
掉电后EEPROM的数据丢失了吗

你有用过NUC029LAN的EEPROM吗?
天灵灵地灵灵 发表于 2017-10-17 18:59 | 显示全部楼层
我去找找看,怎么会没有呢,肯定有的。
天灵灵地灵灵 发表于 2017-10-17 19:04 | 显示全部楼层
  1. /******************************************************************************
  2. * [url=home.php?mod=space&uid=288409]@file[/url]     main.c
  3. * [url=home.php?mod=space&uid=895143]@version[/url]  V3.00
  4. * $Revision: 8 $
  5. * $Date: 14/02/10 2:43p $
  6. * [url=home.php?mod=space&uid=247401]@brief[/url]    FMC erase/program/read sample program for NUC029 series MCU
  7. *
  8. * @note
  9. * Copyright (C) 2014 Nuvoton Technology Corp. All rights reserved.
  10. *****************************************************************************/
  11. #include <stdio.h>
  12. #include "NUC029xAN.h"

  13. #define PLLCON_SETTING      CLK_PLLCON_50MHz_HXT
  14. #define PLL_CLOCK           50000000


  15. #define APROM_TEST_BASE             0x3000
  16. #define APROM_TEST_END              0x10000
  17. #define DATA_FLASH_TEST_BASE        (FMC->DFBADR)
  18. #define DATA_FLASH_TEST_END         (FMC->DFBADR + 0x1000)
  19. #define TEST_PATTERN                0x5A5A5A5A



  20. void SYS_Init(void)
  21. {
  22.     /*---------------------------------------------------------------------------------------------------------*/
  23.     /* Init System Clock                                                                                       */
  24.     /*---------------------------------------------------------------------------------------------------------*/

  25.     /* Enable External XTAL (4~24 MHz) */
  26.     CLK->PWRCON |= CLK_PWRCON_XTL12M_EN_Msk;

  27.     CLK->PLLCON = PLLCON_SETTING;

  28.     /* Waiting for clock ready */
  29.     CLK_WaitClockReady(CLK_CLKSTATUS_PLL_STB_Msk | CLK_CLKSTATUS_XTL12M_STB_Msk);

  30.     /* Switch HCLK clock source to PLL */
  31.     CLK->CLKSEL0 = CLK_CLKSEL0_HCLK_S_PLL;

  32.     /* Enable IP clock */
  33.     CLK->APBCLK = CLK_APBCLK_UART0_EN_Msk;

  34.     /* Select IP clock source */
  35.     CLK->CLKSEL1 = CLK_CLKSEL1_UART_S_HXT;

  36.     /* Update System Core Clock */
  37.     /* User can use SystemCoreClockUpdate() to calculate PllClock, SystemCoreClock and CycylesPerUs automatically. */
  38.     //SystemCoreClockUpdate();
  39.     PllClock        = PLL_CLOCK;            // PLL
  40.     SystemCoreClock = PLL_CLOCK / 1;        // HCLK
  41.     CyclesPerUs     = PLL_CLOCK / 1000000;  // For SYS_SysTickDelay()

  42.     /*---------------------------------------------------------------------------------------------------------*/
  43.     /* Init I/O Multi-function                                                                                 */
  44.     /*---------------------------------------------------------------------------------------------------------*/
  45.     /* Set P3 multi-function pins for UART0 RXD and TXD  */
  46.     SYS->P3_MFP = SYS_MFP_P30_RXD0 | SYS_MFP_P31_TXD0;

  47. }

  48. void UART_Init()
  49. {
  50.     /*---------------------------------------------------------------------------------------------------------*/
  51.     /* Init UART                                                                                               */
  52.     /*---------------------------------------------------------------------------------------------------------*/
  53.     UART_Open(UART0, 115200);
  54. }


  55. static int  SetDataFlashBase(uint32_t u32DFBA)
  56. {
  57.     uint32_t au32Config[1];

  58.     /* Read current User Configuration */
  59.     FMC_ReadConfig(au32Config, 1);

  60.     /* Just return when Data Flash has been enabled */
  61.     if(!(au32Config[0] & 0x1))
  62.         return 0;

  63.     /* Enable User Configuration Update */
  64.     FMC_EnableConfigUpdate();

  65.     /* Erase User Configuration */
  66.     FMC_Erase(FMC_CONFIG_BASE);

  67.     /* Write User Configuration to Enable Data Flash */
  68.     au32Config[0] &= ~0x1;
  69.     if(FMC_WriteConfig(au32Config, 1))
  70.         return -1;

  71.     printf("\nSet Data Flash base as 0x%x.\n", FMC_ReadDataFlashBaseAddr());

  72.     /* Perform chip reset to make new User Config take effect */
  73.     SYS->IPRSTC1 |= SYS_IPRSTC1_CHIP_RST_Msk;

  74.     return 0;
  75. }


  76. int32_t FillDataPattern(uint32_t u32StartAddr, uint32_t u32EndAddr, uint32_t u32Pattern)
  77. {
  78.     uint32_t u32Addr;

  79.     for(u32Addr = u32StartAddr; u32Addr < u32EndAddr; u32Addr += 4)
  80.     {
  81.         FMC_Write(u32Addr, u32Pattern);
  82.     }
  83.     return 0;
  84. }


  85. int32_t  VerifyData(uint32_t u32StartAddr, uint32_t u32EndAddr, uint32_t u32Pattern)
  86. {
  87.     uint32_t    u32Addr;
  88.     uint32_t    u32Data;

  89.     for(u32Addr = u32StartAddr; u32Addr < u32EndAddr; u32Addr += 4)
  90.     {
  91.         u32Data = FMC_Read(u32Addr);
  92.         if(u32Data != u32Pattern)
  93.         {
  94.             printf("\nFMC_Read data verify failed at address 0x%x, read=0x%x, expect=0x%x\n", u32Addr, u32Data, u32Pattern);
  95.             return -1;
  96.         }
  97.     }
  98.     return 0;
  99. }


  100. int32_t  FlashTest(uint32_t u32StartAddr, uint32_t u32EndAddr, uint32_t u32Pattern)
  101. {
  102.     uint32_t    u32Addr;

  103.     for(u32Addr = u32StartAddr; u32Addr < u32EndAddr; u32Addr += FMC_FLASH_PAGE_SIZE)
  104.     {
  105.         printf("    Flash test address: 0x%x    \r", u32Addr);

  106.         // Erase page
  107.         FMC_Erase(u32Addr);

  108.         // Verify if page contents are all 0xFFFFFFFF
  109.         if(VerifyData(u32Addr, u32Addr + FMC_FLASH_PAGE_SIZE, 0xFFFFFFFF) < 0)
  110.         {
  111.             printf("\nPage 0x%x erase verify failed!\n", u32Addr);
  112.             return -1;
  113.         }

  114.         // Write test pattern to fill the whole page
  115.         if(FillDataPattern(u32Addr, u32Addr + FMC_FLASH_PAGE_SIZE, u32Pattern) < 0)
  116.         {
  117.             printf("Failed to write page 0x%x!\n", u32Addr);
  118.             return -1;
  119.         }

  120.         // Verify if page contents are all equal to test pattern
  121.         if(VerifyData(u32Addr, u32Addr + FMC_FLASH_PAGE_SIZE, u32Pattern) < 0)
  122.         {
  123.             printf("\nData verify failed!\n ");
  124.             return -1;
  125.         }

  126.         FMC_Erase(u32Addr);

  127.         // Verify if page contents are all 0xFFFFFFFF
  128.         if(VerifyData(u32Addr, u32Addr + FMC_FLASH_PAGE_SIZE, 0xFFFFFFFF) < 0)
  129.         {
  130.             printf("\nPage 0x%x erase verify failed!\n", u32Addr);
  131.             return -1;
  132.         }
  133.     }
  134.     printf("\r    Flash Test Passed.          \n");
  135.     return 0;
  136. }


  137. int main()
  138. {
  139.     uint32_t i, u32Data;

  140.     /* Unlock protected registers */
  141.     SYS_UnlockReg();

  142.     SYS_Init();
  143.     UART_Init();

  144.     /*
  145.         This sample code is used to show how to use StdDriver API to implement ISP functions.
  146.     */

  147.     printf("\n\n");
  148.     printf("+----------------------------------------+\n");
  149.     printf("|          NUC029 FMC Sample Code          |\n");
  150.     printf("+----------------------------------------+\n");

  151.     SYS_UnlockReg();

  152.     /* Enable FMC ISP function */
  153.     FMC_Open();

  154.     if(SetDataFlashBase(DATA_FLASH_TEST_BASE) < 0)
  155.     {
  156.         printf("Failed to set Data Flash base address!\n");
  157.         goto lexit;
  158.     }

  159.     /* Read BS */
  160.     printf("  Boot Mode ............................. ");
  161.     if(FMC_GetBootSource() == 0)
  162.         printf("[APROM]\n");
  163.     else
  164.     {
  165.         printf("[LDROM]\n");
  166.         printf("  WARNING: The driver sample code must execute in AP mode!\n");
  167.         goto lexit;
  168.     }

  169.     u32Data = FMC_ReadCID();
  170.     printf("  Company ID ............................ [0x%08x]\n", u32Data);

  171.     u32Data = FMC_ReadDID();
  172.     printf("  Device ID ............................. [0x%08x]\n", u32Data);

  173.     u32Data = FMC_ReadPID();
  174.     printf("  Product ID ............................ [0x%08x]\n", u32Data);

  175.     for(i = 0; i < 3; i++)
  176.     {
  177.         u32Data = FMC_ReadUID(i);
  178.         printf("  Unique ID %d ........................... [0x%08x]\n", i, u32Data);
  179.     }

  180.     for(i = 0; i < 4; i++)
  181.     {
  182.         u32Data = FMC_ReadUCID(i);
  183.         printf("  Unique Customer ID %d .................. [0x%08x]\n", i, u32Data);
  184.     }

  185.     /* Read User Configuration */
  186.     printf("  User Config 0 ......................... [0x%08x]\n", FMC_Read(FMC_CONFIG_BASE));
  187.     printf("  User Config 1 ......................... [0x%08x]\n", FMC_Read(FMC_CONFIG_BASE + 4));

  188.     /* Read Data Flash base address */
  189.     u32Data = FMC_ReadDataFlashBaseAddr();
  190.     printf("  Data Flash Base Address ............... [0x%08x]\n", u32Data);

  191.     printf("\n\nLDROM test =>\n");
  192.     FMC_EnableLDUpdate();
  193.     if(FlashTest(FMC_LDROM_BASE, FMC_LDROM_BASE + FMC_LDROM_SIZE, TEST_PATTERN) < 0)
  194.     {
  195.         printf("\n\nLDROM test failed!\n");
  196.         goto lexit;
  197.     }
  198.     FMC_DisableLDUpdate();

  199.     printf("\n\nAPROM test =>\n");
  200.     FMC_EnableAPUpdate();
  201.     if(FlashTest(APROM_TEST_BASE, APROM_TEST_END, TEST_PATTERN) < 0)
  202.     {
  203.         printf("\n\nAPROM test failed!\n");
  204.         goto lexit;
  205.     }
  206.     FMC_DisableAPUpdate();

  207.     printf("\n\nData Flash test =>\n");
  208.     if(FlashTest(DATA_FLASH_TEST_BASE, DATA_FLASH_TEST_END, TEST_PATTERN) < 0)
  209.     {
  210.         printf("\n\nData Flash test failed!\n");
  211.         goto lexit;
  212.     }

  213. lexit:

  214.     /* Disable FMC ISP function */
  215.     FMC_Close();

  216.     /* Lock protected registers */
  217.     SYS_LockReg();

  218.     printf("\nFMC Sample Code Completed.\n");

  219.     while(1);
  220. }

  221. /*** (C) COPYRIGHT 2014 Nuvoton Technology Corp. ***/
天灵灵地灵灵 发表于 2017-10-17 19:05 | 显示全部楼层
应该就是上面这个例子吧,FMC读写操作。
5574293 发表于 2017-10-18 11:26 | 显示全部楼层
NUC029的官方BSP里面都有,你找找。它有几个例程,新唐里面叫dataflash,和EEEProm是一个东西。使用时需要注意几点:
1、数据的写入之前要进行擦除
2、擦除都是按照一页一页来擦除
3、BSP包里提供了三个函数,FMC_Erase  FMC_Write FMC_Read.即擦除,写入,以及读取三个函数。
4、使用三个函数时,注意其入口参数的类型以及入口参数的格式。举个例子:
FMC_Erase(0x1f000);
FMC_Write(0x1f000+i*4,change_data[i]);
FMC_Read(0x1f000+i*4);
哦 对了还需要注意的是读取和写入的时候需要考虑数据的大小。根据大小来取两个数据之间的地址间隔。同时在三个操作之前要对系统进行一系列的使能配置,比如解除系统锁定,使能LDu,APU等配置。

评分

参与人数 1威望 +2 收起 理由
18123784178 + 2 很给力!

查看全部评分

 楼主| 18123784178 发表于 2017-10-23 08:52 | 显示全部楼层
5574293 发表于 2017-10-18 11:26
NUC029的官方BSP里面都有,你找找。它有几个例程,新唐里面叫dataflash,和EEEProm是一个东西。使用时需要 ...

好的,感谢,非常感谢
 楼主| 18123784178 发表于 2017-10-23 09:02 | 显示全部楼层
天灵灵地灵灵 发表于 2017-10-17 19:05
应该就是上面这个例子吧,FMC读写操作。

谢谢,上面那个例子我在BSP里有看到,但是不我还是有很多不明白的地方。比如我如果写数据到APROM,那里是程序存储的地方,会不会把我的数据搞没了,或者把程序搞乱了?
 楼主| 18123784178 发表于 2017-10-23 09:18 | 显示全部楼层
5574293 发表于 2017-10-18 11:26
NUC029的官方BSP里面都有,你找找。它有几个例程,新唐里面叫dataflash,和EEEProm是一个东西。使用时需要 ...

你说的读写擦除等函数我找到了,但是你说的那些注意什么的,能说得具体详细点吗,谢谢
 楼主| 18123784178 发表于 2017-10-23 10:27 | 显示全部楼层
天灵灵地灵灵 发表于 2017-10-17 19:05
应该就是上面这个例子吧,FMC读写操作。

我是需要掉电保存数据,应该是对地址0X0001F000开始,0x0001ffff结束的这段地址进行读写操作吧。但是楼下的说还要注意其他的,不知道还有哪些,BSP里的貌似没有对这一块进行举例,全是读写ISP那段的还有用户配置信息那段的。你那有具体的应用例程吗?
 楼主| 18123784178 发表于 2017-10-23 10:33 | 显示全部楼层
5574293 发表于 2017-10-18 11:26
NUC029的官方BSP里面都有,你找找。它有几个例程,新唐里面叫dataflash,和EEEProm是一个东西。使用时需要 ...

比如如何解除系统锁定?LDu,APU跟我要使用的data Flash貌似不是同一段空间啊?
5574293 发表于 2017-10-23 11:50 | 显示全部楼层
18123784178 发表于 2017-10-23 10:33
比如如何解除系统锁定?LDu,APU跟我要使用的data Flash貌似不是同一段空间啊? ...

系统解除锁定BSP包里也有,就是一个函数的引用,sys_unlock(),具体我忘了。新唐的资料还是比较多的。网上找好找,你最好是搜一搜新唐data flash 的相关问题,就可以看到很多详细的了。我记得我有一个PDF的文档,关于data flash 的,但是找不到了。
你说的APU和LDU,其实和data flash是不在同一段空间,但是你要操作data flash就必须使能这些模块。然后你也知道data flash 的操作地址是从0x0001f000开始进行操作。只要你对应的地址没错,是不会把其他数据给擦除的。调试的时候你再结合那个软件上的内存看看,你写入的数据正不正确,就可以知道你的程序对不对了
 楼主| 18123784178 发表于 2017-10-23 14:10 | 显示全部楼层
5574293 发表于 2017-10-23 11:50
系统解除锁定BSP包里也有,就是一个函数的引用,sys_unlock(),具体我忘了。新唐的资料还是比较多的。 ...

好的,谢谢
您需要登录后才可以回帖 登录 | 注册

本版积分规则

3

主题

22

帖子

0

粉丝
快速回复 在线客服 返回列表 返回顶部