fsfw/osal/rtems/PollingTask.cpp

138 lines
4.2 KiB
C++
Raw Normal View History

2021-01-27 18:03:56 +01:00
#include "PollingTask.h"
#include "RtemsBasic.h"
2020-11-17 19:25:57 +01:00
#include "../../tasks/FixedSequenceSlot.h"
2020-08-13 20:53:35 +02:00
#include "../../objectmanager/SystemObjectIF.h"
2020-11-17 19:25:57 +01:00
#include "../../objectmanager/ObjectManagerIF.h"
2020-08-13 20:53:35 +02:00
#include "../../returnvalues/HasReturnvaluesIF.h"
2021-01-27 18:03:56 +01:00
#include "../../serviceinterface/ServiceInterface.h"
2018-07-13 18:28:26 +02:00
#include <rtems/bspIo.h>
#include <rtems/rtems/ratemon.h>
#include <rtems/rtems/status.h>
#include <rtems/rtems/tasks.h>
#include <rtems/rtems/types.h>
#include <sys/_stdint.h>
2021-01-27 18:03:56 +01:00
#if FSFW_CPP_OSTREAM_ENABLED == 1
2018-07-13 18:28:26 +02:00
#include <iostream>
2021-01-27 18:03:56 +01:00
#endif
#include <cstddef>
2018-07-13 18:28:26 +02:00
#include <list>
uint32_t PollingTask::deadlineMissedCount = 0;
PollingTask::PollingTask(const char *name, rtems_task_priority setPriority,
size_t setStack, uint32_t setOverallPeriod,
void (*setDeadlineMissedFunc)()) :
2021-01-27 00:59:34 +01:00
RTEMSTaskBase(setPriority, setStack, name), periodId(0), pst(
setOverallPeriod) {
// All additional attributes are applied to the object.
this->deadlineMissedFunc = setDeadlineMissedFunc;
}
PollingTask::~PollingTask() {
}
rtems_task PollingTask::taskEntryPoint(rtems_task_argument argument) {
//The argument is re-interpreted as PollingTask.
PollingTask *originalTask(reinterpret_cast<PollingTask*>(argument));
//The task's functionality is called.
originalTask->taskFunctionality();
2021-01-04 15:34:08 +01:00
#if FSFW_CPP_OSTREAM_ENABLED == 1
2020-11-17 19:25:57 +01:00
sif::debug << "Polling task " << originalTask->getId()
<< " returned from taskFunctionality." << std::endl;
2021-01-04 15:34:08 +01:00
#endif
}
void PollingTask::missedDeadlineCounter() {
PollingTask::deadlineMissedCount++;
if (PollingTask::deadlineMissedCount % 10 == 0) {
2021-01-04 15:34:08 +01:00
#if FSFW_CPP_OSTREAM_ENABLED == 1
2020-11-17 19:25:57 +01:00
sif::error << "PST missed " << PollingTask::deadlineMissedCount
<< " deadlines." << std::endl;
2021-01-04 15:34:08 +01:00
#endif
}
}
ReturnValue_t PollingTask::startTask() {
rtems_status_code status = rtems_task_start(id, PollingTask::taskEntryPoint,
rtems_task_argument((void *) this));
if (status != RTEMS_SUCCESSFUL) {
2021-01-04 15:34:08 +01:00
#if FSFW_CPP_OSTREAM_ENABLED == 1
2020-11-17 19:25:57 +01:00
sif::error << "PollingTask::startTask for " << std::hex << this->getId()
<< std::dec << " failed." << std::endl;
2021-01-04 15:34:08 +01:00
#endif
}
2018-07-13 18:28:26 +02:00
switch(status){
case RTEMS_SUCCESSFUL:
//ask started successfully
return HasReturnvaluesIF::RETURN_OK;
default:
/* RTEMS_INVALID_ADDRESS - invalid task entry point
RTEMS_INVALID_ID - invalid task id
RTEMS_INCORRECT_STATE - task not in the dormant state
RTEMS_ILLEGAL_ON_REMOTE_OBJECT - cannot start remote task */
return HasReturnvaluesIF::RETURN_FAILED;
}
}
2020-05-05 18:34:04 +02:00
ReturnValue_t PollingTask::addSlot(object_id_t componentId,
2020-05-05 18:32:50 +02:00
uint32_t slotTimeMs, int8_t executionStep) {
2020-11-17 19:25:57 +01:00
ExecutableObjectIF* object = objectManager->get<ExecutableObjectIF>(componentId);
if (object != nullptr) {
pst.addSlot(componentId, slotTimeMs, executionStep, object, this);
2020-05-05 18:32:50 +02:00
return HasReturnvaluesIF::RETURN_OK;
}
2021-01-04 15:34:08 +01:00
#if FSFW_CPP_OSTREAM_ENABLED == 1
2020-11-17 19:25:57 +01:00
sif::error << "Component " << std::hex << componentId <<
2020-05-05 18:32:50 +02:00
" not found, not adding it to pst" << std::endl;
2021-01-04 15:34:08 +01:00
#endif
2020-05-05 18:32:50 +02:00
return HasReturnvaluesIF::RETURN_FAILED;
}
2020-05-05 18:36:18 +02:00
uint32_t PollingTask::getPeriodMs() const {
return pst.getLengthMs();
}
ReturnValue_t PollingTask::checkSequence() const {
return pst.checkSequence();
}
2018-07-13 18:28:26 +02:00
#include <rtems/io.h>
void PollingTask::taskFunctionality() {
// A local iterator for the Polling Sequence Table is created to find the start time for the first entry.
2020-11-17 19:25:57 +01:00
FixedSlotSequence::SlotListIter it = pst.current;
//The start time for the first entry is read.
2020-11-17 19:25:57 +01:00
rtems_interval interval = RtemsBasic::convertMsToTicks(it->pollingTimeMs);
2021-01-27 00:59:34 +01:00
RTEMSTaskBase::setAndStartPeriod(interval,&periodId);
//The task's "infinite" inner loop is entered.
while (1) {
if (pst.slotFollowsImmediately()) {
//Do nothing
} else {
//The interval for the next polling slot is selected.
2018-07-13 15:56:37 +02:00
interval = RtemsBasic::convertMsToTicks(this->pst.getIntervalToNextSlotMs());
//The period is checked and restarted with the new interval.
//If the deadline was missed, the deadlineMissedFunc is called.
2021-01-27 00:59:34 +01:00
rtems_status_code status = RTEMSTaskBase::restartPeriod(interval,periodId);
if (status == RTEMS_TIMEOUT) {
2020-11-17 19:35:37 +01:00
if (this->deadlineMissedFunc != nullptr) {
this->deadlineMissedFunc();
}
}
}
//The device handler for this slot is executed and the next one is chosen.
this->pst.executeAndAdvance();
}
}
ReturnValue_t PollingTask::sleepFor(uint32_t ms){
2021-01-27 00:59:34 +01:00
return RTEMSTaskBase::sleepFor(ms);
};