#include "DFTState.h" #include "DFTElements.h" #include "DFT.h" namespace storm { namespace storage { template DFTState::DFTState(DFT const& dft, DFTStateGenerationInfo const& stateGenerationInfo, size_t id) : mStatus(dft.stateVectorSize()), mId(id), mPseudoState(false), mDft(dft), mStateGenerationInfo(stateGenerationInfo), exploreHeuristic() { // TODO Matthias: use construct() // Initialize uses for(size_t spareId : mDft.getSpareIndices()) { std::shared_ptr const> elem = mDft.getGate(spareId); STORM_LOG_ASSERT(elem->isSpareGate(), "Element is no spare gate."); STORM_LOG_ASSERT(elem->nrChildren() > 0, "Element has no child."); this->setUses(spareId, elem->children()[0]->id()); } for (auto elem : mDft.getBasicElements()) { mCurrentlyNotFailableBE.push_back(elem->id()); } // Initialize activation propagateActivation(mDft.getTopLevelIndex()); std::vector alwaysActiveBEs = mDft.nonColdBEs(); mCurrentlyFailableBE.insert(mCurrentlyFailableBE.end(), alwaysActiveBEs.begin(), alwaysActiveBEs.end()); // Remove always active BEs from currently not failable BEs for (size_t id : alwaysActiveBEs) { auto it = std::find(mCurrentlyNotFailableBE.begin(), mCurrentlyNotFailableBE.end(), id); STORM_LOG_ASSERT(it != mCurrentlyNotFailableBE.end(), "Id not found."); mCurrentlyNotFailableBE.erase(it); } sortFailableBEs(); } template DFTState::DFTState(storm::storage::BitVector const& status, DFT const& dft, DFTStateGenerationInfo const& stateGenerationInfo, size_t id) : mStatus(status), mId(id), mPseudoState(true), mDft(dft), mStateGenerationInfo(stateGenerationInfo), exploreHeuristic() { // Intentionally left empty } template void DFTState::construct() { STORM_LOG_TRACE("Construct concrete state from pseudo state " << mDft.getStateString(mStatus, mStateGenerationInfo, mId)); // Clear information from pseudo state mCurrentlyFailableBE.clear(); mCurrentlyNotFailableBE.clear(); mFailableDependencies.clear(); mUsedRepresentants.clear(); STORM_LOG_ASSERT(mPseudoState, "Only pseudo states can be constructed."); for(size_t index = 0; index < mDft.nrElements(); ++index) { // Initialize currently failable BE if (mDft.isBasicElement(index) && isOperational(index)) { std::shared_ptr> be = mDft.getBasicElement(index); if ((!be->isColdBasicElement() && be->canFail()) || !mDft.hasRepresentant(index) || isActive(mDft.getRepresentant(index))) { mCurrentlyFailableBE.push_back(index); STORM_LOG_TRACE("Currently failable: " << mDft.getBasicElement(index)->toString()); } else { // BE currently is not failable mCurrentlyNotFailableBE.push_back(index); STORM_LOG_TRACE("Currently not failable: " << mDft.getBasicElement(index)->toString()); } } else if (mDft.getElement(index)->isSpareGate()) { // Initialize used representants uint_fast64_t useId = uses(index); mUsedRepresentants.push_back(useId); STORM_LOG_TRACE("Spare " << index << " uses " << useId); } } sortFailableBEs(); // Initialize failable dependencies for (size_t dependencyId : mDft.getDependencies()) { std::shared_ptr const> dependency = mDft.getDependency(dependencyId); STORM_LOG_ASSERT(dependencyId == dependency->id(), "Ids do not match."); if (hasFailed(dependency->triggerEvent()->id()) && getElementState(dependency->dependentEvent()->id()) == DFTElementState::Operational) { mFailableDependencies.push_back(dependencyId); STORM_LOG_TRACE("New dependency failure: " << dependency->toString()); } } mPseudoState = false; } template std::shared_ptr> DFTState::copy() const { std::shared_ptr> stateCopy = std::make_shared>(*this); stateCopy->exploreHeuristic = storm::builder::DFTExplorationHeuristic(); return stateCopy; } template DFTElementState DFTState::getElementState(size_t id) const { return static_cast(getElementStateInt(id)); } template DFTDependencyState DFTState::getDependencyState(size_t id) const { return static_cast(getElementStateInt(id)); } template int DFTState::getElementStateInt(size_t id) const { return mStatus.getAsInt(mStateGenerationInfo.getStateIndex(id), 2); } template int DFTState::getElementStateInt(storm::storage::BitVector const& state, size_t indexId) { return state.getAsInt(indexId, 2); } template size_t DFTState::getId() const { return mId; } template void DFTState::setId(size_t id) { mId = id; } template bool DFTState::isOperational(size_t id) const { return getElementState(id) == DFTElementState::Operational; } template bool DFTState::hasFailed(size_t id) const { return mStatus[mStateGenerationInfo.getStateIndex(id)]; } template bool DFTState::hasFailed(storm::storage::BitVector const& state, size_t indexId) { return state[indexId]; } template bool DFTState::isFailsafe(size_t id) const { return mStatus[mStateGenerationInfo.getStateIndex(id)+1]; } template bool DFTState::isFailsafe(storm::storage::BitVector const& state, size_t indexId) { return state[indexId+1]; } template bool DFTState::dontCare(size_t id) const { return getElementState(id) == DFTElementState::DontCare; } template bool DFTState::dependencyTriggered(size_t id) const { return getElementStateInt(id) > 0; } template bool DFTState::dependencySuccessful(size_t id) const { return mStatus[mStateGenerationInfo.getStateIndex(id)]; } template bool DFTState::dependencyUnsuccessful(size_t id) const { return mStatus[mStateGenerationInfo.getStateIndex(id)+1]; } template void DFTState::setFailed(size_t id) { mStatus.set(mStateGenerationInfo.getStateIndex(id)); } template void DFTState::setFailsafe(size_t id) { mStatus.set(mStateGenerationInfo.getStateIndex(id)+1); } template void DFTState::setDontCare(size_t id) { if (mDft.isRepresentative(id)) { // Activate dont care element activate(id); } mStatus.setFromInt(mStateGenerationInfo.getStateIndex(id), 2, static_cast(DFTElementState::DontCare) ); } template void DFTState::setDependencySuccessful(size_t id) { // Only distinguish between passive and dont care dependencies //mStatus.set(mStateGenerationInfo.getStateIndex(id)); setDependencyDontCare(id); } template void DFTState::setDependencyUnsuccessful(size_t id) { // Only distinguish between passive and dont care dependencies //mStatus.set(mStateGenerationInfo.getStateIndex(id)+1); setDependencyDontCare(id); } template void DFTState::setDependencyDontCare(size_t id) { mStatus.setFromInt(mStateGenerationInfo.getStateIndex(id), 2, static_cast(DFTDependencyState::DontCare)); } template void DFTState::beNoLongerFailable(size_t id) { auto it = std::find(mCurrentlyFailableBE.begin(), mCurrentlyFailableBE.end(), id); if(it != mCurrentlyFailableBE.end()) { mCurrentlyFailableBE.erase(it); } } template bool DFTState::updateFailableDependencies(size_t id) { if (!hasFailed(id)) { return false; } for (auto dependency : mDft.getElement(id)->outgoingDependencies()) { STORM_LOG_ASSERT(dependency->triggerEvent()->id() == id, "Ids do not match."); if (getElementState(dependency->dependentEvent()->id()) == DFTElementState::Operational) { STORM_LOG_ASSERT(!isFailsafe(dependency->dependentEvent()->id()), "Dependent event is failsafe."); mFailableDependencies.push_back(dependency->id()); STORM_LOG_TRACE("New dependency failure: " << dependency->toString()); } } return nrFailableDependencies() > 0; } template void DFTState::updateDontCareDependencies(size_t id) { STORM_LOG_ASSERT(mDft.isBasicElement(id), "Element is no BE."); STORM_LOG_ASSERT(hasFailed(id), "Element has not failed."); for (auto dependency : mDft.getBasicElement(id)->ingoingDependencies()) { STORM_LOG_ASSERT(dependency->dependentEvent()->id() == id, "Ids do not match."); setDependencyDontCare(dependency->id()); } } template ValueType DFTState::getBERate(size_t id, bool considerPassive) const { STORM_LOG_ASSERT(mDft.isBasicElement(id), "Element is no BE."); if (considerPassive && mDft.hasRepresentant(id) && !isActive(mDft.getRepresentant(id))) { return mDft.getBasicElement(id)->passiveFailureRate(); } else { return mDft.getBasicElement(id)->activeFailureRate(); } } template ValueType DFTState::getFailableBERate(size_t index) const { STORM_LOG_ASSERT(index < nrFailableBEs(), "Index invalid."); return getBERate(mCurrentlyFailableBE[index], true); } template ValueType DFTState::getNotFailableBERate(size_t index) const { STORM_LOG_ASSERT(index < nrNotFailableBEs(), "Index invalid."); STORM_LOG_ASSERT(storm::utility::isZero(mDft.getBasicElement(mCurrentlyNotFailableBE[index])->activeFailureRate()) || (mDft.hasRepresentant(mCurrentlyNotFailableBE[index]) && !isActive(mDft.getRepresentant(mCurrentlyNotFailableBE[index]))), "BE " << mCurrentlyNotFailableBE[index] << " can fail"); // Use active failure rate as passive failure rate is 0. return getBERate(mCurrentlyNotFailableBE[index], false); } template std::pair const>, bool> DFTState::letNextBEFail(size_t index) { STORM_LOG_TRACE("currently failable: " << getCurrentlyFailableString()); if (nrFailableDependencies() > 0) { // Consider failure due to dependency STORM_LOG_ASSERT(index < nrFailableDependencies(), "Index invalid."); std::shared_ptr const> dependency = mDft.getDependency(mFailableDependencies[index]); std::pair const>,bool> res(mDft.getBasicElement(dependency->dependentEvent()->id()), true); mFailableDependencies.erase(mFailableDependencies.begin() + index); setFailed(res.first->id()); setDependencySuccessful(dependency->id()); return res; } else { // Consider "normal" failure STORM_LOG_ASSERT(index < nrFailableBEs(), "Index invalid."); std::pair const>,bool> res(mDft.getBasicElement(mCurrentlyFailableBE[index]), false); STORM_LOG_ASSERT(res.first->canFail(), "Element cannot fail."); mCurrentlyFailableBE.erase(mCurrentlyFailableBE.begin() + index); setFailed(res.first->id()); return res; } } template void DFTState::letDependencyBeUnsuccessful(size_t index) { STORM_LOG_ASSERT(nrFailableDependencies() > 0 && index < nrFailableDependencies(), "Index invalid."); std::shared_ptr const> dependency = mDft.getDependency(getDependencyId(index)); mFailableDependencies.erase(mFailableDependencies.begin() + index); setDependencyUnsuccessful(dependency->id()); } template void DFTState::activate(size_t repr) { size_t activationIndex = mStateGenerationInfo.getSpareActivationIndex(repr); mStatus.set(activationIndex); } template bool DFTState::isActive(size_t id) const { STORM_LOG_ASSERT(mDft.isRepresentative(id), "Element is no representative."); return mStatus[mStateGenerationInfo.getSpareActivationIndex(id)]; } template void DFTState::propagateActivation(size_t representativeId) { if (representativeId != mDft.getTopLevelIndex()) { activate(representativeId); } for(size_t elem : mDft.module(representativeId)) { if(mDft.isBasicElement(elem) && isOperational(elem)) { std::shared_ptr> be = mDft.getBasicElement(elem); if (be->isColdBasicElement() && be->canFail()) { // Add to failable BEs mCurrentlyFailableBE.push_back(elem); // Remove from not failable BEs auto it = std::find(mCurrentlyNotFailableBE.begin(), mCurrentlyNotFailableBE.end(), elem); STORM_LOG_ASSERT(it != mCurrentlyNotFailableBE.end(), "Element " << mDft.getElement(elem)->name() << " not found."); mCurrentlyNotFailableBE.erase(it); } } else if (mDft.getElement(elem)->isSpareGate() && !isActive(uses(elem))) { propagateActivation(uses(elem)); } } sortFailableBEs(); } template uint_fast64_t DFTState::uses(size_t id) const { size_t nrUsedChild = extractUses(mStateGenerationInfo.getSpareUsageIndex(id)); if (nrUsedChild == mDft.getMaxSpareChildCount()) { return id; } else { return mDft.getChild(id, nrUsedChild); } } template uint_fast64_t DFTState::extractUses(size_t from) const { STORM_LOG_ASSERT(mStateGenerationInfo.usageInfoBits() < 64, "UsageInfoBit size too large."); return mStatus.getAsInt(from, mStateGenerationInfo.usageInfoBits()); } template bool DFTState::isUsed(size_t child) const { return (std::find(mUsedRepresentants.begin(), mUsedRepresentants.end(), child) != mUsedRepresentants.end()); } template void DFTState::setUses(size_t spareId, size_t child) { mStatus.setFromInt(mStateGenerationInfo.getSpareUsageIndex(spareId), mStateGenerationInfo.usageInfoBits(), mDft.getNrChild(spareId, child)); mUsedRepresentants.push_back(child); } template void DFTState::finalizeUses(size_t spareId) { STORM_LOG_ASSERT(hasFailed(spareId), "Spare has not failed."); mStatus.setFromInt(mStateGenerationInfo.getSpareUsageIndex(spareId), mStateGenerationInfo.usageInfoBits(), mDft.getMaxSpareChildCount()); } template bool DFTState::hasOperationalPostSeqElements(size_t id) const { STORM_LOG_ASSERT(!mDft.isDependency(id), "Element is dependency."); STORM_LOG_ASSERT(!mDft.isRestriction(id), "Element is restriction."); auto const& postIds = mStateGenerationInfo.seqRestrictionPostElements(id); for(size_t id : postIds) { if(isOperational(id)) { return true; } } return false; } template bool DFTState::claimNew(size_t spareId, size_t currentlyUses, std::vector>> const& children) { auto it = children.begin(); while ((*it)->id() != currentlyUses) { STORM_LOG_ASSERT(it != children.end(), "Currently used element not found."); ++it; } ++it; while(it != children.end()) { size_t childId = (*it)->id(); if(!hasFailed(childId) && !isUsed(childId)) { setUses(spareId, childId); if(isActive(currentlyUses)) { propagateActivation(childId); } return true; } ++it; } return false; } template bool DFTState::orderBySymmetry() { bool changed = false; for (size_t pos = 0; pos < mStateGenerationInfo.getSymmetrySize(); ++pos) { // Check each symmetry size_t length = mStateGenerationInfo.getSymmetryLength(pos); std::vector symmetryIndices = mStateGenerationInfo.getSymmetryIndices(pos); // Sort symmetry group in decreasing order by bubble sort // TODO use better algorithm? size_t tmp; size_t n = symmetryIndices.size(); do { tmp = 0; for (size_t i = 1; i < n; ++i) { if (mStatus.compareAndSwap(symmetryIndices[i-1], symmetryIndices[i], length)) { tmp = i; changed = true; } } n = tmp; } while (n > 0); } if (changed) { mPseudoState = true; } return changed; } template void DFTState::sortFailableBEs() { std::sort(mCurrentlyFailableBE.begin( ), mCurrentlyFailableBE.end( ), [&](size_t const& lhs, size_t const& rhs) { // Sort decreasing return getBERate(rhs, true) < getBERate(lhs, true); }); } // Explicitly instantiate the class. template class DFTState; #ifdef STORM_HAVE_CARL template class DFTState; #endif } }