STM32串口实现1-wire(一)
2015-10-20 16:36
204 查看
STM32串口实现1-wire(一)
接线
只需要将STM32的串口的TX线和18B20的DQ相连接,通过4.7K上拉电阻让TX线拉到3.3V即可(PS:就不贴图了)。实现原理分析
STM32支持单线半双工通信,看手册大概是说内部TX和RX相连接。也就是发出去的数据能够直接收到。在TX线上没有连接任何设备时发生什么就能接收到什么。串口TTL起始位为低电平,空闲位高电平,这个和1-wire一样。
串口1btye数据包括1bit起始位(低电平)、8个数据位、1停止位(高电平)。可以通过串口1byte模拟出1-wire的1bit。
1-wire协议写信令,整个写需要60-120us,设备(18B20)在15-60us进行采样。如果将串口的波特率设置成115200,每个bit时间为1000000/115200=8.6us,串口1btye有9位数据组合成t=8.6*9=78us,刚好在1-wire协议范围内,并且采样时间在15-60us,8.6us(串口起始信号)还没有开始采样,通过控制串口发送的数据可以完成写0或者写1。如果串口发生0x00数据,就是1-wire写0信令,如果串口发生0xFF数据,就是1-wire写1信令。
1-wire协议读信令,串口发送0xFF,如果串口收到还是0xFF(单线半双工模式,TX和RX内部连接),那么就是读取1,或者就是读到0。
1-wire协议复位信令,由于复位的时序要求比较长,115200的波特率无法满足,复位时需要先将波特率设置成9600,复位完成后改成115200。发送0xF0进行复位(串口发生低位在前,发生5个连续的0),复位时长位1000000/9600 * 5us = 520us。如果收到0xF0说明设备没有应答,其他说明设备应答了。
代码实现
初始化代码/* *配置串口2 */ void DS18B20_init() { GPIO_InitTypeDef GPIO_InitStructure; USART_InitTypeDef USART_InitStructure; GPIO_PinRemapConfig(GPIO_Remap_USART2, ENABLE); RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_AFIO, ENABLE); RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2 , ENABLE); //USART2_TX PA.2 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_OD; GPIO_Init(GPIOA, &GPIO_InitStructure); //USART 初始化设置 USART_InitStructure.USART_BaudRate = 115200;//默认设置成115200 USART_InitStructure.USART_WordLength = USART_WordLength_8b; USART_InitStructure.USART_StopBits = USART_StopBits_1; USART_InitStructure.USART_Parity = USART_Parity_No; USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None; USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx; USART_Init(USART2, &USART_InitStructure); // USART_ITConfig(USART2, USART_IT_RXNE, ENABLE);//开启中断 USART_Cmd(USART2, ENABLE); //使能串口 USART_HalfDuplexCmd(USART2, ENABLE); }
只配置了TX管脚,使能单线半双工模式(USART_HalfDuplexCmd函数)
复位操作
/* *复位18B20, *返回0:找到设备,设备响应。 *返回1:没有找到设备,设置无响应 */ int DS18B20_Reset()//复位18B20 { unsigned char status ; USART_InitTypeDef USART_InitStructure; USART_InitStructure.USART_BaudRate = 9600;//一般设置为9600; USART_InitStructure.USART_WordLength = USART_WordLength_8b; USART_InitStructure.USART_StopBits = USART_StopBits_1; USART_InitStructure.USART_Parity = USART_Parity_No; USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None; USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx; USART_Init(USART2, &USART_InitStructure); USART_Cmd(USART2, ENABLE); USART_SendData(USART2,0xF0); //发送0xF0 低电平时长位 1/9600 * 1000 * 1000 * 5 us = 520us while (USART_GetFlagStatus(USART2, USART_FLAG_TC) == RESET) //等待传输完成 { } while (USART_GetFlagStatus(USART2, USART_FLAG_RXNE) == RESET) //等待发送完成 { } while(USART_GetFlagStatus(USART2, USART_FLAG_RXNE) != RESET)//等待收到数据,TX和RX内部连接,肯定能收到数据 { status = USART_ReceiveData(USART2); USART_ClearFlag(USART2,USART_FLAG_RXNE); } USART_InitStructure.USART_BaudRate = 115200;//将波特率改为115200 USART_InitStructure.USART_WordLength = USART_WordLength_8b; USART_InitStructure.USART_StopBits = USART_StopBits_1; USART_InitStructure.USART_Parity = USART_Parity_No; USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None; USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx; USART_Init(USART2, &USART_InitStructure); if(status == 0xF0) return 1; else return 0; }
写1Byte操作
void DS18B20_WriteBit(unsigned char bit) { if(bit) USART_SendData(USART2,0xFF); else USART_SendData(USART2,0x00); while (USART_GetFlagStatus(USART2, USART_FLAG_TC) == RESET) //等待传输完成 { } while (USART_GetFlagStatus(USART2, USART_FLAG_RXNE) == RESET) //等待收到数据 { } USART_ReceiveData(USART2); USART_ClearFlag(USART2,USART_FLAG_RXNE); } void DS18B20_WriteByte(unsigned char byte) { int i; for(i = 0 ; i < 8 ; i++)//8字节的数据组合成1字节 1-wire数据 { DS18B20_WriteBit((byte>>i) &0x01); } }
读1Byte操作
char DS18B20_ReadBit()//读取一位数据 { unsigned char status; USART_SendData(USART2,0xFF); while (USART_GetFlagStatus(USART2, USART_FLAG_TC) == RESET) //等待传输完成 { } while (USART_GetFlagStatus(USART2, USART_FLAG_RXNE) == RESET) //等待传输完成 { } status = USART_ReceiveData(USART2); USART_ClearFlag(USART2,USART_FLAG_RXNE); if(status == 0xFF) status = 1; else status = 0; return status; } char DS18B20_ReadByte() { unsigned char status = 0; int i; for(i = 0 ; i < 8 ; i++) { status |= (DS18B20_ReadBit() << i); } return status; }
18B20启动转换
void DS18B20_Start() { int i = 0xFFFF; DS18B20_reset(); DS18B20_WriteByte(0xCC);//跳过网络地址 DS18B20_WriteByte(0x44);//启动转换 while(i--);//等待转换完成 }
读取18B20温度值
float DS18B20_GetTemp() { unsigned char TL,TH ,t; short tem; float temp; DS18B20_Start(); DS18B20_reset(); DS18B20_WriteByte(0xCC);//跳过网络地址 DS18B20_WriteByte(0xBE);//读取温度值 TL=DS18B20_ReadByte(); // LSB TH=DS18B20_ReadByte(); // MSB if(TH>7) { TH=~TH; TL=~TL; t=0;//温度为负 }else t=1;//温度为正 tem=TH; //获得高八位 tem<<=8; tem+=TL;//获得底八位 temp = (float)tem*0.0625; if(t)return temp; //返回温度值 else return -temp; }
问题
在读取数据由于串口发送的是0xFF,将1-wire总线拉高,如果1-wire设备输出0时,此时会强制将1-wire总线拉低,即是TX线输出高电平,1-wire设备输出低电平,有可能将TX管脚烧掉(类似直接把TX线接到地上了),下一篇改进问题。相关文章推荐
- memset()的用法
- uva 10420
- iOS开发 MPMoviePlayerController
- reverse mapping checking getaddrinfo for bogon failed – POSSIBLE BREAK-IN ATTEMPT! 错误
- Win7上IIS发布网站系统\部署项目
- JSON 教程
- TCP/IP网络编程 基于Linux编程_3 --优于select的epoll
- layer---口碑极佳的web弹层组件
- ubuntu mysql使用
- SD卡中查找指定文件
- jQuery MiniUI开发系列之:UI和数据分离
- 第八周--项目3顺序串算法
- 局域网p2p终结者之类流氓软件抢占网速的原理
- android Binder的优点
- 装饰器设计模式
- python 中map、reduce、zip、filter内置函数作用说明
- 清理ambari安装的hadoop集群
- 指向指针的指针 写的非常好 适合0基础菜鸟
- coderforce 527A Arrays
- JAVA设计模式(10) —<结构型>桥接模式(Bridge)