关于EXTSEL[2:0]的说明:
位19:17 EXTSEL[2:0]:选择启动规则通道组转换的外部事件
这些位选择用于启动规则通道组转换的外部事件
ADC1和ADC2的触发配置如下
000:定时器1的CC1事件 100:定时器3的TRGO事件
001:定时器1的CC2事件 101:定时器4的CC4事件
010:定时器1的CC3事件 110:EXTI线11/ TIM8_TRGO,
仅大容量产品具有TIM8_TRGO功能
011:定时器2的CC2事件 111:SWSTART
ADC3的触发配置如下
000:定时器3的CC1事件 100:定时器8的TRGO事件
001:定时器2的CC3事件 101:定时器5的CC1事件
010:定时器1的CC3事件 110:定时器5的CC3事件
011:定时器8的CC1事件 111:SWSTART
*/
ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
/*
这个是用来设定数据对齐模式的,有两种可能:
#define ADC_DataAlign_Right ((uint32_t)0x00000000)
#define ADC_DataAlign_Left ((uint32_t)0x00000800)
找到数据手册上的相关说明:
位11:ALIGN:数据对齐
该位由软件设置和清除。
0:右对齐 1:左对齐
*/
ADC_InitStructure.ADC_NbrOfChannel = 1;
/* ADC_NbrOfChannel的定义如下:
uint8_t ADC_NbrOfChannel;
指定有多少个通道会被转换,它的值可以是1~16,这个数据将会影响到寄存器ADC_SQR1,下面是stm32f10x_adc.c中的相关代码:
。。.。。.
tmpreg2 |= (uint8_t) (ADC_InitStruct-》ADC_NbrOfChannel - (uint8_t)1);
tmpreg1 |= (uint32_t)tmpreg2 《《 20;
ADCx-》SQR1 = tmpreg1;
看到mpreg1 |= (uint32_t)tmpreg2 《《 20;中的:20,用上面我们刚理解到的原则,这个值的低位将在ADC_SQR1的20位,而它的值是1~16,从代码中可以看到这里又减去1,则其设置值为:0~15,即4bit就够了,那么从20往前数,也就是[23:20],那么SQR1中这几位的用途是什么呢?顺这条线索我们去找SQR1中的23:20位,看它是怎么用的。
位23:20 L[3:0]:规则通道序列长度
这些位定义了在规则通道转
0000:1个转换
0001:2个转换
……
1111:16个转换
也就是设置一次进行几个通道的转换,看来我们的理解完全正确。
*/
ADC_Init(ADC1, &ADC_InitStructure);
//通过前面一系列的设置,可以执行ADC_Init函数了。
/* ADC1 规则通道15(Channel15)配置(规则通道见文章开头)*/
ADC_RegularChannelConfig(ADC1, ADC_Channel_15, 1, ADC_SampleTime_55Cycles5);
/* 这个函数一共有4个参数,第一个是指定转换器,根据所采用的器件的不同,可以是ADC1,ADC2,ADC3;第二个参数是指定通道号;第三个参数是指定该通道在转换序列中第几个开始转换,第四个参数是指定转换时间
第一、二个参数不难理解,这里就不再多说了,看一看第三个参数。
先看一看这个函数的内容,它在stm32f10x_adc.c中,这是STM库提供的一个函数:
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) //这个Rand就是第三个参数
{
/* 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每个都是占5位。
这下理解了:如果这个Rank是1,那么tmpreg2这个变量第[4:0]这5位将会是11111(即SQR3_SQ_Set的初始值:0x0000001f),如果Rank是2,那么tmpreg2这个变量的第[9:5]将会是11111,即tmpreg2将等于:0x00001f00,依此类推。
/* Clear the old SQx bits for the selected rank */
tmpreg1 &= ~tmpreg2;
/* tmpreg2取反再与,即清掉tmpreg1中相应的5位*/
tmpreg2 = (uint32_t)ADC_Channel 《《 (5 * (Rank - 1));
/*这次tmpreg2取的是通道值了,然后同相根据Rank的值左移5、10或更多位 */
tmpreg1 |= tmpreg2;
/* Store the new register value */
ADCx-》SQR3 = tmpreg1;
}
*/
第四个参数是采样时间设定,代码如下:
tmpreg2 = (uint32_t)ADC_SampleTime 《《 (3 * ADC_Channel);
/* 设定新的采样时间,这里为什么用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转换配置完毕。很麻烦。。.。。.也许功能强大的副产品就是麻烦吧,没有办法。
评论
查看更多