說在前面的話:
1.
請勿盲目抄襲。這個協議使用了一個定時器,所以在別處請不要再使用,如果定時器不夠用,可以做虛擬定時器。也就是采用一個物理的定時器產生時基。在這個定時器的中斷函數中可以給相應的多個定時器自加1.每個虛擬定時器可以用兩個變量分別控制打開關閉,和計時。這個已經試驗通過了可行的。其實就跟我們使用物理的定時器一樣,只不過物理的定時器是用晶振產生時基。
2.
這段代碼已經調試通過了,也硬件試驗過,沒有問題,如果你出現問題了,看看你在主函數的的各種基本配置有沒有完成。如果要使用06和10號功能,你還需要在主函數中建立一個100個元素的數組,每個元素是16位。
3.
寫這個文檔的時候,這個協議已經是半年前完成的了。所以有些東西記得不是很清楚了,如果說錯了,請以實際為準。只是不想讓這份代碼死在電腦中了,所以才想起來要拿出來分享,支持開源精神。
4. 如果實在實在是沒有弄出來,請聯系我,可以共同交流,我的QQ:810663503
電話:18046771801
#include"stm32f10x.h"
//modbus用通訊參數
u8 Tim_Out;//大于3.5個字符時間,保守取3ms (波特率9600的時候大約2點幾毫秒)
u8 Rcv_Complete;//一幀是否已經接受完成
u8 Send_Complete;//一幀是否已經發送完成
u8 Com_busy;//通訊繁忙,表示上一幀還未處理結束
u8
Rcv_Buffer[210];//用來存放接收到的完整的一幀數據(第一個字節用來存放接收到的有效字節數,也就是數組中的有效字節數)
u8
Send_Buffer[210];//用來存放待發送的完整的一幀數據(第一個字節用來存放待發送的有效字節數,也就是數組中的有效字節數)
u8 Rcv_Data;//用來存放接收的一個字節
u8 Send_Data;//用來存放要發送的一字節
u8 Mod_Id;//用來標志作為從站的站號
u8 Rcv_Num;//用來表示接收的一幀的有效字節數(從功能碼到CRC校驗)
u8 Send_Num;//用來表示待發送的一幀的字節數
u8 *PointToRcvBuf;//用來指向接收的數據緩存
u8 *PointToSendBuf;//用來指向帶發送的數據緩存
u8 Comu_Busy;//用來表示能否接收下一幀數據
u8 HaveMes;
extern u16 HoldReg[100];
//CRC校驗查表用參數
static u8 auchCRCHi[] = {
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
} ;
static u8 auchCRCLo[] = {
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
} ;
//聲明modbus的函數
void ModInit(u8 Id);//用于Modbus初始化,參數Id為站號(1-255)
void ModRcv(void);//用于modbus信息接收
void ModSend(void);//用于modbus信息接收
void MessageHandle(u8 *pointer_in,u8
*pointer_out);//處理收到的信息幀
void ReadOutputBit(u8 *pointer_1,u8
*pointer_2);//讀線圈
void ReadInputBit(u8 *pointer_1,u8
*pointer_2);//讀輸入位
void ReadHoldingReg(u8 *pointer_1,u8
*pointer_2);//讀保持寄存器
void ReadInputReg(u8 *pointer_1,u8
*pointer_2);//讀輸入寄存器
void ForceSingleCoil(u8 *pointer_1,u8
*pointer_2);//強制單個線圈
void PresetSingleReg(u8 *pointer_1,u8
*pointer_2);//預制單個寄存器
void ForceMulCoil(u8 *pointer_1,u8
*pointer_2);//強制多個線圈
void PresetMulReg(u8 *pointer_1,u8
*pointer_2);//預制多個寄存器
void ErrorHandle(u8 Mode,u8 *Pointer);//錯誤信息幀處理
u16 CRC16(u8 *puchMsgg,u8 usDataLen);//用于計算CRC校驗碼
void ModInit(u8 Id)
{
//modbus參數初始化
PointToRcvBuf=Rcv_Buffer;
PointToSendBuf=Send_Buffer;
Send_Num=1;//發送的數據順序(輸出數組的第幾個數)
Mod_Id=Id;//站號設置
Rcv_Buffer[1]=Mod_Id;
Send_Buffer[1]=Mod_Id;
Comu_Busy=0;
}
void ModRcv(void)
{
HaveMes=1;//表示接收到了信息
Rcv_Data=USART_ReceiveData(USART1);
if(Comu_Busy!=1)//如果不忙,可以接收下一幀信息
{
TIM_Cmd(TIM2, DISABLE);
TIM_SetCounter(TIM2,0);
if((Tim_Out!=0)&&(Rcv_Data==Mod_Id))//如果間隔時間超過了3.5個字符,同時接受的字節和自己的站號一致,則認為接收開始
{
Rcv_Complete=0;//表示數據幀接收開始
Rcv_Num=0;//接收數據個數初始化
Rcv_Num++;//同時個數加一
}
if((0==Tim_Out)&&(0==Rcv_Complete))//如果處于接收一幀的正常過程中
{
if(Rcv_Num<100)
{
Rcv_Buffer[Rcv_Num+1]=Rcv_Data;//將數據放入接收數組中
Rcv_Num++;//同時個數加一
}
else
{
Rcv_Complete=1;
Comu_Busy=1;
Rcv_Buffer[0]=Rcv_Num;
*(PointToSendBuf+2)=*(PointToRcvBuf+2);//獲取功能碼
ErrorHandle(6,PointToSendBuf);//表示超出了字節數(從機設備忙碌)
Rcv_Num=0;
}
}
Tim_Out=0;
TIM_Cmd(TIM2, ENABLE);//開啟4.5ms計時(三個半字符的保守估計)
}
}
void ModSend(void)
{
Send_Data=*(PointToSendBuf+Send_Num);
USART_SendData(USART1,Send_Data);
Send_Num++;
if(Send_Num>(*PointToSendBuf))//發送已經完成
{
Comu_Busy=0;
*PointToSendBuf=0;
Rcv_Num=0;
Send_Num=1;
//啟動數據發送
USART_ITConfig(USART1, USART_IT_TC,
DISABLE);//關閉數據發送中斷
}
}
void TIM2_IRQHandler(void)
{
if (TIM_GetITStatus(TIM2, TIM_IT_Update) != RESET)
//檢查指定的TIM中斷發生與否:TIM 中斷源
{
TIM_ClearITPendingBit(TIM2, TIM_IT_Update ); //清除TIMx的中斷待處理位:TIM
中斷源
Tim_Out=1;
TIM_Cmd(TIM2,DISABLE);
TIM_SetCounter(TIM2,0);
Rcv_Complete=1;
Rcv_Buffer[0]=Rcv_Num;
函數輸入:兩個指針,pointer_1指向用來存放輸入信息幀的數組,
pointer_2用來指向存放輸出信息幀的數組(兩個數組的第一個元素都用來存放信息幀的有效字節個數)
后面的元素按照Modbus協議組織。
函數輸出:無。 */
void ForceMulCoil(u8 *pointer_1,u8
*pointer_2)//pointer_1用作輸入,pointer_2用作輸出
{
u16
Address=0;//待強制線圈起始地址(GPIO_X,X為C,D兩個端口,每個端口16位,對應地址0——31)
u16 Num=0;//要強制的線圈個數
u8 ByteCount;//命令值的字節個數
u32 CommandValue;//命令的數值,有效的每個位用來置位或者復位指定地址的線圈
u32 PortTemp;//用來存放從端口取過來的數據
u32 CalTemp;//臨時計算用
u16 ReadData=0;//用來臨時存放從端口讀來的數據
u16 SendKey;//要發送數據的校驗值
u8 CountTemp;//計算實際需要的命令字節數
Address=(u16)(*(pointer_1+3))*256+(*(pointer_1+4));//先得到線圈地址
Num=(u16)(*(pointer_1+5))*256+(*(pointer_1+6));//先得到線圈數量
ByteCount= *(pointer_1+7);//表示命令值的字節數
CountTemp=(u8)(Num/8);
if(Num%8!=0)
CountTemp++;
*(pointer_2+2)=0x0F;//第三個字節為功能碼
if((*(pointer_1)==9+ByteCount)&&ByteCount>0&&ByteCount<=4&&CountTemp==ByteCount)//如果接收到的字節數不是預定的個數,或者命令字節數超出允許范圍就是一個錯誤幀
{
if(Address<32) //只要地址小于32,就是合法地址
{
if(Address+Num<=32&&Num>0)
//只要地址加數量大于0小于32,就是合法數量
{
//用于for循環
u8 i;
u8 j;
//把命令值做一些處理,存入CommandValue
CommandValue=0;
for(i=0,j=ByteCount;j>0;i++,j--)
{
CommandValue|=((u32)*(pointer_1+8+i))<<8*i;//將輸入數據緩存中的數據存入CommandValue
}
CommandValue=CommandValue<<Address;//移動到對應起始位位置
*(pointer_2)=1+1+2+2+2;//有效字節個數等于叢機地址+功能碼+線圈起始地址+線圈數量+CRC校驗
*(pointer_2+3)=*(pointer_1+3);//將地址值寫入輸出的寄存器中
*(pointer_2+4)=*(pointer_1+4);
*(pointer_2+5)=*(pointer_1+5);//將線圈數量值寫入輸出的寄存器中
*(pointer_2+6)=*(pointer_1+6);
//將端口C和D的數據預先讀入到臨時的數據緩存中
ReadData=GPIO_ReadOutputData(GPIOD);
PortTemp=(u32)(ReadData);
PortTemp=PortTemp<<16;
ReadData=GPIO_ReadOutputData(GPIOC);
PortTemp=PortTemp|(u32)(ReadData);
//將需要讀出來的數據按要求處理
CalTemp=0xFFFFFFFF<<32-Num;
CalTemp=CalTemp>>32-Num-Address;
CalTemp=~CalTemp;
PortTemp&=CalTemp;
PortTemp|=CommandValue;
//再將數據寫入
GPIO_Write(GPIOC,(u16)(PortTemp&0x0000FFFF));
GPIO_Write(GPIOD,(u16)(PortTemp>>16));
//寫入校驗碼
SendKey=CRC16(pointer_2+1,*pointer_2-2);
//將計算出來的校驗碼裝入輸出數據緩存中
*(pointer_2+(*pointer_2-1))=(u8)(SendKey>>8);
*(pointer_2+(*pointer_2))=(u8)(SendKey&0x00FF);
//啟動數據發送
USART_ITConfig(USART1, USART_IT_TC,
ENABLE);//開啟數據發送中斷
}
else
{
ErrorHandle(3,pointer_2);//錯誤讀取數量
}
}
else
{
ErrorHandle(2,pointer_2);//錯誤起始地址
}
}
else
{
Comu_Busy=0;
}
}
void PresetMulReg(u8 *pointer_1,u8
*pointer_2)//pointer_1用作輸入,pointer_2用作輸出
{
u16
Address=0;//待預制寄存器的起始地址(HoldReg[i],i為0-99對應地址從0到99)
u16 Num=0;//要預制的寄存器數量
u8 ByteCount;//預制值的字節個數
u16 PresetValue=0;//預制數值
u16 SendKey;//要發送數據的校驗值
Address=(u16)(*(pointer_1+3))*256+(*(pointer_1+4));//先得到寄存器地址
Num=(u16)(*(pointer_1+5))*256+(*(pointer_1+6));//先得到待預制寄存器數量
*(pointer_2+2)=0x10;//第三個字節為功能碼
ByteCount= *(pointer_1+7);//表示命令值的字節數
if((*(pointer_1)==9+ByteCount)&&ByteCount>0&&ByteCount<=200&&ByteCount==(u8)(Num*2))//如果接收到的字節數不是預定的個數,或者命令字節數超出允許范圍就是一個錯誤幀
{
if(Address<100) //只要地址小于100,就是合法地址
{
if(Address+Num<=100&&Num>0)
//只要地址加數量大于0小于100,就是合法數量
{
//用于for循環
u8 i;
u8 j;
*(pointer_2)=1+1+2+2+2;//有效字節個數等于叢機地址+功能碼+寄存器地址+寄存器數量+CRC校驗
*(pointer_2+3)=*(pointer_1+3);//將地址值寫入輸出的寄存器中
*(pointer_2+4)=*(pointer_1+4);
*(pointer_2+5)=*(pointer_1+5);//將數量寫入輸出寄存器中
*(pointer_2+6)=*(pointer_1+6);
for(i=0,j=0;i
{
PresetValue=(u16)(*(pointer_1+8+j))*256+(*(pointer_1+9+j));//先得到預制值
HoldReg[Address+i]=PresetValue;//將預制值寫入保持寄存器
}
//寫入校驗碼
SendKey=CRC16(pointer_2+1,*pointer_2-2);
//將計算出來的校驗碼裝入輸出數據緩存中
*(pointer_2+(*pointer_2-1))=(u8)(SendKey>>8);
*(pointer_2+(*pointer_2))=(u8)(SendKey&0x00FF);
//啟動數據發送
USART_ITConfig(USART1, USART_IT_TC,
ENABLE);//開啟數據發送中斷
}
else
{
ErrorHandle(3,pointer_2);//錯誤讀取數量
}
}
else
{
ErrorHandle(2,pointer_2);//錯誤起始地址
}
}
else
{
Comu_Busy=0;
}
}
void ErrorHandle(u8 Mode,u8 *Pointer)
{
u16 SendKey;//要發送數據的校驗值
HaveMes=0;//清除信息位
TIM_Cmd(TIM2,DISABLE);
TIM_SetCounter(TIM3,0);
Rcv_Complete=1;
Comu_Busy=1;
Rcv_Buffer[0]=Rcv_Num;
switch(Mode)
{
case 1:*(Pointer+3)=0x01;//錯誤功能碼
break;
case 2:*(Pointer+3)=0x02;//錯誤地址
break;
case 3:*(Pointer+3)=0x03;//錯誤數據
break;
case 6:*(Pointer+3)=0x06;//從設備忙
break;
}
*Pointer=0x05;//輸出寄存器有效數據個數
*(Pointer+2)|=0x80;//功能碼最高位置一
//寫入校驗碼
SendKey=CRC16(Pointer+1,*Pointer-2);
//將計算出來的校驗碼裝入輸出數據緩存中
*(Pointer+(*Pointer-1))=(u8)(SendKey>>8);
*(Pointer+(*Pointer))=(u8)(SendKey&0x00FF);
//啟動數據發送
USART_ITConfig(USART1, USART_IT_TC, ENABLE);//開啟數據發送中斷
}
函數輸入:兩個指針,pointer_1指向用來存放輸入信息幀的數組,
pointer_2用來指向存放輸出信息幀的數組(兩個數組的第一個元素都用來存放信息幀的有效字節個數)
后面的元素按照Modbus協議組織。
函數輸出:無。 */
void ForceMulCoil(u8 *pointer_1,u8
*pointer_2)//pointer_1用作輸入,pointer_2用作輸出
{
u16
Address=0;//待強制線圈起始地址(GPIO_X,X為C,D兩個端口,每個端口16位,對應地址0——31)
u16 Num=0;//要強制的線圈個數
u8 ByteCount;//命令值的字節個數
u32 CommandValue;//命令的數值,有效的每個位用來置位或者復位指定地址的線圈
u32 PortTemp;//用來存放從端口取過來的數據
u32 CalTemp;//臨時計算用
u16 ReadData=0;//用來臨時存放從端口讀來的數據
u16 SendKey;//要發送數據的校驗值
u8 CountTemp;//計算實際需要的命令字節數
Address=(u16)(*(pointer_1+3))*256+(*(pointer_1+4));//先得到線圈地址
Num=(u16)(*(pointer_1+5))*256+(*(pointer_1+6));//先得到線圈數量
ByteCount= *(pointer_1+7);//表示命令值的字節數
CountTemp=(u8)(Num/8);
if(Num%8!=0)
CountTemp++;
*(pointer_2+2)=0x0F;//第三個字節為功能碼
if((*(pointer_1)==9+ByteCount)&&ByteCount>0&&ByteCount<=4&&CountTemp==ByteCount)//如果接收到的字節數不是預定的個數,或者命令字節數超出允許范圍就是一個錯誤幀
{
if(Address<32) //只要地址小于32,就是合法地址
{
if(Address+Num<=32&&Num>0)
//只要地址加數量大于0小于32,就是合法數量
{
//用于for循環
u8 i;
u8 j;
//把命令值做一些處理,存入CommandValue
CommandValue=0;
for(i=0,j=ByteCount;j>0;i++,j--)
{
CommandValue|=((u32)*(pointer_1+8+i))<<8*i;//將輸入數據緩存中的數據存入CommandValue
}
CommandValue=CommandValue<<Address;//移動到對應起始位位置
*(pointer_2)=1+1+2+2+2;//有效字節個數等于叢機地址+功能碼+線圈起始地址+線圈數量+CRC校驗
*(pointer_2+3)=*(pointer_1+3);//將地址值寫入輸出的寄存器中
*(pointer_2+4)=*(pointer_1+4);
*(pointer_2+5)=*(pointer_1+5);//將線圈數量值寫入輸出的寄存器中
*(pointer_2+6)=*(pointer_1+6);
//將端口C和D的數據預先讀入到臨時的數據緩存中
ReadData=GPIO_ReadOutputData(GPIOD);
PortTemp=(u32)(ReadData);
PortTemp=PortTemp<<16;
ReadData=GPIO_ReadOutputData(GPIOC);
PortTemp=PortTemp|(u32)(ReadData);
//將需要讀出來的數據按要求處理
CalTemp=0xFFFFFFFF<<32-Num;
CalTemp=CalTemp>>32-Num-Address;
CalTemp=~CalTemp;
PortTemp&=CalTemp;
PortTemp|=CommandValue;
//再將數據寫入
GPIO_Write(GPIOC,(u16)(PortTemp&0x0000FFFF));
GPIO_Write(GPIOD,(u16)(PortTemp>>16));
//寫入校驗碼
SendKey=CRC16(pointer_2+1,*pointer_2-2);
//將計算出來的校驗碼裝入輸出數據緩存中
*(pointer_2+(*pointer_2-1))=(u8)(SendKey>>8);
*(pointer_2+(*pointer_2))=(u8)(SendKey&0x00FF);
//啟動數據發送
USART_ITConfig(USART1, USART_IT_TC,
ENABLE);//開啟數據發送中斷
}
else
{
ErrorHandle(3,pointer_2);//錯誤讀取數量
}
}
else
{
ErrorHandle(2,pointer_2);//錯誤起始地址
}
}
else
{
Comu_Busy=0;
}
}
void PresetMulReg(u8 *pointer_1,u8
*pointer_2)//pointer_1用作輸入,pointer_2用作輸出
{
u16
Address=0;//待預制寄存器的起始地址(HoldReg[i],i為0-99對應地址從0到99)
u16 Num=0;//要預制的寄存器數量
u8 ByteCount;//預制值的字節個數
u16 PresetValue=0;//預制數值
u16 SendKey;//要發送數據的校驗值
Address=(u16)(*(pointer_1+3))*256+(*(pointer_1+4));//先得到寄存器地址
Num=(u16)(*(pointer_1+5))*256+(*(pointer_1+6));//先得到待預制寄存器數量
*(pointer_2+2)=0x10;//第三個字節為功能碼
ByteCount= *(pointer_1+7);//表示命令值的字節數
if((*(pointer_1)==9+ByteCount)&&ByteCount>0&&ByteCount<=200&&ByteCount==(u8)(Num*2))//如果接收到的字節數不是預定的個數,或者命令字節數超出允許范圍就是一個錯誤幀
{
if(Address<100) //只要地址小于100,就是合法地址
{
if(Address+Num<=100&&Num>0)
//只要地址加數量大于0小于100,就是合法數量
{
//用于for循環
u8 i;
u8 j;
*(pointer_2)=1+1+2+2+2;//有效字節個數等于叢機地址+功能碼+寄存器地址+寄存器數量+CRC校驗
*(pointer_2+3)=*(pointer_1+3);//將地址值寫入輸出的寄存器中
*(pointer_2+4)=*(pointer_1+4);
*(pointer_2+5)=*(pointer_1+5);//將數量寫入輸出寄存器中
*(pointer_2+6)=*(pointer_1+6);
for(i=0,j=0;i
{
PresetValue=(u16)(*(pointer_1+8+j))*256+(*(pointer_1+9+j));//先得到預制值
HoldReg[Address+i]=PresetValue;//將預制值寫入保持寄存器
}
//寫入校驗碼
SendKey=CRC16(pointer_2+1,*pointer_2-2);
//將計算出來的校驗碼裝入輸出數據緩存中
*(pointer_2+(*pointer_2-1))=(u8)(SendKey>>8);
*(pointer_2+(*pointer_2))=(u8)(SendKey&0x00FF);
//啟動數據發送
USART_ITConfig(USART1, USART_IT_TC,
ENABLE);//開啟數據發送中斷
}
else
{
ErrorHandle(3,pointer_2);//錯誤讀取數量
}
}
else
{
ErrorHandle(2,pointer_2);//錯誤起始地址
}
}
else
{
Comu_Busy=0;
}
}
void ErrorHandle(u8 Mode,u8 *Pointer)
{
u16 SendKey;//要發送數據的校驗值
HaveMes=0;//清除信息位
TIM_Cmd(TIM2,DISABLE);
TIM_SetCounter(TIM3,0);
Rcv_Complete=1;
Comu_Busy=1;
Rcv_Buffer[0]=Rcv_Num;
switch(Mode)
{
case 1:*(Pointer+3)=0x01;//錯誤功能碼
break;
case 2:*(Pointer+3)=0x02;//錯誤地址
break;
case 3:*(Pointer+3)=0x03;//錯誤數據
break;
case 6:*(Pointer+3)=0x06;//從設備忙
break;
}
*Pointer=0x05;//輸出寄存器有效數據個數
*(Pointer+2)|=0x80;//功能碼最高位置一
//寫入校驗碼
SendKey=CRC16(Pointer+1,*Pointer-2);
//將計算出來的校驗碼裝入輸出數據緩存中
*(Pointer+(*Pointer-1))=(u8)(SendKey>>8);
*(Pointer+(*Pointer))=(u8)(SendKey&0x00FF);
//啟動數據發送
USART_ITConfig(USART1, USART_IT_TC, ENABLE);//開啟數據發送中斷
} |