1. 前言
- ADC 模块包含 2 个 12 位的逐次逼近型的模拟数字转换器,最高 14MHz 的输入时钟。支持 16 个
- 外部通道和 2 个内部信号源采样源。可完成通道的单次转换、连续转换,通道间自动扫描模式、间
- 断模式、外部触发模式、双重采样等功能。可以通过模拟看门狗功能监测通道电压是否在阈值范围
- 内,本次实验采用两路各一通道的ADC间隔采样,然后打印输出采样值,熟悉STM32开发用易上手配置。✨✨✨
这是使用MounRiver Studio开发的项目,支持在RISC-V核心基础硬件CH32V307评估板上使用带有msh Shell的RTOS快速原型。
MCU:CH32V307VCT6,主频 144MHz,FLASH和RAM可配置
l 12 位分辨率
l 支持 16 个外部通道和 2 个内部信号源采样
l 多通道的多种采样转换方式:单次、连续、扫描、触发、间断等
l 数据对齐模式:左对齐、右对齐
l 采样时间可按通道分别编程
l 规则转换和注入转换均支持外部触发
l 模拟看门狗监测通道电压,自校准功能
l 双重模式
l ADC 通道输入范围:0≤VIN≤VDDA
l 输入增益可调,可实现小信号放大采样
首先,应安装 CH32V307 评估板的驱动程序,打开设备管理器查看USB 端口和外部接口已准备就绪。
2. 软件配置
2.1 安装MounRiver Studio
环境搭建教程:https://blog.csdn.net/VOR234/article/details/128932474
3. ADC项目测试
3.1 打开ADC工程
评估板说明及参考例程:https://www.wch.cn/downloads/CH32V307EVT_ZIP.html
进入EXAM目录,就有对应的外设教程
进入DualADC_InjectionSimul文件下,双击DualADC_InjectionSimul.wvproj,
打开项目工程如下,main.c在user文件夹下
main.c
/********************************** (C) COPYRIGHT ******************************* * File Name : main.c * Author : WCH * Version : V1.0.0 * Date : 2021/06/06 * Description : Main program body. ********************************************************************************* * Copyright (c) 2021 Nanjing Qinheng Microelectronics Co., Ltd. * Attention: This software (modified or not) and binary are used for * microcontroller manufactured by Nanjing Qinheng Microelectronics. *******************************************************************************/ /* *@Note Dual ADC injection simultaneous sampling routine: ADC1 channel 1 (PA1), ADC2 channel 2 (PA3). */ #include "debug.h" /* Global Variable */ u16 ADC_val1,ADC_val2; s16 Calibrattion_Val1 = 0; s16 Calibrattion_Val2 = 0; /********************************************************************* * @fn ADC_Function_Init * * @brief Initializes ADC collection. * * @return none */ void ADC_Function_Init(void) { ADC_InitTypeDef ADC_InitStructure={0}; GPIO_InitTypeDef GPIO_InitStructure={0}; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA , ENABLE ); RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC1 , ENABLE ); RCC_APB2PeriphClockCmd(RCC_APB2Periph_ADC2 , ENABLE ); RCC_ADCCLKConfig(RCC_PCLK2_Div8); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1 |GPIO_Pin_3; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AIN; GPIO_Init(GPIOA, &GPIO_InitStructure); ADC_DeInit(ADC1); ADC_DeInit(ADC2); ADC_InitStructure.ADC_Mode = ADC_Mode_InjecSimult; ADC_InitStructure.ADC_ScanConvMode = DISABLE; ADC_InitStructure.ADC_ContinuousConvMode = DISABLE; ADC_InitStructure.ADC_ExternalTrigConv = ADC_ExternalTrigInjecConv_None; ADC_InitStructure.ADC_DataAlign = ADC_DataAlign_Right; ADC_InitStructure.ADC_NbrOfChannel = 1; ADC_InitStructure.ADC_OutputBuffer = ADC_OutputBuffer_Disable; ADC_InitStructure.ADC_Pga = ADC_Pga_1; ADC_Init(ADC1, &ADC_InitStructure); ADC_InjectedSequencerLengthConfig(ADC1, 1); ADC_InjectedChannelConfig(ADC1, ADC_Channel_1, 1, ADC_SampleTime_239Cycles5 ); ADC_DMACmd(ADC1, ENABLE); ADC_Cmd(ADC1, ENABLE); ADC_BufferCmd(ADC1, DISABLE); //disable buffer ADC_ResetCalibration(ADC1); while(ADC_GetResetCalibrationStatus(ADC1)); ADC_StartCalibration(ADC1); while(ADC_GetCalibrationStatus(ADC1)); Calibrattion_Val1 = Get_CalibrationValue(ADC1); ADC_Init(ADC2, &ADC_InitStructure); ADC_InjectedSequencerLengthConfig(ADC2, 1); ADC_InjectedChannelConfig(ADC2, ADC_Channel_3, 1, ADC_SampleTime_239Cycles5 ); ADC_SoftwareStartInjectedConvCmd(ADC2, ENABLE); ADC_Cmd(ADC2, ENABLE); ADC_BufferCmd(ADC2, DISABLE); //disable buffer ADC_ResetCalibration(ADC2); while(ADC_GetResetCalibrationStatus(ADC2)); ADC_StartCalibration(ADC2); while(ADC_GetCalibrationStatus(ADC2)); Calibrattion_Val2 = Get_CalibrationValue(ADC2); } /********************************************************************* * @fn Get_ConversionVal1 * * @brief Get Conversion Value. * * @param val - Sampling value * * @return val+Calibrattion_Val - Conversion Value. */ u16 Get_ConversionVal1(s16 val) { if((val+Calibrattion_Val1)<0) return 0; if((Calibrattion_Val2+val)>4095||val==4095) return 4095; return (val+Calibrattion_Val1); } /********************************************************************* * @fn Get_ConversionVal2 * * @brief Get Conversion Value. * * @param val - Sampling value * * @return val+Calibrattion_Val - Conversion Value. */ u16 Get_ConversionVal2(s16 val) { if((val+Calibrattion_Val2)<0) return 0; if((Calibrattion_Val2+val)>4095||val==4095) return 4095; return (val+Calibrattion_Val2); } /********************************************************************* * @fn main * * @brief Main program. * * @return none */ int main(void) { USART_Printf_Init(115200); SystemCoreClockUpdate(); Delay_Init(); printf("SystemClk:%d\r\n",SystemCoreClock); printf( "ChipID:%08x\r\n", DBGMCU_GetCHIPID() ); ADC_Function_Init(); printf("CalibrattionValue1:%d\n", Calibrattion_Val1); printf("CalibrattionValue2:%d\n", Calibrattion_Val2); while(1) { ADC_SoftwareStartInjectedConvCmd(ADC1, ENABLE); Delay_Ms(100); ADC_SoftwareStartInjectedConvCmd(ADC1, DISABLE); Delay_Ms(100); while(!ADC_GetFlagStatus(ADC1, ADC_FLAG_JEOC )); ADC_val1 = ADC_GetInjectedConversionValue(ADC1, ADC_InjectedChannel_1); ADC_val2 = ADC_GetInjectedConversionValue(ADC2, ADC_InjectedChannel_1); printf( "JADC1 ch2=%04d\r\n", Get_ConversionVal1(ADC_val1)); printf( "JADC2 ch3=%04d\r\n", Get_ConversionVal2(ADC_val2)); } }
3.2 编译项目
开发板数据线连接电脑就可以开始连接调试🛹🛹🛹,首先开始编译,编译成功如下
然后下载,下载成功如下
4. 下载验证
4.1 接线
根据程序设计调试,可以需要用杜邦线分别采用GND和VCC与PA1和PA3连接起来,即可输出12位
4.2 演示效果
代码下载后验证,点击串口调试器,设置串口参数确认。
复位运行成功如下打印
SystemClk:96000000 ChipID:30700518 CalibrattionValue1:9 CalibrattionValue2:-2 JADC1 ch2=2125 JADC2 ch3=4095 JADC1 ch2=2071 JADC2 ch3=4095 JADC1 ch2=2104 JADC2 ch3=4095 JADC1 ch2=2172 JADC2 ch3=4095 JADC1 ch2=2186 JADC2 ch3=4095 JADC1 ch2=2134 JADC2 ch3=4095 JADC1 ch2=2045
5. 小结
🥳🥳🥳通过对这篇文章我们掌握了沁恒WCH CH32V307V-R1开发板两路ADC读取实验,接下来会有许多有趣的实验,尝试与Arduino通讯做更加好玩的实验,进而丰富我们的生活。🛹🛹🛹从而实现对外部世界进行感知,充分认识这个有机与无机的环境,🥳🥳🥳科学地合理地进行创作和发挥效益,然后为人类社会发展贡献一点微薄之力。🤣🤣🤣