#ifndef EXPLICITDFTMODELBUILDERAPPROX_H #define EXPLICITDFTMODELBUILDERAPPROX_H #include "src/builder/DftExplorationHeuristic.h" #include "src/models/sparse/StateLabeling.h" #include "src/models/sparse/StandardRewardModel.h" #include "src/models/sparse/Model.h" #include "src/generator/DftNextStateGenerator.h" #include "src/storage/SparseMatrix.h" #include "src/storage/sparse/StateStorage.h" #include "src/storage/dft/DFT.h" #include "src/storage/dft/SymmetricUnits.h" #include #include #include #include #include namespace storm { namespace builder { /*! * Build a Markov chain from DFT. */ template class ExplicitDFTModelBuilderApprox { using DFTElementPointer = std::shared_ptr>; using DFTElementCPointer = std::shared_ptr const>; using DFTGatePointer = std::shared_ptr>; using DFTStatePointer = std::shared_ptr>; using DFTRestrictionPointer = std::shared_ptr>; // A structure holding the individual components of a model. struct ModelComponents { // Constructor ModelComponents(); // The transition matrix. storm::storage::SparseMatrix transitionMatrix; // The state labeling. storm::models::sparse::StateLabeling stateLabeling; // The Markovian states. storm::storage::BitVector markovianStates; // The exit rates. std::vector exitRates; // A vector that stores a labeling for each choice. boost::optional>> choiceLabeling; // A flag indicating if the model is deterministic. bool deterministicModel; }; // A class holding the information for building the transition matrix. class MatrixBuilder { public: // Constructor MatrixBuilder(bool canHaveNondeterminism); /*! * Set a mapping from a state id to the index in the matrix. * * @param id Id of the state. */ void setRemapping(StateType id) { STORM_LOG_ASSERT(id < stateRemapping.size(), "Invalid index for remapping."); stateRemapping[id] = currentRowGroup; } /*! * Create a new row group if the model is nondeterministic. */ void newRowGroup() { if (canHaveNondeterminism) { builder.newRowGroup(currentRow); } ++currentRowGroup; } /*! * Add a transition from the current row. * * @param index Target index * @param value Value of transition */ void addTransition(StateType index, ValueType value) { builder.addNextValue(currentRow, index, value); } /*! * Finish the current row. */ void finishRow() { ++currentRow; } /*! * Remap the columns in the matrix. */ void remap() { builder.replaceColumns(stateRemapping, mappingOffset); } /*! * Get the current row group. * * @return The current row group. */ StateType getCurrentRowGroup() { return currentRowGroup; } // Matrix builder. storm::storage::SparseMatrixBuilder builder; // Offset to distinguish states which will not be remapped anymore and those which will. size_t mappingOffset; // A mapping from state ids to the row group indices in which they actually reside. // TODO Matthias: avoid hack with fixed int type std::vector stateRemapping; private: // Index of the current row group. StateType currentRowGroup; // Index of the current row. StateType currentRow; // Flag indicating if row groups are needed. bool canHaveNondeterminism; }; public: // A structure holding the labeling options. struct LabelOptions { bool buildFailLabel = true; bool buildFailSafeLabel = false; std::set beLabels = {}; }; /*! * Constructor. * * @param dft DFT. * @param symmetries Symmetries in the dft. * @param enableDC Flag indicating if dont care propagation should be used. */ ExplicitDFTModelBuilderApprox(storm::storage::DFT const& dft, storm::storage::DFTIndependentSymmetries const& symmetries, bool enableDC); /*! * Build model from DFT. * * @param labelOpts Options for labeling. * @param firstTime Flag indicating if the model is built for the first time or rebuilt. * @param approximationThreshold Threshold determining when to skip exploring states. */ void buildModel(LabelOptions const& labelOpts, bool firstTime, double approximationThreshold = 0.0); /*! * Get the built model. * * @return The model built from the DFT. */ std::shared_ptr> getModel(); /*! * Get the built approximation model for either the lower or upper bound. * * @param lowerBound If true, the lower bound model is returned, else the upper bound model * * @return The model built from the DFT. */ std::shared_ptr> getModelApproximation(bool lowerBound = true); private: /*! * Explore state space of DFT. * * @param approximationThreshold Threshold to determine when to skip states. */ void exploreStateSpace(double approximationThreshold); /*! * Initialize the matrix for a refinement iteration. */ void initializeNextIteration(); /*! * Build the labeling. * * @param labelOpts Options for labeling. */ void buildLabeling(LabelOptions const& labelOpts); /*! * Add a state to the explored states (if not already there). It also handles pseudo states. * * @param state The state to add. * * @return Id of state. */ StateType getOrAddStateIndex(DFTStatePointer const& state); /*! * Set markovian flag for the current state. * * @param markovian Flag indicating if the state is markovian. */ void setMarkovian(bool markovian); /** * Change matrix to reflect the lower approximation bound. * * @param matrix Matrix to change. The change are reflected here. */ void changeMatrixLowerBound(storm::storage::SparseMatrix & matrix) const; /*! * Change matrix to reflect the upper approximation bound. * * @param matrix Matrix to change. The change are reflected here. */ void changeMatrixUpperBound(storm::storage::SparseMatrix & matrix) const; /*! * Create the model model from the model components. * * @param copy If true, all elements of the model component are copied (used for approximation). If false * they are moved to reduce the memory overhead. * * @return The model built from the model components. */ std::shared_ptr> createModel(bool copy); // Initial size of the bitvector. const size_t INITIAL_BITVECTOR_SIZE = 20000; // Offset used for pseudo states. const StateType OFFSET_PSEUDO_STATE = std::numeric_limits::max() / 2; // Dft storm::storage::DFT const& dft; // General information for state generation // TODO Matthias: use const reference std::shared_ptr stateGenerationInfo; // Flag indication if dont care propagation should be used. bool enableDC = true; //TODO Matthias: make changeable const bool mergeFailedStates = true; // Current id for new state size_t newIndex = 0; // Id of failed state size_t failedStateId = 0; // Id of initial state size_t initialStateIndex = 0; // Mapping from pseudo states to (id of concrete state, bitvector representation) std::vector> mPseudoStatesMapping; // Next state generator for exploring the state space storm::generator::DftNextStateGenerator generator; // Structure for the components of the model. ModelComponents modelComponents; // Structure for the transition matrix builder. MatrixBuilder matrixBuilder; // Internal information about the states that were explored. storm::storage::sparse::StateStorage stateStorage; // A pniority queue of states that still need to be explored. std::priority_queue, std::function> statesToExplore; // Holds all skipped states which were not yet expanded. More concretely it is a mapping from matrix indices // to the corresponding skipped states. // Notice that we need an ordered map here to easily iterate in increasing order over state ids. std::map skippedStates; }; } } #endif /* EXPLICITDFTMODELBUILDERAPPROX_H */