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

📄 i2csupport.c

📁 Zigbee2006入门(源代码+文档讲解+系统推荐)
💻 C
字号:
/**************************************************************************************************
    Filename:       i2csupport.c
    Revised:        $Date: 2006-10-19 13:43:54 -0700 (Thu, 19 Oct 2006) $
    Revision:       $Revision: 12357 $

    Description:

    This file contains the I2C interface to off-chip serial EEPROM. While
    much of the driver is generic, certain portions are specific to the
    Atmel part AT24C1024. For example, the 17th address bit occurs in the
    device address byte where one of the address pins should be. Different
    1 Mb parts use a different bit location. This could be abstracted at
    a later time if this is the only differentce among these parts.

    Copyright (c) 2006 by Texas Instruments, Inc.
    All Rights Reserved.  Permission to use, reproduce, copy, prepare
    derivative works, modify, distribute, perform, display or sell this
    software and/or its documentation for any purpose is prohibited
    without the express written consent of Texas Instruments, Inc.
**************************************************************************************************/
#include "ioCC2430.h"
#include "zcomdef.h"
#include "i2cSupport.h"
#include "flashutils.h"

#define STATIC  static

// the default cofiguration below usses P1.3 for SDA and P1.2 for SCL.
// change these as needed.
#ifndef OCM_CLK_PORT
  #define OCM_CLK_PORT  1
#endif

#ifndef OCM_DATA_PORT
  #define OCM_DATA_PORT   1
#endif

#ifndef OCM_CLK_PIN
  #define OCM_CLK_PIN   2
#endif

#ifndef OCM_DATA_PIN
  #define OCM_DATA_PIN    3
#endif

// General I/O definitions
#define IO_GIO  0  // General purpose I/O
#define IO_PER  1  // Peripheral function
#define IO_IN   0  // Input pin
#define IO_OUT  1  // Output pin
#define IO_PUD  0  // Pullup/pulldn input
#define IO_TRI  1  // Tri-state input
#define IO_PUP  0  // Pull-up input pin
#define IO_PDN  1  // Pull-down input pin

#define OCM_ADDRESS  (0xA0)
#define OCM_READ     (0x01)
#define OCM_WRITE    (0x00)
#define SMB_ACK      (0)
#define SMB_NAK      (1)
#define SEND_STOP    (0)
#define NOSEND_STOP  (1)
#define SEND_START   (0)
#define NOSEND_START (1)

// device specific as to where the 17th address bit goes...
#define OCM_ADDRESS_BYTE(target,rw)  (OCM_ADDRESS | rw | ((target > 0xFFFF) ? 0x02 : 0x00))

// *************************   MACROS   ************************************
#undef P

  /* I/O PORT CONFIGURATION */
#define CAT1(x,y) x##y  // Concatenates 2 strings
#define CAT2(x,y) CAT1(x,y)  // Forces evaluation of CAT1

// OCM port I/O defintions
// Builds I/O port name: PNAME(1,INP) ==> P1INP
#define PNAME(y,z) CAT2(P,CAT2(y,z))
// Builds I/O bit name: BNAME(1,2) ==> P1_2
#define BNAME(port,pin) CAT2(CAT2(P,port),CAT2(_,pin))

// OCM port I/O defintions
#define OCM_SCL BNAME(OCM_CLK_PORT, OCM_CLK_PIN)
#define OCM_SDA BNAME(OCM_DATA_PORT, OCM_DATA_PIN)

#define IO_DIR_PORT_PIN(port, pin, dir) \
{\
  if ( dir == IO_OUT ) \
    PNAME(port,DIR) |= (1<<(pin)); \
  else \
    PNAME(port,DIR) &= ~(1<<(pin)); \
}

#define OCM_DATA_HIGH()\
{ \
  IO_DIR_PORT_PIN(OCM_DATA_PORT, OCM_DATA_PIN, IO_IN); \
}

#define OCM_DATA_LOW() \
{ \
  IO_DIR_PORT_PIN(OCM_DATA_PORT, OCM_DATA_PIN, IO_OUT); \
  OCM_SDA = 0;\
}

#define IO_FUNC_PORT_PIN(port, pin, func) \
{ \
  if( port < 2 ) \
  { \
    if ( func == IO_PER ) \
      PNAME(port,SEL) |= (1<<(pin)); \
    else \
      PNAME(port,SEL) &= ~(1<<(pin)); \
  } \
  else \
  { \
    if ( func == IO_PER ) \
      P2SEL |= (1<<(pin>>1)); \
    else \
      P2SEL &= ~(1<<(pin>>1)); \
  } \
}

#define IO_IMODE_PORT_PIN(port, pin, mode) \
{ \
  if ( mode == IO_TRI ) \
    PNAME(port,INP) |= (1<<(pin)); \
  else \
    PNAME(port,INP) &= ~(1<<(pin)); \
}

#define IO_PUD_PORT(port, dir) \
{ \
  if ( dir == IO_PDN ) \
    P2INP |= (1<<(port+5)); \
  else \
    P2INP &= ~(1<<(port+5)); \
}

STATIC void   smbSend( uint8 *buffer, uint16 len, uint8 sendStart, uint8 sendStop );
STATIC _Bool  smbSendByte( uint8 dByte );
STATIC void   smbWrite( bool dBit );
STATIC void   smbClock( bool dir );
STATIC void   smbStart( void );
STATIC void   smbStop( void );
STATIC void   smbReceive( uint32 address, uint8 *buffer, uint16 len );
STATIC uint8  smbReceiveByte( void );
STATIC _Bool  smbRead( void );
STATIC void   smbSendDeviceAddress(uint32 address);
STATIC int8   SMBReceive(uint32 address, uint8 *buf, uint16 len);
STATIC int8   SMBSend(uint32 address, uint8 *buf, uint16 len);
STATIC int8   SMBReadWrite(uint8 which, uint32 address, uint8 *buf, uint16 len);


STATIC __near_func void   smbWait( uint8 );

STATIC uint8 s_xmemIsInit;

/*********************************************************************
 * @fn      DF_i2cInit
 * @brief   Initializes two-wire serial I/O bus
 * @param   void
 * @return  void
 */
void DF_i2cInit( int8 (**readWrite)(uint8, uint32, uint8 *, uint16))
{
  if (!s_xmemIsInit)  {
    s_xmemIsInit = 1;

    // Set port pins as inputs
    IO_DIR_PORT_PIN( OCM_CLK_PORT, OCM_CLK_PIN, IO_IN );
    IO_DIR_PORT_PIN( OCM_DATA_PORT, OCM_DATA_PIN, IO_IN );

    // Set for general I/O operation
    IO_FUNC_PORT_PIN( OCM_CLK_PORT, OCM_CLK_PIN, IO_GIO );
    IO_FUNC_PORT_PIN( OCM_DATA_PORT, OCM_DATA_PIN, IO_GIO );

    // Set I/O mode for pull-up/pull-down
    IO_IMODE_PORT_PIN( OCM_CLK_PORT, OCM_CLK_PIN, IO_PUD );
    IO_IMODE_PORT_PIN( OCM_DATA_PORT, OCM_DATA_PIN, IO_PUD );

    // Set pins to pull-up
    IO_PUD_PORT( OCM_CLK_PORT, IO_PUP );
    IO_PUD_PORT( OCM_DATA_PORT, IO_PUP );
  }

  *readWrite = SMBReadWrite;
}

int8 SMBReadWrite(uint8 which, uint32 address, uint8 *buf, uint16 len)
{
  if (which == XMEM_READ)  {
    return SMBReceive(address, buf, len);
  }
  else  {
    return SMBSend(address, buf, len);
  }
}

int8 SMBReceive(uint32 address, uint8 *buf, uint16 len)
{
  smbReceive(address, buf, len);

  return 0;
}

int8 SMBSend(uint32 address, uint8 *buf, uint16 len)
{
  uint16 rollover = (uint16)address;
  uint16  locLen;

  // The AT24C1024 writes 256 byte 'pages'. If the starting address
  // plus the number of bytes to write exceed the 256 byte page boundary
  // the address will wrap and overwrite from the beginning of the page.
  // Presumably this is not what is intended under this scenario. We
  // have to deal with rollover explicitly. If it occurs we need to
  // re-address the target byte so we need to stop and re-start.
  do  {
    if (((rollover+len-1)&0xFF00) != (rollover&0xFF00))  {
      // We're going to exceed the current page. Write only
      // enough bytes to fill the current page.
      locLen = 0x100 - (address & 0xFF);
    }
    else  {
      locLen = len;
    }
    // begin the write sequence with the address byte
    smbSendDeviceAddress(address);
    smbSend(buf, locLen, NOSEND_START, SEND_STOP);
    len      -= locLen;
    address  += locLen;
    rollover += locLen;
    buf      += locLen;
  } while (len);

  return 0;
}
/*********************************************************************
 * @fn      smbSend
 * @brief   Sends buffer contents to SM-Bus device
 * @param   buffer - ptr to buffered data to send
 * @param   len - number of bytes in buffer
 * @param   sendStart - whether or not to send start condition.
 * @param   sendStop - whether or not to send stop condition.
 * @return  void
 */
STATIC void smbSend( uint8 *buffer, uint16 len, uint8 sendStart, uint8 sendStop )
{
  uint16 i;

  if (!len)  {
    return;
  }

  if (sendStart == SEND_START)  {
    smbStart();
  }

  for ( i = 0; i < len; i++ )
  {
    while ( !smbSendByte( buffer[i] ) );  // takes care of ack polling
  }

  if (sendStop == SEND_STOP) {
    smbStop();
  }
}

/*********************************************************************
 * @fn      smbSendByte
 * @brief   Serialize and send one byte to SM-Bus device
 * @param   dByte - data byte to send
 * @return  ACK status - 0=none, 1=received
 */
STATIC _Bool smbSendByte( uint8 dByte )
{
  uint8 i;

  for ( i = 0; i < 8; i++ )
  {
    // Send the MSB
    smbWrite( dByte & 0x80 );
    // Next bit into MSB
    dByte <<= 1;
  }
  // need clock low so if the SDA transitions on the next statement the
  // slave doesn't stop. Also give opportunity for slave to set SDA
  smbClock( 0 );
  OCM_DATA_HIGH(); // set to input to receive ack...
  smbClock( 1 );
  smbWait(1);

  return ( !OCM_SDA );  // Return ACK status
}

/*********************************************************************
 * @fn      smbWrite
 * @brief   Send one bit to SM-Bus device
 * @param   dBit - data bit to clock onto SM-Bus
 * @return  void
 */
STATIC void smbWrite( bool dBit )
{
  smbClock( 0 );
  smbWait(1);
  if ( dBit )
  {
    OCM_DATA_HIGH();
  }
  else
  {
    OCM_DATA_LOW();
  }

  smbClock( 1 );
  smbWait(1);
}

/*********************************************************************
 * @fn      smbClock
 * @brief   Clocks the SM-Bus. If a negative edge is going out, the
 *          I/O pin is set as an output and driven low. If a positive
 *          edge is going out, the pin is set as an input and the pin
 *          pull-up drives the line high. This way, the slave device
 *          can hold the node low if longer setup time is desired.
 * @param   dir - clock line direction
 * @return  void
 */
STATIC void smbClock( bool dir )
{
  if ( dir )
  {
    IO_DIR_PORT_PIN( OCM_CLK_PORT, OCM_CLK_PIN, IO_IN );
  }
  else
  {
    IO_DIR_PORT_PIN( OCM_CLK_PORT, OCM_CLK_PIN, IO_OUT );
    OCM_SCL = 0;
  }
  smbWait(1);
}

/*********************************************************************
 * @fn      smbStart
 * @brief   Initiates SM-Bus communication. Makes sure that both the
 *          clock and data lines of the SM-Bus are high. Then the data
 *          line is set high and clock line is set low to start I/O.
 * @param   void
 * @return  void
 */
STATIC void smbStart( void )
{
  // wait for slave to release clock line...
  // set SCL to input but with pull-up. if slave is pulling down it will stay down.
  smbClock(1);
  while (!OCM_SCL) ; // wait until the line is high...

  // SCL low to set SDA high so the transition will be correct.
  smbClock(0);
  OCM_DATA_HIGH();  // SDA high
  smbClock(1);      // set up for transition
  smbWait(1);
  OCM_DATA_LOW();   // start

  smbWait(1);
  smbClock( 0 );
}

/*********************************************************************
 * @fn      smbStop
 * @brief   Terminates SM-Bus communication. Waits unitl the data line
 *          is low and the clock line is high. Then sets the data line
 *          high, keeping the clock line high to stop I/O.
 * @param   void
 * @return  void
 */
STATIC void smbStop( void )
{
  // Wait for clock high and data low
  smbClock(0);
  OCM_DATA_LOW();  // force low with SCL low
  smbWait(1);

  smbClock( 1 );
  OCM_DATA_HIGH(); // stop condition
  smbWait(1);

}

/*********************************************************************
 * @fn      smbWait
 * @brief   Wastes a an amount of time.
 * @param   count: down count in busy-wait
 * @return  void
 */
STATIC __near_func void smbWait( uint8 count )
{
  while ( count-- );
}

/*********************************************************************
 * @fn      smbReceiveByte
 * @brief   Read the 8 data bits.
 * @param   void
 * @return  character read
 */
STATIC uint8 smbReceiveByte()
{
  int8 i, rval = 0;

  for (i=7; i>=0; --i)  {
    if (smbRead())  {
      rval |= 1<<i;
    }
  }

  return rval;
}
/**************************************************************************************************
**************************************************************************************************/
/*********************************************************************
 * @fn      smbReceive
 * @brief   reads data into a buffer
 * @param   address: linear address on part from which to read
 * @param   buffer: target array for read characters
 * @param   len: max number of characters to read
 * @return  void
 */
STATIC void smbReceive( uint32 address, uint8 *buffer, uint16 len )
{
  uint8  ch;
  uint16 i;

  if (!len)  {
    return;
  }

  smbSendDeviceAddress(address);

  ch = OCM_ADDRESS_BYTE(0, OCM_READ);
  smbSend(&ch, 1, SEND_START, NOSEND_STOP);

  for ( i = 0; i < len-1; i++ )
  {
    // SCL may be high. set SCL low. If SDA goes high when input
    // mode is set the slave won't see a STOP
    smbClock(0);
    OCM_DATA_HIGH();

    buffer[i] = smbReceiveByte();
    smbWrite(SMB_ACK);           // write leaves SCL high
  }

  // condition SDA one more time...
  smbClock(0);
  OCM_DATA_HIGH();
  buffer[i] = smbReceiveByte();
  smbWrite(SMB_NAK);

  smbStop();
}

/*********************************************************************
 * @fn      smbRead
 * @brief   Toggle the clock line to let the slave set the data line.
 *          Then read the data line.
 * @param   void
 * @return  TRUE if bit read is 1 else FALSE
 */
STATIC _Bool smbRead( void )
{
  // SCL low to let slave set SDA. SCL high for SDA
  // valid and then get bit
  smbClock( 0 );
  smbWait(1);
  smbClock( 1 );
  smbWait(1);

  return OCM_SDA;
}

/*********************************************************************
 * @fn      smbSendDeviceAddress
 * @brief   Send onlythe device address. Do ack polling
 *
 * @param   void
 * @return  none
 */
STATIC void smbSendDeviceAddress(uint32 address)
{
  uint8 addr[2];

  addr[0] = (address>>8) & 0xFF;
  addr[1] = address & 0xFF;

  // do ack polling...
  do  {
    smbStart();
  } while (!smbSendByte(OCM_ADDRESS_BYTE(address, OCM_WRITE)));

  // OK. send memory address
  smbSend(addr, 2, NOSEND_START, NOSEND_STOP);
}

⌨️ 快捷键说明

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