- ✨采用寄存器操作方式编程。
✨本示例基于
STM8S903K3
,使用通道6(PD6),作为ADC采集引脚。使用10K可调定位器作为输入端测试。这里直接读取ADC数据寄存器数据进行输出。
- 🎉 如果需要添加数据平移算法,可以直接采用多次数据累加然后取平均值即可。
AIN0->PB6,
AIN1->PB5,
AIN2->PB4,
AIN3->PD3,
AIN4->PD5,
AIN5->PD6
🌿ADC 配置寄存器 1 (ADC_CR1)
🌿ADC 配置寄存器 2 (ADC_CR2)
🌿ADC控制/状态寄存器(ADC_CSR)
🌿ADC 施密特触发器禁止寄存器低位 (ADC_TDRL)
🌿ADC 数据低位寄存器(ADC_DRL)
/**************************************************************************
* 函数名:ADC_conf
* 描述 :ADC模块初始化
* ADC通道 :PD2(AIN3),PD3(AIN4),
* 输出 :无
* 返回 :无
* 调用 :外部调用
*************************************************************************/
void ADC_conf(unsigned char ch )
{
ADC_CR1 = (0<<4)|(1<<1)|(0<<0); //ADC时钟输入频率为16MHz 这里设置分频系数为2 连续转换模式 先禁止ADC转换
// ADC_CR2 = (1<<3)|(0<<1); //设置数据右对齐 禁止扫描模式
ADC_CR2 = 0x08; // 同上,右对齐
ADC_CSR =(0<<5)|(ch + 1); //不用外部触发 禁止转换结束中断 设置转换通道为AIN4(PD3)
ADC_TDRL = ( 1 << (ch + 1 )); //禁止相应通道 施密特触发功能 1左移ch+1位
// ADC_TDRH = 4; //禁止AIN4施密特触发器功能
ADC_CR1 |= 1; //第一次写1是从低功耗模式唤醒
ADC_CR1 |= 1; //在这一位是1的情况下再次写1启动ADC转换
}
/**************************************************************************
* 函数名:ADC_GetConversionValue
* 描述 :获取ADC转换结果
* 输入 :无
* 输出 :无
* 返回 :无
* 调用 :内部调用
*************************************************************************/
uint16_t ADC_GetConversionValue(void)
{
uint16_t value=0;
uint8_t templ,temph ; // 定义templ存储低8位数据 temph存储高8位数据
while(!(ADC_CSR & 0x80)); //等待转换完成
templ = ADC_DRL;
temph = ADC_DRH; //读取ADC转换 在左对齐和右对齐模式下 读取数据的顺序不同
value = (uint16_t)(templ + (uint16_t)(temph << (uint8_t)8)) ; //得到十位精度的数据 0--1024
//注意是10位的转换精度 value、temph应为unsigned int 变量
return (uint16_t)value;
}
**************************************************************************
* 函数名:ADC_GetConversionValue
* 描述 :获取ADC转换结果
* 输入 :无
* 输出 :无
* 返回 :无
* 调用 :内部调用
*************************************************************************/
uint16_t ADC_GetConversionValue(void)
{
uint16_t value=0;
uint8_t templ,temph ; // 定义templ存储低8位数据 temph存储高8位数据
while(!(ADC_CSR & 0x80)); //等待转换完成
if ((ADC_CR2 & 0x08) != 0) /* Right alignment */
{
templ = ADC_DRL;
temph = ADC_DRH; //读取ADC转换 在左对齐和右对齐模式下 读取数据的顺序不同
value = (uint16_t)(templ + (uint16_t)(temph << (uint8_t)8)) ; //得到十位精度的数据 0--1024
}
else
{//注意是10位的转换精度 value、temph应为unsigned int 变量
temph = ADC_DRH;
templ = ADC_DRL;
value = ((uint16_t)temph << 2 ) + (uint16_t)templ ; //得到十位精度的数据 0--1024
}
return (uint16_t)value;
}
🔰由于stm8s903k3,AIN6通道在PD6引脚上,需要将串口功能映射到其他地方。否则,串口无法打印。
Save
保存格式后缀名位.obc
格式。.obc
格式文件。
/****************************************
* 文件名 :main.c
* 描述 :AD转换实验
* 程序现象:本程序通过初始化ADC的AIN6通道,将传感器的ADC值通过串口进行打印
* 寄存器版本
*****************************************/
/* Includes ------------------------------------------------------------------*/
#include "clk_conf.h"
#include "uart.h"
#include "adc.h"
/* Private defines -----------------------------------------------------------*/
/* Private function prototypes -----------------------------------------------*/
/* Private functions ---------------------------------------------------------*/
void delay(u32 nCount)
{
/* Decrement nCount value */
while (nCount != 0)
{
nCount--;
}
}
int main(void)
{
/* Infinite loop */
/*设置内部高速时钟16M为主时钟*/
Clk_conf();
/* 串口初始化 */
uart_conf();
/* ADC模块初始化AIN0->PB6, AIN1->PB5,AIN2->PB4,AIN3->PD3,AIN4->PD5,AIN5->PD6*/
ADC_conf(5);
printf("\r\n:%s\r\n","Using STM8s903k3");
while(1)
{
printf("ADC_value=%u\r\n",ADC_GetConversionValue());
delay(0xffff);
delay(0xffff);
delay(0xffff);
}
}
#ifdef USE_FULL_ASSERT
/**
* @brief Reports the name of the source file and the source line number
* where the assert_param error has occurred.
* @param file: pointer to the source file name
* @param line: assert_param error line source number
* @retval : None
*/
void assert_failed(u8* file, u32 line)
{
/* User can add his own implementation to report the file name and line number,
ex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */
/* Infinite loop */
while (1)
{
}
}
#endif
链接:https://pan.baidu.com/s/1vdxOTbBcCYHYgUGTV41-lg
提取码:nzap