2020-08-13 20:53:35 +02:00
|
|
|
#include "DeviceHandlerBase.h"
|
2020-08-27 19:50:02 +02:00
|
|
|
#include "AcceptsDeviceResponsesIF.h"
|
|
|
|
#include "DeviceTmReportingWrapper.h"
|
|
|
|
|
2021-01-08 16:23:57 +01:00
|
|
|
#include "../serviceinterface/ServiceInterface.h"
|
2020-08-13 20:53:35 +02:00
|
|
|
#include "../objectmanager/ObjectManager.h"
|
|
|
|
#include "../storagemanager/StorageManagerIF.h"
|
|
|
|
#include "../thermal/ThermalComponentIF.h"
|
|
|
|
#include "../globalfunctions/CRC.h"
|
2020-10-12 18:18:41 +02:00
|
|
|
#include "../housekeeping/HousekeepingMessage.h"
|
|
|
|
#include "../ipc/MessageQueueMessage.h"
|
2020-08-13 20:53:35 +02:00
|
|
|
#include "../ipc/QueueFactory.h"
|
2020-10-12 18:18:41 +02:00
|
|
|
#include "../subsystem/SubsystemBase.h"
|
2020-12-08 15:59:30 +01:00
|
|
|
#include "../datapoollocal/LocalPoolVariable.h"
|
2018-07-12 16:29:32 +02:00
|
|
|
|
2020-07-16 11:56:48 +02:00
|
|
|
object_id_t DeviceHandlerBase::powerSwitcherId = objects::NO_OBJECT;
|
|
|
|
object_id_t DeviceHandlerBase::rawDataReceiverId = objects::NO_OBJECT;
|
|
|
|
object_id_t DeviceHandlerBase::defaultFdirParentId = objects::NO_OBJECT;
|
2016-06-15 23:48:41 +02:00
|
|
|
|
2020-04-19 15:48:17 +02:00
|
|
|
DeviceHandlerBase::DeviceHandlerBase(object_id_t setObjectId,
|
|
|
|
object_id_t deviceCommunication, CookieIF * comCookie,
|
2020-07-16 11:56:48 +02:00
|
|
|
FailureIsolationBase* fdirInstance, size_t cmdQueueSize) :
|
2020-04-19 15:15:33 +02:00
|
|
|
SystemObject(setObjectId), mode(MODE_OFF), submode(SUBMODE_NONE),
|
2020-04-19 15:57:39 +02:00
|
|
|
wiretappingMode(OFF), storedRawData(StorageManagerIF::INVALID_ADDRESS),
|
2020-05-17 15:28:00 +02:00
|
|
|
deviceCommunicationId(deviceCommunication), comCookie(comCookie),
|
2020-04-19 15:57:39 +02:00
|
|
|
healthHelper(this,setObjectId), modeHelper(this), parameterHelper(this),
|
2021-01-12 20:59:15 +01:00
|
|
|
actionHelper(this, nullptr), poolManager(this, nullptr),
|
2020-10-12 18:18:41 +02:00
|
|
|
childTransitionFailure(RETURN_OK), fdirInstance(fdirInstance),
|
|
|
|
hkSwitcher(this), defaultFDIRUsed(fdirInstance == nullptr),
|
|
|
|
switchOffWasReported(false), childTransitionDelay(5000),
|
|
|
|
transitionSourceMode(_MODE_POWER_DOWN),
|
2020-07-16 11:56:48 +02:00
|
|
|
transitionSourceSubMode(SUBMODE_NONE) {
|
2018-07-12 16:29:32 +02:00
|
|
|
commandQueue = QueueFactory::instance()->createMessageQueue(cmdQueueSize,
|
2020-07-16 11:56:48 +02:00
|
|
|
MessageQueueMessage::MAX_MESSAGE_SIZE);
|
2016-06-15 23:48:41 +02:00
|
|
|
insertInCommandMap(RAW_COMMAND_ID);
|
2020-06-06 20:56:09 +02:00
|
|
|
cookieInfo.state = COOKIE_UNUSED;
|
|
|
|
cookieInfo.pendingCommand = deviceCommandMap.end();
|
2020-05-17 17:46:27 +02:00
|
|
|
if (comCookie == nullptr) {
|
2021-01-12 13:18:18 +01:00
|
|
|
printWarningOrError(sif::OutputTypes::OUT_ERROR, "DeviceHandlerBase",
|
2021-01-08 16:14:11 +01:00
|
|
|
HasReturnvaluesIF::RETURN_FAILED, "Invalid cookie");
|
2020-05-17 17:46:27 +02:00
|
|
|
}
|
2020-04-19 15:15:33 +02:00
|
|
|
if (this->fdirInstance == nullptr) {
|
2018-07-12 16:29:32 +02:00
|
|
|
this->fdirInstance = new DeviceHandlerFailureIsolation(setObjectId,
|
2020-07-16 11:56:48 +02:00
|
|
|
defaultFdirParentId);
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-12 18:18:41 +02:00
|
|
|
void DeviceHandlerBase::setHkDestination(object_id_t hkDestination) {
|
|
|
|
this->hkDestination = hkDestination;
|
|
|
|
}
|
|
|
|
|
2020-07-16 11:56:48 +02:00
|
|
|
void DeviceHandlerBase::setThermalStateRequestPoolIds(
|
2020-12-08 15:59:30 +01:00
|
|
|
lp_id_t thermalStatePoolId, lp_id_t heaterRequestPoolId,
|
|
|
|
uint32_t thermalSetId) {
|
|
|
|
thermalSet = new DeviceHandlerThermalSet(this, thermalSetId,
|
|
|
|
thermalStatePoolId, heaterRequestPoolId);
|
2020-07-16 11:56:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-06-15 23:48:41 +02:00
|
|
|
DeviceHandlerBase::~DeviceHandlerBase() {
|
2020-05-17 15:28:00 +02:00
|
|
|
delete comCookie;
|
2016-06-15 23:48:41 +02:00
|
|
|
if (defaultFDIRUsed) {
|
|
|
|
delete fdirInstance;
|
|
|
|
}
|
2018-07-12 16:29:32 +02:00
|
|
|
QueueFactory::instance()->deleteMessageQueue(commandQueue);
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
|
|
|
|
2018-07-12 16:29:32 +02:00
|
|
|
ReturnValue_t DeviceHandlerBase::performOperation(uint8_t counter) {
|
2016-06-15 23:48:41 +02:00
|
|
|
this->pstStep = counter;
|
2020-10-12 18:28:07 +02:00
|
|
|
this->lastStep = this->pstStep;
|
2016-06-15 23:48:41 +02:00
|
|
|
|
2020-10-12 18:28:07 +02:00
|
|
|
if (getComAction() == CommunicationAction::NOTHING) {
|
|
|
|
return HasReturnvaluesIF::RETURN_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (getComAction() == CommunicationAction::PERFORM_OPERATION) {
|
2016-06-15 23:48:41 +02:00
|
|
|
cookieInfo.state = COOKIE_UNUSED;
|
|
|
|
readCommandQueue();
|
|
|
|
doStateMachine();
|
|
|
|
checkSwitchState();
|
|
|
|
decrementDeviceReplyMap();
|
|
|
|
fdirInstance->checkForFailures();
|
2018-07-12 16:29:32 +02:00
|
|
|
hkSwitcher.performOperation();
|
2020-04-19 22:17:14 +02:00
|
|
|
performOperationHook();
|
2020-10-12 18:28:07 +02:00
|
|
|
return RETURN_OK;
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
2020-10-12 18:28:07 +02:00
|
|
|
|
2016-06-15 23:48:41 +02:00
|
|
|
if (mode == MODE_OFF) {
|
2018-07-12 16:29:32 +02:00
|
|
|
return RETURN_OK;
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
2020-10-12 18:28:07 +02:00
|
|
|
|
2020-04-19 15:16:44 +02:00
|
|
|
switch (getComAction()) {
|
2020-10-12 18:28:07 +02:00
|
|
|
case CommunicationAction::SEND_WRITE:
|
|
|
|
if (cookieInfo.state == COOKIE_UNUSED) {
|
|
|
|
// if no external command was specified, build internal command.
|
2016-06-15 23:48:41 +02:00
|
|
|
buildInternalCommand();
|
|
|
|
}
|
|
|
|
doSendWrite();
|
|
|
|
break;
|
2020-10-12 18:28:07 +02:00
|
|
|
case CommunicationAction::GET_WRITE:
|
2016-06-15 23:48:41 +02:00
|
|
|
doGetWrite();
|
|
|
|
break;
|
2020-10-12 18:28:07 +02:00
|
|
|
case CommunicationAction::SEND_READ:
|
2016-06-15 23:48:41 +02:00
|
|
|
doSendRead();
|
|
|
|
break;
|
2020-10-12 18:28:07 +02:00
|
|
|
case CommunicationAction::GET_READ:
|
2016-06-15 23:48:41 +02:00
|
|
|
doGetRead();
|
2020-12-08 15:59:30 +01:00
|
|
|
// This will be performed after datasets have been updated by the
|
|
|
|
// custom device implementation.
|
2021-01-12 20:59:15 +01:00
|
|
|
poolManager.performHkOperation();
|
2016-06-15 23:48:41 +02:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2018-07-12 16:29:32 +02:00
|
|
|
return RETURN_OK;
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
|
|
|
|
2020-04-19 14:03:47 +02:00
|
|
|
ReturnValue_t DeviceHandlerBase::initialize() {
|
|
|
|
ReturnValue_t result = SystemObject::initialize();
|
|
|
|
if (result != RETURN_OK) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
communicationInterface = objectManager->get<DeviceCommunicationIF>(
|
|
|
|
deviceCommunicationId);
|
2020-07-16 11:56:48 +02:00
|
|
|
if (communicationInterface == nullptr) {
|
2021-01-12 13:18:18 +01:00
|
|
|
printWarningOrError(sif::OutputTypes::OUT_ERROR, "initialize",
|
2021-01-08 16:14:11 +01:00
|
|
|
ObjectManagerIF::CHILD_INIT_FAILED,
|
|
|
|
"Passed communication IF invalid");
|
2020-07-16 11:56:48 +02:00
|
|
|
return ObjectManagerIF::CHILD_INIT_FAILED;
|
2020-04-19 14:03:47 +02:00
|
|
|
}
|
|
|
|
|
2020-04-19 15:48:17 +02:00
|
|
|
result = communicationInterface->initializeInterface(comCookie);
|
2020-04-19 14:03:47 +02:00
|
|
|
if (result != RETURN_OK) {
|
2021-01-12 13:18:18 +01:00
|
|
|
printWarningOrError(sif::OutputTypes::OUT_ERROR, "initialize",
|
2021-01-08 16:14:11 +01:00
|
|
|
ObjectManagerIF::CHILD_INIT_FAILED,
|
|
|
|
"ComIF initialization failed");
|
2020-10-12 18:18:41 +02:00
|
|
|
return result;
|
2020-04-19 14:03:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
IPCStore = objectManager->get<StorageManagerIF>(objects::IPC_STORE);
|
2020-07-16 11:56:48 +02:00
|
|
|
if (IPCStore == nullptr) {
|
2021-01-12 13:18:18 +01:00
|
|
|
printWarningOrError(sif::OutputTypes::OUT_ERROR, "initialize",
|
2021-01-08 16:14:11 +01:00
|
|
|
ObjectManagerIF::CHILD_INIT_FAILED, "IPC Store not set up");
|
2020-07-16 11:56:48 +02:00
|
|
|
return ObjectManagerIF::CHILD_INIT_FAILED;
|
2020-04-19 14:03:47 +02:00
|
|
|
}
|
|
|
|
|
2020-07-16 11:56:48 +02:00
|
|
|
if(rawDataReceiverId != objects::NO_OBJECT) {
|
|
|
|
AcceptsDeviceResponsesIF *rawReceiver = objectManager->get<
|
|
|
|
AcceptsDeviceResponsesIF>(rawDataReceiverId);
|
2020-04-19 14:03:47 +02:00
|
|
|
|
2020-07-16 11:56:48 +02:00
|
|
|
if (rawReceiver == nullptr) {
|
2021-01-12 13:18:18 +01:00
|
|
|
printWarningOrError(sif::OutputTypes::OUT_ERROR,
|
2021-01-08 16:14:11 +01:00
|
|
|
"initialize", ObjectManagerIF::CHILD_INIT_FAILED,
|
|
|
|
"Raw receiver object ID set but no valid object found.");
|
2021-01-03 14:16:52 +01:00
|
|
|
#if FSFW_CPP_OSTREAM_ENABLED == 1
|
2020-07-16 11:56:48 +02:00
|
|
|
sif::error << "Make sure the raw receiver object is set up properly"
|
|
|
|
" and implements AcceptsDeviceResponsesIF" << std::endl;
|
2021-01-08 16:14:11 +01:00
|
|
|
#else
|
2021-01-12 13:18:18 +01:00
|
|
|
sif::printError("Make sure the raw receiver object is set up "
|
2021-01-08 16:14:11 +01:00
|
|
|
"properly and implements AcceptsDeviceResponsesIF\n");
|
2021-01-03 13:58:18 +01:00
|
|
|
#endif
|
2020-07-16 11:56:48 +02:00
|
|
|
return ObjectManagerIF::CHILD_INIT_FAILED;
|
|
|
|
}
|
|
|
|
defaultRawReceiver = rawReceiver->getDeviceQueue();
|
2020-04-19 14:03:47 +02:00
|
|
|
}
|
|
|
|
|
2020-07-16 11:56:48 +02:00
|
|
|
if(powerSwitcherId != objects::NO_OBJECT) {
|
|
|
|
powerSwitcher = objectManager->get<PowerSwitchIF>(powerSwitcherId);
|
|
|
|
if (powerSwitcher == nullptr) {
|
2021-01-12 13:18:18 +01:00
|
|
|
printWarningOrError(sif::OutputTypes::OUT_ERROR,
|
2021-01-08 16:14:11 +01:00
|
|
|
"initialize", ObjectManagerIF::CHILD_INIT_FAILED,
|
|
|
|
"Power switcher set but no valid object found.");
|
2021-01-03 14:16:52 +01:00
|
|
|
#if FSFW_CPP_OSTREAM_ENABLED == 1
|
2021-01-08 16:14:11 +01:00
|
|
|
sif::error << "Make sure the power switcher object is set up "
|
|
|
|
<< "properly and implements PowerSwitchIF" << std::endl;
|
|
|
|
#else
|
2021-01-12 13:18:18 +01:00
|
|
|
sif::printError("Make sure the power switcher object is set up "
|
2021-01-08 16:14:11 +01:00
|
|
|
"properly and implements PowerSwitchIF\n");
|
2021-01-03 13:58:18 +01:00
|
|
|
#endif
|
2020-07-16 11:56:48 +02:00
|
|
|
return ObjectManagerIF::CHILD_INIT_FAILED;
|
|
|
|
}
|
2020-04-19 14:03:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
result = healthHelper.initialize();
|
2020-07-16 12:54:30 +02:00
|
|
|
if (result != RETURN_OK) {
|
|
|
|
return result;
|
2020-04-19 14:03:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
result = modeHelper.initialize();
|
|
|
|
if (result != RETURN_OK) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
result = actionHelper.initialize(commandQueue);
|
|
|
|
if (result != RETURN_OK) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
result = fdirInstance->initialize();
|
|
|
|
if (result != HasReturnvaluesIF::RETURN_OK) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
result = parameterHelper.initialize();
|
|
|
|
if (result != HasReturnvaluesIF::RETURN_OK) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
result = hkSwitcher.initialize();
|
|
|
|
if (result != HasReturnvaluesIF::RETURN_OK) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2021-01-12 20:59:15 +01:00
|
|
|
result = poolManager.initialize(commandQueue);
|
2020-10-12 18:18:41 +02:00
|
|
|
if (result != HasReturnvaluesIF::RETURN_OK) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2020-04-19 14:03:47 +02:00
|
|
|
fillCommandAndReplyMap();
|
|
|
|
|
2020-12-08 15:59:30 +01:00
|
|
|
if(thermalSet != nullptr) {
|
|
|
|
//Set temperature target state to NON_OP.
|
|
|
|
result = thermalSet->read();
|
|
|
|
if(result == HasReturnvaluesIF::RETURN_OK) {
|
|
|
|
thermalSet->heaterRequest.value =
|
|
|
|
ThermalComponentIF::STATE_REQUEST_NON_OPERATIONAL;
|
2021-01-06 21:14:23 +01:00
|
|
|
thermalSet->heaterRequest.setValid(true);
|
|
|
|
thermalSet->commit();
|
2020-12-08 15:59:30 +01:00
|
|
|
}
|
2020-04-19 14:03:47 +02:00
|
|
|
|
2020-12-08 15:59:30 +01:00
|
|
|
}
|
2020-04-19 14:03:47 +02:00
|
|
|
|
2020-12-08 15:59:30 +01:00
|
|
|
return RETURN_OK;
|
2020-04-19 14:03:47 +02:00
|
|
|
}
|
|
|
|
|
2016-06-15 23:48:41 +02:00
|
|
|
void DeviceHandlerBase::decrementDeviceReplyMap() {
|
|
|
|
for (std::map<DeviceCommandId_t, DeviceReplyInfo>::iterator iter =
|
|
|
|
deviceReplyMap.begin(); iter != deviceReplyMap.end(); iter++) {
|
|
|
|
if (iter->second.delayCycles != 0) {
|
|
|
|
iter->second.delayCycles--;
|
|
|
|
if (iter->second.delayCycles == 0) {
|
2020-06-19 01:05:51 +02:00
|
|
|
if (iter->second.periodic) {
|
2016-06-15 23:48:41 +02:00
|
|
|
iter->second.delayCycles = iter->second.maxDelayCycles;
|
|
|
|
}
|
|
|
|
replyToReply(iter, TIMEOUT);
|
|
|
|
missedReply(iter->first);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::readCommandQueue() {
|
|
|
|
|
|
|
|
if (dontCheckQueue()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-07-16 11:56:48 +02:00
|
|
|
CommandMessage command;
|
|
|
|
ReturnValue_t result = commandQueue->receiveMessage(&command);
|
2016-06-15 23:48:41 +02:00
|
|
|
if (result != RETURN_OK) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-07-16 12:54:30 +02:00
|
|
|
result = healthHelper.handleHealthCommand(&command);
|
|
|
|
if (result == RETURN_OK) {
|
|
|
|
return;
|
2020-07-16 11:56:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
result = modeHelper.handleModeCommand(&command);
|
2016-06-15 23:48:41 +02:00
|
|
|
if (result == RETURN_OK) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-07-16 11:56:48 +02:00
|
|
|
result = actionHelper.handleActionMessage(&command);
|
2016-06-15 23:48:41 +02:00
|
|
|
if (result == RETURN_OK) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-07-16 11:56:48 +02:00
|
|
|
result = parameterHelper.handleParameterMessage(&command);
|
2016-06-15 23:48:41 +02:00
|
|
|
if (result == RETURN_OK) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-01-12 20:59:15 +01:00
|
|
|
result = poolManager.handleHousekeepingMessage(&command);
|
2020-10-12 18:18:41 +02:00
|
|
|
if (result == RETURN_OK) {
|
|
|
|
return;
|
|
|
|
}
|
2016-06-15 23:48:41 +02:00
|
|
|
|
2020-07-16 11:56:48 +02:00
|
|
|
result = handleDeviceHandlerMessage(&command);
|
2016-06-15 23:48:41 +02:00
|
|
|
if (result == RETURN_OK) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-07-16 11:56:48 +02:00
|
|
|
result = letChildHandleMessage(&command);
|
2016-06-15 23:48:41 +02:00
|
|
|
if (result == RETURN_OK) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-07-16 11:56:48 +02:00
|
|
|
replyReturnvalueToCommand(CommandMessage::UNKNOWN_COMMAND);
|
2016-06-15 23:48:41 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::doStateMachine() {
|
|
|
|
switch (mode) {
|
|
|
|
case _MODE_START_UP:
|
|
|
|
case _MODE_SHUT_DOWN:
|
|
|
|
case _MODE_TO_NORMAL:
|
|
|
|
case _MODE_TO_ON:
|
|
|
|
case _MODE_TO_RAW: {
|
|
|
|
Mode_t currentMode = mode;
|
|
|
|
callChildStatemachine();
|
|
|
|
//Only do timeout if child did not change anything
|
|
|
|
if (mode != currentMode) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
uint32_t currentUptime;
|
2018-07-12 16:29:32 +02:00
|
|
|
Clock::getUptime(¤tUptime);
|
2016-06-15 23:48:41 +02:00
|
|
|
if (currentUptime - timeoutStart >= childTransitionDelay) {
|
|
|
|
triggerEvent(MODE_TRANSITION_FAILED, childTransitionFailure, 0);
|
|
|
|
setMode(transitionSourceMode, transitionSourceSubMode);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case _MODE_POWER_DOWN:
|
|
|
|
commandSwitch(PowerSwitchIF::SWITCH_OFF);
|
|
|
|
setMode(_MODE_WAIT_OFF);
|
|
|
|
break;
|
|
|
|
case _MODE_POWER_ON:
|
|
|
|
commandSwitch(PowerSwitchIF::SWITCH_ON);
|
|
|
|
setMode(_MODE_WAIT_ON);
|
|
|
|
break;
|
|
|
|
case _MODE_WAIT_ON: {
|
|
|
|
uint32_t currentUptime;
|
2018-07-12 16:29:32 +02:00
|
|
|
Clock::getUptime(¤tUptime);
|
2020-07-16 11:56:48 +02:00
|
|
|
if (powerSwitcher != nullptr and currentUptime - timeoutStart >=
|
|
|
|
powerSwitcher->getSwitchDelayMs()) {
|
2016-06-15 23:48:41 +02:00
|
|
|
triggerEvent(MODE_TRANSITION_FAILED, PowerSwitchIF::SWITCH_TIMEOUT,
|
|
|
|
0);
|
|
|
|
setMode(_MODE_POWER_DOWN);
|
|
|
|
callChildStatemachine();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
ReturnValue_t switchState = getStateOfSwitches();
|
|
|
|
if ((switchState == PowerSwitchIF::SWITCH_ON)
|
|
|
|
|| (switchState == NO_SWITCH)) {
|
|
|
|
//NOTE: TransitionSourceMode and -SubMode are set by handleCommandedModeTransition
|
|
|
|
childTransitionFailure = CHILD_TIMEOUT;
|
|
|
|
setMode(_MODE_START_UP);
|
|
|
|
callChildStatemachine();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case _MODE_WAIT_OFF: {
|
|
|
|
uint32_t currentUptime;
|
2018-07-12 16:29:32 +02:00
|
|
|
Clock::getUptime(¤tUptime);
|
2020-07-16 11:56:48 +02:00
|
|
|
|
|
|
|
if(powerSwitcher == nullptr) {
|
|
|
|
setMode(MODE_OFF);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2016-06-15 23:48:41 +02:00
|
|
|
if (currentUptime - timeoutStart >= powerSwitcher->getSwitchDelayMs()) {
|
|
|
|
triggerEvent(MODE_TRANSITION_FAILED, PowerSwitchIF::SWITCH_TIMEOUT,
|
|
|
|
0);
|
|
|
|
setMode(MODE_ERROR_ON);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
ReturnValue_t switchState = getStateOfSwitches();
|
|
|
|
if ((switchState == PowerSwitchIF::SWITCH_OFF)
|
|
|
|
|| (switchState == NO_SWITCH)) {
|
|
|
|
setMode(_MODE_SWITCH_IS_OFF);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case MODE_OFF:
|
|
|
|
doOffActivity();
|
|
|
|
break;
|
|
|
|
case MODE_ON:
|
|
|
|
doOnActivity();
|
|
|
|
break;
|
|
|
|
case MODE_RAW:
|
|
|
|
case MODE_NORMAL:
|
|
|
|
case MODE_ERROR_ON:
|
|
|
|
break;
|
|
|
|
case _MODE_SWITCH_IS_OFF:
|
|
|
|
setMode(MODE_OFF, SUBMODE_NONE);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
triggerEvent(OBJECT_IN_INVALID_MODE, mode, submode);
|
|
|
|
setMode(_MODE_POWER_DOWN, 0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ReturnValue_t DeviceHandlerBase::isModeCombinationValid(Mode_t mode,
|
|
|
|
Submode_t submode) {
|
|
|
|
switch (mode) {
|
|
|
|
case MODE_OFF:
|
|
|
|
case MODE_ON:
|
|
|
|
case MODE_NORMAL:
|
|
|
|
case MODE_RAW:
|
|
|
|
if (submode == SUBMODE_NONE) {
|
|
|
|
return RETURN_OK;
|
|
|
|
} else {
|
|
|
|
return INVALID_SUBMODE;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return HasModesIF::INVALID_MODE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-16 11:56:48 +02:00
|
|
|
ReturnValue_t DeviceHandlerBase::insertInCommandAndReplyMap(
|
|
|
|
DeviceCommandId_t deviceCommand, uint16_t maxDelayCycles,
|
2020-10-12 18:18:41 +02:00
|
|
|
LocalPoolDataSetBase* replyDataSet, size_t replyLen, bool periodic,
|
|
|
|
bool hasDifferentReplyId, DeviceCommandId_t replyId) {
|
2020-04-19 15:01:27 +02:00
|
|
|
//No need to check, as we may try to insert multiple times.
|
2016-06-15 23:48:41 +02:00
|
|
|
insertInCommandMap(deviceCommand);
|
|
|
|
if (hasDifferentReplyId) {
|
2020-10-12 18:18:41 +02:00
|
|
|
return insertInReplyMap(replyId, maxDelayCycles,
|
|
|
|
replyDataSet, replyLen, periodic);
|
2016-06-15 23:48:41 +02:00
|
|
|
} else {
|
2020-10-12 18:18:41 +02:00
|
|
|
return insertInReplyMap(deviceCommand, maxDelayCycles,
|
|
|
|
replyDataSet, replyLen, periodic);
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ReturnValue_t DeviceHandlerBase::insertInReplyMap(DeviceCommandId_t replyId,
|
2020-10-12 18:18:41 +02:00
|
|
|
uint16_t maxDelayCycles, LocalPoolDataSetBase* dataSet,
|
|
|
|
size_t replyLen, bool periodic) {
|
2016-06-15 23:48:41 +02:00
|
|
|
DeviceReplyInfo info;
|
|
|
|
info.maxDelayCycles = maxDelayCycles;
|
|
|
|
info.periodic = periodic;
|
|
|
|
info.delayCycles = 0;
|
2020-04-19 15:01:27 +02:00
|
|
|
info.replyLen = replyLen;
|
2020-10-12 18:18:41 +02:00
|
|
|
info.dataSet = dataSet;
|
2016-06-15 23:48:41 +02:00
|
|
|
info.command = deviceCommandMap.end();
|
2020-05-17 15:47:17 +02:00
|
|
|
auto resultPair = deviceReplyMap.emplace(replyId, info);
|
|
|
|
if (resultPair.second) {
|
2016-06-15 23:48:41 +02:00
|
|
|
return RETURN_OK;
|
|
|
|
} else {
|
|
|
|
return RETURN_FAILED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-07-16 11:56:48 +02:00
|
|
|
ReturnValue_t DeviceHandlerBase::insertInCommandMap(
|
|
|
|
DeviceCommandId_t deviceCommand) {
|
2016-06-15 23:48:41 +02:00
|
|
|
DeviceCommandInfo info;
|
|
|
|
info.expectedReplies = 0;
|
|
|
|
info.isExecuting = false;
|
|
|
|
info.sendReplyTo = NO_COMMANDER;
|
2020-05-17 15:47:17 +02:00
|
|
|
auto resultPair = deviceCommandMap.emplace(deviceCommand, info);
|
|
|
|
if (resultPair.second) {
|
2016-06-15 23:48:41 +02:00
|
|
|
return RETURN_OK;
|
|
|
|
} else {
|
|
|
|
return RETURN_FAILED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-19 15:01:27 +02:00
|
|
|
ReturnValue_t DeviceHandlerBase::updateReplyMapEntry(DeviceCommandId_t deviceReply,
|
2020-06-19 01:05:51 +02:00
|
|
|
uint16_t delayCycles, uint16_t maxDelayCycles, bool periodic) {
|
2020-10-12 18:18:41 +02:00
|
|
|
auto replyIter = deviceReplyMap.find(deviceReply);
|
|
|
|
if (replyIter == deviceReplyMap.end()) {
|
2016-06-15 23:48:41 +02:00
|
|
|
triggerEvent(INVALID_DEVICE_COMMAND, deviceReply);
|
|
|
|
return RETURN_FAILED;
|
|
|
|
} else {
|
2020-10-12 18:18:41 +02:00
|
|
|
DeviceReplyInfo *info = &(replyIter->second);
|
2016-06-15 23:48:41 +02:00
|
|
|
if (maxDelayCycles != 0) {
|
|
|
|
info->maxDelayCycles = maxDelayCycles;
|
|
|
|
}
|
|
|
|
info->delayCycles = delayCycles;
|
|
|
|
info->periodic = periodic;
|
|
|
|
return RETURN_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-12 18:18:41 +02:00
|
|
|
|
|
|
|
ReturnValue_t DeviceHandlerBase::setReplyDataset(DeviceCommandId_t replyId,
|
|
|
|
LocalPoolDataSetBase *dataSet) {
|
|
|
|
auto replyIter = deviceReplyMap.find(replyId);
|
|
|
|
if(replyIter == deviceReplyMap.end()) {
|
|
|
|
return HasReturnvaluesIF::RETURN_FAILED;
|
|
|
|
}
|
|
|
|
replyIter->second.dataSet = dataSet;
|
|
|
|
return HasReturnvaluesIF::RETURN_OK;
|
|
|
|
}
|
|
|
|
|
2016-06-15 23:48:41 +02:00
|
|
|
void DeviceHandlerBase::callChildStatemachine() {
|
|
|
|
if (mode == _MODE_START_UP) {
|
|
|
|
doStartUp();
|
|
|
|
} else if (mode == _MODE_SHUT_DOWN) {
|
|
|
|
doShutDown();
|
|
|
|
} else if (mode & TRANSITION_MODE_CHILD_ACTION_MASK) {
|
|
|
|
doTransition(transitionSourceMode, transitionSourceSubMode);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::setTransition(Mode_t modeTo, Submode_t submodeTo) {
|
|
|
|
triggerEvent(CHANGING_MODE, modeTo, submodeTo);
|
|
|
|
childTransitionDelay = getTransitionDelayMs(mode, modeTo);
|
|
|
|
transitionSourceMode = mode;
|
|
|
|
transitionSourceSubMode = submode;
|
|
|
|
childTransitionFailure = CHILD_TIMEOUT;
|
|
|
|
|
2020-07-16 11:56:48 +02:00
|
|
|
// transitionTargetMode is set by setMode
|
2016-06-15 23:48:41 +02:00
|
|
|
setMode((modeTo | TRANSITION_MODE_CHILD_ACTION_MASK), submodeTo);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::setMode(Mode_t newMode, uint8_t newSubmode) {
|
2018-07-12 16:29:32 +02:00
|
|
|
changeHK(mode, submode, false);
|
2016-06-15 23:48:41 +02:00
|
|
|
submode = newSubmode;
|
|
|
|
mode = newMode;
|
|
|
|
modeChanged();
|
|
|
|
setNormalDatapoolEntriesInvalid();
|
|
|
|
if (!isTransitionalMode()) {
|
|
|
|
modeHelper.modeChanged(newMode, newSubmode);
|
|
|
|
announceMode(false);
|
|
|
|
}
|
2018-07-12 16:29:32 +02:00
|
|
|
Clock::getUptime(&timeoutStart);
|
2016-06-15 23:48:41 +02:00
|
|
|
|
2020-12-08 15:59:30 +01:00
|
|
|
if (mode == MODE_OFF and thermalSet != nullptr) {
|
|
|
|
ReturnValue_t result = thermalSet->read();
|
|
|
|
if(result == HasReturnvaluesIF::RETURN_OK) {
|
|
|
|
if (thermalSet->heaterRequest.value !=
|
|
|
|
ThermalComponentIF::STATE_REQUEST_IGNORE) {
|
|
|
|
thermalSet->heaterRequest.value = ThermalComponentIF::
|
|
|
|
STATE_REQUEST_NON_OPERATIONAL;
|
|
|
|
}
|
|
|
|
thermalSet->heaterRequest.commit(PoolVariableIF::VALID);
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
2020-12-08 15:59:30 +01:00
|
|
|
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
2018-07-12 16:29:32 +02:00
|
|
|
changeHK(mode, submode, true);
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::setMode(Mode_t newMode) {
|
|
|
|
setMode(newMode, submode);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::replyReturnvalueToCommand(ReturnValue_t status,
|
|
|
|
uint32_t parameter) {
|
2019-08-28 14:50:24 +02:00
|
|
|
//This is actually the reply protocol for raw and misc DH commands.
|
2016-06-15 23:48:41 +02:00
|
|
|
if (status == RETURN_OK) {
|
|
|
|
CommandMessage reply(CommandMessage::REPLY_COMMAND_OK, 0, parameter);
|
2018-07-12 16:29:32 +02:00
|
|
|
commandQueue->reply(&reply);
|
2016-06-15 23:48:41 +02:00
|
|
|
} else {
|
|
|
|
CommandMessage reply(CommandMessage::REPLY_REJECTED, status, parameter);
|
2018-07-12 16:29:32 +02:00
|
|
|
commandQueue->reply(&reply);
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::replyToCommand(ReturnValue_t status,
|
|
|
|
uint32_t parameter) {
|
2021-01-08 16:14:11 +01:00
|
|
|
// Check if we reply to a raw command.
|
2016-06-15 23:48:41 +02:00
|
|
|
if (cookieInfo.pendingCommand->first == RAW_COMMAND_ID) {
|
|
|
|
if (status == NO_REPLY_EXPECTED) {
|
|
|
|
status = RETURN_OK;
|
|
|
|
}
|
|
|
|
replyReturnvalueToCommand(status, parameter);
|
2021-01-08 16:14:11 +01:00
|
|
|
// Always delete data from a raw command.
|
2016-06-15 23:48:41 +02:00
|
|
|
IPCStore->deleteData(storedRawData);
|
|
|
|
return;
|
|
|
|
}
|
2021-01-08 16:14:11 +01:00
|
|
|
// Check if we were externally commanded.
|
2016-06-15 23:48:41 +02:00
|
|
|
if (cookieInfo.pendingCommand->second.sendReplyTo != NO_COMMANDER) {
|
|
|
|
MessageQueueId_t queueId = cookieInfo.pendingCommand->second.sendReplyTo;
|
|
|
|
if (status == NO_REPLY_EXPECTED) {
|
|
|
|
actionHelper.finish(queueId, cookieInfo.pendingCommand->first,
|
|
|
|
RETURN_OK);
|
|
|
|
} else {
|
|
|
|
actionHelper.step(1, queueId, cookieInfo.pendingCommand->first,
|
|
|
|
status);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::replyToReply(DeviceReplyMap::iterator iter,
|
|
|
|
ReturnValue_t status) {
|
2021-01-08 16:14:11 +01:00
|
|
|
// No need to check if iter exists, as this is checked by callers.
|
|
|
|
// If someone else uses the method, add check.
|
2016-06-15 23:48:41 +02:00
|
|
|
if (iter->second.command == deviceCommandMap.end()) {
|
|
|
|
//Is most likely periodic reply. Silent return.
|
|
|
|
return;
|
|
|
|
}
|
2021-01-08 16:14:11 +01:00
|
|
|
// Check if more replies are expected. If so, do nothing.
|
2016-06-15 23:48:41 +02:00
|
|
|
DeviceCommandInfo* info = &(iter->second.command->second);
|
|
|
|
if (--info->expectedReplies == 0) {
|
2021-01-08 16:14:11 +01:00
|
|
|
// Check if it was transition or internal command.
|
|
|
|
// Don't send any replies in that case.
|
2016-06-15 23:48:41 +02:00
|
|
|
if (info->sendReplyTo != NO_COMMANDER) {
|
|
|
|
actionHelper.finish(info->sendReplyTo, iter->first, status);
|
|
|
|
}
|
|
|
|
info->isExecuting = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::doSendWrite() {
|
|
|
|
if (cookieInfo.state == COOKIE_WRITE_READY) {
|
|
|
|
|
2020-04-19 15:48:17 +02:00
|
|
|
ReturnValue_t result = communicationInterface->sendMessage(comCookie,
|
2016-06-15 23:48:41 +02:00
|
|
|
rawPacket, rawPacketLen);
|
|
|
|
|
|
|
|
if (result == RETURN_OK) {
|
|
|
|
cookieInfo.state = COOKIE_WRITE_SENT;
|
|
|
|
} else {
|
2021-01-08 16:14:11 +01:00
|
|
|
// always generate a failure event, so that FDIR knows what's up
|
2016-06-15 23:48:41 +02:00
|
|
|
triggerEvent(DEVICE_SENDING_COMMAND_FAILED, result,
|
|
|
|
cookieInfo.pendingCommand->first);
|
|
|
|
replyToCommand(result);
|
|
|
|
cookieInfo.state = COOKIE_UNUSED;
|
|
|
|
cookieInfo.pendingCommand->second.isExecuting = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::doGetWrite() {
|
|
|
|
if (cookieInfo.state != COOKIE_WRITE_SENT) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
cookieInfo.state = COOKIE_UNUSED;
|
2020-04-19 15:48:17 +02:00
|
|
|
ReturnValue_t result = communicationInterface->getSendSuccess(comCookie);
|
2016-06-15 23:48:41 +02:00
|
|
|
if (result == RETURN_OK) {
|
|
|
|
if (wiretappingMode == RAW) {
|
2018-07-12 16:29:32 +02:00
|
|
|
replyRawData(rawPacket, rawPacketLen, requestedRawTraffic, true);
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
2020-04-19 14:03:47 +02:00
|
|
|
|
|
|
|
//We need to distinguish here, because a raw command never expects a reply.
|
|
|
|
//(Could be done in eRIRM, but then child implementations need to be careful.
|
2016-06-15 23:48:41 +02:00
|
|
|
result = enableReplyInReplyMap(cookieInfo.pendingCommand);
|
|
|
|
} else {
|
|
|
|
//always generate a failure event, so that FDIR knows what's up
|
|
|
|
triggerEvent(DEVICE_SENDING_COMMAND_FAILED, result,
|
|
|
|
cookieInfo.pendingCommand->first);
|
|
|
|
}
|
|
|
|
if (result != RETURN_OK) {
|
|
|
|
cookieInfo.pendingCommand->second.isExecuting = false;
|
|
|
|
}
|
|
|
|
replyToCommand(result);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::doSendRead() {
|
|
|
|
ReturnValue_t result;
|
|
|
|
|
2020-04-19 15:01:27 +02:00
|
|
|
size_t requestLen = 0;
|
2020-06-06 20:56:09 +02:00
|
|
|
if(cookieInfo.pendingCommand != deviceCommandMap.end()) {
|
|
|
|
DeviceReplyIter iter = deviceReplyMap.find(
|
|
|
|
cookieInfo.pendingCommand->first);
|
|
|
|
if(iter != deviceReplyMap.end()) {
|
|
|
|
requestLen = iter->second.replyLen;
|
|
|
|
}
|
2020-04-19 15:01:27 +02:00
|
|
|
}
|
|
|
|
|
2020-04-19 15:48:17 +02:00
|
|
|
result = communicationInterface->requestReceiveMessage(comCookie, requestLen);
|
2020-04-19 15:01:27 +02:00
|
|
|
|
2016-06-15 23:48:41 +02:00
|
|
|
if (result == RETURN_OK) {
|
|
|
|
cookieInfo.state = COOKIE_READ_SENT;
|
|
|
|
} else {
|
|
|
|
triggerEvent(DEVICE_REQUESTING_REPLY_FAILED, result);
|
|
|
|
//We can't inform anyone, because we don't know which command was sent last.
|
|
|
|
//So, we need to wait for a timeout.
|
|
|
|
//but I think we can allow to ignore one missedReply.
|
|
|
|
ignoreMissedRepliesCount++;
|
|
|
|
cookieInfo.state = COOKIE_UNUSED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::doGetRead() {
|
2020-07-16 11:56:48 +02:00
|
|
|
size_t receivedDataLen = 0;
|
|
|
|
uint8_t *receivedData = nullptr;
|
2016-06-15 23:48:41 +02:00
|
|
|
|
|
|
|
if (cookieInfo.state != COOKIE_READ_SENT) {
|
|
|
|
cookieInfo.state = COOKIE_UNUSED;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
cookieInfo.state = COOKIE_UNUSED;
|
|
|
|
|
2020-07-16 11:56:48 +02:00
|
|
|
ReturnValue_t result = communicationInterface->readReceivedMessage(
|
|
|
|
comCookie, &receivedData, &receivedDataLen);
|
2016-06-15 23:48:41 +02:00
|
|
|
|
|
|
|
if (result != RETURN_OK) {
|
|
|
|
triggerEvent(DEVICE_REQUESTING_REPLY_FAILED, result);
|
|
|
|
//I think we can allow to ignore one missedReply.
|
|
|
|
ignoreMissedRepliesCount++;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-04-19 14:52:27 +02:00
|
|
|
if (receivedDataLen == 0 or result == DeviceCommunicationIF::NO_REPLY_RECEIVED)
|
2016-06-15 23:48:41 +02:00
|
|
|
return;
|
|
|
|
|
|
|
|
if (wiretappingMode == RAW) {
|
2018-07-12 16:29:32 +02:00
|
|
|
replyRawData(receivedData, receivedDataLen, requestedRawTraffic);
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
|
|
|
|
2020-12-30 22:34:40 +01:00
|
|
|
if (mode == MODE_RAW) {
|
|
|
|
if (defaultRawReceiver != MessageQueueIF::NO_QUEUE) {
|
|
|
|
replyRawReplyIfnotWiretapped(receivedData, receivedDataLen);
|
|
|
|
}
|
2020-07-16 11:56:48 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
parseReply(receivedData, receivedDataLen);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::parseReply(const uint8_t* receivedData,
|
|
|
|
size_t receivedDataLen) {
|
|
|
|
ReturnValue_t result = HasReturnvaluesIF::RETURN_FAILED;
|
2020-12-23 20:17:10 +01:00
|
|
|
DeviceCommandId_t foundId = DeviceHandlerIF::NO_COMMAND_ID;
|
2020-07-16 11:56:48 +02:00
|
|
|
size_t foundLen = 0;
|
|
|
|
// The loop may not execute more often than the number of received bytes
|
|
|
|
// (worst case). This approach avoids infinite loops due to buggy
|
|
|
|
// scanForReply routines.
|
|
|
|
uint32_t remainingLength = receivedDataLen;
|
|
|
|
for (uint32_t count = 0; count < receivedDataLen; count++) {
|
|
|
|
result = scanForReply(receivedData, remainingLength, &foundId,
|
|
|
|
&foundLen);
|
|
|
|
switch (result) {
|
|
|
|
case RETURN_OK:
|
|
|
|
handleReply(receivedData, foundId, foundLen);
|
2020-10-12 18:18:41 +02:00
|
|
|
if(foundLen == 0) {
|
2021-01-12 13:18:18 +01:00
|
|
|
printWarningOrError(sif::OutputTypes::OUT_WARNING,
|
2021-01-08 16:23:57 +01:00
|
|
|
"parseReply", ObjectManagerIF::CHILD_INIT_FAILED,
|
|
|
|
"Found length is one, parsing might be stuck");
|
2020-10-12 18:18:41 +02:00
|
|
|
}
|
2020-07-16 11:56:48 +02:00
|
|
|
break;
|
|
|
|
case APERIODIC_REPLY: {
|
|
|
|
result = interpretDeviceReply(foundId, receivedData);
|
|
|
|
if (result != RETURN_OK) {
|
2020-10-12 18:18:41 +02:00
|
|
|
replyRawReplyIfnotWiretapped(receivedData, foundLen);
|
|
|
|
triggerEvent(DEVICE_INTERPRETING_REPLY_FAILED, result,
|
|
|
|
foundId);
|
|
|
|
}
|
|
|
|
if(foundLen == 0) {
|
2021-01-12 13:18:18 +01:00
|
|
|
printWarningOrError(sif::OutputTypes::OUT_ERROR,
|
2021-01-08 16:14:11 +01:00
|
|
|
"parseReply", ObjectManagerIF::CHILD_INIT_FAILED,
|
|
|
|
"Power switcher set but no valid object found.");
|
2021-01-03 14:16:52 +01:00
|
|
|
#if FSFW_CPP_OSTREAM_ENABLED == 1
|
2020-10-12 18:18:41 +02:00
|
|
|
sif::warning << "DeviceHandlerBase::parseReply: foundLen is 0!"
|
|
|
|
" Packet parsing will be stuck." << std::endl;
|
2021-01-03 13:58:18 +01:00
|
|
|
#endif
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
2020-10-12 18:18:41 +02:00
|
|
|
break;
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
2020-07-16 11:56:48 +02:00
|
|
|
case IGNORE_REPLY_DATA:
|
2020-10-12 18:18:41 +02:00
|
|
|
continue;
|
2020-07-16 11:56:48 +02:00
|
|
|
case IGNORE_FULL_PACKET:
|
|
|
|
return;
|
|
|
|
default:
|
2021-01-08 16:18:08 +01:00
|
|
|
// We need to wait for timeout.. don't know what command failed
|
|
|
|
// and who sent it.
|
2020-07-16 11:56:48 +02:00
|
|
|
replyRawReplyIfnotWiretapped(receivedData, foundLen);
|
|
|
|
triggerEvent(DEVICE_READING_REPLY_FAILED, result, foundLen);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
receivedData += foundLen;
|
|
|
|
if (remainingLength > foundLen) {
|
|
|
|
remainingLength -= foundLen;
|
|
|
|
} else {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-06-15 23:48:41 +02:00
|
|
|
|
2020-07-16 11:56:48 +02:00
|
|
|
void DeviceHandlerBase::handleReply(const uint8_t* receivedData,
|
|
|
|
DeviceCommandId_t foundId, uint32_t foundLen) {
|
|
|
|
ReturnValue_t result;
|
|
|
|
DeviceReplyMap::iterator iter = deviceReplyMap.find(foundId);
|
|
|
|
|
|
|
|
if (iter == deviceReplyMap.end()) {
|
|
|
|
replyRawReplyIfnotWiretapped(receivedData, foundLen);
|
|
|
|
triggerEvent(DEVICE_UNKNOWN_REPLY, foundId);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
DeviceReplyInfo *info = &(iter->second);
|
|
|
|
|
|
|
|
if (info->delayCycles != 0) {
|
2020-10-12 18:18:41 +02:00
|
|
|
result = interpretDeviceReply(foundId, receivedData);
|
|
|
|
|
|
|
|
if(result == IGNORE_REPLY_DATA) {
|
|
|
|
return;
|
|
|
|
}
|
2020-07-16 11:56:48 +02:00
|
|
|
|
2020-10-12 18:18:41 +02:00
|
|
|
if (info->periodic) {
|
2020-07-16 11:56:48 +02:00
|
|
|
info->delayCycles = info->maxDelayCycles;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
info->delayCycles = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (result != RETURN_OK) {
|
|
|
|
// Report failed interpretation to FDIR.
|
|
|
|
replyRawReplyIfnotWiretapped(receivedData, foundLen);
|
|
|
|
triggerEvent(DEVICE_INTERPRETING_REPLY_FAILED, result, foundId);
|
|
|
|
}
|
|
|
|
replyToReply(iter, result);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// Other completion failure messages are created by timeout.
|
|
|
|
// Powering down the device might take some time during which periodic
|
|
|
|
// replies may still come in.
|
|
|
|
if (mode != _MODE_WAIT_OFF) {
|
|
|
|
triggerEvent(DEVICE_UNREQUESTED_REPLY, foundId);
|
|
|
|
}
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ReturnValue_t DeviceHandlerBase::getStorageData(store_address_t storageAddress,
|
2020-07-16 11:56:48 +02:00
|
|
|
uint8_t** data, uint32_t * len) {
|
2020-05-04 16:53:04 +02:00
|
|
|
size_t lenTmp;
|
2016-06-15 23:48:41 +02:00
|
|
|
|
2020-05-17 17:54:21 +02:00
|
|
|
if (IPCStore == nullptr) {
|
|
|
|
*data = nullptr;
|
2016-06-15 23:48:41 +02:00
|
|
|
*len = 0;
|
|
|
|
return RETURN_FAILED;
|
|
|
|
}
|
|
|
|
ReturnValue_t result = IPCStore->modifyData(storageAddress, data, &lenTmp);
|
|
|
|
if (result == RETURN_OK) {
|
|
|
|
*len = lenTmp;
|
|
|
|
return RETURN_OK;
|
|
|
|
} else {
|
|
|
|
triggerEvent(StorageManagerIF::GET_DATA_FAILED, result,
|
|
|
|
storageAddress.raw);
|
2020-05-17 17:54:21 +02:00
|
|
|
*data = nullptr;
|
2016-06-15 23:48:41 +02:00
|
|
|
*len = 0;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::replyRawData(const uint8_t *data, size_t len,
|
|
|
|
MessageQueueId_t sendTo, bool isCommand) {
|
2020-07-16 11:56:48 +02:00
|
|
|
if (IPCStore == nullptr or len == 0 or sendTo == MessageQueueIF::NO_QUEUE) {
|
2016-06-15 23:48:41 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
store_address_t address;
|
|
|
|
ReturnValue_t result = IPCStore->addData(&address, data, len);
|
|
|
|
|
|
|
|
if (result != RETURN_OK) {
|
|
|
|
triggerEvent(StorageManagerIF::STORE_DATA_FAILED, result);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-07-16 11:56:48 +02:00
|
|
|
CommandMessage command;
|
2016-06-15 23:48:41 +02:00
|
|
|
|
2020-07-16 11:56:48 +02:00
|
|
|
DeviceHandlerMessage::setDeviceHandlerRawReplyMessage(&command,
|
2016-06-15 23:48:41 +02:00
|
|
|
getObjectId(), address, isCommand);
|
|
|
|
|
2020-07-16 11:56:48 +02:00
|
|
|
result = commandQueue->sendMessage(sendTo, &command);
|
2016-06-15 23:48:41 +02:00
|
|
|
|
|
|
|
if (result != RETURN_OK) {
|
|
|
|
IPCStore->deleteData(address);
|
2020-07-16 11:56:48 +02:00
|
|
|
// Silently discard data, this indicates heavy TM traffic which
|
|
|
|
// should not be increased by additional events.
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//Default child implementations
|
2020-10-12 18:28:07 +02:00
|
|
|
DeviceHandlerIF::CommunicationAction DeviceHandlerBase::getComAction() {
|
2016-06-15 23:48:41 +02:00
|
|
|
switch (pstStep) {
|
|
|
|
case 0:
|
2020-10-12 18:28:07 +02:00
|
|
|
return CommunicationAction::PERFORM_OPERATION;
|
2016-06-15 23:48:41 +02:00
|
|
|
break;
|
|
|
|
case 1:
|
2020-10-12 18:28:07 +02:00
|
|
|
return CommunicationAction::SEND_WRITE;
|
|
|
|
break;
|
2016-06-15 23:48:41 +02:00
|
|
|
case 2:
|
2020-10-12 18:28:07 +02:00
|
|
|
return CommunicationAction::GET_WRITE;
|
2016-06-15 23:48:41 +02:00
|
|
|
break;
|
|
|
|
case 3:
|
2020-10-12 18:28:07 +02:00
|
|
|
return CommunicationAction::SEND_READ;
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
return CommunicationAction::GET_READ;
|
2016-06-15 23:48:41 +02:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2020-10-12 18:28:07 +02:00
|
|
|
return CommunicationAction::NOTHING;
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
MessageQueueId_t DeviceHandlerBase::getCommandQueue() const {
|
2018-07-12 16:29:32 +02:00
|
|
|
return commandQueue->getId();
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::buildRawDeviceCommand(CommandMessage* commandMessage) {
|
|
|
|
storedRawData = DeviceHandlerMessage::getStoreAddress(commandMessage);
|
|
|
|
ReturnValue_t result = getStorageData(storedRawData, &rawPacket,
|
|
|
|
&rawPacketLen);
|
|
|
|
if (result != RETURN_OK) {
|
|
|
|
replyReturnvalueToCommand(result, RAW_COMMAND_ID);
|
|
|
|
storedRawData.raw = StorageManagerIF::INVALID_ADDRESS;
|
|
|
|
} else {
|
|
|
|
cookieInfo.pendingCommand = deviceCommandMap.find(
|
|
|
|
(DeviceCommandId_t) RAW_COMMAND_ID);
|
|
|
|
cookieInfo.pendingCommand->second.isExecuting = true;
|
|
|
|
cookieInfo.state = COOKIE_WRITE_READY;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::commandSwitch(ReturnValue_t onOff) {
|
2020-07-16 11:56:48 +02:00
|
|
|
if(powerSwitcher == nullptr) {
|
|
|
|
return;
|
|
|
|
}
|
2016-06-15 23:48:41 +02:00
|
|
|
const uint8_t *switches;
|
|
|
|
uint8_t numberOfSwitches = 0;
|
|
|
|
ReturnValue_t result = getSwitches(&switches, &numberOfSwitches);
|
|
|
|
if (result == RETURN_OK) {
|
|
|
|
while (numberOfSwitches > 0) {
|
|
|
|
powerSwitcher->sendSwitchCommand(switches[numberOfSwitches - 1],
|
|
|
|
onOff);
|
|
|
|
numberOfSwitches--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ReturnValue_t DeviceHandlerBase::getSwitches(const uint8_t **switches,
|
|
|
|
uint8_t *numberOfSwitches) {
|
2020-07-16 11:56:48 +02:00
|
|
|
return DeviceHandlerBase::NO_SWITCH;
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::modeChanged(void) {
|
|
|
|
}
|
|
|
|
|
|
|
|
ReturnValue_t DeviceHandlerBase::enableReplyInReplyMap(
|
|
|
|
DeviceCommandMap::iterator command, uint8_t expectedReplies,
|
|
|
|
bool useAlternativeId, DeviceCommandId_t alternativeReply) {
|
|
|
|
DeviceReplyMap::iterator iter;
|
|
|
|
if (useAlternativeId) {
|
|
|
|
iter = deviceReplyMap.find(alternativeReply);
|
|
|
|
} else {
|
|
|
|
iter = deviceReplyMap.find(command->first);
|
|
|
|
}
|
|
|
|
if (iter != deviceReplyMap.end()) {
|
|
|
|
DeviceReplyInfo *info = &(iter->second);
|
|
|
|
info->delayCycles = info->maxDelayCycles;
|
|
|
|
info->command = command;
|
|
|
|
command->second.expectedReplies = expectedReplies;
|
|
|
|
return RETURN_OK;
|
|
|
|
} else {
|
|
|
|
return NO_REPLY_EXPECTED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::doTransition(Mode_t modeFrom, Submode_t subModeFrom) {
|
|
|
|
setMode(getBaseMode(mode));
|
|
|
|
}
|
|
|
|
|
|
|
|
ReturnValue_t DeviceHandlerBase::getStateOfSwitches(void) {
|
2020-07-16 11:56:48 +02:00
|
|
|
if(powerSwitcher == nullptr) {
|
|
|
|
return NO_SWITCH;
|
|
|
|
}
|
2016-06-15 23:48:41 +02:00
|
|
|
uint8_t numberOfSwitches = 0;
|
|
|
|
const uint8_t *switches;
|
|
|
|
|
|
|
|
ReturnValue_t result = getSwitches(&switches, &numberOfSwitches);
|
|
|
|
if ((result == RETURN_OK) && (numberOfSwitches != 0)) {
|
|
|
|
while (numberOfSwitches > 0) {
|
|
|
|
if (powerSwitcher->getSwitchState(switches[numberOfSwitches - 1])
|
|
|
|
== PowerSwitchIF::SWITCH_OFF) {
|
|
|
|
return PowerSwitchIF::SWITCH_OFF;
|
|
|
|
}
|
|
|
|
numberOfSwitches--;
|
|
|
|
}
|
|
|
|
return PowerSwitchIF::SWITCH_ON;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NO_SWITCH;
|
|
|
|
}
|
|
|
|
|
|
|
|
Mode_t DeviceHandlerBase::getBaseMode(Mode_t transitionMode) {
|
2021-01-08 16:14:11 +01:00
|
|
|
// only child action special modes are handled, as a child should
|
|
|
|
// never see any base action modes
|
2016-06-15 23:48:41 +02:00
|
|
|
if (transitionMode == _MODE_START_UP) {
|
|
|
|
return _MODE_TO_ON;
|
|
|
|
}
|
|
|
|
if (transitionMode == _MODE_SHUT_DOWN) {
|
|
|
|
return _MODE_POWER_DOWN;
|
|
|
|
}
|
|
|
|
return transitionMode
|
|
|
|
& ~(TRANSITION_MODE_BASE_ACTION_MASK
|
|
|
|
| TRANSITION_MODE_CHILD_ACTION_MASK);
|
|
|
|
}
|
|
|
|
|
|
|
|
//SHOULDDO: Allow transition from OFF to NORMAL to reduce complexity in assemblies. And, by the way, throw away DHB and write a new one:
|
|
|
|
// - Include power and thermal completely, but more modular :-)
|
|
|
|
// - Don't use modes for state transitions, reduce FSM (Finte State Machine) complexity.
|
|
|
|
// - Modularization?
|
|
|
|
ReturnValue_t DeviceHandlerBase::checkModeCommand(Mode_t commandedMode,
|
|
|
|
Submode_t commandedSubmode, uint32_t* msToReachTheMode) {
|
|
|
|
if (isTransitionalMode()) {
|
|
|
|
return IN_TRANSITION;
|
|
|
|
}
|
|
|
|
if ((mode == MODE_ERROR_ON) && (commandedMode != MODE_OFF)) {
|
|
|
|
return TRANS_NOT_ALLOWED;
|
|
|
|
}
|
|
|
|
if ((commandedMode == MODE_NORMAL) && (mode == MODE_OFF)) {
|
|
|
|
return TRANS_NOT_ALLOWED;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((commandedMode == MODE_ON) && (mode == MODE_OFF)
|
2020-12-08 15:59:30 +01:00
|
|
|
and (thermalSet != nullptr)) {
|
|
|
|
ReturnValue_t result = thermalSet->read();
|
|
|
|
if(result == HasReturnvaluesIF::RETURN_OK) {
|
|
|
|
if((thermalSet->heaterRequest.value !=
|
|
|
|
ThermalComponentIF::STATE_REQUEST_IGNORE) and (not
|
|
|
|
ThermalComponentIF::isOperational(
|
|
|
|
thermalSet->thermalState.value))) {
|
2018-07-12 16:29:32 +02:00
|
|
|
triggerEvent(ThermalComponentIF::TEMP_NOT_IN_OP_RANGE,
|
2020-12-08 15:59:30 +01:00
|
|
|
thermalSet->thermalState.value);
|
2016-06-15 23:48:41 +02:00
|
|
|
return NON_OP_TEMPERATURE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return isModeCombinationValid(commandedMode, commandedSubmode);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::startTransition(Mode_t commandedMode,
|
|
|
|
Submode_t commandedSubmode) {
|
|
|
|
switch (commandedMode) {
|
|
|
|
case MODE_ON:
|
2020-12-08 15:59:30 +01:00
|
|
|
handleTransitionToOnMode(commandedMode, commandedSubmode);
|
2016-06-15 23:48:41 +02:00
|
|
|
break;
|
|
|
|
case MODE_OFF:
|
|
|
|
if (mode == MODE_OFF) {
|
|
|
|
triggerEvent(CHANGING_MODE, commandedMode, commandedSubmode);
|
|
|
|
setMode(_MODE_POWER_DOWN, commandedSubmode);
|
|
|
|
} else {
|
2020-12-08 15:59:30 +01:00
|
|
|
// already set the delay for the child transition
|
|
|
|
// so we don't need to call it twice
|
2016-06-15 23:48:41 +02:00
|
|
|
childTransitionDelay = getTransitionDelayMs(mode, _MODE_POWER_DOWN);
|
|
|
|
transitionSourceMode = _MODE_POWER_DOWN;
|
|
|
|
transitionSourceSubMode = commandedSubmode;
|
|
|
|
childTransitionFailure = CHILD_TIMEOUT;
|
|
|
|
setMode(_MODE_SHUT_DOWN, commandedSubmode);
|
|
|
|
triggerEvent(CHANGING_MODE, commandedMode, commandedSubmode);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case MODE_RAW:
|
|
|
|
if (mode != MODE_OFF) {
|
|
|
|
setTransition(MODE_RAW, commandedSubmode);
|
|
|
|
} else {
|
|
|
|
setMode(MODE_RAW, commandedSubmode);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case MODE_NORMAL:
|
|
|
|
if (mode != MODE_OFF) {
|
|
|
|
setTransition(MODE_NORMAL, commandedSubmode);
|
|
|
|
} else {
|
|
|
|
replyReturnvalueToCommand(HasModesIF::TRANS_NOT_ALLOWED);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-08 15:59:30 +01:00
|
|
|
void DeviceHandlerBase::handleTransitionToOnMode(Mode_t commandedMode,
|
|
|
|
Submode_t commandedSubmode) {
|
|
|
|
if (mode == MODE_OFF) {
|
|
|
|
transitionSourceMode = _MODE_POWER_DOWN;
|
|
|
|
transitionSourceSubMode = SUBMODE_NONE;
|
|
|
|
setMode(_MODE_POWER_ON, commandedSubmode);
|
|
|
|
// already set the delay for the child transition so we don't
|
|
|
|
// need to call it twice
|
|
|
|
childTransitionDelay = getTransitionDelayMs(_MODE_START_UP,
|
|
|
|
MODE_ON);
|
|
|
|
triggerEvent(CHANGING_MODE, commandedMode, commandedSubmode);
|
|
|
|
if(thermalSet != nullptr) {
|
|
|
|
ReturnValue_t result = thermalSet->read();
|
|
|
|
if(result == HasReturnvaluesIF::RETURN_OK) {
|
|
|
|
if(thermalSet->heaterRequest !=
|
|
|
|
ThermalComponentIF::STATE_REQUEST_IGNORE) {
|
|
|
|
thermalSet->heaterRequest =
|
|
|
|
ThermalComponentIF::STATE_REQUEST_OPERATIONAL;
|
|
|
|
thermalSet->commit();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
setTransition(MODE_ON, commandedSubmode);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-15 23:48:41 +02:00
|
|
|
void DeviceHandlerBase::getMode(Mode_t* mode, Submode_t* submode) {
|
|
|
|
*mode = this->mode;
|
|
|
|
*submode = this->submode;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::setToExternalControl() {
|
2020-07-16 12:54:30 +02:00
|
|
|
healthHelper.setHealth(EXTERNAL_CONTROL);
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::announceMode(bool recursive) {
|
|
|
|
triggerEvent(MODE_INFO, mode, submode);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DeviceHandlerBase::dontCheckQueue() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::missedReply(DeviceCommandId_t id) {
|
|
|
|
if (ignoreMissedRepliesCount > 0) {
|
|
|
|
ignoreMissedRepliesCount--;
|
|
|
|
} else {
|
|
|
|
triggerEvent(DEVICE_MISSED_REPLY, id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
HasHealthIF::HealthState DeviceHandlerBase::getHealth() {
|
2020-07-16 12:54:30 +02:00
|
|
|
return healthHelper.getHealth();
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ReturnValue_t DeviceHandlerBase::setHealth(HealthState health) {
|
2020-07-16 12:54:30 +02:00
|
|
|
healthHelper.setHealth(health);
|
|
|
|
return HasReturnvaluesIF::RETURN_OK;
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::checkSwitchState() {
|
|
|
|
if ((mode == MODE_ON || mode == MODE_NORMAL)) {
|
|
|
|
//We only check in ON and NORMAL, ignore RAW and ERROR_ON.
|
|
|
|
ReturnValue_t result = getStateOfSwitches();
|
|
|
|
if (result == PowerSwitchIF::SWITCH_OFF && !switchOffWasReported) {
|
|
|
|
triggerEvent(PowerSwitchIF::SWITCH_WENT_OFF);
|
|
|
|
switchOffWasReported = true;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
switchOffWasReported = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::doOnActivity() {
|
|
|
|
}
|
|
|
|
|
|
|
|
ReturnValue_t DeviceHandlerBase::acceptExternalDeviceCommands() {
|
|
|
|
if ((mode != MODE_ON) && (mode != MODE_NORMAL)) {
|
|
|
|
return WRONG_MODE_FOR_COMMAND;
|
|
|
|
}
|
|
|
|
return RETURN_OK;
|
|
|
|
}
|
|
|
|
|
2018-07-12 16:29:32 +02:00
|
|
|
void DeviceHandlerBase::replyRawReplyIfnotWiretapped(const uint8_t* data,
|
|
|
|
size_t len) {
|
|
|
|
if ((wiretappingMode == RAW)
|
|
|
|
&& (defaultRawReceiver == requestedRawTraffic)) {
|
|
|
|
//The raw packet was already sent by the wiretapping service
|
|
|
|
} else {
|
|
|
|
replyRawData(data, len, defaultRawReceiver);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-15 23:48:41 +02:00
|
|
|
ReturnValue_t DeviceHandlerBase::handleDeviceHandlerMessage(
|
|
|
|
CommandMessage * message) {
|
|
|
|
switch (message->getCommand()) {
|
|
|
|
case DeviceHandlerMessage::CMD_WIRETAPPING:
|
|
|
|
switch (DeviceHandlerMessage::getWiretappingMode(message)) {
|
|
|
|
case RAW:
|
|
|
|
wiretappingMode = RAW;
|
2018-07-12 16:29:32 +02:00
|
|
|
requestedRawTraffic = commandQueue->getLastPartner();
|
2016-06-15 23:48:41 +02:00
|
|
|
break;
|
|
|
|
case TM:
|
|
|
|
wiretappingMode = TM;
|
2018-07-12 16:29:32 +02:00
|
|
|
requestedRawTraffic = commandQueue->getLastPartner();
|
2016-06-15 23:48:41 +02:00
|
|
|
break;
|
|
|
|
case OFF:
|
|
|
|
wiretappingMode = OFF;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
replyReturnvalueToCommand(INVALID_COMMAND_PARAMETER);
|
|
|
|
wiretappingMode = OFF;
|
|
|
|
return RETURN_OK;
|
|
|
|
}
|
|
|
|
replyReturnvalueToCommand(RETURN_OK);
|
|
|
|
return RETURN_OK;
|
|
|
|
case DeviceHandlerMessage::CMD_RAW:
|
|
|
|
if ((mode != MODE_RAW)) {
|
|
|
|
DeviceHandlerMessage::clear(message);
|
|
|
|
replyReturnvalueToCommand(WRONG_MODE_FOR_COMMAND);
|
|
|
|
} else {
|
|
|
|
buildRawDeviceCommand(message);
|
|
|
|
}
|
|
|
|
return RETURN_OK;
|
|
|
|
default:
|
|
|
|
return RETURN_FAILED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::setParentQueue(MessageQueueId_t parentQueueId) {
|
|
|
|
modeHelper.setParentQueue(parentQueueId);
|
2020-06-25 18:09:32 +02:00
|
|
|
healthHelper.setParentQueue(parentQueueId);
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool DeviceHandlerBase::isAwaitingReply() {
|
|
|
|
std::map<DeviceCommandId_t, DeviceReplyInfo>::iterator iter;
|
|
|
|
for (iter = deviceReplyMap.begin(); iter != deviceReplyMap.end(); ++iter) {
|
|
|
|
if (iter->second.delayCycles != 0) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
ReturnValue_t DeviceHandlerBase::letChildHandleMessage(
|
|
|
|
CommandMessage * message) {
|
|
|
|
return RETURN_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::handleDeviceTM(SerializeIF* data,
|
2018-07-12 16:29:32 +02:00
|
|
|
DeviceCommandId_t replyId, bool neverInDataPool, bool forceDirectTm) {
|
2016-06-15 23:48:41 +02:00
|
|
|
DeviceReplyMap::iterator iter = deviceReplyMap.find(replyId);
|
|
|
|
if (iter == deviceReplyMap.end()) {
|
|
|
|
triggerEvent(DEVICE_UNKNOWN_REPLY, replyId);
|
|
|
|
return;
|
|
|
|
}
|
2018-07-12 16:29:32 +02:00
|
|
|
DeviceTmReportingWrapper wrapper(getObjectId(), replyId, data);
|
2020-07-16 11:56:48 +02:00
|
|
|
//replies to a command
|
|
|
|
if (iter->second.command != deviceCommandMap.end())
|
|
|
|
{
|
2016-06-15 23:48:41 +02:00
|
|
|
MessageQueueId_t queueId = iter->second.command->second.sendReplyTo;
|
2018-07-12 16:29:32 +02:00
|
|
|
|
2016-06-15 23:48:41 +02:00
|
|
|
if (queueId != NO_COMMANDER) {
|
2018-07-12 16:29:32 +02:00
|
|
|
//This may fail, but we'll ignore the fault.
|
2016-06-15 23:48:41 +02:00
|
|
|
actionHelper.reportData(queueId, replyId, data);
|
|
|
|
}
|
2020-07-16 11:56:48 +02:00
|
|
|
|
2016-06-15 23:48:41 +02:00
|
|
|
//This check should make sure we get any TM but don't get anything doubled.
|
2018-07-12 16:29:32 +02:00
|
|
|
if (wiretappingMode == TM && (requestedRawTraffic != queueId)) {
|
|
|
|
actionHelper.reportData(requestedRawTraffic, replyId, &wrapper);
|
2020-07-16 11:56:48 +02:00
|
|
|
}
|
|
|
|
else if (forceDirectTm and (defaultRawReceiver != queueId) and
|
|
|
|
(defaultRawReceiver != MessageQueueIF::NO_QUEUE))
|
|
|
|
{
|
|
|
|
// hiding of sender needed so the service will handle it as
|
|
|
|
// unexpected Data, no matter what state (progress or completed)
|
|
|
|
// it is in
|
2018-07-12 16:29:32 +02:00
|
|
|
actionHelper.reportData(defaultRawReceiver, replyId, &wrapper,
|
2020-07-16 11:56:48 +02:00
|
|
|
true);
|
2018-07-12 16:29:32 +02:00
|
|
|
}
|
2020-07-16 11:56:48 +02:00
|
|
|
}
|
|
|
|
//unrequested/aperiodic replies
|
|
|
|
else
|
|
|
|
{
|
2018-07-12 16:29:32 +02:00
|
|
|
if (wiretappingMode == TM) {
|
|
|
|
actionHelper.reportData(requestedRawTraffic, replyId, &wrapper);
|
2020-07-16 11:56:48 +02:00
|
|
|
}
|
|
|
|
else if (forceDirectTm and defaultRawReceiver !=
|
|
|
|
MessageQueueIF::NO_QUEUE)
|
|
|
|
{
|
|
|
|
// hiding of sender needed so the service will handle it as
|
|
|
|
// unexpected Data, no matter what state (progress or completed)
|
|
|
|
// it is in
|
2018-07-12 16:29:32 +02:00
|
|
|
actionHelper.reportData(defaultRawReceiver, replyId, &wrapper,
|
2020-07-16 11:56:48 +02:00
|
|
|
true);
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
|
|
|
}
|
2020-07-16 11:56:48 +02:00
|
|
|
//Try to cast to GlobDataSet and commit data.
|
2020-12-08 15:59:30 +01:00
|
|
|
if (not neverInDataPool) {
|
|
|
|
LocalPoolDataSetBase* dataSet =
|
|
|
|
dynamic_cast<LocalPoolDataSetBase*>(data);
|
|
|
|
if (dataSet != nullptr) {
|
|
|
|
dataSet->setValidity(true, true);
|
|
|
|
dataSet->commit();
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ReturnValue_t DeviceHandlerBase::executeAction(ActionId_t actionId,
|
2020-07-16 11:56:48 +02:00
|
|
|
MessageQueueId_t commandedBy, const uint8_t* data, size_t size) {
|
2016-06-15 23:48:41 +02:00
|
|
|
ReturnValue_t result = acceptExternalDeviceCommands();
|
|
|
|
if (result != HasReturnvaluesIF::RETURN_OK) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
DeviceCommandMap::iterator iter = deviceCommandMap.find(actionId);
|
|
|
|
if (iter == deviceCommandMap.end()) {
|
|
|
|
result = COMMAND_NOT_SUPPORTED;
|
|
|
|
} else if (iter->second.isExecuting) {
|
|
|
|
result = COMMAND_ALREADY_SENT;
|
|
|
|
} else {
|
|
|
|
result = buildCommandFromCommand(actionId, data, size);
|
|
|
|
}
|
|
|
|
if (result == RETURN_OK) {
|
|
|
|
iter->second.sendReplyTo = commandedBy;
|
|
|
|
iter->second.isExecuting = true;
|
|
|
|
cookieInfo.pendingCommand = iter;
|
|
|
|
cookieInfo.state = COOKIE_WRITE_READY;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::buildInternalCommand(void) {
|
|
|
|
//Neither Raw nor Direct could build a command
|
|
|
|
ReturnValue_t result = NOTHING_TO_SEND;
|
|
|
|
DeviceCommandId_t deviceCommandId = NO_COMMAND_ID;
|
|
|
|
if (mode == MODE_NORMAL) {
|
|
|
|
result = buildNormalDeviceCommand(&deviceCommandId);
|
|
|
|
if (result == BUSY) {
|
2021-01-08 16:14:11 +01:00
|
|
|
// so we can track misconfigurations
|
2021-01-12 13:18:18 +01:00
|
|
|
printWarningOrError(sif::OutputTypes::OUT_WARNING,
|
2021-01-08 16:14:11 +01:00
|
|
|
"buildInternalCommand",
|
|
|
|
HasReturnvaluesIF::RETURN_FAILED,
|
|
|
|
"Busy.");
|
2016-06-15 23:48:41 +02:00
|
|
|
result = NOTHING_TO_SEND; //no need to report this
|
|
|
|
}
|
2020-07-16 11:56:48 +02:00
|
|
|
}
|
|
|
|
else if (mode == MODE_RAW) {
|
2016-06-15 23:48:41 +02:00
|
|
|
result = buildChildRawCommand();
|
|
|
|
deviceCommandId = RAW_COMMAND_ID;
|
2020-07-16 11:56:48 +02:00
|
|
|
}
|
|
|
|
else if (mode & TRANSITION_MODE_CHILD_ACTION_MASK) {
|
2016-06-15 23:48:41 +02:00
|
|
|
result = buildTransitionDeviceCommand(&deviceCommandId);
|
2020-07-16 11:56:48 +02:00
|
|
|
}
|
|
|
|
else {
|
2016-06-15 23:48:41 +02:00
|
|
|
return;
|
|
|
|
}
|
2020-07-16 11:56:48 +02:00
|
|
|
|
2016-06-15 23:48:41 +02:00
|
|
|
if (result == NOTHING_TO_SEND) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (result == RETURN_OK) {
|
|
|
|
DeviceCommandMap::iterator iter = deviceCommandMap.find(
|
|
|
|
deviceCommandId);
|
|
|
|
if (iter == deviceCommandMap.end()) {
|
|
|
|
result = COMMAND_NOT_SUPPORTED;
|
|
|
|
} else if (iter->second.isExecuting) {
|
2021-01-08 16:23:57 +01:00
|
|
|
#if FSFW_DISABLE_PRINTOUT == 0
|
2021-01-08 16:16:17 +01:00
|
|
|
char output[36];
|
2021-01-08 21:03:47 +01:00
|
|
|
sprintf(output, "Command 0x%08x is executing",
|
|
|
|
static_cast<unsigned int>(deviceCommandId));
|
2021-01-08 16:14:11 +01:00
|
|
|
// so we can track misconfigurations
|
2021-01-12 13:18:18 +01:00
|
|
|
printWarningOrError(sif::OutputTypes::OUT_WARNING,
|
2021-01-08 16:14:11 +01:00
|
|
|
"buildInternalCommand",
|
|
|
|
HasReturnvaluesIF::RETURN_FAILED,
|
|
|
|
output);
|
2021-01-08 16:23:57 +01:00
|
|
|
#endif
|
2020-10-12 18:18:41 +02:00
|
|
|
// this is an internal command, no need to report a failure here,
|
|
|
|
// missed reply will track if a reply is too late, otherwise, it's ok
|
|
|
|
return;
|
2016-06-15 23:48:41 +02:00
|
|
|
} else {
|
|
|
|
iter->second.sendReplyTo = NO_COMMANDER;
|
|
|
|
iter->second.isExecuting = true;
|
|
|
|
cookieInfo.pendingCommand = iter;
|
|
|
|
cookieInfo.state = COOKIE_WRITE_READY;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (result != RETURN_OK) {
|
|
|
|
triggerEvent(DEVICE_BUILDING_COMMAND_FAILED, result, deviceCommandId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ReturnValue_t DeviceHandlerBase::buildChildRawCommand() {
|
|
|
|
return NOTHING_TO_SEND;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t DeviceHandlerBase::getReplyDelayCycles(
|
|
|
|
DeviceCommandId_t deviceCommand) {
|
|
|
|
DeviceReplyMap::iterator iter = deviceReplyMap.find(deviceCommand);
|
|
|
|
if (iter == deviceReplyMap.end()) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return iter->second.delayCycles;
|
|
|
|
}
|
|
|
|
|
|
|
|
Mode_t DeviceHandlerBase::getTransitionSourceMode() const {
|
|
|
|
return transitionSourceMode;
|
|
|
|
}
|
|
|
|
|
|
|
|
Submode_t DeviceHandlerBase::getTransitionSourceSubMode() const {
|
|
|
|
return transitionSourceSubMode;
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::triggerEvent(Event event, uint32_t parameter1,
|
|
|
|
uint32_t parameter2) {
|
|
|
|
fdirInstance->triggerEvent(event, parameter1, parameter2);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::forwardEvent(Event event, uint32_t parameter1,
|
|
|
|
uint32_t parameter2) const {
|
|
|
|
fdirInstance->triggerEvent(event, parameter1, parameter2);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::doOffActivity() {
|
|
|
|
}
|
|
|
|
|
|
|
|
ReturnValue_t DeviceHandlerBase::getParameter(uint8_t domainId,
|
|
|
|
uint16_t parameterId, ParameterWrapper* parameterWrapper,
|
|
|
|
const ParameterWrapper* newValues, uint16_t startAtIndex) {
|
|
|
|
ReturnValue_t result = fdirInstance->getParameter(domainId, parameterId,
|
|
|
|
parameterWrapper, newValues, startAtIndex);
|
|
|
|
if (result != INVALID_DOMAIN_ID) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
return INVALID_DOMAIN_ID;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
bool DeviceHandlerBase::isTransitionalMode() {
|
2018-07-12 16:29:32 +02:00
|
|
|
return ((mode
|
|
|
|
& (TRANSITION_MODE_BASE_ACTION_MASK
|
|
|
|
| TRANSITION_MODE_CHILD_ACTION_MASK)) != 0);
|
2016-06-15 23:48:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool DeviceHandlerBase::commandIsExecuting(DeviceCommandId_t commandId) {
|
|
|
|
auto iter = deviceCommandMap.find(commandId);
|
|
|
|
if (iter != deviceCommandMap.end()) {
|
|
|
|
return iter->second.isExecuting;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2018-07-12 16:29:32 +02:00
|
|
|
|
|
|
|
void DeviceHandlerBase::changeHK(Mode_t mode, Submode_t submode, bool enable) {
|
|
|
|
}
|
2018-07-13 18:28:26 +02:00
|
|
|
|
2020-12-08 15:59:30 +01:00
|
|
|
void DeviceHandlerBase::setTaskIF(PeriodicTaskIF* task){
|
|
|
|
executingTask = task;
|
2018-07-13 18:28:26 +02:00
|
|
|
}
|
2020-04-19 22:10:02 +02:00
|
|
|
|
2020-04-19 22:17:14 +02:00
|
|
|
// Default implementations empty.
|
2020-04-19 22:10:02 +02:00
|
|
|
void DeviceHandlerBase::debugInterface(uint8_t positionTracker,
|
|
|
|
object_id_t objectId, uint32_t parameter) {}
|
2020-04-19 22:17:14 +02:00
|
|
|
|
2020-07-16 11:56:48 +02:00
|
|
|
void DeviceHandlerBase::performOperationHook() {
|
|
|
|
}
|
|
|
|
|
2020-10-12 18:18:41 +02:00
|
|
|
ReturnValue_t DeviceHandlerBase::initializeLocalDataPool(
|
2021-01-12 19:58:29 +01:00
|
|
|
localpool::DataPool &localDataPoolMap,
|
2020-10-12 18:18:41 +02:00
|
|
|
LocalDataPoolManager& poolManager) {
|
2020-12-08 15:59:30 +01:00
|
|
|
if(thermalSet != nullptr) {
|
|
|
|
localDataPoolMap.emplace(thermalSet->thermalStatePoolId,
|
|
|
|
new PoolEntry<DeviceHandlerIF::dh_thermal_state_t>);
|
|
|
|
localDataPoolMap.emplace(thermalSet->heaterRequestPoolId,
|
|
|
|
new PoolEntry<DeviceHandlerIF::dh_heater_request_t>);
|
|
|
|
}
|
2020-10-12 18:18:41 +02:00
|
|
|
return RETURN_OK;
|
|
|
|
}
|
|
|
|
|
2020-07-16 11:56:48 +02:00
|
|
|
ReturnValue_t DeviceHandlerBase::initializeAfterTaskCreation() {
|
|
|
|
// In this function, the task handle should be valid if the task
|
|
|
|
// was implemented correctly. We still check to be 1000 % sure :-)
|
|
|
|
if(executingTask != nullptr) {
|
|
|
|
pstIntervalMs = executingTask->getPeriodMs();
|
|
|
|
}
|
2021-01-12 20:59:15 +01:00
|
|
|
this->poolManager.initializeAfterTaskCreation();
|
2020-10-12 18:18:41 +02:00
|
|
|
|
|
|
|
if(setStartupImmediately) {
|
|
|
|
startTransition(MODE_ON, SUBMODE_NONE);
|
|
|
|
}
|
2020-07-16 11:56:48 +02:00
|
|
|
return HasReturnvaluesIF::RETURN_OK;
|
|
|
|
}
|
|
|
|
|
2020-10-12 18:18:41 +02:00
|
|
|
LocalPoolDataSetBase* DeviceHandlerBase::getDataSetHandle(sid_t sid) {
|
|
|
|
auto iter = deviceReplyMap.find(sid.ownerSetId);
|
|
|
|
if(iter != deviceReplyMap.end()) {
|
|
|
|
return iter->second.dataSet;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
object_id_t DeviceHandlerBase::getObjectId() const {
|
|
|
|
return SystemObject::getObjectId();
|
|
|
|
}
|
|
|
|
|
|
|
|
void DeviceHandlerBase::setStartUpImmediately() {
|
|
|
|
this->setStartupImmediately = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
dur_millis_t DeviceHandlerBase::getPeriodicOperationFrequency() const {
|
|
|
|
return pstIntervalMs;
|
|
|
|
}
|
|
|
|
|
2020-12-08 15:59:30 +01:00
|
|
|
DeviceCommandId_t DeviceHandlerBase::getPendingCommand() const {
|
|
|
|
if(cookieInfo.pendingCommand != deviceCommandMap.end()) {
|
|
|
|
return cookieInfo.pendingCommand->first;
|
|
|
|
}
|
2020-12-23 20:17:10 +01:00
|
|
|
return DeviceHandlerIF::NO_COMMAND_ID;
|
2020-12-08 15:59:30 +01:00
|
|
|
}
|
2020-12-22 12:44:55 +01:00
|
|
|
|
|
|
|
void DeviceHandlerBase::setNormalDatapoolEntriesInvalid() {
|
|
|
|
for(const auto& reply: deviceReplyMap) {
|
|
|
|
if(reply.second.dataSet != nullptr) {
|
|
|
|
reply.second.dataSet->setValidity(false, true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-01-08 16:14:11 +01:00
|
|
|
|
2021-01-12 14:08:51 +01:00
|
|
|
AccessPoolManagerIF* DeviceHandlerBase::getAccessorHandle() {
|
2021-01-12 20:59:15 +01:00
|
|
|
return &poolManager;
|
2021-01-12 00:13:49 +01:00
|
|
|
}
|
|
|
|
|
2021-01-12 13:18:18 +01:00
|
|
|
void DeviceHandlerBase::printWarningOrError(sif::OutputTypes errorType,
|
2021-01-08 16:14:11 +01:00
|
|
|
const char *functionName, ReturnValue_t errorCode,
|
|
|
|
const char *errorPrint) {
|
|
|
|
if(errorPrint == nullptr) {
|
|
|
|
if(errorCode == ObjectManagerIF::CHILD_INIT_FAILED) {
|
|
|
|
errorPrint = "Initialization error";
|
|
|
|
}
|
|
|
|
if(errorCode == HasReturnvaluesIF::RETURN_FAILED) {
|
2021-01-12 13:18:18 +01:00
|
|
|
if(errorType == sif::OutputTypes::OUT_WARNING) {
|
2021-01-08 16:14:11 +01:00
|
|
|
errorPrint = "Generic Warning";
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
errorPrint = "Generic Error";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
errorPrint = "Unknown error";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-12 13:18:18 +01:00
|
|
|
if(errorType == sif::OutputTypes::OUT_WARNING) {
|
2021-01-08 16:14:11 +01:00
|
|
|
#if FSFW_CPP_OSTREAM_ENABLED == 1
|
|
|
|
sif::warning << "DeviceHandlerBase::" << functionName << ": Object ID "
|
|
|
|
<< std::hex << std::setw(8) << std::setfill('0')
|
|
|
|
<< this->getObjectId() << " | " << errorPrint << std::dec
|
|
|
|
<< std::setfill(' ') << std::endl;
|
|
|
|
#else
|
2021-01-12 13:18:18 +01:00
|
|
|
sif::printWarning("DeviceHandlerBase::%s: Object ID 0x%08x | %s\n",
|
2021-01-08 16:14:11 +01:00
|
|
|
this->getObjectId(), errorPrint);
|
|
|
|
#endif
|
|
|
|
}
|
2021-01-12 13:18:18 +01:00
|
|
|
else if(errorType == sif::OutputTypes::OUT_ERROR) {
|
2021-01-08 16:14:11 +01:00
|
|
|
#if FSFW_CPP_OSTREAM_ENABLED == 1
|
|
|
|
sif::error << "DeviceHandlerBase::" << functionName << ": Object ID "
|
|
|
|
<< std::hex << std::setw(8) << std::setfill('0')
|
|
|
|
<< this->getObjectId() << " | " << errorPrint << std::dec
|
|
|
|
<< std::setfill(' ') << std::endl;
|
|
|
|
#else
|
2021-01-12 13:18:18 +01:00
|
|
|
sif::printError("DeviceHandlerBase::%s: Object ID 0x%08x | %s\n",
|
2021-01-08 16:14:11 +01:00
|
|
|
this->getObjectId(), errorPrint);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2021-01-11 21:31:03 +01:00
|
|
|
|
2021-01-11 22:25:39 +01:00
|
|
|
ProvidesDataPoolSubscriptionIF* DeviceHandlerBase::getSubscriptionInterface() {
|
2021-01-12 20:59:15 +01:00
|
|
|
return &poolManager;
|
2021-01-11 21:31:03 +01:00
|
|
|
}
|