基于STM8的DS18B20检测

论坛 期权论坛 编程之家     
选择匿名的用户   2021-6-2 16:12   2232   0

本人实力菜,望大佬手下留情,随手一记。

最近要鼓捣出5V的ds18b20用stm8采集,网上一搜发现大部分都是把数据端直接接到3.3的引脚,我也这么做的。

用的DS18B20为淘宝外面有钢管的那种,系统板为STM8L051,第一次接触stm8先去找的库函数,以附件形式粘贴到下面。用IAR建工程一搜一大把。

单总线协议因为只有一条线,时序很重要,肯定要使用延时函数,用的原子哥的延时函数,,照搬原子哥的代码:http://www.openedv.com/posts/list/17347/htm

记得定义头文件时加上 #include "stm8l15x_conf.h",不然会报错

#include "delay.h"

volatile u8 fac_us=0;

void delay_init(u8 clk)
{
  if(clk>16)fac_us=(16-4)/4;
  else if(clk>4)fac_us=(clk-4)/4;
  else fac_us=1;
}

void delay_us(u16 nus)
{
  __asm(
"PUSH A          \n"    //1T,压栈
"DELAY_XUS:      \n"
"LD A,fac_us     \n"    //1T,fac_us加载到累加器A
"DELAY_US_1:     \n"
"NOP             \n"    //1T,nop延时
"DEC A           \n"    //1T,A--
"JRNE DELAY_US_1 \n"    //不等于0,则跳转(2T)到DELAY_US_1继续执行,若等于0,则不跳转(1T).
"NOP             \n"    //1T,nop延时
"DECW X          \n"    //1T,x--
"JRNE DELAY_XUS  \n"    //不等于0,则跳转(2T)到DELAY_XUS继续执行,若等于0,则不跳转(1T).
"POP A           \n"    //1T,出栈
);
}

void delay_ms(u32 nms)
{
  u8 t;
  if(nms>65)
  {
    t=nms/65;
    while(t--)delay_us(65000);
    nms=nms%65;
  }
  delay_us(nms*1000);
}

因为我没有显示的东西,只能用printf,在参考大佬「C_Aya」的文章,原文链接:https://blog.csdn.net/baweiyaoji/article/details/72812045 成功弄出来后,可以正常打印字符串,但一遇到带参数的printf立马内存爆炸,然后又在参考大佬「Dancer__Sky」的文章,原文链接:https://blog.csdn.net/Dancer__Sky/article/details/82284961得以成功打印。记得自己的头文件不要定义成这个名字"stdarg.h",我都不记得C有这个头文件了,不然你无法调用C的stdarg.h,将会出现错误。

现在就到了最关键的时候了,这一部分我参考的大佬石破天开 https://blog.csdn.net/u012166958/article/details/84872480

// 复位,主机给从机发送复位脉冲
void DS18B20_Rst(void)
{
  DS18B20_Mode_Out();
  delay_us(2);
  DS18B20_DQ_OUT_Low;  // 产生至少480us的低电平复位信号
  delay_us(700);
  DS18B20_DQ_OUT_High; // 在产生复位信号后,需将总线拉高
  delay_us(3);
}

// 写1字节到DS18B20
void DS18B20_Write_Byte(u8 dat)
{
  u8 j;
  u8 testb;
  DS18B20_Mode_Out();
  for (j = 1; j <= 8; j++)
  {
    testb = dat & 0x01;
    dat = dat >> 1;
    if (testb)
    {
      DS18B20_DQ_OUT_Low;// 写1
      delay_us(2);
      DS18B20_DQ_OUT_High;
      delay_us(80);
    }
    else
    {
      DS18B20_DQ_OUT_Low;// 写0
      delay_us(78);
      DS18B20_DQ_OUT_High;// 释放总线
      delay_us(2);
    }
}

这两部分延时不太一样,贴出来大家可以试一试。注意引脚输入不要弄错成浮空输入。

DS18B20 2.0版本,在之前基础上添加了查询序列号

当只有一个DS18B20的时候

void DS18B20_Search_Rom(void) //函数位置放在初始化下方 ,读取单个设备的序列号
{
  DS18B20_Write_Byte(0x33);
  delay_us(2);
  for(u8 j = 0; j <8; j++)
  {
    address[j] = DS18B20_Read_Byte();
  }
}    //重复读取可能有点问题,建议放在while上边

有多个DS18B20时,参考大佬 https://blog.csdn.net/u012166958/article/details/84872480

void DS18B20_Search_Rom(void)
{
  u8 ReadBit,data;
  u8 i,j,t = 0 ;
  u8 chongtuwei;
  u8 zhan[5];
  u8 Sequence[64];//序列码 
  DS18B20_Struct.ID_Num = 0;
  do
  {
//    DS18B20_Rst();   //这里我加上会出问题,所以我注释掉了
//    delay_us(450);
    DS18B20_Write_Byte(0xF0);
    delay_us(2);
    for(i = 0;i < 8;i++)
    {
      data =0;
      for(j = 0;j < 8;j++)
      {
        ReadBit = DS18B20_Read_2Bit();
        ReadBit = ReadBit & 0x03;
        data >>=1;
        if(ReadBit == 0x01) //读到的数据位0 写0 此位为0的器件响应
        {
          DS18B20_Write_Bit(0);
          Sequence[(i * 8 + j)] = 0;
        }else if(ReadBit == 0x02)//读到的数据位1 写1 此位为1的器件响应
        {
          data = data|0x80;
          DS18B20_Write_Bit(1);
          Sequence[i * 8 + j] = 1;
        }else if(ReadBit == 0x00) //读到的数据位0,有冲突位,判断冲突位
        {
                //如果冲突位大于栈顶写0,小于栈顶写以前数据,等于栈顶写1
          chongtuwei = i * 8 + j + 1;
          if(chongtuwei > zhan[t])
          {
                  DS18B20_Write_Bit(0);
                  Sequence[(i * 8 + j)] = 0;
                  zhan[++t] = chongtuwei;
          }else if (chongtuwei < zhan[t])
          {
                  data = data|((Sequence[(i * 8 + j)]&0x01)<<7);
                  DS18B20_Write_Bit(Sequence[(i * 8 + j)]);
          }else if(chongtuwei == zhan[t])
          {
                  data = data|0x80;
                  DS18B20_Write_Bit(1);
                  Sequence[(i * 8 + j)] = 1;
                  t--;
          }
        }else  //没搜索到
        {
        
        }
      } 
      DS18B20_Struct.DS18B20_ID[DS18B20_Struct.ID_Num][i] = data;
    }
    DS18B20_Struct.ID_Num++;//保存搜索到的个数
  }while(zhan[t] !=0&&(DS18B20_Struct.ID_Num < ROM_ID)); 
}

主函数

void main(void)
{
  sys_clock_init();//16MHZ工作
  GPIO_Init(GPIOD,GPIO_Pin_0,GPIO_Mode_Out_PP_Low_Fast);//配置D0的工作模式  
  usart_init(115200);
  delay_init(16);
  while(DS18B20_Init());//初始化加检测
  DS18B20_Search_Rom();
  while (1)
  { 
    
    //DS18B20_Init();
    i = DS18B20_Get_Temp();
    
    mprintf("temp=%f",i);
    
  }
}

最后大体是这个效果 版本1.0

分享到 :
0 人收藏
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

积分:3875789
帖子:775174
精华:0
期权论坛 期权论坛
发布
内容

下载期权论坛手机APP