您的位置:首页 > 其它

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线接到地上了),下一篇改进问题。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: