Non Blocking SD Card Initialization #75
@ -12,6 +12,8 @@
|
|||||||
#include "fsfw/osal/common/TcpTmTcBridge.h"
|
#include "fsfw/osal/common/TcpTmTcBridge.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#include "fsfw/osal/linux/Timer.h"
|
||||||
|
|
||||||
#include "bsp_q7s/memory/scratchApi.h"
|
#include "bsp_q7s/memory/scratchApi.h"
|
||||||
#include "bsp_q7s/memory/SdCardManager.h"
|
#include "bsp_q7s/memory/SdCardManager.h"
|
||||||
|
|
||||||
@ -24,8 +26,7 @@ CoreController::Chip CoreController::currentChip = Chip::NO_CHIP;
|
|||||||
CoreController::Copy CoreController::currentCopy = Copy::NO_COPY;
|
CoreController::Copy CoreController::currentCopy = Copy::NO_COPY;
|
||||||
|
|
||||||
CoreController::CoreController(object_id_t objectId):
|
CoreController::CoreController(object_id_t objectId):
|
||||||
ExtendedControllerBase(objectId, objects::NO_OBJECT, 5),
|
ExtendedControllerBase(objectId, objects::NO_OBJECT, 5), opDivider(5) {
|
||||||
opDivider(5) {
|
|
||||||
ReturnValue_t result = HasReturnvaluesIF::RETURN_OK;
|
ReturnValue_t result = HasReturnvaluesIF::RETURN_OK;
|
||||||
try {
|
try {
|
||||||
result = initWatchdogFifo();
|
result = initWatchdogFifo();
|
||||||
@ -33,6 +34,13 @@ CoreController::CoreController(object_id_t objectId):
|
|||||||
sif::warning << "CoreController::CoreController: Watchdog FIFO init failed" <<
|
sif::warning << "CoreController::CoreController: Watchdog FIFO init failed" <<
|
||||||
std::endl;
|
std::endl;
|
||||||
}
|
}
|
||||||
|
sdcMan = SdCardManager::instance();
|
||||||
|
if(sdcMan == nullptr) {
|
||||||
|
sif::error << "CoreController::CoreController: SD card manager invalid!" << std::endl;
|
||||||
|
}
|
||||||
|
if(not BLOCKING_SD_INIT) {
|
||||||
|
initSdCardNonBlocking();
|
||||||
|
}
|
||||||
|
|
||||||
result = initBootCopy();
|
result = initBootCopy();
|
||||||
if(result != HasReturnvaluesIF::RETURN_OK) {
|
if(result != HasReturnvaluesIF::RETURN_OK) {
|
||||||
@ -50,6 +58,9 @@ ReturnValue_t CoreController::handleCommandMessage(CommandMessage *message) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void CoreController::performControlOperation() {
|
void CoreController::performControlOperation() {
|
||||||
|
if(not BLOCKING_SD_INIT and not sdInitFinished) {
|
||||||
|
initSdCardNonBlocking();
|
||||||
|
}
|
||||||
performWatchdogControlOperation();
|
performWatchdogControlOperation();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -71,6 +82,10 @@ ReturnValue_t CoreController::initialize() {
|
|||||||
"count failed" << std::endl;
|
"count failed" << std::endl;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(not BLOCKING_SD_INIT) {
|
||||||
|
initSdCardNonBlocking();
|
||||||
|
}
|
||||||
|
|
||||||
return ExtendedControllerBase::initialize();
|
return ExtendedControllerBase::initialize();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -79,16 +94,11 @@ ReturnValue_t CoreController::checkModeCommand(Mode_t mode, Submode_t submode,
|
|||||||
return HasReturnvaluesIF::RETURN_OK;
|
return HasReturnvaluesIF::RETURN_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
ReturnValue_t CoreController::initSdCard() {
|
ReturnValue_t CoreController::initSdCardBlocking() {
|
||||||
#if Q7S_SD_CARD_CONFIG == Q7S_SD_NONE
|
#if Q7S_SD_CARD_CONFIG == Q7S_SD_NONE
|
||||||
sif::info << "No SD card initialization will be performed" << std::endl;
|
sif::info << "No SD card initialization will be performed" << std::endl;
|
||||||
return HasReturnvaluesIF::RETURN_OK;
|
return HasReturnvaluesIF::RETURN_OK;
|
||||||
#else
|
#else
|
||||||
SdCardManager* sdcMan = SdCardManager::instance();
|
|
||||||
if(sdcMan == nullptr) {
|
|
||||||
return HasReturnvaluesIF::RETURN_FAILED;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create update status file
|
// Create update status file
|
||||||
ReturnValue_t result = sdcMan->updateSdCardStateFile();
|
ReturnValue_t result = sdcMan->updateSdCardStateFile();
|
||||||
if(result != HasReturnvaluesIF::RETURN_OK) {
|
if(result != HasReturnvaluesIF::RETURN_OK) {
|
||||||
@ -96,19 +106,24 @@ ReturnValue_t CoreController::initSdCard() {
|
|||||||
<< std::endl;
|
<< std::endl;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto statusPair = SdCardManager::SdStatusPair(sd::SdStatus::OFF, sd::SdStatus::OFF);
|
auto statusPair = SdCardManager::SdStatusPair(sd::SdState::OFF, sd::SdState::OFF);
|
||||||
result = sdcMan->getSdCardActiveStatus(statusPair);
|
result = sdcMan->getSdCardActiveStatus(statusPair);
|
||||||
if(result != HasReturnvaluesIF::RETURN_OK) {
|
if(result != HasReturnvaluesIF::RETURN_OK) {
|
||||||
sif::warning << "Getting SD card activity status failed" << std::endl;
|
sif::warning << "Getting SD card activity status failed" << std::endl;
|
||||||
}
|
}
|
||||||
|
|
||||||
#if Q7S_SD_CARD_CONFIG == Q7S_SD_COLD_REDUNDANT
|
#if Q7S_SD_CARD_CONFIG == Q7S_SD_COLD_REDUNDANT
|
||||||
return sdCardColdRedundantInit(sdcMan, statusPair);
|
determinePreferredSdCard();
|
||||||
|
sif::info << "Cold redundant SD card configuration, preferred SD card: " <<
|
||||||
|
static_cast<int>(preferredSdCard) << std::endl;
|
||||||
|
result = sdCardColdRedundantInit(sdcMan, statusPair);
|
||||||
|
// Update status file
|
||||||
|
sdcMan->updateSdCardStateFile();
|
||||||
|
return result;
|
||||||
#elif Q7S_SD_CARD_CONFIG == Q7S_SD_HOT_REDUNDANT
|
#elif Q7S_SD_CARD_CONFIG == Q7S_SD_HOT_REDUNDANT
|
||||||
sif::info << "Hot redundant SD card configuration" << std::endl;
|
sif::info << "Hot redundant SD card configuration" << std::endl;
|
||||||
|
sdCardSetup(statusPair, sd::SdCard::SLOT_0, "0");
|
||||||
setUpSdCard(sd::SdCard::SLOT_0, sdStatus.first, "0");
|
sdCardSetup(statusPair, sd::SdCard::SLOT_1, "1");
|
||||||
setUpSdCard(sd::SdCard::SLOT_1, sdStatus.second, "1");
|
|
||||||
// Update status file
|
// Update status file
|
||||||
sdcMan->updateSdCardStateFile();
|
sdcMan->updateSdCardStateFile();
|
||||||
return HasReturnvaluesIF::RETURN_OK;
|
return HasReturnvaluesIF::RETURN_OK;
|
||||||
@ -118,9 +133,130 @@ ReturnValue_t CoreController::initSdCard() {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ReturnValue_t CoreController::sdCardSetup(SdCardManager& sdcMan,
|
ReturnValue_t CoreController::initSdCardNonBlocking() {
|
||||||
SdCardManager::SdStatusPair& statusPair,sd::SdCard sdCard, sd::SdStatus status,
|
#if Q7S_SD_CARD_CONFIG == Q7S_SD_NONE
|
||||||
std::string sdString) {
|
sif::info << "No SD card initialization will be performed" << std::endl;
|
||||||
|
return HasReturnvaluesIF::RETURN_OK;
|
||||||
|
#else
|
||||||
|
ReturnValue_t result = HasReturnvaluesIF::RETURN_OK;
|
||||||
|
if(state == SdStates::START) {
|
||||||
|
sdcInitCycleCount = 0;
|
||||||
|
commandExecuted = false;
|
||||||
|
state = SdStates::GET_INFO;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(state == SdStates::GET_INFO) {
|
||||||
|
if(not commandExecuted) {
|
||||||
|
// Create update status file
|
||||||
|
result = sdcMan->updateSdCardStateFile();
|
||||||
|
if(result != HasReturnvaluesIF::RETURN_OK) {
|
||||||
|
sif::warning << "CoreController::initialize: Updating SD card state file failed"
|
||||||
|
<< std::endl;
|
||||||
|
}
|
||||||
|
commandExecuted = true;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
SdCardManager::Operations operation;
|
||||||
|
SdCardManager::OpStatus status = sdcMan->checkCurrentOp(operation);
|
||||||
|
if(status == SdCardManager::OpStatus::SUCCESS) {
|
||||||
|
state = SdStates::SET_STATE;
|
||||||
|
commandExecuted = false;
|
||||||
|
sdcInitCycleCount = 0;
|
||||||
|
}
|
||||||
|
else if(sdcInitCycleCount > 2) {
|
||||||
|
sif::warning << "CoreController::initSdCardNonBlocking: "
|
||||||
|
"Updating SDC file takes too long" << std::endl;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if(state == SdStates::SET_STATE) {
|
||||||
|
if(not commandExecuted) {
|
||||||
|
determinePreferredSdCard();
|
||||||
|
statusPair = SdCardManager::SdStatusPair(sd::SdState::OFF, sd::SdState::OFF);
|
||||||
|
result = sdcMan->getSdCardActiveStatus(statusPair);
|
||||||
|
if(result != HasReturnvaluesIF::RETURN_OK) {
|
||||||
|
sif::warning << "Getting SD card activity status failed" << std::endl;
|
||||||
|
}
|
||||||
|
sif::info << "Cold redundant SD card configuration, preferred SD card: " <<
|
||||||
|
static_cast<int>(preferredSdCard) << std::endl;
|
||||||
|
result = sdCardColdRedundantInit(sdcMan, statusPair);
|
||||||
|
if(result == SdCardManager::ALREADY_ON) {
|
||||||
|
state = SdStates::MOUNT;
|
||||||
|
}
|
||||||
|
if(result == SdCardManager::ALREADY_MOUNTED) {
|
||||||
|
state = SdStates::UPDATE_INFO;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
commandExecuted = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
SdCardManager::Operations operation;
|
||||||
|
SdCardManager::OpStatus status = sdcMan->checkCurrentOp(operation);
|
||||||
|
if(status == SdCardManager::OpStatus::SUCCESS) {
|
||||||
|
state = SdStates::MOUNT;
|
||||||
|
commandExecuted = false;
|
||||||
|
sdcInitCycleCount = 0;
|
||||||
|
}
|
||||||
|
else if(sdcInitCycleCount > 10) {
|
||||||
|
sif::warning << "CoreController::initSdCardNonBlocking: "
|
||||||
|
"Setting SDC state takes too long" << std::endl;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if(state == SdStates::MOUNT) {
|
||||||
|
if(not commandExecuted) {
|
||||||
|
result = sdCardColdRedundantInit(sdcMan, statusPair);
|
||||||
|
commandExecuted = false;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
SdCardManager::Operations operation;
|
||||||
|
SdCardManager::OpStatus status = sdcMan->checkCurrentOp(operation);
|
||||||
|
if(status == SdCardManager::OpStatus::SUCCESS) {
|
||||||
|
state = SdStates::UPDATE_INFO;
|
||||||
|
commandExecuted = false;
|
||||||
|
sdcInitCycleCount = 0;
|
||||||
|
}
|
||||||
|
else if(sdcInitCycleCount > 5) {
|
||||||
|
sif::warning << "CoreController::initSdCardNonBlocking: "
|
||||||
|
"Mounting SD card takes too long" << std::endl;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if(state == SdStates::UPDATE_INFO) {
|
||||||
|
if(not commandExecuted) {
|
||||||
|
// Update status file
|
||||||
|
result = sdcMan->updateSdCardStateFile();
|
||||||
|
if(result != HasReturnvaluesIF::RETURN_OK) {
|
||||||
|
sif::warning << "CoreController::initialize: Updating SD card state file failed"
|
||||||
|
<< std::endl;
|
||||||
|
}
|
||||||
|
commandExecuted = true;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
SdCardManager::Operations operation;
|
||||||
|
SdCardManager::OpStatus status = sdcMan->checkCurrentOp(operation);
|
||||||
|
if(status == SdCardManager::OpStatus::SUCCESS) {
|
||||||
|
sdInitFinished = true;
|
||||||
|
state = SdStates::DONE;
|
||||||
|
}
|
||||||
|
else if(sdcInitCycleCount > 2) {
|
||||||
|
sif::warning << "CoreController::initSdCardNonBlocking: "
|
||||||
|
"Updating SDC file takes too long" << std::endl;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return HasReturnvaluesIF::RETURN_OK;
|
||||||
|
#endif /* Q7S_SD_CARD_CONFIG != Q7S_SD_NONE */
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
ReturnValue_t CoreController::sdCardSetup(SdCardManager::SdStatusPair& statusPair,
|
||||||
|
sd::SdCard sdCard, std::string sdString) {
|
||||||
std::string mountString;
|
std::string mountString;
|
||||||
if(sdCard == sd::SdCard::SLOT_0) {
|
if(sdCard == sd::SdCard::SLOT_0) {
|
||||||
mountString = SdCardManager::SD_0_MOUNT_POINT;
|
mountString = SdCardManager::SD_0_MOUNT_POINT;
|
||||||
@ -129,27 +265,60 @@ ReturnValue_t CoreController::sdCardSetup(SdCardManager& sdcMan,
|
|||||||
mountString = SdCardManager::SD_1_MOUNT_POINT;
|
mountString = SdCardManager::SD_1_MOUNT_POINT;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(status == sd::SdStatus::OFF) {
|
sd::SdState status = sd::SdState::OFF;
|
||||||
sif::info << "Switching on and mounting SD card " << sdString << " at " <<
|
if(sdCard == sd::SdCard::SLOT_0) {
|
||||||
mountString << std::endl;
|
status = statusPair.first;
|
||||||
return sdcMan.switchOnSdCard(sdCard, true, &statusPair);
|
|
||||||
}
|
|
||||||
else if(status == sd::SdStatus::ON) {
|
|
||||||
sif::info << "Mounting SD card " << sdString << " at " << mountString << std::endl;
|
|
||||||
return sdcMan.mountSdCard(sdCard);
|
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
status = statusPair.second;
|
||||||
|
}
|
||||||
|
if(status == sd::SdState::MOUNTED) {
|
||||||
if(std::filesystem::exists(mountString)) {
|
if(std::filesystem::exists(mountString)) {
|
||||||
sif::info << "SD card " << sdString << " already on and mounted at " <<
|
sif::info << "SD card " << sdString << " already on and mounted at " <<
|
||||||
mountString << std::endl;
|
mountString << std::endl;
|
||||||
return SdCardManager::ALREADY_MOUNTED;
|
return SdCardManager::ALREADY_MOUNTED;
|
||||||
}
|
}
|
||||||
sif::error << "SD card mounted but expected mount point " << mountString << " not found!"
|
sif::error << "SD card mounted but expected mount point " <<
|
||||||
<< std::endl;
|
mountString << " not found!" << std::endl;
|
||||||
return SdCardManager::MOUNT_ERROR;
|
return SdCardManager::MOUNT_ERROR;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(status == sd::SdState::OFF) {
|
||||||
|
if(BLOCKING_SD_INIT) {
|
||||||
|
sif::info << "Switching on and mounting SD card " << sdString << " at " <<
|
||||||
|
mountString << std::endl;
|
||||||
|
return sdcMan->switchOnSdCard(sdCard, true, &statusPair);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
if(state == SdStates::SET_STATE) {
|
||||||
|
sif::info << "Switching on SD card" << std::endl;
|
||||||
|
return sdcMan->switchOnSdCard(sdCard, false, &statusPair);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
else if(status == sd::SdState::ON) {
|
||||||
|
if(BLOCKING_SD_INIT) {
|
||||||
|
sif::info << "Mounting SD card " << sdString << " at " << mountString << std::endl;
|
||||||
|
return sdcMan->mountSdCard(sdCard);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
if(state == SdStates::SET_STATE) {
|
||||||
|
return SdCardManager::ALREADY_ON;
|
||||||
|
}
|
||||||
|
if(state == SdStates::MOUNT) {
|
||||||
|
sif::info << "Mounting SD card " << sdString << " at " << mountString << std::endl;
|
||||||
|
return sdcMan->mountSdCard(sdCard);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
sif::warning << "CoreController::sdCardSetup: Invalid state for this call" << std::endl;
|
||||||
|
}
|
||||||
|
return HasReturnvaluesIF::RETURN_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
ReturnValue_t CoreController::executeAction(ActionId_t actionId, MessageQueueId_t commandedBy,
|
ReturnValue_t CoreController::executeAction(ActionId_t actionId, MessageQueueId_t commandedBy,
|
||||||
const uint8_t *data, size_t size) {
|
const uint8_t *data, size_t size) {
|
||||||
switch(actionId) {
|
switch(actionId) {
|
||||||
@ -166,16 +335,20 @@ ReturnValue_t CoreController::executeAction(ActionId_t actionId, MessageQueueId_
|
|||||||
}
|
}
|
||||||
|
|
||||||
ReturnValue_t CoreController::initializeAfterTaskCreation() {
|
ReturnValue_t CoreController::initializeAfterTaskCreation() {
|
||||||
|
ReturnValue_t result = HasReturnvaluesIF::RETURN_OK;
|
||||||
// TODO: Do this here because this can take a longer time. Later, use non-blocking mode
|
// TODO: Do this here because this can take a longer time. Later, use non-blocking mode
|
||||||
// to start doing stuff in ctor and initialize function.
|
// to start doing stuff in ctor and initialize function.
|
||||||
// This operation can take 1-2 seconds on reboot, and around 50-100ms if the SD
|
// This operation can take 1-2 seconds on reboot, and around 50-100ms if the SD
|
||||||
// card is already on.
|
// card is already on.
|
||||||
ReturnValue_t result = initSdCard();
|
if(BLOCKING_SD_INIT) {
|
||||||
if(result != HasReturnvaluesIF::RETURN_OK) {
|
ReturnValue_t result = initSdCardBlocking();
|
||||||
sif::warning << "CoreController::CoreController: SD card init failed" << std::endl;
|
if(result != HasReturnvaluesIF::RETURN_OK) {
|
||||||
|
sif::warning << "CoreController::CoreController: SD card init failed" << std::endl;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
initSdCardNonBlocking();
|
||||||
}
|
}
|
||||||
|
|
||||||
result = initVersionFile();
|
result = initVersionFile();
|
||||||
if(result != HasReturnvaluesIF::RETURN_OK) {
|
if(result != HasReturnvaluesIF::RETURN_OK) {
|
||||||
sif::warning << "CoreController::initialize: Version initialization failed" << std::endl;
|
sif::warning << "CoreController::initialize: Version initialization failed" << std::endl;
|
||||||
@ -185,29 +358,15 @@ ReturnValue_t CoreController::initializeAfterTaskCreation() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
ReturnValue_t CoreController::sdCardColdRedundantInit(SdCardManager* sdcMan,
|
ReturnValue_t CoreController::sdCardColdRedundantInit(SdCardManager* sdcMan,
|
||||||
SdCardManager::SdStatusPair& statusPair) {
|
SdCardManager::SdStatusPair& statusPair, SdStates step) {
|
||||||
sd::SdCard preferredSdCard = sd::SdCard::SLOT_0;
|
ReturnValue_t result = HasReturnvaluesIF::RETURN_OK;
|
||||||
ReturnValue_t result = sdcMan->getPreferredSdCard(preferredSdCard);
|
|
||||||
if(result != HasReturnvaluesIF::RETURN_OK) {
|
|
||||||
if(result == scratch::KEY_NOT_FOUND) {
|
|
||||||
sif::warning << "CoreController::sdCardInit: "
|
|
||||||
"Preferred SD card not set. Setting to 0" << std::endl;
|
|
||||||
sdcMan->setPreferredSdCard(preferredSdCard);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
sif::warning << "CoreController::sdCardInit: Could not get preferred SD card"
|
|
||||||
"information from the scratch buffer" << std::endl;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
std::string preferredString;
|
std::string preferredString;
|
||||||
sd::SdStatus preferredStatus = sd::SdStatus::OFF;
|
|
||||||
|
|
||||||
sd::SdStatus otherStatus = sd::SdStatus::OFF;
|
sd::SdState otherStatus = sd::SdState::OFF;
|
||||||
std::string otherString;
|
std::string otherString;
|
||||||
sd::SdCard otherSdc = sd::SdCard::SLOT_0;
|
sd::SdCard otherSdc = sd::SdCard::SLOT_0;
|
||||||
|
|
||||||
if(preferredSdCard == sd::SdCard::SLOT_0) {
|
if(preferredSdCard == sd::SdCard::SLOT_0) {
|
||||||
preferredStatus = statusPair.first;
|
|
||||||
preferredString = "0";
|
preferredString = "0";
|
||||||
otherSdc = sd::SdCard::SLOT_1;
|
otherSdc = sd::SdCard::SLOT_1;
|
||||||
otherStatus = statusPair.second;
|
otherStatus = statusPair.second;
|
||||||
@ -215,25 +374,21 @@ ReturnValue_t CoreController::sdCardColdRedundantInit(SdCardManager* sdcMan,
|
|||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
preferredString = "1";
|
preferredString = "1";
|
||||||
preferredStatus = statusPair.second;
|
|
||||||
otherStatus = statusPair.first;
|
otherStatus = statusPair.first;
|
||||||
otherSdc = sd::SdCard::SLOT_0;
|
otherSdc = sd::SdCard::SLOT_0;
|
||||||
otherString = "0";
|
otherString = "0";
|
||||||
}
|
}
|
||||||
|
|
||||||
sif::info << "Cold redundant SD card configuration, preferred SD card " <<
|
result = sdCardSetup(statusPair, preferredSdCard, preferredString);
|
||||||
preferredString << std::endl;
|
|
||||||
|
|
||||||
result = sdCardSetup(*sdcMan, statusPair, preferredSdCard, preferredStatus, preferredString);
|
|
||||||
if(result != SdCardManager::ALREADY_MOUNTED and result != HasReturnvaluesIF::RETURN_OK) {
|
if(result != SdCardManager::ALREADY_MOUNTED and result != HasReturnvaluesIF::RETURN_OK) {
|
||||||
sif::warning << "Setting up preferred card " << otherString <<
|
sif::warning << "Setting up preferred card " << otherString <<
|
||||||
" in cold redundant mode failed" << std::endl;
|
" in cold redundant mode failed" << std::endl;
|
||||||
// Try other SD card and mark set up operation as failed
|
// Try other SD card and mark set up operation as failed
|
||||||
sdCardSetup(*sdcMan, statusPair, preferredSdCard, preferredStatus, preferredString);
|
sdCardSetup(statusPair, preferredSdCard, preferredString);
|
||||||
result = HasReturnvaluesIF::RETURN_FAILED;
|
result = HasReturnvaluesIF::RETURN_FAILED;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(result != HasReturnvaluesIF::RETURN_FAILED and otherStatus != sd::SdStatus::OFF) {
|
if(result != HasReturnvaluesIF::RETURN_FAILED and otherStatus != sd::SdState::OFF) {
|
||||||
sif::info << "Switching off secondary SD card " << otherString << std::endl;
|
sif::info << "Switching off secondary SD card " << otherString << std::endl;
|
||||||
// Switch off other SD card in cold redundant mode if setting up preferred one walked
|
// Switch off other SD card in cold redundant mode if setting up preferred one walked
|
||||||
// without issues
|
// without issues
|
||||||
@ -243,10 +398,7 @@ ReturnValue_t CoreController::sdCardColdRedundantInit(SdCardManager* sdcMan,
|
|||||||
" in cold redundant mode failed" << std::endl;
|
" in cold redundant mode failed" << std::endl;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return result;
|
||||||
// Update status file
|
|
||||||
sdcMan->updateSdCardStateFile();
|
|
||||||
return HasReturnvaluesIF::RETURN_OK;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ReturnValue_t CoreController::incrementAllocationFailureCount() {
|
ReturnValue_t CoreController::incrementAllocationFailureCount() {
|
||||||
@ -501,6 +653,26 @@ ReturnValue_t CoreController::actionPerformReboot(const uint8_t *data, size_t si
|
|||||||
return HasActionsIF::EXECUTION_FINISHED;
|
return HasActionsIF::EXECUTION_FINISHED;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
CoreController::~CoreController() {
|
||||||
|
}
|
||||||
|
|
||||||
|
void CoreController::determinePreferredSdCard() {
|
||||||
|
if(preferredSdCard == sd::SdCard::NONE) {
|
||||||
|
ReturnValue_t result = sdcMan->getPreferredSdCard(preferredSdCard);
|
||||||
|
if(result != HasReturnvaluesIF::RETURN_OK) {
|
||||||
|
if(result == scratch::KEY_NOT_FOUND) {
|
||||||
|
sif::warning << "CoreController::sdCardInit: "
|
||||||
|
"Preferred SD card not set. Setting to 0" << std::endl;
|
||||||
|
sdcMan->setPreferredSdCard(preferredSdCard);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
sif::warning << "CoreController::sdCardInit: Could not get preferred SD card"
|
||||||
|
"information from the scratch buffer" << std::endl;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void CoreController::performWatchdogControlOperation() {
|
void CoreController::performWatchdogControlOperation() {
|
||||||
// Only perform each fifth iteration
|
// Only perform each fifth iteration
|
||||||
if(watchdogFifoFd != 0 and opDivider.checkAndIncrement()) {
|
if(watchdogFifoFd != 0 and opDivider.checkAndIncrement()) {
|
||||||
|
@ -8,6 +8,8 @@
|
|||||||
#include "events/subsystemIdRanges.h"
|
#include "events/subsystemIdRanges.h"
|
||||||
|
|
||||||
|
|
||||||
|
class Timer;
|
||||||
|
class SdCardManager;
|
||||||
|
|
||||||
class CoreController: public ExtendedControllerBase {
|
class CoreController: public ExtendedControllerBase {
|
||||||
public:
|
public:
|
||||||
@ -32,6 +34,7 @@ public:
|
|||||||
|
|
||||||
|
|
||||||
CoreController(object_id_t objectId);
|
CoreController(object_id_t objectId);
|
||||||
|
virtual~ CoreController();
|
||||||
|
|
||||||
ReturnValue_t initialize() override;
|
ReturnValue_t initialize() override;
|
||||||
|
|
||||||
@ -56,11 +59,32 @@ private:
|
|||||||
ReturnValue_t checkModeCommand(Mode_t mode, Submode_t submode,
|
ReturnValue_t checkModeCommand(Mode_t mode, Submode_t submode,
|
||||||
uint32_t *msToReachTheMode);
|
uint32_t *msToReachTheMode);
|
||||||
|
|
||||||
ReturnValue_t initSdCard();
|
// States for non-blocking setup
|
||||||
ReturnValue_t sdCardSetup(SdCardManager& sdcMan, SdCardManager::SdStatusPair& statusPair,
|
enum class SdStates {
|
||||||
sd::SdCard sdCard, sd::SdStatus status, std::string sdString);
|
NONE,
|
||||||
|
START,
|
||||||
|
GET_INFO,
|
||||||
|
SET_STATE,
|
||||||
|
MOUNT,
|
||||||
|
UPDATE_INFO,
|
||||||
|
DONE
|
||||||
|
};
|
||||||
|
static constexpr bool BLOCKING_SD_INIT = true;
|
||||||
|
|
||||||
|
SdCardManager* sdcMan = nullptr;
|
||||||
|
uint16_t sdcInitCycleCount = 0;
|
||||||
|
ReturnValue_t initSdCardBlocking();
|
||||||
|
ReturnValue_t initSdCardNonBlocking();
|
||||||
|
sd::SdCard preferredSdCard = sd::SdCard::NONE;
|
||||||
|
SdStates state = SdStates::START;
|
||||||
|
SdCardManager::SdStatusPair statusPair;
|
||||||
|
bool commandExecuted = true;
|
||||||
|
bool sdInitFinished = false;
|
||||||
|
ReturnValue_t sdCardSetup(SdCardManager::SdStatusPair& statusPair,
|
||||||
|
sd::SdCard sdCard, std::string sdString);
|
||||||
ReturnValue_t sdCardColdRedundantInit(SdCardManager* sdcMan,
|
ReturnValue_t sdCardColdRedundantInit(SdCardManager* sdcMan,
|
||||||
SdCardManager::SdStatusPair& statusPair);
|
SdCardManager::SdStatusPair& statusPair, SdStates step = SdStates::NONE);
|
||||||
|
void determinePreferredSdCard();
|
||||||
|
|
||||||
ReturnValue_t initVersionFile();
|
ReturnValue_t initVersionFile();
|
||||||
ReturnValue_t initBootCopy();
|
ReturnValue_t initBootCopy();
|
||||||
|
@ -77,11 +77,11 @@ void FileSystemHandler::fileSystemCheckup() {
|
|||||||
sd::SdCard preferredSdCard;
|
sd::SdCard preferredSdCard;
|
||||||
sdcMan->getPreferredSdCard(preferredSdCard);
|
sdcMan->getPreferredSdCard(preferredSdCard);
|
||||||
if((preferredSdCard == sd::SdCard::SLOT_0) and
|
if((preferredSdCard == sd::SdCard::SLOT_0) and
|
||||||
(statusPair.first == sd::SdStatus::MOUNTED)) {
|
(statusPair.first == sd::SdState::MOUNTED)) {
|
||||||
currentMountPrefix = SdCardManager::SD_0_MOUNT_POINT;
|
currentMountPrefix = SdCardManager::SD_0_MOUNT_POINT;
|
||||||
}
|
}
|
||||||
else if((preferredSdCard == sd::SdCard::SLOT_1) and
|
else if((preferredSdCard == sd::SdCard::SLOT_1) and
|
||||||
(statusPair.second == sd::SdStatus::MOUNTED)) {
|
(statusPair.second == sd::SdState::MOUNTED)) {
|
||||||
currentMountPrefix = SdCardManager::SD_1_MOUNT_POINT;
|
currentMountPrefix = SdCardManager::SD_1_MOUNT_POINT;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
#include <fsfw/osal/linux/Timer.h>
|
||||||
#include "SdCardManager.h"
|
#include "SdCardManager.h"
|
||||||
#include "scratchApi.h"
|
#include "scratchApi.h"
|
||||||
|
|
||||||
@ -52,39 +53,35 @@ ReturnValue_t SdCardManager::switchOnSdCard(sd::SdCard sdCard, bool doMountSdCar
|
|||||||
return HasReturnvaluesIF::RETURN_FAILED;
|
return HasReturnvaluesIF::RETURN_FAILED;
|
||||||
}
|
}
|
||||||
|
|
||||||
sd::SdStatus targetStatus;
|
sd::SdState currentState;
|
||||||
if(sdCard == sd::SdCard::SLOT_0) {
|
if(sdCard == sd::SdCard::SLOT_0) {
|
||||||
targetStatus = statusPair->first;
|
currentState = statusPair->first;
|
||||||
}
|
}
|
||||||
else if(sdCard == sd::SdCard::SLOT_1) {
|
else if(sdCard == sd::SdCard::SLOT_1) {
|
||||||
targetStatus = statusPair->second;
|
currentState = statusPair->second;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
// Should not happen
|
// Should not happen
|
||||||
targetStatus = sd::SdStatus::OFF;
|
currentState = sd::SdState::OFF;
|
||||||
}
|
}
|
||||||
|
|
||||||
auto switchCall = [&]() {
|
if(currentState == sd::SdState::ON) {
|
||||||
if(targetStatus == sd::SdStatus::ON) {
|
if(not doMountSdCard) {
|
||||||
if(not doMountSdCard) {
|
return ALREADY_ON;
|
||||||
return ALREADY_ON;
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
return mountSdCard(sdCard);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if(targetStatus == sd::SdStatus::MOUNTED) {
|
|
||||||
return ALREADY_MOUNTED;
|
|
||||||
}
|
|
||||||
else if(targetStatus == sd::SdStatus::OFF) {
|
|
||||||
return setSdCardState(sdCard, true);
|
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
return HasReturnvaluesIF::RETURN_FAILED;
|
return mountSdCard(sdCard);
|
||||||
}
|
}
|
||||||
};
|
}
|
||||||
|
else if(currentState == sd::SdState::MOUNTED) {
|
||||||
result = switchCall();
|
result = ALREADY_MOUNTED;
|
||||||
|
}
|
||||||
|
else if(currentState == sd::SdState::OFF) {
|
||||||
|
result = setSdCardState(sdCard, true);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
result = HasReturnvaluesIF::RETURN_FAILED;
|
||||||
|
}
|
||||||
|
|
||||||
if(result != HasReturnvaluesIF::RETURN_OK or not doMountSdCard) {
|
if(result != HasReturnvaluesIF::RETURN_OK or not doMountSdCard) {
|
||||||
return result;
|
return result;
|
||||||
@ -95,7 +92,7 @@ ReturnValue_t SdCardManager::switchOnSdCard(sd::SdCard sdCard, bool doMountSdCar
|
|||||||
|
|
||||||
ReturnValue_t SdCardManager::switchOffSdCard(sd::SdCard sdCard, bool doUnmountSdCard,
|
ReturnValue_t SdCardManager::switchOffSdCard(sd::SdCard sdCard, bool doUnmountSdCard,
|
||||||
SdStatusPair* statusPair) {
|
SdStatusPair* statusPair) {
|
||||||
std::pair<sd::SdStatus, sd::SdStatus> active;
|
std::pair<sd::SdState, sd::SdState> active;
|
||||||
ReturnValue_t result = getSdCardActiveStatus(active);
|
ReturnValue_t result = getSdCardActiveStatus(active);
|
||||||
if(result != HasReturnvaluesIF::RETURN_OK) {
|
if(result != HasReturnvaluesIF::RETURN_OK) {
|
||||||
return result;
|
return result;
|
||||||
@ -107,12 +104,12 @@ ReturnValue_t SdCardManager::switchOffSdCard(sd::SdCard sdCard, bool doUnmountSd
|
|||||||
return HasReturnvaluesIF::RETURN_FAILED;
|
return HasReturnvaluesIF::RETURN_FAILED;
|
||||||
}
|
}
|
||||||
if(sdCard == sd::SdCard::SLOT_0) {
|
if(sdCard == sd::SdCard::SLOT_0) {
|
||||||
if(active.first == sd::SdStatus::OFF) {
|
if(active.first == sd::SdState::OFF) {
|
||||||
return ALREADY_OFF;
|
return ALREADY_OFF;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if(sdCard == sd::SdCard::SLOT_1) {
|
else if(sdCard == sd::SdCard::SLOT_1) {
|
||||||
if(active.second == sd::SdStatus::OFF) {
|
if(active.second == sd::SdState::OFF) {
|
||||||
return ALREADY_OFF;
|
return ALREADY_OFF;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -268,7 +265,7 @@ ReturnValue_t SdCardManager::sanitizeState(SdStatusPair* statusPair, sd::SdCard
|
|||||||
getSdCardActiveStatus(*statusPair);
|
getSdCardActiveStatus(*statusPair);
|
||||||
}
|
}
|
||||||
|
|
||||||
if(statusPair->first == sd::SdStatus::ON) {
|
if(statusPair->first == sd::SdState::ON) {
|
||||||
return mountSdCard(prefSdCard);
|
return mountSdCard(prefSdCard);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -280,7 +277,7 @@ void SdCardManager::resetState() {
|
|||||||
currentOp = Operations::IDLE;
|
currentOp = Operations::IDLE;
|
||||||
}
|
}
|
||||||
|
|
||||||
void SdCardManager::processSdStatusLine(std::pair<sd::SdStatus, sd::SdStatus> &active,
|
void SdCardManager::processSdStatusLine(std::pair<sd::SdState, sd::SdState> &active,
|
||||||
std::string& line, uint8_t& idx, sd::SdCard& currentSd) {
|
std::string& line, uint8_t& idx, sd::SdCard& currentSd) {
|
||||||
using namespace std;
|
using namespace std;
|
||||||
istringstream iss(line);
|
istringstream iss(line);
|
||||||
@ -302,28 +299,28 @@ void SdCardManager::processSdStatusLine(std::pair<sd::SdStatus, sd::SdStatus> &a
|
|||||||
|
|
||||||
if(word == "on") {
|
if(word == "on") {
|
||||||
if(currentSd == sd::SdCard::SLOT_0) {
|
if(currentSd == sd::SdCard::SLOT_0) {
|
||||||
active.first = sd::SdStatus::ON;
|
active.first = sd::SdState::ON;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
active.second = sd::SdStatus::ON;
|
active.second = sd::SdState::ON;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (word == "off") {
|
else if (word == "off") {
|
||||||
if(currentSd == sd::SdCard::SLOT_0) {
|
if(currentSd == sd::SdCard::SLOT_0) {
|
||||||
active.first = sd::SdStatus::OFF;
|
active.first = sd::SdState::OFF;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
active.second = sd::SdStatus::OFF;
|
active.second = sd::SdState::OFF;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if(mountLine) {
|
if(mountLine) {
|
||||||
if(currentSd == sd::SdCard::SLOT_0) {
|
if(currentSd == sd::SdCard::SLOT_0) {
|
||||||
active.first = sd::SdStatus::MOUNTED;
|
active.first = sd::SdState::MOUNTED;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
active.second = sd::SdStatus::MOUNTED;
|
active.second = sd::SdState::MOUNTED;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -389,8 +386,16 @@ SdCardManager::OpStatus SdCardManager::checkCurrentOp(Operations ¤tOp) {
|
|||||||
currentOp = this->currentOp;
|
currentOp = this->currentOp;
|
||||||
bool bytesRead = false;
|
bool bytesRead = false;
|
||||||
|
|
||||||
|
Timer timer;
|
||||||
|
timer.setTimer(200);
|
||||||
|
uint32_t remainingTimeMs = 0;
|
||||||
while(true) {
|
while(true) {
|
||||||
ReturnValue_t result = cmdExecutor.check(bytesRead);
|
ReturnValue_t result = cmdExecutor.check(bytesRead);
|
||||||
|
timer.getTimer(&remainingTimeMs);
|
||||||
|
if(remainingTimeMs == 0) {
|
||||||
|
sif::error << "SdCardManager::checkCurrentOp: Timeout!" << std::endl;
|
||||||
|
return OpStatus::FAIL;
|
||||||
|
}
|
||||||
switch(result) {
|
switch(result) {
|
||||||
case(CommandExecutor::BYTES_READ): {
|
case(CommandExecutor::BYTES_READ): {
|
||||||
continue;
|
continue;
|
||||||
|
@ -42,7 +42,7 @@ public:
|
|||||||
FAIL
|
FAIL
|
||||||
};
|
};
|
||||||
|
|
||||||
using SdStatusPair = std::pair<sd::SdStatus, sd::SdStatus>;
|
using SdStatusPair = std::pair<sd::SdState, sd::SdState>;
|
||||||
|
|
||||||
static constexpr uint8_t INTERFACE_ID = CLASS_ID::SD_CARD_MANAGER;
|
static constexpr uint8_t INTERFACE_ID = CLASS_ID::SD_CARD_MANAGER;
|
||||||
|
|
||||||
@ -197,7 +197,7 @@ public:
|
|||||||
private:
|
private:
|
||||||
CommandExecutor cmdExecutor;
|
CommandExecutor cmdExecutor;
|
||||||
Operations currentOp = Operations::IDLE;
|
Operations currentOp = Operations::IDLE;
|
||||||
bool blocking = true;
|
bool blocking = false;
|
||||||
|
|
||||||
SdCardManager();
|
SdCardManager();
|
||||||
|
|
||||||
|
@ -5,7 +5,7 @@
|
|||||||
|
|
||||||
namespace sd {
|
namespace sd {
|
||||||
|
|
||||||
enum SdStatus: uint8_t {
|
enum SdState: uint8_t {
|
||||||
OFF = 0,
|
OFF = 0,
|
||||||
ON = 1,
|
ON = 1,
|
||||||
// A mounted SD card is on as well
|
// A mounted SD card is on as well
|
||||||
|
2
fsfw
2
fsfw
@ -1 +1 @@
|
|||||||
Subproject commit 4202205182aa007b9be5279ce40056e18f0b8782
|
Subproject commit 6073abb12db23decbc9d69f4b6342f4f6f1247bd
|
Loading…
Reference in New Issue
Block a user