//OpenSCADA module DAQ.ModBus file: modbus_daq.h /*************************************************************************** * Copyright (C) 2007-2024 by Roman Savochenko, * * * * This program is free software; you can redistribute it and/or modify * * it under the terms of the GNU General Public License as published by * * the Free Software Foundation; version 2 of the License. * * * * This program 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 General Public License for more details. * * * * You should have received a copy of the GNU General Public License * * along with this program; if not, write to the * * Free Software Foundation, Inc., * * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. * ***************************************************************************/ #ifndef MODBUS_DAQ_H #define MODBUS_DAQ_H #include #include #include #include #include #undef _ #define _(mess) mod->I18N(mess).c_str() #undef trS #define trS(mess) mod->I18N(mess,mess_PreSave) using std::string; using std::vector; using std::map; using std::deque; using namespace OSCADA; //************************************************* //* DAQ modul info! * #define DAQ_ID "ModBus" #define DAQ_NAME trS("Client ModBus") #define DAQ_TYPE SDAQ_ID #define DAQ_SUBVER SDAQ_VER #define DAQ_MVER "3.8.2" #define DAQ_AUTHORS trS("Roman Savochenko") #define DAQ_DESCR trS("Provides implementation of the client ModBus service. ModBus/TCP, ModBus/RTU and ModBus/ASCII protocols are supported.") #define DAQ_LICENSE "GPL2" //************************************************* namespace ModBus { //****************************************************** //* TMdPrm * //****************************************************** class TMdContr; class TMdPrm: public TParamContr { public: //Methods TMdPrm( string name, TTypeParam *tp_prm ); ~TMdPrm( ); TCntrNode &operator=( const TCntrNode &node ); bool isStd( ) const; bool isLogic( ) const; void loadDATA( bool incl = false ); void enable( ); void disable( ); void upValStd( ); void upValLog( bool first, bool last, double frq ); TElem *dynElCntr( ) { return &pEl; } TElem &elem( ) { return pEl; } TVariant objFuncCall( const string &id, vector &prms, const string &user_lang ); AutoHD at( const string &nm ) { return TParamContr::at(nm); } TMdContr &owner( ) const; //Attributes MtxString acqErr; protected: //Methods void load_( ); void save_( ); void cntrCmdProc( XMLNode *opt ); //Control interface command process void setType( const string &tpId ); private: //Methods void postEnable( int flag ); void postDisable( int flag ); void vlGet( TVal &vo ); void vlSet( TVal &vo, const TVariant &vl, const TVariant &pvl ); void vlArchMake( TVal &vo ); //Attributes TElem pEl; //Work atribute elements // Logical type by template //Data //*************************************************** //* Logical type parameter's context * class TLogCtx : public TPrmTempl::Impl { public: //Methods TLogCtx( TCntrNode *iobj, const string &name ); //void lnkAdd( int num, const SLnk &l ); bool lnkInit( int num, bool toRecnt = false ); bool lnkActive( int num ); TVariant lnkInput( int num ); bool lnkOutput( int num, const TVariant &vl ); void cleanLnks( bool andFunc = false ); //Attributes bool chkLnkNeed; // Check lnks need flag int idFreq, idStart, idStop, idErr, idSh, idNm, idDscr; //Fixed system attributes identifiers protected: //Methods string lnkHelp( ); }; //Methods void loadIO( ); void saveIO( ); //Attributes TLogCtx *lCtx; }; //****************************************************** //* TMdContr * //****************************************************** class TMdContr: public TController { friend class TMdPrm; public: //Methods TMdContr( string name_c, const string &daq_db, TElem *cfgelem ); ~TMdContr( ); string getStatus( ); int64_t period( ) { return mPer; } string cron( ) { return mSched; } string addr( ) const { return mAddr; } int prior( ) { return mPrior; } AutoHD at( const string &nm ) { return TController::at(nm); } void regVal( int reg, const string &dt = "R", bool separate = false ); //Register value for acquisition TVariant getVal( const string &addr, MtxString &err ); //Unified value request from string address int64_t getValR( int addr, MtxString &err, bool in = false, bool isLE = false ); //Get register value char getValC( int addr, MtxString &err, bool in = false ); //Get coins value bool setVal( const TVariant &val, const string &addr, MtxString &err, bool isGeneric = false ); //Unified value set by string address bool setValR( uint16_t val, int addr, MtxString &err, bool isLE = false ); //Set register value bool setValRs( const map ®s, MtxString &err, bool isLE = false ); //Set multiply registers bool setValC( char val, int addr, MtxString &err ); //Set coins value string modBusReq( string &pdu ); void redntDataUpdate( ); protected: //Methods void postDisable( int flag ); //Delete all DB if flag 1 void disable_( ); void load_( ); void start_( ); void stop_( ); void cntrCmdProc( XMLNode *opt ); //Control interface command process bool cfgChange( TCfg &co, const TVariant &pc ); void prmEn( TMdPrm *prm, bool val ); TVariant objFuncCall( const string &id, vector &prms, const string &user_lang ); bool inWr( const string &addr ); private: //Data class SDataRec { public: SDataRec( int ioff, int v_rez ); int off; //Data block start offset string val; //Data block values kadr MtxString err; //Acquisition error text }; //Methods TParamContr *ParamAttach( const string &name, int type ); static void *Task( void *icntr ); void setCntrDelay( const string &err ); //Attributes ResMtx enRes, aWrRes; ResRW reqRes; int64_t &mPrior, //Process task priority &mNode, //Node &blkMaxSz; //Maximum request block size TCfg &mSched, //Calc schedule &mPrt, //Protocol &mAddr; //Transport device address char &mMerge, //Fragments of register merge &mMltWr, //Use multi-write functions(15,16) &mAsynchWr; //Asynchronous write int64_t &mOmitRdCyclForWr, //Omit cycles for read back of written &reqTm, //Request timeout in milliseconds &restTm, //Restore timeout in seconds &connTry; //Connections try int64_t mPer; bool prcSt, //Process task active callSt, //Calc now stat endrunReq; //Request to stop of the Process task int8_t alSt; //Alarm state vector acqBlks; //Acquisition data blocks for registers vector acqBlksIn; //Acquisition data blocks for input registers vector acqBlksCoil; //Acquisition data blocks for coils vector acqBlksCoilIn; //Acquisition data blocks for input coils map asynchWrs; //Asynchronous writers list map omitRdCycl; //Omit cycles for read back of written float tmDelay; //Delay time for next try connect vector< AutoHD > pHd; uint32_t numRReg, numRRegIn, numRCoil, numRCoilIn, numWReg, numWCoil, numErrCon, numErrResp; }; //************************************************* //* TTpContr * //************************************************* class TTpContr: public TTypeDAQ { public: //Methods TTpContr( string name ); ~TTpContr( ); TElem &prmIOE( ) { return elPrmIO; } //Attributes int8_t tPrmId, tPrmLId; protected: //Methods void load_( ); void save_( ); bool redntAllow( ) { return true; } private: //Methods void postEnable( int flag ); TController *ContrAttach( const string &name, const string &daq_db ); //Attributes TElem elPrmIO; }; extern TTpContr *mod; } //End namespace #endif //MODBUS_DAQ_H