diff --git a/silecs-communication-cpp/src/silecs-communication/interface/core/PLCRecvAction.cpp b/silecs-communication-cpp/src/silecs-communication/interface/core/PLCRecvAction.cpp index b45a4f84c08f5e3696886a5438331952d45fc021..9046bd6dacb71bc70cf36da6b240141ba16c820f 100644 --- a/silecs-communication-cpp/src/silecs-communication/interface/core/PLCRecvAction.cpp +++ b/silecs-communication-cpp/src/silecs-communication/interface/core/PLCRecvAction.cpp @@ -190,6 +190,12 @@ int PLCRecvBlockMode::execute(Context* pContext) for (pDeviceIter = deviceCol.begin(); pDeviceIter != deviceCol.end(); ++pDeviceIter) { Device* pDev = pDeviceIter->second; + + // Only extract if this device actually contains this block, otherwise skip it. + if (!pDev->hasBlock(theBlock_->getName())) + { + continue; + } unsigned long deviceOffset = pDev->getInputAddress(area) * theBlock_->getMemSize(); unsigned char* pBuffer = ((unsigned char*)theBlock_->getBuffer()) + deviceOffset; pDev->importRegisters(theBlock_, pBuffer, tod, pContext); diff --git a/silecs-communication-cpp/src/silecs-communication/interface/equipment/PLCBlock.cpp b/silecs-communication-cpp/src/silecs-communication/interface/equipment/PLCBlock.cpp index f01d53c9d9d5c98c43b1856e89706ece469ce259..11a4b9298a464b9475f3013e33401d432475d677 100644 --- a/silecs-communication-cpp/src/silecs-communication/interface/equipment/PLCBlock.cpp +++ b/silecs-communication-cpp/src/silecs-communication/interface/equipment/PLCBlock.cpp @@ -36,7 +36,8 @@ PLCBlock::PLCBlock(PLC* thePLC, ElementXML blockNode, AccessType accessType) : bufferSize_ = memSize_; //size of one block type (including alignements) if (getPLC()->getProtocolModeID() == BlockMode) { //BlockMode ==> access all devices in once - bufferSize_ *= getPLC()->getDeviceMap().size(); + unsigned int devicesWithBlock = getPLC()->getNrDevicesWithBlock(blockNode.getAttribute("name")); + bufferSize_ *= devicesWithBlock; } pBuffer_ = (unsigned char*)calloc(bufferSize_, sizeof(unsigned char)); } diff --git a/silecs-communication-cpp/src/silecs-communication/interface/equipment/SilecsPLC.cpp b/silecs-communication-cpp/src/silecs-communication/interface/equipment/SilecsPLC.cpp index f30ecf43b7b23780f92f7de9da5642b77d66fb79..317416f943fad51df677dc78a50bca9fa3b7d5dc 100644 --- a/silecs-communication-cpp/src/silecs-communication/interface/equipment/SilecsPLC.cpp +++ b/silecs-communication-cpp/src/silecs-communication/interface/equipment/SilecsPLC.cpp @@ -204,6 +204,21 @@ const std::string PLC::getParamsFileName() return parameterFile_; } +unsigned int PLC::getNrDevicesWithBlock(const std::string& blockName) +{ + unsigned int devicesWithBlock = 0; + deviceVectorType::iterator pDeviceIter; + for (pDeviceIter = deviceCol_.begin(); pDeviceIter != deviceCol_.end(); ++pDeviceIter) + { + Device* pDev = pDeviceIter->second; + if (pDev->hasBlock(blockName)) + { + devicesWithBlock++; + } + } + return devicesWithBlock; +} + Device* PLC::getDevice(std::string deviceName) { deviceVectorType::iterator iter; diff --git a/silecs-communication-cpp/src/silecs-communication/interface/equipment/SilecsPLC.h b/silecs-communication-cpp/src/silecs-communication/interface/equipment/SilecsPLC.h index 67c317c1c93ec967fb59c51b5b95c30720baf200..0386fc32d28a59380dab8da11f37c91591b0b2f0 100644 --- a/silecs-communication-cpp/src/silecs-communication/interface/equipment/SilecsPLC.h +++ b/silecs-communication-cpp/src/silecs-communication/interface/equipment/SilecsPLC.h @@ -440,6 +440,13 @@ public: const std::string getParamsFileName(); + /*! + * \brief Used for detemining the required block buffer size in case of BLOCK_MODE. In BLOCK_MODE a single block contains a specific register for all the devices, + * hence when calculating the required buffer size, we need to only reserve the space for the devices which actually contain this block. + * \param blockName name of the block which should be used to count the devices + */ + unsigned int getNrDevicesWithBlock(const std::string& blockName); + private: friend class Cluster; friend class Device;