eive-obsw/mission/controller/PowerController.cpp

219 lines
6.7 KiB
C++
Raw Normal View History

2023-07-28 11:36:20 +02:00
2023-06-07 11:33:09 +02:00
#include <mission/controller/PowerController.h>
PowerController::PowerController(object_id_t objectId, bool enableHkSets)
2023-07-26 11:18:15 +02:00
: ExtendedControllerBase(objectId),
enableHkSets(enableHkSets),
parameterHelper(this),
2023-07-28 11:36:20 +02:00
pwrCtrlCoreHk(this) {}
2023-06-07 11:33:09 +02:00
ReturnValue_t PowerController::initialize() {
ReturnValue_t result = parameterHelper.initialize();
if (result != returnvalue::OK) {
return result;
}
2023-07-26 10:08:40 +02:00
sif::debug << "Rush B, no stop" << std::endl;
2023-06-07 11:33:09 +02:00
return ExtendedControllerBase::initialize();
}
ReturnValue_t PowerController::handleCommandMessage(CommandMessage *message) {
ReturnValue_t result = actionHelper.handleActionMessage(message);
if (result == returnvalue::OK) {
return result;
}
result = parameterHelper.handleParameterMessage(message);
if (result == returnvalue::OK) {
return result;
}
return result;
}
2023-06-15 17:51:32 +02:00
MessageQueueId_t PowerController::getCommandQueue() const { return commandQueue->getId(); }
ReturnValue_t PowerController::getParameter(uint8_t domainId, uint8_t parameterId,
ParameterWrapper *parameterWrapper,
const ParameterWrapper *newValues,
uint16_t startAtIndex) {
2023-07-26 17:01:48 +02:00
switch (domainId) {
case 0x0: // direct members
switch (parameterId) {
case 0x0:
parameterWrapper->set(batteryInternalResistance);
break;
2023-07-28 11:36:20 +02:00
case 0x1:
parameterWrapper->set(batteryMaximumCapacity);
break;
2023-07-26 17:01:48 +02:00
default:
return INVALID_IDENTIFIER_ID;
}
break;
default:
return INVALID_DOMAIN_ID;
};
return returnvalue::OK;
2023-06-15 17:51:32 +02:00
}
2023-06-07 11:33:09 +02:00
void PowerController::performControlOperation() {
switch (internalState) {
case InternalState::STARTUP: {
initialCountdown.resetTimer();
internalState = InternalState::INITIAL_DELAY;
return;
}
case InternalState::INITIAL_DELAY: {
if (initialCountdown.hasTimedOut()) {
internalState = InternalState::READY;
}
return;
}
case InternalState::READY: {
2023-07-26 11:18:15 +02:00
// if (mode != MODE_OFF) {
sif::debug << "oh shit, now i gotta do something" << std::endl;
2023-07-26 17:01:48 +02:00
calculateStateOfCharge();
2023-07-26 11:18:15 +02:00
// do something
//}
2023-06-07 11:33:09 +02:00
break;
}
default:
break;
}
}
ReturnValue_t PowerController::initializeLocalDataPool(localpool::DataPool &localDataPoolMap,
LocalDataPoolManager &poolManager) {
2023-07-28 11:36:20 +02:00
localDataPoolMap.emplace(pwrctrl::PoolIds::TOTAL_BATTERY_CURRENT, new PoolEntry<int16_t>({0}));
localDataPoolMap.emplace(pwrctrl::PoolIds::OPEN_CIRCUIT_VOLTAGE_CHARGE,
new PoolEntry<float>({0.0}));
localDataPoolMap.emplace(pwrctrl::PoolIds::COULOMB_COUNTER_CHARGE, new PoolEntry<float>({0.0}));
poolManager.subscribeForRegularPeriodicPacket({pwrCtrlCoreHk.getSid(), enableHkSets, 60.0});
2023-06-07 11:33:09 +02:00
return returnvalue::OK;
}
2023-06-10 15:14:20 +02:00
LocalPoolDataSetBase *PowerController::getDataSetHandle(sid_t sid) {
switch (sid.ownerSetId) {
2023-07-28 11:36:20 +02:00
case pwrctrl::CORE_HK:
return &pwrCtrlCoreHk;
2023-06-10 15:14:20 +02:00
default:
return nullptr;
}
return nullptr;
}
ReturnValue_t PowerController::checkModeCommand(Mode_t mode, Submode_t submode,
uint32_t *msToReachTheMode) {
if (mode == MODE_OFF) {
if (submode == SUBMODE_NONE) {
return returnvalue::OK;
} else {
return INVALID_SUBMODE;
}
}
return INVALID_MODE;
}
2023-07-26 17:01:48 +02:00
void PowerController::calculateStateOfCharge() {
// get time
Clock::getClock_timeval(&now);
2023-07-28 11:36:20 +02:00
// update EPS HK values
2023-07-26 17:01:48 +02:00
ReturnValue_t result = updateEpsData();
2023-07-28 11:36:20 +02:00
if (result != returnvalue::OK) {
triggerEvent(power::DATASET_READ_FAILED);
sif::info << "Power Controller::Reading of Datasets has failed" << std::endl;
return;
}
2023-07-26 17:01:48 +02:00
2023-07-28 11:36:20 +02:00
// calculate total battery current
iBat = p60CoreHk.batteryCurrent.value + bpxBatteryHk.heaterCurrent.value +
bpxBatteryHk.dischargeCurrent.value;
2023-07-26 17:01:48 +02:00
2023-07-28 11:36:20 +02:00
calculateOpenCircuitVoltageCharge();
calculateCoulombCounterCharge();
// commit to dataset
{
PoolReadGuard pg(&pwrCtrlCoreHk);
if (pg.getReadResult() == returnvalue::OK) {
pwrCtrlCoreHk.totalBatteryCurrent.value = iBat;
pwrCtrlCoreHk.openCircuitVoltageCharge.value = charge2stateOfCharge(openCircuitVoltageCharge);
pwrCtrlCoreHk.coulombCounterCharge.value = charge2stateOfCharge(coulombCounterCharge);
pwrCtrlCoreHk.setValidity(true, true);
2023-07-26 17:01:48 +02:00
}
}
2023-07-28 11:36:20 +02:00
// store time for next run
oldTime = now;
}
void PowerController::calculateOpenCircuitVoltageCharge() {
float vBatCorrected = p60CoreHk.batteryVoltage.value - iBat * batteryInternalResistance;
if (vBatCorrected >= lookUpTableOcv[1][100]) {
triggerEvent(power::VOLTAGE_OUT_OF_BOUNDS, 0);
sif::error << "Power Controller::Voltage is too high" << std::endl;
return;
} else if (vBatCorrected <= lookUpTableOcv[1][0]) {
triggerEvent(power::VOLTAGE_OUT_OF_BOUNDS, 1);
sif::error << "Power Controller::Voltage is too low" << std::endl;
return;
}
uint8_t lookUpTableIdx = 99;
while (lookUpTableOcv[1][lookUpTableIdx] > vBatCorrected) {
lookUpTableIdx--;
2023-07-26 17:01:48 +02:00
}
2023-07-28 11:36:20 +02:00
openCircuitVoltageCharge = linearInterpolation(
vBatCorrected, lookUpTableOcv[1][lookUpTableIdx], lookUpTableOcv[1][lookUpTableIdx + 1],
lookUpTableOcv[0][lookUpTableIdx], lookUpTableOcv[0][lookUpTableIdx + 1]);
}
void PowerController::calculateCoulombCounterCharge() {
if ((pwrCtrlCoreHk.coulombCounterCharge.value == 0) or
(p60CoreHk.batteryVoltage.value > coulombCounterVoltageUpperThreshold and
pwrCtrlCoreHk.coulombCounterCharge.value >= )) {
coulombCounterCharge = openCircuitVoltageCharge;
2023-07-26 17:01:48 +02:00
}
2023-07-28 11:36:20 +02:00
else {
double timeDiff = timevalOperations::toDouble(now - oldTime);
coulombCounterCharge = pwrCtrlCoreHk.coulombCounterCharge.value + iBat * timeDiff;
}
2023-07-26 17:01:48 +02:00
}
ReturnValue_t PowerController::updateEpsData() {
std::vector<ReturnValue_t> results;
{
PoolReadGuard pgBat(&bpxBatteryHk);
results.push_back(pgBat.getReadResult());
}
{
PoolReadGuard pgP60(&p60CoreHk);
results.push_back(pgP60.getReadResult());
}
{
PoolReadGuard pgPdu1(&pdu1CoreHk);
results.push_back(pgPdu1.getReadResult());
}
{
PoolReadGuard pgPdu2(&pdu2CoreHk);
results.push_back(pgPdu2.getReadResult());
}
{
PoolReadGuard pgAcu(&acuCoreHk);
results.push_back(pgAcu.getReadResult());
}
for (const auto &result : results) {
if (result != returnvalue::OK) {
return result;
}
}
return returnvalue::OK;
}
float PowerController::charge2stateOfCharge(float capacity) {
return capacity / batteryMaximumCapacity;
}
2023-07-28 11:36:20 +02:00
float PowerController::linearInterpolation(float x, float x0, float x1, float y0, float y1) {
return y0 + (x - x0) * (y1 - y0) / (x1 - x0);
}