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.

I2C Nothing

Other Parts Discussed in Thread: CONTROLSUITE, TMS320F28069

Well I have been trying to get my TMS320F28069U I2C module working for a few days now and all I can do is generate a start condition on the bus...I have implemented and used plenty of other I2C modules but I am very displeased with the lack of documentation in this feature.

For example there is no explanation about FIFO mode and its usage verse non-FIFO mode. There is a register description and that is it...

So here is my code...it is nearly exactly (functionally it is exact, except it doesn't work) as what the example is in the ControlSuite so please tell me what I am doing wrong.

// Setup for I2C pins and peripheral clock.

    // GPIO32 - uC_CSAC_SDA
    GpioCtrlRegs.GPBPUD.bit.GPIO32 = 0;        // Enable pull-up for GPIO32 (SDAA)
    GpioCtrlRegs.GPBQSEL1.bit.GPIO32 = 3;      // Asynch input GPIO32 (SDAA)
    GpioCtrlRegs.GPBMUX1.bit.GPIO32 = 1;    // 0=GPIO,  1=I2CSDA,  2=SYNCI,  3=ADCSOCA

    // GPIO33 - uC_CSAC_SCL
    GpioCtrlRegs.GPBPUD.bit.GPIO33 = 0;        // Enable pull-up for GPIO32 (SCLA)
    GpioCtrlRegs.GPBQSEL1.bit.GPIO33 = 3;      // Asynch input GPIO32 (SLCA)
    GpioCtrlRegs.GPBMUX1.bit.GPIO33 = 1;    // 0=GPIO,  1=I2CSCL,  2=SYNCO,  3=ADCSOCB

    SysCtrlRegs.PCLKCR0.bit.I2CAENCLK = 1;         // I2C-A

// I2C initialization

    void I2C_Init(void)
    {
        // Just like the example...
        I2C_DISABLE;                                              // Disable I2C module for IPSC initialization (IRS = 0).
        I2caRegs.I2COAR = 0x0007;                    // I2C_ master address.
        I2caRegs.I2CPSC.all = 7;                          // Set I2C module clock (not SCL) to 10MHz for specified operation, systemclock = 80MHz.
        I2caRegs.I2CCLKH = 45;                          // Set I2C SCL to 100kHz operation (high time multipler).
        I2caRegs.I2CCLKL = 45;                           // Set I2C SCL to 100kHz operation (low time multipler).
        I2caRegs.I2CIER.all = I2C_INTS;             // No interrupts set.
        I2C_ENABLE;                                                // Enable I2C module for initialization (IRS = 1).
        I2caRegs.I2CFFTX.all = 0x6000;               // Enable FIFO mode and TXFIFO.
        I2caRegs.I2CFFTX.bit.TXFFINTCLR = 1; // Does not clear the INT although it says it will.
    }

// I2C Write (only generates start condition)

    char I2C_Write(char *local_buffer_ptr, char local_buffer_cnt)
    {
        unsigned char i;

        if(I2caRegs.I2CMDR.bit.STP == 1)
        {
            return I2C_STP_NOT_READY_ERROR;
        }
        I2caRegs.I2CSAR = 0x0024;            // Load I2C slave address.

        if(I2caRegs.I2CSTR.bit.BB == 1)
        {
            return I2C_BUS_BUSY_ERROR;
        }
        I2caRegs.I2CCNT = local_buffer_cnt;    // Load number of bytes to be transmitted.

        for(i = local_buffer_cnt; i > 0; i--)
        {
            I2caRegs.I2CDXR = *local_buffer_ptr;
            local_buffer_ptr++;
            local_buffer_cnt--;
        }
        I2caRegs.I2CMDR.all = 0x6E20; // Master, transmitter, start and stop.
        I2C_Tran_Flag = 0;
        return 0;
    }

So I see all the initializations and such go into the appropriate registers, I see the slave address and the count set but all I get out is a start condition. Please advise!

  • I feel your pain!  I've spent the better part of 2 days getting my own I2C code working on the F28335 (and I spent a good 2 days getting a project to successfully merge SYS-BIOS hello world example to play nice with all the experimenter board peripheral libraries of the I2C EEPROM example, but that's another story).

    As for documentation of I2C for C2000, it takes a bit of looking.  First, I started out with the manual for my chip:

    http://www.ti.com/lit/ds/sprs439m/sprs439m.pdf

    Unfortunately, they seemed to have forgotten to list the peripherals manual on the F28335 webpage in the listing of documents.  That was a bit tricky to find:

    http://www.ti.com/lit/ug/spru566j/spru566j.pdf

    On page 14 section 3.10 of the 28x DSP Peripherals manual, it will refer you to even more specific I2C manuals.  In my case:

    http://www.ti.com/lit/ug/sprug03b/sprug03b.pdf

    This 3rd manual goes into the detail you need to cover all the registers and how to setup them up.  Well....except it glosses over FIFO as you said.  But it turns out the EEPROM example uses FIFO if you look carefully at it.  It's really not too hard.  You set the counter register I2CCNT to the byte count you have (with a maximum of up to 16 bytes at a time), then load your data into I2CDXR one byte at a time, and finally trigger a transmit with I2CMDR.

    I had the issue of seeing just the start condition at first and no data sent.  I can't quite backtrace my debuggins, but I think my MDR settings before and after weren't quite right.

    Here's the bulk of my code to do a successful I2C write to a 4-digit 7-segment LED driver to test the LEDs being lit.

    void I2CA_Init(void);

    //
    // RAM Function references
    // These are defined by the linker (see F28335.cmd)
    //
    extern Uint16 RamfuncsLoadStart;
    extern Uint16 RamfuncsLoadEnd;
    extern Uint16 RamfuncsRunStart;



    /*
     *  ======== main ========
     */
    Void main()
    {

        MemCopy(&RamfuncsLoadStart, &RamfuncsLoadEnd, &RamfuncsRunStart);  // needed if you use the DELAY_US, especially ADC setup


        // Step 1. Initialize System Control:
        // PLL, WatchDog, enable Peripheral Clocks
        // This example function is found in the DSP2833x_SysCtrl.c file.
        InitSysCtrl();


        // Step 2. Initalize GPIO:
        // This example function is found in the DSP2833x_Gpio.c file and
        // illustrates how to set the GPIO to it's default state.
        // InitGpio();
        // Setup only the GP I/O only for I2C functionality
        InitI2CGpio();



        // Step 4. Initialize all the Device Peripherals:
        // This function is found in DSP2833x_InitPeripherals.c
        // InitPeripherals(); // Not required for this example
        I2CA_Init();


        DELAY_US(1000000);


        // Check for stop condition
        while (I2caRegs.I2CMDR.bit.STP == 1)
        {
            uiTemp = 1;
        }


        I2caRegs.I2CSAR = 0x38;

        // Check if bus busy
        while (I2caRegs.I2CSTR.bit.BB == 1)
        {
            uiTemp = 1;
        }



        I2caRegs.I2CCNT = 6;

        I2caRegs.I2CDXR = 0x00;
        I2caRegs.I2CDXR = 0x7F;
        I2caRegs.I2CDXR = 0xFF;
        I2caRegs.I2CDXR = 0xFF;
        I2caRegs.I2CDXR = 0xFF;
        I2caRegs.I2CDXR = 0xFF;


        //  Engage FIFO transmit
        I2caRegs.I2CMDR.all = 0x6E20;

        //  Infinite loop to catch after successful transmit
        while (1)
        {
            uiTemp = 1;
        }

    } // end main


    void I2CA_Init(void)
    {
       // Initialize I2C
       I2caRegs.I2CSAR = 0x3C;            // Slave address of first I2C 4-digit display controller

       #if (CPU_FRQ_150MHZ)             // Default - For 150MHz SYSCLKOUT
         I2caRegs.I2CPSC.all = 14;       // Prescaler - need 7-12 Mhz on module clk (150/15 = 10MHz)
       #endif
       #if (CPU_FRQ_100MHZ)             // For 100 MHz SYSCLKOUT
         I2caRegs.I2CPSC.all = 9;        // Prescaler - need 7-12 Mhz on module clk (100/10 = 10MHz)
       #endif

       I2caRegs.I2CCLKL = 76;            // NOTE: must be non zero
       I2caRegs.I2CCLKH = 38;            // NOTE: must be non zero
       I2caRegs.I2CIER.all = 0x0;        // Disable interrupts



       I2caRegs.I2CMDR.all = 0x0020;    // Take I2C out of reset
                                           // Stop I2C when suspended



       I2caRegs.I2CFFTX.all = 0x6000;    // Enable FIFO mode and TXFIFO
       I2caRegs.I2CFFRX.all = 0x2040;    // Enable RXFIFO, clear RXFFINT,

       return;
    }

    The above settings for CCLKL and CCLKH will get a 86 kHz SCL rate for the I2C and the predivisor gets you 10 MHz (150 MHz system / (14+1) ) for the I2C module clock.

    I will probably start a separate thread if this isn't acknowledged, as it needs to be fixed.  On page 34 section 5.7.1 of TMS320x280x Inter-Integrated Circuit Module, the formula is WRONG!  If you refer to the flowchart on page 14, it is clear that the prescaler gets you the I2C clock frequency.  So the denominator in that equation is the system clock (150 MHz in my case) NOT the I2C clock.  The prescaler is already in that equation!   

    UPDATE/EDIT: See Trey's post below.  I was confused thinking the equation referred to the I2C module clock.  It refers to the I2C input clock, which is the SYSCLK.

    That means that the controlSUITE EEPROM I2C example probably has an unintended clock rate 15 X higher than it should be, it's well over 100 kHz.

    Now a read might be a bit trickier to do.

    Hope that helps.

    I'm in a bit of a hurry, so I didn't get to look to closely at your code.

  • Andrew thank you very much for the prompt response and code snippets...I believe from what you have posted and what I have studied in the example I should be able to get this interface working. Definitely seems like I am not setting something at the appropriate time.

    And your information at the end on the clocking formula error I think is something else I will have to look at. When I looked at the example the master clock was at a very weird frequency (~454kHz).

    I have thoroughly reviewed most of the documentation, think I just need to revisit the MDR and the clocking.

    Many thanks again! Hopefully I can shake out the bugs from here...

  • Andrew,

    We will take a look at the forumla error you mentioned and get it rectified.  Thank you for bringing it to our attention.

    Regards,

    Peggy

  • Andrew,

    I can understand your frustration with the documentation.  Much of it is written from a design perspective and not from the perspective of an engineer actually trying to use the peripherals.

    That said, while I believe the formula could be clearer, the formula is correct.  If we look at page 14 (as you suggested) we see that the I2c input clock is the SYSCLK.  Then if we look at the formula on page 34, we see that the denominator is in fact the I2c input clock which is the SYSCLK just as you suggested.

    Regards,

  • Ahh, you're right.  I misread that as I2C module clock, the I2C input clock confused me.

  • Dear all,

    I'm trying to use the I2C pheripheral of the TMS320F28069, my code run exactly but the SCL pins is always high, I don't see the clock signal in the SCK output, and also the data in the SDA output.

    My code snippet is the follow:

      //  GPIO-28 - PIN FUNCTION = --Spare--
        GpioCtrlRegs.GPAMUX2.bit.GPIO28 = 2;   // Configure GPIO28 for SDAA operation
        GpioCtrlRegs.GPAPUD.bit.GPIO28 = 0;    // Enable pull-up for GPIO28 (SDAA)
        GpioCtrlRegs.GPAQSEL2.bit.GPIO28 = 3;  // Asynch input GPIO28 (SDAA)
        //--------------------------------------------------------------------------------------

    //  GPIO-29 - PIN FUNCTION = --Spare--
        GpioCtrlRegs.GPAMUX2.bit.GPIO29 = 2;   // Configure GPIO29 for SCLA operation
        GpioCtrlRegs.GPAPUD.bit.GPIO29 = 0;    // Enable pull-up for GPIO29 (SCLA)
        GpioCtrlRegs.GPAQSEL2.bit.GPIO29 = 3;  // Asynch input GPIO29 (SCLA)

        SysCtrlRegs.PCLKCR0.bit.I2CAENCLK = 1;   // I2C

    void main (void){

       I2caRegs.I2CSAR = 0x0050;        // Slave address - EEPROM control code

       I2caRegs.I2CPSC.all = 7;            // Prescaler - need 7-12 Mhz on module clk
       I2caRegs.I2CCLKL = 76;            // NOTE: must be non zero
       I2caRegs.I2CCLKH = 38;            // NOTE: must be non zero
       I2caRegs.I2CIER.all = 0x0;        // disable interrupts

       I2caRegs.I2CMDR.all = 0x0020;    // Take I2C out of reset
                                           // Stop I2C when suspended

       I2caRegs.I2CFFTX.all = 0x6000;    // Enable FIFO mode and TXFIFO
       I2caRegs.I2CFFRX.all = 0x2040;    // Enable RXFIFO, clear RXFFINT,

    while(1){        

                while (I2caRegs.I2CMDR.bit.STP == 1)
                {
                    uiTemp = 1;
                }


                I2caRegs.I2CSAR = 0x38;

                // Check if bus busy
                while (I2caRegs.I2CSTR.bit.BB == 1)
                {
                    uiTemp = 1;
                }



                I2caRegs.I2CCNT = 6;

                I2caRegs.I2CDXR = 0x00;
                I2caRegs.I2CDXR = 0x7F;
                I2caRegs.I2CDXR = 0xFF;
                I2caRegs.I2CDXR = 0xFF;
                I2caRegs.I2CDXR = 0xFF;
                I2caRegs.I2CDXR = 0xFF;


                //  Engage FIFO transmit
                I2caRegs.I2CMDR.all = 0x6E20;

        }// end of while(1)    
    }// end of main

    Can someone help me, please?

  • Nicola,

    Have you verified that the GPIO mux values are being written to the register successfully?  I'm guessing you're missing an EALLOW somewhere and some of the values aren't being actually written.

    BR,

  • Thanks a lot but I have solved now.

  • Hi, i have the same problem, the SCL and SDA pins are high all the time, dont have clock signal... how do you fix??? im working in simulink with the library embedded coder

    how can i change that "mux values"????
  • Hi,

    my problem was thas I tried the I2C module without a slave device connected in the bus, so the acknowledge condition is missed and 

    the I2C module wait indefinitely.

    I work with Code Composer Studio and I dont't now how you can change "mux value" with simulink.

    I'm sorry

  • SImulink create a .out

    then ccs create a code that will build in the target (c28027)

    in the worst case i will change directly in the code, but i dont know where... if i upload the code can help to me?