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.
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)
{
}
}
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
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