Browse Source

Try to improve performance for relevant events

tempestpy_adaptions
Matthias Volk 6 years ago
parent
commit
187d47d8ac
  1. 17
      src/storm-dft/generator/DftNextStateGenerator.cpp
  2. 260
      src/storm-dft/storage/dft/DFT.cpp
  3. 3
      src/storm-dft/storage/dft/DFT.h
  4. 20
      src/storm-dft/storage/dft/DFTState.cpp
  5. 16
      src/storm-dft/storage/dft/DFTState.h

17
src/storm-dft/generator/DftNextStateGenerator.cpp

@ -60,10 +60,23 @@ namespace storm {
//size_t failableCount = hasDependencies ? state->nrFailableDependencies() : state->nrFailableBEs(); //size_t failableCount = hasDependencies ? state->nrFailableDependencies() : state->nrFailableBEs();
//size_t currentFailable = 0; //size_t currentFailable = 0;
state->getFailableElements().init(exploreDependencies); state->getFailableElements().init(exploreDependencies);
// Check whether operational relevant event remains
bool remainingRelevantEvent = true;
if (mDft.hasFailed(state)) {
remainingRelevantEvent = false;
// Toplevel has failed already -> check for possible other relevant events
for (auto const& event : mDft.getRelevantEvents()) {
if (state->isOperational(event)) {
remainingRelevantEvent = true;
break;
}
}
}
// Check for absorbing state: // Check for absorbing state:
// - either no relevant event remains (i.e., all relevant events have failed already), or // - either no relevant event remains (i.e., all relevant events have failed already), or
// - no BE can fail // - no BE can fail
if (!state->getFailableElements().hasRemainingRelevantEvent() || state->getFailableElements().isEnd()) {
if (!remainingRelevantEvent || state->getFailableElements().isEnd()) {
Choice<ValueType, StateType> choice(0, true); Choice<ValueType, StateType> choice(0, true);
// Add self loop // Add self loop
choice.addProbability(state->getId(), storm::utility::one<ValueType>()); choice.addProbability(state->getId(), storm::utility::one<ValueType>());
@ -122,8 +135,6 @@ namespace storm {
newState->updateFailableInRestrictions(next->id()); newState->updateFailableInRestrictions(next->id());
} }
newState->updateRemainingRelevantEvents();
bool transient = false; bool transient = false;
if (nextBE->type() == storm::storage::DFTElementType::BE_EXP) { if (nextBE->type() == storm::storage::DFTElementType::BE_EXP) {
auto beExp = std::static_pointer_cast<storm::storage::BEExponential<ValueType> const>(nextBE); auto beExp = std::static_pointer_cast<storm::storage::BEExponential<ValueType> const>(nextBE);

260
src/storm-dft/storage/dft/DFT.cpp

@ -17,7 +17,8 @@ namespace storm {
namespace storage { namespace storage {
template<typename ValueType> template<typename ValueType>
DFT<ValueType>::DFT(DFTElementVector const& elements, DFTElementPointer const& tle) : mElements(elements), mNrOfBEs(0), mNrOfSpares(0), mNrRepresentatives(0), mTopLevelIndex(tle->id()), mMaxSpareChildCount(0) {
DFT<ValueType>::DFT(DFTElementVector const& elements, DFTElementPointer const& tle) : mElements(elements), mNrOfBEs(0), mNrOfSpares(0), mNrRepresentatives(0),
mTopLevelIndex(tle->id()), mMaxSpareChildCount(0) {
// Check that ids correspond to indices in the element vector // Check that ids correspond to indices in the element vector
STORM_LOG_ASSERT(elementIndicesCorrect(), "Ids incorrect."); STORM_LOG_ASSERT(elementIndicesCorrect(), "Ids incorrect.");
@ -25,19 +26,20 @@ namespace storm {
if (isRepresentative(elem->id())) { if (isRepresentative(elem->id())) {
++mNrRepresentatives; ++mNrRepresentatives;
} }
if(elem->isBasicElement()) {
if (elem->isBasicElement()) {
++mNrOfBEs; ++mNrOfBEs;
} else if (elem->isSpareGate()) { } else if (elem->isSpareGate()) {
// Build spare modules by setting representatives and representants // Build spare modules by setting representatives and representants
++mNrOfSpares; ++mNrOfSpares;
mMaxSpareChildCount = std::max(mMaxSpareChildCount, std::static_pointer_cast<DFTSpare<ValueType>>(elem)->children().size()); mMaxSpareChildCount = std::max(mMaxSpareChildCount, std::static_pointer_cast<DFTSpare<ValueType>>(elem)->children().size());
for(auto const& spareReprs : std::static_pointer_cast<DFTSpare<ValueType>>(elem)->children()) {
STORM_LOG_THROW(spareReprs->isGate() || spareReprs->isBasicElement(), storm::exceptions::WrongFormatException, "Child '" << spareReprs->name() << "' of spare '" << elem->name() << "' must be gate or BE.");
for (auto const& spareReprs : std::static_pointer_cast<DFTSpare<ValueType>>(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<size_t> module = {spareReprs->id()}; std::set<size_t> module = {spareReprs->id()};
spareReprs->extendSpareModule(module); spareReprs->extendSpareModule(module);
std::vector<size_t> sparesAndBes; std::vector<size_t> sparesAndBes;
for(size_t modelem : module) {
if(mElements[modelem]->isSpareGate() || mElements[modelem]->isBasicElement()) {
for (size_t modelem : module) {
if (mElements[modelem]->isSpareGate() || mElements[modelem]->isBasicElement()) {
sparesAndBes.push_back(modelem); sparesAndBes.push_back(modelem);
mRepresentants.insert(std::make_pair(modelem, spareReprs->id())); mRepresentants.insert(std::make_pair(modelem, spareReprs->id()));
} }
@ -52,14 +54,14 @@ namespace storm {
// For the top module, we assume, contrary to [Jun15], that we have all spare gates and basic elements which are not in another module. // 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<size_t> topModuleSet; std::set<size_t> topModuleSet;
// Initialize with all ids. // Initialize with all ids.
for(auto const& elem : mElements) {
for (auto const& elem : mElements) {
if (elem->isBasicElement() || elem->isSpareGate()) { if (elem->isBasicElement() || elem->isSpareGate()) {
topModuleSet.insert(elem->id()); topModuleSet.insert(elem->id());
} }
} }
// Erase spare modules // Erase spare modules
for(auto const& module : mSpareModules) {
for(auto const& index : module.second) {
for (auto const& module : mSpareModules) {
for (auto const& index : module.second) {
topModuleSet.erase(index); topModuleSet.erase(index);
} }
} }
@ -72,7 +74,8 @@ namespace storm {
if (!mTopModule.empty()) { if (!mTopModule.empty()) {
for (auto& module : mSpareModules) { for (auto& module : mSpareModules) {
if (std::find(module.second.begin(), module.second.end(), mTopModule.front()) != module.second.end()) { 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.");
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(); module.second.clear();
} }
} }
@ -86,10 +89,10 @@ namespace storm {
template<typename ValueType> template<typename ValueType>
DFTStateGenerationInfo DFT<ValueType>::buildStateGenerationInfo(storm::storage::DFTIndependentSymmetries const& symmetries) const { DFTStateGenerationInfo DFT<ValueType>::buildStateGenerationInfo(storm::storage::DFTIndependentSymmetries const& symmetries) const {
DFTStateGenerationInfo generationInfo(nrElements(), mNrOfSpares, mNrRepresentatives, mMaxSpareChildCount); DFTStateGenerationInfo generationInfo(nrElements(), mNrOfSpares, mNrRepresentatives, mMaxSpareChildCount);
// Generate Pre and Post info for restrictions, and mutexes // Generate Pre and Post info for restrictions, and mutexes
for(auto const& elem : mElements) {
if(!elem->isDependency() && !elem->isRestriction()) {
for (auto const& elem : mElements) {
if (!elem->isDependency() && !elem->isRestriction()) {
generationInfo.setRestrictionPreElements(elem->id(), elem->seqRestrictionPres()); generationInfo.setRestrictionPreElements(elem->id(), elem->seqRestrictionPres());
generationInfo.setRestrictionPostElements(elem->id(), elem->seqRestrictionPosts()); generationInfo.setRestrictionPostElements(elem->id(), elem->seqRestrictionPosts());
generationInfo.setMutexElements(elem->id(), elem->mutexRestrictionElements()); generationInfo.setMutexElements(elem->id(), elem->mutexRestrictionElements());
@ -121,7 +124,7 @@ namespace storm {
} }
} }
size_t offset = stateIndex - groupIndex; size_t offset = stateIndex - groupIndex;
// Mirror symmetries // Mirror symmetries
size_t noSymmetricElements = symmetryGroup.front().size(); size_t noSymmetricElements = symmetryGroup.front().size();
STORM_LOG_ASSERT(noSymmetricElements > 1, "No symmetry available."); STORM_LOG_ASSERT(noSymmetricElements > 1, "No symmetry available.");
@ -141,7 +144,7 @@ namespace storm {
size_t index = generationInfo.getStateIndex(originalElement); size_t index = generationInfo.getStateIndex(originalElement);
size_t activationIndex = isRepresentative(originalElement) ? generationInfo.getSpareActivationIndex(originalElement) : 0; size_t activationIndex = isRepresentative(originalElement) ? generationInfo.getSpareActivationIndex(originalElement) : 0;
size_t usageIndex = mElements[originalElement]->isSpareGate() ? generationInfo.getSpareUsageIndex(originalElement) : 0; size_t usageIndex = mElements[originalElement]->isSpareGate() ? generationInfo.getSpareUsageIndex(originalElement) : 0;
// Mirror symmetry for each element // Mirror symmetry for each element
for (size_t i = 1; i < symmetricElements.size(); ++i) { for (size_t i = 1; i < symmetricElements.size(); ++i) {
size_t symmetricElement = symmetricElements[i]; size_t symmetricElement = symmetricElements[i];
@ -179,7 +182,8 @@ namespace storm {
std::shared_ptr<DFTDependency<ValueType> const> dependency = getDependency(idDependency); std::shared_ptr<DFTDependency<ValueType> const> dependency = getDependency(idDependency);
visitQueue.push(dependency->id()); visitQueue.push(dependency->id());
visitQueue.push(dependency->triggerEvent()->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.");
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()); visitQueue.push(dependency->dependentEvents()[0]->id());
} }
stateIndex = performStateGenerationInfoDFS(generationInfo, visitQueue, visited, stateIndex); stateIndex = performStateGenerationInfoDFS(generationInfo, visitQueue, visited, stateIndex);
@ -191,7 +195,7 @@ namespace storm {
stateIndex = performStateGenerationInfoDFS(generationInfo, visitQueue, visited, stateIndex); stateIndex = performStateGenerationInfoDFS(generationInfo, visitQueue, visited, stateIndex);
} }
} }
generationInfo.generateSymmetries(symmetries); generationInfo.generateSymmetries(symmetries);
STORM_LOG_TRACE(generationInfo); STORM_LOG_TRACE(generationInfo);
@ -201,31 +205,32 @@ namespace storm {
return generationInfo; return generationInfo;
} }
template<typename ValueType> template<typename ValueType>
size_t DFT<ValueType>::generateStateInfo(DFTStateGenerationInfo& generationInfo, size_t id, storm::storage::BitVector& visited, size_t stateIndex) const { size_t DFT<ValueType>::generateStateInfo(DFTStateGenerationInfo& generationInfo, size_t id, storm::storage::BitVector& visited, size_t stateIndex) const {
STORM_LOG_ASSERT(!visited[id], "Element already visited."); STORM_LOG_ASSERT(!visited[id], "Element already visited.");
visited.set(id); visited.set(id);
// Reserve bits for element // Reserve bits for element
generationInfo.addStateIndex(id, stateIndex); generationInfo.addStateIndex(id, stateIndex);
stateIndex += 2; stateIndex += 2;
if (isRepresentative(id)) { if (isRepresentative(id)) {
generationInfo.addSpareActivationIndex(id, stateIndex); generationInfo.addSpareActivationIndex(id, stateIndex);
++stateIndex; ++stateIndex;
} }
if (mElements[id]->isSpareGate()) { if (mElements[id]->isSpareGate()) {
generationInfo.addSpareUsageIndex(id, stateIndex); generationInfo.addSpareUsageIndex(id, stateIndex);
stateIndex += generationInfo.usageInfoBits(); stateIndex += generationInfo.usageInfoBits();
} }
return stateIndex; return stateIndex;
} }
template<typename ValueType> template<typename ValueType>
size_t DFT<ValueType>::performStateGenerationInfoDFS(DFTStateGenerationInfo& generationInfo, std::queue<size_t>& visitQueue, storm::storage::BitVector& visited, size_t stateIndex) const {
size_t DFT<ValueType>::performStateGenerationInfoDFS(DFTStateGenerationInfo& generationInfo, std::queue<size_t>& visitQueue, storm::storage::BitVector& visited,
size_t stateIndex) const {
while (!visitQueue.empty()) { while (!visitQueue.empty()) {
size_t id = visitQueue.front(); size_t id = visitQueue.front();
visitQueue.pop(); visitQueue.pop();
@ -234,7 +239,7 @@ namespace storm {
continue; continue;
} }
stateIndex = generateStateInfo(generationInfo, id, visited, stateIndex); stateIndex = generateStateInfo(generationInfo, id, visited, stateIndex);
// Insert children // Insert children
if (mElements[id]->isGate()) { if (mElements[id]->isGate()) {
for (auto const& child : std::static_pointer_cast<DFTGate<ValueType>>(mElements[id])->children()) { for (auto const& child : std::static_pointer_cast<DFTGate<ValueType>>(mElements[id])->children()) {
@ -244,15 +249,15 @@ namespace storm {
} }
return stateIndex; return stateIndex;
} }
template<typename ValueType> template<typename ValueType>
std::vector<DFT<ValueType>> DFT<ValueType>::topModularisation() const {
std::vector<DFT<ValueType>> DFT<ValueType>::topModularisation() const {
STORM_LOG_ASSERT(isGate(mTopLevelIndex), "Top level element is no gate."); STORM_LOG_ASSERT(isGate(mTopLevelIndex), "Top level element is no gate.");
auto const& children = getGate(mTopLevelIndex)->children(); auto const& children = getGate(mTopLevelIndex)->children();
std::map<size_t, std::vector<size_t>> subdfts; std::map<size_t, std::vector<size_t>> subdfts;
for(auto const& child : children) {
for (auto const& child : children) {
std::vector<size_t> isubdft; std::vector<size_t> isubdft;
if(child->nrParents() > 1 || child->hasOutgoingDependencies() || child->hasRestrictions()) {
if (child->nrParents() > 1 || child->hasOutgoingDependencies() || child->hasRestrictions()) {
STORM_LOG_TRACE("child " << child->name() << " does not allow modularisation."); STORM_LOG_TRACE("child " << child->name() << " does not allow modularisation.");
return {*this}; return {*this};
} }
@ -260,40 +265,40 @@ namespace storm {
isubdft = getGate(child->id())->independentSubDft(false); isubdft = getGate(child->id())->independentSubDft(false);
} else { } else {
STORM_LOG_ASSERT(isBasicElement(child->id()), "Child is no BE."); STORM_LOG_ASSERT(isBasicElement(child->id()), "Child is no BE.");
if(getBasicElement(child->id())->hasIngoingDependencies()) {
if (getBasicElement(child->id())->hasIngoingDependencies()) {
STORM_LOG_TRACE("child " << child->name() << "does not allow modularisation."); STORM_LOG_TRACE("child " << child->name() << "does not allow modularisation.");
return {*this}; return {*this};
} else { } else {
isubdft = {child->id()}; isubdft = {child->id()};
} }
} }
if(isubdft.empty()) {
if (isubdft.empty()) {
return {*this}; return {*this};
} else { } else {
subdfts[child->id()] = isubdft; subdfts[child->id()] = isubdft;
} }
} }
std::vector<DFT<ValueType>> res; std::vector<DFT<ValueType>> res;
for(auto const& subdft : subdfts) {
for (auto const& subdft : subdfts) {
storm::builder::DFTBuilder<ValueType> builder; storm::builder::DFTBuilder<ValueType> builder;
for(size_t id : subdft.second) {
for (size_t id : subdft.second) {
builder.copyElement(mElements[id]); builder.copyElement(mElements[id]);
} }
builder.setTopLevel(mElements[subdft.first]->name()); builder.setTopLevel(mElements[subdft.first]->name());
res.push_back(builder.build()); res.push_back(builder.build());
} }
return res; return res;
} }
template<typename ValueType> template<typename ValueType>
uint64_t DFT<ValueType>::maxRank() const { uint64_t DFT<ValueType>::maxRank() const {
uint64_t max = 0; uint64_t max = 0;
for (auto const& e : mElements) { for (auto const& e : mElements) {
if(e->rank() > max) {
if (e->rank() > max) {
max = e->rank(); max = e->rank();
} }
} }
@ -338,7 +343,8 @@ namespace storm {
// Accumulate children names // Accumulate children names
std::vector<std::string> childrenNames; std::vector<std::string> childrenNames;
for (size_t i = 1; i < rewrites.size(); ++i) { for (size_t i = 1; i < rewrites.size(); ++i) {
STORM_LOG_ASSERT(mElements[rewrites[i]]->parents().front()->id() == originalParent->id(), "Children have not the same father for rewrite " << mElements[rewrites[i]]->name());
STORM_LOG_ASSERT(mElements[rewrites[i]]->parents().front()->id() == originalParent->id(),
"Children have not the same father for rewrite " << mElements[rewrites[i]]->name());
childrenNames.push_back(mElements[rewrites[i]]->name()); childrenNames.push_back(mElements[rewrites[i]]->name());
} }
@ -369,7 +375,7 @@ namespace storm {
childrenNames.clear(); childrenNames.clear();
childrenNames.push_back(newParentName); childrenNames.push_back(newParentName);
for (auto const& child : originalParent->children()) { for (auto const& child : originalParent->children()) {
if (std::find(rewrites.begin()+1, rewrites.end(), child->id()) == rewrites.end()) {
if (std::find(rewrites.begin() + 1, rewrites.end(), child->id()) == rewrites.end()) {
// Child was not rewritten and must be kept // Child was not rewritten and must be kept
childrenNames.push_back(child->name()); childrenNames.push_back(child->name());
} }
@ -466,21 +472,21 @@ namespace storm {
STORM_LOG_ASSERT(it != mTopModule.end(), "Element not found."); STORM_LOG_ASSERT(it != mTopModule.end(), "Element not found.");
stream << mElements[(*it)]->name(); stream << mElements[(*it)]->name();
++it; ++it;
while(it != mTopModule.end()) {
stream << ", " << mElements[(*it)]->name();
while (it != mTopModule.end()) {
stream << ", " << mElements[(*it)]->name();
++it; ++it;
} }
stream << "}" << std::endl; stream << "}" << std::endl;
for(auto const& spareModule : mSpareModules) {
for (auto const& spareModule : mSpareModules) {
stream << "[" << mElements[spareModule.first]->name() << "] = {"; stream << "[" << mElements[spareModule.first]->name() << "] = {";
if (!spareModule.second.empty()) { if (!spareModule.second.empty()) {
std::vector<size_t>::const_iterator it = spareModule.second.begin(); std::vector<size_t>::const_iterator it = spareModule.second.begin();
STORM_LOG_ASSERT(it != spareModule.second.end(), "Element not found."); STORM_LOG_ASSERT(it != spareModule.second.end(), "Element not found.");
stream << mElements[(*it)]->name(); stream << mElements[(*it)]->name();
++it; ++it;
while(it != spareModule.second.end()) {
stream << ", " << mElements[(*it)]->name();
while (it != spareModule.second.end()) {
stream << ", " << mElements[(*it)]->name();
++it; ++it;
} }
} }
@ -490,7 +496,7 @@ namespace storm {
} }
template<typename ValueType> template<typename ValueType>
std::string DFT<ValueType>::getElementsWithStateString(DFTStatePointer const& state) const{
std::string DFT<ValueType>::getElementsWithStateString(DFTStatePointer const& state) const {
std::stringstream stream; std::stringstream stream;
for (auto const& elem : mElements) { for (auto const& elem : mElements) {
stream << "[" << elem->id() << "]"; stream << "[" << elem->id() << "]";
@ -499,9 +505,9 @@ namespace storm {
stream << "\t** " << storm::storage::toChar(state->getDependencyState(elem->id())) << "[dep]"; stream << "\t** " << storm::storage::toChar(state->getDependencyState(elem->id())) << "[dep]";
} else { } else {
stream << "\t** " << storm::storage::toChar(state->getElementState(elem->id())); stream << "\t** " << storm::storage::toChar(state->getElementState(elem->id()));
if(elem->isSpareGate()) {
if (elem->isSpareGate()) {
size_t useId = state->uses(elem->id()); size_t useId = state->uses(elem->id());
if(useId == elem->id() || state->isActive(useId)) {
if (useId == elem->id() || state->isActive(useId)) {
stream << "actively "; stream << "actively ";
} }
stream << "using " << useId; stream << "using " << useId;
@ -511,7 +517,7 @@ namespace storm {
} }
return stream.str(); return stream.str();
} }
template<typename ValueType> template<typename ValueType>
std::string DFT<ValueType>::getStateString(DFTStatePointer const& state) const { std::string DFT<ValueType>::getStateString(DFTStatePointer const& state) const {
std::stringstream stream; std::stringstream stream;
@ -521,10 +527,10 @@ namespace storm {
stream << storm::storage::toChar(state->getDependencyState(elem->id())) << "[dep]"; stream << storm::storage::toChar(state->getDependencyState(elem->id())) << "[dep]";
} else { } else {
stream << storm::storage::toChar(state->getElementState(elem->id())); stream << storm::storage::toChar(state->getElementState(elem->id()));
if(elem->isSpareGate()) {
if (elem->isSpareGate()) {
stream << "["; stream << "[";
size_t useId = state->uses(elem->id()); size_t useId = state->uses(elem->id());
if(useId == elem->id() || state->isActive(useId)) {
if (useId == elem->id() || state->isActive(useId)) {
stream << "actively "; stream << "actively ";
} }
stream << "using " << useId << "]"; stream << "using " << useId << "]";
@ -532,7 +538,7 @@ namespace storm {
} }
} }
return stream.str(); return stream.str();
}
}
template<typename ValueType> template<typename ValueType>
std::string DFT<ValueType>::getStateString(storm::storage::BitVector const& status, DFTStateGenerationInfo const& stateGenerationInfo, size_t id) const { std::string DFT<ValueType>::getStateString(storm::storage::BitVector const& status, DFTStateGenerationInfo const& stateGenerationInfo, size_t id) const {
@ -543,7 +549,7 @@ namespace storm {
stream << storm::storage::toChar(DFTState<ValueType>::getDependencyState(status, stateGenerationInfo, elem->id())) << "[dep]"; stream << storm::storage::toChar(DFTState<ValueType>::getDependencyState(status, stateGenerationInfo, elem->id())) << "[dep]";
} else { } else {
stream << storm::storage::toChar(DFTState<ValueType>::getElementState(status, stateGenerationInfo, elem->id())); stream << storm::storage::toChar(DFTState<ValueType>::getElementState(status, stateGenerationInfo, elem->id()));
if(elem->isSpareGate()) {
if (elem->isSpareGate()) {
stream << "["; stream << "[";
size_t nrUsedChild = status.getAsInt(stateGenerationInfo.getSpareUsageIndex(elem->id()), stateGenerationInfo.usageInfoBits()); size_t nrUsedChild = status.getAsInt(stateGenerationInfo.getSpareUsageIndex(elem->id()), stateGenerationInfo.usageInfoBits());
size_t useId; size_t useId;
@ -552,7 +558,7 @@ namespace storm {
} else { } else {
useId = getChild(elem->id(), nrUsedChild); useId = getChild(elem->id(), nrUsedChild);
} }
if(useId == elem->id() || status[stateGenerationInfo.getSpareActivationIndex(useId)]) {
if (useId == elem->id() || status[stateGenerationInfo.getSpareActivationIndex(useId)]) {
stream << "actively "; stream << "actively ";
} }
stream << "using " << useId << "]"; stream << "using " << useId << "]";
@ -561,13 +567,13 @@ namespace storm {
} }
return stream.str(); return stream.str();
} }
template<typename ValueType> template<typename ValueType>
size_t DFT<ValueType>::getChild(size_t spareId, size_t nrUsedChild) const { size_t DFT<ValueType>::getChild(size_t spareId, size_t nrUsedChild) const {
STORM_LOG_ASSERT(mElements[spareId]->isSpareGate(), "Element is no spare."); STORM_LOG_ASSERT(mElements[spareId]->isSpareGate(), "Element is no spare.");
return getGate(spareId)->children()[nrUsedChild]->id(); return getGate(spareId)->children()[nrUsedChild]->id();
} }
template<typename ValueType> template<typename ValueType>
size_t DFT<ValueType>::getNrChild(size_t spareId, size_t childId) const { size_t DFT<ValueType>::getNrChild(size_t spareId, size_t childId) const {
STORM_LOG_ASSERT(mElements[spareId]->isSpareGate(), "Element is no spare."); STORM_LOG_ASSERT(mElements[spareId]->isSpareGate(), "Element is no spare.");
@ -580,8 +586,8 @@ namespace storm {
STORM_LOG_ASSERT(false, "Child not found."); STORM_LOG_ASSERT(false, "Child not found.");
return 0; return 0;
} }
template <typename ValueType>
template<typename ValueType>
std::vector<size_t> DFT<ValueType>::getIndependentSubDftRoots(size_t index) const { std::vector<size_t> DFT<ValueType>::getIndependentSubDftRoots(size_t index) const {
auto elem = getElement(index); auto elem = getElement(index);
auto ISD = elem->independentSubDft(false); auto ISD = elem->independentSubDft(false);
@ -608,7 +614,7 @@ namespace storm {
bool wellformed = true; bool wellformed = true;
// Check independence of spare modules // Check independence of spare modules
// TODO: comparing one element of each spare module sufficient? // TODO: comparing one element of each spare module sufficient?
for (auto module1 = mSpareModules.begin() ; module1 != mSpareModules.end() ; ++module1) {
for (auto module1 = mSpareModules.begin(); module1 != mSpareModules.end(); ++module1) {
size_t firstElement = module1->second.front(); size_t firstElement = module1->second.front();
for (auto module2 = std::next(module1); module2 != mSpareModules.end(); ++module2) { for (auto module2 = std::next(module1); module2 != mSpareModules.end(); ++module2) {
if (std::find(module2->second.begin(), module2->second.end(), firstElement) != module2->second.end()) { if (std::find(module2->second.begin(), module2->second.end(), firstElement) != module2->second.end()) {
@ -621,6 +627,7 @@ namespace storm {
} }
} }
// TODO check VOT gates // TODO check VOT gates
// TODO check only one constant failed event
return wellformed; return wellformed;
} }
@ -634,7 +641,7 @@ namespace storm {
STORM_LOG_TRACE("Considering ids " << index1 << ", " << index2 << " for isomorphism."); STORM_LOG_TRACE("Considering ids " << index1 << ", " << index2 << " for isomorphism.");
bool sharedSpareMode = false; bool sharedSpareMode = false;
std::map<size_t, size_t> bijection; std::map<size_t, size_t> bijection;
if (isBasicElement(index1)) { if (isBasicElement(index1)) {
if (!isBasicElement(index2)) { if (!isBasicElement(index2)) {
return {}; return {};
@ -646,18 +653,18 @@ namespace storm {
return {}; return {};
} }
} }
STORM_LOG_ASSERT(isGate(index1), "Element is no gate."); STORM_LOG_ASSERT(isGate(index1), "Element is no gate.");
STORM_LOG_ASSERT(isGate(index2), "Element is no gate."); STORM_LOG_ASSERT(isGate(index2), "Element is no gate.");
std::vector<size_t> isubdft1 = getGate(index1)->independentSubDft(false); std::vector<size_t> isubdft1 = getGate(index1)->independentSubDft(false);
std::vector<size_t> isubdft2 = getGate(index2)->independentSubDft(false); std::vector<size_t> isubdft2 = getGate(index2)->independentSubDft(false);
if(isubdft1.empty() || isubdft2.empty() || isubdft1.size() != isubdft2.size()) {
if (isubdft1.empty() || isubdft2.empty() || isubdft1.size() != isubdft2.size()) {
if (isubdft1.empty() && isubdft2.empty() && sparesAsLeaves) { if (isubdft1.empty() && isubdft2.empty() && sparesAsLeaves) {
// Check again for shared spares // Check again for shared spares
sharedSpareMode = true; sharedSpareMode = true;
isubdft1 = getGate(index1)->independentSubDft(false, true); isubdft1 = getGate(index1)->independentSubDft(false, true);
isubdft2 = getGate(index2)->independentSubDft(false, true); isubdft2 = getGate(index2)->independentSubDft(false, true);
if(isubdft1.empty() || isubdft2.empty() || isubdft1.size() != isubdft2.size()) {
if (isubdft1.empty() || isubdft2.empty() || isubdft1.size() != isubdft2.size()) {
return {}; return {};
} }
} else { } else {
@ -668,7 +675,7 @@ namespace storm {
auto LHS = colouring.colourSubdft(isubdft1); auto LHS = colouring.colourSubdft(isubdft1);
auto RHS = colouring.colourSubdft(isubdft2); auto RHS = colouring.colourSubdft(isubdft2);
auto IsoCheck = DFTIsomorphismCheck<ValueType>(LHS, RHS, *this); auto IsoCheck = DFTIsomorphismCheck<ValueType>(LHS, RHS, *this);
while (IsoCheck.findNextIsomorphism()) { while (IsoCheck.findNextIsomorphism()) {
bijection = IsoCheck.getIsomorphism(); bijection = IsoCheck.getIsomorphism();
if (sharedSpareMode) { if (sharedSpareMode) {
@ -677,7 +684,7 @@ namespace storm {
if (getElement(elementId)->isSpareGate()) { if (getElement(elementId)->isSpareGate()) {
std::shared_ptr<DFTSpare<ValueType>> spareLeft = std::static_pointer_cast<DFTSpare<ValueType>>(mElements[elementId]); std::shared_ptr<DFTSpare<ValueType>> spareLeft = std::static_pointer_cast<DFTSpare<ValueType>>(mElements[elementId]);
std::shared_ptr<DFTSpare<ValueType>> spareRight = std::static_pointer_cast<DFTSpare<ValueType>>(mElements[bijection.at(elementId)]); std::shared_ptr<DFTSpare<ValueType>> spareRight = std::static_pointer_cast<DFTSpare<ValueType>>(mElements[bijection.at(elementId)]);
if (spareLeft->nrChildren() != spareRight->nrChildren()) { if (spareLeft->nrChildren() != spareRight->nrChildren()) {
bijectionSpareCompatible = false; bijectionSpareCompatible = false;
break; break;
@ -692,13 +699,13 @@ namespace storm {
// Ignore shared child // Ignore shared child
continue; continue;
} }
// TODO generalize for more than one parent // TODO generalize for more than one parent
if (spareLeft->children().at(i)->nrParents() != 1 || spareRight->children().at(i)->nrParents() != 1) { if (spareLeft->children().at(i)->nrParents() != 1 || spareRight->children().at(i)->nrParents() != 1) {
bijectionSpareCompatible = false; bijectionSpareCompatible = false;
break; break;
} }
std::map<size_t, size_t> tmpBijection = findBijection(childLeftId, childRightId, colouring, false); std::map<size_t, size_t> tmpBijection = findBijection(childLeftId, childRightId, colouring, false);
if (!tmpBijection.empty()) { if (!tmpBijection.empty()) {
bijection.insert(tmpBijection.begin(), tmpBijection.end()); bijection.insert(tmpBijection.begin(), tmpBijection.end());
@ -731,12 +738,12 @@ namespace storm {
std::map<size_t, std::vector<std::vector<size_t>>> res; std::map<size_t, std::vector<std::vector<size_t>>> res;
// Find symmetries for gates // Find symmetries for gates
for(auto const& colourClass : completeCategories.gateCandidates) {
for (auto const& colourClass : completeCategories.gateCandidates) {
findSymmetriesHelper(colourClass.second, colouring, res); findSymmetriesHelper(colourClass.second, colouring, res);
} }
// Find symmetries for BEs // Find symmetries for BEs
for(auto const& colourClass : completeCategories.beCandidates) {
for (auto const& colourClass : completeCategories.beCandidates) {
findSymmetriesHelper(colourClass.second, colouring, res); findSymmetriesHelper(colourClass.second, colouring, res);
} }
@ -744,41 +751,42 @@ namespace storm {
} }
template<typename ValueType> template<typename ValueType>
void DFT<ValueType>::findSymmetriesHelper(std::vector<size_t> const& candidates, DFTColouring<ValueType> const& colouring, std::map<size_t, std::vector<std::vector<size_t>>>& result) const {
if(candidates.size() <= 0) {
void DFT<ValueType>::findSymmetriesHelper(std::vector<size_t> const& candidates, DFTColouring<ValueType> const& colouring,
std::map<size_t, std::vector<std::vector<size_t>>>& result) const {
if (candidates.size() <= 0) {
return; return;
} }
std::set<size_t> foundEqClassFor; std::set<size_t> foundEqClassFor;
for(auto it1 = candidates.cbegin(); it1 != candidates.cend(); ++it1) {
for (auto it1 = candidates.cbegin(); it1 != candidates.cend(); ++it1) {
std::vector<std::vector<size_t>> symClass; std::vector<std::vector<size_t>> symClass;
if(foundEqClassFor.count(*it1) > 0) {
if (foundEqClassFor.count(*it1) > 0) {
// This item is already in a class. // This item is already in a class.
continue; continue;
} }
if(!getElement(*it1)->hasOnlyStaticParents()) {
if (!getElement(*it1)->hasOnlyStaticParents()) {
continue; continue;
} }
std::tuple<std::vector<size_t>, std::vector<size_t>, std::vector<size_t>> influencedElem1Ids = getSortedParentAndDependencyIds(*it1); std::tuple<std::vector<size_t>, std::vector<size_t>, std::vector<size_t>> influencedElem1Ids = getSortedParentAndDependencyIds(*it1);
auto it2 = it1; auto it2 = it1;
for(++it2; it2 != candidates.cend(); ++it2) {
if(!getElement(*it2)->hasOnlyStaticParents()) {
for (++it2; it2 != candidates.cend(); ++it2) {
if (!getElement(*it2)->hasOnlyStaticParents()) {
continue; continue;
} }
if(influencedElem1Ids == getSortedParentAndDependencyIds(*it2)) {
if (influencedElem1Ids == getSortedParentAndDependencyIds(*it2)) {
std::map<size_t, size_t> bijection = findBijection(*it1, *it2, colouring, true); std::map<size_t, size_t> bijection = findBijection(*it1, *it2, colouring, true);
if (!bijection.empty()) { if (!bijection.empty()) {
STORM_LOG_TRACE("Subdfts are symmetric"); STORM_LOG_TRACE("Subdfts are symmetric");
foundEqClassFor.insert(*it2); foundEqClassFor.insert(*it2);
if(symClass.empty()) {
for(auto const& i : bijection) {
if (symClass.empty()) {
for (auto const& i : bijection) {
symClass.push_back(std::vector<size_t>({i.first})); symClass.push_back(std::vector<size_t>({i.first}));
} }
} }
auto symClassIt = symClass.begin(); auto symClassIt = symClass.begin();
for(auto const& i : bijection) {
for (auto const& i : bijection) {
symClassIt->emplace_back(i.second); symClassIt->emplace_back(i.second);
++symClassIt; ++symClassIt;
@ -787,7 +795,7 @@ namespace storm {
} }
} }
if(!symClass.empty()) {
if (!symClass.empty()) {
result.emplace(*it1, symClass); result.emplace(*it1, symClass);
} }
} }
@ -795,31 +803,32 @@ namespace storm {
template<typename ValueType> template<typename ValueType>
std::vector<size_t> DFT<ValueType>::findModularisationRewrite() const { std::vector<size_t> DFT<ValueType>::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<DFTGate<ValueType>>(e)->children();
for(auto const& child : children) {
auto ISD = std::static_pointer_cast<DFTGate<ValueType>>(child)->independentSubDft(true);
// In the ISD, check for other children:
std::vector<size_t> 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<DFTElement<ValueType>> const& e) { return e->id() == isdElemId; } ) != children.end()) {
// element in subtree is also child
rewrite.push_back(isdElemId);
}
}
if(rewrite.size() > 2 && rewrite.size() < children.size() - 1) {
return rewrite;
}
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<DFTGate<ValueType>>(e)->children();
for (auto const& child : children) {
auto ISD = std::static_pointer_cast<DFTGate<ValueType>>(child)->independentSubDft(true);
// In the ISD, check for other children:
std::vector<size_t> 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<DFTElement<ValueType>> const& e) { return e->id() == isdElemId; }) !=
children.end()) {
// element in subtree is also child
rewrite.push_back(isdElemId);
}
}
if (rewrite.size() > 2 && rewrite.size() < children.size() - 1) {
return rewrite;
}
}
}
}
return {};
}
}
}
return {};
} }
@ -831,14 +840,14 @@ namespace storm {
// Ingoing dependencies // Ingoing dependencies
std::vector<size_t> ingoingDeps; std::vector<size_t> ingoingDeps;
if (isBasicElement(index)) { if (isBasicElement(index)) {
for(auto const& dep : getBasicElement(index)->ingoingDependencies()) {
for (auto const& dep : getBasicElement(index)->ingoingDependencies()) {
ingoingDeps.push_back(dep->id()); ingoingDeps.push_back(dep->id());
} }
std::sort(ingoingDeps.begin(), ingoingDeps.end()); std::sort(ingoingDeps.begin(), ingoingDeps.end());
} }
// Outgoing dependencies // Outgoing dependencies
std::vector<size_t> outgoingDeps; std::vector<size_t> outgoingDeps;
for(auto const& dep : getElement(index)->outgoingDependencies()) {
for (auto const& dep : getElement(index)->outgoingDependencies()) {
outgoingDeps.push_back(dep->id()); outgoingDeps.push_back(dep->id());
} }
std::sort(outgoingDeps.begin(), outgoingDeps.end()); std::sort(outgoingDeps.begin(), outgoingDeps.end());
@ -863,10 +872,12 @@ namespace storm {
template<typename ValueType> template<typename ValueType>
void DFT<ValueType>::setRelevantEvents(std::set<size_t> const& relevantEvents, bool allowDCForRelevantEvents) const { void DFT<ValueType>::setRelevantEvents(std::set<size_t> const& relevantEvents, bool allowDCForRelevantEvents) const {
mRelevantEvents.clear();
for (auto const& elem : mElements) { for (auto const& elem : mElements) {
if (relevantEvents.find(elem->id()) != relevantEvents.end()) { if (relevantEvents.find(elem->id()) != relevantEvents.end()) {
elem->setRelevance(true); elem->setRelevance(true);
elem->setAllowDC(allowDCForRelevantEvents); elem->setAllowDC(allowDCForRelevantEvents);
mRelevantEvents.insert(elem->id());
} else { } else {
elem->setRelevance(false); elem->setRelevance(false);
elem->setAllowDC(true); elem->setAllowDC(true);
@ -876,29 +887,20 @@ namespace storm {
template<typename ValueType> template<typename ValueType>
std::set<size_t> DFT<ValueType>::getRelevantEvents() const { std::set<size_t> DFT<ValueType>::getRelevantEvents() const {
std::set<size_t> relevantEvents;
for (auto const& elem : mElements) {
if (elem->isRelevant()) {
relevantEvents.insert(elem->id());
}
}
return relevantEvents;
return mRelevantEvents;
} }
template<typename ValueType> template<typename ValueType>
std::string DFT<ValueType>::getRelevantEventsString() const { std::string DFT<ValueType>::getRelevantEventsString() const {
std::stringstream stream; std::stringstream stream;
bool first = true; bool first = true;
for (auto const& elem : mElements) {
if (elem->isRelevant()) {
if (first) {
first = false;
} else {
stream << ", ";
}
stream << elem->name() << " [" << elem->id() << "]";
for (size_t relevant_id : getRelevantEvents()) {
if (first) {
first = false;
} else {
stream << ", ";
} }
stream << getElement(relevant_id)->name() << " [" << relevant_id << "]";
} }
return stream.str(); return stream.str();
} }
@ -998,12 +1000,16 @@ namespace storm {
} }
stream << "=========================================" << std::endl; stream << "=========================================" << std::endl;
} }
// Explicitly instantiate the class. // Explicitly instantiate the class.
template class DFT<double>;
template
class DFT<double>;
#ifdef STORM_HAVE_CARL #ifdef STORM_HAVE_CARL
template class DFT<RationalFunction>;
template
class DFT<RationalFunction>;
#endif #endif
} }

3
src/storm-dft/storage/dft/DFT.h

@ -68,6 +68,7 @@ namespace storm {
std::map<size_t, size_t> mRepresentants; // id element -> id representative std::map<size_t, size_t> mRepresentants; // id element -> id representative
std::vector<std::vector<size_t>> mSymmetries; std::vector<std::vector<size_t>> mSymmetries;
std::map<size_t, DFTLayoutInfo> mLayoutInfo; std::map<size_t, DFTLayoutInfo> mLayoutInfo;
mutable std::set<size_t> mRelevantEvents;
public: public:
DFT(DFTElementVector const& elements, DFTElementPointer const& tle); DFT(DFTElementVector const& elements, DFTElementPointer const& tle);
@ -327,7 +328,7 @@ namespace storm {
/*! /*!
* Set the relevance flag for all elements according to the given relevant events. * Set the relevance flag for all elements according to the given relevant events.
* @param relevantEvents All elements which should be to relevant. All elements not occuring are set to irrelevant.
* @param relevantEvents All elements which should be to relevant. All elements not occurring are set to irrelevant.
* @param allowDCForRelevantEvents Flag whether Don't Care propagation is allowed even for relevant events. * @param allowDCForRelevantEvents Flag whether Don't Care propagation is allowed even for relevant events.
*/ */
void setRelevantEvents(std::set<size_t> const& relevantEvents, bool allowDCForRelevantEvents) const; void setRelevantEvents(std::set<size_t> const& relevantEvents, bool allowDCForRelevantEvents) const;

20
src/storm-dft/storage/dft/DFTState.cpp

@ -7,7 +7,7 @@ namespace storm {
namespace storage { namespace storage {
template<typename ValueType> template<typename ValueType>
DFTState<ValueType>::DFTState(DFT<ValueType> const& dft, DFTStateGenerationInfo const& stateGenerationInfo, size_t id) : mStatus(dft.stateBitVectorSize()), mId(id), failableElements(dft.nrElements(), dft.getRelevantEvents()), mPseudoState(false), mDft(dft), mStateGenerationInfo(stateGenerationInfo) {
DFTState<ValueType>::DFTState(DFT<ValueType> const& dft, DFTStateGenerationInfo const& stateGenerationInfo, size_t id) : mStatus(dft.stateBitVectorSize()), mId(id), failableElements(dft.nrElements()), mPseudoState(false), mDft(dft), mStateGenerationInfo(stateGenerationInfo) {
// TODO: use construct() // TODO: use construct()
// Initialize uses // Initialize uses
@ -33,7 +33,7 @@ namespace storm {
} }
template<typename ValueType> template<typename ValueType>
DFTState<ValueType>::DFTState(storm::storage::BitVector const& status, DFT<ValueType> const& dft, DFTStateGenerationInfo const& stateGenerationInfo, size_t id) : mStatus(status), mId(id), failableElements(dft.nrElements(), dft.getRelevantEvents()), mPseudoState(true), mDft(dft), mStateGenerationInfo(stateGenerationInfo) {
DFTState<ValueType>::DFTState(storm::storage::BitVector const& status, DFT<ValueType> const& dft, DFTStateGenerationInfo const& stateGenerationInfo, size_t id) : mStatus(status), mId(id), failableElements(dft.nrElements()), mPseudoState(true), mDft(dft), mStateGenerationInfo(stateGenerationInfo) {
// Intentionally left empty // Intentionally left empty
} }
@ -87,10 +87,6 @@ namespace storm {
} }
} }
// Initialize remaining relevant events
failableElements.remainingRelevantEvents = mDft.getRelevantEvents();
this->updateRemainingRelevantEvents();
mPseudoState = false; mPseudoState = false;
} }
@ -316,18 +312,6 @@ namespace storm {
} }
} }
template<typename ValueType>
void DFTState<ValueType>::updateRemainingRelevantEvents() {
for (auto it = failableElements.remainingRelevantEvents.begin(); it != failableElements.remainingRelevantEvents.end(); ) {
if (isOperational(*it)) {
++it;
} else {
// Element is not relevant anymore
it = failableElements.remainingRelevantEvents.erase(it);
}
}
}
template<typename ValueType> template<typename ValueType>
ValueType DFTState<ValueType>::getBERate(size_t id) const { ValueType DFTState<ValueType>::getBERate(size_t id) const {
STORM_LOG_ASSERT(mDft.isBasicElement(id), "Element is no BE."); STORM_LOG_ASSERT(mDft.isBasicElement(id), "Element is no BE.");

16
src/storm-dft/storage/dft/DFTState.h

@ -25,7 +25,7 @@ namespace storm {
struct FailableElements { struct FailableElements {
FailableElements(size_t nrElements, std::set<size_t> relevantEvents) : currentlyFailableBE(nrElements), remainingRelevantEvents(relevantEvents), it(currentlyFailableBE.begin()) {
FailableElements(size_t nrElements) : currentlyFailableBE(nrElements), it(currentlyFailableBE.begin()) {
// Intentionally left empty // Intentionally left empty
} }
@ -103,19 +103,10 @@ namespace storm {
return !currentlyFailableBE.empty(); return !currentlyFailableBE.empty();
} }
/*!
* Check whether at least one relevant event has not failed yet.
* @return True iff one relevant event is still operational.
*/
bool hasRemainingRelevantEvent() const {
return !remainingRelevantEvents.empty();
}
mutable bool dependency; mutable bool dependency;
storm::storage::BitVector currentlyFailableBE; storm::storage::BitVector currentlyFailableBE;
std::vector<size_t> mFailableDependencies; std::vector<size_t> mFailableDependencies;
std::set<size_t> remainingRelevantEvents;
mutable storm::storage::BitVector::const_iterator it; mutable storm::storage::BitVector::const_iterator it;
mutable std::vector<size_t>::const_iterator itDep; mutable std::vector<size_t>::const_iterator itDep;
@ -312,11 +303,6 @@ namespace storm {
*/ */
void updateDontCareDependencies(size_t id); void updateDontCareDependencies(size_t id);
/*!
* Update remaining relevant events.
*/
void updateRemainingRelevantEvents();
/** /**
* Sets the next BE as failed * Sets the next BE as failed
* @param index Index in currentlyFailableBE of BE to fail * @param index Index in currentlyFailableBE of BE to fail

Loading…
Cancel
Save