#include "src/builder/ExplicitDFTModelBuilder.h" #include #include #include namespace storm { namespace builder { template ExplicitDFTModelBuilder::ModelComponents::ModelComponents() : transitionMatrix(), stateLabeling(), rewardModels(), choiceLabeling() { // Intentionally left empty. } template std::shared_ptr> ExplicitDFTModelBuilder::buildCTMC() { // Initialize storm::storage::DFTState state(mDft, newIndex++); mStates.insert(state); std::queue> stateQueue; stateQueue.push(state); bool deterministicModel = true; storm::storage::SparseMatrixBuilder transitionMatrixBuilder(0, 0, 0, false, !deterministicModel, 0); // Begin model generation exploreStates(stateQueue, transitionMatrixBuilder); STORM_LOG_DEBUG("Generated " << mStates.size() << " states"); // Build CTMC ModelComponents modelComponents; // Build transition matrix modelComponents.transitionMatrix = transitionMatrixBuilder.build(); STORM_LOG_DEBUG("TransitionMatrix: " << std::endl << modelComponents.transitionMatrix); // Build state labeling modelComponents.stateLabeling = storm::models::sparse::StateLabeling(mStates.size()); // Initial state is always first state without any failure modelComponents.stateLabeling.addLabel("init"); modelComponents.stateLabeling.addLabelToState("init", 0); // Label all states corresponding to their status (failed, failsafe, failed BE) modelComponents.stateLabeling.addLabel("failed"); modelComponents.stateLabeling.addLabel("failsafe"); // Collect labels for all BE std::vector>> basicElements = mDft.getBasicElements(); for (std::shared_ptr> elem : basicElements) { modelComponents.stateLabeling.addLabel(elem->name() + "_fail"); } for (storm::storage::DFTState state : mStates) { if (mDft.hasFailed(state)) { modelComponents.stateLabeling.addLabelToState("failed", state.getId()); } if (mDft.isFailsafe(state)) { modelComponents.stateLabeling.addLabelToState("failsafe", state.getId()); }; // Set fail status for each BE for (std::shared_ptr> elem : basicElements) { if (state.hasFailed(elem->id())) { modelComponents.stateLabeling.addLabelToState(elem->name() + "_fail", state.getId()); } } } std::shared_ptr> model; model = std::shared_ptr>(new storm::models::sparse::Ctmc(std::move(modelComponents.transitionMatrix), std::move(modelComponents.stateLabeling))); model->printModelInformationToStream(std::cout); return model; } template void ExplicitDFTModelBuilder::exploreStates(std::queue>& stateQueue, storm::storage::SparseMatrixBuilder& transitionMatrixBuilder) { std::map outgoingTransitions; while (!stateQueue.empty()) { // Initialization outgoingTransitions.clear(); ValueType sum = storm::utility::zero(); // Consider next state storm::storage::DFTState state = stateQueue.front(); stateQueue.pop(); size_t smallest = 0; // Add self loop for target states if (mDft.hasFailed(state) || mDft.isFailsafe(state)) { transitionMatrixBuilder.addNextValue(state.getId(), state.getId(), storm::utility::one()); STORM_LOG_TRACE("Added self loop for " << state.getId()); // No further exploration required continue; } // Let BE fail while (smallest < state.nrFailableBEs()) { STORM_LOG_TRACE("exploring from: " << mDft.getStateString(state)); storm::storage::DFTState newState(state); std::pair>, bool> nextBEPair = newState.letNextBEFail(smallest++); std::shared_ptr> nextBE = nextBEPair.first; if (nextBE == nullptr) { break; } STORM_LOG_TRACE("with the failure of: " << nextBE->name() << " [" << nextBE->id() << "]"); storm::storage::DFTStateSpaceGenerationQueues queues; for (DFTGatePointer parent : nextBE->parents()) { if (newState.isOperational(parent->id())) { queues.propagateFailure(parent); } } while (!queues.failurePropagationDone()) { DFTGatePointer next = queues.nextFailurePropagation(); next->checkFails(newState, queues); } while (!queues.failsafePropagationDone()) { DFTGatePointer next = queues.nextFailsafePropagation(); next->checkFailsafe(newState, queues); } while (!queues.dontCarePropagationDone()) { DFTElementPointer next = queues.nextDontCarePropagation(); next->checkDontCareAnymore(newState, queues); } auto itState = mStates.find(newState); if (itState == mStates.end()) { // New state newState.setId(newIndex++); auto itInsert = mStates.insert(newState); assert(itInsert.second); itState = itInsert.first; STORM_LOG_TRACE("New state " << mDft.getStateString(newState)); // Add state to search stateQueue.push(newState); } else { // State already exists STORM_LOG_TRACE("State " << mDft.getStateString(*itState) << " already exists"); } // Set failure rate according to usage bool isUsed = true; if (mDft.hasRepresentant(nextBE->id())) { DFTElementPointer representant = mDft.getRepresentant(nextBE->id()); isUsed = newState.isUsed(representant->id()); } STORM_LOG_TRACE("BE " << nextBE->name() << " is " << (isUsed ? "used" : "not used")); ValueType rate = isUsed ? nextBE->activeFailureRate() : nextBE->passiveFailureRate(); auto resultFind = outgoingTransitions.find(itState->getId()); if (resultFind != outgoingTransitions.end()) { // Add to existing transition resultFind->second += rate; } else { // Insert new transition outgoingTransitions.insert(std::make_pair(itState->getId(), rate)); } sum += rate; } // end while failing BE // Add all transitions for (auto it = outgoingTransitions.begin(); it != outgoingTransitions.end(); ++it) { // TODO Matthias: correct? ValueType rate = it->second;// / sum; transitionMatrixBuilder.addNextValue(state.getId(), it->first, rate); STORM_LOG_TRACE("Added transition from " << state.getId() << " to " << it->first << " with " << rate); } } // end while queue } // Explicitly instantiate the class. template class ExplicitDFTModelBuilder, uint32_t>; #ifdef STORM_HAVE_CARL template class ExplicitDFTModelBuilder, uint32_t>; template class ExplicitDFTModelBuilder, uint32_t>; #endif } // namespace builder } // namespace storm