This thread has been locked.

If you have a related question, please click the "Ask a related question" button in the top right corner. The newly created question will be automatically linked to this question.

LAUNCHXL-F28069M: F28069 I2C communication with polling and without FIFO

Part Number: LAUNCHXL-F28069M

Hello,

I am trying to estaiblish I2C communication from one F28069 to another. I see that the slave address is 0x7900.

I have a 10kohm resistor as pull up. I have disabled interrupts and FIFO.

Using the code below I see that one byte of data is getting transferred fine but I am not able to see any clock on my SCL lines when using my logic analyzer. Would the SCL clock stop after the data transfer is over?

Does this code seem okay for starters?

Here is my code - 

#include "DSP28x_Project.h"

#include <stdint.h>

uint16_t Readi2c();

void Writei2c();

void Initi2c();

void Initi2c(void){

EALLOW;

// GpioCtrlRegs.GPBPUD.bit.GPIO32 = 0x0000;  //I am using external pull up resistors

// GpioCtrlRegs.GPBPUD.bit.GPIO33 = 0x0000;

GpioCtrlRegs.GPBQSEL1.bit.GPIO32 = 0x0003;

GpioCtrlRegs.GPBQSEL1.bit.GPIO33 = 0x0003;

GpioCtrlRegs.GPBMUX1.bit.GPIO32 = 0x0001;

GpioCtrlRegs.GPBMUX1.bit.GPIO33 = 0x0001;

EDIS;

I2caRegs.I2CMDR.all = 0x0000;                 // Keep I2C in reset before setting prescalar

I2caRegs.I2CSAR = 0x7900;        //Address of F28069

I2caRegs.I2CPSC.all = 0x0008;    // for 400kHz

I2caRegs.I2CCLKL = 20;  

I2caRegs.I2CCLKH = 10; 

I2caRegs.I2CIER.all = 0x0000; // Disable all interrupts

// I2caRegs.I2CFFTX.all = 0x6000; // Enable FIFO mode and TXFIFO

// I2caRegs.I2CFFRX.all = 0x2040; // Enable RXFIFO, clear RXFFINT,

I2caRegs.I2CMDR.all = 0x0020; // 1: Take I2C out of reset

// 0: Stop I2C when suspended

}

uint16_t Readi2c()

{

uint16_t tempdata;

I2caRegs.I2CSAR = 0x7900;

I2caRegs.I2CCNT = 0x0001;

I2caRegs.I2CMDR.all = 0x2020;       //Start in receive slave mode

while(!I2caRegs.I2CSTR.bit.RRDY); 

I2caRegs.I2CCNT = 0x0001;

I2caRegs.I2CMDR.all = 0x2C20;       // Master receive. 

I2caRegs.I2CMDR.all = 0x2820;      // Other F28069 (slave) has 0x2820 for slave mode receive

tempdata = I2caRegs.I2CDRR;

return(tempdata);

}

void Writei2c()

{

// I2caRegs.I2CMDR.all = 0x2620; //Send Start condition

I2caRegs.I2CSAR = 0x7900;

I2caRegs.I2CCNT = 0x0001;

I2caRegs.I2CDXR = 0x05; //Write one byte on bus

I2caRegs.I2CMDR.all = 0x2E20; //Send stop bit and set as master transmitter

while(!I2caRegs.I2CSTR.bit.SCD);

I2caRegs.I2CSTR.bit.SCD = 1;

}

void main()

{

uint16_t recvdata;

InitSysCtrl();    //I2C clock is initialized in this function

Initi2c();

Writei2c();      

recvdata = Readi2c(); // Line in slave F28069 to read data from F28069 master

while(1)

  {

     }

}

  • Hello Sowmya,

    I've looked into your code a little and it seems to be OK for the most part. One thing to note is that I don't believe 0x7900 is the F28069 slave address, but actually I2COAR's register address. The value of this register defaults to zero on start up, so you will need to set the device's slave address yourself as shown below...

    I2caRegs.I2COAR.all = 0x0060; // Sets device slave address to 0x0060, or whatever you'd like it to be

    I would suggest keeping the address you set to one byte and using the lower 7 bits, as the I2C module only handles 7 and 10 bit addressing.

    The SCL clock will pulse during transmission and when receiving (Reading/writing). It will not always be pulsing and can be held high/low.

    Hope this helps,
    Kevin
  • Thanks, the SCL clock part is clear now.
    Yes, you are right, 0x7900 is F28069's own address.
    My settings are now ->
    I2caRegs.I2COAR.all = 0x0060 in the Initi2c( ) function.
    I2caRegs.I2CSAR.all = 0x0060 in the Readi2c ( ) function.

    I have an issue when I send multiple bytes though. I am able to read only one byte into my slave.

    I hope polling on while(!I2caRegs.I2CSTR.bit.RRDY); is the right way. I haven't explicity set any NACK bits.

    This thread says that we would need to poll on SCD bit and then read the bytes from DDR one by one. Is that the right approach, instead of the RRDY flag ?

    https://e2e.ti.com/support/microcontrollers/c2000/f/171/t/416428?tisearch=e2e-quicksearch&keymatch=f28069%20i2c

  • Hi Sowmya,

    Having I2caRegs.I2COAR.all = 0x0060 in your code will set the slave address of the current F28069 to 0x60 (The device that runs the I2COAR line), if that makes sense.

    Since you are wanting I2C communication between two F28069s, what I think you will want to do is set a different slave address for each device within the Initi2c function. The master device will want to set I2caRegs.I2CSAR equal to the other device's slave address when it is reading/writing to it. Each slave has a different address because you can have multiple I2C slaves on one line, it lets the specific slave know that it's being written to or is being read from by a master.

    For example lets say you want to set the master f28069's "slave address" to 0x05 and the slave f28069's "slave address" to 0x60.

    You will have I2caRegs.I2COAR.all = 0x0005 in the Initi2c function of the master device

    You will have I2caRegs.I2COAR.all = 0x0060 in the Initi2c function of the slave device

    If the master wants to read or write to the slave device it will set I2caRegs.I2CSAR = 0x0060, the address of the slave it wants to talk to, at the beginning of the read/write functions.

    This was sort of a quick explanation of how this process works and I hope it made sense!

    Best,
    Kevin
  • Thanks it makes sense.

    I just made an edit to my question while you were sending your reply. Could you have a look? Its regarding receiving multiple bytes.

    I have a pretty simple application so do not prefer to use any interrupts or FIFO. 

    Any quick help would be great!

  • Hi Sowmya,

    Yes, this looks good.

    Regarding sending/receiving multiple bytes, you will want to look into the I2caRegs.I2CCNT register. This sets how many bytes you expect to read or write. Below is a link to the I2C manual that explains this concept well (Check pg. 35 for starters), it is for X2802x & X2803x but will work for f2806x all the same.

    www.ti.com/.../sprufz9d.pdf

    You will probably also want to read a little into STOP conditions, NACKs, START conditions, and maybe even repeated START conditions. I have not tested the below code, but maybe something similar to it could work for your read function.

    uint16_t data1, data2, data3; // Could also utilize an array for this
    
    I2caRegs.I2CSAR = 0x0060;
    
    I2caRegs.I2CCNT = 0x0001;
    
    I2caRegs.I2CMDR.all = 0x2020; //Start in receive slave mode
    
    while(!I2caRegs.I2CSTR.bit.RRDY);
    
    I2caRegs.I2CCNT = 0x0003;
    
    I2caRegs.I2CMDR.all = 0x2C20; // Master receive.
    
    // I2caRegs.I2CMDR.all = 0x2820; // Other F28069 (slave) has 0x2820 for slave mode receive
    
    // May need some sort of start condition here...
    
    while(!I2caRegs.I2CSTR.bit.RRDY);
    data1 = I2caRegs.I2CDRR;
    while(!I2caRegs.I2CSTR.bit.RRDY);
    data2 = I2caRegs.I2CDRR;
    while(!I2caRegs.I2CSTR.bit.RRDY);
    data3 = I2caRegs.I2CDRR;


    Hope this helps,
    Kevin