[技术问答] 使用FMC指令调整Data Flash的大小?

[复制链接]
982|6
 楼主| antusheng 发表于 2020-3-12 10:33 | 显示全部楼层 |阅读模式
对于支持可设定Data Flash大小的NuMicro®系列,用户可以在程序中,通过修改DFEN(Config0[0])来使能或禁止Data Flash,以及DFBADR(Config1)设定Data Flash的基地址,调整Data Flash的大小。
在设置后,需要程序中写1到CHIPRST (SYS_IPRST0[0]),进行软件的芯片重启(Chip Reset),新的设定才会生效。

示范程序如下:
uint32_t   au32Config[2];

SYS_UnlockReg();
FMC_Open();
FMC_ReadConfig(au32Config, 2);
FMC_ENABLE_CFG_UPDATE();
au32Config[0] &= ~0x1;
au32Config[1] = u32DFBA;
FMC_WriteConfig(au32Config, 2)
SYS_ResetChip();
 楼主| antusheng 发表于 2020-3-12 10:33 | 显示全部楼层
基本上全系列的都支持。
 楼主| antusheng 发表于 2020-3-12 10:35 | 显示全部楼层
  1. /******************************************************************************
  2. * [url=home.php?mod=space&uid=288409]@file[/url]     main.c
  3. * [url=home.php?mod=space&uid=895143]@version[/url]  V1.00
  4. * $Revision: 4 $
  5. * $Date: 13/10/07 3:56p $
  6. * [url=home.php?mod=space&uid=247401]@brief[/url]    FMC erase/program/read sample program for MINI51 series MCU
  7. *
  8. * @note
  9. * Copyright (C) 2013 Nuvoton Technology Corp. All rights reserved.
  10. *****************************************************************************/   
  11. #include <stdio.h>
  12. #include "Mini51Series.h"
  13. #include "uart.h"
  14. #include "fmc.h"

  15. #define APROM_TEST_BASE             0x3000
  16. #define DATA_FLASH_TEST_BASE        0x3000
  17. #define DATA_FLASH_TEST_END         0x4000
  18. #define TEST_PATTERN 0x12345678


  19. void SYS_Init(void)
  20. {
  21. /*---------------------------------------------------------------------------------------------------------*/
  22. /* Init System Clock                                                                                       */
  23. /*---------------------------------------------------------------------------------------------------------*/
  24.     /* Unlock protected registers */
  25.     SYS_UnlockReg();

  26.     /* Enable External XTAL (4~24 MHz) */
  27.     CLK->PWRCON &= ~CLK_PWRCON_XTLCLK_EN_Msk;
  28.     CLK->PWRCON |= (0x1 << CLK_PWRCON_XTLCLK_EN_Pos); // XTAL12M (HXT) Enabled
  29.    
  30.     /* Waiting for 12MHz clock ready */
  31.     while (!(CLK->CLKSTATUS & CLK_CLKSTATUS_XTL_STB_Msk));

  32.     /* Switch HCLK clock source to XTAL */
  33.     CLK->CLKSEL0 &= ~CLK_CLKSEL0_HCLK_S_Msk;
  34.     CLK->CLKSEL0 |= CLK_CLKSEL0_HCLK_S_XTAL;

  35.     /* Enable IP clock */        
  36.     CLK->APBCLK |= CLK_APBCLK_UART_EN_Msk; // UART Clock Enable
  37.    
  38.     /* Select IP clock source */
  39.     CLK->CLKSEL1 &= ~CLK_CLKSEL1_UART_S_Msk;
  40.     CLK->CLKSEL1 |= CLK_CLKSEL1_UART_S_XTAL;// Clock source from external 12 MHz or 32 KHz crystal clock
  41.                      
  42.     /* Update System Core Clock */
  43.     /* User can use SystemCoreClockUpdate() to calculate PllClock, SystemCoreClock and CycylesPerUs automatically. */
  44.     SystemCoreClockUpdate();

  45. /*---------------------------------------------------------------------------------------------------------*/
  46. /* Init I/O Multi-function                                                                                 */
  47. /*---------------------------------------------------------------------------------------------------------*/
  48.     /* Set P0.0,P0.1 multi-function pins for UART RXD and TXD  */
  49.     SYS->P0_MFP = SYS_MFP_P00_TXD | SYS_MFP_P01_RXD;

  50.     /* Lock protected registers */
  51.     SYS_LockReg();
  52. }

  53. void UART_Init()
  54. {
  55. /*---------------------------------------------------------------------------------------------------------*/
  56. /* Init UART                                                                                               */
  57. /*---------------------------------------------------------------------------------------------------------*/
  58.     UART_Open(UART, 115200);
  59. }


  60. static int  set_data_flash_base(uint32_t u32DFBA)
  61. {
  62.     uint32_t   au32Config[2];
  63.    
  64.     if (FMC_ReadConfig(au32Config, 2) < 0)
  65.     {
  66.         printf("\nRead User Config failed!\n");
  67.         return -1;
  68.     }
  69.         
  70.     if ((!(au32Config[0] & 0x1)) && (au32Config[1] == u32DFBA))
  71.         return 0;
  72.         
  73.     FMC_EnableConfigUpdate();
  74.         
  75.     au32Config[0] &= ~0x1;
  76.     au32Config[1] = u32DFBA;

  77.     if (FMC_WriteConfig(au32Config, 2) < 0)
  78.         return -1;
  79.         
  80.     printf("\nSet Data Flash base as 0x%x.\n", DATA_FLASH_TEST_BASE);

  81.     // Perform chip reset to make new User Config take effect
  82.     SYS->IPRSTC1 = SYS_IPRSTC1_CHIP_RST_Msk;
  83.     return 0;
  84. }


  85. int32_t fill_data_pattern(uint32_t u32StartAddr, uint32_t u32EndAddr, uint32_t u32Pattern)
  86. {
  87.     uint32_t u32Addr;
  88.    
  89.     for (u32Addr = u32StartAddr; u32Addr < u32EndAddr; u32Addr += 4)
  90.     {
  91.         FMC_Write(u32Addr, u32Pattern);
  92.     }
  93.     return 0;
  94. }


  95. int32_t  verify_data(uint32_t u32StartAddr, uint32_t u32EndAddr, uint32_t u32Pattern)
  96. {
  97.     uint32_t    u32Addr;
  98.     uint32_t    u32data;
  99.    
  100.     for (u32Addr = u32StartAddr; u32Addr < u32EndAddr; u32Addr += 4)
  101.     {     
  102.         u32data = FMC_Read(u32Addr);
  103.         if (u32data != u32Pattern)
  104.         {
  105.            printf("\nFMC_Read data verify failed at address 0x%x, read=0x%x, expect=0x%x\n", u32Addr, u32data, u32Pattern);
  106.            return -1;
  107.         }         
  108.     }
  109.     return 0;
  110. }


  111. int32_t  flash_test(uint32_t u32StartAddr, uint32_t u32EndAddr, uint32_t u32Pattern)
  112. {
  113.     uint32_t    u32Addr;
  114.    
  115.     for (u32Addr = u32StartAddr; u32Addr < u32EndAddr; u32Addr += FMC_FLASH_PAGE_SIZE)
  116.     {
  117.         printf("    Flash test address: 0x%x    \r", u32Addr);

  118.         // Erase page
  119.         FMC_Erase(u32Addr);
  120.         
  121.         // Verify if page contents are all 0xFFFFFFFF
  122.         if (verify_data(u32Addr, u32Addr + FMC_FLASH_PAGE_SIZE, 0xFFFFFFFF) < 0)
  123.         {
  124.             printf("\nPage 0x%x erase verify failed!\n", u32Addr);
  125.             return -1;
  126.         }
  127.         
  128.         // Write test pattern to fill the whole page
  129.         if (fill_data_pattern(u32Addr, u32Addr + FMC_FLASH_PAGE_SIZE, u32Pattern) < 0)
  130.         {
  131.             printf("Failed to write page 0x%x!\n", u32Addr);
  132.             return -1;
  133.         }

  134.         // Verify if page contents are all equal to test pattern
  135.         if (verify_data(u32Addr, u32Addr + FMC_FLASH_PAGE_SIZE, u32Pattern) < 0)
  136.         {
  137.             printf("\nData verify failed!\n ");
  138.             return -1;
  139.         }

  140.         FMC_Erase(u32Addr);
  141.         
  142.         // Verify if page contents are all 0xFFFFFFFF
  143.         if (verify_data(u32Addr, u32Addr + FMC_FLASH_PAGE_SIZE, 0xFFFFFFFF) < 0)
  144.         {
  145.             printf("\nPage 0x%x erase verify failed!\n", u32Addr);
  146.             return -1;
  147.         }
  148.                                
  149.     }
  150.                                
  151.     printf("\r    Flash Test Passed.          \n");
  152.     return 0;
  153. }

  154. #define DATA_FLAHSH_OFFSET  0x3000

  155. unsigned int temp1;

  156. unsigned char SIM_EEPROM_READ(unsigned int address)
  157. {
  158. unsigned int temp;
  159.   temp=FMC_Read(((address/4)*4)+DATA_FLAHSH_OFFSET);
  160.         return (temp>>((address%4)*8))&0xff;;
  161. }

  162. void SIM_EEPROM_WRITE(unsigned int address,unsigned char data)
  163. {
  164.         unsigned int temp_data;
  165.         unsigned int i,j;
  166.   unsigned int buffer[512/4];       
  167.        
  168.        
  169.         if((SIM_EEPROM_READ(address)==0xff)|((SIM_EEPROM_READ(address)&data)==data))
  170.         {
  171.                 temp_data=FMC_Read(((address/4)*4)+DATA_FLAHSH_OFFSET);
  172.                 temp_data=~(0x0ff<<((address%4)*8));
  173.                 temp_data|=data<<((address%4)*8);
  174.                 FMC_Write(((address/4)*4)+DATA_FLAHSH_OFFSET, temp_data);
  175.         }
  176.         else
  177.              {
  178.                                  temp_data=FMC_Read(((address/4)*4)+DATA_FLAHSH_OFFSET);
  179.                                  temp_data&=~(0x0ff<<((address%4)*8));
  180.                                  temp_data|=data<<((address%4)*8);
  181.                                  //backup date
  182.                                  j=0;
  183.                                  
  184.                                 // printf("0x%x\n\r",DATA_FLAHSH_OFFSET+(address/512)+(512));
  185.                                  for(i=DATA_FLAHSH_OFFSET+((address/512)*512);i<DATA_FLAHSH_OFFSET+((address/512)*512)+512;i=i+4)
  186.                                   {
  187.                                                 //printf("0x%x\n\r",i);
  188.                                                 //printf("0x%x\n\r",FMC_Read(i));
  189.                                          buffer[j]=FMC_Read(i);
  190.                                                
  191.                                                 j++;
  192.                                         }                                 
  193.                                         //erase page
  194.                       FMC_Erase(DATA_FLAHSH_OFFSET+((address/512)*512));
  195.                                        
  196.                                         buffer[(address%512)/4]=temp_data;
  197.                                         j=0;
  198.                                         for(i=DATA_FLAHSH_OFFSET+((address/512)*512);i<DATA_FLAHSH_OFFSET+((address/512)*512)+512;i=i+4)
  199.                                   {
  200.                                                 FMC_Write(i,buffer[j]);
  201.                                                 j++;
  202.                                         }
  203.                    }
  204. }


  205. int main()
  206. {
  207.     uint32_t    i, u32Data;

  208.     SYS_Init();
  209.     UART_Init();

  210.     printf("\n\n");
  211.     printf("+----------------------------------------+\n");
  212.     printf("|         MINI51 FMC Sample Code         |\n");
  213.     printf("+----------------------------------------+\n");

  214.     SYS_UnlockReg();

  215.     /* Enable FMC ISP function */
  216.     FMC_Open();

  217.     if (set_data_flash_base(DATA_FLASH_TEST_BASE) < 0)
  218.     {
  219.         printf("Failed to set Data Flash base address!\n");
  220.         goto lexit;
  221.     }

  222.     /* Read Data Flash base address */
  223.     u32Data = FMC_ReadDataFlashBaseAddr();
  224.     printf("  Data Flash Base Address ............... [0x%08x]\n", u32Data);
  225.                 FMC_Erase(DATA_FLASH_TEST_BASE);               
  226.                
  227.                 for(i=0;i<1024;i++)
  228.                 {
  229.                 SIM_EEPROM_WRITE(i,0);
  230.                                 if(SIM_EEPROM_READ(i)!=0)
  231.                         {
  232.                 printf("0x%x",SIM_EEPROM_READ(i));
  233.                                 while(1);
  234.                         }
  235.                 }
  236.                 for(i=0;i<1024;i++)
  237.           {
  238.                 SIM_EEPROM_WRITE(i,(i&0xff));
  239.                         if(SIM_EEPROM_READ(i)!=(i&0xff))
  240.                         {
  241.                 printf("0x%x",SIM_EEPROM_READ(i));
  242.                                 while(1);
  243.                         }
  244.                 }
  245.         #if 0
  246.     printf("\n\nData Flash test =>\n");
  247.     if (flash_test(DATA_FLASH_TEST_BASE, DATA_FLASH_TEST_END, TEST_PATTERN) < 0)
  248.     {
  249.         printf("\n\nUHB test failed!\n");
  250.         goto lexit;
  251.     }
  252.     #endif
  253. lexit:

  254.     /* Disable FMC ISP function */
  255.     FMC_Close();

  256.     /* Lock protected registers */
  257.     SYS_LockReg();
  258.    
  259.     printf("\nFMC Sample Code Completed.\n");
  260.    
  261.     while (1);
  262. }

  263. /*** (C) COPYRIGHT 2013 Nuvoton Technology Corp. ***/
wahahaheihei 发表于 2020-3-15 21:49 | 显示全部楼层
多谢分享。
wanduzi 发表于 2020-3-15 22:07 | 显示全部楼层
提供的例子非常容易入门。
小灵通2018 发表于 2020-3-16 11:15 | 显示全部楼层
FMC的操作要熟练起来。
643757107 发表于 2020-3-20 13:36 | 显示全部楼层
还要复位芯片的操作。
您需要登录后才可以回帖 登录 | 注册

本版积分规则

86

主题

1521

帖子

5

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