#pragma once #include "DFTElement.h" namespace storm { namespace storage { template class DFTGate : public DFTElement { using DFTElementPointer = std::shared_ptr>; using DFTElementVector = std::vector; protected: DFTElementVector mChildren; public: DFTGate(size_t id, std::string const& name, DFTElementVector const& children) : DFTElement(id, name), mChildren(children) {} virtual ~DFTGate() {} void pushBackChild(DFTElementPointer elem) { return mChildren.push_back(elem); } size_t nrChildren() const override { return mChildren.size(); } DFTElementVector const& children() const { return mChildren; } virtual bool isGate() const override { return true; } bool isDynamicGate() const { return !isStaticGateType(this->type()); } virtual std::string typestring() const = 0; virtual void checkFails(storm::storage::DFTState& state, DFTStateSpaceGenerationQueues& queues) const = 0; virtual void checkFailsafe(storm::storage::DFTState& state, DFTStateSpaceGenerationQueues& queues) const = 0; virtual void extendSpareModule(std::set& elementsInSpareModule) const override { if (!this->isSpareGate()) { DFTElement::extendSpareModule(elementsInSpareModule); for (auto const& child : mChildren) { if (elementsInSpareModule.count(child->id()) == 0) { elementsInSpareModule.insert(child->id()); child->extendSpareModule(elementsInSpareModule); } } } } virtual std::vector independentUnit() const override { std::set unit = {this->mId}; for(auto const& child : mChildren) { child->extendUnit(unit); } return std::vector(unit.begin(), unit.end()); } virtual void extendUnit(std::set& unit) const override { DFTElement::extendUnit(unit); for(auto const& child : mChildren) { child->extendUnit(unit); } } virtual std::vector independentSubDft(bool blockParents, bool sparesAsLeaves = false) const override { auto prelRes = DFTElement::independentSubDft(blockParents); if(prelRes.empty()) { // No elements (especially not this->id) in the prelimanry result, so we know already that it is not a subdft. return prelRes; } std::set unit(prelRes.begin(), prelRes.end()); std::vector pids = this->parentIds(); for(auto const& child : mChildren) { child->extendSubDft(unit, pids, blockParents, sparesAsLeaves); if(unit.empty()) { // Parent in the subdft, ie it is *not* a subdft break; } } return std::vector(unit.begin(), unit.end()); } virtual void extendSubDft(std::set& elemsInSubtree, std::vector const& parentsOfSubRoot, bool blockParents, bool sparesAsLeaves) const override { if(elemsInSubtree.count(this->id()) > 0) return; DFTElement::extendSubDft(elemsInSubtree, parentsOfSubRoot, blockParents, sparesAsLeaves); if(elemsInSubtree.empty()) { // Parent in the subdft, ie it is *not* a subdft return; } for(auto const& child : mChildren) { child->extendSubDft(elemsInSubtree, parentsOfSubRoot, blockParents, sparesAsLeaves); if(elemsInSubtree.empty()) { // Parent in the subdft, ie it is *not* a subdft return; } } } virtual std::string toString() const override { std::stringstream stream; stream << "{" << this->name() << "} " << typestring() << "( "; typename DFTElementVector::const_iterator it = mChildren.begin(); stream << (*it)->name(); ++it; while(it != mChildren.end()) { stream << ", " << (*it)->name(); ++it; } stream << ")"; return stream.str(); } virtual bool checkDontCareAnymore(storm::storage::DFTState& state, DFTStateSpaceGenerationQueues& queues) const override { if(DFTElement::checkDontCareAnymore(state, queues)) { childrenDontCare(state, queues); return true; } return false; } protected: void fail(DFTState& state, DFTStateSpaceGenerationQueues& queues) const { for(std::shared_ptr parent : this->mParents) { if(state.isOperational(parent->id())) { queues.propagateFailure(parent); } } for(std::shared_ptr> restr : this->mRestrictions) { queues.checkRestrictionLater(restr); } state.setFailed(this->mId); this->childrenDontCare(state, queues); } void failsafe(DFTState& state, DFTStateSpaceGenerationQueues& queues) const { for(std::shared_ptr parent : this->mParents) { if(state.isOperational(parent->id())) { queues.propagateFailsafe(parent); } } state.setFailsafe(this->mId); this->childrenDontCare(state, queues); } void childrenDontCare(DFTState& state, DFTStateSpaceGenerationQueues& queues) const { queues.propagateDontCare(mChildren); } bool hasFailsafeChild(DFTState& state) const { for(auto const& child : mChildren) { if(state.isFailsafe(child->id())) { return true; } } return false; } bool hasFailedChild(DFTState& state) const { for(auto const& child : mChildren) { if(state.hasFailed(child->id())) { return true; } } return false; } }; } }