将数据存储在EEPROM,MICROCONTROLLER中

时间:2016-03-22 13:01:10

标签: c memory microcontroller eeprom

我正在使用MCF51EM256飞思卡尔微控制器,而且我将一些数据存储在EEPROM中以使其持久存在。

我需要将这个结构存储在EEPROM中:

typedef struct {
  ui64_s Ea_ps; 
  ui64_s Ea_ng;  
  ui64_s Er_q1;
  ui64_s Er_q2;
  ui64_s Er_q3;
  ui64_s Er_q4;
  uint16 F_ea;
  uint16 F_er;
}Ws_EnergyAcc64;

其中:

typedef union{    
  uint64 v;
  uint32 p[2];  
} ui64_s;

typedef unsigned long long int uint64;
typedef unsigned long int uint32;
typedef unsigned short int uint16;

为了做到这一点,我实现了这个功能:

void Save_EEPROM_WsEnergyAcc(long addr, Ws_EnergyAcc64* Acc) {

  // WsEnergyAcc struct needs 56 bytes in EEPROM

  uint32 F_ea_32 = (uint32) Acc->F_ea;
  uint32 F_er_32 = (uint32) Acc->F_er;

  Flash_Burst(addr, 2, Acc->Ea_ps.p);
  Flash_Burst(addr + 8, 2, Acc->Ea_ng.p);
  Flash_Burst(addr + 16, 2, Acc->Er_q1.p);
  Flash_Burst(addr + 24, 2, Acc->Er_q2.p);
  Flash_Burst(addr + 32, 2, Acc->Er_q3.p);
  Flash_Burst(addr + 40, 2, Acc->Er_q4.p);

  Flash_Burst(addr + 48, 2, &F_ea_32);
  Flash_Burst(addr + 52, 2, &F_er_32);

}

其中" Flash_Burst":

#define Flash_Burst(Address, Size, DataPtr) \
      Flash_Cmd((UINT32)Address, (UINT16)Size, (UINT32*)DataPtr, FLASH_BURST_CMD)

UINT8 /*far*/ 
Flash_Cmd(UINT32 FlashAddress, 
      UINT16 FlashDataCounter, 
      UINT32 *pFlashDataPtr, 
      UINT8 FlashCommand)
{
  /* Check to see if FACCERR or PVIOL is set */
  if (FSTAT &0x30)  
  {         
      /* Clear Flags if set*/
      FSTAT = 0x30;  
  }

  if (FlashDataCounter)
  {
    do
    {
        /* Wait for the Last Busrt Command to complete */
        while(!(FSTAT&FSTAT_FCBEF_MASK)){};/*wait until termination*/

        /* Write Data into Flash*/
        (*((volatile unsigned long *)(FlashAddress))) = *pFlashDataPtr;
        FlashAddress += 4;
        pFlashDataPtr++;

        /* Write Command */
        FCMD = FlashCommand;

        /* Put FCBEF at 1 */
        FSTAT = FSTAT_FCBEF_MASK;

        asm (NOP);
        asm (NOP);
        asm (NOP);

         /* Check if Flash Access Error or Protection Violation Error are Set */
        if (FSTAT&0x30)
        {     
          /* If so, finish the function returning 1 to indicate error */
          return (1);
        }

    }while (--FlashDataCounter);
  }
  /* wait for the last command to complete */
  while ((FSTAT&FSTAT_FCCF_MASK)==0){};/*wait until termination*/

  /* Return zero to indicate that the function executed OK */
  return (0);
}

void Init_EEPROM_WsEnergyAcc(long addr, Ws_EnergyAcc64* Acc) {

  uint32 F_ea_32;
  uint32 F_er_32;

  Acc->Ea_ps.p[0] = *(uint32 *)addr;
  addr = addr + 4;
  Acc->Ea_ps.p[1] = *(uint32 *)addr;
  Acc->Ea_ps.v = (uint64) Acc->Ea_ps.p[0] << 32 | Acc->Ea_ps.p[1];

  addr = addr + 4;
  Acc->Ea_ng.p[0] = *(uint32 *)addr;
  addr = addr + 4;
  Acc->Ea_ng.p[1] = *(uint32 *)addr;
  Acc->Ea_ng.v = (uint64) Acc->Ea_ng.p[0] << 32 | Acc->Ea_ng.p[1];

  addr = addr + 4;
  Acc->Er_q1.p[0] = *(uint32*)addr;
  addr = addr + 4;
  Acc->Er_q1.p[1] = *(uint32*)addr;
  Acc->Er_q1.v = (uint64) Acc->Er_q1.p[0] << 32 | Acc->Er_q1.p[1];

  addr = addr + 4;
  Acc->Er_q2.p[0] = *(uint32*)addr;
  addr = addr + 4;
  Acc->Er_q2.p[1] = *(uint32*)addr;
  Acc->Er_q2.v = (uint64) Acc->Er_q2.p[0] << 32 | Acc->Er_q2.p[1];

  addr = addr + 4;
  Acc->Er_q3.p[0] = *(uint32*)addr;
  addr = addr + 4;
  Acc->Er_q3.p[1] = *(uint32*)addr;
  Acc->Er_q3.v = (uint64) Acc->Er_q3.p[0] << 32 | Acc->Er_q3.p[1];  

  addr = addr + 4;
  Acc->Er_q4.p[0] = *(uint32*)addr;
  addr = addr + 4;
  Acc->Er_q4.p[1] = *(uint32*)addr;
  Acc->Er_q4.v = (uint64) Acc->Er_q4.p[0] << 32 | Acc->Er_q4.p[1];  

  addr = addr + 4;
  F_ea_32 = *(uint32*) addr;
  Acc->F_ea = (uint16) F_ea_32;

  addr = addr + 4;
  F_er_32 = *(uint32*) addr;
  Acc->F_er = (uint16) F_er_32;

}

为了测试我的功能,我做了一个小测试程序:

void testEEPROM(Ws_EnergyAcc64* Acc){

      Ws_EnergyAcc64 checkStruct;

      Acc->Ea_ps.p[0] = 0x10000000;
      Acc->Ea_ps.p[1] = 0x10000000;
      Acc->Ea_ng.p[0] = 0x10000000;
      Acc->Ea_ng.p[1] = 0x10000000;
      Acc->Er_q1.p[0] = 0x10000000;
      Acc->Er_q1.p[1] = 0x10000000;
      Acc->Er_q2.p[0] = 0x10000000;
      Acc->Er_q2.p[1] = 0x10000000;
      Acc->Er_q3.p[0] = 0x10000000;
      Acc->Er_q3.p[1] = 0x10000000;
      Acc->Er_q4.p[0] = 0x10000000;
      Acc->Er_q4.p[1] = 0x10000000;
      Acc->F_ea = 0x0000;
      Acc->F_er = 0x0000;

     Save_EEPROM_WsEnergyAcc(PhR_ABS_Ws_addr, Acc);

     Acc->Ea_ps.p[0] = 0x10011001;
     Acc->Ea_ps.p[1] = 0x10011002;
     Acc->Ea_ng.p[0] = 0x10011003;
     Acc->Ea_ng.p[1] = 0x10011004;
     Acc->Er_q1.p[0] = 0x10011005;
     Acc->Er_q1.p[1] = 0x10011006;
     Acc->Er_q2.p[0] = 0x10011007;
     Acc->Er_q2.p[1] = 0x10011008;
     Acc->Er_q3.p[0] = 0x10011009;
     Acc->Er_q3.p[1] = 0x1001100A;
     Acc->Er_q4.p[0] = 0x1001100B;
     Acc->Er_q4.p[1] = 0x1001100C;
     Acc->F_ea = 0x0000;
     Acc->F_er = 0x0000;

     Save_EEPROM_WsEnergyAcc(PhR_ABS_Ws_addr, Acc);     
     Init_EEPROM_WsEnergyAcc(PhR_ABS_Ws_addr, &checkStruct);

     printf("%d\n", checkStruct.Ea_ps.p[0]);
     printf("%d\n", checkStruct.Ea_ps.p[1]);
     printf("%d\n", checkStruct.Ea_ng.p[0]);
     printf("%d\n", checkStruct.Ea_ng.p[1]);
     printf("%d\n", checkStruct.Er_q1.p[0]);
     printf("%d\n", checkStruct.Er_q1.p[1]);
     printf("%d\n", checkStruct.Er_q2.p[0]);
     printf("%d\n", checkStruct.Er_q2.p[1]);
     printf("%d\n", checkStruct.Er_q3.p[0]);
     printf("%d\n", checkStruct.Er_q3.p[1]);
     printf("%d\n", checkStruct.Er_q4.p[0]);
     printf("%d\n", checkStruct.Er_q4.p[1]);
}

如果我只在EEPROM写一次,它可以工作,但是如果我写EEPROM多次,就像例子一样,它不起作用,打印总是0x1000000。

有人可以帮助我吗?为什么不起作用?如果我能写一次,为什么我不能再写一次?

谢谢大家!

1 个答案:

答案 0 :(得分:1)

每当你想写一个EEPROM时,你就会删除它。第一次它起作用,可能是因为JTAG执行了大量擦除。

另请注意,现代MCU中的EEPROM是模拟的,并使用片上闪存的一部分。这更好地解释了为什么你必须在写之前擦除它。