manwjh 发表于 2015-10-27 20:45

modbus项目,进行中

今天看了modbus的协议规范,同时找了一堆网上各种源代码。不满意,还是自己写吧。

如下代码不可能编译,因为我是边看规格书,边开始规划协议程序框架。CRC部分是摘抄自规范文档里面,结构借鉴了一份不完整的代码,我觉得命名符合我的审美。

/**
******************************************************************************
* @file    modbus.c
* @authorSZQVC
* @version V1.0.0
* @date    2015.10.27
* @brief   modbus协议解析
******************************************************************************
* @Attention                                                               *
*                                                                            *
* <h2><center>&copy; COPYRIGHT 2015 SZQVC</center></h2>                      *
*                                                                            *
* 文件版权归“深圳权成安视科技有限公司”(简称SZQVC)所有。                      *
*                                                                            *
*      QQ:936318687    http://www.szqvc.com                               *
*                                                                            *
******************************************************************************
**/

/*
MODBUS 长度
RS232 / RS485 ADU = 253字节+服务器地址(1 byte) + CRC (2字节) = 256字节。
TCP MODBUS ADU    = 249字节+ MBAP (7字节) = 256字节。

MODBUS请求PDU,   mb_req_pdu
MODBUS响应PDU,   mb_rsp_pdu
MODBUS异常响应PDU mb_excep_rsp_pdu
*/

/* define */


/* public */


/* extern */
//定义MODBUS服务器的存储单元
#define   MAX_I_COIL    1             //输入线圈数量
#define   MAX_O_COIL    1             //输出线圈数量
#define   MAX_I_REGI    1             //输入寄存器数量
#define   MAX_H_REGI    1             //保持寄存器数量
uint8_t   I_Coil;   //输入线圈
uint8_t   O_Coil;   //输出线圈
uint16_tI_Regi;         //输入寄存器
uint16_tH_Regi;         //保持寄存器

/* private */
#defineMAX_PDU_LEN   64
uint16_t PDU_Len;                     //缓冲接收数据
uint8_tPDU_Buffer;      //协议数据单元指针


/* function */
uint16_t CRC16( uint8_t *puchMsg, uint16_t usDataLen );


/*------------------------------------------------------------------------------*/
/*                              Modbus TASK                                     */
/*------------------------------------------------------------------------------*/
void ModbusTask(void)
{
uint8_t err_code; //错误码

//ADU模块(完成数据接收并传递到PDU缓冲,兼容RS485和TCP模式)


//PDU模块(针对PDU缓冲进行解析)
if( PDU_Len>MAX_PDU_LEN ){
    mb_excep_rsp_pdu(PDU_Buffer,SLAVE_DEVICE_BUSY);//数据长度超出能力范围
}else if( PDU_Len>0 ){
    switch(PDU_Buffer){
      //公共功能码
      case 1: //读取输出线圈状态
      ReadCoilState('O');
      break;
      case 2: //读取输入线圈状态
      ReadCoilState('I');
      break;
      case 3: //读取保持寄存器
      ReadRegiState('H');
      break;
      case 4: //读取输入积存器
      ReadRegiState('I');
      break;
      case 5: //设置单个线圈状态
      SetSingleCoil();
      break;
      case 6: //设置单个寄存器
      SetSingleRegi();
      break;
      case 15://设置多个线圈
      SetMultCoil();
      break;
      case 16://预置多个寄存器
      SetMultRegVal();
      break;
      default:
      mb_excep_rsp_pdu(PDU_Buffer,ILLEGAL_FUNCTION);
      break;
    }
}
}

void mb_rsp_pdu(uint8_t f_code)
{

}

void mb_excep_rsp_pdu(uint8_t f_code, uint8_t err_code)
{

}

void ReadCoilState(char type)
{
uint16_t coil_addr = PDU_Buffer<<8|PDU_Buffer;
uint16_t coil_num= PDU_Buffer<<8|PDU_Buffer;
   
switch(type)
{
    case 'I':
      if( addr>MAX_I_COIL ){
      mb_excep_rsp_pdu(PDU_Buffer,ILLEGAL_DATA_ADDRESS);   //地址超过界限
      }else if( (addr+coil_num)>MAX_I_COIL ){
      mb_excep_rsp_pdu(PDU_Buffer,ILLEGAL_DATA_VALUE);   //读取数量超过界限
      }else{
      }
      break;
    case 'O':
      if( addr>MAX_O_COIL ){
      mb_excep_rsp_pdu(PDU_Buffer,ILLEGAL_DATA_ADDRESS);
      }else if( (addr+coil_num)>MAX_O_COIL ){
      mb_excep_rsp_pdu(PDU_Buffer,ILLEGAL_DATA_VALUE);
      }else{
      }
      break;
}
}

void ReadRegiState(char type)
{
switch(type)
{
    case 'I':
      break;
    case 'H':
      break;
}
}

void SetSingleCoil(void)
{
}

void SetSingleRegi(char type)
{
}

void SetMultCoil(void)
{
}

void SetMultRegVal(char type)
{
}

/*------------------------------------------------------------------------------*/
/*                              CRC                                           */
/*------------------------------------------------------------------------------*/
//High-Order Byte Table
/* Table of CRC values for high–order byte */
const uint8_t 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};

//Low-Order Byte Table
/* Table of CRC values for low–order byte */
const uint8_t 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};

/* The function returns the CRC as a unsigned short type */
uint16_t CRC16( uint8_t *puchMsg, uint16_t usDataLen )
{
uint8_t uchCRCHi = 0xFF/* high byte of CRC initialized*/
uint8_t uchCRCLo = 0xFF/* low byte of CRC initialized*/
uint16_t uIndex   /* will index into CRC lookup table   */   

while (usDataLen--) /* pass through message buffer*/
{
    uIndex = uchCRCLo ^ *puchMsg++    /* calculate the CRC   */
    uchCRCLo = uchCRCHi ^ auchCRCHiuchCRCHi = auchCRCLo   
}
return (uchCRCHi << 8 | uchCRCLo);
}


ngyg12 发表于 2015-10-27 21:10

libmodbus    freemodbus现成的很多哦

manwjh 发表于 2015-10-29 10:53

ngyg12 发表于 2015-10-27 21:10
libmodbus    freemodbus现成的很多哦

能给个链接吗?

我搜索到的代码,许多都糟糕透了。

ngyg12 发表于 2015-10-29 14:10

manwjh 发表于 2015-10-29 10:53
能给个链接吗?

我搜索到的代码,许多都糟糕透了。

http://libmodbus.org/

cqwangsf 发表于 2016-5-29 10:24

自己写一个吧。又不难!

shizaigaole 发表于 2016-5-30 15:32

modbus协议比较简单,难度不大。
但是要写一个比较完全的版本,各种错误状态都有返回码的程序,
还是要花点力气的。

qianrushistm32 发表于 2016-6-2 09:19

freemodbus已经搞好给你了,现在还有人把slave 和master都做到了一块,master也是参照slave的框架编写的,阅读起来很顺畅

ziiiro 发表于 2017-7-13 23:07

标记下
页: [1]
查看完整版本: modbus项目,进行中