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.

MSP430F5529: DMA not trigger by ADC

Part Number: MSP430F5529

Hello All, I am trying to do a simple program which configure ADC to read temperature and use DMA to send the result to UART for transmit. It looks like my ADC is not triggering my DMA, so when I look at the register UCATXBUF is always 0x00.

I am aware that I don't have the UART proportion of the code yet. I have this portion working another project. So here I am concentrating on getting the ADC read out to the UCATXBUF by DMA.


Fullscreen
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
#include <msp430.h>
#include <stdint.h>
#define ADC_RESULTION_BITS 12
void DMA_config (void)
// configure DMA0
{
// Setup DMA0
DMACTL0 = DMA0TSEL_24; // ADC12IFGx triggered
DMACTL4 = DMARMWDIS; // Read-modify-write disable
DMA0CTL &= ~DMAIFG;
DMA0CTL = DMADT_4+DMAEN+DMADSTINCR_0+DMASRCINCR_0+DMAIE; // Rpt single tranfer, unchanged src & dst, Int
DMA0SZ = 1; // Block size
__data20_write_long((uintptr_t) &DMA0SA,(uintptr_t) &ADC12MEM0); // Source block address
__data20_write_long((uintptr_t) &DMA0DA,(uintptr_t) &UCA1TXBUF); // Destination single address
}
void ADC_config()
{
// ADC Core control
XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX

ADC Interrupt flag is been set:

but DMA didn't transfer the data to the TxBuffer:

I am fully aware that the reading from 12bit ADC is 1 word wide (16bit) while the destination register UCATXBUF is only 8bit. According to the MSP430 user guide "When transferring word to byte, only the lower byte of the source word is transfered." So in this example, I am expecting 0xED to be written to UCATXBUF, but instead, it's reporting 0x00

  • I think the ADC configuration is right. The problem may lies on the DMA setting.

    Can you find some help from our example code:https://dev.ti.com/tirex/explore/node?node=ALkqBIa6R80LFOC7XAng.Q__IOGqZri__LATEST

    You can use the int value in ram to check whether the DMA send the value or it just send 0x00. Or you just decrease the input voltage.

  • I am a bit confused about how interrupt and DMA works together. Say if I have a ADC generating interrupt and this interrupt is configured as the trigger for DMA. When the interrupt is triggered, does ISR run or DMA execute? Afterall, DMA also takes a few clock cycle to run.

  • Following code is semi-working:

    Fullscreen
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    #include <msp430.h>
    #include <stdint.h>
    #define ADC_RESULTION_BITS 12
    void IO_config (void)
    // configure IOs
    {
    P1DIR |= 0x01; // configure P1.0 as output (RED LED)
    P4DIR |= 0x80; // configure P4.7 as output (GREEN LED)
    P1OUT = 0x00; // initialize LED to off
    P4OUT = 0x00; // initialize LED to off
    P4SEL = BIT4 + BIT5; // Port Configuration for UART
    }
    void DMA_config (void)
    // configure DMA0
    {
    // Setup DMA0
    DMACTL0 = DMA0TSEL_24; // ADC12IFGx triggered
    DMACTL4 = DMARMWDIS; // Read-modify-write disable
    XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX

    However, when I probe the UART Tx line, I see there's a dip on the Tx line. I figure this has nothing to do with the DMA and ADC, rather, it's somehow related to the UART. But the same UART portion of the code in a separate program work just fine without this glitch. Any idea? Thanks all

  • 1. As I double check, the code you posted before doesn't set the UART. Sorry, I miss that one.

    2. For the glitch, I have no idea. It doesn't seem to cause from digital signal. I advise you move out the other UART device to check if it is caused from MSP430.

  • It looks like the glitch was somehow related to watch dog reset. While, trying to fix this watch dog reset induced issue, I found out that I have a deeper problem. As I added the statement to halt watch dog timer, the program stopped running after the 1st DMA, meaning that DMA will happen once, and then no further DMA will be initialized. I have configured the DMA ISR to restart an ADC conversion by setting ADC12SC bit in ADC12CTL0. I thought this will trigger another ADC conversion and once ADC finished (signal by ADC12IFG), the result will be loaded to the UA1TXBUF by DMA and then the cycle repeat itself indefinitely. But it looks like that after the 1st time, ADC didn't trigger the DMA to transfer data over.

    Fullscreen
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    #include <msp430.h>
    #include <stdint.h>
    #define ADC_RESULTION_BITS 12
    void IO_config (void)
    // configure IOs
    {
    P1DIR |= 0x01; // configure P1.0 as output (RED LED)
    P4DIR |= 0x80; // configure P4.7 as output (GREEN LED)
    P1OUT = 0x00; // initialize LED to off
    P4OUT = 0x00; // initialize LED to off
    P4SEL = BIT4 + BIT5; // Port Configuration for UART
    }
    void DMA_config (void)
    // configure DMA0
    {
    // Setup DMA0
    DMACTL0 = DMA0TSEL_24; // ADC12IFGx triggered
    DMACTL4 = DMARMWDIS; // Read-modify-write disable
    XXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXXX

    For example, the 1st time, ADC12MEM0 is 0x04ED and the lower byte is been copied into UCA1TXBUF

               

    Scope also captured this data been sent out on UART

    If I continue execution of the program, no more transfer will happen. If I manually pause the program and look at the register values, I see ADC12MEM0 is loaded with new data and ADC12IFG is been updated, but no more DMA happens

         

    MSP430 user guide specifies "A [DMA] transfer is triggered by an ADC12IFG flag with the corresponding ADC12IE bit reset." Since I just want to copy the data out of ADC, so I don't have an ISR for ADC, therefore, ADC12IE is not set to '1' and it was kept at 0. I tried to set it to '1', but it doesn't solve the problem. Anyway what does it mean that "corresponding ADC12IE bit reset"...This is not clear to me... 

    I must have a silly mistake in my code. Can someone explain to me why the DMA transfer wasn't triggered again after the 1st time? Thanks everyone..

  • I see why, the DMAEN bit was cleared after each transfer in Single transfer. Should have put it in Repeated single transfer mode and then everything works.

**Attention** This is a public forum