分享使用串口操作DS18B20的代码,简化时序,提高系统实时性

2020-01-11 18:15发布


原理见帖子: https://www.amobbs.com/thread-5624824-1-1.html
从上次发帖子后, 一值没有写这个代码,直到最近要用到18B20,才写了一个,测试成功
用硬件操作的好处是全程不需要禁止中断,另外还可以用串口中断方式或DMA来操作

硬件连接: DS18B20连接到STM32F407VE的PA0, PA0是UART4接口的TX脚, 也是该串口单线半双工模式下时的TXRX脚,可以用此单引脚来发送和接收。该脚外接上拉电阻。

下面上代码:

  1. static const unsigned char crc_tab[256]={
  2.         0, 94,188,226, 97, 63,221,131,194,156,126, 32,163,253, 31, 65,
  3.         157,195, 33,127,252,162, 64, 30, 95,  1,227,189, 62, 96,130,220,
  4.         35,125,159,193, 66, 28,254,160,225,191, 93,  3,128,222, 60, 98,
  5.         190,224,  2, 92,223,129, 99, 61,124, 34,192,158, 29, 67,161,255,
  6.         70, 24,250,164, 39,121,155,197,132,218, 56,102,229,187, 89,  7,
  7.         219,133,103, 57,186,228,  6, 88, 25, 71,165,251,120, 38,196,154,
  8.         101, 59,217,135,  4, 90,184,230,167,249, 27, 69,198,152,122, 36,
  9.         248,166, 68, 26,153,199, 37,123, 58,100,134,216, 91,  5,231,185,
  10.         140,210, 48,110,237,179, 81, 15, 78, 16,242,172, 47,113,147,205,
  11.         17, 79,173,243,112, 46,204,146,211,141,111, 49,178,236, 14, 80,
  12.         175,241, 19, 77,206,144,114, 44,109, 51,209,143, 12, 82,176,238,
  13.         50,108,142,208, 83, 13,239,177,240,174, 76, 18,145,207, 45,115,
  14.         202,148,118, 40,171,245, 23, 73,  8, 86,180,234,105, 55,213,139,
  15.         87,  9,235,181, 54,104,138,212,149,203, 41,119,244,170, 72, 22,
  16.         233,183, 85, 11,136,214, 52,106, 43,117,151,201, 74, 20,246,168,
  17.         116, 42,200,150, 21, 75,169,247,182,232, 10, 84,215,137,107, 53
  18. };

  19. unsigned char Check_crc8(unsigned char *p,unsigned char n)
  20. {
  21.         unsigned char i,crc8=0;
  22.         for(i=0;i<n;i++)
  23.         crc8=crc_tab[crc8^p[i]];                //连续查表计算CRC
  24.         return crc8;
  25. }

  26. void DS18B20Init()
  27. {
  28.         //时钟,UART4, PA0
  29.         RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA,ENABLE);
  30.         RCC_APB1PeriphClockCmd(RCC_APB1Periph_UART4,ENABLE);
  31.         //GPIO->PA0
  32.         GPIO_InitTypeDef Gpio = {GPIO_Pin_0,GPIO_Mode_AF,GPIO_Low_Speed,GPIO_OType_OD,GPIO_PuPd_NOPULL};
  33.         GPIO_Init(GPIOA,&Gpio);
  34.     GPIO_PinAFConfig(GPIOA,GPIO_PinSource0,GPIO_AF_UART4);  
  35.         //UART4
  36.         USART_InitTypeDef Uart = {        115200,
  37.                                                                 USART_WordLength_8b,
  38.                                                                 USART_StopBits_1,
  39.                                                                 USART_Parity_No,
  40.                                                                 USART_Mode_Rx|USART_Mode_Tx,
  41.                                                                 USART_HardwareFlowControl_None};
  42.         USART_Init(UART4,&Uart);
  43.         USART_HalfDuplexCmd(UART4,ENABLE);
  44.         USART_Cmd(UART4,ENABLE);
  45.         //
  46. }

  47. bool DS18B20Reset()
  48. {
  49.         u8 rx;
  50.         UART4->BRR = 0x1117;                        //9600
  51.         UART4->DR = 0xF0;
  52.         while (!(UART4->SR&USART_SR_RXNE));
  53.         rx = UART4->DR;
  54.         UART4->BRR = 0x16C;                                //115200
  55.         return rx!=0xF0;
  56. }

  57. void DS18B20WriteByte(u8 byte)
  58. {
  59.         for (int i=0;i<8;i++)
  60.         {
  61.                 u8 c=(byte&0x01) ? 0xff:0x00;
  62.                 UART4->DR = c;
  63.                 byte>>=1;
  64.                 while (!(UART4->SR&USART_SR_RXNE));
  65.                 c=UART4->DR;
  66.         }
  67. }

  68. u8 DS18B20ReadByte()
  69. {
  70.         u8 byte,rx;
  71.         for (int i=0;i<8;i++)
  72.         {
  73.                 UART4->DR = 0xff;
  74.                 byte>>=1;
  75.                 while (!(UART4->SR&USART_SR_RXNE));
  76.                 rx=UART4->DR;
  77.                 if (rx==0xff) byte|=0x80;
  78.         }
  79.         return byte;
  80. }

  81. int DS18B20GetTemp()
  82. {
  83.         u8 Buf[9];
  84.         //发送复位信号
  85.         if (!DS18B20Reset()) return -1;
  86.         //发送SKIP_ROM       
  87.         DS18B20WriteByte(0xCC);
  88.         //发送转换温度命令
  89.         DS18B20WriteByte(0x44);
  90.         //等待转换, 最大750ms
  91.         OSTimeDly(800);
  92.        
  93.         //发送复位信号
  94.         if (!DS18B20Reset()) return -1;
  95.         //发送SKIP_ROM       
  96.         DS18B20WriteByte(0xCC);
  97.         //发送读取温度命令
  98.         DS18B20WriteByte(0xBE);
  99.         //读取9字节内存
  100.         for (int i=0;i<9;i++)
  101.                 Buf[i]=DS18B20ReadByte();
  102.         //CRC
  103.         if (Check_crc8(Buf,9)!=0)
  104.                 return -1;
  105.         return Buf[1]*256+Buf[0];
  106. }
复制代码
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
该问题目前已经被作者或者管理员关闭, 无法添加新回复
33条回答
dwj0
1楼-- · 2020-01-14 01:45
JJKwong 发表于 2018-1-8 16:25
看到CSDN博客上:http://blog.csdn.net/u010339445/article/details/49279177

在读取数据由于串口发送的是 ...

TX管脚配置为开漏输出,不存在此问题
chenchaoting
2楼-- · 2020-01-14 04:52
来个51的看看
huangqi412
3楼-- · 2020-01-14 10:06
 精彩回答 2  元偷偷看……
huangqi412
4楼-- · 2020-01-14 11:55
JJKwong 发表于 2018-1-8 16:25
看到CSDN博客上:http://blog.csdn.net/u010339445/article/details/49279177

在读取数据由于串口发送的是 ...

开漏就不怕  没开漏那就tx rx都用上 用二极管
youkebing
5楼-- · 2020-01-14 12:49
其实用spi口感觉也比串口更好,我就写过一个spi口的,
wctmdgcd
6楼-- · 2020-01-14 15:14
这个不错。。。
收藏一下

一周热门 更多>