
#include <avr/ioh>
#include<avr/interrupth>
#include<avr/signalh>
#define uint unsigned int
#define uchar unsigned char
#define INT8U unsigned char
/18B20/
#define CLR_DIR_1WIRE DDRD&=~(0x80) //只要修改这里的参数就可以了!呵呵!
#define SET_DIR_1WIRE DDRD|=(0x80) //里面什么都不用该!
#define CLR_OP_1WIRE PORTD&=~(0x80)
#define SET_OP_1WIRE PORTD|=(0x80)
#define CHECK_IP_1WIRE (PIND & 0x80) //检测
unsigned char tem;
unsigned char H_tem,L_tem;
unsigned char wmh,wml;
uchar s[10]={0x30,0x31,0x32,0x33,0x34,0x35,0x36,0x37,0x38,0x39};
/延时函数 /
void delayus(unsigned int i)
{
while(i--);
}
void delayms(unsigned int ms) //10us延时函数
{
while(ms--)
{
delayus(1000);
}
}
/ 18B20 函数/
unsigned char wmh,wml;
//-------------延时程序------------------------
void delay_3us(void) //3us延时函数
{
;
}
void delay_10us(void) //10us延时函数
{
unsigned int i;
for (i=0;i<2;i++);
}
void delay_30us(void) //30us延时函数
{
unsigned int n;
for(n=10;n>0;n--)
asm("nop");
}
void delay_600us(void) //600us延时函数
{
unsigned
int i;
for(i = 0;i < 49;i ++)
delay_10us();
}
void delay_1ms(void) //1ms延时函数
{
unsigned int i;
for (i=0;i<407;i++);
}
void delay_nms(unsigned int n) //N ms延时函数
{
unsigned int i=0;
for (i=0;i<n;i++)
delay_1ms();
}
///////////18b20初始化/////////////////////
void init_1820()
{
SET_DIR_1WIRE; //设置PD0 为输出
SET_OP_1WIRE;
CLR_OP_1WIRE;
delay_600us(); //480us以上
SET_OP_1WIRE;
CLR_DIR_1WIRE;
delay_30us(); //15~60us
while(CHECK_IP_1WIRE);
SET_DIR_1WIRE;
SET_OP_1WIRE;
delay_30us();
delay_30us();
delay_30us();
delay_30us(); //60~240us
}
/////////写入///////////////////
void write_1820(unsigned char x)
{
unsigned char m;
for(m=0;m<8;m++)
{
CLR_OP_1WIRE;
if (x&(1<<m)) //写数据了,先写低位的!
SET_OP_1WIRE;
else
{CLR_OP_1WIRE;}
delay_30us(); //15~60us
SET_OP_1WIRE;
}
SET_OP_1WIRE;
}
////////////读出//////////////////////
unsigned char read_1820()
{
unsigned char temp,k,n;
temp=0;
for(n=0;n<8;n++)
{
CLR_OP_1WIRE;
SET_OP_1WIRE;
CLR_DIR_1WIRE;
k=(CHECK_IP_1WIRE); //读数据,从低位开始
if(k)
temp|=(1<<n);
else
temp&=~(1<<n);
delay_30us();
delay_30us(); //60~120us
SET_DIR_1WIRE;
}
return (temp);
}
///////////////读取温度值////////////////////////////
void gettemp(void)
{
unsigned char temh,teml,wm0,wm1,wm2,wm3;
init_1820(); //复位18b20
write_1820(0xcc); // 发出转换命令
write_1820(0x44);
init_1820();
write_1820(0xcc); //发出读命令
write_1820(0xbe);
teml=read_1820(); //读数据
temh=read_1820();
wm0=teml>>4; //只要高8位的低四位和低8位的高四位,温度范围0~99啦!
wm1=temh<<4;
wm2=wm1+wm0; //16进制转10进制
wm3=wm2/100;
wmh=(wm2%100)/10; //出口参数了!wmh是显示的高位,wml使显示的低位
wml=(wm2%100)%10;
}
这个是我做过的 一个关于18B20的 驱动程序 是AVR 的 楼主可以参考一下
//DS18B20的读写程序,数据脚P27 //
//温度传感器18B20汇编程序,采用器件默认的12位转化 //
//最大转化时间750微秒,显示温度-55到+125度,显示精度 //
//为01度,显示采用4位LED共阳显示测温值 //
//P0口为段码输入,P34~P37为位选 //
//
#include "reg51h"
#include "intrinsh" //_nop_();延时函数用
#define Disdata P0 //段码输出口
#define discan P2 //扫描口
#define uchar unsigned char
#define uint unsigned int
sbit DQ=P1^2; //温度输入口
sbit DIN=P0^7; //LED小数点控制
uint h;
uint temp;
//
//
//温度小数部分用查表法//
uchar code ditab[16]=
{0x00,0x01,0x01,0x02,0x03,0x03,0x04,0x04,0x05,0x06,0x06,0x07,0x08,0x08,0x09,0x09};
//
uchar code dis_7[12]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90,0xff,0xbf};
//共阳LED段码表 "0" "1" "2" "3" "4" "5" "6" "7" "8" "9" "不亮" "-"
uchar code scan_con[4]={0xfe,0xfd,0xfb,0xf7}; //列扫描控制字
uchar data temp_data[2]={0x00,0x00}; //读出温度暂放
uchar data display[5]={0x00,0x00,0x00,0x00,0x00}; //显示单元数据,共4个数据和一个运算暂用
//
//
//
/11us延时函数/
//
void delay(uint t)
{
for (;t>0;t--);
}
//
/显示扫描函数/
scan()
{
char k;
for(k=0;k<4;k++) //4位LED扫描控制
{
Disdata=dis_7[display[k]]; //数据显示
if (k==1){DIN=0;} //小数点显示
discan=scan_con[k]; //位选
delay(300);
P2=0xff;
}
}
//
//
/DS18B20复位函数/
ow_reset(void)
{
char presence=1;
while(presence)
{
while(presence)
{
DQ=1;_nop_();_nop_();//从高拉倒低
DQ=0;
delay(50); //550 us
DQ=1;
delay(6); //66 us
presence=DQ; //presence=0 复位成功,继续下一步
}
delay(45); //延时500 us
presence=~DQ;
}
DQ=1; //拉高电平
}
//
//
/DS18B20写命令函数/
//向1-WIRE 总线上写1个字节
void write_byte(uchar val)
{
uchar i;
for(i=8;i>0;i--)
{
DQ=1;_nop_();_nop_(); //从高拉倒低
DQ=0;_nop_();_nop_();_nop_();_nop_(); //5 us
DQ=val&0x01; //最低位移出
delay(6); //66 us
val=val/2; //右移1位
}
DQ=1;
delay(1);
}
//
/DS18B20读1字节函数/
//从总线上取1个字节
uchar read_byte(void)
{
uchar i;
uchar value=0;
for(i=8;i>0;i--)
{
DQ=1;_nop_();_nop_();
value>>=1;
DQ=0;_nop_();_nop_();_nop_();_nop_(); //4 us
DQ=1;_nop_();_nop_();_nop_();_nop_(); //4 us
if(DQ)value|=0x80;
delay(6); //66 us
}
DQ=1;
return(value);
}
//
/读出温度函数/
//
read_temp()
{
ow_reset(); //总线复位
delay(200);
write_byte(0xcc); //发命令
write_byte(0x44); //发转换命令
ow_reset();
delay(1);
write_byte(0xcc); //发命令
write_byte(0xbe);
temp_data[0]=read_byte(); //读温度值的第字节
temp_data[1]=read_byte(); //读温度值的高字节
temp=temp_data[1];
temp<<=8;
temp=temp|temp_data[0]; // 两字节合成一个整型变量。
return temp; //返回温度值
}
//
/温度数据处理函数/
//二进制高字节的低半字节和低字节的高半字节组成一字节,这个
//字节的二进制转换为十进制后,就是温度值的百、十、个位值,而剩
//下的低字节的低半字节转化成十进制后,就是温度值的小数部分
//
work_temp(uint tem)
{
uchar n=0;
if(tem>6348) // 温度值正负判断
{tem=65536-tem;n=1;} // 负温度求补码,标志位置1
display[4]=tem&0x0f; // 取小数部分的值
display[0]=ditab[display[4]]; // 存入小数部分显示值
display[4]=tem>>4; // 取中间八位,即整数部分的值
display[3]=display[4]/100; // 取百位数据暂存
display[1]=display[4]%100; // 取后两位数据暂存
display[2]=display[1]/10; // 取十位数据暂存
display[1]=display[1]%10;
/符号位显示判断/
if(!display[3])
{
display[3]=0; //最高位为0时不显示
if(!display[2])
{
display[2]=0; //次高位为0时不显示
}
}
if(n){display[3]=0x0b;} //负温度时最高位显示"-"
}
//
//
/主函数/
main()
{
Disdata=0xff; //初始化端口
discan=0xff;
for(h=0;h<4;h++) //开机显示"0000"
{display[h]=0;}
ow_reset(); //开机先转换一次
write_byte(0xcc); //Skip ROM
write_byte(0x44); //发转换命令
for(h=0;h<100;h++) //开机显示"0000"
{scan();}
while(1)
{
work_temp(read_temp()); //处理温度数据
scan(); //显示温度值
}
}
//
//结束//
以上就是关于求助:DS18B20温度测试,数码管显示。但是没有任何反应,不知道哪里出错了。请教高手指点。(附上程序)全部的内容,包括:求助:DS18B20温度测试,数码管显示。但是没有任何反应,不知道哪里出错了。请教高手指点。(附上程序)、求STC89C52单片机和DS18B20温度传感器的程序代码,用四位共阴数码管显示温度值、等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!
欢迎分享,转载请注明来源:内存溢出
微信扫一扫
支付宝扫一扫
评论列表(0条)