⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 eeprom.c

📁 TI(德州仪器公司)的CC2430芯片的库函数以及一些应用
💻 C
字号:
/******************************************************************************
*                                                                             *
*        **********                                                           *
*       ************                                                          *
*      ***        ***                                                         *
*     ***    ++    ***                                                        *
*     ***   +  +   ***                      CHIPCON                           *
*     ***   +                                                                 *
*     ***   +  +   ***                                                        *
*     ***    ++    ***                                                        *
*      ***        ***                                                         *
*       ************                                                          *
*        **********                                                           *
*                                                                             *
*******************************************************************************

Filename:     eeprom.c
Target:       cc2430
Author:       EFU
Revised:      16/12-2005
Revision:     1.0

Description:
Function implementations for common EEPROM functions for use with the CC2430DB.

******************************************************************************/


#include "eeprom.h"
#include "ioCC2430.h"
#include "hal.h"
#include "CC2430DB.h"

// Prototypes for internal EEPROM functions
BYTE writeEEPromBlock(BYTE* buffer, INT8 length, WORD address);
void smbEESend(BYTE *buffer, const UINT8 n);
BOOL smbEESendByte(BYTE b);
void smbEEWrite(BOOL value);
void smbEEReceive(BYTE address, UINT16 length, BYTE *buffer);
BYTE smbEEReceiveByte(BOOL lastByte);
BOOL EEPromWrapCheck(INT8 length, WORD address);
void smbEEClock(BOOL value);
void smbEEStart();
void smbEEStop();
void waitEE();


#define DATA_HIGH()    do{IO_DIR_PORT_PIN(0, 2, IO_IN);} while(0)
#define DATA_LOW()     do{IO_DIR_PORT_PIN(0, 2, IO_OUT); EE_SDA = 0;}while(0)

#define EEPROM_WRITEADDRESS   0xA0
#define EEPROM_READADDRESS    0xA1

#define PAGE_SIZE 32
// 32 k bit = 4096 BYTE
#define PROM_SIZE (0x1000)

#define WAIT_FOR_WRITE_DELAY 0x03


/******************************************************************************
* See eeprom.h for a description of this function.
******************************************************************************/
void initEEProm(void)
{
   // Setting the ports as inputs.
   IO_DIR_PORT_PIN(0, 2, IO_IN);
   IO_DIR_PORT_PIN(0, 3, IO_IN);

   // Setting P0_2 and P0_3 for general IO operation.
   IO_FUNC_PORT_PIN(0, 2, IO_FUNC_GIO);
   IO_FUNC_PORT_PIN(0, 3, IO_FUNC_GIO);

   // Setting ports for pull-up.
   IO_IMODE_PORT_PIN(0, 2, IO_IMODE_PUD);
   IO_IMODE_PORT_PIN(0, 3, IO_IMODE_PUD);
   IO_PUD_PORT(0, IO_PULLUP);
}


/******************************************************************************
* See eeprom.h for a description of this function.
******************************************************************************/
BOOL readEEProm(BYTE* buffer, WORD address, UINT8 length)
{
   BYTE i;
   BYTE transmitBuffer[3];

   transmitBuffer[0] = EEPROM_WRITEADDRESS;
   transmitBuffer[1] = (BYTE) (address >> 8);
   transmitBuffer[2] = (BYTE) (address);

   smbEEStart();
   for(i = 0; i < 3; i++){
      while(!smbEESendByte(transmitBuffer[i])); //Keep sending until ACK received
   }

   smbEEClock(0);
   waitEE();
   DATA_HIGH();
   smbEEClock(1);

   smbEEReceive(EEPROM_READADDRESS, length, buffer);

   return TRUE;
}


/******************************************************************************
* See eeprom.h for a description of this function.
******************************************************************************/
BYTE writeEEProm(BYTE* buffer, WORD address, UINT8 length)
{
   UINT8 i;
   UINT8 currentPageLength;
   UINT8 pageOffset;

   if(((WORD)address + length) > (WORD)PROM_SIZE){
      // Trying to write to a higher address than max address
      return FALSE;
   }

   if( pageOffset = (address & 0x1F) ){
      // Write to first page if address is not 32 bytes aligned
      if( pageOffset + length > PAGE_SIZE ){
         currentPageLength = PAGE_SIZE - pageOffset;
      }
      else{
         currentPageLength = length;
      }

      writeEEPromBlock(&buffer[0], currentPageLength, address);
      // Wait for internal write cycle to finish
      halWait(WAIT_FOR_WRITE_DELAY);
   }
   else{
      // Address is 32 bytes aligned
      pageOffset = 0;
      currentPageLength = 0;
   }

   for( i = currentPageLength; i < length; i += PAGE_SIZE ){
      if( i + PAGE_SIZE < length ) {
         currentPageLength = PAGE_SIZE;
      }
      else{
         currentPageLength = length - i;
      }
      writeEEPromBlock(&buffer[i], currentPageLength, address + i);
      // Wait for internal write cycle to finish
      halWait(WAIT_FOR_WRITE_DELAY);
   }

   return TRUE;
}


/******************************************************************************
* See eeprom.h for a description of this function.
* length must be less or equal 32
******************************************************************************/
BYTE writeEEPromBlock(BYTE* buffer, INT8 length, WORD address)
{
   UINT8 i = 0;
   UINT8 j;
   BYTE transmitBuffer[35];

   transmitBuffer[i++] = EEPROM_WRITEADDRESS;
   transmitBuffer[i++] = (BYTE) (address >> 8);
   transmitBuffer[i++] = (BYTE) (address);

   for(j = 0; j < length; j++)
   {
      transmitBuffer[i++] = buffer[j];
   }

   smbEESend(transmitBuffer, i);
   return TRUE;
}


/******************************************************************************
* See eeprom.h for a description of this function.
******************************************************************************/
void eraseEEProm(void)
{
   WORD j;
   UINT8 i;
   BYTE transmitBuffer[35];

   transmitBuffer[0] = EEPROM_WRITEADDRESS;
   for(i = 0; i < PAGE_SIZE; i++)
   {
      transmitBuffer[i + 3] = 0;
   }

   for(j = 0; j < PROM_SIZE; j += PAGE_SIZE)
   {
      transmitBuffer[1] = (BYTE) (j >> 8);
      transmitBuffer[2] = (BYTE) (j);

      smbEESend(transmitBuffer, 35);
      // Wait for internal write cycle to finish
      halWait(WAIT_FOR_WRITE_DELAY);
   }
}


/******************************************************************************
* Internal function for eeprom.c
******************************************************************************/
void smbEESend(BYTE *buffer, const UINT8 n){
   UINT8 i = 0;
   smbEEStart();
   for(i = 0; i < n; i++){
      while(!smbEESendByte(buffer[i])); //Keep sending until ACK received
   }
   smbEEClock(0);
   smbEEClock(1);
   DATA_LOW();
   smbEEStop();
}


/******************************************************************************
* Internal function for eeprom.c
******************************************************************************/
BOOL smbEESendByte(BYTE b){
   UINT8 i;
   BOOL ack;
   for (i = 0; i < 8; i++){
      smbEEWrite(b & 0x80);
      b = (b <<  1);
   }
   smbEEClock(0);
   DATA_LOW();
   smbEEClock(1);
   ack = !EE_SDA;
   return ack; //high = ACK received, else ACK not received
}


/******************************************************************************
* Internal function for eeprom.c
* Function for writing bit to the data line. Setting the port as input
* make the SMBus go high because of the pull-up resistors.
*
******************************************************************************/
void smbEEWrite(BOOL value){
   smbEEClock(0);
   if(value){
      DATA_HIGH();
   }
   else{
      DATA_LOW();
   }
   smbEEClock(1);
}


/******************************************************************************
* Internal function for eeprom.c
******************************************************************************/
void smbEEReceive(BYTE address, UINT16 length, BYTE *buffer){
   UINT16 i;

   smbEEStart();
   DATA_HIGH();
   //smbEEClock(1);

   while (!smbEESendByte(address)); //Keep sending address until ACK'ed
   smbEEClock(0);
   DATA_HIGH();

   // (CLK is high and data is high)
   for(i = 0; i < length; i++){
      buffer[i] = smbEEReceiveByte( i==(length-1) ); //TRUE if last byte
      halWait(0x01);
   }

   smbEEClock(1);
   smbEEStop();
}


/******************************************************************************
* Internal function for eeprom.c
******************************************************************************/
BYTE smbEEReceiveByte(BOOL lastByte){
   INT16 i;
   BYTE inData = 0;

   DATA_HIGH();

   for(i = 0; i < 8 * 2; i++){
      if(EE_SCL == 1){ 	//clk high
         if(EE_SDA){ 	//read '1'
            inData = (inData<<1) | 0x01;
         } else {
            inData = (inData<<1) & ~0x01;
         }
         smbEEClock(0);
      }
      else{
         smbEEClock(1);
      }
   }
   smbEEClock(0);

   //do not ACK if last byte
   //lastByte ? smbEEWrite(1) :  smbEEWrite(0);

   DATA_LOW();
   smbEEClock(1);
   smbEEClock(0);

   return inData;
}


/******************************************************************************
* Internal function for eeprom.c
* This function is used to clock the SMBus connected to the LCD. If a negative
* edge is to follow, the pin is set as an output and driven low. If a positive
* edge is to follow, the pin is set as an input and the pull-up resistor is
* to drive the node high. This way the slave device can hold the node low if
* a longer setup time is desired.
*
******************************************************************************/
void smbEEClock(BOOL value){
   if(!value){
      IO_DIR_PORT_PIN(0, 3, IO_OUT);
      EE_SCL = 0;
   }
   else {
      IO_DIR_PORT_PIN(0, 3, IO_IN);
   }
   waitEE();
}

/******************************************************************************
* Internal function for eeprom.c
* This function initiates SMBus communication. It makes sure that both the
* data and the clock of the SMBus are high. Then the data pin is set low
* while the clock is kept high. This initializes SMBus transfer.
*
******************************************************************************/
void smbEEStart(){
   while (! (EE_SDA && EE_SCL) )
   {
      DATA_HIGH();
      smbEEClock(0);
      smbEEClock(1);
   }//wait for Data and clk high
   DATA_LOW();
   waitEE();
   smbEEClock(0);
}

/******************************************************************************
* Internal function for eeprom.c
* This function terminates SMBus communication. It makes sure that the data
* and clock of the SMBus are low and high, respectively. Then the data pin is
* set high while the clock is kept high. This terminates SMBus transfer.
*
******************************************************************************/
void smbEEStop(){
   while (! (!EE_SDA && EE_SCL))
   {
      DATA_LOW();
      smbEEClock(0);
      smbEEClock(1);
   }
   DATA_HIGH();
   smbEEClock(1);
   waitEE();
}


/******************************************************************************
* Internal function for eeprom.c
******************************************************************************/
void waitEE(){
   UINT8 j = 0x01;
   while(j--);
}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -