UART控制器LIN功能模式(NUC029LAN阅读笔记)
- LIN总线帧格式
- 字节域格式(LIN标准)
- 帧头(Header)
- 同步间隔域(Break Field)
- 同步域(Sync Byte Field)
- 受保护ID域(Protected Identifier Field)
- 响应域(Response)
- 数据域(Data1~N)
- 校验和(Checksum)
- LIN总线编程流程
LIN总线帧格式
一个LIN信息帧包含帧头(Header)和响应(Response)。
帧头(Header) 包含:间隔域(Break Field),同步域(Sync Field)和受保护的标识符域(Protected Indentifier Field);
响应(Response) 包含:数据域(Data1~N),校验和(Checksum)。
字节之间有字节间隔(InterByteSpace),在头信息和响应之间有一个响应间隔(ResponseSpace),这两个间隔的最小值为0。
下图是LIN功能模式的帧结构:
下图是LIN功能模式的帧格式:
字节域格式(LIN标准)
除起始域(Header)与响应间隔(ResponseSpace),其他部分都是以字节为单位传送,每个字节都有自己的格式,称之为字节域(ByteField)。
在LIN 模式下,根据LIN的标准,每个字节域的开始都是由一个值为0的起始位(显性的),后面跟随8个位的数据(LSB 优先),最后以一个值为1的停止位(隐性的)结束。
字节域(Byte Field):1位起始位(Start Bit,显性,值为0)+ 8位数据位 + 1位停止位(Stop Bit,隐性,值为1),是一种标准UART 数据传输格式。
下图是LIN功能模式的字节域格式:
帧头(Header)
同步间隔域(Break Field)
下图是LIN功能模式的同步间隔域格式:
同步域(Sync Byte Field)
同步域(Sync Byte Field),是固定值(0x55),二进制是01010101,由于是先发送低位,也就是先发送1。
下图是LIN功能模式的同步域格式:
受保护ID域(Protected Identifier Field)
受保护ID域(Protected Identifier Field),前6位为帧ID(Frame ID),最高两位是奇偶校验。
帧ID(Frame ID),前6位为帧ID,范围在0x00~0x3F之间。帧ID标识了帧的类别。从机任务对于帧头作出的反应(接收/发送/忽略应答部分)都是依据帧ID判断的。如果帧ID传输错误,将会导致信号无法正确到达目的地。因此引入奇偶校验位。
奇偶校验(Parity Check),最高两位是奇偶校验,其中ID6 是ID0、ID1、ID2、ID4 的奇校验,ID7 是ID1、ID3、ID4、ID5 的偶校验。
校验公式如下,其中“⊕”代表“异或”运算,“¬”代表“取非”运算。
P0 = ID0 ⊕ID1 ⊕ID2 ⊕ID4
P1 = ¬ (ID1 ⊕ID3 ⊕ID4 ⊕ID5)
(由公式可以看出,PID不会出现全0或全1的情况,因此,如果从机节点收到了“0xFF”或“0x00”,可判断为传输错误。)
下图是LIN功能模式的ID域格式:
下列是NUC029xANSeriesBSP固件库中,ID域奇偶校验位求值函数的代码:
uint8_t GetParityValue(uint32_t u32id)
{
uint32_t u32Res = 0, ID[6], p_Bit[2] , mask = 0;
for(mask = 0; mask < 6; mask++)
ID[mask] = (u32id & (1 << mask)) >> mask;
p_Bit[0] = (ID[0] + ID[1] + ID[2] + ID[4]) % 2;
p_Bit[1] = (!((ID[1] + ID[3] + ID[4] + ID[5]) % 2));
u32Res = u32id + (p_Bit[0] << 6) + (p_Bit[1] << 7);
return u32Res;
}
【校验和求值示例】
ID域对0x12的ID求校验位
xx01 0010 (0x12)
ID6 = (ID[0] + ID[1] + ID[2] + ID[4]) % 2; = 0
ID7 = (!((ID[1] + ID[3] + ID[4] + ID[5]) % 2)); = 1
得到ID域为
1001 0010 (0x92)
响应域(Response)
LIN协议中,一帧信息的响应域由数据域和校验码域两部分构成。
数据域(Data1~N)
每个数据域的开始都是由一个值为0的起始位 (显性的),后面跟随8个位的数据(LSB 优先),最后以一个值为1的停止位(隐性的)结束。
下图是LIN功能模式的数据域格式:
校验和(Checksum)
校验和(Checksum),是所有数据(可能包含ID域)带进位的8位和(即所有数据累加,大于等于256时,减去255)。
校验和计算方法:将校验对象的各字节作带进位二进制加法(每当结果大于等于256 时就减去255),并将所得最终的和逐位取反,以该结果作为要发送的校验和。
接收方根据校验和类型,对接收数据作相同的带进位二进制加法,最终的和不取反,并将该和与接收到的校验和作加法,如果结果为0xFF,则校验和无误。这在一定程度上保证了数据传输的正确性。
下图是LIN功能模式的校验和格式:
下列是NUC029xANSeriesBSP固件库中,校验和求值函数的代码:
#define MODE_CLASSIC 2
#define MODE_ENHANCED 1
uint32_t GetCheckSumValue(uint8_t *pu8Buf, uint32_t u32ModeSel)
{
uint32_t i, CheckSum = 0;
for(i = u32ModeSel; i <= 9; i++)
{
CheckSum += pu8Buf[i];
if(CheckSum >= 256)
CheckSum -= 255;
}
return (255 - CheckSum);
}
【校验和求值示例】
数据1域0x1:g_u8SendData[3] = 01
数据2域0x2:g_u8SendData[4] = 02
数据3域0x3:g_u8SendData[5] = 03
数据4域0x4:g_u8SendData[6] = 04
数据5域0x5:g_u8SendData[7] = 05
数据6域0x6:g_u8SendData[8] = 06
数据7域0x7:g_u8SendData[9] = 07
数据8域0x8:g_u8SendData[10] = 08
校验和含ID:g_u8SendData[11] = 49
求校验和(不含ID)
01+02+03+04+05+06+07+08=0x24
~0x24=0xDB
得g_u8SendData[11] = DB
求校验和(含ID)
92+01+02+03+04+05+06+07+08=1011 0110=0xB6
~0xB6=0100 1001=0x49
得g_u8SendData[11] = 49
LIN总线编程流程
LIN总线发送(TX)编程流程如下:
- 设置 UA_FUN_SEL[1:0] 为 01 来使能 LIN 总线模式。
- 设置 UA_LIN_BKFL(UA_ALT_CSR[3:0])寄存器来选择 break 域长度。(break 域的长度为UA_LIN_BKFL+2)
- 将THR(UA_THR[7:0])设置为0x55,来请求发送同步域。
- 通过写受保护的标识值到UA_THR中,来请求标识域传输。
- 设置 LIN_TX_EN(UA_ALT_CSR[7]) 位来开始传输(当break操作完成后,LIN_TX_EN(UA_ALT_CSR[7])将会被自动清除)。
- 当THR寄存器中最后一个字节中的STOP位发送到总线上后,硬件将会设置UA_FSR寄存器中的TE_FLAG位为1。
- 填N个字节的数据和校验码到THR(UA_THR[7:0]),然后重复步骤5和步骤6来发送数据。
下列是NUC029xANSeriesBSP固件库中,LIN总线主机发送LIN帧头、响应域的代码:
uint8_t g_u8SendData[12] = {0};
volatile int32_t g_i32pointer = 0;
void LIN_MasterTest(uint32_t u32id, uint32_t u32ModeSel)
{
uint32_t testPattern[8] = {0x1, 0x2, 0x3, 0x4, 0x5, 0x6, 0x7, 0x8};
LIN_SendHeader(u32id);
LIN_SendResponse(u32ModeSel, &testPattern[0]);
}
void LIN_SendHeader(uint32_t u32id)
{
g_i32pointer = 0 ;
UART_SelectLINMode(UART1, UART_ALT_CSR_LIN_TX_EN_Msk, 11);
g_u8SendData[g_i32pointer++] = 0x55 ;
g_u8SendData[g_i32pointer++] = GetParityValue(u32id);
UART_Write(UART1, g_u8SendData, 2);
}
void LIN_SendResponse(int32_t checkSumOption, uint32_t *pu32TxBuf)
{
int32_t i32;
for(i32 = 0; i32 < 8; i32++)
g_u8SendData[g_i32pointer++] = pu32TxBuf[i32] ;
g_u8SendData[g_i32pointer++] = GetCheckSumValue(g_u8SendData, checkSumOption) ;
UART_Write(UART1, g_u8SendData + 2, 9);
}
LIN 总结接收(RX)编程流程如下:
- 设置 UA_FUN_SEL[1:0] 为 01 来使能 LIN 总线模式。
- 设置 LIN_RX_EN(UA_ALT_CSR[6]) 来使能 LIN RX模式。
- 等待标志 LIN_RX_BREAK_IF(UA_ISR[7]) 来检测RX是否接收到 break 域。
- 等待标志RDA_IF(UA_ISR[0])并且从RBR(UA_RBR[7:0])寄存器读回数据。
参考:
- 《TRM_NUC029_Series_SC_Rev1.04.pdf》
- 《 LIN协议中的帧结构与同步域(0x55)》杭州山不高
声明:[笔记整理] 内容整理自网络,版权归原作者所有,若有侵权请联系删除。
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)