找回密碼
 立即注冊(cè)

QQ登錄

只需一步,快速開始

搜索
查看: 1688|回復(fù): 5
打印 上一主題 下一主題
收起左側(cè)

STM32單片機(jī)TIM觸發(fā)ADC采樣

[復(fù)制鏈接]
跳轉(zhuǎn)到指定樓層
樓主
ID:1095470 發(fā)表于 2023-10-20 11:43 | 只看該作者 回帖獎(jiǎng)勵(lì) |倒序?yàn)g覽 |閱讀模式
本人最近在學(xué)習(xí)使用TIM來觸發(fā)ADC采樣,使用的是TIM1_CC4事件來觸發(fā)ADC1通道4的采樣,TIM_CH4通道對(duì)應(yīng)的是PA11引腳,ADC1_CH4對(duì)應(yīng)的是PA4引腳,由變量DC_ADC來存儲(chǔ)ADC1通道4的采樣值。使用ST-Link進(jìn)行仿真,由示波器測(cè)量得到TIM1_CH4的輸出波形是50%占空比的方波,符合預(yù)期,說明TIM1的輸出信號(hào)(即ADC1的觸發(fā)信號(hào))是沒有問題的。但是ADC1_DR寄存器存放的值一直是校準(zhǔn)完成后校準(zhǔn)因子的值,且程序一直運(yùn)行在主函數(shù)的等待EOC轉(zhuǎn)換完成標(biāo)志循環(huán)內(nèi),說明ADC1并沒有被觸發(fā),求高人指點(diǎn)這是哪里出了問題,萬分感謝!
程序如下:
#include "main.h"
uint32_t DC_ADC = 0;

void MyGPIO_Init(void)
{
        RCC_AHBPeriphClockCmd(RCC_AHBPeriph_GPIOA,ENABLE);  //開啟GPIO時(shí)鐘
        
        GPIO_InitTypeDef GPIO_InitStruct1;
        GPIO_InitTypeDef GPIO_InitStruct2;
        
        GPIO_InitStruct1.GPIO_Mode = GPIO_Mode_AN;                        //設(shè)置PA4為模擬輸入
        GPIO_InitStruct1.GPIO_Pin = GPIO_Pin_4;
        GPIO_InitStruct1.GPIO_Speed = GPIO_Speed_50MHz;
        GPIO_Init(GPIOA,&GPIO_InitStruct1);
        
        GPIO_InitStruct2.GPIO_Mode = GPIO_Mode_AF;                        //設(shè)置PA11為復(fù)用功能
        GPIO_InitStruct2.GPIO_Pin = GPIO_Pin_11;
        GPIO_InitStruct2.GPIO_Speed = GPIO_Speed_50MHz;
        GPIO_InitStruct2.GPIO_OType = GPIO_OType_PP;                       //設(shè)置輸出為推挽輸出
        GPIO_InitStruct2.GPIO_PuPd = GPIO_PuPd_UP;                       //上拉
        GPIO_Init(GPIOA,&GPIO_InitStruct2);
        GPIO_PinAFConfig(GPIOA,GPIO_PinSource11,GPIO_AF_2);    //PC6設(shè)置為復(fù)用功能AF2(TIM1)
}

void MyADC1_Init(void)
{
        /*==================1.開啟ADC1時(shí)鐘===============*/
        RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1,ENABLE);


        
        /*==================2.復(fù)位ADC1===============*/
        ADC_DeInit(ADC1)


        
        /*==================3.ADC1初始化===============*/
        ADC_InitTypeDef ADC_InitStruct;
        ADC_InitStruct.ADC_Resolution = ADC_Resolution_12b ;                                           //分辨率12位
        ADC_InitStruct.ADC_ExternalTrigConv = ADC_ExternalTrigConv_T1_CC4;                    //外部觸發(fā)源為TIM1_CC4
        ADC_InitStruct.ADC_ContinuousConvMode = DISABLE;                                             //不開啟連續(xù)轉(zhuǎn)換模式
        ADC_InitStruct.ADC_ScanDirection = ADC_ScanDirection_Upward;                             //向前掃描
        ADC_InitStruct.ADC_ExternalTrigConvEdge = ADC_ExternalTrigConvEdge_Rising;        //外部觸發(fā)為上升沿
        ADC_InitStruct.ADC_DataAlign = ADC_DataAlign_Right;                                            //數(shù)據(jù)對(duì)齊模式:右對(duì)齊
        ADC_Init(ADC1,&ADC_InitStruct);
        
        ADC_ChannelConfig(ADC1,ADC_Channel_4,ADC_SampleTime_239_5Cycles);               //設(shè)置通道4和采樣周期


        
        /*==================4.校準(zhǔn)并使能ADC1===============*/
        ADC_GetCalibrationFactor(ADC1);                                                  //ADC打開校準(zhǔn)并等待完成
        ADC_Cmd(ADC1, ENABLE);                                                               //開啟ADC
        while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_ADRDY));                           //等待ADC準(zhǔn)備完成
}

void MyTIM1_Init(uint16_t arr,uint16_t psc)
{
        /*==============1.打開TIM1時(shí)鐘=============*/
        RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1,ENABLE);


        
        /*==============2.初始化TIM1=============*/
        TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStruct;
        TIM_TimeBaseInitStruct.TIM_ClockDivision = TIM_CKD_DIV1;
        TIM_TimeBaseInitStruct.TIM_CounterMode = TIM_CounterMode_Up;    //向上計(jì)數(shù)
        TIM_TimeBaseInitStruct.TIM_Period = arr - 1;
        TIM_TimeBaseInitStruct.TIM_Prescaler = psc -1 ;
        TIM_TimeBaseInit(TIM1,&TIM_TimeBaseInitStruct);


        
        /*==============3.配置TIM1 PWM1模式=============*/
        TIM_OCInitTypeDef TIM_OCInitStruct;
        TIM_OCInitStruct.TIM_OCMode = TIM_OCMode_PWM1;                        //設(shè)置輸出模式:PWM1模式
        TIM_OCInitStruct.TIM_OutputState = TIM_OutputState_Enable;                //設(shè)置比較輸出使能:
        TIM_OCInitStruct.TIM_OCPolarity = TIM_OCPolarity_High;                        //設(shè)置ref為有效電平時(shí) 輸出高電平
        TIM_OCInitStruct.TIM_Pulse = arr/2;                                                        //輸出占空比:50%的PWM波
        TIM_OC4Init(TIM1,&TIM_OCInitStruct);                                                //通道4輸出


        
        TIM_CtrlPWMOutputs(TIM1,ENABLE);                   //PWM輸出使能
        
        TIM_Cmd(TIM1,ENABLE);                              //使能定時(shí)器 輸出上升沿時(shí)觸發(fā)ADC采樣



int main()
{
        MyGPIO_Init();
        MyADC1_Init();
        MyTIM1_Init(10000,8);
        while(1)
        {
                while(ADC_GetFlagStatus(ADC1,ADC_FLAG_EOC)==RESET) ;     //等待轉(zhuǎn)換完成標(biāo)志
                DC_ADC = ADC_GetConversionValue(ADC1)*3300/0xfff ;            //由ADC采樣轉(zhuǎn)換值計(jì)算出真實(shí)電壓
        }
        
}

分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏2 分享淘帖 頂 踩
回復(fù)

使用道具 舉報(bào)

沙發(fā)
ID:891041 發(fā)表于 2023-10-20 22:07 | 只看該作者
EXTTRIG 寄存器沒配置這一位
回復(fù)

使用道具 舉報(bào)

板凳
ID:526543 發(fā)表于 2023-10-22 16:51 | 只看該作者
ADC配置問題 ADC校準(zhǔn)問題 TIM1配置問題 GPIO配置問題 時(shí)鐘配置 錯(cuò)誤檢查這些好好查查
回復(fù)

使用道具 舉報(bào)

地板
ID:1095470 發(fā)表于 2023-10-23 11:23 | 只看該作者
abob 發(fā)表于 2023-10-20 22:07
EXTTRIG 寄存器沒配置這一位

謝謝你的解答!我查了一下你說的EXTTRIG,F(xiàn)103的板子開啟外部觸發(fā)確實(shí)是需要將CR2寄存器的EXTTRIG置1。但是我使用的是F070的板子,找遍了ADC的寄存器也沒有發(fā)現(xiàn)外部觸發(fā)的開關(guān)控制,CFGR寄存器有個(gè)EXTEN可以選擇四種設(shè)置:關(guān)閉硬件觸發(fā)檢測(cè)(可由軟件啟動(dòng)轉(zhuǎn)換),以及在上升沿、下降沿和上升下降沿進(jìn)行硬件觸發(fā)檢測(cè)。類似于F103和F070的這種區(qū)別還有ADC通道的掃描,F(xiàn)103的ADC是可以選擇通道掃描開啟和關(guān)閉的,而F070的只能選擇掃描方向,默認(rèn)就是開啟的。
回復(fù)

使用道具 舉報(bào)

5#
ID:1095470 發(fā)表于 2023-10-23 11:43 | 只看該作者
123456ZXC1 發(fā)表于 2023-10-22 16:51
ADC配置問題 ADC校準(zhǔn)問題 TIM1配置問題 GPIO配置問題 時(shí)鐘配置 錯(cuò)誤檢查這些好好查查

都查過了,TIM可以輸出預(yù)期的波形,TIM配置應(yīng)該沒有問題,通過逐步調(diào)試感覺應(yīng)該是ADC觸發(fā)的問題,大佬幫忙看看我的觸發(fā)配置對(duì)不對(duì)
回復(fù)

使用道具 舉報(bào)

6#
ID:1095470 發(fā)表于 2023-10-25 09:20 | 只看該作者
已解決。
個(gè)人理解:ADC的外部觸發(fā)(包括TIM的Update和CCx事件)也是需要操作ADC_CR寄存器的開始轉(zhuǎn)換命令A(yù)DSTART位來實(shí)現(xiàn)的,是通過軟件層面進(jìn)行開始轉(zhuǎn)換的控制,而不是在硬件層面直接通過內(nèi)部邏輯電路觸發(fā)。
解決方法:
在main函數(shù)的 while(ADC_GetFlagStatus(ADC1,ADC_FLAG_EOC)==RESET) ;     //等待轉(zhuǎn)換完成標(biāo)志
這一行前面加上ADC_StartOfConversion(ADC1);     //開啟ADC_CR寄存器的ADSTART位,即開始轉(zhuǎn)換命令
回復(fù)

使用道具 舉報(bào)

本版積分規(guī)則

手機(jī)版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術(shù)交流QQ群281945664

Powered by 單片機(jī)教程網(wǎng)

快速回復(fù) 返回頂部 返回列表