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.

TMS320C6748: Problem with SPI transfer using EDMA3

Part Number: TMS320C6748

Hello TI engineers,

I'm using EDMA3 to hold the SPI data transfer base on th C6748 starter-ware. SPI DMA event is enabled in Timer ISR function every 125us and disabled in DMA ISR after 4 16-bit words were transferred. But it does not work as I expected. 

For example, when the SPI_CLK was 2MHz, I monitored  the SPI bus with logic analyzer, I found that 5 16-bit words were transferred in the first turn and 4 16-bit words were transferred in other turns.

For another example, when the SPI_CLK was 6MHz, I monitored  the SPI bus with logic analyzer, I found that 9 16-bit words were transferred in the first turn and 8 16-bit words were transferred in other turns.

I only want to transfer 4 16-bit words every 125us, but it doesn't work as expected. Please give me some suggestions. Thanks!

Below is my code.

1. This function is executed every 125us in Timer ISR to enable the SPI DMA event.

void SPI0Transfer(void)
{

    EDMA3EnableTransfer(SOC_EDMA30CC_0_REGS, EDMA3_CHA_SPI0_TX, EDMA3_TRIG_MODE_EVENT);

    EDMA3EnableTransfer(SOC_EDMA30CC_0_REGS, EDMA3_CHA_SPI0_RX, EDMA3_TRIG_MODE_EVENT);

    CSHoldAssert();

    SPIIntEnable(SOC_SPI_0_REGS, SPI_DMA_REQUEST_ENA_INT);  

}
 

2.This part is SPI configuration.

#define CHAR_LENGTH             (16)

static void SPI0Config(void)
{
    unsigned int  val = SIMO_SOMI_CLK_CS;

    SPIReset(SOC_SPI_0_REGS);

    SPIOutOfReset(SOC_SPI_0_REGS);

    SPIModeConfigure(SOC_SPI_0_REGS, SPI_MASTER_MODE);

    SPIClkConfigure(SOC_SPI_0_REGS, 228000000, 2000000, SPI_DATA_FORMAT0);

    SPIDelayConfigure(SOC_SPI_0_REGS,0,0,20,20);

    SPIPinControl(SOC_SPI_0_REGS, 0, 0, &val);

    SPIDefaultCSSet(SOC_SPI_0_REGS, (CS_ADC0 | CS_ADC1));

    /* Configures the polarity and phase of SPI clock */

    SPIConfigClkFormat(SOC_SPI_0_REGS, (SPI_CLK_POL_HIGH | SPI_CLK_INPHASE), SPI_DATA_FORMAT0);

    /* Configures SPI to transmit MSB bit First during data transfer */
    SPIShiftMsbFirst(SOC_SPI_0_REGS, SPI_DATA_FORMAT0);

    /* Sets the Charcter length */
    SPICharLengthSet(SOC_SPI_0_REGS, CHAR_LENGTH, SPI_DATA_FORMAT0);
    /* Enable SPI communication */
    SPIEnable(SOC_SPI_0_REGS);
}

 

3. This function is executed in callback function in DMA ISR to disable SPI DMA event.

void Spi0Pause(void)

{

    SPIIntDisable(SOC_SPI_0_REGS, SPI_DMA_REQUEST_ENA_INT);  

    CSHoldDeassert();

}

static void callback(unsigned int tccNum, unsigned int status)

{

    if(tccNum == EDMA3_CHA_SPI0_RX)

    {

    }

    else if(tccNum == EDMA3_CHA_SPI0_TX)

    {

        Spi0Pause();

    }

}

 

4. This part is DMA configuration.

void EDMA3Initial(void)

{

    /* Power up EDMA3CC_0 and EDMA3TC_0 */

    PSCModuleControl(SOC_PSC_0_REGS, HW_PSC_CC0, PSC_POWERDOMAIN_ALWAYS_ON, PSC_MDCTL_NEXT_ENABLE);

    PSCModuleControl(SOC_PSC_0_REGS, HW_PSC_TC0, PSC_POWERDOMAIN_ALWAYS_ON,  PSC_MDCTL_NEXT_ENABLE);

    EDMA3Init(SOC_EDMA30CC_0_REGS, 0);

    EDMA3IntSetup();

    EDMA3Configure();

}

 

static void EDMA3Configure(void)

{

    EDMA3RequestChannel(SOC_EDMA30CC_0_REGS, EDMA3_CHANNEL_TYPE_DMA, EDMA3_CHA_SPI0_TX, EDMA3_CHA_SPI0_TX, 0);

    EDMA3RequestChannel(SOC_EDMA30CC_0_REGS, EDMA3_CHANNEL_TYPE_DMA, EDMA3_CHA_SPI0_RX, EDMA3_CHA_SPI0_RX, 0);

    DMAParamInit();

    cb_Fxn[EDMA3_CHA_SPI0_TX] = &callback;

    cb_Fxn[EDMA3_CHA_SPI0_RX] = &callback;

}

 

void DMAParamInit(void)     
{
    EDMA3CCPaRAMEntry paramSet; 

    /* Initialize the paRAM set for receive */
    memcpy(&paramSet, &spi0rxDefaultPar, SIZE_PARAMSET - 2);
    EDMA3SetPaRAM(SOC_EDMA30CC_0_REGS, EDMA3_CHA_SPI0_RX, &paramSet);//PaRAM 14
    EDMA3SetPaRAM(SOC_EDMA30CC_0_REGS, 41, &paramSet);//PaRAM 41
    paramSet.destAddr = (unsigned int)spi0RxBuf1;
    paramSet.linkAddr = (unsigned short)(41u * 32u);
    EDMA3SetPaRAM(SOC_EDMA30CC_0_REGS, 40, &paramSet);//PaRAM 40

/* Initialize the paRAM set for transmit */
    memcpy(&paramSet, &spi0txDefaultPar, SIZE_PARAMSET - 2);
    EDMA3SetPaRAM(SOC_EDMA30CC_0_REGS, EDMA3_CHA_SPI0_TX, &paramSet);//PaRAM 15
    paramSet.linkAddr = 42*32u;
    EDMA3SetPaRAM(SOC_EDMA30CC_0_REGS, 42, &paramSet);//PaRAM 42
}
  

/*

** Sets up the interrupts for EDMA in AINTC

*/

static void EDMA3IntSetup(void)

{

    IntRegister(C674X_MASK_INT5, EDMA3CCComplIsr);

    IntEventMap(C674X_MASK_INT5, SYS_INT_EDMA3_0_CC0_INT1);

    IntEnable(C674X_MASK_INT5);

}

 

/*

** EDMA transfer completion ISR

*/

static void EDMA3CCComplIsr(void)

{

    volatile unsigned int pendingIrqs;

    volatile unsigned int isIPR = 0;

 

    volatile unsigned int indexl;

    volatile unsigned int Cnt = 0;

    indexl = 1;

 

    IntEventClear(SYS_INT_EDMA3_0_CC0_INT1);

 

    isIPR = EDMA3GetIntrStatus(SOC_EDMA30CC_0_REGS);

    if(isIPR)

    {

        while ((Cnt < EDMA3CC_COMPL_HANDLER_RETRY_COUNT)&& (indexl != 0))

        {

            indexl = 0;

            pendingIrqs = EDMA3GetIntrStatus(SOC_EDMA30CC_0_REGS);

            while (pendingIrqs)

            {

                if((pendingIrqs & 1) == TRUE)

                {

                    /**

                     * If the user has not given any callback function

                     * while requesting the TCC, its TCC specific bit

                     * in the IPR register will NOT be cleared.

                     */

                    /* Here write to ICR to clear the corresponding IPR bits. */

                    EDMA3ClrIntr(SOC_EDMA30CC_0_REGS, indexl);

                    (*cb_Fxn[indexl])(indexl, EDMA3_XFER_COMPLETE);

                }

                ++indexl;

                pendingIrqs >>= 1;

            }

            Cnt++;

        }

    }

}

 

5. This part is the ParamSet for SPI transfer and receive.

#define SPI0TX_DMA_INT_ENABLE                     (EDMA3CC_OPT_TCC_SET(EDMA3_CHA_SPI0_TX) | (1  << EDMA3CC_OPT_TCINTEN_SHIFT))

#define SPI0RX_DMA_INT_ENABLE                    (EDMA3CC_OPT_TCC_SET(EDMA3_CHA_SPI0_RX) | (1 << EDMA3CC_OPT_TCINTEN_SHIFT))

 

/*

** Default paRAM for Receive section.

*/

static struct EDMA3CCPaRAMEntry const spi0rxDefaultPar =

       {

           (unsigned int)(SPI0RX_DMA_INT_ENABLE), /* Opt field */

           (unsigned int)(SOC_SPI_0_REGS + SPI_SPIBUF), /* source address */

           (unsigned short)(2), /* aCnt */

           (unsigned short)(4), /* bCnt */

           (unsigned int)spi0RxBuf0, /* dest address */

           (short) (0), /* source bIdx */

           (short)(2), /* dest bIdx */

           (unsigned short)(40u * 32u), /* link address */

           (unsigned short)(0), /* bCnt reload value */

           (short)(0), /* source cIdx */

           (short)(0), /* dest cIdx */

           (unsigned short)1 /* cCnt */

       };

/*

 * Default paRAM for Transmit section.

 */

static struct EDMA3CCPaRAMEntry const spi0txDefaultPar =

       {

           (unsigned int)(SPI0TX_DMA_INT_ENABLE), /* Opt field */

           (unsigned int)spi0TxBuf0, /* source address */

           (unsigned short)(2), /* aCnt */

           (unsigned short)(4), /* bCnt */

           (unsigned int)(SOC_SPI_0_REGS + SPI_SPIDAT1), /* dest address */

           (short) (2), /* source bIdx */

           (short)(0), /* dest bIdx */

           (unsigned short)(42u * 32u), /* link address */

           (unsigned short)(0), /* bCnt reload value */

           (short)(0), /* source cIdx */

           (short)(0), /* dest cIdx */

           (unsigned short)1 /* cCnt */

       };

  • Hello, can someone help?

  • Hello,

    Starterware is no longer supported by TI, so our forum support will be limited.

    If you can, TI recommends upgrading to Processor SDK, which is the current supported software offering by TI today and for many years to come. Processor SDK provides unified API for various, current processors and for all processors in the future. This, combined with the abstracted board library and OS abstraction layer, allows customers to easily port their software to newer devices and other OSes if/when the time comes. 

    That being said, have you tried performing SPI transfers without EDMA involved? This way you can narrow down whether the issue lies in your SPI configuration or EDMA configuration. I suspect it is in your EDMA configuration, but it would be good if you can run this test to confirm. 

    By the way, have you referred to this wiki page already? 

    https://processors.wiki.ti.com/index.php/StarterWare_SPI

    Regards,
    Sahin

  • Thanks for your reply.

    Yes, I successfully performed SPI transfers without EDMA. And I also successfully performed SPI transfers with EDMA by setting the linkaddr of transfer paRam to 0xFFFF and loading new paRam of SPI transfers in EDMA ISR. But if setting the linkaddr to another paRam for auto reloading, the EDMA transfer complete interrupt won't happen properly when paRam is exhausted. So it appears problem above and I wonder why.

    And Yes, I have referred to that wiki.

  • Hi,

    SPI 0 Rx event is EDMA CC0 #14 and SPI 0 Tx event is EDMA CC0 #15. Can you clarify how the ParamSet 14, 15, 40, 41, 42 are linked? If you provide a memory dump from 0x01c0_4000 to 0x01c0_4fff may explain this. A working case ParamSet dump from: 0x01c0_4000 + 0x20 * 14 and 15 also help.

    Do you know all the transfers happened with linked EDMA but just interrupt didn't fire? You may check 0x01c0_1068 (IPR) to see if any bit is set (this bit should match your TCC code).

    Regards, Eric

  • Hi,

    Just as the code shown in part 4 and 5.

    The ParamSet 14 link to 41, ParamSet 41 link to 40, ParamSet 40 link to 41.

    The ParamSet 15 link to 42, ParamSet 42 link to itself.

    The memory is provided in below pictures.

    ParamSet 14 & 15:

    ParamSet 40 to 42, address begin at 0x1c0_4500;

    And Yes, the data transferred is correct but in wrong order because of 5 words were transferred in first turn other than 4 words in 2MHz clock case. When I check the IPR, it indicates 0x8000 or 0xC000 and matches the TCC code(14 & 15).

  • Hello, should I provide more other details?