|
发表于 2023-9-3 16:05:54
|
显示全部楼层
/************* 参数定义声明 **************/
#define MAIN_Fosc 24000000L //定义主时钟
#define Baudrate 115200L
#define TM (65536 -(MAIN_Fosc/Baudrate/4))
/************* 本地常量声明 **************/
#define ADC_SPEED 15 /* 0~15, ADC转换时间(CPU时钟数) = (n+1)*32 ADCCFG */
#define RES_FMT (1<<5) /* ADC结果格式 0: 左对齐, ADC_RES: D11 D10 D9 D8 D7 D6 D5 D4, ADC_RESL: D3 D2 D1 D0 0 0 0 0 */
/* ADCCFG 1: 右对齐, ADC_RES: 0 0 0 0 D11 D10 D9 D8, ADC_RESL: D7 D6 D5 D4 D3 D2 D1 D0 */
/************* 本地函数声明 **************/
void delay_ms(u8 ms);
void ADC_convert(u8 chn); //chn=0~7对应P1.0~P1.7, chn=8~14对应P0.0~P0.6, chn=15对应BandGap电压
u16 Get_ADC12bitResult(u8 channel);
/******************** 串口打印函数 ********************/
void UartInit(void)
{
P_SW1 = (P_SW1 & 0x3f) | 0x00; //USART1 switch to, 0x00: P3.0 P3.1, 0x40: P3.6 P3.7, 0x80: P1.6 P1.7, 0xC0: P4.3 P4.4
SCON = (SCON & 0x3f) | 0x40;
AUXR |= 0x40; //定时器时钟1T模式
AUXR &= 0xFE; //串口1选择定时器1为波特率发生器
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)
{
P_SW2 |= 0x80; //扩展寄存器(XFR)访问使能
P0M1 = 0x00; P0M0 = 0x00; //设置为准双向口
P1M1 = 0x40; P1M0 = 0x00; //P1.6设置为高阻输入
P2M1 = 0x00; P2M0 = 0x00; //设置为准双向口
P3M1 = 0x00; P3M0 = 0x00; //设置为准双向口
P4M1 = 0x00; P4M0 = 0x00; //设置为准双向口
P5M1 = 0x00; P5M0 = 0x00; //设置为准双向口
P6M1 = 0x00; P6M0 = 0x00; //设置为准双向口
P7M1 = 0x00; P7M0 = 0x00; //设置为准双向口
ADCTIM = 0x3f; //设置通道选择时间、保持时间、采样时间
//ADC模块电源打开后,需等待1ms,MCU内部ADC电源稳定后再进行AD转换
ADC_CONTR = 0x80 + 6; //ADC on + channel
ADCCFG = RES_FMT + ADC_SPEED;
UartInit();
EA = 1;
printf("STC8H系列测试程序!\r\n");
while (1)
{
delay_ms(250);
delay_ms(250);
ADC_convert(6); //发送固定通道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;
if((ADC_CONTR & 0x20) == 0)
ADC_CONTR = (ADC_CONTR & 0xF0) | 0x40 | channel; //启动 AD 转换
_nop_();
_nop_();
_nop_();
while((ADC_CONTR & 0x20) == 0); //等待ADC结束
ADC_CONTR &= ~0x20; //清除ADC结束标志
return (((u16)ADC_RES << 8) | ADC_RESL);
}
/***********************************
查询方式做一次ADC, chn为通道号, chn=0~7对应P1.0~P1.7, chn=8~14对应P0.0~P0.6, chn=15对应BandGap电压.
***********************************/
void ADC_convert(u8 chn)
{
u16 j;
u8 k; //平均值滤波时使用
float vADC;
Get_ADC12bitResult(chn); //参数i=0~15,查询方式做一次ADC, 切换通道后第一次转换结果丢弃. 避免采样电容的残存电压影响.
Get_ADC12bitResult(chn); //参数i=0~15,查询方式做一次ADC, 切换通道后第二次转换结果丢弃. 避免采样电容的残存电压影响.
j = Get_ADC12bitResult(chn);
printf("12bit: ADC%02bd=%04u ",chn,j); //输出ADC值
vADC = ((float)j * 3.3 / 4096.0); //
printf("P13=%fV ",vADC);
//过采样例子
for(k=0, j=0; k<16; k++) j += Get_ADC12bitResult(chn); // 采样累加和 参数0~15,查询方式做一次ADC, 返回值就是结果
vADC = j / 4; // 12位ADC,采样16次数值累加后除以4,结果便为14位过采样ADC数值
printf("14bit: ADC=%f ",vADC);
vADC = (vADC * 3.3 / 16384.0);
printf("P13=%fV\r\n",vADC);
}
//========================================================================
// 函数: void delay_ms(u8 ms)
// 描述: 延时函数。
// 参数: ms,要延时的ms数, 这里只支持1~255ms. 自动适应主时钟.
// 返回: none.
// 版本: VER1.0
// 日期: 2013-4-1
// 备注:
//========================================================================
void delay_ms(u8 ms)
{
u16 i;
do
{
i = MAIN_Fosc / 10000;
while(--i);
}while(--ms);
} |
|