ADCDMA技術解析

一、ADCDMA概述

ADCDMA是指將模擬信號轉換成數字信號,然後通過DMA技術將數字信號直接傳輸到內存中的一種技術。

ADCDMA技術在分散式控制系統、自動化控制、通信系統等領域得到廣泛應用。

二、ADCDMA顯示波形

通過ADCDMA技術可以獲取模擬信號的數字信號,然後將其顯示成波形。

/* ADCDMA顯示波形代碼示例 */
#include 
#include "inc/hw_types.h"
#include "inc/hw_memmap.h"
#include "inc/hw_adc.h"
#include "inc/hw_ints.h"
#include "driverlib/adc.h"
#include "driverlib/gpio.h"
#include "driverlib/sysctl.h"
#include "driverlib/interrupt.h"

/* ADC初始化,設置採樣率為250KSPS */
void ADC_Init(void)
{
    SysCtlPeripheralEnable(SYSCTL_PERIPH_ADC0);
    SysCtlPeripheralEnable(SYSCTL_PERIPH_GPIOE);
    GPIOPinTypeADC(GPIO_PORTE_BASE, GPIO_PIN_3);
    ADCSequenceConfigure(ADC0_BASE, 1, ADC_TRIGGER_PROCESSOR, 0);
    ADCSequenceStepConfigure(ADC0_BASE, 1, 0, ADC_CTL_CH0);
    ADCSequenceStepConfigure(ADC0_BASE, 1, 1, ADC_CTL_CH0);
    ADCSequenceStepConfigure(ADC0_BASE, 1, 2, ADC_CTL_CH0);
    ADCSequenceStepConfigure(ADC0_BASE, 1, 3, ADC_CTL_CH0|ADC_CTL_IE|ADC_CTL_END);
    ADCSequenceEnable(ADC0_BASE, 1);
}

/* DMA初始化 */
void DMA_Init(void)
{
    SysCtlPeripheralEnable(SYSCTL_PERIPH_DMA);
    SysCtlPeripheralEnable(SYSCTL_PERIPH_UDMA);
    uDMAEnable();
    uDMAControlBaseSet(g_psDMAControlTable);
    uDMAChannelAssign(UDMA_CH13_ADC0_1);
    uDMAChannelAttributeDisable(UDMA_CHANNEL_ADC0, UDMA_ATTR_ALL);
    uDMAChannelControlSet(UDMA_CHANNEL_ADC0 | UDMA_PRI_SELECT,
                          UDMA_SIZE_16 | UDMA_SRC_INC_NONE | UDMA_DST_INC_16 |
                          UDMA_ARB_64);
    uDMAChannelTransferSet(UDMA_CHANNEL_ADC0 | UDMA_PRI_SELECT,
                           UDMA_MODE_PINGPONG, (void*)(ADC0_BASE + ADC_O_SSFIFO1),
                           g_adcBuffer, ADC_BUFFER_SIZE/2);
    uDMAChannelEnable(UDMA_CHANNEL_ADC0);
}

/* 開始採樣 */
void ADC_Sample(void)
{
    /* ADCDMA採樣開始 */
    ADCProcessorTrigger(ADC0_BASE, 1);
    /* 等待DMA傳輸完成 */
    while(!uDMAChannelModeGet(UDMA_CHANNEL_ADC0 | UDMA_PRI_SELECT));
}

三、ADCDMA中斷多通道採集

ADCDMA中斷採集時,可以同步採集多個通道的信號。

/* ADCDMA中斷多通道採集代碼示例 */
#include 
#include "inc/hw_types.h"
#include "inc/hw_memmap.h"
#include "inc/hw_adc.h"
#include "inc/hw_ints.h"
#include "driverlib/adc.h"
#include "driverlib/gpio.h"
#include "driverlib/sysctl.h"
#include "driverlib/interrupt.h"

#define ADC_CHANNEL_NUM 2
uint32_t g_adcValues[ADC_CHANNEL_NUM];

void ADCInit(void)
{
    /* ADC0採樣率配置 */
    SysCtlPeripheralEnable(SYSCTL_PERIPH_ADC0);
    SysCtlPeripheralEnable(SYSCTL_PERIPH_GPIOE);
    GPIOPinTypeADC(GPIO_PORTE_BASE, GPIO_PIN_3);
    ADCSequenceConfigure(ADC0_BASE, 3, ADC_TRIGGER_PROCESSOR, 0);
    ADCSequenceStepConfigure(ADC0_BASE, 3, 0, ADC_CTL_CH0);
    ADCSequenceStepConfigure(ADC0_BASE, 3, 1, ADC_CTL_CH1|ADC_CTL_IE|ADC_CTL_END);
    ADCSequenceEnable(ADC0_BASE, 3);

    /* 中斷配置 */
    ADCIntEnable(ADC0_BASE, 3);
    ADCIntRegister(ADC0_BASE, 3, ADCIntHandler);
    IntPrioritySet(INT_ADC0SS3, 0x00);
    IntEnable(INT_ADC0SS3);
}

void ADCIntHandler(void)
{
    /* 清除中斷標誌位 */
    ADCIntClear(ADC0_BASE, 3);
    /* 獲取ADC數值 */
    ADCSequenceDataGet(ADC0_BASE, 3, g_adcValues);
}

void ADCCollect(void)
{
    /* ADC中斷採樣開始 */
    ADCProcessorTrigger(ADC0_BASE, 3);
    /* 等待中斷完成 */
    while(ADCIntStatus(ADC0_BASE, 3, false) == 0) {}
}

四、ADCDMA數據錯位

當ADCDMA採集數據出現錯位時,需要對數據進行調整或再次採樣。

五、ADCDMA只轉換了一次

如果只進行一次採集和轉換,可以採用CPU讀取方式,不需要使用DMA。

六、ADCDMA應用實例

以ADCDMA採集電流電壓信號並實時顯示為例,代碼如下:

/* ADCDMA電流電壓採集和顯示代碼示例 */
#include "inc/hw_memmap.h"
#include "inc/hw_types.h"
#include "inc/hw_adc.h"
#include "driverlib/adc.h"
#include "driverlib/gpio.h"
#include "driverlib/pin_map.h"
#include "driverlib/sysctl.h"
#include "driverlib/timer.h"
#include "utils/uartstdio.h"

/* 定義ADC通道以及數組 */
#define ADC_SEQUENCE_NUM             0
#define ADC_SEQUENCE_PRIORITY        ADC_CTL_CH0
#define ADC_SEQUENCE_SAMPLE_RATE     8000
#define ADC_STATUS_BUF_DEPTH         8
#define ADC_RESULT_BUF_DEPTH         8
int32_t pui32ADC0Status[ADC_STATUS_BUF_DEPTH];
int32_t pui32ADC0Value[ADC_RESULT_BUF_DEPTH];
int32_t i32ADCSum;
uint32_t ADCCurrentValue;
uint32_t ADCVoltageValue;
uint32_t ui32ADCSampleCount = 0;

void ADCIntHandler(void)
{
    ADCSequenceDataGet(ADC0_BASE, ADC_SEQUENCE_NUM, pui32ADC0Value);
    ADCSequenceDataGet(ADC0_BASE, ADC_SEQUENCE_NUM | ADC_CTL_CMP0, pui32ADC0Status);
    ui32ADCSampleCount++;
    /* 計算平均值 */
    for(uint8_t i = 0; i < ADC_STATUS_BUF_DEPTH; i++)
    {
        i32ADCSum += pui32ADC0Status[i];
    }
    ADCCurrentValue = i32ADCSum/ADC_STATUS_BUF_DEPTH;
    i32ADCSum = 0;
    for(uint8_t i = 0; i < ADC_RESULT_BUF_DEPTH; i++)
    {
        i32ADCSum += pui32ADC0Value[i];
    }
    ADCVoltageValue = i32ADCSum/ADC_RESULT_BUF_DEPTH;
    i32ADCSum = 0;
}
void TimerIntHandler(void)
{
    /* 顯示電流電壓 */
    UARTprintf("Current is %d mA, voltage is %d mV\n", ADCCurrentValue, ADCVoltageValue);
    /* 清除中斷標誌位 */
    TimerIntClear(TIMER0_BASE, TIMER_TIMA_TIMEOUT);
}
int main(void)
{
    /* 系統時鐘設置至80MHz */
    SysCtlClockSet(SYSCTL_SYSDIV_2_5|SYSCTL_USE_PLL|SYSCTL_OSC_MAIN|SYSCTL_XTAL_16MHZ);

    /* IO口設置為ADC輸入 */
    SysCtlPeripheralEnable(SYSCTL_PERIPH_GPIOE);
    GPIOPinTypeADC(GPIO_PORTE_BASE, GPIO_PIN_3);
    GPIOPinTypeADC(GPIO_PORTE_BASE, GPIO_PIN_2);

    /* ADC採樣初始化 */
    SysCtlPeripheralEnable(SYSCTL_PERIPH_ADC0);
    ADCClockConfigSet(ADC0_BASE, ADC_CLOCK_SRC_PLL | ADC_CLOCK_RATE_FULL, 8);
    ADCSequenceConfigure(ADC0_BASE, ADC_SEQUENCE_NUM, ADC_TRIGGER_PROCESSOR | ADC_SEQUENCE_PRIORITY, 0);
    ADCSequenceStepConfigure(ADC0_BASE, ADC_SEQUENCE_NUM, 0, ADC_CTL_CH0 | ADC_CTL_IE | ADC_CTL_END);
    ADCSequenceStepConfigure(ADC0_BASE, ADC_SEQUENCE_NUM | ADC_CTL_CMP0, 0, ADC_CTL_CH1 | ADC_CTL_IE | ADC_CTL_END);
    ADCSequenceOverflowConfigure(ADC0_BASE, ADC_SEQUENCE_NUM, ADC_OVS_FIELD_NONE | ADC_OVS_RATE_HALF);
    ADCSequenceOverflowConfigure(ADC0_BASE, ADC_SEQUENCE_NUM | ADC_CTL_CMP0, ADC_OVS_FIELD_NONE | ADC_OVS_RATE_HALF);
    ADCSequenceDMAEnable(ADC0_BASE, ADC_SEQUENCE_NUM);
    ADCSequenceDMAEnable(ADC0_BASE, ADC_SEQUENCE_NUM | ADC_CTL_CMP0);
    ADCIntRegister(ADC0_BASE, ADC_SEQUENCE_NUM, ADCIntHandler);
    ADCIntEnable(ADC0_BASE, ADC_SEQUENCE_NUM | ADC_CTL_CMP0);
    ADCSequenceEnable(ADC0_BASE, ADC_SEQUENCE_NUM | ADC_CTL_CMP0);
    ADCProcessorTrigger(ADC0_BASE, ADC_SEQUENCE_NUM);
    int32_t i32Err;
    i32Err = TimerConfigure(TIMER0_BASE, TIMER_CFG_PERIODIC);
    TimerLoadSet(TIMER0_BASE, TIMER_A, SysCtlClockGet()/100); // 1秒鐘100次
    TimerIntRegister(TIMER0_BASE, TIMER_A, TimerIntHandler);
    TimerIntEnable(TIMER0_BASE, TIMER_TIMA_TIMEOUT);
    TimerEnable(TIMER0_BASE, TIMER_A);

    /* 初始化UART輸出 */
    SysCtlPeripheralEnable(SYSCTL_PERIPH_UART0);
    SysCtlPeripheralEnable(SYSCTL_PERIPH_GPIOA);
    GPIOPinConfigure(GPIO_PA0_U0RX);
    GPIOPinConfigure(GPIO_PA1_U0TX);
    GPIOPinTypeUART(GPIO_PORTA_BASE, GPIO_PIN_0 | GPIO_PIN_1);
    UARTStdioConfig(0, 115200, SysCtlClockGet());

    while(1)
    {
    }
}

原創文章,作者:小藍,如若轉載,請註明出處:https://www.506064.com/zh-tw/n/242841.html

(0)
打賞 微信掃一掃 微信掃一掃 支付寶掃一掃 支付寶掃一掃
小藍的頭像小藍
上一篇 2024-12-12 12:52
下一篇 2024-12-12 12:52

相關推薦

  • Python熱重載技術

    Python熱重載技術是現代編程的關鍵功能之一。它可以幫助我們在程序運行的過程中,更新代碼而無需重新啟動程序。本文將會全方位地介紹Python熱重載的實現方法和應用場景。 一、實現…

    編程 2025-04-29
  • Python包絡平滑技術解析

    本文將從以下幾個方面對Python包絡平滑技術進行詳細的闡述,包括: 什麼是包絡平滑技術? Python中使用包絡平滑技術的方法有哪些? 包絡平滑技術在具體應用中的實際效果 一、包…

    編程 2025-04-29
  • parent.$.dialog是什麼技術的語法

    parent.$.dialog是一種基於jQuery插件的彈出式對話框技術,它提供了一個方便快捷的方式來創建各種類型和樣式的彈出式對話框。它是對於在網站開發中常見的彈窗、提示框等交…

    編程 2025-04-28
  • 微信小程序重構H5技術方案設計 Github

    本文旨在探討如何在微信小程序中重構H5技術方案,以及如何結合Github進行代碼存儲和版本管理。我們將從以下幾個方面進行討論: 一、小程序與H5技術對比 微信小程序與H5技術都可以…

    編程 2025-04-28
  • HTML sprite技術

    本文將從多個方面闡述HTML sprite技術,包含基本概念、使用示例、實現原理等。 一、基本概念 1、什麼是HTML sprite? HTML sprite,也稱CSS spri…

    編程 2025-04-28
  • Python工作需要掌握什麼技術

    Python是一種高級編程語言,它因其簡單易學、高效可靠、可擴展性強而成為最流行的編程語言之一。在Python開發中,需要掌握許多技術才能讓開發工作更加高效、準確。本文將從多個方面…

    編程 2025-04-28
  • 開源腦電波技術

    本文將會探討開源腦電波技術的應用、原理和示例。 一、腦電波簡介 腦電波(Electroencephalogram,簡稱EEG),是一種用於檢測人腦電活動的無創性技術。它通過在頭皮上…

    編程 2025-04-27
  • 阿里Python技術手冊

    本文將從多個方面對阿里Python技術手冊進行詳細闡述,包括規範、大數據、Web應用、安全和調試等方面。 一、規範 Python的編寫規範對於代碼的可讀性和可維護性有很大的影響。阿…

    編程 2025-04-27
  • TaintGraphTraversal – 使用數據流分析技術解決污點問題

    TaintGraphTraversal是一種數據流分析技術,旨在解決應用程序中污點問題。通過在程序中跟蹤數據流和標記數據源,TaintGraphTraversal可以確定哪些數據被…

    編程 2025-04-27
  • 網路數據爬蟲技術用法介紹

    網路數據爬蟲技術是指通過一定的策略、方法和技術手段,獲取互聯網上的數據信息並進行處理的一種技術。本文將從以下幾個方面對網路數據爬蟲技術做詳細的闡述。 一、爬蟲原理 網路數據爬蟲技術…

    編程 2025-04-27

發表回復

登錄後才能評論