【转】STM32之ADC-2
续 http://club.gizwits.com/thread-3645-1-1.html
3、初始化ADC1的参数、设置ADC1的工作模式和规则序列的相关信息; 大家通过打开"stm32f10.adc.h"可以看到:
[*]<font color="rgb(64, 50, 38)"><font face="verdana,"><i>typedef struct
[*]{
[*]uint32_t ADC_Mode; //设置ADC模式-->独立模式
[*]FunctionalState ADC_ScanConvMode; //设置是否开启扫描模式 --->否
[*]FunctionalState ADC_ContinuousConvMode; //设置是否开启连续转换模式 ---->否
[*]uint32_t ADC_ExternalTrigConv; //设置启动规则转换组转换模式---->软件触发
[*]uint32_t ADC_DataAlign; //设置数据对齐方式----->右对齐
[*]uint8_t ADC_NbrOfChannel; //设置规则序列的长度---->顺序进行规则转换的ADC通道数目1
[*]}ADC_InitTypeDef;</i></font></font>
复制代码
4、使能ADC并校准注:在设置完了以上信息后,使能AD转换器,执行复位校准和AD校准(这两步校准一定要,否则数据将不准)还有记住,每次进行校准之后都要等待校准结束,但是通过什么方式知道校准结束呢?这里是通过获取校准状态来判断是否校准结束,相关的库函数请看代码 分别的库函数请看待会的代码。(请用比较老外的方式去看,也就是用英语啦,为什么呢?请看下文) 5、读取AD的值 当然,这里说读取AD值并不是那么的简单,以上我们只是准备好了AD,还没有设置相关的规则序列通道,采样顺序,以及采样周期,设置完之后启动AD转换就行了、然后才直接读取哈、、
相关的库函数请看代码、
[*]<font color="rgb(64, 50, 38)"><font face="verdana,">void Adc_Init(void)
[*]{
[*]
[*] ADC_InitTypeDef ADC_InitStructure;
[*] GPIO_InitTypeDef GPIO_InitStructure;
[*]
[*] /* Enable ADC1 and GPIOA clock */
[*] RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 | RCC_APB2Periph_GPIOA, ENABLE);
[*]
[*] RCC_ADCCLKConfig(RCC_PCLK2_Div6);//12MHZ
[*]
[*]/* Configure PA.1 (ADC Channel) as analog input -------------------------*/
[*]GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1;
[*]GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN;
[*]GPIO_Init(GPIOA, &GPIO_InitStructure);
[*]
[*] //ADC_DeInit(ADC1);//在这里复位被我注释掉了、至于为什么,我待会会说
[*]
[*] /* ADC1 configuration ------------------------------------------------------*/
[*]ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;//这里对应上面所讲的配置,在这里就不给出注释了
[*]ADC_InitStructure.ADC_ScanConvMode = DISABLE;
[*]ADC_InitStructure.ADC_ContinuousConvMode = DISABLE;
[*]ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;
[*]ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;
[*]ADC_InitStructure.ADC_NbrOfChannel = 1;
[*]ADC_Init(ADC1, &ADC_InitStructure);
[*]
[*] /* Enable ADC1 *///知道我为啥要在上面提醒大家要用老外的方式来看了吧、因为这里的注释都是用英文的
[*]//请不要以为我装逼,我这样做是有原因的、、原因我待会会说、你也会明白我最初的标题为何那样写
[*]
[*]ADC_Cmd(ADC1, ENABLE);
[*]
[*]/* Enable ADC1 reset calibration register */
[*]ADC_ResetCalibration(ADC1);
[*]/* Check the end of ADC1 reset calibration register */
[*]while(ADC_GetResetCalibrationStatus(ADC1));
[*]
[*]/* Start ADC1 calibration */
[*]ADC_StartCalibration(ADC1);
[*]/* Check the end of ADC1 calibration */
[*]while(ADC_GetCalibrationStatus(ADC1));
[*]}</font></font>
复制代码
[*]u16 Get_val(u8 ch)
[*]{
[*] u16 DataValue; //又是英文注释、、啊哈
[*]/* ADC1 regular channel14 configuration */
[*]ADC_RegularChannelConfig(ADC1, ch, 1, ADC_SampleTime_239Cycles5);
[*]
[*] /* Start ADC1 Software Conversion */
[*]ADC_SoftwareStartConvCmd(ADC1, ENABLE);
[*]
[*] /* Test if the ADC1 EOC flag is set or not */
[*]
[*] while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC));
[*]
[*] //FlagStatus Status;
[*] //Status = ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC);
[*] //while(!Status);---->这样做实现不了、请注意
[*]/*Returns the ADC1 Master data value of the last converted channel*/
[*]DataValue = ADC_GetConversionValue(ADC1);
[*]return DataValue;
[*]}
[*]
[*]u16 ADC_Get_aveg(u8 ch,u8 n)
[*]{
[*]u32 ad_sum = 0;
[*]u8 i;
[*]for(i=0;i<n;i++)
[*]{
[*]ad_sum += Get_val(ch);
[*]delay_ms(5);
[*]}
[*]return (ad_sum / n);
[*]}
[*]
[*]adcx=ADC_Get_aveg(ADC_Channel_1,10);//获取AD数值(0~4095)
[*]
[*]temp=(float)adcx*(3.3/4096);//获取相应的电压值
复制代码
到了这一步,我们已经完成了AD采集数据的任务、接下来,有人可能有时候会觉得很纳闷,为什么有些人知道要完成特定的功能,它的步骤是怎么样的、为什么我就不知道??这个问题嘛、、接下来我讲的希望能稍微帮你,也希望你能好好的借鉴、 步骤小技巧:其实也没啥的、大家知道下载库的文件的时候,里面都有包含每个模块的例子和一个模版、拿ADC这个模块来举例: http://bbs.elecfans.com/data/attachment/forum/201502/12/145016vtcqis7hyhbqht7m.png.thumb.jpg
点击main.c可以看到神奇的一幕:http://bbs.elecfans.com/data/attachment/forum/201502/12/145016e3ocvvevyb9ygevq.png.thumb.jpg
大家仔细看看,可以发现在官方给的历程中的步骤里并没有复位ADC的函数,个人觉得所以没有必要去复位当然复位也不是什么坏事哈、看你个人、、看到这、应该明白了我前面的说法了吧、还有、大家应该也注意到了、都是英文的注释、、所以看到这大家也清楚了,前面不是我装逼、、所以呢、其实英语对于我们来说还是很重要的、、那有人问,时钟的分频因子呢?怎么没有设置??不急哈、、请看:对于分频因子的设置,也在这个函数里:而这个RCC_Configuration()在最开始已经使用 了、、
所以大家要好好利用官方给的历程、说到这、你猜我词穷了吗?
楼主在STM32底板上试过吗?做好实例教程就好啊:lol
页:
[1]