如何在linux中写ds18b20驱动思维框图

如何在linux中写ds18b20驱动思维框图,第1张

(1)先DS18B20与处理器的连接接口设计,数据端要接一个5.1K的上位电阻。 (2)找到DS18B20的驱动程序。 (3)然后在主程序中,初始化DS18B20,接着读取温度值,进行处理就可以了。

/*使用举例:数码管

scan()

{

char k

for(k=0k<4k++) //4位LED扫描控制

{

discan=0x00

Disdata=dis_7[_1820display[k]] //数据显示

if (k==1){DIN=0} //小数点显示

discan=scan_con[k]//位选

_18B20_delay(100)

}

}

main()

_18B20_init()//18B20初始化

while(1)

{

EA=0//在利用18B20测试温度时,要严格遵循时序,禁止一切中断

_18B20_work(_18B20_read())//处理温度数据

EA=1//测试完毕,恢复系统中断

scan()//显示温度值

}

*/

#include "intrins.h"//_nop_()延时函数

//*****************//

//以下是DS18B20驱动程序

//*****************//

/**************************************************

** 功能描述: DS18B20驱动程序,使用12M晶体

** DQ占用引脚资源P1^7

****************************************************/sbit DQ=P1^7 //温度输入口unsigned char data temp_data[2]={0x00,0x00}//读出温度暂放

unsigned char data _1820display[5]={0x00,0x00,0x00,0x00,0x00}//显示单元数据,共4个数据和一个运算暂用

unsigned int temp

//**************温度小数部分用查表法***********//

unsigned char code ditab[16]=

{

0x00,0x01,0x01,0x02,

0x03,0x03,0x04,0x04,

0x05,0x06,0x06,0x07,

0x08,0x08,0x09,0x09

}/*****************11us延时函数*************************/

//

void _18B20_delay(unsigned int t)

{

for (t>0t--)

}/****************DS18B20复位函数************************/

_18B20_reset(void)

{

char presence=1

while(presence)

{

while(presence)

{

DQ=1

_nop_()_nop_()//从高拉倒低

DQ=0

_18B20_delay(50) //550 us

DQ=1

_18B20_delay(6) //66 us

presence=DQ//presence=0 复位成功,继续下一步

}

_18B20_delay(45) //延时500 us

presence=~DQ

}

DQ=1 //拉高电平

}/****************DS18B20写命令函数************************/

//向1-WIRE 总线上写1个字节

void _18B20_write(unsigned char val)

{

unsigned char i

for(i=8i>0i--)

{

DQ=1

_nop_()_nop_() //从高拉倒低

DQ=0

_nop_()_nop_()_nop_()_nop_() //5 us

DQ=val&0x01 //最低位移出

_18B20_delay(6) //66 us

val=val/2//右移1位

}

DQ=1

_18B20_delay(1)

}/****************DS18B20读1字节函数************************/

//从总线上取1个字节

unsigned char _18B20read_byte(void)

{

unsigned char i

unsigned char value=0

for(i=8i>0i--)

{

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

_18B20_delay(6)//66 us

}

DQ=1

return(value)

}

_18B20_read() //读出温度函数

{

_18B20_reset() //总线复位

_18B20_delay(200)

_18B20_write(0xcc) //发命令

_18B20_write(0x44) //发转换命令

_18B20_reset()

_18B20_delay(1)

_18B20_write(0xcc) //发命令

_18B20_write(0xbe)

temp_data[0]=_18B20read_byte() //读温度值的低字节

temp_data[1]=_18B20read_byte() //读温度值的高字节

temp=temp_data[1]

temp<<=8

temp=temp|temp_data[0] // 两字节合成一个整型变量。

return temp//返回温度值

}/****************温度数据处理函数************************///二进制高字节的低半字节和低字节的高半字节组成一字节,这个

//字节的二进制转换为十进制后,就是温度值的百、十、个位值,而剩

//下的低字节的低半字节转化成十进制后,就是温度值的小数部分/********************************************************/

_18B20_work(unsigned int tem)

{

unsigned char n=0

if(tem>6348)// 温度值正负判断

{

tem=65536-tem

n=1

}// 负温度求补码,标志位置1

_1820display[4]=tem&0x0f // 取小数部分的值

_1820display[0]=ditab[_1820display[4]] // 存入小数部分显示值

_1820display[4]=tem>>4// 取中间八位,即整数部分的值

_1820display[3]=_1820display[4]/100// 取百位数据暂存

_1820display[1]=_1820display[4]%100// 取后两位数据暂存

_1820display[2]=_1820display[1]/10 // 取十位数据暂存

_1820display[1]=_1820display[1]%10

/******************数码管符号位显示判断**************************/

if(!_1820display[3])

{

_1820display[3]=0x0a //最高位为0时不显示

if(!_1820display[2])

_1820display[2]=0x0a //次高位为0时不显示

}

if(n)

_1820display[3]=0x0b //负温度时最高位显示"-"

}

/******************1602液晶符号位显示判断**************************/

if(!_1820display[3])

{

_1820display[3]=' '-'0' //最高位为0时不显示

if(!_1820display[2])

_1820display[2]=' '-'0' //次高位为0时不显示

}

if(n)

_1820display[3]='-'-'0' //负温度时最高位显示"-"

} _18B20_init()//18B20初始化

{

_18B20_reset() //开机先转换一次

_18B20_write(0xcc)//Skip ROM

_18B20_write(0x44)//发转换命令

}

供你参考

#include <REGX51.H>

#define uchar unsigned char

#define uint unsigned int

#define OFF 1

#define ON 0

sbit LedRed=P2^7

sbit LedGreen=P2^6

//18B20函数声明

bit STA44(uchar i)//启动第i个1820的温度转换

bit STAbe(uchar i,uint *tc) //读第i个1820的温度存到指针tc处

uint TC//温度采样暂存

/* 温度发送缓存,S起始,ABCD组号,E结束,中间是温度 */

uchar TCA[4][6]={{'S','A',0,0,0,'E'},

{'S','B',0,0,0,'E'},

{'S','C',0,0,0,'E'},

{'S','D',0,0,0,'E'}}

uchar TCArow=0,TCAi=0//二维数组行、列索引

//串行口初始化

void init_S(void)

{

SCON=0x50 //串行口方式1

TMOD=(TMOD&0x0F)|0x20//T/C1 方式2

TH1=TL1=0xfe //波特率重装值

TR1=1 //启动T/C1

}

main(){

uint count

init_S()

while(1)

{

STA44((TCArow+1)%4) //启动下一路温度转换

if(STAbe(TCArow,&TC)){ //读上一路温度转换值

TC=TC*0.0625 //温度换算

TCA[TCArow][2]=TC/100%10+'0'//转存为ASCII码

TCA[TCArow][3]=TC/10%10+'0'

TCA[TCArow][4]=TC%10+'0'

}

#include <REGX51.H>

#define uchar unsigned char

#define uint unsigned int

void delay(uint i){while(--i)}

/* 指定DQ线 *** 作常数表,查表比计算快 */

uchar code DQa[3][4]={{0x00,0x00,0x00,0x00},

{0x01,0x02,0x04,0x08},

{0xFE,0xFD,0xFB,0xF7}}

/*给指定DQ线赋值*/

#define DQ(i,b) P1=P1&(DQa[2][i])|DQa[b][i]

/*读指定DQ线*/

#define DQack(i) P1&(DQa[1][i])

/* 初始化DS18B20,成功返回1,失败返回0 */

/* 参数i取值0~3,对应4只DS18B20,下同 */

bit STA_1820(uchar i){

bit ack

DQ(i,1)delay(5)

DQ(i,0)delay(80)

DQ(i,1)delay(5)

ack=DQack(i)delay(50)

return ack

}

//读DS18B20一字节,返回读到的数

char Read_1820(uchar i){

uchar j,dat=0

for(j=0j<8j++){

DQ(i,0)dat>>=1

DQ(i,1)if(DQack(i))dat|=0x80

delay(7)

}

return(dat)

}

//向DS18B20写一字节,参数为要写的数

void Write_1820(uchar i,uchar dat){

uchar j

for(j=0j<8j++) {

DQ(i,0)

DQ(i,dat&0x01)delay(7)

DQ(i,1)dat>>=1

}

}

/* 启动温度转换,成功返回1 */

bit STA44(uchar i){

if(!STA_1820(i)){

Write_1820(i,0xcc)

Write_1820(i,0x44)

return 1

}

else return 0

}

/* 读温度转换值,成功返回1,读到的值存到TC指针处 */

bit STAbe(uchar i,uint *TC){

uchar TCL,TCH //缓存采集值

if(!STA_1820(i)){ //重新启动总线

Write_1820(i,0xcc)//跳过ROM

Write_1820(i,0xbe)//准备读

TCL = Read_1820(i)//读0号

TCH = Read_1820(i)//读1号

*TC=(TCH*0x100+TCL)

return 1

}

else return 0

}


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/yw/11067236.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-05-13
下一篇 2023-05-13

发表评论

登录后才能评论

评论列表(0条)

保存