關(guān)于EXTSEL[2:0]的說明:
位19:17 EXTSEL[2:0]:選擇啟動(dòng)規(guī)則通道組轉(zhuǎn)換的外部事件
這些位選擇用于啟動(dòng)規(guī)則通道組轉(zhuǎn)換的外部事件
ADC1和ADC2的觸發(fā)配置如下
000:定時(shí)器1的CC1事件 100:定時(shí)器3的TRGO事件
001:定時(shí)器1的CC2事件 101:定時(shí)器4的CC4事件
010:定時(shí)器1的CC3事件 110:EXTI線11/ TIM8_TRGO,
僅大容量產(chǎn)品具有TIM8_TRGO功能
011:定時(shí)器2的CC2事件 111:SWSTART
ADC3的觸發(fā)配置如下
000:定時(shí)器3的CC1事件 100:定時(shí)器8的TRGO事件
001:定時(shí)器2的CC3事件 101:定時(shí)器5的CC1事件
010:定時(shí)器1的CC3事件 110:定時(shí)器5的CC3事件
011:定時(shí)器8的CC1事件 111:SWSTART
*/
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
/*
這個(gè)是用來設(shè)定數(shù)據(jù)對齊模式的,有兩種可能:
#define ADC_DataAlign_Right ((uint32_t)0x00000000)
#define ADC_DataAlign_Left ((uint32_t)0x00000800)
找到數(shù)據(jù)手冊上的相關(guān)說明:
位11:ALIGN:數(shù)據(jù)對齊
該位由軟件設(shè)置和清除。
0:右對齊 1:左對齊
*/
ADC_InitStructure.ADC_NbrOfChannel = 1;
/* ADC_NbrOfChannel的定義如下:
uint8_t ADC_NbrOfChannel;
指定有多少個(gè)通道會被轉(zhuǎn)換,它的值可以是1~16,這個(gè)數(shù)據(jù)將會影響到寄存器ADC_SQR1,下面是stm32f10x_adc.c中的相關(guān)代碼:
。。.。。.
tmpreg2 |= (uint8_t) (ADC_InitStruct-》ADC_NbrOfChannel - (uint8_t)1);
tmpreg1 |= (uint32_t)tmpreg2 《《 20;
ADCx-》SQR1 = tmpreg1;
看到mpreg1 |= (uint32_t)tmpreg2 《《 20;中的:20,用上面我們剛理解到的原則,這個(gè)值的低位將在ADC_SQR1的20位,而它的值是1~16,從代碼中可以看到這里又減去1,則其設(shè)置值為:0~15,即4bit就夠了,那么從20往前數(shù),也就是[23:20],那么SQR1中這幾位的用途是什么呢?順這條線索我們?nèi)フ襍QR1中的23:20位,看它是怎么用的。
位23:20 L[3:0]:規(guī)則通道序列長度
這些位定義了在規(guī)則通道轉(zhuǎn)
0000:1個(gè)轉(zhuǎn)換
0001:2個(gè)轉(zhuǎn)換
……
1111:16個(gè)轉(zhuǎn)換
也就是設(shè)置一次進(jìn)行幾個(gè)通道的轉(zhuǎn)換,看來我們的理解完全正確。
*/
ADC_Init(ADC1, &ADC_InitStructure);
//通過前面一系列的設(shè)置,可以執(zhí)行ADC_Init函數(shù)了。
/* ADC1 規(guī)則通道15(Channel15)配置(規(guī)則通道見文章開頭)*/
ADC_RegularChannelConfig(ADC1, ADC_Channel_15, 1, ADC_SampleTime_55Cycles5);
/* 這個(gè)函數(shù)一共有4個(gè)參數(shù),第一個(gè)是指定轉(zhuǎn)換器,根據(jù)所采用的器件的不同,可以是ADC1,ADC2,ADC3;第二個(gè)參數(shù)是指定通道號;第三個(gè)參數(shù)是指定該通道在轉(zhuǎn)換序列中第幾個(gè)開始轉(zhuǎn)換,第四個(gè)參數(shù)是指定轉(zhuǎn)換時(shí)間
第一、二個(gè)參數(shù)不難理解,這里就不再多說了,看一看第三個(gè)參數(shù)。
先看一看這個(gè)函數(shù)的內(nèi)容,它在stm32f10x_adc.c中,這是STM庫提供的一個(gè)函數(shù):
void ADC_RegularChannelConfig(ADC_TypeDef* ADCx, uint8_t ADC_Channel, uint8_t Rank, uint8_t ADC_SampleTime)
{ 。。.。。.前面的不寫了
/* For Rank 1 to 6 */
if (Rank 《 7) //這個(gè)Rand就是第三個(gè)參數(shù)
{
/* Get the old register value */
tmpreg1 = ADCx-》SQR3;
/* Calculate the mask to clear */
tmpreg2 = SQR3_SQ_Set 《《 (5 * (Rank - 1));
SQR3的值如下:
//#define SQR3_SQ_Set ((uint32_t)0x0000001F)
之所以用5去乘,看下圖中的表格:ADC_SQ3中SQ1~SQ6每個(gè)都是占5位。
這下理解了:如果這個(gè)Rank是1,那么tmpreg2這個(gè)變量第[4:0]這5位將會是11111(即SQR3_SQ_Set的初始值:0x0000001f),如果Rank是2,那么tmpreg2這個(gè)變量的第[9:5]將會是11111,即tmpreg2將等于:0x00001f00,依此類推。
/* Clear the old SQx bits for the selected rank */
tmpreg1 &= ~tmpreg2;
/* tmpreg2取反再與,即清掉tmpreg1中相應(yīng)的5位*/
tmpreg2 = (uint32_t)ADC_Channel 《《 (5 * (Rank - 1));
/*這次tmpreg2取的是通道值了,然后同相根據(jù)Rank的值左移5、10或更多位 */
tmpreg1 |= tmpreg2;
/* Store the new register value */
ADCx-》SQR3 = tmpreg1;
}
*/
?
第四個(gè)參數(shù)是采樣時(shí)間設(shè)定,代碼如下:
tmpreg2 = (uint32_t)ADC_SampleTime 《《 (3 * ADC_Channel);
/* 設(shè)定新的采樣時(shí)間,這里為什么用3,理由和上面的5一樣,看下圖。*/
tmpreg1 |= tmpreg2;
/* Store the new register value */
ADCx-》SMPR2 = tmpreg1;
/* Enable ADC1 DMA */
ADC_DMACmd(ADC1, ENABLE);
/* Enable ADC1 */
ADC_Cmd(ADC1, ENABLE);
至此一次ADC轉(zhuǎn)換配置完畢。很麻煩。。.。。.也許功能強(qiáng)大的副產(chǎn)品就是麻煩吧,沒有辦法。
評論
查看更多