#include "DFT.h" #include #include #include "storm/exceptions/InvalidArgumentException.h" #include "storm/exceptions/NotSupportedException.h" #include "storm/exceptions/WrongFormatException.h" #include "storm/utility/iota_n.h" #include "storm/utility/vector.h" #include "storm-dft/builder/DFTBuilder.h" #include "storm-dft/storage/dft/DFTIsomorphism.h" namespace storm { namespace storage { template DFT::DFT(DFTElementVector const& elements, DFTElementPointer const& tle) : mElements(elements), mNrOfBEs(0), mNrOfSpares(0), mTopLevelIndex(tle->id()), mMaxSpareChildCount(0) { // Check that ids correspond to indices in the element vector STORM_LOG_ASSERT(elementIndicesCorrect(), "Ids incorrect."); size_t nrRepresentatives = 0; for (auto& elem : mElements) { if (isRepresentative(elem->id())) { ++nrRepresentatives; } if(elem->isBasicElement()) { ++mNrOfBEs; } else if (elem->isSpareGate()) { // Build spare modules by setting representatives and representants ++mNrOfSpares; mMaxSpareChildCount = std::max(mMaxSpareChildCount, std::static_pointer_cast>(elem)->children().size()); for(auto const& spareReprs : std::static_pointer_cast>(elem)->children()) { STORM_LOG_THROW(spareReprs->isGate() || spareReprs->isBasicElement(), storm::exceptions::WrongFormatException, "Child '" << spareReprs->name() << "' of spare '" << elem->name() << "' must be gate or BE."); std::set module = {spareReprs->id()}; spareReprs->extendSpareModule(module); std::vector sparesAndBes; for(size_t modelem : module) { if(mElements[modelem]->isSpareGate() || mElements[modelem]->isBasicElement()) { sparesAndBes.push_back(modelem); mRepresentants.insert(std::make_pair(modelem, spareReprs->id())); } } mSpareModules.insert(std::make_pair(spareReprs->id(), sparesAndBes)); } } else if (elem->isDependency()) { mDependencies.push_back(elem->id()); } } // For the top module, we assume, contrary to [Jun15], that we have all spare gates and basic elements which are not in another module. std::set topModuleSet; // Initialize with all ids. for(auto const& elem : mElements) { if (elem->isBasicElement() || elem->isSpareGate()) { topModuleSet.insert(elem->id()); } } // Erase spare modules for(auto const& module : mSpareModules) { for(auto const& index : module.second) { topModuleSet.erase(index); } } // Extend top module and insert those elements which are part of the top module and a spare module mElements[mTopLevelIndex]->extendSpareModule(topModuleSet); mTopModule = std::vector(topModuleSet.begin(), topModuleSet.end()); // Clear all spare modules where at least one element is also in the top module. // These spare modules will be activated from the beginning. if (!mTopModule.empty()) { for (auto& module : mSpareModules) { if (std::find(module.second.begin(), module.second.end(), mTopModule.front()) != module.second.end()) { STORM_LOG_WARN("Elements of spare module '" << getElement(module.first)->name() << "' also contained in top module. All elements of this spare module will be activated from the beginning on."); module.second.clear(); } } } //Reserve space for failed spares ++mMaxSpareChildCount; size_t usageInfoBits = storm::utility::math::uint64_log2(mMaxSpareChildCount) + 1; mStateVectorSize = nrElements() * 2 + mNrOfSpares * usageInfoBits + nrRepresentatives; } template DFTStateGenerationInfo DFT::buildStateGenerationInfo(storm::storage::DFTIndependentSymmetries const& symmetries) const { DFTStateGenerationInfo generationInfo(nrElements(), mMaxSpareChildCount); // Generate Pre and Post info for restrictions for(auto const& elem : mElements) { if(!elem->isDependency() && !elem->isRestriction()) { generationInfo.setRestrictionPreElements(elem->id(), elem->seqRestrictionPres()); generationInfo.setRestrictionPostElements(elem->id(), elem->seqRestrictionPosts()); } } size_t stateIndex = 0; std::queue visitQueue; storm::storage::BitVector visited(nrElements(), false); if (symmetries.groups.empty()) { // Perform DFS for whole tree visitQueue.push(mTopLevelIndex); stateIndex = performStateGenerationInfoDFS(generationInfo, visitQueue, visited, stateIndex); } else { // Generate information according to symmetries for (size_t symmetryIndex : symmetries.sortedSymmetries) { STORM_LOG_ASSERT(!visited[symmetryIndex], "Element already considered for symmetry."); auto const& symmetryGroup = symmetries.groups.at(symmetryIndex); STORM_LOG_ASSERT(!symmetryGroup.empty(), "No symmetry available."); // Insert all elements of first subtree of each symmetry size_t groupIndex = stateIndex; for (std::vector const& symmetryElement : symmetryGroup) { if (visited[symmetryElement[0]]) { groupIndex = std::min(groupIndex, generationInfo.getStateIndex(symmetryElement[0])); } else { stateIndex = generateStateInfo(generationInfo, symmetryElement[0], visited, stateIndex); } } size_t offset = stateIndex - groupIndex; // Mirror symmetries size_t noSymmetricElements = symmetryGroup.front().size(); STORM_LOG_ASSERT(noSymmetricElements > 1, "No symmetry available."); for (std::vector symmetricElements : symmetryGroup) { STORM_LOG_ASSERT(symmetricElements.size() == noSymmetricElements, "No. of symmetric elements do not coincide."); if (visited[symmetricElements[1]]) { // Elements already mirrored for (size_t index : symmetricElements) { STORM_LOG_ASSERT(visited[index], "Element not mirrored."); } continue; } // Initialize for original element size_t originalElement = symmetricElements[0]; size_t index = generationInfo.getStateIndex(originalElement); size_t activationIndex = isRepresentative(originalElement) ? generationInfo.getSpareActivationIndex(originalElement) : 0; size_t usageIndex = mElements[originalElement]->isSpareGate() ? generationInfo.getSpareUsageIndex(originalElement) : 0; // Mirror symmetry for each element for (size_t i = 1; i < symmetricElements.size(); ++i) { size_t symmetricElement = symmetricElements[i]; visited.set(symmetricElement); generationInfo.addStateIndex(symmetricElement, index + offset * i); stateIndex += 2; STORM_LOG_ASSERT((activationIndex > 0) == isRepresentative(symmetricElement), "Bits for representative incorrect."); if (activationIndex > 0) { generationInfo.addSpareActivationIndex(symmetricElement, activationIndex + offset * i); ++stateIndex; } STORM_LOG_ASSERT((usageIndex > 0) == mElements[symmetricElement]->isSpareGate(), "Bits for usage incorrect."); if (usageIndex > 0) { generationInfo.addSpareUsageIndex(symmetricElement, usageIndex + offset * i); stateIndex += generationInfo.usageInfoBits(); } } } // Store starting indices of symmetry groups std::vector symmetryIndices; for (size_t i = 0; i < noSymmetricElements; ++i) { symmetryIndices.push_back(groupIndex + i * offset); } generationInfo.addSymmetry(offset, symmetryIndices); } } // TODO symmetries in dependencies? // Consider dependencies for (size_t idDependency : getDependencies()) { std::shared_ptr const> dependency = getDependency(idDependency); visitQueue.push(dependency->id()); visitQueue.push(dependency->triggerEvent()->id()); STORM_LOG_THROW(dependency->dependentEvents().size() == 1, storm::exceptions::NotSupportedException, "Direct state generation does not support n-ary dependencies. Consider rewriting them by setting the binary dependency flag."); visitQueue.push(dependency->dependentEvents()[0]->id()); } stateIndex = performStateGenerationInfoDFS(generationInfo, visitQueue, visited, stateIndex); // Visit all remaining states for (size_t i = 0; i < visited.size(); ++i) { if (!visited[i]) { visitQueue.push(i); stateIndex = performStateGenerationInfoDFS(generationInfo, visitQueue, visited, stateIndex); } } generationInfo.generateSymmetries(symmetries); STORM_LOG_TRACE(generationInfo); STORM_LOG_ASSERT(stateIndex == mStateVectorSize, "Id incorrect."); STORM_LOG_ASSERT(visited.full(), "Not all elements considered."); return generationInfo; } template size_t DFT::generateStateInfo(DFTStateGenerationInfo& generationInfo, size_t id, storm::storage::BitVector& visited, size_t stateIndex) const { STORM_LOG_ASSERT(!visited[id], "Element already visited."); visited.set(id); // Reserve bits for element generationInfo.addStateIndex(id, stateIndex); stateIndex += 2; if (isRepresentative(id)) { generationInfo.addSpareActivationIndex(id, stateIndex); ++stateIndex; } if (mElements[id]->isSpareGate()) { generationInfo.addSpareUsageIndex(id, stateIndex); stateIndex += generationInfo.usageInfoBits(); } return stateIndex; } template size_t DFT::performStateGenerationInfoDFS(DFTStateGenerationInfo& generationInfo, std::queue& visitQueue, storm::storage::BitVector& visited, size_t stateIndex) const { while (!visitQueue.empty()) { size_t id = visitQueue.front(); visitQueue.pop(); if (visited[id]) { // Already visited continue; } stateIndex = generateStateInfo(generationInfo, id, visited, stateIndex); // Insert children if (mElements[id]->isGate()) { for (auto const& child : std::static_pointer_cast>(mElements[id])->children()) { visitQueue.push(child->id()); } } } return stateIndex; } template std::vector> DFT::topModularisation() const { STORM_LOG_ASSERT(isGate(mTopLevelIndex), "Top level element is no gate."); auto const& children = getGate(mTopLevelIndex)->children(); std::map> subdfts; for(auto const& child : children) { std::vector isubdft; if(child->nrParents() > 1 || child->hasOutgoingDependencies() || child->hasRestrictions()) { STORM_LOG_TRACE("child " << child->name() << " does not allow modularisation."); return {*this}; } if (isGate(child->id())) { isubdft = getGate(child->id())->independentSubDft(false); } else { STORM_LOG_ASSERT(isBasicElement(child->id()), "Child is no BE."); if(getBasicElement(child->id())->hasIngoingDependencies()) { STORM_LOG_TRACE("child " << child->name() << "does not allow modularisation."); return {*this}; } else { isubdft = {child->id()}; } } if(isubdft.empty()) { return {*this}; } else { subdfts[child->id()] = isubdft; } } std::vector> res; for(auto const& subdft : subdfts) { storm::builder::DFTBuilder builder; for(size_t id : subdft.second) { builder.copyElement(mElements[id]); } builder.setTopLevel(mElements[subdft.first]->name()); res.push_back(builder.build()); } return res; } template uint64_t DFT::maxRank() const { uint64_t max = 0; for (auto const& e : mElements) { if(e->rank() > max) { max = e->rank(); } } return max; } template DFT DFT::optimize() const { std::vector modIdea = findModularisationRewrite(); STORM_LOG_DEBUG("Modularisation idea: " << storm::utility::vector::toString(modIdea)); if (modIdea.empty()) { // No rewrite needed return *this; } std::vector> rewriteIds; rewriteIds.push_back(modIdea); storm::builder::DFTBuilder builder; // Accumulate elements which must be rewritten std::set rewriteSet; for (std::vector rewrites : rewriteIds) { rewriteSet.insert(rewrites.front()); } // Copy all other elements which do not change for (auto elem : mElements) { if (rewriteSet.count(elem->id()) == 0) { builder.copyElement(elem); } } // Add rewritten elements for (std::vector rewrites : rewriteIds) { STORM_LOG_ASSERT(rewrites.size() > 1, "No rewritten elements."); STORM_LOG_ASSERT(mElements[rewrites[1]]->hasParents(), "Rewritten elements has no parents."); STORM_LOG_ASSERT(mElements[rewrites[1]]->parents().front()->isGate(), "Rewritten element has no parent gate."); DFTGatePointer originalParent = std::static_pointer_cast>(mElements[rewrites[1]]->parents().front()); std::string newParentName = builder.getUniqueName(originalParent->name()); // Accumulate children names std::vector childrenNames; for (size_t i = 1; i < rewrites.size(); ++i) { STORM_LOG_ASSERT(mElements[rewrites[i]]->parents().front()->id() == originalParent->id(), "Children have the same father"); childrenNames.push_back(mElements[rewrites[i]]->name()); } // Add element inbetween parent and children switch (originalParent->type()) { case DFTElementType::AND: builder.addAndElement(newParentName, childrenNames); break; case DFTElementType::OR: builder.addOrElement(newParentName, childrenNames); break; case DFTElementType::BE_EXP: case DFTElementType::BE_CONST: case DFTElementType::VOT: case DFTElementType::PAND: case DFTElementType::SPARE: case DFTElementType::POR: case DFTElementType::PDEP: case DFTElementType::SEQ: case DFTElementType::MUTEX: // Other elements are not supported default: STORM_LOG_ASSERT(false, "Dft type can not be rewritten."); break; } // Add parent with the new child newParent and all its remaining children childrenNames.clear(); childrenNames.push_back(newParentName); for (auto const& child : originalParent->children()) { if (std::find(rewrites.begin()+1, rewrites.end(), child->id()) == rewrites.end()) { // Child was not rewritten and must be kept childrenNames.push_back(child->name()); } } builder.copyGate(originalParent, childrenNames); } builder.setTopLevel(mElements[mTopLevelIndex]->name()); // TODO use reference? DFT newDft = builder.build(); STORM_LOG_TRACE(newDft.getElementsString()); return newDft.optimize(); } template size_t DFT::nrDynamicElements() const { size_t noDyn = 0; for (auto const& elem : mElements) { switch (elem->type()) { case DFTElementType::AND: case DFTElementType::OR: case DFTElementType::VOT: case DFTElementType::BE_EXP: case DFTElementType::BE_CONST: break; case DFTElementType::PAND: case DFTElementType::SPARE: case DFTElementType::POR: case DFTElementType::SEQ: case DFTElementType::MUTEX: case DFTElementType::PDEP: noDyn += 1; break; default: STORM_LOG_ASSERT(false, "DFT element type " << elem->type() << " not known."); break; } } return noDyn; } template size_t DFT::nrStaticElements() const { size_t noStatic = 0; for (auto const& elem : mElements) { switch (elem->type()) { case DFTElementType::AND: case DFTElementType::OR: case DFTElementType::VOT: ++noStatic; break; case DFTElementType::BE_EXP: case DFTElementType::BE_CONST: case DFTElementType::PAND: case DFTElementType::SPARE: case DFTElementType::POR: case DFTElementType::SEQ: case DFTElementType::MUTEX: case DFTElementType::PDEP: break; default: STORM_LOG_ASSERT(false, "DFT element type " << elem->type() << " not known."); break; } } return noStatic; } template std::string DFT::getElementsString() const { std::stringstream stream; for (auto const& elem : mElements) { stream << "[" << elem->id() << "]" << *elem << std::endl; } return stream.str(); } template std::string DFT::getInfoString() const { std::stringstream stream; stream << "Top level index: " << mTopLevelIndex << ", Nr BEs" << mNrOfBEs; return stream.str(); } template std::string DFT::getSpareModulesString() const { std::stringstream stream; stream << "[" << mElements[mTopLevelIndex]->id() << "] {"; std::vector::const_iterator it = mTopModule.begin(); if (it == mTopModule.end()) { stream << "}" << std::endl; return stream.str(); } STORM_LOG_ASSERT(it != mTopModule.end(), "Element not found."); stream << mElements[(*it)]->name(); ++it; while(it != mTopModule.end()) { stream << ", " << mElements[(*it)]->name(); ++it; } stream << "}" << std::endl; for(auto const& spareModule : mSpareModules) { stream << "[" << mElements[spareModule.first]->name() << "] = {"; if (!spareModule.second.empty()) { std::vector::const_iterator it = spareModule.second.begin(); STORM_LOG_ASSERT(it != spareModule.second.end(), "Element not found."); stream << mElements[(*it)]->name(); ++it; while(it != spareModule.second.end()) { stream << ", " << mElements[(*it)]->name(); ++it; } } stream << "}" << std::endl; } return stream.str(); } template std::string DFT::getElementsWithStateString(DFTStatePointer const& state) const{ std::stringstream stream; for (auto const& elem : mElements) { stream << "[" << elem->id() << "]"; stream << *elem; if (elem->isDependency()) { stream << "\t** " << storm::storage::toChar(state->getDependencyState(elem->id())) << "[dep]"; } else { stream << "\t** " << storm::storage::toChar(state->getElementState(elem->id())); if(elem->isSpareGate()) { size_t useId = state->uses(elem->id()); if(useId == elem->id() || state->isActive(useId)) { stream << "actively "; } stream << "using " << useId; } } stream << std::endl; } return stream.str(); } template std::string DFT::getStateString(DFTStatePointer const& state) const { std::stringstream stream; stream << "(" << state->getId() << ") "; for (auto const& elem : mElements) { if (elem->isDependency()) { stream << storm::storage::toChar(state->getDependencyState(elem->id())) << "[dep]"; } else { stream << storm::storage::toChar(state->getElementState(elem->id())); if(elem->isSpareGate()) { stream << "["; size_t useId = state->uses(elem->id()); if(useId == elem->id() || state->isActive(useId)) { stream << "actively "; } stream << "using " << useId << "]"; } } } return stream.str(); } template std::string DFT::getStateString(storm::storage::BitVector const& status, DFTStateGenerationInfo const& stateGenerationInfo, size_t id) const { std::stringstream stream; stream << "(" << id << ") "; for (auto const& elem : mElements) { size_t elemIndex = stateGenerationInfo.getStateIndex(elem->id()); int elementState = DFTState::getElementStateInt(status, elemIndex); if (elem->isDependency()) { stream << storm::storage::toChar(static_cast(elementState)) << "[dep]"; } else { stream << storm::storage::toChar(static_cast(elementState)); if(elem->isSpareGate()) { stream << "["; size_t nrUsedChild = status.getAsInt(stateGenerationInfo.getSpareUsageIndex(elem->id()), stateGenerationInfo.usageInfoBits()); size_t useId; if (nrUsedChild == getMaxSpareChildCount()) { useId = elem->id(); } else { useId = getChild(elem->id(), nrUsedChild); } if(useId == elem->id() || status[stateGenerationInfo.getSpareActivationIndex(useId)]) { stream << "actively "; } stream << "using " << useId << "]"; } } } return stream.str(); } template size_t DFT::getChild(size_t spareId, size_t nrUsedChild) const { STORM_LOG_ASSERT(mElements[spareId]->isSpareGate(), "Element is no spare."); return getGate(spareId)->children()[nrUsedChild]->id(); } template size_t DFT::getNrChild(size_t spareId, size_t childId) const { STORM_LOG_ASSERT(mElements[spareId]->isSpareGate(), "Element is no spare."); DFTElementVector children = getGate(spareId)->children(); for (size_t nrChild = 0; nrChild < children.size(); ++nrChild) { if (children[nrChild]->id() == childId) { return nrChild; } } STORM_LOG_ASSERT(false, "Child not found."); return 0; } template std::vector DFT::getIndependentSubDftRoots(size_t index) const { auto elem = getElement(index); auto ISD = elem->independentSubDft(false); return ISD; } template std::vector DFT::immediateFailureCauses(size_t index) const { if (isGate(index)) { STORM_LOG_ASSERT(false, "Is gate."); return {}; } else { return {index}; } } template bool DFT::canHaveNondeterminism() const { return !getDependencies().empty(); } template bool DFT::checkWellFormedness(std::ostream& stream) const { bool wellformed = true; // Check independence of spare modules // TODO: comparing one element of each spare module sufficient? for (auto module1 = mSpareModules.begin() ; module1 != mSpareModules.end() ; ++module1) { size_t firstElement = module1->second.front(); for (auto module2 = std::next(module1); module2 != mSpareModules.end(); ++module2) { if (std::find(module2->second.begin(), module2->second.end(), firstElement) != module2->second.end()) { if (!wellformed) { stream << std::endl; } stream << "Spare modules of '" << getElement(module1->first)->name() << "' and '" << getElement(module2->first)->name() << "' should not overlap."; wellformed = false; } } } // TODO check VOT gates return wellformed; } template DFTColouring DFT::colourDFT() const { return DFTColouring(*this); } template std::map DFT::findBijection(size_t index1, size_t index2, DFTColouring const& colouring, bool sparesAsLeaves) const { STORM_LOG_TRACE("Considering ids " << index1 << ", " << index2 << " for isomorphism."); bool sharedSpareMode = false; std::map bijection; if (isBasicElement(index1)) { if (!isBasicElement(index2)) { return {}; } if (colouring.hasSameColour(index1, index2)) { bijection[index1] = index2; return bijection; } else { return {}; } } STORM_LOG_ASSERT(isGate(index1), "Element is no gate."); STORM_LOG_ASSERT(isGate(index2), "Element is no gate."); std::vector isubdft1 = getGate(index1)->independentSubDft(false); std::vector isubdft2 = getGate(index2)->independentSubDft(false); if(isubdft1.empty() || isubdft2.empty() || isubdft1.size() != isubdft2.size()) { if (isubdft1.empty() && isubdft2.empty() && sparesAsLeaves) { // Check again for shared spares sharedSpareMode = true; isubdft1 = getGate(index1)->independentSubDft(false, true); isubdft2 = getGate(index2)->independentSubDft(false, true); if(isubdft1.empty() || isubdft2.empty() || isubdft1.size() != isubdft2.size()) { return {}; } } else { return {}; } } STORM_LOG_TRACE("Checking subdfts from " << index1 << ", " << index2 << " for isomorphism."); auto LHS = colouring.colourSubdft(isubdft1); auto RHS = colouring.colourSubdft(isubdft2); auto IsoCheck = DFTIsomorphismCheck(LHS, RHS, *this); while (IsoCheck.findNextIsomorphism()) { bijection = IsoCheck.getIsomorphism(); if (sharedSpareMode) { bool bijectionSpareCompatible = true; for (size_t elementId : isubdft1) { if (getElement(elementId)->isSpareGate()) { std::shared_ptr> spareLeft = std::static_pointer_cast>(mElements[elementId]); std::shared_ptr> spareRight = std::static_pointer_cast>(mElements[bijection.at(elementId)]); if (spareLeft->nrChildren() != spareRight->nrChildren()) { bijectionSpareCompatible = false; break; } // Check bijection for spare children for (size_t i = 0; i < spareLeft->nrChildren(); ++i) { size_t childLeftId = spareLeft->children().at(i)->id(); size_t childRightId = spareRight->children().at(i)->id(); STORM_LOG_ASSERT(bijection.count(childLeftId) == 0, "Child already part of bijection."); if (childLeftId == childRightId) { // Ignore shared child continue; } // TODO generalize for more than one parent if (spareLeft->children().at(i)->nrParents() != 1 || spareRight->children().at(i)->nrParents() != 1) { bijectionSpareCompatible = false; break; } std::map tmpBijection = findBijection(childLeftId, childRightId, colouring, false); if (!tmpBijection.empty()) { bijection.insert(tmpBijection.begin(), tmpBijection.end()); } else { bijectionSpareCompatible = false; break; } } if (!bijectionSpareCompatible) { break; } } } if (bijectionSpareCompatible) { return bijection; } } else { return bijection; } } // end while return {}; } template DFTIndependentSymmetries DFT::findSymmetries(DFTColouring const& colouring) const { std::vector vec; vec.reserve(nrElements()); storm::utility::iota_n(std::back_inserter(vec), nrElements(), 0); BijectionCandidates completeCategories = colouring.colourSubdft(vec); std::map>> res; // Find symmetries for gates for(auto const& colourClass : completeCategories.gateCandidates) { findSymmetriesHelper(colourClass.second, colouring, res); } // Find symmetries for BEs for(auto const& colourClass : completeCategories.beCandidates) { findSymmetriesHelper(colourClass.second, colouring, res); } return DFTIndependentSymmetries(res); } template void DFT::findSymmetriesHelper(std::vector const& candidates, DFTColouring const& colouring, std::map>>& result) const { if(candidates.size() <= 0) { return; } std::set foundEqClassFor; for(auto it1 = candidates.cbegin(); it1 != candidates.cend(); ++it1) { std::vector> symClass; if(foundEqClassFor.count(*it1) > 0) { // This item is already in a class. continue; } if(!getElement(*it1)->hasOnlyStaticParents()) { continue; } std::tuple, std::vector, std::vector> influencedElem1Ids = getSortedParentAndDependencyIds(*it1); auto it2 = it1; for(++it2; it2 != candidates.cend(); ++it2) { if(!getElement(*it2)->hasOnlyStaticParents()) { continue; } if(influencedElem1Ids == getSortedParentAndDependencyIds(*it2)) { std::map bijection = findBijection(*it1, *it2, colouring, true); if (!bijection.empty()) { STORM_LOG_TRACE("Subdfts are symmetric"); foundEqClassFor.insert(*it2); if(symClass.empty()) { for(auto const& i : bijection) { symClass.push_back(std::vector({i.first})); } } auto symClassIt = symClass.begin(); for(auto const& i : bijection) { symClassIt->emplace_back(i.second); ++symClassIt; } } } } if(!symClass.empty()) { result.emplace(*it1, symClass); } } } template std::vector DFT::findModularisationRewrite() const { for(auto const& e : mElements) { if(e->isGate() && (e->type() == DFTElementType::AND || e->type() == DFTElementType::OR) ) { // suitable parent gate! - Lets check the independent submodules of the children auto const& children = std::static_pointer_cast>(e)->children(); for(auto const& child : children) { auto ISD = std::static_pointer_cast>(child)->independentSubDft(true); // In the ISD, check for other children: std::vector rewrite = {e->id(), child->id()}; for(size_t isdElemId : ISD) { if(isdElemId == child->id()) continue; if(std::find_if(children.begin(), children.end(), [&isdElemId](std::shared_ptr> const& e) { return e->id() == isdElemId; } ) != children.end()) { rewrite.push_back(isdElemId); } } if(rewrite.size() > 2 && rewrite.size() < children.size() - 1) { return rewrite; } } } } return {}; } template std::tuple, std::vector, std::vector> DFT::getSortedParentAndDependencyIds(size_t index) const { // Parents std::vector parents = getElement(index)->parentIds(); std::sort(parents.begin(), parents.end()); // Ingoing dependencies std::vector ingoingDeps; if (isBasicElement(index)) { for(auto const& dep : getBasicElement(index)->ingoingDependencies()) { ingoingDeps.push_back(dep->id()); } std::sort(ingoingDeps.begin(), ingoingDeps.end()); } // Outgoing dependencies std::vector outgoingDeps; for(auto const& dep : getElement(index)->outgoingDependencies()) { outgoingDeps.push_back(dep->id()); } std::sort(outgoingDeps.begin(), outgoingDeps.end()); return std::make_tuple(parents, ingoingDeps, outgoingDeps); } template std::set DFT::getAllIds() const { std::set ids; for (auto const& elem : mElements) { ids.insert(elem->id()); } return ids; } template size_t DFT::getIndex(std::string const& name) const { auto iter = std::find_if(mElements.begin(), mElements.end(), [&name](DFTElementPointer const& e) { return e->name() == name; }); STORM_LOG_THROW(iter != mElements.end(), storm::exceptions::InvalidArgumentException, "Event name '" << name << "' not known."); return (*iter)->id(); } template void DFT::writeStatsToStream(std::ostream& stream) const { // Count individual types of elements size_t noBE = 0; size_t noAnd = 0; size_t noOr = 0; size_t noVot = 0; size_t noPand = 0; size_t noPor = 0; size_t noSpare = 0; size_t noDependency = 0; size_t noRestriction = 0; for (auto const& elem : mElements) { switch (elem->type()) { case DFTElementType::BE_EXP: case DFTElementType::BE_CONST: ++noBE; break; case DFTElementType::AND: ++noAnd; break; case DFTElementType::OR: ++noOr; break; case DFTElementType::VOT: ++noVot; break; case DFTElementType::PAND: ++noPand; break; case DFTElementType::POR: ++noPor; break; case DFTElementType::SPARE: ++noSpare; break; case DFTElementType::PDEP: ++noDependency; break; case DFTElementType::SEQ: case DFTElementType::MUTEX: ++noRestriction; break; default: STORM_LOG_ASSERT(false, "DFT element type " << elem->type() << " not known."); break; } } size_t noStatic = nrStaticElements(); size_t noDynamic = nrDynamicElements(); // Check whether numbers are correct STORM_LOG_ASSERT(noBE == nrBasicElements(), "No. of BEs does not match."); STORM_LOG_ASSERT(noSpare == mNrOfSpares, "No. of SPAREs does not match."); STORM_LOG_ASSERT(noDependency == mDependencies.size(), "No. of Dependencies does not match."); STORM_LOG_ASSERT(noAnd + noOr + noVot == noStatic, "No. of static gates does not match."); STORM_LOG_ASSERT(noPand + noPor + noSpare + noDependency + noRestriction == noDynamic, "No. of dynamic gates does not match."); STORM_LOG_ASSERT(noBE + noStatic + noDynamic == nrElements(), "No. of elements does not match."); // Print output stream << "=============DFT Statistics==============" << std::endl; stream << "Number of BEs: " << nrBasicElements() << std::endl; stream << "Number of static elements: " << noStatic << std::endl; stream << "Number of dynamic elements: " << noDynamic << std::endl; stream << "Number of elements: " << nrElements() << std::endl; stream << "-----------------------------------------" << std::endl; if (noBE > 0) { stream << "Number of BEs: " << noBE << std::endl; } if (noAnd > 0) { stream << "Number of AND gates: " << noAnd << std::endl; } if (noOr > 0) { stream << "Number of OR gates: " << noOr << std::endl; } if (noVot > 0) { stream << "Number of VOT gates: " << noVot << std::endl; } if (noPand > 0) { stream << "Number of PAND gates: " << noPand << std::endl; } if (noPor > 0) { stream << "Number of POR gates: " << noPor << std::endl; } if (noSpare > 0) { stream << "Number of SPARE gates: " << noSpare << std::endl; } if (noDependency > 0) { stream << "Number of Dependencies: " << noDependency << std::endl; } if (noRestriction > 0) { stream << "Number of Restrictions: " << noRestriction << std::endl; } stream << "=========================================" << std::endl; } // Explicitly instantiate the class. template class DFT; #ifdef STORM_HAVE_CARL template class DFT; #endif } }