打印
[技术问答]

m058写dataflash几次后MCU会宕机

[复制链接]
1055|13
手机看帖
扫描二维码
随时随地手机跟帖
跳转到指定楼层
楼主
shizheng080|  楼主 | 2019-8-16 11:48 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
大家好!我现在使用的是m058LDN 没有使用操作系统,现在有个问题是将片内dataflash当做EEPROM来用,存储配置参数,但是发现在操作几次或者几十次后M058会死机,大家有没有遇到过此问题,各位大牛帮忙看看是哪里出了问题,谢谢。
代码如下:

#define PAGE_SIZE             512
#define FMC_DATAFLASH_BASE    0x0001F000
#define DATAFLASH_SIZE        0x00001000

int32_t FMC_WriteData(uint32_t *u32Config, uint32_t u32Count,uint32_t page)
{
    int32_t i;
        __set_PRIMASK(1);//关闭全局中断
        SYS_UnlockReg();
        FMC_Open();//Enable FMC ISP function
        FMC_EnableLDUpdate();
       
        FMC_Erase(FMC_DATAFLASH_BASE + page * PAGE_SIZE);

    for(i = 0; i < u32Count; i++)
    {
        FMC_Write(FMC_DATAFLASH_BASE + page * PAGE_SIZE + i * 4, u32Config[i]);
        if(FMC_Read(FMC_DATAFLASH_BASE + page * PAGE_SIZE + i * 4) != u32Config[i])
            return -1;
    }
        FMC_Close();//Disable ISP Functions
        SYS_LockReg();
        __set_PRIMASK(0);//打开全局中断
    return 0;
}

void FMC_ReadData(uint32_t *u32Config, uint32_t u32Count,uint32_t page)
{
    int32_t i;
        __set_PRIMASK(1);//关闭全局中断
        SYS_UnlockReg();
        FMC_Open();//Enable FMC ISP function
        FMC_EnableLDUpdate();

    for(i = 0; i < u32Count; i++)
        {
                u32Config[i] = FMC_Read(FMC_DATAFLASH_BASE + page * PAGE_SIZE + i * 4);
        }
        FMC_Close();//Disable ISP Functions
        SYS_LockReg();
        __set_PRIMASK(0);//打开全局中断
}

使用特权

评论回复
沙发
dongnanxibei| | 2019-8-16 16:56 | 只看该作者
你是不是操作了不该操作的地方。先规划好存储位置。

使用特权

评论回复
板凳
shizheng080|  楼主 | 2019-8-16 17:11 | 只看该作者
dongnanxibei 发表于 2019-8-16 16:56
你是不是操作了不该操作的地方。先规划好存储位置。

应该是没错的,**作的page是在0-7,都试过,都有死机现象

使用特权

评论回复
地板
shizheng080|  楼主 | 2019-8-16 17:15 | 只看该作者
shizheng080 发表于 2019-8-16 17:11
应该是没错的,**作的page是在0-7,都试过,都有死机现象

后来把#define PAGE_SIZE             512
改为#define PAGE_SIZE             128  了
可是还是不行

使用特权

评论回复
5
玛尼玛尼哄| | 2019-8-18 13:45 | 只看该作者
这个型号官方有提供类似操作的例子没,看看怎么实现。

使用特权

评论回复
6
zhuotuzi| | 2019-8-18 21:03 | 只看该作者
规划地址从后面规划试试看

使用特权

评论回复
7
zhuotuzi| | 2019-8-18 21:03 | 只看该作者
彻底避免修改前面存储程序的可能。

使用特权

评论回复
8
dongnanxibei| | 2019-8-19 12:51 | 只看该作者
地址跟例子中的不同啊。
M051的官方BSP例子是这样吗定义的

使用特权

评论回复
9
dongnanxibei| | 2019-8-19 12:51 | 只看该作者
#define APROM_TEST_BASE             0x3000
#define APROM_TEST_END              0x10000
#define DATA_FLASH_TEST_BASE        (FMC->DFBADR)
#define DATA_FLASH_TEST_END         (FMC->DFBADR + 0x1000)
#define TEST_PATTERN                0x5A5A5A5A

使用特权

评论回复
10
dongnanxibei| | 2019-8-19 12:54 | 只看该作者
而很多地址,头文件定义好了,你另外定义,肯定容易错。

使用特权

评论回复
11
dongnanxibei| | 2019-8-19 12:54 | 只看该作者
/*---------------------------------------------------------------------------------------------------------*/
/* Define Base Address                                                                                     */
/*---------------------------------------------------------------------------------------------------------*/
#define FMC_APROM_BASE          0x00000000UL    /*!< APROM  Base Address         */
#define FMC_LDROM_BASE          0x00100000UL    /*!< LDROM  Base Address         */
#define FMC_CONFIG_BASE         0x00300000UL    /*!< CONFIG Base Address         */

#define FMC_FLASH_PAGE_SIZE     0x200           /*!< Flash Page Size (512 Bytes) */
#define FMC_LDROM_SIZE          0x1000          /*!< LDROM Size (4 KBytes)       */

使用特权

评论回复
12
dongnanxibei| | 2019-8-19 12:54 | 只看该作者
/******************************************************************************
* [url=home.php?mod=space&uid=288409]@file[/url]     main.c
* [url=home.php?mod=space&uid=895143]@version[/url]  V3.00
* $Revision: 10 $
* $Date: 15/05/22 5:13p $
* [url=home.php?mod=space&uid=247401]@brief[/url]    FMC erase/program/read sample program for M051 series MCU
*
* @note
* Copyright (C) 2014 Nuvoton Technology Corp. All rights reserved.
*****************************************************************************/
#include <stdio.h>
#include "M051Series.h"

#define PLLCON_SETTING      CLK_PLLCON_50MHz_HXT
#define PLL_CLOCK           50000000


#define APROM_TEST_BASE             0x3000
#define APROM_TEST_END              0x10000
#define DATA_FLASH_TEST_BASE        (FMC->DFBADR)
#define DATA_FLASH_TEST_END         (FMC->DFBADR + 0x1000)
#define TEST_PATTERN                0x5A5A5A5A



void SYS_Init(void)
{
    /*---------------------------------------------------------------------------------------------------------*/
    /* Init System Clock                                                                                       */
    /*---------------------------------------------------------------------------------------------------------*/

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

    CLK->PLLCON = PLLCON_SETTING;

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

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

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

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

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

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

}

void UART_Init()
{
    /*---------------------------------------------------------------------------------------------------------*/
    /* Init UART                                                                                               */
    /*---------------------------------------------------------------------------------------------------------*/
    UART_Open(UART0, 115200);
}

int32_t FillDataPattern(uint32_t u32StartAddr, uint32_t u32EndAddr, uint32_t u32Pattern)
{
    uint32_t u32Addr;

    for(u32Addr = u32StartAddr; u32Addr < u32EndAddr; u32Addr += 4)
    {
        FMC_Write(u32Addr, u32Pattern);
    }
    return 0;
}


int32_t  VerifyData(uint32_t u32StartAddr, uint32_t u32EndAddr, uint32_t u32Pattern)
{
    uint32_t    u32Addr;
    uint32_t    u32Data;

    for(u32Addr = u32StartAddr; u32Addr < u32EndAddr; u32Addr += 4)
    {
        u32Data = FMC_Read(u32Addr);
        if(u32Data != u32Pattern)
        {
            printf("\nFMC_Read data verify failed at address 0x%x, read=0x%x, expect=0x%x\n", u32Addr, u32Data, u32Pattern);
            return -1;
        }
    }
    return 0;
}


int32_t  FlashTest(uint32_t u32StartAddr, uint32_t u32EndAddr, uint32_t u32Pattern)
{
    uint32_t    u32Addr;

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

        // Erase page
        FMC_Erase(u32Addr);

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

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

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

        FMC_Erase(u32Addr);

        // Verify if page contents are all 0xFFFFFFFF
        if(VerifyData(u32Addr, u32Addr + FMC_FLASH_PAGE_SIZE, 0xFFFFFFFF) < 0)
        {
            printf("\nPage 0x%x erase verify failed!\n", u32Addr);
            return -1;
        }
    }
    printf("\r    Flash Test Passed.          \n");
    return 0;
}


int main()
{
    uint32_t i, u32Data;

    /* Unlock protected registers */
    SYS_UnlockReg();

    SYS_Init();
    UART_Init();

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

    printf("\n\n");
    printf("+----------------------------------------+\n");
    printf("|          M051 FMC Sample Code          |\n");
    printf("+----------------------------------------+\n");

    SYS_UnlockReg();

    /* Enable FMC ISP function */
    FMC_Open();

    /* Read BS */
    printf("  Boot Mode ............................. ");
    if(FMC_GetBootSource() == 0)
        printf("[APROM]\n");
    else
    {
        printf("[LDROM]\n");
        printf("  WARNING: The driver sample code must execute in AP mode!\n");
        goto lexit;
    }

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

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

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

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

    /* Read User Configuration */
    printf("  User Config 0 ......................... [0x%08x]\n", FMC_Read(FMC_CONFIG_BASE));

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

    printf("\n\nLDROM test =>\n");
    FMC_EnableLDUpdate();
    if(FlashTest(FMC_LDROM_BASE, FMC_LDROM_BASE + FMC_LDROM_SIZE, TEST_PATTERN) < 0)
    {
        printf("\n\nLDROM test failed!\n");
        goto lexit;
    }
    FMC_DisableLDUpdate();

    printf("\n\nAPROM test =>\n");
    FMC_EnableAPUpdate();
    if(FlashTest(APROM_TEST_BASE, APROM_TEST_END, TEST_PATTERN) < 0)
    {
        printf("\n\nAPROM test failed!\n");
        goto lexit;
    }
    FMC_DisableAPUpdate();

    printf("\n\nData Flash test =>\n");
    if(FlashTest(DATA_FLASH_TEST_BASE, DATA_FLASH_TEST_END, TEST_PATTERN) < 0)
    {
        printf("\n\nData Flash test failed!\n");
        goto lexit;
    }

lexit:

    /* Disable FMC ISP function */
    FMC_Close();

    /* Lock protected registers */
    SYS_LockReg();

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

    while(1);
}

/*** (C) COPYRIGHT 2014 Nuvoton Technology Corp. ***/

使用特权

评论回复
13
dongnanxibei| | 2019-8-19 12:55 | 只看该作者
参考上面这个BSP带的例子

使用特权

评论回复
14
shizheng080|  楼主 | 2019-8-20 12:18 | 只看该作者
谢谢大家的回复,后来按照官方的BSP来做了一下,但还是会出现问题,后来发现其实并不是MCU死掉了,而是不知道什么原因,把我的显示屏给搞死掉了,后来发现好像和时钟有关,现在暂时通过操作dataflash后再将时钟部分重新初始化一遍得以暂时解决,还有待继续查找根源

使用特权

评论回复
发新帖 我要提问
您需要登录后才可以回帖 登录 | 注册

本版积分规则

1

主题

11

帖子

1

粉丝