/* * FreeModbus Libary: Win32 Demo Application * Copyright (C) 2006 Christian Walter * * 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 * * File: $Id$ */ /********************************************************** * Linux TCP support. * Based on Walter's project. * Modified by Steven Guo ***********************************************************/ /* ----------------------- Standard C Libs includes --------------------------*/ #include #include #include #include #include #include #include /* ----------------------- Modbus includes ----------------------------------*/ #include "mb.h" #include "mbport.h" /* ----------------------- Defines ------------------------------------------*/ #define PROG "freemodbus" #define REG_INPUT_START 1000 #define REG_INPUT_NREGS 4 #define REG_HOLDING_START 2000 #define REG_HOLDING_NREGS 130 /* ----------------------- Static variables ---------------------------------*/ static USHORT usRegInputStart = REG_INPUT_START; static USHORT usRegInputBuf[REG_INPUT_NREGS]; static USHORT usRegHoldingStart = REG_HOLDING_START; static USHORT usRegHoldingBuf[REG_HOLDING_NREGS]; static pthread_mutex_t xLock = PTHREAD_MUTEX_INITIALIZER; static enum ThreadState { STOPPED, RUNNING, SHUTDOWN } ePollThreadState; /* ----------------------- Static functions ---------------------------------*/ static BOOL bCreatePollingThread( void ); static enum ThreadState eGetPollingThreadState( void ); static void eSetPollingThreadState( enum ThreadState eNewState ); static void* pvPollingThread( void *pvParameter ); /* ----------------------- Start implementation -----------------------------*/ int main( int argc, char *argv[] ) { int iExitCode; CHAR cCh; BOOL bDoExit; if( eMBTCPInit( MB_TCP_PORT_USE_DEFAULT ) != MB_ENOERR ) { fprintf( stderr, "%s: can't initialize modbus stack!\r\n", PROG ); iExitCode = EXIT_FAILURE; } else { eSetPollingThreadState( STOPPED ); /* CLI interface. */ printf( "Type 'q' for quit or 'h' for help!\r\n" ); bDoExit = FALSE; do { printf( "> " ); cCh = getchar( ); switch ( cCh ) { case 'q' : bDoExit = TRUE; break; case 'd' : eSetPollingThreadState( SHUTDOWN ); break; case 'e' : if( bCreatePollingThread( ) != TRUE ) { printf( "Can't start protocol stack! Already running?\r\n" ); } break; case 's' : switch ( eGetPollingThreadState( ) ) { case RUNNING: printf( "Protocol stack is running.\r\n" ); break; case STOPPED: printf( "Protocol stack is stopped.\r\n" ); break; case SHUTDOWN: printf( "Protocol stack is shuting down.\r\n" ); break; } break; case 'h': printf( "FreeModbus demo application help:\r\n" ); printf( " 'd' ... disable protocol stack.\r\n" ); printf( " 'e' ... enabled the protocol stack\r\n" ); printf( " 's' ... show current status\r\n" ); printf( " 'q' ... quit applicationr\r\n" ); printf( " 'h' ... this information\r\n" ); printf( "\r\n" ); printf( "Copyright 2007 Steven Guo \r\n" ); break; default: if( cCh != '\n' ) { printf( "illegal command '%c'!\r\n" , cCh ); } break; } /* eat up everything untill return character. */ while( cCh != '\n' ) { cCh = getchar( ); } } while( !bDoExit ); /* Release hardware resources. */ ( void )eMBClose( ); iExitCode = EXIT_SUCCESS; } return iExitCode; } BOOL bCreatePollingThread( void ) { BOOL bResult; pthread_t xThread; if( eGetPollingThreadState( ) == STOPPED ) { if( pthread_create( &xThread, NULL, pvPollingThread, NULL ) != 0 ) { /* Can't create the polling thread. */ bResult = FALSE; } else { bResult = TRUE; } } else { bResult = FALSE; } return bResult; } void* pvPollingThread( void *pvParameter ) { eSetPollingThreadState( RUNNING ); if( eMBEnable( ) == MB_ENOERR ) { do { if( eMBPoll( ) != MB_ENOERR ) break; } while( eGetPollingThreadState( ) != SHUTDOWN ); } ( void )eMBDisable( ); eSetPollingThreadState( STOPPED ); return 0; } enum ThreadState eGetPollingThreadState( ) { enum ThreadState eCurState; ( void )pthread_mutex_lock( &xLock ); eCurState = ePollThreadState; ( void )pthread_mutex_unlock( &xLock ); return eCurState; } void eSetPollingThreadState( enum ThreadState eNewState ) { ( void )pthread_mutex_lock( &xLock ); ePollThreadState = eNewState; ( void )pthread_mutex_unlock( &xLock ); } eMBErrorCode eMBRegInputCB( UCHAR * pucRegBuffer, USHORT usAddress, USHORT usNRegs ) { eMBErrorCode eStatus = MB_ENOERR; int iRegIndex; if( ( usAddress >= REG_INPUT_START ) && ( usAddress + usNRegs <= REG_INPUT_START + REG_INPUT_NREGS ) ) { iRegIndex = ( int )( usAddress - usRegInputStart ); while( usNRegs > 0 ) { *pucRegBuffer++ = ( unsigned char )( usRegInputBuf[iRegIndex] >> 8 ); *pucRegBuffer++ = ( unsigned char )( usRegInputBuf[iRegIndex] & 0xFF ); iRegIndex++; usNRegs--; } } else { eStatus = MB_ENOREG; } return eStatus; } eMBErrorCode eMBRegHoldingCB( UCHAR * pucRegBuffer, USHORT usAddress, USHORT usNRegs, eMBRegisterMode eMode ) { eMBErrorCode eStatus = MB_ENOERR; int iRegIndex; if( ( usAddress >= REG_HOLDING_START ) && ( usAddress + usNRegs <= REG_HOLDING_START + REG_HOLDING_NREGS ) ) { iRegIndex = ( int )( usAddress - usRegHoldingStart ); switch ( eMode ) { /* Pass current register values to the protocol stack. */ case MB_REG_READ: while( usNRegs > 0 ) { *pucRegBuffer++ = ( UCHAR ) ( usRegHoldingBuf[iRegIndex] >> 8 ); *pucRegBuffer++ = ( UCHAR ) ( usRegHoldingBuf[iRegIndex] & 0xFF ); iRegIndex++; usNRegs--; } break; /* Update current register values with new values from the * protocol stack. */ case MB_REG_WRITE: while( usNRegs > 0 ) { usRegHoldingBuf[iRegIndex] = *pucRegBuffer++ << 8; usRegHoldingBuf[iRegIndex] |= *pucRegBuffer++; iRegIndex++; usNRegs--; } } } else { eStatus = MB_ENOREG; } return eStatus; } eMBErrorCode eMBRegCoilsCB( UCHAR * pucRegBuffer, USHORT usAddress, USHORT usNCoils, eMBRegisterMode eMode ) { return MB_ENOREG; } eMBErrorCode eMBRegDiscreteCB( UCHAR * pucRegBuffer, USHORT usAddress, USHORT usNDiscrete ) { return MB_ENOREG; }