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.

MSP432P401R: 32-Bit DMA over SPI

Part Number: MSP432P401R

My goal is to achieve an I2S-compatible PCM-audio transfer using SPI and an external Codec (Maxim 9867). Main concept is:

  • Short audio sample is located in memory
  • Have a timer providing the frame sync signal
  • Use SPI to feed the digital audio interface
  • Configure DMA with timer as trigger to transfer 4 bytes:
    • 2 bytes (16 bits) left channel
    • 2 bytes for right channel (according to I2S specification)

Hence I setup TIMER_A3 with period of 543 ticks according to audio sampling frequency of 44.1 kHz. This does produce a proper frame sync signal visible in logic analyzer. The SPI on EUSCI_B1_BASE is configured properly (we will see later). So far so good...

Then I use the TIMER_A3_CCR0 as trigger for DMA_CH6 which leads to the following source code for DMA:

#define MAXIM_PLAY_DMA_CHANNEL         DMA_CH6_TIMERA3CCR0
#define MAXIM_PLAY_DMA_CH_NUMBER       DMA_CHANNEL_6
#define AUDIO_BUFFER_SIZE				0x10

#define DMA_ARBITRATION_VALUE   UDMA_ARB_2

volatile uint8_t audioBuffer[2 * AUDIO_BUFFER_SIZE];

/* Later audioBuffer will be parameter of this function */
void maximPlayDMAinit(volatile uint8_t * pData)
{
   MAP_DMA_disableChannel(MAXIM_PLAY_DMA_CH_NUMBER);
   MAP_DMA_assignChannel(MAXIM_PLAY_DMA_CHANNEL);
   MAP_DMA_disableChannelAttribute(MAXIM_PLAY_DMA_CHANNEL
                                   , UDMA_ATTR_ALTSELECT | UDMA_ATTR_USEBURST | UDMA_ATTR_HIGH_PRIORITY | UDMA_ATTR_REQMASK
                                   );
   MAP_DMA_setChannelControl(MAXIM_PLAY_DMA_CHANNEL | UDMA_PRI_SELECT
                         , UDMA_SIZE_8 | UDMA_SRC_INC_8 | UDMA_DST_INC_NONE | DMA_ARBITRATION_VALUE
                         );
   MAP_DMA_setChannelControl(MAXIM_PLAY_DMA_CHANNEL | UDMA_ALT_SELECT
                         , UDMA_SIZE_8 | UDMA_SRC_INC_8 | UDMA_DST_INC_NONE | DMA_ARBITRATION_VALUE
                         );
   MAP_DMA_setChannelTransfer(MAXIM_PLAY_DMA_CHANNEL | UDMA_PRI_SELECT
                         , UDMA_MODE_PINGPONG
                         , (void *)pData
                         , (void *)MAP_SPI_getTransmitBufferAddressForDMA(mSPIbase)
                         , AUDIO_BUFFER_SIZE
                         );
   MAP_DMA_setChannelTransfer(MAXIM_PLAY_DMA_CHANNEL | UDMA_ALT_SELECT
                         , UDMA_MODE_PINGPONG
                         , (void *)(pData + AUDIO_BUFFER_SIZE)
                         , (void *)MAP_SPI_getTransmitBufferAddressForDMA(mSPIbase)
                         , AUDIO_BUFFER_SIZE
                         );
   MAP_DMA_clearInterruptFlag(MAXIM_PLAY_DMA_CH_NUMBER);
}

void maximStartPlayDMA()
{
   MAP_DMA_clearInterruptFlag(mPlayDMAconfig.channelNumber);
   MAP_DMA_assignInterrupt(DMA_INT1, mPlayDMAconfig.channelNumber);
   MAP_DMA_enableChannel(mPlayDMAconfig.channelNumber);
   MAP_Interrupt_enableInterrupt(INT_DMA_INT1);
   MAP_DMA_enableInterrupt(DMA_INT1);
}

void dma_1_interrupt(void)
{
   uint32_t ui32Mode = MAP_DMA_getChannelMode( MAXIM_PLAY_DMA_CHANNEL | UDMA_PRI_SELECT);
   if (ui32Mode == UDMA_MODE_STOP)
   {
      /* Ping Pong Primary */
      fill_buf(audioBuffer);
      MAP_DMA_setChannelTransfer(MAXIM_PLAY_DMA_CHANNEL | UDMA_PRI_SELECT
                         , UDMA_MODE_PINGPONG
                         , (void *)audioBuffer
                         , (void *)MAP_SPI_getTransmitBufferAddressForDMA(mSPIbase)
                         , AUDIO_BUFFER_SIZE
                         );
   }
   ui32Mode = MAP_DMA_getChannelMode( MAXIM_PLAY_DMA_CHANNEL | UDMA_ALT_SELECT);
   if (ui32Mode == UDMA_MODE_STOP)
   {
      /* Ping Pong Secondary */
      fill_buf(audioBuffer + AUDIO_BUFFER_SIZE);
      MAP_DMA_setChannelTransfer(MAXIM_PLAY_DMA_CHANNEL | UDMA_ALT_SELECT
                         , UDMA_MODE_PINGPONG
                         , (void *)(audioBuffer + AUDIO_BUFFER_SIZE)
                         , (void *)MAP_SPI_getTransmitBufferAddressForDMA(mSPIbase)
                         , AUDIO_BUFFER_SIZE
                         );
} }

When I give it a try I can see the audio samples in logic analyzer in right order, but there are only 2 Bytes being transfered upon each trigger from frame sync signal: (please disregard the last 8 not matching bits in decoded data as a configuration issue - the signals are the same, obviously):

Of course, you might say, I simply need to change the define to UDMA_ARB_4. But this is where the real odds (and major topic of this post) are coming up: still only 16 bits are being transferred, but the 16 bits from right channel have vanished!

And this is something I neither understand nor having a clue how to resolve it.

  • I can take a look at this, but it would help to have the rest of the code. Can you send me the main(), or at least a sufficient snippet to replicate what you are trying to do?
    Regards,
    Bob L.
  • The main() is from TI-Bluetopia examples, i.e. the A3DPDemo_Sink. I did hook the initialization of the Maxim codec and the DMA stuff into HAL_EnableAudioCodec function. The code is here:

    /* DMA Control Table */
    #if defined(__TI_COMPILER_VERSION__)
    #pragma DATA_ALIGN(1024)
    #elif defined(__IAR_SYSTEMS_ICC__)
    #pragma data_alignment=1024
    #elif defined(__GNUC__)
    __attribute__ ((aligned (1024)))
    #elif defined(__CC_ARM)
    __align(1024)
    #endif
    static DMA_ControlTable MSP_EXP432P401RLP_DMAControlTable[32];  // from dma_eusci_spi_loopback example
    
    static eUSCI_SPI_MasterConfig spiConfig;
    
    void Init_Maxim(void)
    {
       // Switch on master clock generation for Maxim Audio codec
       GPIO_setAsPeripheralModuleFunctionOutputPin(GPIO_PORT_P7, GPIO_PIN0, GPIO_PRIMARY_MODULE_FUNCTION);
       
       fill_buf(audioBuffer);
       fill_buf(audioBuffer + AUDIO_BUFFER_SIZE);
       
       MAP_DMA_enableModule();
       MAP_DMA_setControlBase(MSP_EXP432P401RLP_DMAControlTable);
       
       maximPlayDMAinit(audioBuffer);
       maximStartPlayDMA(); // Will not really start before first trigger occurs
    
       spiConfig.selectClockSource = EUSCI_B_SPI_CLOCKSOURCE_SMCLK;
       spiConfig.clockSourceFrequency = CS_getSMCLK();   // const
       spiConfig.desiredSpiClock = 3000000UL;  // preliminary maximum
       spiConfig.msbFirst = EUSCI_B_SPI_MSB_FIRST;
       spiConfig.clockPhase = EUSCI_B_SPI_PHASE_DATA_CHANGED_ONFIRST_CAPTURED_ON_NEXT;
       spiConfig.clockPolarity = EUSCI_B_SPI_CLOCKPOLARITY_INACTIVITY_LOW;
       spiConfig.spiMode = EUSCI_B_SPI_3PIN;
    
       MAP_SPI_initMaster(EUSCI_B1_BASE, &spiConfig);
       MAP_SPI_enableModule(EUSCI_B1_BASE);
       
       maximInit();   // configure Maxim Audio codec over I2C using EUSCI_B0
       
       maximFramesyncInit();   // Configure TIMER_TA3 for frame sync signal
       
       maximStartSampling();   // Enable counter of TIMER_TA3 - which generates trigger for DMA
    }
    

    A few words to the hardware: I attached the module BOOST-CC2564MODA to the TI-MSP-EXP432P401R launchpad, and on top there is my own module with the Audio codec. I made sure that no pins overlap in terms of functionality. The Maxim Codec uses completely different pins than the audio from Bluetooth module.


    OT: Why doesn't get the code get highlighted even while I used the "Insert code using Syntaxhighlighter"? For me it does only draw a frame around (in Firefox latest greatest).

  • Can't answer the code highlighting question. I've never seen any actual syntax highlighting- just the gray box and fixed-width text.
    -Bob
  • Meanwhile I did a lot of try and error but it always ends with error. Looking at this thread I think my own description sounds too complicated to me. To keep the things short and better understandable I would summarize my question today in the following way:

    I'm using SPI with DMA as in dma_eusci_spi_loopback example from TI. Initialization of DMA channel is:

    #define DMA_ARBITRATION_VALUE   UDMA_ARB_2
    
    MAP_DMA_disableChannel(DMA_CHANNEL_6);
    MAP_DMA_assignChannel(DMA_CH6_TIMERA3CCR0);
    MAP_DMA_disableChannelAttribute(DMA_CH6_TIMERA3CCR0
                                    , UDMA_ATTR_ALTSELECT | UDMA_ATTR_USEBURST | UDMA_ATTR_HIGH_PRIORITY | UDMA_ATTR_REQMASK
                                    );
    MAP_DMA_setChannelControl(DMA_CH6_TIMERA3CCR0 | UDMA_PRI_SELECT
                          , UDMA_SIZE_8 | UDMA_SRC_INC_8 | UDMA_DST_INC_NONE | DMA_ARBITRATION_VALUE
                          );
    MAP_DMA_setChannelControl(MAXIM_PLAY_DMA_CHANNEL | UDMA_ALT_SELECT
                          , UDMA_SIZE_8 | UDMA_SRC_INC_8 | UDMA_DST_INC_NONE | DMA_ARBITRATION_VALUE
                          );
    

    If the #define DMA_ARBITRATION_VALUE is either UDMA_ARB_1 or UDMA_ARB_2 the transfer does work as expected, flawlessly.

    As soon as I switch to UDMA_ARB_4 it doesn't work well - every 3rd and 4th byte are missing. Why does UDMA_ARB_4 break the functionality?

  • multi_channel_02.c
    Fullscreen
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    63
    64
    65
    66
    67
    //*****************************************************************************
    //
    // Copyright (C) 2015 - 2016 Texas Instruments Incorporated - http://www.ti.com/
    //
    // Redistribution and use in source and binary forms, with or without
    // modification, are permitted provided that the following conditions
    // are met:
    //
    // Redistributions of source code must retain the above copyright
    // notice, this list of conditions and the following disclaimer.
    //
    // Redistributions in binary form must reproduce the above copyright
    // notice, this list of conditions and the following disclaimer in the
    // documentation and/or other materials provided with the
    // distribution.
    //
    // Neither the name of Texas Instruments Incorporated nor the names of
    // its contributors may be used to endorse or promote products derived
    // from this software without specific prior written permission.
    //
    // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
    // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
    // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
    // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
    // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
    // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
    // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
    // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
    // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
    // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
    // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
    //
    //*****************************************************************************
    //*****************************************************************************
    //
    // MSP432
    //
    // 4, I2S, 16Bit, 200Khz
    //
    //
    //
    // 4, 160us Period Timers: TA0,TA1,TA2,TA3
    // 4, 200Khz SPIs: USCA0,USCA1,USCA2,USCA3
    // CPU Load: (rising edge processing + falling edge processing)/(total)
    // ( 11.54us + 11.54us )/160us
    // ~15%
    //
    //****************************************************************************
    #include "msp.h"
    /* Standard Includes */
    #include <stdint.h>
    #include <string.h>
    #include <stdbool.h>
    /* DriverLib Includes */
    #include "driverlib.h"
    #define ARRAY_LENGTH 32
    #define TRANSMIT_LENGTH 2
    /*
    *
    * TA0.1 -> P2.0
    * TA1.1 -> P2.3
    *
    XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX

    I hope this example is of some value.

    Chris

**Attention** This is a public forum