DS18B20的时序操作

1.DS18B20的时序操作

DS18B20参考连接图:
在这里插入图片描述
通讯方式:单总线模式通讯,DS18B20为从机;

1.1 复位

主机拉低18B20_DQ至少480us后,释放,将会被DS18B20视为复位脉冲,DS18B20将会对此复位脉冲回应,故:主机拉低18B20_DQ至少480us后延时15us进入接受模式等待DS18B20返回的数据。

//DS18B20复位
void DS18B20_Rest(void)	   
{                 
	DS18B20_IO_OUT(); //设置主机信号线对应IO为推挽输出模式 GPIOB->CRL&=0XFFFFFF3F
  	DS18B20_DQ_OUT=0; //主机拉低数据线
  	delay_us(600);    //延时600us
 	DS18B20_DQ_OUT=1; //拉高DQ信号线延时后接收数据 
	delay_us(15);     //延时15us
}

1.2 检查

//DS18B20检查
u8 DS18B20_check(void)	   
{                 
	u8 retry=0; //创建u8类型累加数据 
	DS18B20_IO_IN(); //设置主机数据线对应IO为浮空输入 GPIOB->CRL&=0XFFFFFF8F
  	while (DS18B20_DQ_IN&&retry<200) //数据线高电平时间超过200us则超时,跳出while循环,数据线高电平时间少于200us,跳出while循环
	{
		retry++;
		delay_us(1);
	};	 
	if(retry>=200) //超时认为DS18B20不存在 返回1
		return 1;
	else  //数据线200us内被拉低,retry置0
		retry=0;
    while (!DS18B20_DQ_IN&&retry<240) //数据线低电平持续时间超过240us超时,跳出while循环,数据线低电平保持时间少于240us跳出while循环
	{
		retry++;
		delay_us(1);
	};
	if(retry>=240) //数据线低电平持续时间超过240us超时,通讯异常,认为DS18B20不存在,返回1
		return 1;	    
	return 0; //数据线低电平保持时间少于240us跳出while循环,DS18B20存在,返回0
}

1.3 按字节写入

// DS18B20写入1个Byte
void DS18B20_Write_Byte(u8 dat)     
{             
    u8 j;
    u8 testb;
	DS18B20_IO_OUT(); //设置主机信号线对应IO为推挽输出模式 GPIOB->CRL&=0XFFFFFF3F
    for (j=1;j<=8;j++)   //1-8写入8个bit
	{
        testb=dat&0x01; //dat的第(0)位,赋值8次 bit0~bit7
        dat=dat>>1; //第1位变为第(0)位,共执行8次,7次有效,最后一次右移不赋值
        /*以下为写的位操作,执行时间62us*/
        if (testb) 
        {
            DS18B20_DQ_OUT=0; // 写1
            delay_us(2);                            
            DS18B20_DQ_OUT=1; 
            delay_us(60);             
        }
        else 
        {
            DS18B20_DQ_OUT=0;// 写0
            delay_us(60);             
            DS18B20_DQ_OUT=1;
            delay_us(2);                          
        }
    }
}

1.4 按位读取

//读取1bit,先写后读
u8 DS18B20_Read_Bit(void) 			 // 
{
    u8 data;
	DS18B20_IO_OUT(); //设置主机信号线对应IO为推挽输出模式 GPIOB->CRL&=0XFFFFFF3F
    DS18B20_DQ_OUT=0; //拉低数据线
	delay_us(2);
    DS18B20_DQ_OUT=1;  //拉高数据线
	DS18B20_IO_IN();   //设置主机数据线对应IO为浮空输入 GPIOB->CRL&=0XFFFFFF8F
	delay_us(12); //12us后读取电平状态
	/*高电平为1,低电平为0*/
	if(DS18B20_DQ_IN)
		data=1;
    else 
    	data=0;	 
    delay_us(50);  //读取操作时间62us,此时延时50us结束本次操作,(主机读取从机的bit写入)          
    return data; //返回读取到的bit
}

1.5 按字节读取

// 按字节读取,返回读到的字节数据
u8 DS18B20_Read_Byte(void)   
{        
    u8 i,j,dat;
    dat=0;
	for (i=1;i<=8;i++) 
	{
        j=DS18B20_Read_Bit(); //按位读取,先读低位,再读高位
        dat=(j<<7)|(dat>>1); //j为收到的低数据位,dat第一次操作为无效操作,后dat操作7次,第一次接收到的数据由7bit-0bit
    }						    
    return dat; //返回接收单字节
}

2.DS18B20的温度读取

操作过程:

DS18B20_Rest(); //DS18B20复位
DS18B20_check(); //检查是否存在DS18B20
DS18B20_Write_Byte(0xCC); //总线设备唯一,写入SKIP ROM命令,忽略设备选取的过程,直接进行后续的操作
DS18B20_Write_Byte(0xBE); //写入转换温度指令
DATA_CRL = DS18B20_Read_Byte(); //读取低8位
DATA_CRH = DS18B20_Read_Byte(); //读取高8位

3.DS18B20的温度数据结构

在这里插入图片描述
DS18B20数据:0bit~15bit,共计16位;
12bit~15bit:符号位,0000 -> +号 ; 1111 -> -号;
4bit~11bit: 整数位,
0bit~3bit: 小数位,

若:符号位为+,则温度等于:
(4bit-11bit)+( 0bit-3bit)*0.0625;
例如:+25.0625 = (0001 1001)+(0001)*0.0625 = 25+0.0625 = 25.0625

若:符号位为-,则温度等于正温度值取补码:
[ ~(4bit-11bit) ]+ [ ~(0bit-3bit) ]*0.0625 = -25.0625;

  • 9
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

LuDvei

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值