#include "P60DockHandler.h" #include #include "OBSWConfig.h" P60DockHandler::P60DockHandler(object_id_t objectId, object_id_t comIF, CookieIF *comCookie, FailureIsolationBase *customFdir) : GomspaceDeviceHandler(objectId, comIF, comCookie, cfg, customFdir), coreHk(this), auxHk(this) { cfg.maxConfigTableAddress = P60Dock::MAX_CONFIGTABLE_ADDRESS; cfg.maxHkTableAddress = P60Dock::MAX_HKTABLE_ADDRESS; cfg.hkTableSize = P60Dock::HK_TABLE_SIZE; cfg.cfgTableSize = P60Dock::CONFIG_TABLE_SIZE; } P60DockHandler::~P60DockHandler() {} ReturnValue_t P60DockHandler::buildNormalDeviceCommand(DeviceCommandId_t *id) { *id = GOMSPACE::REQUEST_HK_TABLE; return buildCommandFromCommand(*id, NULL, 0); } void P60DockHandler::letChildHandleHkReply(DeviceCommandId_t id, const uint8_t *packet) { parseHkTableReply(packet); /** * Hk table will be sent to the commander if hk table request was not triggered by the * P60DockHandler itself. */ handleDeviceTM(&coreHk, id, true); } void P60DockHandler::parseHkTableReply(const uint8_t *packet) { using namespace P60Dock; PoolReadGuard pg0(&coreHk); PoolReadGuard pg1(&auxHk); if (pg0.getReadResult() != returnvalue::OK or pg1.getReadResult() != returnvalue::OK) { coreHk.setValidity(false, true); auxHk.setValidity(false, true); return; } /** * Fist 10 bytes contain the gomspace header. Each variable is preceded by the 16-bit table * address. */ for (uint8_t idx = 0; idx < hk::CHNLS_LEN; idx++) { coreHk.currents[idx] = as(packet + (idx * 2)); } for (uint8_t idx = 0; idx < hk::CHNLS_LEN; idx++) { coreHk.voltages[idx] = as(packet + 0x1a + (idx * 2)); } for (uint8_t idx = 0; idx < hk::CHNLS_LEN; idx++) { coreHk.outputEnables[idx] = *(packet + 0x34 + idx); } coreHk.temperature1 = as(packet + 0x44) * 0.1; coreHk.temperature2 = as(packet + 0x44 + 2) * 0.1; auxHk.bootcause = as(packet + 0x48); coreHk.bootCount = as(packet + 0x4c); if (firstHk) { triggerEvent(P60_BOOT_COUNT, coreHk.bootCount.value); } auxHk.uptime = as(packet + 0x50); auxHk.resetcause = as(packet + 0x54); uint8_t newBattMode = packet[0x56]; if (firstHk) { triggerEvent(BATT_MODE, newBattMode); } else if (newBattMode != coreHk.battMode.value) { triggerEvent(BATT_MODE_CHANGED, coreHk.battMode.value, newBattMode); } coreHk.battMode = newBattMode; auxHk.heaterOn = *(packet + 0x57); auxHk.converter5VStatus = *(packet + 0x58); for (uint8_t idx = 0; idx < hk::CHNLS_LEN; idx++) { auxHk.latchups[idx] = as(packet + 0x5a + (idx * 2)); } auxHk.dockVbatVoltageValue = as(packet + 0x74); auxHk.dockVccCurrent = as(packet + 0x76); coreHk.batteryCurrent = as(packet + 0x78); coreHk.batteryVoltage = as(packet + 0x7a); auxHk.batteryTemperature1 = as(packet + 0x7c); auxHk.batteryTemperature2 = as(packet + 0x7c + 2); for (uint8_t idx = 0; idx < NUM_DEVS; idx++) { auxHk.devicesType[idx] = *(packet + 0x80 + idx); } for (uint8_t idx = 0; idx < NUM_DEVS; idx++) { auxHk.devicesStatus[idx] = *(packet + 0x88 + idx); } auxHk.dearmStatus = *(packet + 0x90); auxHk.wdtCntGnd = as(packet + 0x94); auxHk.wdtCntI2c = as(packet + 0x98); auxHk.wdtCntCan = as(packet + 0x9c); auxHk.wdtCntCsp1 = as(packet + 0xa0); auxHk.wdtCntCsp2 = as(packet + 0xa0 + 4); auxHk.wdtGndLeft = as(packet + 0xa8); auxHk.wdtI2cLeft = as(packet + 0xac); auxHk.wdtCanLeft = as(packet + 0xb0); auxHk.wdtCspLeft1 = *(packet + 0xb4); auxHk.wdtCspLeft2 = *(packet + 0xb4 + 1); auxHk.batteryChargeCurrent = as(packet + 0xb6); auxHk.batteryDischargeCurrent = as(packet + 0xb8); auxHk.ant6Depl = *(packet + 0xba); auxHk.ar6Depl = *(packet + 0xbb); if (firstHk) { firstHk = false; } coreHk.setValidity(true, true); auxHk.setValidity(true, true); } ReturnValue_t P60DockHandler::initializeLocalDataPool(localpool::DataPool &localDataPoolMap, LocalDataPoolManager &poolManager) { using namespace P60Dock; localDataPoolMap.emplace(pool::P60_CURRENTS, &hkCurrents); localDataPoolMap.emplace(pool::P60_VOLTAGES, &hkVoltages); localDataPoolMap.emplace(pool::P60_OUTPUT_ENABLE, &outputEnables); localDataPoolMap.emplace(pool::P60DOCK_TEMPERATURE_1, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_TEMPERATURE_2, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_BOOT_CAUSE, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_BOOT_CNT, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_UPTIME, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_RESETCAUSE, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_BATT_MODE, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_HEATER_ON, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_CONV_5V_ENABLE_STATUS, new PoolEntry({0})); localDataPoolMap.emplace(pool::LATCHUPS, &latchups); localDataPoolMap.emplace(pool::P60DOCK_DOCK_VBAT, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_DOCK_VCC_CURRENT, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_BATTERY_CURRENT, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_BATTERY_VOLTAGE, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_BATTERY_TEMPERATURE_1, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_BATTERY_TEMPERATURE_2, new PoolEntry({0})); localDataPoolMap.emplace(pool::DEVICES_TYPE, &devicesType); localDataPoolMap.emplace(pool::DEVICES_STATUS, &devicesStatus); localDataPoolMap.emplace(pool::P60DOCK_DEARM_STATUS, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_WDT_CNT_GND, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_WDT_CNT_I2C, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_WDT_CNT_CAN, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_WDT_CNT_CSP_1, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_WDT_CNT_CSP_2, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_WDT_GND_LEFT, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_WDT_I2C_LEFT, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_WDT_CAN_LEFT, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_WDT_CSP_LEFT_1, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_WDT_CSP_LEFT_2, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_BATT_CHARGE_CURRENT, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_BATT_DISCHARGE_CURRENT, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_ANT6_DEPL, new PoolEntry({0})); localDataPoolMap.emplace(pool::P60DOCK_AR6_DEPL, new PoolEntry({0})); poolManager.subscribeForDiagPeriodicPacket( subdp::DiagnosticsHkPeriodicParams(coreHk.getSid(), false, 10.0)); poolManager.subscribeForRegularPeriodicPacket( subdp::RegularHkPeriodicParams(auxHk.getSid(), false, 30.0)); return returnvalue::OK; } ReturnValue_t P60DockHandler::printStatus(DeviceCommandId_t cmd) { ReturnValue_t result = returnvalue::OK; switch (cmd) { case (GOMSPACE::PRINT_SWITCH_V_I): { PoolReadGuard pg0(&coreHk); PoolReadGuard pg1(&auxHk); if (pg0.getReadResult() != returnvalue::OK or pg1.getReadResult() != returnvalue::OK) { break; } printHkTableSwitchIV(); return returnvalue::OK; } case (GOMSPACE::PRINT_LATCHUPS): { PoolReadGuard pg(&auxHk); result = pg.getReadResult(); printHkTableLatchups(); if (result != returnvalue::OK) { break; } return returnvalue::OK; } default: { return DeviceHandlerIF::COMMAND_NOT_SUPPORTED; } } sif::warning << "Reading P60 Dock HK table failed" << std::endl; return returnvalue::FAILED; } void P60DockHandler::printHkTableSwitchIV() { using namespace P60Dock; sif::info << "P60 Dock Info:" << std::endl; sif::info << "Boot Cause: " << auxHk.bootcause << " | Boot Count: " << std::setw(4) << std::right << coreHk.bootCount << std::endl; sif::info << "Reset Cause: " << auxHk.resetcause << " | Battery Mode: " << static_cast(coreHk.battMode.value) << std::endl; sif::info << "SwitchState, Currents [mA], Voltages [mV]:" << std::endl; sif::info << std::setw(MAX_CHANNEL_STR_WIDTH) << std::left << "Dock VBAT VCC" << std::dec << "| -, " << std::setw(4) << std::right << auxHk.dockVccCurrent << ", " << std::setw(5) << auxHk.dockVbatVoltageValue << std::endl; sif::info << std::setw(MAX_CHANNEL_STR_WIDTH) << std::left << "BATT" << std::dec << "| -, " << std::setw(4) << std::right << coreHk.batteryCurrent.value << ", " << std::setw(5) << coreHk.batteryVoltage.value << std::endl; auto genericPrintoutHandler = [&](std::string name, uint8_t idx) { sif::info << std::setw(MAX_CHANNEL_STR_WIDTH) << std::left << name << std::dec << "| " << unsigned(coreHk.outputEnables[idx]) << ", " << std::setw(4) << std::right << coreHk.currents[idx] << ", " << std::setw(5) << coreHk.voltages[idx] << std::endl; }; genericPrintoutHandler("ACU VCC", hk::ACU_VCC); genericPrintoutHandler("ACU VBAT", hk::ACU_VBAT); genericPrintoutHandler("PDU1 VCC", hk::PDU1_VCC); genericPrintoutHandler("PDU1 VBAT", hk::PDU1_VBAT); genericPrintoutHandler("PDU2 VCC", hk::PDU2_VCC); genericPrintoutHandler("PDU2 VBAT", hk::PDU2_VBAT); genericPrintoutHandler("Stack VBAT", hk::STACK_VBAT); genericPrintoutHandler("Stack 3V3", hk::STACK_3V3); genericPrintoutHandler("Stack 5V", hk::STACK_5V); } LocalPoolDataSetBase *P60DockHandler::getDataSetHandle(sid_t sid) { if (sid == coreHk.getSid()) { return &coreHk; } else if (sid == auxHk.getSid()) { return &auxHk; } return nullptr; } void P60DockHandler::printHkTableLatchups() { using namespace P60Dock; sif::info << "P60 Latchup Information" << std::endl; auto genericPrintoutHandler = [&](std::string name, uint8_t idx) { sif::info << std::setw(MAX_CHANNEL_STR_WIDTH) << std::left << name << std::dec << "| " << std::setw(4) << std::right << auxHk.latchups[idx] << std::endl; }; genericPrintoutHandler("ACU VCC", hk::ACU_VCC); genericPrintoutHandler("ACU VBAT", hk::ACU_VBAT); genericPrintoutHandler("PDU1 VCC", hk::PDU1_VCC); genericPrintoutHandler("PDU1 VBAT", hk::PDU1_VBAT); genericPrintoutHandler("PDU2 VCC", hk::PDU2_VCC); genericPrintoutHandler("PDU2 VBAT", hk::PDU2_VBAT); genericPrintoutHandler("STACK VBAT", hk::STACK_VBAT); genericPrintoutHandler("STACK 3V3", hk::STACK_3V3); genericPrintoutHandler("STACK 5V", hk::STACK_5V); genericPrintoutHandler("GS 3V3", hk::GS3V3); genericPrintoutHandler("GS 5V", hk::GS5V); genericPrintoutHandler("X3 VBAT", hk::X3_IDLE_VBAT); genericPrintoutHandler("X3 VCC", hk::X3_IDLE_VCC); } void P60DockHandler::setDebugMode(bool enable) { this->debugMode = enable; } void P60DockHandler::letChildHandleConfigReply(DeviceCommandId_t id, const uint8_t *packet) { handleDeviceTM(packet, P60Dock::CONFIG_TABLE_SIZE, id); }