Hello,
It doesn't seem that we're able to ready multiple bytes from a i2C slave device from offsets other than zero.
Writing registers to our i2c device works fine, but when we want to read multiple bytes from a specific address, it always reads from an offset of zero. With a previous slave i2C device (MMA8453Q accelerometer), we got lucky since the registers we wanted to read just happened to be at 0x01, 0x02, and 0x03, (so we just read 4 bytes) and we threw away the first one, but now, that is no longer the case and we'd really like to get this to work.
Actually, I believe we saw the same problem even for regular one-byte reads too. We think the problem is somewhere in how the reg byte gets written first, then we read the value at the offset of reg.
See code snippet below from our driver file for the mma8453. And attached hal_i2c.c that we pulled in, from some example code we found for a different TI chip, I believe.
/************************************************************************************************** Filename: hal_i2c.c Revised: $Date: 2012-09-21 06:30:38 -0700 (Fri, 21 Sep 2012) $ Revision: $Revision: 31581 $ Description: This module defines the HAL I2C API for the CC2541ST. It implements the I2C master. Copyright 2012 Texas Instruments Incorporated. All rights reserved. IMPORTANT: Your use of this Software is limited to those specific rights granted under the terms of a software license agreement between the user who downloaded the software, his/her employer (which must be your employer) and Texas Instruments Incorporated (the "License"). You may not use this Software unless you agree to abide by the terms of the License. The License limits your use, and you acknowledge, that the Software may not be modified, copied or distributed unless embedded on a Texas Instruments microcontroller or used solely and exclusively in conjunction with a Texas Instruments radio frequency transceiver, which is integrated into your product. Other than for the foregoing purpose, you may not use, reproduce, copy, prepare derivative works of, modify, distribute, perform, display or sell this Software and/or its documentation for any purpose. YOU FURTHER ACKNOWLEDGE AND AGREE THAT THE SOFTWARE AND DOCUMENTATION ARE PROVIDED �AS IS� WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING WITHOUT LIMITATION, ANY WARRANTY OF MERCHANTABILITY, TITLE, NON-INFRINGEMENT AND FITNESS FOR A PARTICULAR PURPOSE. IN NO EVENT SHALL TEXAS INSTRUMENTS OR ITS LICENSORS BE LIABLE OR OBLIGATED UNDER CONTRACT, NEGLIGENCE, STRICT LIABILITY, CONTRIBUTION, BREACH OF WARRANTY, OR OTHER LEGAL EQUITABLE THEORY ANY DIRECT OR INDIRECT DAMAGES OR EXPENSES INCLUDING BUT NOT LIMITED TO ANY INCIDENTAL, SPECIAL, INDIRECT, PUNITIVE OR CONSEQUENTIAL DAMAGES, LOST PROFITS OR LOST DATA, COST OF PROCUREMENT OF SUBSTITUTE GOODS, TECHNOLOGY, SERVICES, OR ANY CLAIMS BY THIRD PARTIES (INCLUDING BUT NOT LIMITED TO ANY DEFENSE THEREOF), OR OTHER SIMILAR COSTS. Should you have any questions regarding your right to use this Software, contact Texas Instruments Incorporated at www.TI.com. **************************************************************************************************/ /* ------------------------------------------------------------------------------------------------ * Includes * ------------------------------------------------------------------------------------------------ */ #include "hal_assert.h" #include "hal_board_cfg.h" #include "hal_i2c.h" /* ------------------------------------------------------------------------------------------------ * Constants * ------------------------------------------------------------------------------------------------ */ #define I2C_ENS1 BV(6) #define I2C_STA BV(5) #define I2C_STO BV(4) #define I2C_SI BV(3) #define I2C_AA BV(2) #define I2C_MST_RD_BIT BV(0) // Master RD/WRn bit to be OR'ed with Slave address. #define I2C_CLOCK_MASK 0x83 #define I2C_PXIFG P2IFG #define I2C_IF P2IF #define I2C_IE BV(1) /* ------------------------------------------------------------------------------------------------ * Typedefs * ------------------------------------------------------------------------------------------------ */ typedef enum { // HAL_I2C_MASTER mode statuses. mstStarted = 0x08, mstRepStart = 0x10, mstAddrAckW = 0x18, mstAddrNackW = 0x20, mstDataAckW = 0x28, mstDataNackW = 0x30, mstLostArb = 0x38, mstAddrAckR = 0x40, mstAddrNackR = 0x48, mstDataAckR = 0x50, mstDataNackR = 0x58, } i2cStatus_t; /* ------------------------------------------------------------------------------------------------ * Macros * ------------------------------------------------------------------------------------------------ */ //bgk The I2C module is configured as an I2C master //bgk by setting the I2CCFG.ENS1 and I2CCFG.STA bits. #define I2C_WRAPPER_DISABLE() st( I2CWC = 0x00; ) #define I2C_CLOCK_RATE(x) st( I2CCFG &= ~I2C_CLOCK_MASK; \ I2CCFG |= x; ) #define I2C_SET_NACK() st( I2CCFG &= ~I2C_AA; ) #define I2C_SET_ACK() st( I2CCFG |= I2C_AA; ) // Enable I2C bus #define I2C_ENABLE() st( I2CCFG |= (I2C_ENS1); ) #define I2C_DISABLE() st( I2CCFG &= ~(I2C_ENS1); ) // Must clear SI before setting STA and then STA must be manually cleared. #define I2C_STRT() st ( \ I2CCFG &= ~I2C_SI; \ I2CCFG |= I2C_STA; \ while ((I2CCFG & I2C_SI) == 0); \ I2CCFG &= ~I2C_STA; \ ) // Must set STO before clearing SI. #define I2C_STOP() st ( \ I2CCFG |= I2C_STO; \ I2CCFG &= ~I2C_SI; \ while ((I2CCFG & I2C_STO) != 0); \ ) // Stop clock-stretching and then read when it arrives. #define I2C_READ(_X_) st ( \ I2CCFG &= ~I2C_SI; \ while ((I2CCFG & I2C_SI) == 0); \ (_X_) = I2CDATA; \ ) // First write new data and then stop clock-stretching. #define I2C_WRITE(_X_) st ( \ I2CDATA = (_X_); \ I2CCFG &= ~I2C_SI; \ while ((I2CCFG & I2C_SI) == 0); \ ) /* ------------------------------------------------------------------------------------------------ * Local Variables * ------------------------------------------------------------------------------------------------ */ static uint8 i2cAddr; // Target Slave address pre-shifted up by one leaving RD/WRn LSB as zero. /************************************************************************************************** * @fn i2cMstStrt * * @brief Attempt to send an I2C bus START and Slave Address as an I2C bus Master. * * input parameters * * @param RD_WRn - The LSB of the Slave Address as Read/~Write. * * @return The I2C status of the START request or of the Slave Address Ack. */ static uint8 i2cMstStrt(uint8 RD_WRn) { I2C_STRT(); if (I2CSTAT == mstStarted) /* A start condition has been transmitted */ { I2C_WRITE(i2cAddr | RD_WRn); } return I2CSTAT; } /************************************************************************************************** * @fn HalI2CInit * * @brief Initialize the I2C bus as a Master. * * input parameters * * @param address - I2C slave address. * @param clockRate - I2C clock rate. * * output parameters * * None. * * @return None. */ void HalI2CInit(uint8 address, i2cClock_t clockRate) { i2cAddr = address << 1; I2C_WRAPPER_DISABLE(); I2CADDR = 0; // no multi master support at this time I2C_CLOCK_RATE(clockRate); I2C_ENABLE(); } /************************************************************************************************** * @fn HalI2CRead * * @brief Read from the I2C bus as a Master. * * input parameters * * @param len - Number of bytes to read. * @param pBuf - Pointer to the data buffer to put read bytes. * * output parameters * * None. * * @return The number of bytes successfully read. */ uint8 HalI2CRead(uint8 len, uint8 *pBuf) { uint8 cnt = 0; if (i2cMstStrt(I2C_MST_RD_BIT) != mstAddrAckR) { len = 0; } // All bytes are ACK'd except for the last one which is NACK'd. If only // 1 byte is being read, a single NACK will be sent. Thus, we only want // to enable ACK if more than 1 byte is going to be read. if (len > 1) { I2C_SET_ACK(); } while (len > 0) { // slave devices require NACK to be sent after reading last byte if (len == 1) { I2C_SET_NACK(); } // read a byte from the I2C interface I2C_READ(*pBuf++); cnt++; len--; if (I2CSTAT != mstDataAckR) { if (I2CSTAT != mstDataNackR) { // something went wrong, so don't count last byte cnt--; } break; } } I2C_STOP(); return cnt; } /************************************************************************************************** * @fn HalI2CWrite * * @brief Write to the I2C bus as a Master. * * input parameters * * @param len - Number of bytes to write. * @param pBuf - Pointer to the data buffer to write. * * output parameters * * None. * * @return The number of bytes successfully written. */ uint8 HalI2CWrite(uint8 len, uint8 *pBuf) { if (i2cMstStrt(0) != mstAddrAckW) { len = 0; } for (uint8 cnt = 0; cnt < len; cnt++) { I2C_WRITE(*pBuf++); if (I2CSTAT != mstDataAckW) { if (I2CSTAT == mstDataNackW) { len = cnt + 1; } else { len = cnt; } break; } } I2C_STOP(); return len; } uint8 HalI2CWrite2(uint8 len, uint8 *pBuf) { if (i2cMstStrt(0) != mstAddrAckW) { len = 0; } for (uint8 cnt = 0; cnt < len; cnt++) { I2C_WRITE(*pBuf++); if (I2CSTAT != mstDataAckW) { if (I2CSTAT == mstDataNackW) { len = cnt + 1; } else { len = cnt; } break; } } //I2C_STOP(); return len; } /************************************************************************************************** * @fn HalI2CDisable * * @brief Places the I2C bus in inactive mode * * input parameters * * None. * * output parameters * * None. * * @return None. */ void HalI2CDisable(void) { I2C_DISABLE(); } /********************************************************************* *********************************************************************/
Any thoughts would be super helpful!
Many thanks guys if anyone has any suggestions!
Brian
In file mma8453.c (our driver)
/**************************************************************************************************
* @fn ReadReg
*
* @brief This function implements the I2C protocol to read from an I2C reigster(s)
*
* @param reg - which register to read
* @param pBuf - pointer to buffer to place data
* @param nBytes - numbver of bytes to read
*
* @return TRUE if the required number of bytes are reveived
**************************************************************************************************/
bool ReadReg(uint8 reg, uint8 *pBuf, uint8 nBytes)
{
uint8 i = 0;
/* Send address we're reading from */
HalI2CInit(MMA8453_ADDRESS,i2cClock_123KHZ);
if (HalI2CWrite(1,®) == 1) //this line passes the if-statement, but makes no difference if reg is zero, or something else, we always get bytes from 0x00.
{
/* Now read data */
HalI2CInit(MMA8453_ADDRESS,i2cClock_123KHZ);
i = HalI2CRead(nBytes,pBuf);
}
return i == nBytes;
}