清风迎明月 发表于 2024-4-24 15:38:26

关于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);
}


乘风飞扬 发表于 2024-4-24 16:58:46

使用 printf 打印,需要重写 putchar 函数,在 UartInit 初始化函数后面添加以下两个函数:
void UartPutc(unsigned char dat)
{
    SBUF = dat;
    while(TI==0);
    TI = 0;
}

char putchar(char c)
{
    UartPutc(c);
    return c;
}

xxxevery 发表于 2024-4-24 17:29:30

函数ADC_convert(u8 chn)中的 j 你赋值了吗,怎么可能有结果

清风迎明月 发表于 2024-4-24 21:52:49

乘风飞扬 发表于 2024-4-24 16:58
使用 printf 打印,需要重写 putchar 函数,在 UartInit 初始化函数后面添加以下两个函数:
...

谢谢,我试试

清风迎明月 发表于 2024-4-24 22:11:25

xxxevery 发表于 2024-4-24 17:29
函数ADC_convert(u8 chn)中的 j 你赋值了吗,怎么可能有结果

谢谢,确实j无赋值

清风迎明月 发表于 2024-4-24 22:13:16

本帖最后由 清风迎明月 于 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 14:54:35

用你的代码进行验证,是可以正常输出采集数据的:

如果你那里还是不行的话,建议排查硬件连接以及串口工具是否正常。
附件是我验证的项目文件,请参考。

清风迎明月 发表于 2024-4-25 20:53:00

本帖最后由 清风迎明月 于 2024-4-25 21:03 编辑

怪了,我收不到数据,不过我检查程序好像没有问题的,除了多个k,我换板子试试

清风迎明月 发表于 2024-4-25 21:03:40

乘风飞扬 发表于 2024-4-25 14:54
用你的代码进行验证,是可以正常输出采集数据的:

如果你那里还是不行的话,建议排查硬件连接以及串口工具 ...

万分感谢哈{:4_196:}
页: [1]
查看完整版本: 关于16路ADC转换程序问题