📄 mbfuncholding.c
字号:
/* * FreeRTOS Modbus Libary: A Modbus serial implementation for FreeRTOS * Copyright (C) 2006 Christian Walter <wolti@sil.at> * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA *//* ----------------------- System includes ----------------------------------*/#include "stdlib.h"#include "string.h"/* ----------------------- Platform includes --------------------------------*/#include "port.h"/* ----------------------- Modbus includes ----------------------------------*/#include "mb.h"#include "mbframe.h"#include "mbproto.h"#include "mbconfig.h"/* ----------------------- Defines ------------------------------------------*/#define MB_PDU_FUNC_READ_ADDR_OFF ( MB_PDU_DATA_OFF + 0)#define MB_PDU_FUNC_READ_REGCNT_OFF ( MB_PDU_DATA_OFF + 2 )#define MB_PDU_FUNC_READ_SIZE ( 4 )#define MB_PDU_FUNC_READ_REGCNT_MAX ( 0x007D )#define MB_PDU_FUNC_WRITE_ADDR_OFF ( MB_PDU_DATA_OFF + 0)#define MB_PDU_FUNC_WRITE_VALUE_OFF ( MB_PDU_DATA_OFF + 2 )#define MB_PDU_FUNC_WRITE_SIZE ( 4 )#define MB_PDU_FUNC_WRITE_MUL_ADDR_OFF ( MB_PDU_DATA_OFF_+ 0 )#define MB_PDU_FUNC_WRITE_MUL_REGCNT_OFF ( MB_PDU_DATA_OFF + 2 )#define MB_PDU_FUNC_WRITE_MUL_BYTECNT_OFF ( MB_PDU_DATA_OFF + 4 )#define MB_PDU_FUNC_WRITE_MUL_VALUES_OFF ( MB_PDU_DATA_OFF + 5 )#define MB_PDU_FUNC_WRITE_MUL_SIZE_MIN ( 5 )#define MB_PDU_FUNC_WRITE_MUL_REGCNT_MAX ( 0x0078 )#define MB_PDU_FUNC_READWRITE_READ_ADDR_OFF ( MB_PDU_DATA_OFF + 0)#define MB_PDU_FUNC_READWRITE_READ_REGCNT_OFF ( MB_PDU_DATA_OFF + 2)#define MB_PDU_FUNC_READWRITE_WRITE_ADDR_OFF ( MB_PDU_DATA_OFF + 4)#define MB_PDU_FUNC_READWRITE_WRITE_REGCNT_OFF ( MB_PDU_DATA_OFF + 6)#define MB_PDU_FUNC_READWRITE_BYTECNT_OFF ( MB_PDU_DATA_OFF + 8)#define MB_PDU_FUNC_READWRITE_WRITE_VALUES_OFF ( MB_PDU_DATA_OFF + 9)#define MB_PDU_FUNC_READWRITE_SIZE_MIN ( 9 )/* ----------------------- Static functions ---------------------------------*/eMBException prveMBError2Exception( eMBErrorCode eErrorCode );/* ----------------------- Start implementation -----------------------------*/#if MB_FUNC_WRITE_HOLDING_ENABLED > 0eMBExceptioneMBFuncWriteHoldingRegister( UCHAR * pucFrame, USHORT * usLen ){ USHORT usRegAddress; eMBException eStatus = MB_EX_NONE; eMBErrorCode eRegStatus; if( *usLen == ( MB_PDU_FUNC_WRITE_SIZE + MB_PDU_SIZE_MIN ) ) { usRegAddress = pucFrame[MB_PDU_FUNC_WRITE_ADDR_OFF] << 8; usRegAddress |= pucFrame[MB_PDU_FUNC_WRITE_ADDR_OFF + 1]; usRegAddress++; /* Make callback to update the value. */ eRegStatus = eMBRegHoldingCB( &pucFrame[MB_PDU_FUNC_WRITE_VALUE_OFF], usRegAddress, 1, MB_REG_WRITE ); /* If an error occured convert it into a Modbus exception. */ if( eRegStatus != MB_ENOERR ) { eStatus = prveMBError2Exception( eRegStatus ); } } else { /* Can't be a valid request because the length is incorrect. */ eStatus = MB_EX_ILLEGAL_DATA_VALUE; } return eStatus;}#endif#if MB_FUNC_WRITE_MULTIPLE_HOLDING_ENABLED > 0eMBExceptioneMBFuncWriteMultipleHoldingRegister( UCHAR * pucFrame, USHORT * usLen ){ USHORT usRegAddress; USHORT usRegCount; UCHAR ucRegByteCount; eMBException eStatus = MB_EX_NONE; eMBErrorCode eRegStatus; if( *usLen >= ( MB_PDU_FUNC_WRITE_MUL_SIZE_MIN + MB_PDU_SIZE_MIN ) ) { usRegAddress = pucFrame[MB_PDU_FUNC_READ_ADDR_OFF] << 8; usRegAddress |= pucFrame[MB_PDU_FUNC_READ_ADDR_OFF + 1]; usRegAddress++; usRegCount = pucFrame[MB_PDU_FUNC_WRITE_MUL_REGCNT_OFF] << 8; usRegCount |= pucFrame[MB_PDU_FUNC_WRITE_MUL_REGCNT_OFF + 1]; ucRegByteCount = pucFrame[MB_PDU_FUNC_WRITE_MUL_BYTECNT_OFF]; if( ( usRegCount >= 1 ) && ( usRegCount <= MB_PDU_FUNC_WRITE_MUL_REGCNT_MAX ) && ( ucRegByteCount == ( UCHAR ) ( 2 * usRegCount ) ) ) { /* Make callback to update the register values. */ eRegStatus = eMBRegHoldingCB( &pucFrame[MB_PDU_FUNC_WRITE_MUL_VALUES_OFF], usRegAddress, usRegCount, MB_REG_WRITE ); /* If an error occured convert it into a Modbus exception. */ if( eRegStatus != MB_ENOERR ) { eStatus = prveMBError2Exception( eRegStatus ); } else { /* The response contains the function code, the starting * address and the quantity of registers. We reuse the * old values in the buffer because they are still valid. */ *usLen = MB_PDU_FUNC_WRITE_MUL_BYTECNT_OFF; } } else { eStatus = MB_EX_ILLEGAL_DATA_VALUE; } } else { /* Can't be a valid request because the length is incorrect. */ eStatus = MB_EX_ILLEGAL_DATA_VALUE; } return eStatus;}#endif#if MB_FUNC_READ_HOLDING_ENABLED > 0eMBExceptioneMBFuncReadHoldingRegister( UCHAR * pucFrame, USHORT * usLen ){ USHORT usRegAddress; USHORT usRegCount; UCHAR *pucFrameCur; eMBException eStatus = MB_EX_NONE; eMBErrorCode eRegStatus; if( *usLen == ( MB_PDU_FUNC_READ_SIZE + MB_PDU_SIZE_MIN ) ) { usRegAddress = pucFrame[MB_PDU_FUNC_READ_ADDR_OFF] << 8; usRegAddress |= pucFrame[MB_PDU_FUNC_READ_ADDR_OFF + 1]; usRegAddress++; usRegCount = pucFrame[MB_PDU_FUNC_READ_REGCNT_OFF] << 8; usRegCount = pucFrame[MB_PDU_FUNC_READ_REGCNT_OFF + 1]; /* Check if the number of registers to read is valid. If not * return Modbus illegal data value exception. */ if( ( usRegCount >= 1 ) && ( usRegCount <= MB_PDU_FUNC_READ_REGCNT_MAX ) ) { /* Set the current PDU data pointer to the beginning. */ pucFrameCur = &pucFrame[MB_PDU_FUNC_OFF]; *usLen = MB_PDU_FUNC_OFF; /* First byte contains the function code. */ *pucFrameCur++ = MB_FUNC_READ_HOLDING_REGISTER; *usLen += 1; /* Second byte in the response contain the number of bytes. */ *pucFrameCur++ = ( UCHAR ) ( usRegCount * 2 ); *usLen += 1; /* Make callback to fill the buffer. */ eRegStatus = eMBRegHoldingCB( pucFrameCur, usRegAddress, usRegCount, MB_REG_READ ); /* If an error occured convert it into a Modbus exception. */ if( eRegStatus != MB_ENOERR ) { eStatus = prveMBError2Exception( eRegStatus ); } else { *usLen += usRegCount * 2; } } else { eStatus = MB_EX_ILLEGAL_DATA_VALUE; } } else { /* Can't be a valid request because the length is incorrect. */ eStatus = MB_EX_ILLEGAL_DATA_VALUE; } return eStatus;}#endif#if MB_FUNC_READWRITE_HOLDING_ENABLED > 0eMBExceptioneMBFuncReadWriteMultipleHoldingRegister( UCHAR * pucFrame, USHORT * usLen ){ USHORT usRegReadAddress; USHORT usRegReadCount; USHORT usRegWriteAddress; USHORT usRegWriteCount; UCHAR ucRegWriteByteCount; UCHAR *pucFrameCur; eMBException eStatus = MB_EX_NONE; eMBErrorCode eRegStatus; if( *usLen >= ( MB_PDU_FUNC_READWRITE_SIZE_MIN + MB_PDU_SIZE_MIN ) ) { usRegReadAddress = pucFrame[MB_PDU_FUNC_READWRITE_READ_ADDR_OFF] << 8U; usRegReadAddress |= pucFrame[MB_PDU_FUNC_READWRITE_READ_ADDR_OFF + 1]; usRegReadCount = pucFrame[MB_PDU_FUNC_READWRITE_READ_REGCNT_OFF] << 8U; usRegReadCount |= pucFrame[MB_PDU_FUNC_READWRITE_READ_REGCNT_OFF + 1]; usRegWriteAddress = pucFrame[MB_PDU_FUNC_READWRITE_WRITE_ADDR_OFF] << 8U; usRegWriteAddress |= pucFrame[MB_PDU_FUNC_READWRITE_WRITE_ADDR_OFF + 1]; usRegWriteCount = pucFrame[MB_PDU_FUNC_READWRITE_WRITE_REGCNT_OFF] << 8U; usRegWriteCount |= pucFrame[MB_PDU_FUNC_READWRITE_WRITE_REGCNT_OFF + 1]; ucRegWriteByteCount = pucFrame[MB_PDU_FUNC_READWRITE_BYTECNT_OFF]; if( ( usRegReadCount >= 1 ) && ( usRegReadCount <= 0x7D ) && ( usRegWriteCount >= 1 ) && ( usRegWriteCount <= 0x79 ) && ( ( 2 * usRegWriteCount ) == ucRegWriteByteCount ) ) { /* Make callback to update the register values. */ eRegStatus = eMBRegHoldingCB( &pucFrame[MB_PDU_FUNC_READWRITE_WRITE_VALUES_OFF], usRegWriteAddress, usRegWriteCount, MB_REG_WRITE ); if( eRegStatus == MB_ENOERR ) { /* Set the current PDU data pointer to the beginning. */ pucFrameCur = &pucFrame[MB_PDU_FUNC_OFF]; *usLen = MB_PDU_FUNC_OFF; /* First byte contains the function code. */ *pucFrameCur++ = MB_FUNC_READWRITE_MULTIPLE_REGISTERS; *usLen += 1; /* Second byte in the response contain the number of bytes. */ *pucFrameCur++ = ( UCHAR ) ( usRegReadCount * 2 ); *usLen += 1; /* Make the read callback. */ eRegStatus = eMBRegHoldingCB( pucFrameCur, usRegReadAddress, usRegReadCount, MB_REG_READ ); if( eRegStatus == MB_ENOERR ) { *usLen += 2 * usRegReadCount; } } if( eRegStatus != MB_ENOERR ) { eStatus = prveMBError2Exception( eRegStatus ); } } else { eStatus = MB_EX_ILLEGAL_DATA_VALUE; } } return eStatus;}#endif
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -