#pragma once #include #include "storm/utility/constants.h" #include "storm/utility/macros.h" namespace storm { namespace builder { /*! * Enum representing the heuristic used for deciding which states to expand. */ enum class ApproximationHeuristic { DEPTH, PROBABILITY, BOUNDDIFFERENCE }; /*! * General super class for approximation heuristics. */ template class DFTExplorationHeuristic { public: explicit DFTExplorationHeuristic(size_t id) : id(id), expand(false), lowerBound(storm::utility::zero()), upperBound(storm::utility::infinity()), depth(0), probability(storm::utility::one()) { // Intentionally left empty } DFTExplorationHeuristic(size_t id, DFTExplorationHeuristic const& predecessor, ValueType rate, ValueType exitRate) : id(id), expand(false), lowerBound(storm::utility::zero()), upperBound(storm::utility::infinity()), depth(predecessor.depth + 1), probability(storm::utility::zero()) { this->updateHeuristicValues(predecessor, rate, exitRate); } virtual ~DFTExplorationHeuristic() = default; void setBounds(ValueType lowerBound, ValueType upperBound) { this->lowerBound = lowerBound; this->upperBound = upperBound; } virtual bool updateHeuristicValues(DFTExplorationHeuristic const& predecessor, ValueType rate, ValueType exitRate) { STORM_LOG_ASSERT(!storm::utility::isZero(exitRate), "Exit rate is 0"); probability += predecessor.getProbability() * rate/exitRate; return true; } void markExpand() { expand = true; } size_t getId() const { return id; } bool isExpand() const { return expand; } size_t getDepth() const { return depth; } ValueType getProbability() const { return probability; } ValueType getLowerBound() const { return lowerBound; } ValueType getUpperBound() const { return upperBound; } virtual double getPriority() const = 0; virtual bool isSkip(double approximationThreshold) const { return !this->isExpand() && this->getPriority() < approximationThreshold; } virtual bool operator<(DFTExplorationHeuristic const& other) const { return this->getPriority() < other.getPriority(); } protected: size_t id; bool expand; ValueType lowerBound; ValueType upperBound; size_t depth; ValueType probability; }; template class DFTExplorationHeuristicDepth : public DFTExplorationHeuristic { public: DFTExplorationHeuristicDepth(size_t id) : DFTExplorationHeuristic(id) { // Intentionally left empty } DFTExplorationHeuristicDepth(size_t id, DFTExplorationHeuristic const& predecessor, ValueType rate, ValueType exitRate) : DFTExplorationHeuristic(id, predecessor, rate, exitRate) { // Intentionally left empty } bool updateHeuristicValues(DFTExplorationHeuristic const& predecessor, ValueType, ValueType) override { if (predecessor.getDepth() + 1 < this->depth) { this->depth = predecessor.getDepth() + 1; return true; } return false; } double getPriority() const override { return this->depth; } bool isSkip(double approximationThreshold) const override { return !this->expand && this->getPriority() > approximationThreshold; } bool operator<(DFTExplorationHeuristic const& other) const override { return this->getPriority() > other.getPriority(); } }; template class DFTExplorationHeuristicProbability : public DFTExplorationHeuristic { public: DFTExplorationHeuristicProbability(size_t id) : DFTExplorationHeuristic(id) { // Intentionally left empty } DFTExplorationHeuristicProbability(size_t id, DFTExplorationHeuristic const& predecessor, ValueType rate, ValueType exitRate) : DFTExplorationHeuristic(id, predecessor, rate, exitRate) { // Intentionally left empty } double getPriority() const override; }; template class DFTExplorationHeuristicBoundDifference : public DFTExplorationHeuristic { public: DFTExplorationHeuristicBoundDifference(size_t id) : DFTExplorationHeuristic(id) { // Intentionally left empty } DFTExplorationHeuristicBoundDifference(size_t id, DFTExplorationHeuristic const& predecessor, ValueType rate, ValueType exitRate) : DFTExplorationHeuristic(id, predecessor, rate, exitRate) { // Intentionally left empty } double getPriority() const override; }; } }