/* * File: AbstractSparseRegionModelChecker.cpp * Author: tim * * Created on September 9, 2015, 12:34 PM */ #include "src/modelchecker/region/AbstractSparseRegionModelChecker.h" #include "src/adapters/CarlAdapter.h" #include "src/modelchecker/region/RegionCheckResult.h" #include "src/modelchecker/results/ExplicitQuantitativeCheckResult.h" #include "src/logic/Formulas.h" #include "src/models/sparse/StandardRewardModel.h" #include "src/settings/SettingsManager.h" #include "src/settings/modules/RegionSettings.h" #include "src/utility/constants.h" #include "src/utility/graph.h" #include "src/utility/macros.h" #include "src/exceptions/InvalidArgumentException.h" #include "src/exceptions/InvalidPropertyException.h" #include "src/exceptions/InvalidStateException.h" #include "src/exceptions/InvalidSettingsException.h" #include "src/exceptions/NotImplementedException.h" #include "src/exceptions/UnexpectedException.h" #include "utility/ConversionHelper.h" #include "modelchecker/results/CheckResult.h" #include "modelchecker/results/ExplicitQuantitativeCheckResult.h" namespace storm { namespace modelchecker { namespace region { template AbstractSparseRegionModelChecker::AbstractSparseRegionModelChecker(ParametricSparseModelType const& model) : model(model), specifiedFormula(nullptr){ STORM_LOG_THROW(model.getInitialStates().getNumberOfSetBits() == 1, storm::exceptions::InvalidArgumentException, "Model is required to have exactly one initial state."); } template AbstractSparseRegionModelChecker::~AbstractSparseRegionModelChecker() { //Intentionally left empty } template ParametricSparseModelType const& AbstractSparseRegionModelChecker::getModel() const { return this->model; } template std::shared_ptr const& AbstractSparseRegionModelChecker::getSpecifiedFormula() const { return specifiedFormula; } template ConstantType AbstractSparseRegionModelChecker::getSpecifiedFormulaBound() const { return storm::utility::region::convertNumber(this->getSpecifiedFormula()->getBound()); } template bool AbstractSparseRegionModelChecker::specifiedFormulaHasUpperBound() const { return !storm::logic::isLowerBound(this->getSpecifiedFormula()->getComparisonType()); } template bool const& AbstractSparseRegionModelChecker::isComputeRewards() const { return computeRewards; } template bool const AbstractSparseRegionModelChecker::isResultConstant() const { return this->constantResult.operator bool(); } template std::shared_ptr const& AbstractSparseRegionModelChecker::getSimpleModel() const { return this->simpleModel; } template std::shared_ptr const& AbstractSparseRegionModelChecker::getSimpleFormula() const { return this->simpleFormula; } template void AbstractSparseRegionModelChecker::specifyFormula(std::shared_ptr formula) { std::chrono::high_resolution_clock::time_point timeSpecifyFormulaStart = std::chrono::high_resolution_clock::now(); STORM_LOG_DEBUG("Specifying the formula " << *formula.get()); STORM_LOG_THROW(this->canHandle(*formula), storm::exceptions::InvalidArgumentException, "Tried to specify a formula that can not be handled."); //Initialize the context for this formula if (formula->isProbabilityOperatorFormula()) { this->specifiedFormula = std::make_shared(formula->asProbabilityOperatorFormula()); this->computeRewards = false; } else if (formula->isRewardOperatorFormula()) { this->specifiedFormula = std::make_shared(formula->asRewardOperatorFormula()); this->computeRewards=true; } else { STORM_LOG_THROW(false, storm::exceptions::InvalidPropertyException, "The specified property " << this->getSpecifiedFormula() << "is not supported"); } this->constantResult = boost::none; this->simpleFormula = nullptr; this->isApproximationApplicable = false; this->approximationModel = nullptr; this->samplingModel = nullptr; //stuff for statistics: this->numOfCheckedRegions=0; this->numOfRegionsSolvedThroughSampling=0; this->numOfRegionsSolvedThroughApproximation=0; this->numOfRegionsSolvedThroughSmt=0; this->numOfRegionsExistsBoth=0; this->numOfRegionsAllSat=0; this->numOfRegionsAllViolated=0; this->timeCheckRegion=std::chrono::high_resolution_clock::duration::zero(); this->timeSampling=std::chrono::high_resolution_clock::duration::zero(); this->timeApproximation=std::chrono::high_resolution_clock::duration::zero(); this->timeSmt=std::chrono::high_resolution_clock::duration::zero(); this->timeApproxModelInstantiation=std::chrono::high_resolution_clock::duration::zero(); this->timeComputeReachabilityFunction=std::chrono::high_resolution_clock::duration::zero(); this->timeApproxModelInstantiation=std::chrono::high_resolution_clock::duration::zero(); std::chrono::high_resolution_clock::time_point timePreprocessingStart = std::chrono::high_resolution_clock::now(); this->preprocess(this->simpleModel, this->simpleFormula, isApproximationApplicable, constantResult); std::chrono::high_resolution_clock::time_point timePreprocessingEnd = std::chrono::high_resolution_clock::now(); //Check if the approximation and the sampling model needs to be computed if(!this->isResultConstant()){ if(this->isApproximationApplicable && storm::settings::regionSettings().doApprox()){ initializeApproximationModel(*this->getSimpleModel(), this->getSimpleFormula()); } if(storm::settings::regionSettings().getSampleMode()==storm::settings::modules::RegionSettings::SampleMode::INSTANTIATE || (!storm::settings::regionSettings().doSample() && storm::settings::regionSettings().getApproxMode()==storm::settings::modules::RegionSettings::ApproxMode::TESTFIRST)){ initializeSamplingModel(*this->getSimpleModel(), this->getSimpleFormula()); } } else if (this->isResultConstant() && this->constantResult.get() == storm::utility::region::convertNumber(-1.0)){ //In this case, the result is constant but has not been computed yet. so do it now! STORM_LOG_DEBUG("The Result is constant and will be computed now."); initializeSamplingModel(*this->getSimpleModel(), this->getSimpleFormula()); std::map emptySubstitution; this->getSamplingModel()->instantiate(emptySubstitution); this->constantResult = this->getSamplingModel()->computeInitialStateValue(); } //some more information for statistics... std::chrono::high_resolution_clock::time_point timeSpecifyFormulaEnd = std::chrono::high_resolution_clock::now(); this->timeSpecifyFormula= timeSpecifyFormulaEnd - timeSpecifyFormulaStart; this->timePreprocessing = timePreprocessingEnd - timePreprocessingStart; } template void AbstractSparseRegionModelChecker::initializeApproximationModel(ParametricSparseModelType const& model, std::shared_ptr formula) { std::chrono::high_resolution_clock::time_point timeInitApproxModelStart = std::chrono::high_resolution_clock::now(); STORM_LOG_DEBUG("Initializing the Approximation Model..."); STORM_LOG_THROW(this->isApproximationApplicable, storm::exceptions::UnexpectedException, "Approximation model requested but approximation is not applicable"); this->approximationModel=std::make_shared>(model, formula); std::chrono::high_resolution_clock::time_point timeInitApproxModelEnd = std::chrono::high_resolution_clock::now(); this->timeInitApproxModel=timeInitApproxModelEnd - timeInitApproxModelStart; STORM_LOG_DEBUG("Initialized Approximation Model"); } template void AbstractSparseRegionModelChecker::initializeSamplingModel(ParametricSparseModelType const& model, std::shared_ptr formula) { STORM_LOG_DEBUG("Initializing the Sampling Model...."); std::chrono::high_resolution_clock::time_point timeInitSamplingModelStart = std::chrono::high_resolution_clock::now(); this->samplingModel=std::make_shared>(model, formula); std::chrono::high_resolution_clock::time_point timeInitSamplingModelEnd = std::chrono::high_resolution_clock::now(); this->timeInitSamplingModel = timeInitSamplingModelEnd - timeInitSamplingModelStart; STORM_LOG_DEBUG("Initialized Sampling Model"); } template void AbstractSparseRegionModelChecker::checkRegions(std::vector>& regions) { STORM_LOG_DEBUG("Checking " << regions.size() << "regions."); std::cout << "Checking " << regions.size() << " regions. Progress: "; std::cout.flush(); uint_fast64_t progress=0; uint_fast64_t checkedRegions=0; for(auto& region : regions){ this->checkRegion(region); if((checkedRegions++)*10/regions.size()==progress){ std::cout << progress++; std::cout.flush(); } } std::cout << " done!" << std::endl; } template void AbstractSparseRegionModelChecker::checkRegion(ParameterRegion& region) { std::chrono::high_resolution_clock::time_point timeCheckRegionStart = std::chrono::high_resolution_clock::now(); ++this->numOfCheckedRegions; STORM_LOG_THROW(this->getSpecifiedFormula()!=nullptr, storm::exceptions::InvalidStateException, "Tried to analyze a region although no property has been specified" ); STORM_LOG_DEBUG("Analyzing the region " << region.toString()); //std::cout << "Analyzing the region " << region.toString() << std::endl; //switches for the different steps. bool done=false; STORM_LOG_WARN_COND( (!storm::settings::regionSettings().doApprox() || this->isApproximationApplicable), "the approximation is only correct if the model has only linear functions. As this is not the case, approximation is deactivated"); bool doApproximation=storm::settings::regionSettings().doApprox() && this->isApproximationApplicable; bool doSampling=storm::settings::regionSettings().doSample(); bool doSmt=storm::settings::regionSettings().doSmt(); if(!done && this->isResultConstant()){ STORM_LOG_DEBUG("Checking a region although the result is constant, i.e., independent of the region. This makes sense none."); if(this->valueIsInBoundOfFormula(this->getReachabilityValue(region.getSomePoint()))){ region.setCheckResult(RegionCheckResult::ALLSAT); } else{ region.setCheckResult(RegionCheckResult::ALLVIOLATED); } done=true; } std::chrono::high_resolution_clock::time_point timeApproximationStart = std::chrono::high_resolution_clock::now(); std::vector lowerBounds; std::vector upperBounds; if(!done && doApproximation){ STORM_LOG_DEBUG("Checking approximative values..."); if(this->checkApproximativeValues(region, lowerBounds, upperBounds)){ ++this->numOfRegionsSolvedThroughApproximation; STORM_LOG_DEBUG("Result '" << region.getCheckResult() <<"' obtained through approximation."); done=true; } } std::chrono::high_resolution_clock::time_point timeApproximationEnd = std::chrono::high_resolution_clock::now(); std::chrono::high_resolution_clock::time_point timeSamplingStart = std::chrono::high_resolution_clock::now(); if(!done && doSampling){ STORM_LOG_DEBUG("Checking sample points..."); if(this->checkSamplePoints(region)){ ++this->numOfRegionsSolvedThroughSampling; STORM_LOG_DEBUG("Result '" << region.getCheckResult() <<"' obtained through sampling."); done=true; } } std::chrono::high_resolution_clock::time_point timeSamplingEnd = std::chrono::high_resolution_clock::now(); std::chrono::high_resolution_clock::time_point timeSmtStart = std::chrono::high_resolution_clock::now(); if(!done && doSmt){ STORM_LOG_DEBUG("Checking with Smt Solving..."); if(this->checkSmt(region)){ ++this->numOfRegionsSolvedThroughSmt; STORM_LOG_DEBUG("Result '" << region.getCheckResult() <<"' obtained through Smt Solving."); done=true; } } std::chrono::high_resolution_clock::time_point timeSmtEnd = std::chrono::high_resolution_clock::now(); //some information for statistics... std::chrono::high_resolution_clock::time_point timeCheckRegionEnd = std::chrono::high_resolution_clock::now(); this->timeCheckRegion += timeCheckRegionEnd-timeCheckRegionStart; this->timeSampling += timeSamplingEnd - timeSamplingStart; this->timeApproximation += timeApproximationEnd - timeApproximationStart; this->timeSmt += timeSmtEnd - timeSmtStart; switch(region.getCheckResult()){ case RegionCheckResult::EXISTSBOTH: ++this->numOfRegionsExistsBoth; break; case RegionCheckResult::ALLSAT: ++this->numOfRegionsAllSat; break; case RegionCheckResult::ALLVIOLATED: ++this->numOfRegionsAllViolated; break; default: break; } } template bool AbstractSparseRegionModelChecker::checkApproximativeValues(ParameterRegion& region, std::vector& lowerBounds, std::vector& upperBounds) { std::chrono::high_resolution_clock::time_point timeMDPBuildStart = std::chrono::high_resolution_clock::now(); this->getApproximationModel()->instantiate(region); std::chrono::high_resolution_clock::time_point timeMDPBuildEnd = std::chrono::high_resolution_clock::now(); this->timeApproxModelInstantiation += timeMDPBuildEnd-timeMDPBuildStart; // Decide whether to prove allsat or allviolated. bool proveAllSat; switch (region.getCheckResult()){ case RegionCheckResult::UNKNOWN: switch(storm::settings::regionSettings().getApproxMode()){ case storm::settings::modules::RegionSettings::ApproxMode::TESTFIRST: //Sample a single point to know whether we should try to prove ALLSAT or ALLVIOLATED checkPoint(region,region.getSomePoint(), false); proveAllSat= (region.getCheckResult()==RegionCheckResult::EXISTSSAT); break; case storm::settings::modules::RegionSettings::ApproxMode::GUESSALLSAT: proveAllSat=true; break; case storm::settings::modules::RegionSettings::ApproxMode::GUESSALLVIOLATED: proveAllSat=false; break; default: STORM_LOG_THROW(false, storm::exceptions::InvalidSettingsException, "The specified approxmode is not supported"); } break; case RegionCheckResult::ALLSAT: STORM_LOG_WARN("The checkresult of the current region should not be conclusive (ALLSAT)"); //Intentionally no break; case RegionCheckResult::EXISTSSAT: proveAllSat=true; break; case RegionCheckResult::ALLVIOLATED: STORM_LOG_WARN("The checkresult of the current region should not be conclusive (ALLViolated)"); //Intentionally no break; case RegionCheckResult::EXISTSVIOLATED: proveAllSat=false; break; default: STORM_LOG_WARN("The checkresult of the current region should not be conclusive, i.e. it should be either EXISTSSAT or EXISTSVIOLATED or UNKNOWN in order to apply approximative values"); proveAllSat=true; } bool formulaSatisfied; if((this->specifiedFormulaHasUpperBound() && proveAllSat) || (!this->specifiedFormulaHasUpperBound() && !proveAllSat)){ //these are the cases in which we need to compute upper bounds upperBounds = this->getApproximationModel()->computeValues(storm::solver::OptimizationDirection::Maximize)->template asExplicitQuantitativeCheckResult().getValueVector(); lowerBounds = std::vector(); formulaSatisfied = this->valueIsInBoundOfFormula(upperBounds[*this->getApproximationModel()->getModel()->getInitialStates().begin()]); } else{ //for the remaining cases we compute lower bounds lowerBounds = this->getApproximationModel()->computeValues(storm::solver::OptimizationDirection::Minimize)->template asExplicitQuantitativeCheckResult().getValueVector(); upperBounds = std::vector(); formulaSatisfied = this->valueIsInBoundOfFormula(lowerBounds[*this->getApproximationModel()->getModel()->getInitialStates().begin()]); } //check if approximation was conclusive if(proveAllSat && formulaSatisfied){ region.setCheckResult(RegionCheckResult::ALLSAT); return true; } if(!proveAllSat && !formulaSatisfied){ region.setCheckResult(RegionCheckResult::ALLVIOLATED); return true; } if(region.getCheckResult()==RegionCheckResult::UNKNOWN){ //In this case, it makes sense to try to prove the contrary statement proveAllSat=!proveAllSat; if(lowerBounds.empty()){ lowerBounds = this->getApproximationModel()->computeValues(storm::solver::OptimizationDirection::Minimize)->template asExplicitQuantitativeCheckResult().getValueVector(); formulaSatisfied=this->valueIsInBoundOfFormula(lowerBounds[*this->getApproximationModel()->getModel()->getInitialStates().begin()]); } else{ upperBounds = this->getApproximationModel()->computeValues(storm::solver::OptimizationDirection::Maximize)->template asExplicitQuantitativeCheckResult().getValueVector(); formulaSatisfied=this->valueIsInBoundOfFormula(upperBounds[*this->getApproximationModel()->getModel()->getInitialStates().begin()]); } //check if approximation was conclusive if(proveAllSat && formulaSatisfied){ region.setCheckResult(RegionCheckResult::ALLSAT); return true; } if(!proveAllSat && !formulaSatisfied){ region.setCheckResult(RegionCheckResult::ALLVIOLATED); return true; } } //if we reach this point than the result is still inconclusive. return false; } template std::shared_ptr> const& AbstractSparseRegionModelChecker::getApproximationModel() { if(this->approximationModel==nullptr){ STORM_LOG_WARN("Approximation model requested but it has not been initialized when specifying the formula. Will initialize it now."); initializeApproximationModel(*this->getSimpleModel(), this->getSimpleFormula()); } return this->approximationModel; } template bool AbstractSparseRegionModelChecker::checkSamplePoints(ParameterRegion& region) { auto samplingPoints = region.getVerticesOfRegion(region.getVariables()); //test the 4 corner points for (auto const& point : samplingPoints){ if(checkPoint(region, point)){ return true; } } return false; } template ConstantType AbstractSparseRegionModelChecker::getReachabilityValue(std::map const& point) { if(this->isResultConstant()){ return this->constantResult.get(); } this->getSamplingModel()->instantiate(point); return this->getSamplingModel()->computeInitialStateValue(); } template std::shared_ptr> const& AbstractSparseRegionModelChecker::getSamplingModel() { if(this->samplingModel==nullptr){ STORM_LOG_WARN("Sampling model requested but it has not been initialized when specifying the formula. Will initialize it now."); initializeSamplingModel(*this->getSimpleModel(), this->getSimpleFormula()); } return this->samplingModel; } template bool AbstractSparseRegionModelChecker::valueIsInBoundOfFormula(ConstantType const& value){ STORM_LOG_THROW(this->getSpecifiedFormula()!=nullptr, storm::exceptions::InvalidStateException, "Tried to compare a value to the bound of a formula, but no formula specified."); switch (this->getSpecifiedFormula()->getComparisonType()) { case storm::logic::ComparisonType::Greater: return (value > this->getSpecifiedFormulaBound()); case storm::logic::ComparisonType::GreaterEqual: return (value >= this->getSpecifiedFormulaBound()); case storm::logic::ComparisonType::Less: return (value < this->getSpecifiedFormulaBound()); case storm::logic::ComparisonType::LessEqual: return (value <= this->getSpecifiedFormulaBound()); default: STORM_LOG_THROW(false, storm::exceptions::InvalidArgumentException, "the comparison relation of the formula is not supported"); } } template bool AbstractSparseRegionModelChecker::valueIsInBoundOfFormula(CoefficientType const& value){ return valueIsInBoundOfFormula(storm::utility::region::convertNumber(value)); } template void AbstractSparseRegionModelChecker::printStatisticsToStream(std::ostream& outstream) { STORM_LOG_DEBUG("Printing statistics"); if(this->getSpecifiedFormula()==nullptr){ outstream << "Region Model Checker Statistics Error: No formula specified." << std::endl; return; } std::chrono::milliseconds timeSpecifyFormulaInMilliseconds = std::chrono::duration_cast(this->timeSpecifyFormula); std::chrono::milliseconds timePreprocessingInMilliseconds = std::chrono::duration_cast(this->timePreprocessing); std::chrono::milliseconds timeInitSamplingModelInMilliseconds = std::chrono::duration_cast(this->timeInitSamplingModel); std::chrono::milliseconds timeInitApproxModelInMilliseconds = std::chrono::duration_cast(this->timeInitApproxModel); std::chrono::milliseconds timeComputeReachabilityFunctionInMilliseconds = std::chrono::duration_cast(this->timeComputeReachabilityFunction); std::chrono::milliseconds timeCheckRegionInMilliseconds = std::chrono::duration_cast(this->timeCheckRegion); std::chrono::milliseconds timeSammplingInMilliseconds = std::chrono::duration_cast(this->timeSampling); std::chrono::milliseconds timeApproximationInMilliseconds = std::chrono::duration_cast(this->timeApproximation); std::chrono::milliseconds timeApproxModelInstantiationInMilliseconds = std::chrono::duration_cast(this->timeApproxModelInstantiation); std::chrono::milliseconds timeSmtInMilliseconds = std::chrono::duration_cast(this->timeSmt); std::chrono::high_resolution_clock::duration timeOverall = timeSpecifyFormula + timeCheckRegion; // + ... std::chrono::milliseconds timeOverallInMilliseconds = std::chrono::duration_cast(timeOverall); uint_fast64_t numOfSolvedRegions= this->numOfRegionsExistsBoth + this->numOfRegionsAllSat + this->numOfRegionsAllViolated; outstream << std::endl << "Region Model Checker Statistics:" << std::endl; outstream << "-----------------------------------------------" << std::endl; outstream << "Model: " << this->model.getNumberOfStates() << " states, " << this->model.getNumberOfTransitions() << " transitions." << std::endl; outstream << "Formula: " << *this->getSpecifiedFormula() << std::endl; if(this->isResultConstant()){ outstream << "The requested value is constant (i.e. independent of any parameters)" << std::endl; } else{ outstream << "Simple model: " << this->getSimpleModel()->getNumberOfStates() << " states, " << this->getSimpleModel()->getNumberOfTransitions() << " transitions" << std::endl; } outstream << "Approximation is " << (this->isApproximationApplicable ? "" : "not ") << "applicable" << std::endl; outstream << "Number of checked regions: " << this->numOfCheckedRegions << std::endl; if(this->numOfCheckedRegions>0){ outstream << " Number of solved regions: " << numOfSolvedRegions << "(" << numOfSolvedRegions*100/this->numOfCheckedRegions << "%)" << std::endl; outstream << " AllSat: " << this->numOfRegionsAllSat << "(" << this->numOfRegionsAllSat*100/this->numOfCheckedRegions << "%)" << std::endl; outstream << " AllViolated: " << this->numOfRegionsAllViolated << "(" << this->numOfRegionsAllViolated*100/this->numOfCheckedRegions << "%)" << std::endl; outstream << " ExistsBoth: " << this->numOfRegionsExistsBoth << "(" << this->numOfRegionsExistsBoth*100/this->numOfCheckedRegions << "%)" << std::endl; outstream << " Unsolved: " << this->numOfCheckedRegions - numOfSolvedRegions << "(" << (this->numOfCheckedRegions - numOfSolvedRegions)*100/this->numOfCheckedRegions << "%)" << std::endl; outstream << " -- Note: %-numbers are relative to the NUMBER of regions, not the size of their area --" << std::endl; outstream << " " << this->numOfRegionsSolvedThroughApproximation << " regions solved through Approximation" << std::endl; outstream << " " << this->numOfRegionsSolvedThroughSampling << " regions solved through Sampling" << std::endl; outstream << " " << this->numOfRegionsSolvedThroughSmt << " regions solved through Smt" << std::endl; outstream << std::endl; } outstream << "Running times:" << std::endl; outstream << " " << timeOverallInMilliseconds.count() << "ms overall (excluding model parsing, bisimulation (if applied))" << std::endl; outstream << " " << timeSpecifyFormulaInMilliseconds.count() << "ms Initialization for the specified formula, including... " << std::endl; outstream << " " << timePreprocessingInMilliseconds.count() << "ms for Preprocessing (mainly: state elimination of const transitions), including" << std::endl; outstream << " " << timeComputeReachabilityFunctionInMilliseconds.count() << "ms to compute the reachability function" << std::endl; outstream << " " << timeInitApproxModelInMilliseconds.count() << "ms to initialize the Approximation Model" << std::endl; outstream << " " << timeInitSamplingModelInMilliseconds.count() << "ms to initialize the Sampling Model" << std::endl; outstream << " " << timeCheckRegionInMilliseconds.count() << "ms Region Check including... " << std::endl; outstream << " " << timeApproximationInMilliseconds.count() << "ms Approximation including... " << std::endl; outstream << " " << timeApproxModelInstantiationInMilliseconds.count() << "ms for instantiation of the approximation model" << std::endl; outstream << " " << timeSammplingInMilliseconds.count() << "ms Sampling " << std::endl; outstream << " " << timeSmtInMilliseconds.count() << "ms Smt solving" << std::endl; outstream << "-----------------------------------------------" << std::endl; } //note: for other template instantiations, add rules for the typedefs of VariableType and CoefficientType in utility/regions.h #ifdef STORM_HAVE_CARL template class AbstractSparseRegionModelChecker>, double>; template class AbstractSparseRegionModelChecker>, double>; #endif } // namespace region } //namespace modelchecker } //namespace storm