关于16路ADC转换程序问题
由于开天斧中示例程序:16路ADC转换取平均值程序,直接烧录尝试,并不能上传结果我对那个程序做了简化如下,编译没问题,但是串口助手也不能获得数据,各位看看问题在哪儿
单片机8H8K64U
程序:
#include "STC8h.h"
#include "stdio.h"
#include "intrins.h"
typedef unsigned char u8;
typedef unsigned int u16;
typedef unsigned long u32;
#define MAIN_Fosc 22118400UL //定义主时钟(精确计算115200波特率)
#define Baudrate 115200UL
#define TM (65536-(MAIN_Fosc/Baudrate/4))
/************* 本地常量声明 **************/
#define ADC_START (1<<6) //启动转换,该位在启动后自动清0
#define ADC_FLAG (1<<5) // 转换结束标志位,需软件清0
#define ADC_SPEED 15 /* 0~15, ADC转换时间,在ADCCFG */
#define RES_FMT(1<<5) /* ADC结果格式,右对齐, ADC_RES: 0 0 0 0 D11 D10 D9 D8 */
#define SUM_LENGTH 16 //平均值采样次数,最大值16
/************* 本地函数声明 *************/
u16 Get_ADC12bitResult(u8 channel); //获取一个通道ADC转换结果函数
void delay_ms(u8 ms); //延时函数
void ADC_convert(u8 chn); //对某通道AD转换函数,chn=0~7对应P1.0~P1.7, chn=8~14对应P0.0~P0.6, chn=15对应BandGap电压
/******************* 串口初始化函数 ********************/
void UartInit(void)
{
SCON = (SCON & 0x3f) | 0x40; //串口1的工作方式1,8+2模式
AUXR |= 0x40; //定时器时钟1T模式
AUXR &= 0xFE; //串口1选择定时器1为波特率发生器
TL1 = TM;
TH1 = TM>>8;
TR1 = 1; //定时器1开始计时
}
/**********************************************/
void main(void)
{
u8i;
P_SW2 |= 0x80; //扩展寄存器(XFR)访问使能
P0M1 = 0x7f; P0M0 = 0x00; //设置为高阻输入
P1M1 = 0xff; P1M0 = 0x00; //设置为高阻输入
ADC_CONTR = 0x80 + 0; //ADC on + channel,打开AD电源,并选择通道0
ADCCFG = RES_FMT + ADC_SPEED; //AD结果右对齐,并设定转换时间
ADCTIM = 0x3f; //设置通道选择时间、保持时间、采样时间
ADCEXCFG = 0x07; //转换16次并取平均值 //取16次平均值
UartInit();
while (1)
{
for(i=0; i<16; i++)
{
delay_ms(200);
ADC_convert(i); //发送轮询通道AD值
}
}
}
//==================================================================
// 函数: u16 Get_ADC12bitResult(u8 channel)) //channel = 0~15
// 描述: 查询法读一次ADC结果.
// 参数: channel: 选择要转换的ADC, 0~15.
// 返回: 12位ADC结果.
// 版本: V1.0, 2016-4-28
//=====================================================================
u16 Get_ADC12bitResult(u8 channel) //channel = 0~15
{
ADC_RES = 0;
ADC_RESL = 0;
ADC_CONTR = 0x80 | ADC_START | channel;
NOP(10); //等待稳定电压
while((ADC_CONTR & ADC_FLAG) == 0); //等待ADC转换结束
ADC_CONTR &= ~ADC_FLAG; //清楚转换结束标志位
return ((u16)ADC_RES * 256 + (u16)ADC_RESL);
}
void ADC_convert(u8 chn)
{
u16 j;
Get_ADC12bitResult(chn); //参数i=0~15,查询方式做一次ADC, 切换通道后第一次转换结果丢弃. 避免采样电容的残存电压影响.
Get_ADC12bitResult(chn); //参数i=0~15,查询方式做一次ADC, 切换通道后第二次转换结果丢弃. 避免采样电容的残存电压影响.
if(chn == 15) printf("Bandgap=%04d",j); //内基准1.35V or 1.19V
else //ADC0~ADC14
{
printf("ADC%02bd=%04d",chn,j);
}
}
void delay_ms(u8 ms)
{
u16 i;
do
{
i = MAIN_Fosc / 10000;
while(--i);
}while(--ms);
}
使用 printf 打印,需要重写 putchar 函数,在 UartInit 初始化函数后面添加以下两个函数:
void UartPutc(unsigned char dat)
{
SBUF = dat;
while(TI==0);
TI = 0;
}
char putchar(char c)
{
UartPutc(c);
return c;
}
函数ADC_convert(u8 chn)中的 j 你赋值了吗,怎么可能有结果 乘风飞扬 发表于 2024-4-24 16:58
使用 printf 打印,需要重写 putchar 函数,在 UartInit 初始化函数后面添加以下两个函数:
...
谢谢,我试试 xxxevery 发表于 2024-4-24 17:29
函数ADC_convert(u8 chn)中的 j 你赋值了吗,怎么可能有结果
谢谢,确实j无赋值 本帖最后由 清风迎明月 于 2024-4-25 21:04 编辑
根据两位的建议,程序修改成如下了,串口助手也不能获得输出值,不知道还哪儿有问题
#include "STC8h.h"
#include "stdio.h"
#include "intrins.h"
typedef unsigned char u8;
typedef unsigned int u16;
typedef unsigned long u32;
#define MAIN_Fosc 22118400UL //定义主时钟(精确计算115200波特率)
#define Baudrate 115200UL
#define TM (65536-(MAIN_Fosc/Baudrate/4))
/************* 本地常量声明 **************/
#define ADC_START (1<<6) //启动转换,该位在启动后自动清0
#define ADC_FLAG (1<<5) // 转换结束标志位,需软件清0
#define ADC_SPEED 15 /* 0~15, ADC转换时间,在ADCCFG */
#define RES_FMT(1<<5) /* ADC结果格式,右对齐, ADC_RES: 0 0 0 0 D11 D10 D9 D8 */
#define SUM_LENGTH 16 //平均值采样次数,最大值16
/************* 本地函数声明 *************/
u16 Get_ADC12bitResult(u8 channel); //获取一个通道ADC转换结果函数
void delay_ms(u8 ms); //延时函数
void ADC_convert(u8 chn); //对某通道AD转换函数,chn=0~7对应P1.0~P1.7, chn=8~14对应P0.0~P0.6, chn=15对应BandGap电压
/******************* 串口初始化函数 ********************/
void UartInit(void)
{
SCON = (SCON & 0x3f) | 0x40; //串口1的工作方式1,8+2模式
AUXR |= 0x40; //定时器时钟1T模式
AUXR &= 0xFE; //串口1选择定时器1为波特率发生器
TMOD &=0x0F; //T1方式0,16位自动重载
TL1 = TM;
TH1 = TM>>8;
TR1 = 1; //定时器1开始计时
}
void UartPutc(unsigned char dat)
{
SBUF = dat;
while(TI==0);
TI = 0;
}
char putchar(char c)
{
UartPutc(c);
return c;
}
/**********************************************/
void main(void)
{
u8i;
P_SW2 |= 0x80; //扩展寄存器(XFR)访问使能
P0M1 = 0x7f; P0M0 = 0x00; //设置为高阻输入
P1M1 = 0xff; P1M0 = 0x00; //设置为高阻输入
ADC_CONTR = 0x80 + 0; //ADC on + channel,打开AD电源,并选择通道0
ADCCFG = RES_FMT + ADC_SPEED; //AD结果右对齐,并设定转换时间
ADCTIM = 0x3f; //设置通道选择时间、保持时间、采样时间
ADCEXCFG = 0x07; //转换16次并取平均值 //取16次平均值
UartInit();
while (1)
{
for(i=0; i<16; i++)
{
delay_ms(200);
ADC_convert(i); //发送轮询通道AD值
}
}
}
//==================================================================
// 函数: u16 Get_ADC12bitResult(u8 channel)) //channel = 0~15
// 描述: 查询法读一次ADC结果.
// 参数: channel: 选择要转换的ADC, 0~15.
// 返回: 12位ADC结果.
// 版本: V1.0, 2016-4-28
//=====================================================================
u16 Get_ADC12bitResult(u8 channel) //channel = 0~15
{
ADC_RES = 0;
ADC_RESL = 0;
ADC_CONTR = 0x80 | ADC_START | channel;
NOP(10); //等待稳定电压
while((ADC_CONTR & ADC_FLAG) == 0); //等待ADC转换结束
ADC_CONTR &= ~ADC_FLAG; //清楚转换结束标志位
return ((u16)ADC_RES * 256 + (u16)ADC_RESL);
}
void ADC_convert(u8 chn)
{
u16j;
// u8 k;
Get_ADC12bitResult(chn); //参数i=0~15,查询方式做一次ADC, 切换通道后第一次转换结果丢弃. 避免采样电容的残存电压影响.
j=Get_ADC12bitResult(chn); //参数i=0~15,查询方式做一次ADC, 切换通道后第二次转换结果丢弃. 避免采样电容的残存电压影响.
if(chn == 15) printf("Bandgap=%04d",j); //内基准1.35V or 1.19V
else //ADC0~ADC14
{
printf("ADC%02bd=%04d",chn,j);
}
}
void delay_ms(u8 ms)
{
u16 i;
do
{
i = MAIN_Fosc / 10000;
while(--i);
}while(--ms);
}
用你的代码进行验证,是可以正常输出采集数据的:
如果你那里还是不行的话,建议排查硬件连接以及串口工具是否正常。
附件是我验证的项目文件,请参考。 本帖最后由 清风迎明月 于 2024-4-25 21:03 编辑
怪了,我收不到数据,不过我检查程序好像没有问题的,除了多个k,我换板子试试 乘风飞扬 发表于 2024-4-25 14:54
用你的代码进行验证,是可以正常输出采集数据的:
如果你那里还是不行的话,建议排查硬件连接以及串口工具 ...
万分感谢哈{:4_196:}
页:
[1]