DMA transfer1. Learning objectivesDMA principle2. Hardware Construction3. Experimental steps1. Open the SYSCONFIG configuration tool2. ADC parameter configuration3. DMA parameter configuration4. Write the program5. Compile4. Program Analysis5. Experimental phenomenon
DMA (Direct Memory Access) is a technology used in computer systems that allows peripherals to directly access memory without having to go through the CPU to complete data transmission. It enables data to be exchanged efficiently between peripherals (such as ADC, DAC, memory, etc.) and memory, reducing the burden on the CPU and improving the overall performance of the system.
Principle:
The DMA controller of MSPM0G3507 has the following features:
By looking at TI's data sheet, in addition to the common address addressing mode between memory and peripherals, the DMA function also provides two expansion modes: Fill Mode and Table Mode. DMA channels are divided into basic type and full-function type.
By looking at the data sheet, only the full-function type DMA channel supports repeated transfers, early interrupts, and extended modes. The basic function DMA channel supports basic data transfers and interrupts, but is sufficient to meet simple data transfer requirements.
The DMA of the MSPM0G3507 supports four transfer modes, and each channel can configure its transfer mode separately. For example, channel 0 can be configured as a single word or single byte transfer mode, while channel 1 is configured as a block transfer mode, and channel 2 uses a repeated block transfer mode. The transfer mode is configured independently of the addressing mode. Any addressing mode can be used with any transfer mode. Three types of data can be transferred, which can be selected by the .srcWidth and .destWidth control bits. The source and destination locations can be byte, short word, or word data. It also supports transfers in byte-to-byte, short word-to-short word, word-to-word, or any combination. As follows:
Based on these three modes, MSPM0G3507 provides 6 addressing modes,
Fixed address to fixed address:
Fixed address to block of address :
Block of address to fixed address:
Block of address to block of address:
Fill data to block of address:
Data table to specific address:
In this case, ADC will be used to collect the voltage of the PA27 pin, and the data collected by ADC will be directly moved to the specified address through DMA.
Open the blank project empty in the SDK in KEIL.
Select and open, open the empty.syscfg file in the keil interface, When the empty.syscfg file is open, select Open the SYSCONFIG GUI interface in the Tools column.
The ADC configuration here is the same as the ADC acquisition case.
Configure the resolution and interrupt of the ADC. This case does not use interrupts, so do not select anything here.
Parameter description:
Configure DMA Trigger: Whether to enable the DMA configuration function, select Enable here.
Enable DMA Trigger: Enable DMA trigger.
DMA Samples Count: DMA sample count. Set to 1.
Enable DMA Triggers: DMA trigger mode. Configured as MEM0 result loaded interrupt
ADC result register 0 is loaded interrupt.
Address Mode: DMA addressing mode. Configured as Fixed addr. to Block addr.
Fixed address to data block address.
Source Length: Data width of source address. Configured as Half Word (2 Bytes)
Data width 2 bytes.
Destination Length: Data width configuration of destination address. Same as above.
Destination Address Direction: Destination address direction. Select Increment
to enable increment of data block address.
Configure Transfer size: Whether to configure the amount of data to be transferred. Select Enable here.
Transfer Size: The amount of data to be transferred. Configured as 10, the amount of data transferred in one data block is 10 numbers.
Transfer Mode: DMA transfer mode. Configure to Repeat Single
to repeat a single transfer.
Source Address Increment: Whether the source address is incremented. Not enabled.
Destination Address Increment: Whether the destination address is incremented. Not enabled.
Enable Channel Interrupt: Whether to enable DMA channel interrupt. Not enabled.
Click SAVE to save the configuration in SYSCONFIG, then close SYSCONFIG and return to keil.
Select Yes to All
in the pop-up window
Similarly, we also need to confirm whether the ti_msp_dl_config.c and ti_msp_dl_config.h files are updated. Compile directly, and the compilation will automatically update to keil. If there is no update, we can also copy the file content in SYSCONFIG.
In the empty.c file, write the following code
x#include "ti_msp_dl_config.h"
#include "stdio.h"
volatile unsigned int delay_times = 0;
volatile bool gCheckADC; //ADC采集成功标志位 ADC acquisition success flag
volatile uint16_t ADC_VALUE[20];
void delay_ms(unsigned int ms); //搭配滴答定时器的精准毫秒级延时 Precise millisecond delay with tick timer
unsigned int adc_getValue(unsigned int number);//读取ADC的数据 Read ADC data
/******************************串口重定向 Serial port redirection***************************************/
#if !defined(__MICROLIB)
//不使用微库的话就需要添加下面的函数
// If you don't use the micro library, you need to add the following function
#if (__ARMCLIB_VERSION <= 6000000)
//如果编译器是AC5 就定义下面这个结构体
//If the compiler is AC5, define the following structure
struct __FILE
{
int handle;
};
#endif
FILE __stdout;
//定义_sys_exit()以避免使用半主机模式
// Define _sys_exit() to avoid using semihosting mode
void _sys_exit(int x)
{
x = x;
}
#endif
//printf函数重定义 printf function redefinition
int fputc(int ch, FILE *stream)
{
//当串口0忙的时候等待,不忙的时候再发送传进来的字符
// Wait when serial port 0 is busy, and send the incoming characters when it is not busy
while( DL_UART_isBusy(UART_0_INST) == true );
DL_UART_Main_transmitData(UART_0_INST, ch);
return ch;
}
/*********************************************************************/
int main(void)
{
unsigned int adc_value = 0;
unsigned int voltage_value = 0;
int i = 0;
SYSCFG_DL_init();
//清除串口中断标志 Clear the serial port interrupt flag
NVIC_ClearPendingIRQ(UART_0_INST_INT_IRQN);
//开启串口中断 Enable serial port interrupt
NVIC_EnableIRQ(UART_0_INST_INT_IRQN);
//设置DMA搬运的起始地址 Set the starting address of DMA transfer
DL_DMA_setSrcAddr(DMA, DMA_CH0_CHAN_ID, (uint32_t) &ADC0->ULLMEM.MEMRES[0]);
//设置DMA搬运的目的地址 Set the destination address of DMA transfer
DL_DMA_setDestAddr(DMA, DMA_CH0_CHAN_ID, (uint32_t) &ADC_VALUE[0]);
//开启DMA Enable DMA
DL_DMA_enableChannel(DMA, DMA_CH0_CHAN_ID);
//开启ADC转换 Start ADC conversion
DL_ADC12_startConversion(ADC_VOLTAGE_INST);
printf("adc_dma Demo start\r\n");
while (1)
{
//获取ADC数据 Get ADC data
adc_value = adc_getValue(10);
printf("adc value:%d\r\n", adc_value);
//将ADC采集的数据换算为电压 Convert the data collected by ADC into voltage
voltage_value = (int)((adc_value/4095.0*3.3)*100);
printf("voltage value:%d.%d%d\r\n",
voltage_value/100,
voltage_value/10%10,
voltage_value%10 );
delay_ms(1000);
}
}
//延时函数 Delay function
void delay_ms(unsigned int ms)
{
delay_times = ms;
while( delay_times != 0 );
}
//读取ADC的数据 Read ADC data
unsigned int adc_getValue(unsigned int number)
{
unsigned int gAdcResult = 0;
unsigned char i = 0;
//采集多次累加 Collect multiple times and accumulate
for( i = 0; i < number; i++ )
{
gAdcResult += ADC_VALUE[i];
}
//均值滤波 Mean Filter
gAdcResult /= number;
return gAdcResult;
}
//滴答定时器的中断服务函数 Tick timer interrupt service function
void SysTick_Handler(void)
{
if( delay_times != 0 )
{
delay_times--;
}
}
Click the Rebuild icon. The following prompt appears, indicating that the compilation is complete and there are no errors.
xxxxxxxxxx
//读取ADC的数据 Read ADC data
unsigned int adc_getValue(unsigned int number)
{
unsigned int gAdcResult = 0;
unsigned char i = 0;
//采集多次累加 Collect multiple times and accumulate
for( i = 0; i < number; i++ )
{
gAdcResult += ADC_VALUE[i];
}
//均值滤波 Mean Filter
gAdcResult /= number;
return gAdcResult;
}
This function filters by accumulating the data in the ADC_VALUE
array and calculating the average value. This can reduce the noise of a single ADC sampling. number
is the number of samples, usually set to 10 or more to improve the stability of the data.
xxxxxxxxxx
int main(void)
{
unsigned int adc_value = 0;
unsigned int voltage_value = 0;
int i = 0;
SYSCFG_DL_init();
//清除串口中断标志 Clear the serial port interrupt flag
NVIC_ClearPendingIRQ(UART_0_INST_INT_IRQN);
//开启串口中断 Enable serial port interrupt
NVIC_EnableIRQ(UART_0_INST_INT_IRQN);
//设置DMA搬运的起始地址 Set the starting address of DMA transfer
DL_DMA_setSrcAddr(DMA, DMA_CH0_CHAN_ID, (uint32_t) &ADC0->ULLMEM.MEMRES[0]);
//设置DMA搬运的目的地址 Set the destination address of DMA transfer
DL_DMA_setDestAddr(DMA, DMA_CH0_CHAN_ID, (uint32_t) &ADC_VALUE[0]);
//开启DMA Enable DMA
DL_DMA_enableChannel(DMA, DMA_CH0_CHAN_ID);
//开启ADC转换 Start ADC conversion
DL_ADC12_startConversion(ADC_VOLTAGE_INST);
printf("adc_dma Demo start\r\n");
while (1)
{
//获取ADC数据 Get ADC data
adc_value = adc_getValue(10);
printf("adc value:%d\r\n", adc_value);
//将ADC采集的数据换算为电压 Convert the data collected by ADC into voltage
voltage_value = (int)((adc_value/4095.0*3.3)*100);
printf("voltage value:%d.%d%d\r\n",
voltage_value/100,
voltage_value/10%10,
voltage_value%10 );
delay_ms(1000);
}
}
Call SYSCFG_DL_init()
function to initialize the system, clear and enable serial port interrupt for serial port data transmission. Set DMA source address to ADC0->ULLMEM.MEMRES[0]
, which stores ADC conversion results. Set DMA destination address to ADC_VALUE[0]
, and store DMA results in this array. Enable DMA channel and start ADC conversion. Get 10 ADC sampling results through adc_getValue(10)
and perform mean filtering to return a stable ADC value. Convert ADC value to voltage value, assuming ADC has 12-bit resolution (4095 is the maximum value), and convert it to voltage value in the range of 0-3.3V.
After the program is downloaded, configure the serial port assistant as shown below, adjust the potentiometer knob, output voltage to PA27 pin, and after ADC loads data acquisition into ADC result register 0, DMA is triggered, and DMA will move data to memory variable ADC_VALUE. Data can be obtained by directly calling ADC_VALUE.