色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內(nèi)不再提示

基于MODBUS協(xié)議用STM32F103做從機接收發(fā)送數(shù)據(jù)包實驗

汽車電子技術 ? 來源:qq_34471646 ? 作者:qq_34471646 ? 2022-06-14 17:47 ? 次閱讀

給大家分享一個網(wǎng)友qq_34471646做的實驗,用STM32做從機接收主機發(fā)送過來的數(shù)據(jù)包(也是基于modbus協(xié)議),而后從機將一些數(shù)據(jù)發(fā)送給主機。

首先呢還是介紹下modbus協(xié)議。其實modbus協(xié)議你不需要了解太多。既然是要使用,那么你只需要明白一點,modbus協(xié)議就是在你要發(fā)送的數(shù)據(jù)的基礎上,在數(shù)據(jù)前面加上一個幀頭,數(shù)據(jù)后面加一個幀尾。嗯,是不是還是有點迷?舉個例子吧。


寄存器108的內(nèi)容表示為兩個十六進制字節(jié)值02 2B,或十進制555. 將寄存器109--110的內(nèi)容分別表示為十六進制的00 00和 00 64;或十進制的0 和100.

所以主機會發(fā)過來一幀數(shù)據(jù):01 03 00 6B 00 03 17 74(這個01是我假設主機的地址,這個域名的作用就是用來判斷是否是主機發(fā)送過來的數(shù)據(jù)。因為通信過程可能因為各種原因而導致主機發(fā)送過來的數(shù)據(jù)異常,故而我們從機接收到數(shù)據(jù)之后會先對數(shù)據(jù)進行分析主機發(fā)過來的數(shù)據(jù)是否正常,正常從機再發(fā)送數(shù)據(jù)過去,異常則不對這幀數(shù)據(jù)進行響應即從機不發(fā)數(shù)據(jù)。17 74是根據(jù)01 03 00 6B 00 03計算出來的CRC校驗值。)

當從機接收到這串數(shù)據(jù),并且判斷數(shù)據(jù)正常則發(fā)送一幀數(shù)據(jù)到主機:02 03 06 02 2B 00 00 00 64 11 8A(同樣的02是我假設的這個從機的地址,需注意的是咱們假設自己的從機地址不要與主機的地址相同。)在這幀數(shù)據(jù)中幀頭就是02 03 06,11 8A是根據(jù)02 03 06 02 2B 00 00 00 64計算出來的CRC 校驗值也是幀尾。

那么問題來了。。。程序中我們怎么去計算CRC校驗值呢???這個嘛 ,下方我會貼上整個實驗的例程,其中CRC.c中h函數(shù)unsigned int GetCRC16(unsigned char *ptr, unsigned char len)我們只需要調(diào)用這個函數(shù)就可以算出CRC校驗值了。有興趣的也可以去額外了解下CRC校驗具體是怎么實現(xiàn)的。

整個實驗例程如下:

main.c:

#include "stm32f10x.h"
#include "bsp_485.h"
#include "bsp_led.h"
#include "crc16.h"
/*描述:硬件RS485接口 協(xié)議:Modbus RTU
*功能:采用DMA方式發(fā)送數(shù)據(jù),中斷方式接收數(shù)據(jù)。
*注: 接收到指令之后,判斷是否是相應指令而進行DMA數(shù)據(jù)發(fā)送。
*/
/*DMA:開啟DMA,DMA發(fā)送完一幀數(shù)據(jù)后產(chǎn)生發(fā)送完成中斷,
* 在DMA發(fā)送完成中斷中,開啟USART接收中斷(字節(jié))
* 在USART接收中斷中保存接收到的數(shù)據(jù)。
*注: 本程序額外開啟了USART空閑中斷,在空閑中斷中將
* USART接收中斷中接收到的數(shù)據(jù)發(fā)送至串口調(diào)試助手顯示
* 并開啟DMA請求
*/
extern uint8_t u3Temp;
extern uint16_t uart3_p;
extern uint16_t ReceivedUsart3Flag,tempU3,uart3_RXbuff[];
extern uint8_t SendU3Buff[SENDU3BUFF_SIZE];
uint16_t len,iU3;
uint8_t u2_Temp;
char *pbuf;
void SendUsart3Buff(void);
void SendU3DatatoDebug(void);
void load_U3_SendBuff(void);
int main(void)
{
load_U3_SendBuff();
LED_GPIO_Config();
USART3_Config();
while(1)
{
}
}
/*Description:調(diào)試通信程序?qū)S茫糜趯⒔邮盏降臄?shù)據(jù)再發(fā)送到串口調(diào)試助手顯示*/
void SendU3DatatoDebug()
{
for(uart3_p= 1; uart3_p <= u3Temp; uart3_p++)
{
Usart_SendByte(USART3,uart3_RXbuff[uart3_p]);
}
uart3_p = 1;
RS485_RX_EN();
ReceivedUsart3Flag = 0;
}
/*將USART3需要發(fā)送的數(shù)據(jù)存放在SendU3Buff[]中*/
void load_U3_SendBuff()
{
uint16_t CRCtemp;
SendU3Buff[0]=0x01;//ID
SendU3Buff[1]=0x03;//功能碼
SendU3Buff[2]=0x0C;//內(nèi)容數(shù)據(jù)字節(jié)數(shù)
/*填充將要發(fā)送的數(shù)據(jù)(兩個字節(jié)為一個寄存器的值)*/
SendU3Buff[3]=0x00; //數(shù)據(jù)1(slave地址0,網(wǎng)站上地址40001)
SendU3Buff[4]=0x64;
SendU3Buff[5]=0x00;//數(shù)據(jù)2(slave地址1,網(wǎng)站上地址40002)
SendU3Buff[6]=0x96;
SendU3Buff[7]=0x00;//數(shù)據(jù)3
SendU3Buff[8]=0xC8;
SendU3Buff[9]=0x00; //4
SendU3Buff[10]=0xFA;
SendU3Buff[11]=0x01;//5頁面未添加該數(shù)據(jù)
SendU3Buff[12]=0x2B;
SendU3Buff[13]=0x01;//6
SendU3Buff[14]=0x5E;
CRCtemp = GetCRC16(SendU3Buff, 15);
/*CRC校驗值*/
SendU3Buff[15] = CRCtemp;//存放CRCl
SendU3Buff[16] = (CRCtemp >> 8); //保存CRCh//SendBuff[i] = ((CRCtemp<<8)|(CRCtemp>>8)); //保存CRCh
}
485.c:

#include "bsp_485.h"
uint8_t SendU3Buff[SENDU3BUFF_SIZE];
static void NVIC_USART3_Configuration(void)
{
NVIC_InitTypeDef NVIC_InitStructure;
/* 嵌套向量中斷控制器組選擇 */
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
/* 配置USART為中斷源 */
NVIC_InitStructure.NVIC_IRQChannel = USART3_IRQn;
/* 搶斷優(yōu)先級*/
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
/* 子優(yōu)先級 */
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;
/* 使能中斷 */
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
/* 初始化配置NVIC */
NVIC_Init(&NVIC_InitStructure);
}
static void NVIC_DMA1_2_Configuration(void)
{
NVIC_InitTypeDef NVIC_InitStructure;
/* 嵌套向量中斷控制器組選擇 */
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);
/* 配置USART為中斷源 */
NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel2_IRQn;
/* 搶斷優(yōu)先級*/
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
/* 子優(yōu)先級 */
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
/* 使能中斷 */
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
/* 初始化配置NVIC */
NVIC_Init(&NVIC_InitStructure);
}
//static void NVIC_DMA1_2_Configuration(void)
//{
// NVIC_InitTypeDef NVIC_InitStructure; /* Configure one bit for preemption priority */
// NVIC_PriorityGroupConfig(NVIC_PriorityGroup_1);
// NVIC_InitStructure.NVIC_IRQChannel = DMA1_Channel2_IRQn;
// NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 1;
// NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;
// NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
// NVIC_Init(&NVIC_InitStructure);
//}
/**
* U3:TX:PB10 RX:PB11 TXRXEN:PD3
* U5:TX:PC12 RX:PD1 TXRXEN:PD0
*
*/
void USART3_Config(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
USART_InitTypeDef USART_InitStructure;
// 打開串口GPIO的時鐘
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);//TX、RX是時鐘
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOD, ENABLE);//EN時鐘
// 打開串口外設的時鐘
DEBUG_USART_APBxClkCmd(RCC_APB1Periph_USART3, ENABLE);//USART時鐘
// 將USART Tx的GPIO配置為推挽復用模式
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOB, &GPIO_InitStructure);
// 將USART Rx的GPIO配置為浮空輸入模式
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_11;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING;
GPIO_Init(GPIOB, &GPIO_InitStructure);
/* 設置485收發(fā)控制管腳為推挽輸出Out_PP */
GPIO_InitStructure.GPIO_Pin = RS485_RE_PIN;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOD , &GPIO_InitStructure);
// 配置串口的工作參數(shù)
// 配置波特率
USART_InitStructure.USART_BaudRate = 115200;
// 配置 針數(shù)據(jù)字長
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;
// 配置工作模式,收發(fā)一起
USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
// 完成串口的初始化配置
USART_Init(USART3, &USART_InitStructure);
// 串口中斷優(yōu)先級配置
NVIC_USART3_Configuration();
// 使能串口接收中斷
USART_ITConfig(USART3, USART_IT_RXNE, ENABLE);
// 使能串口空閑中斷(用于檢測一幀數(shù)據(jù)接收完畢)
USART_ITConfig(USART3, USART_IT_IDLE, ENABLE);
// 使能串口
USART_Cmd(USART3, ENABLE);
/*控制 485 芯片進入接收模式*/
RS485_RX_EN();//
}
void USART3_DMA_Config(void)
{
DMA_InitTypeDef DMA_InitStructure;
// 開啟DMA時鐘
RCC_AHBPeriphClockCmd(RCC_AHBPeriph_DMA1, ENABLE);
/* 復位初始化 DMA 數(shù)據(jù)流 */
DMA_DeInit(DMA1_Channel2);
// 設置DMA外設地址:串口數(shù)據(jù)寄存器地址*/
DMA_InitStructure.DMA_PeripheralBaseAddr = USART_DR_ADDRESS;
// 內(nèi)存地址(要傳輸?shù)淖兞康闹羔?
DMA_InitStructure.DMA_MemoryBaseAddr = (u32)SendU3Buff;
// 方向:從內(nèi)存到外設
DMA_InitStructure.DMA_DIR = DMA_DIR_PeripheralDST;
// 傳輸大小
DMA_InitStructure.DMA_BufferSize = SENDU3BUFF_SIZE;
// 外設地址不增
DMA_InitStructure.DMA_PeripheralInc = DMA_PeripheralInc_Disable;
// 內(nèi)存地址自增
DMA_InitStructure.DMA_MemoryInc = DMA_MemoryInc_Enable;
// 外設數(shù)據(jù)單位
DMA_InitStructure.DMA_PeripheralDataSize = DMA_PeripheralDataSize_Byte;
// 內(nèi)存數(shù)據(jù)單位
DMA_InitStructure.DMA_MemoryDataSize = DMA_MemoryDataSize_Byte;
// DMA模式,一次或者循環(huán)模式
DMA_InitStructure.DMA_Mode = DMA_Mode_Normal ;
//DMA_InitStructure.DMA_Mode = DMA_Mode_Circular;
// 優(yōu)先級:中
DMA_InitStructure.DMA_Priority = DMA_Priority_Medium;
// 禁止內(nèi)存到內(nèi)存的傳輸
DMA_InitStructure.DMA_M2M = DMA_M2M_Disable;
// 配置DMA通道
DMA_Init(USART_TX_DMA_CHANNEL, &DMA_InitStructure);
NVIC_DMA1_2_Configuration();
//開啟DMA通道的TC中斷:傳輸完成中斷
DMA_ITConfig(DMA1_Channel2,DMA_IT_TC,ENABLE);
// 使能DMA
DMA_Cmd (DMA1_Channel2,ENABLE);
}
/***************** 發(fā)送一個字符 **********************/
void Usart_SendByte( USART_TypeDef * pUSARTx, uint8_t ch)
{
/* 發(fā)送一個字節(jié)數(shù)據(jù)到USART */
USART_SendData(pUSARTx,ch);
/* 等待發(fā)送數(shù)據(jù)寄存器為空 */
while (USART_GetFlagStatus(pUSARTx, USART_FLAG_TXE) == RESET);
}
/***************** 發(fā)送字符串 **********************/
void Usart_SendString( USART_TypeDef * pUSARTx, char *str)
{
unsigned int k=0;
do
{
Usart_SendByte( pUSARTx, *(str + k) );
k++;
} while(*(str + k)!='\0');
/* 等待發(fā)送完成 */
while(USART_GetFlagStatus(pUSARTx,USART_FLAG_TC)==RESET)
{}
}
/***************** 發(fā)送一個16位數(shù) **********************/
void Usart_SendHalfWord( USART_TypeDef * pUSARTx, uint16_t ch)
{
uint8_t temp_h, temp_l;
/* 取出高八位 */
temp_h = (ch&0XFF00)>>8;
/* 取出低八位 */
temp_l = ch&0XFF;
/* 發(fā)送高八位 */
USART_SendData(pUSARTx,temp_h);
while (USART_GetFlagStatus(pUSARTx, USART_FLAG_TXE) == RESET);
/* 發(fā)送低八位 */
USART_SendData(pUSARTx,temp_l);
while (USART_GetFlagStatus(pUSARTx, USART_FLAG_TXE) == RESET);
}
/*
Desc:接收中斷時,將接收到的所有數(shù)據(jù)用數(shù)組保存。
*/
//中斷緩存串口數(shù)據(jù)
#define UART_BUFF_SIZE 1024
uint16_t uart3_p = 1;
uint16_t uart3_RXbuff[UART_BUFF_SIZE];
uint8_t u3Temp;
uint8_t tempU3;
uint8_t ReceivedUsart3Flag = 0;
uint16_t clr;
uint16_t a=0x00;
void bspU3_RS485_IRQHandler(void)
{
if (USART_GetITStatus( USART3, USART_IT_RXNE) != RESET) //收到一個字節(jié)的數(shù)據(jù)
{//保存接收到的數(shù)據(jù)
uart3_RXbuff[uart3_p] = USART_ReceiveData(USART3);
uart3_p++;
}
if (USART_GetITStatus(USART3, USART_IT_IDLE) != RESET) //收到一幀的數(shù)據(jù)
{//將接收到的數(shù)據(jù)發(fā)送到串口調(diào)試助手上以便觀察數(shù)據(jù)是否正確
u3Temp = uart3_p-1;
clr = USART3->SR;
clr = USART3->DR;
ReceivedUsart3Flag = 1;
RS485_TX_EN() ;
// for(uart3_p= 1; uart3_p <= u3Temp; uart3_p++)//調(diào)試通信程序?qū)S茫糜趯⒔邮盏降臄?shù)據(jù)再發(fā)送到串口調(diào)試助手顯示
// {
// Usart_SendByte(USART3,uart3_RXbuff[uart3_p]);
// }
uart3_p = 1;
// LED1_OFF;
// LED2_OFF;
// LED3_OFF;
// LED4_OFF;
LED1_ON;
LED2_ON;
LED3_ON;
LED4_ON;
if(uart3_RXbuff[1]==0x01&&uart3_RXbuff[2]==0x03&&uart3_RXbuff[3]==0x00&&uart3_RXbuff[4]==0x00&&
uart3_RXbuff[5]==0x00&&uart3_RXbuff[6]==0x06&&uart3_RXbuff[7]==0xC5&&uart3_RXbuff[8]==0xC8)
// uart3_RXbuff[9]==0x00&&uart3_RXbuff[10]==0x01&&uart3_RXbuff[11]==0x03&&uart3_RXbuff[12]==0x9C&&
// uart3_RXbuff[13]==0x41&&uart3_RXbuff[14]==0x00&&uart3_RXbuff[15]==0x02&&uart3_RXbuff[16]==0x17&&
{
/* DMA發(fā)送使能 */
USART3_DMA_Config();
RS485_TX_EN() ;
USART_DMACmd(USART3, USART_DMAReq_Tx, ENABLE);//這個放在最后
}
}
}
void bspU3_RS485_DMA_IRQHandler()
{
if(DMA_GetITStatus(DMA1_IT_TC2))
{
//清TC標志
DMA_ClearITPendingBit(DMA1_IT_GL2); //清除全部中斷標志 //DMA_ClearFLAG(DMA1_FLAG_TC2); //清除全部中斷標志(這種寫法也可以)
while(USART_GetFlagStatus(USART3, USART_FLAG_TC) == RESET); //等待USART1發(fā)送完成標志TC置1
USART_ClearFlag(USART3, USART_FLAG_TC); //清除發(fā)送完成標志
}
//關閉DMA通道
DMA_Cmd(DMA1_Channel2, DISABLE);
RS485_RX_EN();
}

CRC16.c:

#include "crc16.h"
unsigned int GetCRC16(unsigned char *ptr, unsigned char len)
{
uint16_t index;
uint8_t crcl = 0xFF; //高CRC字節(jié)
uint8_t crch = 0xFF; //低CRC字節(jié)
uint8_t TabH[] = { //CRC高位字節(jié)值表
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0,
0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0,
0x80, 0x41, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1,
0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1,
0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0,
0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40,
0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1,
0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0,
0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40,
0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0,
0x80, 0x41, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0,
0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0,
0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0,
0x80, 0x41, 0x00, 0xC1, 0x81, 0x40, 0x00, 0xC1, 0x81, 0x40,
0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0, 0x80, 0x41, 0x00, 0xC1,
0x81, 0x40, 0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41,
0x00, 0xC1, 0x81, 0x40, 0x01, 0xC0, 0x80, 0x41, 0x01, 0xC0,
0x80, 0x41, 0x00, 0xC1, 0x81, 0x40
} ;
u8 TabL[] = { //CRC低位字節(jié)值表
0x00, 0xC0, 0xC1, 0x01, 0xC3, 0x03, 0x02, 0xC2, 0xC6, 0x06,
0x07, 0xC7, 0x05, 0xC5, 0xC4, 0x04, 0xCC, 0x0C, 0x0D, 0xCD,
0x0F, 0xCF, 0xCE, 0x0E, 0x0A, 0xCA, 0xCB, 0x0B, 0xC9, 0x09,
0x08, 0xC8, 0xD8, 0x18, 0x19, 0xD9, 0x1B, 0xDB, 0xDA, 0x1A,
0x1E, 0xDE, 0xDF, 0x1F, 0xDD, 0x1D, 0x1C, 0xDC, 0x14, 0xD4,
0xD5, 0x15, 0xD7, 0x17, 0x16, 0xD6, 0xD2, 0x12, 0x13, 0xD3,
0x11, 0xD1, 0xD0, 0x10, 0xF0, 0x30, 0x31, 0xF1, 0x33, 0xF3,
0xF2, 0x32, 0x36, 0xF6, 0xF7, 0x37, 0xF5, 0x35, 0x34, 0xF4,
0x3C, 0xFC, 0xFD, 0x3D, 0xFF, 0x3F, 0x3E, 0xFE, 0xFA, 0x3A,
0x3B, 0xFB, 0x39, 0xF9, 0xF8, 0x38, 0x28, 0xE8, 0xE9, 0x29,
0xEB, 0x2B, 0x2A, 0xEA, 0xEE, 0x2E, 0x2F, 0xEF, 0x2D, 0xED,
0xEC, 0x2C, 0xE4, 0x24, 0x25, 0xE5, 0x27, 0xE7, 0xE6, 0x26,
0x22, 0xE2, 0xE3, 0x23, 0xE1, 0x21, 0x20, 0xE0, 0xA0, 0x60,
0x61, 0xA1, 0x63, 0xA3, 0xA2, 0x62, 0x66, 0xA6, 0xA7, 0x67,
0xA5, 0x65, 0x64, 0xA4, 0x6C, 0xAC, 0xAD, 0x6D, 0xAF, 0x6F,
0x6E, 0xAE, 0xAA, 0x6A, 0x6B, 0xAB, 0x69, 0xA9, 0xA8, 0x68,
0x78, 0xB8, 0xB9, 0x79, 0xBB, 0x7B, 0x7A, 0xBA, 0xBE, 0x7E,
0x7F, 0xBF, 0x7D, 0xBD, 0xBC, 0x7C, 0xB4, 0x74, 0x75, 0xB5,
0x77, 0xB7, 0xB6, 0x76, 0x72, 0xB2, 0xB3, 0x73, 0xB1, 0x71,
0x70, 0xB0, 0x50, 0x90, 0x91, 0x51, 0x93, 0x53, 0x52, 0x92,
0x96, 0x56, 0x57, 0x97, 0x55, 0x95, 0x94, 0x54, 0x9C, 0x5C,
0x5D, 0x9D, 0x5F, 0x9F, 0x9E, 0x5E, 0x5A, 0x9A, 0x9B, 0x5B,
0x99, 0x59, 0x58, 0x98, 0x88, 0x48, 0x49, 0x89, 0x4B, 0x8B,
0x8A, 0x4A, 0x4E, 0x8E, 0x8F, 0x4F, 0x8D, 0x4D, 0x4C, 0x8C,
0x44, 0x84, 0x85, 0x45, 0x87, 0x47, 0x46, 0x86, 0x82, 0x42,
0x43, 0x83, 0x41, 0x81, 0x80, 0x40
} ;
while (len--) //計算指定長度的CRC
{
index = crcl ^ *ptr++;
crcl = crch ^ TabH[index];
crch = TabL[index];
}
return ((crch<<8) | crcl);??
}

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學習之用,如有內(nèi)容侵權或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • MODBUS
    +關注

    關注

    28

    文章

    1804

    瀏覽量

    76982
  • STM32
    +關注

    關注

    2270

    文章

    10897

    瀏覽量

    355862
  • ModBus協(xié)議

    關注

    3

    文章

    177

    瀏覽量

    33422
  • STM32F103
    +關注

    關注

    33

    文章

    477

    瀏覽量

    63628
收藏 人收藏

    評論

    相關推薦

    DJN Modbus協(xié)議

    DJN Modbus協(xié)議
    的頭像 發(fā)表于 11-27 13:52 ?1107次閱讀
    DJN <b class='flag-5'>Modbus協(xié)議</b>

    如何直接phy mac層發(fā)送接收802.11數(shù)據(jù)包

    我閱讀了完整的文檔(espressif_iot_esp8266ex_development_kit_v0.9.4.zip),但我沒有找到答案: 是否可以訪問 802.11 數(shù)據(jù)包,并通過應用程序處理它們? 我希望能夠直接 phy mac 層
    發(fā)表于 07-15 08:03

    Modbus協(xié)議轉(zhuǎn)Profibus協(xié)議網(wǎng)關接溫控表與PLC通訊

    。它集成了Modbus和Profibus兩種協(xié)議,可以接收Modbus協(xié)議數(shù)據(jù)并將其轉(zhuǎn)換為Profibus
    的頭像 發(fā)表于 06-21 11:31 ?1913次閱讀
    <b class='flag-5'>Modbus協(xié)議</b>轉(zhuǎn)Profibus<b class='flag-5'>協(xié)議</b>網(wǎng)關接溫控表與PLC通訊

    STM32F103CAN調(diào)試器接到該板子后給其發(fā)消息,調(diào)試器會顯示總線數(shù)據(jù)錯誤的原因?

    項目里有個CAN收發(fā)相關的板子,的是STM32F103芯片。在與上位的CAN交互過程中經(jīng)常會出現(xiàn)通訊錯誤的情況,表現(xiàn)在上位
    發(fā)表于 05-27 07:42

    STM32f103 USB雙緩存的時候,發(fā)送4K數(shù)據(jù)里面前64字節(jié)是0,為什么?

    STM32f103 USB 雙緩存的時候, 采取ENP3的TXaddr1和TXaddr0的雙緩存ping-pong發(fā)送機制,每次上位接收
    發(fā)表于 04-28 07:07

    STM32F030串口IDLE中斷接收不定長度數(shù)據(jù)包,中斷異常的原因?

    ;CR1 ~(1<<4); //清除IDLE中斷 rxne_num=0; } STM32F103:當串口調(diào)試工具,發(fā)送E1E2E3E4數(shù)據(jù)時,接收到第四個字節(jié)E4時,程序
    發(fā)表于 04-12 06:08

    Modbus協(xié)議網(wǎng)關是什么?Modbus協(xié)議網(wǎng)關的功能

    數(shù)據(jù)的互聯(lián)互通。Modbus協(xié)議作為一種廣泛應用于工業(yè)控制領域的通信協(xié)議,主要用于在主設備(如PLC、SCADA系統(tǒng))與設備(如傳感器、執(zhí)行器、智能儀表等)之間交換
    的頭像 發(fā)表于 04-11 15:51 ?740次閱讀

    stm32f103如何實現(xiàn)spi接收不定長數(shù)據(jù)

    stm32f103 如何實現(xiàn)spi接收不定長數(shù)據(jù)數(shù)據(jù)首字節(jié)說明了
    發(fā)表于 03-28 07:47

    STM32F103C8T6 SPI發(fā)送數(shù)據(jù)錯誤的原因?

    主機的是STM32F103C8T6,SPI1,的是STM32F051C8T6,SPI2,
    發(fā)表于 03-11 08:24

    STM32H7接收數(shù)據(jù)包異常,一接收數(shù)據(jù)出現(xiàn)兩發(fā)送的內(nèi)容怎么解決?

    );__HAL_UART_DISABLE_IT( huart1, DMA_IT_HT); 2、發(fā)送數(shù)據(jù)包1
    發(fā)表于 03-08 08:05

    ARM系列STM32F103芯片的解密方法

    本文介紹ARM系列STM32F103芯片的解密方法,其內(nèi)核是Cortex-M3,內(nèi)存16K-512K都有。
    發(fā)表于 02-28 11:20 ?1693次閱讀

    modbus協(xié)議功能碼分類 modbus協(xié)議功能碼是什么,有什么作用

    Modbus協(xié)議是一種用于工業(yè)領域常見的通信協(xié)議,它基于主從結(jié)構,用于在不同設備之間實現(xiàn)數(shù)據(jù)交換。Modbus協(xié)議的功能碼是決定數(shù)據(jù)交換類型
    的頭像 發(fā)表于 01-31 14:47 ?7099次閱讀

    modbus協(xié)議與485協(xié)議區(qū)別

    進行詳細探討。 一、協(xié)議層面 Modbus協(xié)議Modbus協(xié)議是一種開放的通信協(xié)議,它定義了一套
    的頭像 發(fā)表于 01-11 11:06 ?2.2w次閱讀

    stm32f103 flash模擬eeprom

    STM32F103是意法半導體(STMicroelectronics)推出的一款32位單片系列,該系列芯片具有高性能和豐富的外設接口,廣泛應用于工業(yè)控制、消費電子、汽車電子等領域。其中
    的頭像 發(fā)表于 01-09 11:21 ?2031次閱讀

    在消防預警系統(tǒng)中Modbus協(xié)議和EthernetIP協(xié)議都發(fā)揮著重要的作用

    Modbus協(xié)議是一種串行通信協(xié)議,最初是為PLC(可編程邏輯控制器)之間的通信而設計的。如今,Modbus協(xié)議在消防預警系統(tǒng)中應用廣泛。它可以實現(xiàn)主從通信,主設備發(fā)送請求,
    發(fā)表于 01-02 19:34
    主站蜘蛛池模板: 久草在在线免视频在线观看| 国产成人免费视频| GOGOGO高清在线播放免费| 99久久网站| 国产ts调教| 久久er99热精品一区二区| 麻豆久久婷婷五月国产| 青青视频国产色偷偷| 偷拍亚洲制服另类无码专区| 亚洲欧美一级久久精品| 在线中文字幕| 成人免费小视频| 国精品产露脸偷拍视频| 蜜桃麻豆WWW久久囤产精品免费| 热中文热国产热综合| 亚洲AV无码乱码A片无码蜜桃| 有码 亚洲 制服 国产 在线| 97在线观看成人免费视频| 公交车被CAO到合不拢腿| 婚后被调教当众高潮H喷水| 免费看片A级毛片免费看| 色多多污版app下载网站| 亚洲三级精品| 草莓视频在线播放视频| 国语对白刺激真实精品| 女人久久WWW免费人成看片| 午夜阳光影院在线观看视频| 综合人妻久久一区二区精品| 国产成人精品免费青青草原app | 他揉捏她两乳不停呻吟口述| 亚洲中文字幕在线精品| 超碰97免费人妻| 久久精品综合电影| 色爱AV综合区| 13一18TV处流血TV| 国产精品99久久久久久WWW| 伦理片秋霞免费影院| 性色欲情网站IWWW| SAO货腿张开JI巴CAO死我| 茎身铃口调教| 色一伦一情一区二区三区|