STM32 ADC單通道與多通道_DMA學習筆記

轉自:https://blog.csdn.net/dmfylb/article/details/72802690

第一部分  ADC單路採集

下面我們將 PC0引腳配置成 AD1的通道10爲例進行講解。下面是ADC通道與 GPIO 對應表 :

3.1 首先我們應將PC0 設置成模擬輸入:

#include "adc.h"
/* 爲 何 定 義 ADC1_DR_Address爲((u32)0x40012400+0x4c), 因 爲 存 放AD 轉 換 結 果 的 寄存 器 的 地址 就 是:0x4001244c*/ #define ADC1_DR_Address ((u32)0x40012400+0x4c)
/*定義變量 ADC_ConvertedValue,放AD1通道 10轉換的數據*/ __IO uint16_t ADC_ConvertedValue;
static void ADC1_GPIO_Config(void)
{
GPIO_InitTypeDef GPIO_InitStructure;    /* Enable ADC1 and GPIOC clock */
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 | RCC_APB2Periph_GPIOC,ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; GPIO_Init(GPIOC, &GPIO_InitStructure);

}

 

3.2 設置完端口後下一步當然是對AD進行初始化:

這裏需要補充一個知識點 DMA,DMA就相當與CPU 的一個祕書,他的作用就是幫CPU減輕負擔的。說的再具體點就是幫 CPU 來轉移數據的。我們都知道,AD每次轉換結束後會將轉換的結果放到一個固定的寄存器裏,以往我們如果想將該寄存器中的值賦給某一變量時會用到賦值語句,如果不用DMA,則賦值語句便要CPU來完成,CPU本來就要忙着處理其他事情,現在還要來解決賦值語句這麼簡單的問題,肯到會蛋疼。所以需要DMA 這個祕書來幫他解決這個問題。由於DMA 只是個祕書,所以比較笨,你只有把任務交代清楚了她才能很好的完成任務。那麼怎樣來給DMA 吩咐任務呢,聰明的人肯定想到了,那當然是“DMA_Init(DMA1_Channel1, &DMA_InitStructure)”這個函數啦。下面就來一步步的來給DMA 交代任務。

/* 函數名:ADC1_Mode_Config
* 描述 :配置 ADC1 的工作模式爲 MDA 模式*輸入 :無* 輸出 :無
* 調用 :內部調用 */
static void ADC1_Mode_Config(void)
{
DMA_InitTypeDef DMA_InitStructure;
ADC_InitTypeDef ADC_InitStructure;
/* 將與DMA 有關的寄存器設我初始值 */
DMA_DeInit(DMA1_Channel1);
/*定義 DMA 外設基地址,這裏的ADC1_DR_Address是用戶自己定義的,即爲存放轉換結果的寄存器 ,他的作用
就是告訴 DMA 取數就到 ADC1_DR_Address 這裏來取。*/
DMA_InitStructure.DMA_PeripheralBaseAddr =ADC1_DR_Address;
/*定義內存基地址,即告訴 DMA要將從AD 中取來的數放到ADC_ConvertedValue中*/
DMA_InitStructure.DMA_MemoryBaseAddr =(u32)&ADC_ConvertedValue;
/*定義 AD 外設作爲數據傳輸的來源,即告訴DMA是將 AD中的數據取出放到內存中,不能反過來*/
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
/*指定 DMA 通道的DMA緩存的大小,即告訴DMA開闢幾個內存空間,由於我們只取通道10的 AD數據所以只需
開闢一個內存空間*/
DMA_InitStructure.DMA_BufferSize = 1;
/*設定寄存器地址固定,即告訴 DMA,只從固定的一個地方取數*/
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
/*設定內存地址固定,即每次 DMA,,只將數搬到固定的內存中*/
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
/*設定外設數據寬度,即告訴 DMA要取的數的大小*/
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;
/*設定內存的的寬度*/
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
/*設定 DMA 工作再循環緩存模式,即告訴DMA要不停的搬運,不能偷懶*/
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
/*設定 DMA 選定的通道軟件優先級*/
DMA_InitStructure.DMA_Priority = DMA_Priority_High;
DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
DMA_Init(DMA1_Channel1, &DMA_InitStructure);
/* Enable DMA channel1,CPU有好幾個DMA 祕書,現在只用DMA1_Channel1這個祕書*/

DMA_Cmd(DMA1_Channel1, ENABLE);

 

/*設置ADC 工作在獨立模式*/
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
/*規定 AD 轉換工作在單次模式,即對一個通道採樣*/
ADC_InitStructure.ADC_ScanConvMode = DISABLE
/*設定 AD 轉化在連續模式*/
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
/*不使用外部促發轉換*/
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
/*採集的數據在寄存器中以右對齊的方式存放*/
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
/*設定要轉換的 AD通道數目*/
ADC_InitStructure.ADC_NbrOfChannel = 1;
ADC_Init(ADC1, &ADC_InitStructure);
/*配置 ADC 時鐘,爲PCLK2的 8分頻,即9MHz*/
RCC_ADCCLKConfig(RCC_PCLK2_Div8);
/*配置 ADC1 的通道 11 爲55.5個採樣週期 */
ADC_RegularChannelConfig(ADC1, ADC_Channel_10, 1, ADC_SampleTime_55Cycles5);
/* Enable ADC1 DMA */
ADC_DMACmd(ADC1, ENABLE);
/* Enable ADC1 */
ADC_Cmd(ADC1, ENABLE);
/*復位校準寄存器 */
ADC_ResetCalibration(ADC1);
/*等待校準寄存器復位完成 */
while(ADC_GetResetCalibrationStatus(ADC1));
/* ADC 校準 */
ADC_StartCalibration(ADC1);
/* 等待校準完成*/
while(ADC_GetCalibrationStatus(ADC1));
/* 由於沒有采用外部觸發,所以使用軟件觸發 ADC轉換*/
ADC_SoftwareStartConvCmd(ADC1, ENABLE);
}
配置完以上的程序,那麼 AD 每轉換一次,DMA 都會將轉換結果搬到變量

ADC_ConvertedValue 中,而不需用每次都用賦值語句來取值AD 轉換的值。

 

第二部分:AD多路採樣

#include "adc.h"
#define ADC1_DR_Address ((u32)0x40012400+0x4c)
/*定義數組變量 ADC_ConvertedValue[2],分別放AD1通道 10和11 轉換的數據*/
__IO uint16_t ADC_ConvertedValue[2];
/* * 函數名: ADC1_GPIO_Config *描述 :使能ADC1 和 DMA1的時鐘,設置PC0,PC1爲模擬輸入 *輸入: 無*

輸出 :無* 調用 :內部調用*/


static void ADC1_GPIO_Config(void)
{ GPIO_InitTypeDef GPIO_InitStructure;
/* Enable DMA clock */
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
/* Enable ADC1 and GPIOC clock */
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1| RCC_APB2Periph_GPIOC,ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 |GPIO_Pin_1;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
GPIO_Init(GPIOC, &GPIO_InitStructure);
}
/*

static void ADC1_GPIO_Config(void)
{ GPIO_InitTypeDef GPIO_InitStructure;
/* Enable DMA clock */
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
/* Enable ADC1 and GPIOC clock */
RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1| RCC_APB2Periph_GPIOC,ENABLE);
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 |GPIO_Pin_1;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
GPIO_Init(GPIOC, &GPIO_InitStructure);
}
/* 函數名: ADC1_Mode_Config *描述 :配置ADC1 的工作模式爲 MDA 模式*輸入 :無* 輸出 :無*調
用 :內部調用 */
static void ADC1_Mode_Config(void)
{
DMA_InitTypeDef DMA_InitStructure;
ADC_InitTypeDef ADC_InitStructure;
/* DMA channel1 configuration */
DMA_DeInit(DMA1_Channel1);
/*定義 DMA 外設基地址,即爲存放轉換結果的寄存器*/
DMA_InitStructure.DMA_PeripheralBaseAddr =ADC1_DR_Address;
/*定義內存基地址*/
DMA_InitStructure.DMA_MemoryBaseAddr =(u32)&ADC_ConvertedValue;
/*定義 AD 外設作爲數據傳輸的來源*/
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;
/*指定 DMA 通道的DMA緩存的大小,即需要開闢幾個內存空間,本實驗有兩個轉換通道,所以開闢兩個*/
DMA_InitStructure.DMA_BufferSize = 2;
/*設定寄存器地址固定*/
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
/*設定內存地址遞加,即每次DMA都是將該外設寄存器中的值傳到兩個內存空間中*/
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
/*設定外設數據寬度*/
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;
/*設定內存的的寬度*/
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
/*設定 DMA 工作再循環緩存模式*/
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
/*設定 DMA 選定的通道軟件優先級*/
DMA_InitStructure.DMA_Priority = DMA_Priority_High;
DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
DMA_Init(DMA1_Channel1, &DMA_InitStructure);
/* Enable DMA channel1 */
DMA_Cmd(DMA1_Channel1, ENABLE);
/*設置 ADC 工作在獨立模式*/
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;
/*規定 AD 轉換工作在掃描模式,即對多個通道採樣*/
ADC_InitStructure.ADC_ScanConvMode = ENABLE;
/*設定 AD轉化在連續模式*/
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;
/*不使用外部促發轉換*/
ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
/*採集的數據在寄存器中以右對齊的方式存放*/
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
/*設定要轉換的 AD 通道數目*/
ADC_InitStructure.ADC_NbrOfChannel = 2;
ADC_Init(ADC1, &ADC_InitStructure);
/*配置 ADC 時鐘,爲PCLK2的 8分頻,即9MHz*/
RCC_ADCCLKConfig(RCC_PCLK2_Div8);
/*配置 ADC1 的通道 10 和11的轉換先後順序以及採樣時間爲爲55.5個採樣週期 */
ADC_RegularChannelConfig(ADC1,ADC_Channel_10,1,ADC_SampleTime_55Cycles5);
ADC_RegularChannelConfig(ADC1, ADC_Channel_11, 2, ADC_SampleTime_55Cycles5);
/* Enable ADC1 DMA */
ADC_DMACmd(ADC1, ENABLE);
/* Enable ADC1 */
ADC_Cmd(ADC1, ENABLE);
/*復位校準寄存器 */
ADC_ResetCalibration(ADC1);
/*等待校準寄存器復位完成 */
while(ADC_GetResetCalibrationStatus(ADC1));
/* ADC 校準 */
ADC_StartCalibration(ADC1);
/* 等待校準完成*/
while(ADC_GetCalibrationStatus(ADC1));
/* 由於沒有采用外部觸發,所以使用軟件觸發 ADC轉換*/

ADC_SoftwareStartConvCmd(ADC1, ENABLE); }

 

!!!!!單通道採樣與多通道採樣的不同點都在第二段程序中用紅色標出來了,注意比較。 總結:DMA就是一個

無私奉獻的搬運工,想將外設寄存器中的值放入內存中原本需要CPU 來完成,現在DMA來幫 CPU完成,這在一定

程度上解放了CPU。

 

 

STM32 ADC 多通道轉換

描述:用 ADC 連續採集 11 路模擬信號,並由 DMA 傳輸到內存。ADC 配置爲掃描並且連續轉換模式,ADC 的時鐘配置爲 12MHZ。在每次轉換結束後,由 DMA循環將轉換的數據傳輸到內存中。ADC 可以連續採集 N 次求平均值。最後通過串口傳輸出最後轉換的結果。

程序如下:
#include "stm32f10x.h"//這個頭文件包括STM32F10x所有外圍寄存器、位、內存映
射的定義
#include "eval.h" //頭文件(包括串口、按鍵、LED的函數聲明)
#include "SysTickDelay.h"
#include "UART_INTERFACE.h"
#include <stdio.h>
#define N 50 //每通道採50次
#define M 12 //爲12個通道
vu16 AD_Value[N][M]; //用來存放ADC轉換結果,也是DMA的目標地址
vu16 After_filter[M]; //用來存放求平均值之後的結果
int i;
/*GPIO 管腳的配置
選用 ADC 的通道0 1 2 8 9 10 11 12 13 14 15,分別對應的管腳爲PA0 PA1
PA2 PB0 PB1 PC0 PC1 PC2 PC3 PC4 PC5
串口使用 USART1其中TX爲PA9,RX爲PA10 */
void GPIO_Configuration(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
/* Configure USART1 Tx (PA.09) as alternate function push‐pull */
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP; //因爲USART1管腳是以複用的形
式接到 GPIO 口上的,所以使用複用推輓式輸出
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOA, &GPIO_InitStructure);
/* Configure USART1 Rx (PA.10) as input floating */
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
GPIO_Init(GPIOA, &GPIO_InitStructure);
//PA0/1/2 作爲模擬通道輸入引腳
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0| GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模擬輸入引腳
GPIO_Init(GPIOA, &GPIO_InitStructure);
//PB0/1 作爲模擬通道輸入引腳
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0|GPIO_Pin_1;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模擬輸入引腳
GPIO_Init(GPIOB, &GPIO_InitStructure);
//PC0/1/2/3/4/5 作爲模擬通道輸入引腳
GPIO_InitStructure.GPIO_Pin =
GPIO_Pin_0|GPIO_Pin_1|GPIO_Pin_2|GPIO_Pin_3|GPIO_Pin_4|GPIO_Pin_5;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; //模擬輸入引腳
GPIO_Init(GPIOC, &GPIO_InitStructure);
}
}
/*配置系統時鐘,使能各外設時鐘*/
void RCC_Configuration(void)
{
ErrorStatus HSEStartUpStatus;
RCC_DeInit(); //RCC 系統復位
RCC_HSEConfig(RCC_HSE_ON); //開啓HSE
HSEStartUpStatus = RCC_WaitForHSEStartUp(); //等待HSE準備好
if(HSEStartUpStatus == SUCCESS)
{
FLASH_PrefetchBufferCmd(FLASH_PrefetchBuffer_Enable); //Enable Prefetch Buffer
FLASH_SetLatency(FLASH_Latency_2); //Set 2 Latency cycles
RCC_HCLKConfig(RCC_SYSCLK_Div1); //AHB clock = SYSCLK
RCC_PCLK2Config(RCC_HCLK_Div1); //APB2 clock = HCLK
RCC_PCLK1Config(RCC_HCLK_Div2); //APB1 clock = HCLK/2
RCC_PLLConfig(RCC_PLLSource_HSE_Div1, RCC_PLLMul_6);//PLLCLK = 12MHz * 6 = 72
MHz
RCC_PLLCmd(ENABLE); //Enable PLL
while(RCC_GetFlagStatus(RCC_FLAG_PLLRDY) == RESET);//Wait till PLL is ready
RCC_SYSCLKConfig(RCC_SYSCLKSource_PLLCLK); //Select PLL as system clock source
while(RCC_GetSYSCLKSource() != 0x08); //Wait till PLL is used as system clock source
/*使能各個外設時鐘*/
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOB
| RCC_APB2Periph_GPIOC |RCC_APB2Periph_ADC1 | RCC_APB2Periph_AFIO
|RCC_APB2Periph_USART1, ENABLE ); //使能ADC1通道時鐘,各個管腳時鐘
/* Configure ADCCLK such as ADCCLK = PCLK2/6 */
RCC_ADCCLKConfig(RCC_PCLK2_Div6); //72M/6=12,ADC最大時間不能超過14M
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);//使能DMA傳輸
}
}
/*配置ADC1*/
void ADC1_Configuration(void)
{
ADC_InitTypeDef ADC_InitStructure;
ADC_DeInit(ADC1); //將外設ADC1的全部寄存器重設爲缺省值
/* ADC1 configuration ‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐‐*/
ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;//ADC工作模式:ADC1和ADC2
工作在獨立模式
ADC_InitStructure.ADC_ScanConvMode =ENABLE; //模數轉換工作在掃描模式
ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;//模數轉換工作在連續轉換模

ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;//外 部 觸 發 轉 換
關閉
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;//ADC數據右對齊
ADC_InitStructure.ADC_NbrOfChannel = M; //順序進行規則轉換的ADC通道的數目
ADC_Init(ADC1, &ADC_InitStructure); //根據ADC_InitStruct中指定的參數初始化外設
ADCx 的寄存器
/* ADC1 regular channel11 configuration */
//設置指定ADC的規則組通道,設置它們的轉化順序和採樣時間
//ADC1,ADC 通道x,規則採樣順序值爲y,採樣時間爲239.5週期
ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_239Cycles5 );
ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 2, ADC_SampleTime_239Cycles5 );
ADC_RegularChannelConfig(ADC1, ADC_Channel_2, 3, ADC_SampleTime_239Cycles5 );
ADC_RegularChannelConfig(ADC1, ADC_Channel_3, 4, ADC_SampleTime_239Cycles5 );
ADC_RegularChannelConfig(ADC1, ADC_Channel_8, 5, ADC_SampleTime_239Cycles5 );
ADC_RegularChannelConfig(ADC1, ADC_Channel_9, 6, ADC_SampleTime_239Cycles5 );
ADC_RegularChannelConfig(ADC1, ADC_Channel_10, 7, ADC_SampleTime_239Cycles5 );
ADC_RegularChannelConfig(ADC1, ADC_Channel_11, 8, ADC_SampleTime_239Cycles5 );
ADC_RegularChannelConfig(ADC1, ADC_Channel_12, 9, ADC_SampleTime_239Cycles5 );
ADC_RegularChannelConfig(ADC1, ADC_Channel_13, 10, ADC_SampleTime_239Cycles5 );
ADC_RegularChannelConfig(ADC1, ADC_Channel_14, 11, ADC_SampleTime_239Cycles5 );
ADC_RegularChannelConfig(ADC1, ADC_Channel_15, 12, ADC_SampleTime_239Cycles5 );
// 開啓ADC的DMA支持(要實現DMA功能,還需獨立配置DMA通道等參數)
ADC_DMACmd(ADC1, ENABLE);
/* Enable ADC1 */
ADC_Cmd(ADC1, ENABLE); //使能指定的ADC1
/* Enable ADC1 reset calibaration register */
ADC_ResetCalibration(ADC1); //復位指定的ADC1的校準寄存器
/* Check the end of ADC1 reset calibration register */
while(ADC_GetResetCalibrationStatus(ADC1)); //獲取ADC1復位校準寄存器的狀態,設置
狀態則等待
/* Start ADC1 calibaration */
ADC_StartCalibration(ADC1); //開始指定ADC1的校準狀態
/* Check the end of ADC1 calibration */
while(ADC_GetCalibrationStatus(ADC1)); //獲取指定ADC1的校準程序,設置狀態則
等待
}
/*配置DMA*/
void DMA_Configuration(void)
{
/* ADC1 DMA1 Channel Config */
DMA_InitTypeDef DMA_InitStructure;
DMA_DeInit(DMA1_Channel1); //將DMA的通道1寄存器重設爲缺省值
DMA_InitStructure.DMA_PeripheralBaseAddr = (u32)&ADC1‐>DR;//DMA外設ADC基地

DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&AD_Value;//DMA內存基地址
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC;//內存作爲數據傳輸的目的地
DMA_InitStructure.DMA_BufferSize = N*M; //DMA通道的DMA緩存的大小
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;//外設地址寄存器
不變
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;//內存地址寄存器遞

DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord;//數
據寬度爲 16 位
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;//數據寬度
爲 16 位
DMA_InitStructure.DMA_Mode = DMA_Mode_Circular; //工作在循環緩存模式
DMA_InitStructure.DMA_Priority = DMA_Priority_High;//DMA通道x擁有高優先級
DMA_InitStructure.DMA_M2M = DMA_M2M_Disable; //DMA 通道x沒有設置爲內存到
內存傳輸
DMA_Init(DMA1_Channel1, &DMA_InitStructure); //根據DMA_InitStruct中指定的參數
初始化 DMA 的通道
}
//配置所有外設
void Init_All_Periph(void)
{
RCC_Configuration();
GPIO_Configuration();
ADC1_Configuration();
DMA_Configuration();
//USART1_Configuration();
USART_Configuration(9600);
}
/*獲取ADC的值,將二進制換算爲十進制*/
u16 GetVolt(u16 advalue)
{
return (u16)(advalue * 330 / 4096); //求的結果擴大了100倍,方便下面求出小數
}
/*求平均值函數*/
void filter(void)
{
int sum = 0;
u8 count;
for(i=0;i<12;i++)
{
for ( count=0;count<N;count++)
{
sum += AD_Value[count][i];
}
After_filter[i]=sum/N;
sum=0;
}
}
int main(void)
{
u16 value[M];
init_All_Periph();
SysTick_Initaize();
/* Start ADC1 Software Conversion */
ADC_SoftwareStartConvCmd(ADC1, ENABLE);
DMA_Cmd(DMA1_Channel1, ENABLE); //啓動DMA通道
while(1)
{
while(USART_GetFlagStatus(USART1,USART_FLAG_TXE)==RESET);//等待傳輸完成否則
第一位數據容易丟失
filter();
for(i=0;i<12;i++)
{
value[i]= GetVolt(After_filter[i]);
printf("value[%d]:\t%d.%dv\n",i,value[i]/100,value[i]%100) ;
delay_ms(100);
}
}

}

 

總結

一、該程序中的兩個宏定義,M 和 N,分別代表有多少個通道,每個通道轉換多少次,可以修改其值。曾出現的問題:配置時鐘時要知道外部晶振是多少,以便準確配置時鐘。將轉換值由二進制轉換爲十進制時,要先擴大 100 倍,方便顯示小數。最後串口輸出 時 在 printf 語 句 之 前 加 這 句 代 碼 , 防 止 輸 出 的 第 一 位 數 據 丟 失 :

while(USART_GetFlagStatus(USART1,USART_FLAG_TXE)==RESET);

 

 

二、另外用多路ADC採集的時候,全部沒接外部電路,每一個ADC都有電壓值,而且還不一定一樣大?這是爲什麼?沒接外電路的時候,電壓值不應該是零嗎?

答:AD口懸空讀入的AD值爲隨機,測靜電。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章