STM32.ADC

ADC实验

原理图:

 

 

1.ADC配置函数

/* enable adc1 and config adc1 to dma mode */
ADC1_Init();

/**
  * @brief  ADC1初始化
  * @param  无
  * @retval 无
  */
void ADC1_Init(void)
{
	ADC1_GPIO_Config();    //端口初始化
	ADC1_Mode_Config();
}

 对于配置ADC1的工作模式为MDA模式

ADC1是挂载到DMA1的通道1的

#define ADC1_DR_Address    ((u32)0x40012400+0x4c)//模拟量转换成数字量后存放到该地址中

看如下存储器映像 寄存器组起始地址(起始地址)

 

便宜地址:0x4c

----------------------------------------------------------------------------------------------------------------------

那么对于ADC自身配置:

采样总时间:

对于STM32最快的ADC转换时间也就是1US了,打死都不会变的了

-----------------------------------------------------------------------------------------------------------------------

代码:

/**
  * @brief  配置ADC1的工作模式为MDA模式
  * @param  无
  * @retval 无
  */
static void ADC1_Mode_Config(void)
{
    DMA_InitTypeDef DMA_InitStructure;
    ADC_InitTypeDef ADC_InitStructure;
    
    /* DMA channel1 configuration */
    DMA_DeInit(DMA1_Channel1);//重新配置
    
    DMA_InitStructure.DMA_PeripheralBaseAddr = ADC1_DR_Address;         //ADC地址
    DMA_InitStructure.DMA_MemoryBaseAddr = (u32)&ADC_ConvertedValue;    //内存地址
    DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralSRC; //方向从#define DMA_DIR_PeripheralSRC 
                                    //((uint32_t)0x00000000)开始发
    DMA_InitStructure.DMA_BufferSize = 1;            //每次只发送一个数据
    DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;    //外设地址固定
    DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Disable;          //内存地址固定
    DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_HalfWord; //半字
    DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_HalfWord;
    DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;                //循环传输
    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);
    
    /* ADC1 configuration */    
    ADC_InitStructure.ADC_Mode = ADC_Mode_Independent;    //独立ADC模式(因为咱们只用一个)
    ADC_InitStructure.ADC_ScanConvMode = DISABLE ;          //禁止扫描模式,扫描模式用于多通道采集
                              //(ADC有3个通道,若几个通道同时采集一个模拟量则用扫描)
                          //若同时采(交叉采集),速率更高,对于32的示波器就是三路同时采集   
    ADC_InitStructure.ADC_ContinuousConvMode = ENABLE;  //开启连续转换模式,即不停地进行ADC转换
    ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigConv_None;    //不使用外部触发转换
    ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right;     
                        //采集数据右对齐,数据时12位,保存数据的寄存器是16位
    ADC_InitStructure.ADC_NbrOfChannel = 1;             //要转换的通道数目1
    ADC_Init(ADC1, &ADC_InitStructure);
    
    /*配置ADC时钟,为PCLK2的8分频,即9MHz(最大是14M)68/9MS*/
    RCC_ADCCLKConfig(RCC_PCLK2_Div8); 
    /*配置ADC1的通道11为55.5个采样周期(模拟量转换成数字量需要多长时间),
     序列为1(有三个通道它排第一) */ 
    ADC_RegularChannelConfig(ADC1, ADC_Channel_11, 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);
}

 

对于main函数

    while (1)
    {
        ADC_ConvertedValueLocal =(float) ADC_ConvertedValue/4096*3.3; // 读取转换的AD值
    
        printf("\r\n The current AD value = 0x%04X \r\n", ADC_ConvertedValue); 
        printf("\r\n The current AD value = %f V \r\n",ADC_ConvertedValueLocal); 

        Delay(0xffffee);  
    }

为什么是4096

设V为我们转换出来的电压 

V/3.3  = 读取的寄存器值/4096  (寄存器满值2^12 = 4096) 当然此时Vref- = 0V  ;Vref+ = 3.3V;若 Vref+ = (x)V  V/(x)  = 读取的寄存器值/4096  

 

 

 

--------------------------------------------------------------------------------------------------------------------------

 

 

 

 

 

 

 

 

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
STM32中,可以使用内置的ADC模块来生成随机数。具体的步骤如下: 1. 配置ADC模块,选择一个合适的通道,比如ADC1的通道1。 2. 开始ADC转换,可以使用单次转换模式或者连续转换模式。 3. 在转换完成后,从ADC数据寄存器中读取转换结果。 4. 取出转换结果中的一部分作为随机数,比如取最低的几位。 下面是一个简单的示例代码,演示如何使用ADC模块生成随机数: ```c #include "stm32f4xx.h" void adc_init(void) { RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1, ENABLE); ADC_InitTypeDef ADC_InitStruct; ADC_InitStruct.ADC_Resolution = ADC_Resolution_12b; ADC_InitStruct.ADC_ScanConvMode = DISABLE; ADC_InitStruct.ADC_ContinuousConvMode = ENABLE; ADC_InitStruct.ADC_ExternalTrigConvEdge = ADC_ExternalTrigConvEdge_None; ADC_InitStruct.ADC_ExternalTrigConv = ADC_ExternalTrigConv_T1_CC1; ADC_InitStruct.ADC_DataAlign = ADC_DataAlign_Right; ADC_InitStruct.ADC_NbrOfConversion = 1; ADC_Init(ADC1, &ADC_InitStruct); ADC_Cmd(ADC1, ENABLE); ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 1, ADC_SampleTime_56Cycles); } uint32_t get_random_number(void) { ADC_SoftwareStartConv(ADC1); while (ADC_GetFlagStatus(ADC1, ADC_FLAG_EOC) == RESET); uint16_t adc_value = ADC_GetConversionValue(ADC1); return adc_value & 0x7; } int main(void) { adc_init(); uint32_t random_number = get_random_number(); return 0; } ``` 在上面的代码中,首先通过`adc_init()`函数配置了ADC模块,选择了ADC1的通道1,并开启了连续转换模式。然后在`get_random_number()`函数中,启动了ADC转换,并等待转换完成。最后从ADC数据寄存器中读取转换结果,并取出最低的3位作为随机数。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值