@ -606,9 +606,53 @@ namespace storm { 
		
	
		
			
				                bool  inputEnabled ;                 bool  inputEnabled ;  
		
	
		
			
				            } ;             } ;  
		
	
		
			
				                         
		
	
		
			
				            struct  ActionIdentification  {  
		
	
		
			
				                ActionIdentification ( uint64_t  actionIndex )  :  actionIndex ( actionIndex ) ,  synchronizationVectorIndex ( boost : : none )  {  
		
	
		
			
				                    // Intentionally left empty.
  
		
	
		
			
				                }  
		
	
		
			
				                 
		
	
		
			
				                ActionIdentification ( uint64_t  actionIndex ,  uint64_t  synchronizationVectorIndex )  :  actionIndex ( actionIndex ) ,  synchronizationVectorIndex ( synchronizationVectorIndex )  {  
		
	
		
			
				                    // Intentionally left empty.
  
		
	
		
			
				                }  
		
	
		
			
				                 
		
	
		
			
				                ActionIdentification ( uint64_t  actionIndex ,  boost : : optional < uint64_t >  synchronizationVectorIndex )  :  actionIndex ( actionIndex ) ,  synchronizationVectorIndex ( synchronizationVectorIndex )  {  
		
	
		
			
				                    // Intentionally left empty.
  
		
	
		
			
				                }  
		
	
		
			
				                 
		
	
		
			
				                bool  operator = = ( ActionIdentification  const &  other )  const  {  
		
	
		
			
				                    bool  result  =  actionIndex  = =  other . actionIndex ;  
		
	
		
			
				                    if  ( synchronizationVectorIndex )  {  
		
	
		
			
				                        if  ( other . synchronizationVectorIndex )  {  
		
	
		
			
				                            result  & =  synchronizationVectorIndex . get ( )  = =  other . synchronizationVectorIndex . get ( ) ;  
		
	
		
			
				                        }  else  {  
		
	
		
			
				                            result  =  false ;  
		
	
		
			
				                        }  
		
	
		
			
				                    }  else  {  
		
	
		
			
				                        if  ( other . synchronizationVectorIndex )  {  
		
	
		
			
				                            result  =  false ;  
		
	
		
			
				                        }  
		
	
		
			
				                    }  
		
	
		
			
				                    return  result ;  
		
	
		
			
				                }  
		
	
		
			
				                 
		
	
		
			
				                uint64_t  actionIndex ;  
		
	
		
			
				                boost : : optional < uint64_t >  synchronizationVectorIndex ;  
		
	
		
			
				            } ;  
		
	
		
			
				             
		
	
		
			
				            struct  ActionIdentificationHash  {  
		
	
		
			
				                std : : size_t  operator ( ) ( ActionIdentification  const &  identification )  const  {  
		
	
		
			
				                    std : : size_t  seed  =  0 ;  
		
	
		
			
				                    boost : : hash_combine ( seed ,  identification . actionIndex ) ;  
		
	
		
			
				                    if  ( identification . synchronizationVectorIndex )  {  
		
	
		
			
				                        boost : : hash_combine ( seed ,  identification . synchronizationVectorIndex . get ( ) ) ;  
		
	
		
			
				                    }  
		
	
		
			
				                    return  seed ;  
		
	
		
			
				                }  
		
	
		
			
				            } ;  
		
	
		
			
				             
		
	
		
			
				            // This structure represents a subcomponent of a composition.
             // This structure represents a subcomponent of a composition.
  
		
	
		
			
				            struct  AutomatonDd  {             struct  AutomatonDd  {  
		
	
		
			
				                AutomatonDd ( storm : : dd : : Add < Type ,  ValueType >  const &  identity ,  std : : map < storm : : expressions : : Variable ,  storm : : dd : : Add < Type ,  ValueType > >  const &  transientLocationAssignments  =  { } )  :  actionIndexToAction ( ) ,  transientLocationAssignments ( transientLocationAssignments ) ,  identity ( identity ) ,  localNondeterminismVariables ( std : : make_pair < uint64_t ,  uint64_t > ( 0 ,  0 ) )  {  
		
	
		
			
				                AutomatonDd ( storm : : dd : : Add < Type ,  ValueType >  const &  identity ,  std : : map < storm : : expressions : : Variable ,  storm : : dd : : Add < Type ,  ValueType > >  const &  transientLocationAssignments  =  { } )  :  actions  ( ) ,  transientLocationAssignments ( transientLocationAssignments ) ,  identity ( identity ) ,  localNondeterminismVariables ( std : : make_pair < uint64_t ,  uint64_t > ( 0 ,  0 ) )  {  
		
	
		
			
				                    // Intentionally left empty.
                     // Intentionally left empty.
  
		
	
		
			
				                }                 }  
		
	
		
			
				                                 
		
	
	
		
			
				
					
					
					
						
							 
					
				 
				@ -633,8 +677,8 @@ namespace storm { 
		
	
		
			
				                    setHighestLocalNondeterminismVariable ( std : : max ( localNondeterminismVariables . second ,  getHighestLocalNondeterminismVariable ( ) ) ) ;                     setHighestLocalNondeterminismVariable ( std : : max ( localNondeterminismVariables . second ,  getHighestLocalNondeterminismVariable ( ) ) ) ;  
		
	
		
			
				                }                 }  
		
	
		
			
				                                 
		
	
		
			
				                // A mapping from action indice s to the action DDs.
  
		
	
		
			
				                std : : map < uint64_t ,  ActionDd >  actionIndexToAction ;  
		
	
		
			
				                // A mapping from action identification s to the action DDs.
  
		
	
		
			
				                std : : unordered_ map< ActionIdentification ,  ActionDd ,  ActionIdentificationHash >  actions ;  
		
	
		
			
				                                 
		
	
		
			
				                // A mapping from transient variables to their location-based transient assignment values.
                 // A mapping from transient variables to their location-based transient assignment values.
  
		
	
		
			
				                std : : map < storm : : expressions : : Variable ,  storm : : dd : : Add < Type ,  ValueType > >  transientLocationAssignments ;                 std : : map < storm : : expressions : : Variable ,  storm : : dd : : Add < Type ,  ValueType > >  transientLocationAssignments ;  
		
	
	
		
			
				
					
					
					
						
							 
					
				 
				@ -644,7 +688,6 @@ namespace storm { 
		
	
		
			
				                                 
		
	
		
			
				                // The local nondeterminism variables used by this action DD, given as the lowest and highest variable index.
                 // The local nondeterminism variables used by this action DD, given as the lowest and highest variable index.
  
		
	
		
			
				                std : : pair < uint64_t ,  uint64_t >  localNondeterminismVariables ;                 std : : pair < uint64_t ,  uint64_t >  localNondeterminismVariables ;  
		
	
		
			
				
 
		
	
		
			
				            } ;             } ;  
		
	
		
			
				                         
		
	
		
			
				            CombinedEdgesSystemComposer ( storm : : jani : : Model  const &  model ,  storm : : jani : : CompositionInformation  const &  actionInformation ,  CompositionVariables < Type ,  ValueType >  const &  variables ,  std : : vector < storm : : expressions : : Variable >  const &  transientVariables )  :  SystemComposer < Type ,  ValueType > ( model ,  variables ,  transientVariables ) ,  actionInformation ( actionInformation )  {             CombinedEdgesSystemComposer ( storm : : jani : : Model  const &  model ,  storm : : jani : : CompositionInformation  const &  actionInformation ,  CompositionVariables < Type ,  ValueType >  const &  variables ,  std : : vector < storm : : expressions : : Variable >  const &  transientVariables )  :  SystemComposer < Type ,  ValueType > ( model ,  variables ,  transientVariables ) ,  actionInformation ( actionInformation )  {  
		
	
	
		
			
				
					
					
					
						
							 
					
				 
				@ -654,65 +697,113 @@ namespace storm { 
		
	
		
			
				            storm : : jani : : CompositionInformation  const &  actionInformation ;             storm : : jani : : CompositionInformation  const &  actionInformation ;  
		
	
		
			
				
 
		
	
		
			
				            ComposerResult < Type ,  ValueType >  compose ( )  override  {             ComposerResult < Type ,  ValueType >  compose ( )  override  {  
		
	
		
			
				                std : : map < uint64_t ,  uint64_t >  actionIndexToLocalNondeterminismVariableOffset ;  
		
	
		
			
				                for  ( auto  const &  actionIndex  :  actionInformation . getNonSilentActionIndices ( ) )  {  
		
	
		
			
				                    actionIndexToLocalNondeterminismVariableOffset [ actionIndex ]  =  0  ;  
		
	
		
			
				                STORM_LOG_THROW ( this - > model . hasStandardCompliantComposition ( ) ,  storm : : exceptions : : WrongFormatException ,  " Model builder only supports non-nested parallel compositions. " ) ;  
		
	
		
			
				                AutomatonDd  globalAutomaton  =  boost : : any_cast < AutomatonDd > ( this - > model . getSystemComposition ( ) . accept ( * this ,  boost : : any ( ) ) ) ;  
		
	
		
			
				                return  buildSystemFromAutomaton ( globalAutomaton ) ;  
		
	
		
			
				            }             }  
		
	
		
			
				                actionIndexToLocalNondeterminismVariableOffset [ storm : : jani : : Model : : SILENT_ACTION_INDEX ]  =  0 ;  
		
	
		
			
				                         
		
	
		
			
				                AutomatonDd  globalAutomaton  =  boost : : any_cast < AutomatonDd > ( this - > model . getSystemComposition ( ) . accept ( * this ,  actionIndexToLocalNondeterminismVariableOffset ) ) ;  
		
	
		
			
				                return  buildSystemFromAutomaton ( globalAutomaton ) ;  
		
	
		
			
				            struct  ActionInstantiation  {  
		
	
		
			
				                ActionInstantiation ( uint64_t  actionIndex ,  uint64_t  synchronizationVectorIndex ,  uint64_t  localNondeterminismVariableOffset )  :  actionIndex ( actionIndex ) ,  synchronizationVectorIndex ( synchronizationVectorIndex ) ,  localNondeterminismVariableOffset ( localNondeterminismVariableOffset )  {  
		
	
		
			
				                    // Intentionally left empty.
  
		
	
		
			
				                }  
		
	
		
			
				
 
		
	
		
			
				                ActionInstantiation ( uint64_t  actionIndex ,  uint64_t  localNondeterminismVariableOffset )  :  actionIndex ( actionIndex ) ,  localNondeterminismVariableOffset ( localNondeterminismVariableOffset )  {  
		
	
		
			
				                    // Intentionally left empty.
  
		
	
		
			
				                }  
		
	
		
			
				
 
		
	
		
			
				                bool  operator = = ( ActionInstantiation  const &  other )  const  {  
		
	
		
			
				                    bool  result  =  actionIndex  = =  other . actionIndex ;  
		
	
		
			
				                    result  & =  localNondeterminismVariableOffset  = =  other . localNondeterminismVariableOffset ;  
		
	
		
			
				                    if  ( synchronizationVectorIndex )  {  
		
	
		
			
				                        if  ( ! other . synchronizationVectorIndex )  {  
		
	
		
			
				                            result  =  false ;  
		
	
		
			
				                        }  else  {  
		
	
		
			
				                            result  & =  synchronizationVectorIndex . get ( )  = =  other . synchronizationVectorIndex . get ( ) ;  
		
	
		
			
				                        }  
		
	
		
			
				                    }  else  {  
		
	
		
			
				                        if  ( other . synchronizationVectorIndex )  {  
		
	
		
			
				                            result  =  false ;  
		
	
		
			
				                        }                         }  
		
	
		
			
				                    }  
		
	
		
			
				                    return  result ;  
		
	
		
			
				                }  
		
	
		
			
				                 
		
	
		
			
				                uint64_t  actionIndex ;  
		
	
		
			
				                boost : : optional < uint64_t >  synchronizationVectorIndex ;  
		
	
		
			
				                uint64_t  localNondeterminismVariableOffset ;  
		
	
		
			
				            } ;  
		
	
		
			
				             
		
	
		
			
				            struct  ActionInstantiationHash  {  
		
	
		
			
				                std : : size_t  operator ( ) ( ActionInstantiation  const &  instantiation )  const  {  
		
	
		
			
				                    std : : size_t  seed  =  0 ;  
		
	
		
			
				                    boost : : hash_combine ( seed ,  instantiation . actionIndex ) ;  
		
	
		
			
				                    boost : : hash_combine ( seed ,  instantiation . localNondeterminismVariableOffset ) ;  
		
	
		
			
				                    if  ( instantiation . synchronizationVectorIndex )  {  
		
	
		
			
				                        boost : : hash_combine ( seed ,  instantiation . synchronizationVectorIndex . get ( ) ) ;  
		
	
		
			
				                    }  
		
	
		
			
				                    return  seed ;  
		
	
		
			
				                }  
		
	
		
			
				            } ;  
		
	
		
			
				             
		
	
		
			
				            typedef  std : : map < uint64_t ,  std : : vector < ActionInstantiation > >  ActionInstantiations ;  
		
	
		
			
				                         
		
	
		
			
				            boost : : any  visit ( storm : : jani : : AutomatonComposition  const &  composition ,  boost : : any  const &  data )  override  {             boost : : any  visit ( storm : : jani : : AutomatonComposition  const &  composition ,  boost : : any  const &  data )  override  {  
		
	
		
			
				                std : : map < uint64_t ,  uint64_t >  const &  actionIndexToLocalNondeterminismVariableOffset  =  boost : : any_cast < std : : map < uint_fast64_t ,  uint_fast64_t >  const & > ( data ) ;  
		
	
		
			
				                ActionInstantiations  actionInstantiations ;  
		
	
		
			
				                if  ( data . empty ( ) )  {  
		
	
		
			
				                    // If no data was provided, this is the top level element in which case we build the full automaton.
  
		
	
		
			
				                    for  ( auto  const &  actionIndex  :  actionInformation . getNonSilentActionIndices ( ) )  {  
		
	
		
			
				                        actionInstantiations [ actionIndex ] . emplace_back ( actionIndex ,  0 ) ;  
		
	
		
			
				                    }  
		
	
		
			
				                    actionInstantiations [ storm : : jani : : Model : : SILENT_ACTION_INDEX ] . emplace_back ( storm : : jani : : Model : : SILENT_ACTION_INDEX ,  0 ) ;  
		
	
		
			
				                }  
		
	
		
			
				                                 
		
	
		
			
				                std : : set < uint64_t >  inputEnabledActionIndices ;                 std : : set < uint64_t >  inputEnabledActionIndices ;  
		
	
		
			
				                for  ( auto  const &  actionName  :  composition . getInputEnabledActions ( ) )  {                 for  ( auto  const &  actionName  :  composition . getInputEnabledActions ( ) )  {  
		
	
		
			
				                    inputEnabledActionIndices . insert ( actionInformation . getActionIndex ( actionName ) ) ;                     inputEnabledActionIndices . insert ( actionInformation . getActionIndex ( actionName ) ) ;  
		
	
		
			
				                }                 }  
		
	
		
			
				                                 
		
	
		
			
				                return  buildAutomatonDd ( composition . getAutomatonName ( ) ,  actionIndexToLocalNondeterminismVariableOffset ,  inputEnabledActionIndices ) ;  
		
	
		
			
				                return  buildAutomatonDd ( composition . getAutomatonName ( ) ,  data . empty ( )  ?  actionInstantiations  :  boost : : any_cast < ActionInstantiations  const & > ( data ) ,  inputEnabledActionIndices ) ;  
		
	
		
			
				            }             }  
		
	
		
			
				                         
		
	
		
			
				            boost : : any  visit ( storm : : jani : : ParallelComposition  const &  composition ,  boost : : any  const &  data )  override  {             boost : : any  visit ( storm : : jani : : ParallelComposition  const &  composition ,  boost : : any  const &  data )  override  {  
		
	
		
			
				                std : : map < uint64_t ,  uint64_t >  const &  actionIndexToLocalNondeterminismVariableOffset  =  boost : : any_cast < std : : map < uint64_t ,  uint64_t >  const & > ( data ) ;  
		
	
		
			
				                STORM_LOG_ASSERT ( data . empty ( ) ,  " Expected parallel composition to be on topmost level to be JANI compliant. " ) ;  
		
	
		
			
				
 
		
	
		
			
				                // Prepare storage for the subautomata of the composition.
  
		
	
		
			
				                std : : vector < AutomatonDd >  subautomata ;                 std : : vector < AutomatonDd >  subautomata ;  
		
	
		
			
				                for  ( uint64_t  subcompositionIndex  =  0 ;  subcompositionIndex  <  composition . getNumberOfSubcompositions ( ) ;  + + subcompositionIndex )  {  
		
	
		
			
				                    // Prepare the new offset mapping.
  
		
	
		
			
				                    std : : map < uint64_t ,  uint64_t >  newSynchronizingActionToOffsetMap  =  actionIndexToLocalNondeterminismVariableOffset ;  
		
	
		
			
				
 
		
	
		
			
				                    if  ( subcompositionIndex  = =  0 )  {  
		
	
		
			
				                        for  ( auto  const &  synchVector  :  composition . getSynchronizationVectors ( ) )  {  
		
	
		
			
				                            auto  it  =  actionIndexToLocalNondeterminismVariableOffset . find ( actionInformation . getActionIndex ( synchVector . getOutput ( ) ) ) ;  
		
	
		
			
				                            STORM_LOG_THROW ( it  ! =  actionIndexToLocalNondeterminismVariableOffset . end ( ) ,  storm : : exceptions : : InvalidArgumentException ,  " Invalid action  "  < <  synchVector . getOutput ( )  < <  " . " ) ;  
		
	
		
			
				                            if  ( synchVector . getInput ( 0 )  ! =  storm : : jani : : SynchronizationVector : : NO_ACTION_INPUT )  {  
		
	
		
			
				                                newSynchronizingActionToOffsetMap [ actionInformation . getActionIndex ( synchVector . getInput ( 0 ) ) ]  =  it - > second ;  
		
	
		
			
				                            }  
		
	
		
			
				                        }  
		
	
		
			
				                    }  else  {  
		
	
		
			
				                        // Based on the previous results, we need to update the offsets.
  
		
	
		
			
				                        for  ( auto  const &  synchVector  :  composition . getSynchronizationVectors ( ) )  {  
		
	
		
			
				                            if  ( synchVector . getInput ( subcompositionIndex )  ! =  storm : : jani : : SynchronizationVector : : NO_ACTION_INPUT )  {  
		
	
		
			
				                // The outer loop iterates over the indices of the subcomposition, because the first subcomposition needs
  
		
	
		
			
				                // to be built before the second and so on.
  
		
	
		
			
				                uint64_t  silentActionIndex  =  actionInformation . getActionIndex ( storm : : jani : : Model : : SILENT_ACTION_NAME ) ;  
		
	
		
			
				                for  ( uint64_t  subcompositionIndex  =  0 ;  subcompositionIndex  <  composition . getNumberOfSubcompositions ( ) ;  + + subcompositionIndex )  {  
		
	
		
			
				                    // Now build a new set of action instantiations for the current subcomposition index.
  
		
	
		
			
				                    ActionInstantiations  actionInstantiations ;  
		
	
		
			
				                    actionInstantiations [ silentActionIndex ] . emplace_back ( silentActionIndex ,  0 ) ;  
		
	
		
			
				                     
		
	
		
			
				                    for  ( uint64_t  synchronizationVectorIndex  =  0 ;  synchronizationVectorIndex  <  composition . getNumberOfSynchronizationVectors ( ) ;  + + synchronizationVectorIndex )  {  
		
	
		
			
				                        auto  const &  synchVector  =  composition . getSynchronizationVector ( synchronizationVectorIndex ) ;  
		
	
		
			
				                         
		
	
		
			
				                        // Determine the first participating subcomposition, because we need to build the corresponding action
  
		
	
		
			
				                        // from all local nondeterminism variable offsets that the output action of the synchronization vector
  
		
	
		
			
				                        // is required to have.
  
		
	
		
			
				                        if  ( subcompositionIndex  = =  synchVector . getPositionOfFirstParticipatingAction ( ) )  {  
		
	
		
			
				                            uint64_t  actionIndex  =  actionInformation . getActionIndex ( synchVector . getInput ( subcompositionIndex ) ) ;  
		
	
		
			
				                            actionInstantiations [ actionIndex ] . emplace_back ( actionIndex ,  synchronizationVectorIndex ,  0 ) ;  
		
	
		
			
				                        }  else  if  ( synchVector . getInput ( subcompositionIndex )  ! =  storm : : jani : : SynchronizationVector : : NO_ACTION_INPUT )  {  
		
	
		
			
				                            uint64_t  actionIndex  =  actionInformation . getActionIndex ( synchVector . getInput ( subcompositionIndex ) ) ;  
		
	
		
			
				
 
		
	
		
			
				                            // If this subcomposition is participating in the synchronization vector, but it's not the first
  
		
	
		
			
				                            // such subcomposition, then we have to retrieve the offset we need for the participating action
  
		
	
		
			
				                            // by looking at the maximal offset used by the preceding participating action.
  
		
	
		
			
				                            boost : : optional < uint64_t >  previousActionPosition  =  synchVector . getPositionOfPrecedingParticipatingAction ( subcompositionIndex ) ;                             boost : : optional < uint64_t >  previousActionPosition  =  synchVector . getPositionOfPrecedingParticipatingAction ( subcompositionIndex ) ;  
		
	
		
			
				                                if  ( previousActionPosition )  {  
		
	
		
			
				                            STORM_LOG_ASSERT ( previousActionPosition ,  " Inconsistent information about synchronization vector. " ) ;  
		
	
		
			
				                            AutomatonDd  const &  previousAutomatonDd  =  subautomata [ previousActionPosition . get ( ) ] ;                             AutomatonDd  const &  previousAutomatonDd  =  subautomata [ previousActionPosition . get ( ) ] ;  
		
	
		
			
				
 
		
	
		
			
				                                    std : : string  const &  previousAction  =  synchVector . getInput ( previousActionPosition . get ( ) ) ;  
		
	
		
			
				                                    auto  it  =  previousAutomatonDd . actionIndexToAction . find ( actionInformation . getActionIndex ( previousAction ) ) ;  
		
	
		
			
				                                    if  ( it  ! =  previousAutomatonDd . actionIndexToAction . end ( ) )  {  
		
	
		
			
				                                        newSynchronizingActionToOffsetMap [ actionInformation . getActionIndex ( synchVector . getInput ( subcompositionIndex ) ) ]  =  it - > second . getHighestLocalNondeterminismVariable ( ) ;  
		
	
		
			
				                                    }  else  {  
		
	
		
			
				                                        STORM_LOG_ASSERT ( false ,  " Subcomposition does not have action that is mentioned in parallel composition. " ) ;  
		
	
		
			
				                                    }  
		
	
		
			
				                                }  
		
	
		
			
				                            }  
		
	
		
			
				                            auto  precedingActionIt  =  previousAutomatonDd . actions . find ( ActionIdentification ( actionInformation . getActionIndex ( synchVector . getInput ( previousActionPosition . get ( ) ) ) ,  synchronizationVectorIndex ) ) ;  
		
	
		
			
				                            STORM_LOG_THROW ( precedingActionIt  ! =  previousAutomatonDd . actions . end ( ) ,  storm : : exceptions : : WrongFormatException ,  " Subcomposition does not have action that is mentioned in parallel composition. " ) ;  
		
	
		
			
				                            actionInstantiations [ actionIndex ] . emplace_back ( actionIndex ,  synchronizationVectorIndex ,  precedingActionIt - > second . getHighestLocalNondeterminismVariable ( ) ) ;  
		
	
		
			
				                        }                         }  
		
	
		
			
				                    }                     }  
		
	
		
			
				                                         
		
	
		
			
				                    // Build the DD for the next element of the composition wrt. to the current offset mapping.
  
		
	
		
			
				                    subautomata . push_back ( boost : : any_cast < AutomatonDd > ( composition . getSubcomposition ( subcompositionIndex ) . accept ( * this ,  newSynchronizingActionToOffsetMap ) ) ) ;  
		
	
		
			
				                    subautomata . push_back ( boost : : any_cast < AutomatonDd > ( composition . getSubcomposition ( subcompositionIndex ) . accept ( * this ,  actionInstantiations ) ) ) ;  
		
	
		
			
				                }                 }  
		
	
		
			
				
 
		
	
		
			
				                return  composeInParallel ( subautomata ,  composition . getSynchronizationVectors ( ) ) ;                 return  composeInParallel ( subautomata ,  composition . getSynchronizationVectors ( ) ) ;  
		
	
	
		
			
				
					
					
					
						
							 
					
				 
				@ -720,142 +811,78 @@ namespace storm { 
		
	
		
			
				                         
		
	
		
			
				        private :         private :  
		
	
		
			
				            AutomatonDd  composeInParallel ( std : : vector < AutomatonDd >  const &  subautomata ,  std : : vector < storm : : jani : : SynchronizationVector >  const &  synchronizationVectors )  {             AutomatonDd  composeInParallel ( std : : vector < AutomatonDd >  const &  subautomata ,  std : : vector < storm : : jani : : SynchronizationVector >  const &  synchronizationVectors )  {  
		
	
		
			
				                typedef  storm : : dd : : Add < Type ,  ValueType >  IdentityAdd ;  
		
	
		
			
				                typedef  std : : pair < ActionDd ,  IdentityAdd >  ActionAndAutomatonIdentity ;  
		
	
		
			
				                typedef  std : : vector < ActionAndAutomatonIdentity >  ActionAndAutomatonIdentities ;  
		
	
		
			
				                typedef  std : : vector < boost : : optional < std : : pair < ActionAndAutomatonIdentities ,  IdentityAdd > > >  SynchronizationVectorActionsAndIdentities ;  
		
	
		
			
				                 
		
	
		
			
				                AutomatonDd  result ( this - > variables . manager - > template  getAddOne < ValueType > ( ) ) ;                 AutomatonDd  result ( this - > variables . manager - > template  getAddOne < ValueType > ( ) ) ;  
		
	
		
			
				
 
		
	
		
			
				                std : : map < uint64_t ,  std : : vector < ActionDd > >  nonSynchronizingActions ;  
		
	
		
			
				                SynchronizationVectorActionsAndIdentities  synchronizationVectorActions ( synchronizationVectors . size ( ) ,  boost : : none ) ;  
		
	
		
			
				                for  ( uint64_t  automaton Index=  0 ;  automatonIndex  <  subautomata . size ( ) ;  + + automaton Index)  {  
		
	
		
			
				                    AutomatonDd  const &  subautomaton  =  subautomata [ automaton Index] ;  
		
	
		
			
				                // Build the results of the synchronization vectors.
  
		
	
		
			
				                std : : map < uint64_t ,  std : : vector < ActionDd > >  actions ;  
		
	
		
			
				                for  ( uint64_t  synchronizationVectorIndex  =  0 ;  synchronizationVectorIndex  <  synchronizationVectors . size ( ) ;  + + synchronizationVectorIndex )  {  
		
	
		
			
				                    auto  const &  synchVector  =  synchronizationVectors [ synchronizationVector Index] ;  
		
	
		
			
				                                         
		
	
		
			
				                    // Add the transient assignments from the new subautomaton.
  
		
	
		
			
				                    addToTransientAssignmentMap ( result . transientLocationAssignments ,  subautomaton . transientLocationAssignments ) ;  
		
	
		
			
				                     
		
	
		
			
				                    // Initilize the used local nondeterminism variables appropriately.
  
		
	
		
			
				                    if  ( automatonIndex  = =  0 )  {  
		
	
		
			
				                        result . setLowestLocalNondeterminismVariable ( subautomaton . getLowestLocalNondeterminismVariable ( ) ) ;  
		
	
		
			
				                        result . setHighestLocalNondeterminismVariable ( subautomaton . getHighestLocalNondeterminismVariable ( ) ) ;  
		
	
		
			
				                    }  
		
	
		
			
				
 
		
	
		
			
				                    // Compose the actions according to the synchronization vectors.
  
		
	
		
			
				                    std : : set < uint64_t >  actionsInSynch ;  
		
	
		
			
				                    for  ( uint64_t  synchVectorIndex  =  0 ;  synchVectorIndex  <  synchronizationVectors . size ( ) ;  + + synchVectorIndex )  {  
		
	
		
			
				                        auto  const &  synchVector  =  synchronizationVectors [ synchVectorIndex ] ;  
		
	
		
			
				                         
		
	
		
			
				                        if  ( synchVector . isNoActionInput ( synchVector . getInput ( automatonIndex ) ) )  {  
		
	
		
			
				                            if  ( automatonIndex  = =  0 )  {  
		
	
		
			
				                                // Create a new action that is the identity over the first automaton.
  
		
	
		
			
				                                synchronizationVectorActions [ synchVectorIndex ]  =  std : : make_pair ( ActionAndAutomatonIdentities { std : : make_pair ( ActionDd ( this - > variables . manager - > template  getAddOne < ValueType > ( ) ,  subautomaton . identity ,  { } ,  subautomaton . localNondeterminismVariables ,  { } ,  this - > variables . manager - > getBddZero ( ) ) ,  subautomaton . identity ) } ,  this - > variables . manager - > template  getAddOne < ValueType > ( ) ) ;  
		
	
		
			
				                            }  else  {  
		
	
		
			
				                                // If there is no action in the output spot, this means that some other subcomposition did
  
		
	
		
			
				                                // not provide the action necessary for the synchronization vector to resolve.
  
		
	
		
			
				                                if  ( synchronizationVectorActions [ synchVectorIndex ] )  {  
		
	
		
			
				                                    synchronizationVectorActions [ synchVectorIndex ] . get ( ) . second  * =  subautomaton . identity ;  
		
	
		
			
				                    boost : : optional < ActionDd >  synchronizingAction  =  combineSynchronizingActions ( subautomata ,  synchVector ,  synchronizationVectorIndex ) ;  
		
	
		
			
				                    if  ( synchronizingAction )  {  
		
	
		
			
				                        actions [ actionInformation . getActionIndex ( synchVector . getOutput ( ) ) ] . emplace_back ( synchronizingAction . get ( ) ) ;  
		
	
		
			
				                    }                     }  
		
	
		
			
				                }                 }  
		
	
		
			
				                        }  else  {  
		
	
		
			
				                            // Determine the indices of input (at the current automaton position) and the output.
  
		
	
		
			
				                            uint64_t  inputActionIndex  =  actionInformation . getActionIndex ( synchVector . getInput ( automatonIndex ) ) ;  
		
	
		
			
				                            actionsInSynch . insert ( inputActionIndex ) ;  
		
	
		
			
				                                 
		
	
		
			
				                            // Either set the action (if it's the first of the ones to compose) or compose the actions directly.
  
		
	
		
			
				                            if  ( automatonIndex  = =  0 )  {  
		
	
		
			
				                                // If the action cannot be found, the particular spot in the output will be left empty.
  
		
	
		
			
				                                auto  inputActionIt  =  subautomaton . actionIndexToAction . find ( inputActionIndex ) ;  
		
	
		
			
				                                if  ( inputActionIt  ! =  subautomaton . actionIndexToAction . end ( ) )  {  
		
	
		
			
				                                    synchronizationVectorActions [ synchVectorIndex ]  =  std : : make_pair ( ActionAndAutomatonIdentities { std : : make_pair ( inputActionIt - > second ,  subautomaton . identity ) } ,  this - > variables . manager - > template  getAddOne < ValueType > ( ) ) ;  
		
	
		
			
				                                }  
		
	
		
			
				                            }  else  {  
		
	
		
			
				                                // If there is no action in the output spot, this means that some other subcomposition did
  
		
	
		
			
				                                // not provide the action necessary for the synchronization vector to resolve.
  
		
	
		
			
				                                if  ( synchronizationVectorActions [ synchVectorIndex ] )  {  
		
	
		
			
				                                    auto  inputActionIt  =  subautomaton . actionIndexToAction . find ( inputActionIndex ) ;  
		
	
		
			
				                                    if  ( inputActionIt  ! =  subautomaton . actionIndexToAction . end ( ) )  {  
		
	
		
			
				                                        synchronizationVectorActions [ synchVectorIndex ] . get ( ) . first . push_back ( std : : make_pair ( inputActionIt - > second ,  subautomaton . identity ) ) ;  
		
	
		
			
				                                    }  else  {  
		
	
		
			
				                                        // If the current subcomposition does not provide the required action for the synchronization
  
		
	
		
			
				                                        // vector, we clear the action.
  
		
	
		
			
				                                        synchronizationVectorActions [ synchVectorIndex ]  =  boost : : none ;  
		
	
		
			
				                                    }  
		
	
		
			
				                                }  
		
	
		
			
				                            }  
		
	
		
			
				                        }  
		
	
		
			
				                // Construct the silent action DDs.
  
		
	
		
			
				                std : : vector < ActionDd >  silentActionDds ;  
		
	
		
			
				                for  ( auto  const &  automaton  :  subautomata )  {  
		
	
		
			
				                    for  ( auto &  actionDd  :  silentActionDds )  {  
		
	
		
			
				                        STORM_LOG_TRACE ( " Extending previous silent action by identity of current automaton. " ) ;  
		
	
		
			
				                        actionDd  =  actionDd . multiplyTransitions ( automaton . identity ) ;  
		
	
		
			
				                    }                     }  
		
	
		
			
				                                         
		
	
		
			
				                    // Now treat all unsynchronizing actions.
  
		
	
		
			
				                    if  ( automatonIndex  = =  0 )  {  
		
	
		
			
				                        // Since it's the first automaton, there is nothing to combine.
  
		
	
		
			
				                        for  ( auto  const &  action  :  subautomaton . actionIndexToAction )  {  
		
	
		
			
				                            if  ( actionsInSynch . find ( action . first )  = =  actionsInSynch . end ( ) )  {  
		
	
		
			
				                                nonSynchronizingActions [ action . first ] . push_back ( action . second ) ;  
		
	
		
			
				                            }  
		
	
		
			
				                        }  
		
	
		
			
				                    }  else  {  
		
	
		
			
				                        // Extend all other non-synchronizing actions with the identity of the current subautomaton.
  
		
	
		
			
				                        for  ( auto &  actions  :  nonSynchronizingActions )  {  
		
	
		
			
				                            for  ( auto &  action  :  actions . second )  {  
		
	
		
			
				                                STORM_LOG_TRACE ( " Extending action ' "  < <  actionInformation . getActionName ( actions . first )  < <  " ' with identity of next composition. " ) ;  
		
	
		
			
				                                action . transitions  * =  subautomaton . identity ;  
		
	
		
			
				                            }  
		
	
		
			
				                    ActionIdentification  silentActionIdentification ( storm : : jani : : Model : : SILENT_ACTION_INDEX ) ;  
		
	
		
			
				                    auto  silentActionIt  =  automaton . actions . find ( silentActionIdentification ) ;  
		
	
		
			
				                    if  ( silentActionIt  ! =  automaton . actions . end ( ) )  {  
		
	
		
			
				                        STORM_LOG_TRACE ( " Extending silent action by running identity. " ) ;  
		
	
		
			
				                        silentActionDds . emplace_back ( silentActionIt - > second . multiplyTransitions ( result . identity ) ) ;  
		
	
		
			
				                    }                     }  
		
	
		
			
				
 
		
	
		
			
				                        // Extend the actions of the current subautomaton with the identity of the previous system and
  
		
	
		
			
				                        // add it to the overall non-synchronizing action result.
  
		
	
		
			
				                        for  ( auto  const &  action  :  subautomaton . actionIndexToAction )  {  
		
	
		
			
				                            if  ( actionsInSynch . find ( action . first )  = =  actionsInSynch . end ( ) )  {  
		
	
		
			
				                                STORM_LOG_TRACE ( " Adding action  "  < <  actionInformation . getActionName ( action . first )  < <  "  to non-synchronizing actions and multiply it with system identity. " ) ;  
		
	
		
			
				                                nonSynchronizingActions [ action . first ] . push_back ( action . second . multiplyTransitions ( result . identity ) ) ;  
		
	
		
			
				                    result . identity  * =  automaton . identity ;  
		
	
		
			
				                }                 }  
		
	
		
			
				                 
		
	
		
			
				                if  ( ! silentActionDds . empty ( ) )  {  
		
	
		
			
				                    auto &  allSilentActionDds  =  actions [ storm : : jani : : Model : : SILENT_ACTION_INDEX ] ;  
		
	
		
			
				                    allSilentActionDds . insert ( actions [ storm : : jani : : Model : : SILENT_ACTION_INDEX ] . end ( ) ,  silentActionDds . begin ( ) ,  silentActionDds . end ( ) ) ;  
		
	
		
			
				                }                 }  
		
	
		
			
				                 
		
	
		
			
				                // Finally, combine (potential) multiple action DDs.
  
		
	
		
			
				                for  ( auto  const &  actionDds  :  actions )  {  
		
	
		
			
				                    ActionDd  combinedAction  =  actionDds . second . size ( )  >  1  ?  combineUnsynchronizedActions ( actionDds . second )  :  actionDds . second . front ( ) ;  
		
	
		
			
				                    result . actions [ ActionIdentification ( actionDds . first ) ]  =  combinedAction ;  
		
	
		
			
				                    result . extendLocalNondeterminismVariables ( combinedAction . getLocalNondeterminismVariables ( ) ) ;  
		
	
		
			
				                }                 }  
		
	
		
			
				                         
		
	
		
			
				                    // Finally, construct combined identity.
  
		
	
		
			
				                // Construct combined identity.
  
		
	
		
			
				                for  ( auto  const &  subautomaton  :  subautomata )  {  
		
	
		
			
				                    result . identity  * =  subautomaton . identity ;                     result . identity  * =  subautomaton . identity ;  
		
	
		
			
				                }                 }  
		
	
		
			
				
 
		
	
		
			
				                // Add the results of the synchronization vectors to that of the non-synchronizing actions.
  
		
	
		
			
				                for  ( uint64_t  synchVectorIndex  =  0 ;  synchVectorIndex  <  synchronizationVectors . size ( ) ;  + + synchVectorIndex )  {  
		
	
		
			
				                    auto  const &  synchVector  =  synchronizationVectors [ synchVectorIndex ] ;  
		
	
		
			
				                     
		
	
		
			
				                    // If there is an action resulting from this combination of actions, add it to the output action.
  
		
	
		
			
				                    if  ( synchronizationVectorActions [ synchVectorIndex ] )  {  
		
	
		
			
				                        uint64_t  outputActionIndex  =  actionInformation . getActionIndex ( synchVector . getOutput ( ) ) ;  
		
	
		
			
				                        nonSynchronizingActions [ outputActionIndex ] . push_back ( combineSynchronizingActions ( synchronizationVectorActions [ synchVectorIndex ] . get ( ) . first ,  synchronizationVectorActions [ synchVectorIndex ] . get ( ) . second ) ) ;  
		
	
		
			
				                    }  
		
	
		
			
				                return  result ;  
		
	
		
			
				            }             }  
		
	
		
			
				                         
		
	
		
			
				                // Now that we have built the individual action DDs for all resulting actions, we need to combine them
  
		
	
		
			
				                // in an unsynchronizing way.
  
		
	
		
			
				                for  ( auto  const &  nonSynchronizingActionDds  :  nonSynchronizingActions )  {  
		
	
		
			
				                    std : : vector < ActionDd >  const &  actionDds  =  nonSynchronizingActionDds . second ;  
		
	
		
			
				                    if  ( actionDds . size ( )  >  1 )  {  
		
	
		
			
				                        ActionDd  combinedAction  =  combineUnsynchronizedActions ( actionDds ) ;  
		
	
		
			
				                        result . actionIndexToAction [ nonSynchronizingActionDds . first ]  =  combinedAction ;  
		
	
		
			
				                        result . extendLocalNondeterminismVariables ( combinedAction . getLocalNondeterminismVariables ( ) ) ;  
		
	
		
			
				            boost : : optional < ActionDd >  combineSynchronizingActions ( std : : vector < AutomatonDd >  const &  subautomata ,  storm : : jani : : SynchronizationVector  const &  synchronizationVector ,  uint64_t  synchronizationVectorIndex )  {  
		
	
		
			
				                std : : vector < std : : pair < uint64_t ,  std : : reference_wrapper < ActionDd  const > > >  actions ;  
		
	
		
			
				                storm : : dd : : Add < Type ,  ValueType >  nonSynchronizingIdentity  =  this - > variables . manager - > template  getAddOne < ValueType > ( ) ;  
		
	
		
			
				                for  ( uint64_t  subautomatonIndex  =  0 ;  subautomatonIndex  <  subautomata . size ( ) ;  + + subautomatonIndex )  {  
		
	
		
			
				                    auto  const &  subautomaton  =  subautomata [ subautomatonIndex ] ;  
		
	
		
			
				                    if  ( synchronizationVector . getInput ( subautomatonIndex )  ! =  storm : : jani : : SynchronizationVector : : NO_ACTION_INPUT )  {  
		
	
		
			
				                        auto  it  =  subautomaton . actions . find ( ActionIdentification ( actionInformation . getActionIndex ( synchronizationVector . getInput ( subautomatonIndex ) ) ,  synchronizationVectorIndex ) ) ;  
		
	
		
			
				                        if  ( it  ! =  subautomaton . actions . end ( ) )  {  
		
	
		
			
				                            actions . emplace_back ( subautomatonIndex ,  it - > second ) ;  
		
	
		
			
				                        }  else  {                         }  else  {  
		
	
		
			
				                        result . actionIndexToAction [ nonSynchronizingActionDds . first ]  =  actionDds . front ( ) ;  
		
	
		
			
				                        result . extendLocalNondeterminismVariables ( actionDds . front ( ) . getLocalNondeterminismVariables ( ) ) ;  
		
	
		
			
				                    }  
		
	
		
			
				                            return  boost : : none ;  
		
	
		
			
				                        }                         }  
		
	
		
			
				                 
		
	
		
			
				                return  result ;  
		
	
		
			
				                    }  else  {  
		
	
		
			
				                        nonSynchronizingIdentity  * =  subautomaton . identity  ;  
		
	
		
			
				                    }                     }  
		
	
		
			
				             
		
	
		
			
				            ActionDd  combineSynchronizingActions ( std : : vector < std : : pair < ActionDd ,  storm : : dd : : Add < Type ,  ValueType > > >  const &  actionsAndIdentities ,  storm : : dd : : Add < Type ,  ValueType >  const &  nonSynchronizingAutomataIdentities )  {  
		
	
		
			
				                // If there is just one action, no need to combine anything.
  
		
	
		
			
				                if  ( actionsAndIdentities . size ( )  = =  1 )  {  
		
	
		
			
				                    return  actionsAndIdentities . front ( ) . first ;  
		
	
		
			
				                }                 }  
		
	
		
			
				                                 
		
	
		
			
				                // If there are only input-enabled actions, we also need to build the disjunction of the guards.
                 // If there are only input-enabled actions, we also need to build the disjunction of the guards.
  
		
	
		
			
				                bool  allActionsInputEnabled  =  true ;                 bool  allActionsInputEnabled  =  true ;  
		
	
		
			
				                for  ( auto  const &  actionIdentityPair  :  actionsAndIdentities )  {  
		
	
		
			
				                    auto  const &  action  =  actionIdentityPair . first ;  
		
	
		
			
				                    if  ( ! action . isInputEnabled ( ) )  {  
		
	
		
			
				                for  ( auto  const &  action  :  actions )  {  
		
	
		
			
				                    if  ( ! action . second . get ( ) . isInputEnabled ( ) )  {  
		
	
		
			
				                        allActionsInputEnabled  =  false ;                         allActionsInputEnabled  =  false ;  
		
	
		
			
				                    }                     }  
		
	
		
			
				                }                 }  
		
	
	
		
			
				
					
					
					
						
							 
					
				 
				@ -875,12 +902,13 @@ namespace storm { 
		
	
		
			
				                storm : : dd : : Add < Type ,  ValueType >  transitions  =  this - > variables . manager - > template  getAddOne < ValueType > ( ) ;                 storm : : dd : : Add < Type ,  ValueType >  transitions  =  this - > variables . manager - > template  getAddOne < ValueType > ( ) ;  
		
	
		
			
				                std : : map < storm : : expressions : : Variable ,  storm : : dd : : Add < Type ,  ValueType > >  transientEdgeAssignments ;                 std : : map < storm : : expressions : : Variable ,  storm : : dd : : Add < Type ,  ValueType > >  transientEdgeAssignments ;  
		
	
		
			
				                                 
		
	
		
			
				                uint64_t  lowestNondeterminismVariable  =  actionsAndIdentities  . front ( ) . first . getLowestLocalNondeterminismVariable ( ) ;  
		
	
		
			
				                uint64_t  highestNondeterminismVariable  =  actionsAndIdentities  . front ( ) . first . getHighestLocalNondeterminismVariable ( ) ;  
		
	
		
			
				                uint64_t  lowestNondeterminismVariable  =  actions . front ( ) . second . get ( ) . getLowestLocalNondeterminismVariable ( ) ;  
		
	
		
			
				                uint64_t  highestNondeterminismVariable  =  actions . front ( ) . second . get ( ) . getHighestLocalNondeterminismVariable ( ) ;  
		
	
		
			
				                                 
		
	
		
			
				                storm : : dd : : Bdd < Type >  newIllegalFragment  =  this - > variables . manager - > getBddZero ( ) ;                 storm : : dd : : Bdd < Type >  newIllegalFragment  =  this - > variables . manager - > getBddZero ( ) ;  
		
	
		
			
				                for  ( auto  const &  actionIdentityPair  :  actionsAndIdentities )  {  
		
	
		
			
				                    auto  const &  action  =  actionIdentityPair . first ;  
		
	
		
			
				                for  ( auto  const &  actionIndexPair  :  actions )  {  
		
	
		
			
				                    auto  componentIndex  =  actionIndexPair . first ;  
		
	
		
			
				                    auto  const &  action  =  actionIndexPair . second . get ( ) ;  
		
	
		
			
				                    storm : : dd : : Bdd < Type >  actionGuard  =  action . guard . toBdd ( ) ;                     storm : : dd : : Bdd < Type >  actionGuard  =  action . guard . toBdd ( ) ;  
		
	
		
			
				                    if  ( guardDisjunction )  {                     if  ( guardDisjunction )  {  
		
	
		
			
				                        guardDisjunction . get ( )  | =  actionGuard ;                         guardDisjunction . get ( )  | =  actionGuard ;  
		
	
	
		
			
				
					
					
					
						
							 
					
				 
				@ -892,7 +920,7 @@ namespace storm { 
		
	
		
			
				                                         
		
	
		
			
				                    if  ( action . isInputEnabled ( ) )  {                     if  ( action . isInputEnabled ( ) )  {  
		
	
		
			
				                        // If the action is input-enabled, we add self-loops to all states.
                         // If the action is input-enabled, we add self-loops to all states.
  
		
	
		
			
				                        transitions  * =  actionGuard . ite ( action . transitions ,  encodeIndex ( 0 ,  action . getLowestLocalNondeterminismVariable ( ) ,  action . getHighestLocalNondeterminismVariable ( )  -  action . getLowestLocalNondeterminismVariable ( ) ,  this - > variables )  *  actionIdentityPair . second ) ;  
		
	
		
			
				                        transitions  * =  actionGuard . ite ( action . transitions ,  encodeIndex ( 0 ,  action . getLowestLocalNondeterminismVariable ( ) ,  action . getHighestLocalNondeterminismVariable ( )  -  action . getLowestLocalNondeterminismVariable ( ) ,  this - > variables )  *  subautomata [ componentIndex ] . identity ) ;  
		
	
		
			
				                    }  else  {                     }  else  {  
		
	
		
			
				                        transitions  * =  action . transitions ;                         transitions  * =  action . transitions ;  
		
	
		
			
				                    }                     }  
		
	
	
		
			
				
					
						
							 
					
					
						
							 
					
					
				 
				@ -948,7 +976,7 @@ namespace storm { 
		
	
		
			
				                // such a combined transition.
                 // such a combined transition.
  
		
	
		
			
				                illegalFragment  & =  inputEnabledGuard ;                 illegalFragment  & =  inputEnabledGuard ;  
		
	
		
			
				                                 
		
	
		
			
				                return  ActionDd ( inputEnabledGuard . template  toAdd < ValueType > ( ) ,  transitions  *  nonSynchronizingAutomataIdentities  ,  transientEdgeAssignments ,  std : : make_pair ( lowestNondeterminismVariable ,  highestNondeterminismVariable ) ,  globalVariableToWritingFragment ,  illegalFragment ) ;  
		
	
		
			
				                return  ActionDd ( inputEnabledGuard . template  toAdd < ValueType > ( ) ,  transitions  *  nonSynchronizingIdentity  ,  transientEdgeAssignments ,  std : : make_pair ( lowestNondeterminismVariable ,  highestNondeterminismVariable ) ,  globalVariableToWritingFragment ,  illegalFragment ) ;  
		
	
		
			
				            }             }  
		
	
		
			
				                         
		
	
		
			
				            ActionDd  combineUnsynchronizedActions ( ActionDd  action1 ,  ActionDd  action2 ,  storm : : dd : : Add < Type ,  ValueType >  const &  identity1 ,  storm : : dd : : Add < Type ,  ValueType >  const &  identity2 )  {             ActionDd  combineUnsynchronizedActions ( ActionDd  action1 ,  ActionDd  action2 ,  storm : : dd : : Add < Type ,  ValueType >  const &  identity1 ,  storm : : dd : : Add < Type ,  ValueType >  const &  identity2 )  {  
		
	
	
		
			
				
					
						
							 
					
					
						
							 
					
					
				 
				@ -1141,7 +1169,7 @@ namespace storm { 
		
	
		
			
				
 
		
	
		
			
				                bool  overlappingGuards  =  false ;                 bool  overlappingGuards  =  false ;  
		
	
		
			
				                for  ( auto  const &  edge  :  edgeDds )  {                 for  ( auto  const &  edge  :  edgeDds )  {  
		
	
		
			
				                    STORM_LOG_THROW ( edge . isMarkovian ,  storm : : exceptions : : InvalidArgumen tException,  " Can only combine Markovian edges. " ) ;  
		
	
		
			
				                    STORM_LOG_THROW ( edge . isMarkovian ,  storm : : exceptions : : WrongForma tException,  " Can only combine Markovian edges. " ) ;  
		
	
		
			
				                                         
		
	
		
			
				                    if  ( ! overlappingGuards )  {                     if  ( ! overlappingGuards )  {  
		
	
		
			
				                        overlappingGuards  | =  ! ( guard  & &  edge . guard . toBdd ( ) ) . isZero ( ) ;                         overlappingGuards  | =  ! ( guard  & &  edge . guard . toBdd ( ) ) . isZero ( ) ;  
		
	
	
		
			
				
					
						
							 
					
					
						
							 
					
					
				 
				@ -1197,7 +1225,7 @@ namespace storm { 
		
	
		
			
				                        }                         }  
		
	
		
			
				                        return  combineEdgesToActionNondeterministic ( nonMarkovianEdges ,  markovianEdge ,  localNondeterminismVariableOffset ) ;                         return  combineEdgesToActionNondeterministic ( nonMarkovianEdges ,  markovianEdge ,  localNondeterminismVariableOffset ) ;  
		
	
		
			
				                    }  else  {                     }  else  {  
		
	
		
			
				                        STORM_LOG_THROW ( false ,  storm : : exceptions : : InvalidArgumen tException,  " Cannot translate model of this type. " ) ;  
		
	
		
			
				                        STORM_LOG_THROW ( false ,  storm : : exceptions : : WrongForma tException,  " Cannot translate model of this type. " ) ;  
		
	
		
			
				                    }                     }  
		
	
		
			
				                }  else  {                 }  else  {  
		
	
		
			
				                    return  ActionDd ( this - > variables . manager - > template  getAddZero < ValueType > ( ) ,  this - > variables . manager - > template  getAddZero < ValueType > ( ) ,  { } ,  std : : make_pair < uint64_t ,  uint64_t > ( 0 ,  0 ) ,  { } ,  this - > variables . manager - > getBddZero ( ) ) ;                     return  ActionDd ( this - > variables . manager - > template  getAddZero < ValueType > ( ) ,  this - > variables . manager - > template  getAddZero < ValueType > ( ) ,  { } ,  std : : make_pair < uint64_t ,  uint64_t > ( 0 ,  0 ) ,  { } ,  this - > variables . manager - > getBddZero ( ) ) ;  
		
	
	
		
			
				
					
						
							 
					
					
						
							 
					
					
				 
				@ -1479,23 +1507,30 @@ namespace storm { 
		
	
		
			
				                }                 }  
		
	
		
			
				            }             }  
		
	
		
			
				                         
		
	
		
			
				            AutomatonDd  buildAutomatonDd ( std : : string  const &  automatonName ,  std : : map < uint_fast64_t ,  uint_fast64_t >  const &  actionIndexToLocalNondeterminismVariableOffset ,  std : : set < uint64_t >  const &  inputEnabledActionIndices )  {  
		
	
		
			
				            AutomatonDd  buildAutomatonDd ( std : : string  const &  automatonName ,  ActionInstantiations  const &  actionInstantiations ,  std : : set < uint64_t >  const &  inputEnabledActionIndices )  {  
		
	
		
			
				                STORM_LOG_TRACE ( " Building DD for automaton ' "  < <  automatonName  < <  " '. " ) ;  
		
	
		
			
				                AutomatonDd  result ( this - > variables . automatonToIdentityMap . at ( automatonName ) ) ;                 AutomatonDd  result ( this - > variables . automatonToIdentityMap . at ( automatonName ) ) ;  
		
	
		
			
				                                 
		
	
		
			
				                storm : : jani : : Automaton  const &  automaton  =  this - > model . getAutomaton ( automatonName ) ;                 storm : : jani : : Automaton  const &  automaton  =  this - > model . getAutomaton ( automatonName ) ;  
		
	
		
			
				                for  ( auto  const &  action  :  this - > model . getActions ( )  )  {  
		
	
		
			
				                    uint64_t  actionIndex  =  this - > model . getActionIndex ( action . getName ( ) ) ;  
		
	
		
			
				                for  ( auto  const &  actionInstantiation  :  actionInstantiations  )  {  
		
	
		
			
				                    uint64_t  actionIndex  =  actionInstantiation . first ;  
		
	
		
			
				                    if  ( ! automaton . hasEdgeLabeledWithActionIndex ( actionIndex ) )  {                     if  ( ! automaton . hasEdgeLabeledWithActionIndex ( actionIndex ) )  {  
		
	
		
			
				                        continue ;                         continue ;  
		
	
		
			
				                    }                     }  
		
	
		
			
				                    ActionDd  actionDd  =  buildActionDdForActionIndex ( automaton ,  actionIndex ,  actionIndexToLocalNondeterminismVariableOffset . at ( actionIndex ) ) ;  
		
	
		
			
				                    bool  inputEnabled  =  false ;  
		
	
		
			
				                    if  ( inputEnabledActionIndices . find ( actionIndex )  ! =  inputEnabledActionIndices . end ( ) )  {                     if  ( inputEnabledActionIndices . find ( actionIndex )  ! =  inputEnabledActionIndices . end ( ) )  {  
		
	
		
			
				                        inputEnabled  =  true ;  
		
	
		
			
				                    }  
		
	
		
			
				                    for  ( auto  const &  instantiationOffset  :  actionInstantiation . second )  {  
		
	
		
			
				                        STORM_LOG_TRACE ( " Building  "  < <  ( actionInformation . getActionName ( actionIndex ) . empty ( )  ?  " silent  "  :  " " )  < <  " action  "  < <  ( actionInformation . getActionName ( actionIndex ) . empty ( )  ?  " "  :  actionInformation . getActionName ( actionIndex )  +  "   " )  < <  " from offset  "  < <  instantiationOffset . localNondeterminismVariableOffset  < <  " . " ) ;  
		
	
		
			
				                        ActionDd  actionDd  =  buildActionDdForActionIndex ( automaton ,  actionIndex ,  instantiationOffset . localNondeterminismVariableOffset ) ;  
		
	
		
			
				                        if  ( inputEnabled )  {  
		
	
		
			
				                            actionDd . setIsInputEnabled ( ) ;                             actionDd . setIsInputEnabled ( ) ;  
		
	
		
			
				                        }                         }  
		
	
		
			
				                     
		
	
		
			
				                    result . actionIndexToAction [ actionIndex ]  =  actionDd ;  
		
	
		
			
				                    result . setLowestLocalNondeterminismVariable ( std : : max ( result . getLowestLocalNondeterminismVariable ( ) ,  actionDd . getLowestLocalNondeterminismVariable ( ) ) ) ;  
		
	
		
			
				                    result . setHighestLocalNondeterminismVariable ( std : : max ( result . getHighestLocalNondeterminismVariable ( ) ,  actionDd . getHighestLocalNondeterminismVariable ( ) ) ) ;  
		
	
		
			
				                        STORM_LOG_TRACE ( " Used local nondeterminism variables are  "  < <  actionDd . getLowestLocalNondeterminismVariable ( )  < <  "  to  "  < <  actionDd . getHighestLocalNondeterminismVariable ( )  < <  " . " ) ;   
		
	
		
			
				                         result . actions [ ActionIdentification ( actionIndex ,  instantiationOffset . synchronizationVectorIndex )  ]  =  actionDd ;  
		
	
		
			
				                         result . extendLocalNondeterminismVariables ( actionDd . getLocalNondeterminismVariables  ( ) ) ;  
		
	
		
			
				                    }  
		
	
		
			
				                }                 }  
		
	
		
			
				                                 
		
	
		
			
				                for  ( uint64_t  locationIndex  =  0 ;  locationIndex  <  automaton . getNumberOfLocations ( ) ;  + + locationIndex )  {                 for  ( uint64_t  locationIndex  =  0 ;  locationIndex  <  automaton . getNumberOfLocations ( ) ;  + + locationIndex )  {  
		
	
	
		
			
				
					
						
							 
					
					
						
							 
					
					
				 
				@ -1539,13 +1574,18 @@ namespace storm { 
		
	
		
			
				                    // First, determine the highest number of nondeterminism variables that is used in any action and make
                     // First, determine the highest number of nondeterminism variables that is used in any action and make
  
		
	
		
			
				                    // all actions use the same amout of nondeterminism variables.
                     // all actions use the same amout of nondeterminism variables.
  
		
	
		
			
				                    uint64_t  numberOfUsedNondeterminismVariables  =  automaton . getHighestLocalNondeterminismVariable ( ) ;                     uint64_t  numberOfUsedNondeterminismVariables  =  automaton . getHighestLocalNondeterminismVariable ( ) ;  
		
	
		
			
				                    STORM_LOG_TRACE ( " Building system from composed automaton; number of used nondeterminism variables is  "  < <  numberOfUsedNondeterminismVariables  < <  " . " ) ;  
		
	
		
			
				                                         
		
	
		
			
				                    // Add missing global variable identities, action and nondeterminism encodings.
                     // Add missing global variable identities, action and nondeterminism encodings.
  
		
	
		
			
				                    std : : map < storm : : expressions : : Variable ,  storm : : dd : : Add < Type ,  ValueType > >  transientEdgeAssignments ;                     std : : map < storm : : expressions : : Variable ,  storm : : dd : : Add < Type ,  ValueType > >  transientEdgeAssignments ;  
		
	
		
			
				                    for  ( auto &  action  :  automaton . actionIndexToAction )  {  
		
	
		
			
				                    std : : unordered_set < uint64_t >  actionIndices ;  
		
	
		
			
				                    for  ( auto &  action  :  automaton . actions )  {  
		
	
		
			
				                        uint64_t  actionIndex  =  action . first . actionIndex ;  
		
	
		
			
				                        STORM_LOG_THROW ( actionIndices . find ( actionIndex )  = =  actionIndices . end ( ) ,  storm : : exceptions : : WrongFormatException ,  " Duplication action  "  < <  actionInformation . getActionName ( actionIndex ) ) ;  
		
	
		
			
				                        actionIndices . insert ( action . first . actionIndex ) ;  
		
	
		
			
				                        illegalFragment  | =  action . second . illegalFragment ;                         illegalFragment  | =  action . second . illegalFragment ;  
		
	
		
			
				                        addMissingGlobalVariableIdentities ( action . second ) ;                         addMissingGlobalVariableIdentities ( action . second ) ;  
		
	
		
			
				                        storm : : dd : : Add < Type ,  ValueType >  actionEncoding  =  encodeAction ( action . first  ! =  storm : : jani : : Model : : SILENT_ACTION_INDEX  ?  boost : : optional < uint64_t > ( action . first )  :  boost : : none ,  this - > variables ) ;  
		
	
		
			
				                        storm : : dd : : Add < Type ,  ValueType >  actionEncoding  =  encodeAction ( actionIndex   ! =  storm : : jani : : Model : : SILENT_ACTION_INDEX  ?  boost : : optional < uint64_t > ( actionIndex  )  :  boost : : none ,  this - > variables ) ;  
		
	
		
			
				                        storm : : dd : : Add < Type ,  ValueType >  missingNondeterminismEncoding  =  encodeIndex ( 0 ,  action . second . getHighestLocalNondeterminismVariable ( ) ,  numberOfUsedNondeterminismVariables  -  action . second . getHighestLocalNondeterminismVariable ( ) ,  this - > variables ) ;                         storm : : dd : : Add < Type ,  ValueType >  missingNondeterminismEncoding  =  encodeIndex ( 0 ,  action . second . getHighestLocalNondeterminismVariable ( ) ,  numberOfUsedNondeterminismVariables  -  action . second . getHighestLocalNondeterminismVariable ( ) ,  this - > variables ) ;  
		
	
		
			
				                        storm : : dd : : Add < Type ,  ValueType >  extendedTransitions  =  actionEncoding  *  missingNondeterminismEncoding  *  action . second . transitions ;                         storm : : dd : : Add < Type ,  ValueType >  extendedTransitions  =  actionEncoding  *  missingNondeterminismEncoding  *  action . second . transitions ;  
		
	
		
			
				                        for  ( auto  const &  transientAssignment  :  action . second . transientEdgeAssignments )  {                         for  ( auto  const &  transientAssignment  :  action . second . transientEdgeAssignments )  {  
		
	
	
		
			
				
					
					
					
						
							 
					
				 
				@ -1562,7 +1602,10 @@ namespace storm { 
		
	
		
			
				                    storm : : dd : : Add < Type ,  ValueType >  result  =  this - > variables . manager - > template  getAddZero < ValueType > ( ) ;                     storm : : dd : : Add < Type ,  ValueType >  result  =  this - > variables . manager - > template  getAddZero < ValueType > ( ) ;  
		
	
		
			
				                    storm : : dd : : Bdd < Type >  illegalFragment  =  this - > variables . manager - > getBddZero ( ) ;                     storm : : dd : : Bdd < Type >  illegalFragment  =  this - > variables . manager - > getBddZero ( ) ;  
		
	
		
			
				                    std : : map < storm : : expressions : : Variable ,  storm : : dd : : Add < Type ,  ValueType > >  transientEdgeAssignments ;                     std : : map < storm : : expressions : : Variable ,  storm : : dd : : Add < Type ,  ValueType > >  transientEdgeAssignments ;  
		
	
		
			
				                    for  ( auto &  action  :  automaton . actionIndexToAction )  {  
		
	
		
			
				                    std : : unordered_set < uint64_t >  actionIndices ;  
		
	
		
			
				                    for  ( auto &  action  :  automaton . actions )  {  
		
	
		
			
				                        STORM_LOG_THROW ( actionIndices . find ( action . first . actionIndex )  = =  actionIndices . end ( ) ,  storm : : exceptions : : WrongFormatException ,  " Duplication action  "  < <  actionInformation . getActionName ( action . first . actionIndex ) ) ;  
		
	
		
			
				                        actionIndices . insert ( action . first . actionIndex ) ;  
		
	
		
			
				                        illegalFragment  | =  action . second . illegalFragment ;                         illegalFragment  | =  action . second . illegalFragment ;  
		
	
		
			
				                        addMissingGlobalVariableIdentities ( action . second ) ;                         addMissingGlobalVariableIdentities ( action . second ) ;  
		
	
		
			
				                        addToTransientAssignmentMap ( transientEdgeAssignments ,  action . second . transientEdgeAssignments ) ;                         addToTransientAssignmentMap ( transientEdgeAssignments ,  action . second . transientEdgeAssignments ) ;  
		
	
	
		
			
				
					
					
					
						
							 
					
				 
				@ -1571,7 +1614,7 @@ namespace storm { 
		
	
		
			
				
 
		
	
		
			
				                    return  ComposerResult < Type ,  ValueType > ( result ,  automaton . transientLocationAssignments ,  transientEdgeAssignments ,  illegalFragment ,  0 ) ;                     return  ComposerResult < Type ,  ValueType > ( result ,  automaton . transientLocationAssignments ,  transientEdgeAssignments ,  illegalFragment ,  0 ) ;  
		
	
		
			
				                }  else  {                 }  else  {  
		
	
		
			
				                    STORM_LOG_THROW ( false ,  storm : : exceptions : : InvalidArgumen tException,  " Illegal model type. " ) ;  
		
	
		
			
				                    STORM_LOG_THROW ( false ,  storm : : exceptions : : WrongForma tException,  " Illegal model type. " ) ;  
		
	
		
			
				                }                 }  
		
	
		
			
				            }             }  
		
	
		
			
				        } ;         } ;  
		
	
	
		
			
				
					
					
					
						
							 
					
				 
				@ -1596,7 +1639,7 @@ namespace storm { 
		
	
		
			
				            }  else  if  ( modelType  = =  storm : : jani : : ModelType : : MDP )  {             }  else  if  ( modelType  = =  storm : : jani : : ModelType : : MDP )  {  
		
	
		
			
				                return  std : : shared_ptr < storm : : models : : symbolic : : Model < Type ,  ValueType > > ( new  storm : : models : : symbolic : : Mdp < Type ,  ValueType > ( variables . manager ,  modelComponents . reachableStates ,  modelComponents . initialStates ,  modelComponents . deadlockStates ,  modelComponents . transitionMatrix ,  variables . rowMetaVariables ,  variables . rowExpressionAdapter ,  variables . columnMetaVariables ,  variables . columnExpressionAdapter ,  variables . rowColumnMetaVariablePairs ,  variables . allNondeterminismVariables ,  modelComponents . labelToExpressionMap ,  modelComponents . rewardModels ) ) ;                 return  std : : shared_ptr < storm : : models : : symbolic : : Model < Type ,  ValueType > > ( new  storm : : models : : symbolic : : Mdp < Type ,  ValueType > ( variables . manager ,  modelComponents . reachableStates ,  modelComponents . initialStates ,  modelComponents . deadlockStates ,  modelComponents . transitionMatrix ,  variables . rowMetaVariables ,  variables . rowExpressionAdapter ,  variables . columnMetaVariables ,  variables . columnExpressionAdapter ,  variables . rowColumnMetaVariablePairs ,  variables . allNondeterminismVariables ,  modelComponents . labelToExpressionMap ,  modelComponents . rewardModels ) ) ;  
		
	
		
			
				            }  else  {             }  else  {  
		
	
		
			
				                STORM_LOG_THROW ( false ,  storm : : exceptions : : InvalidArgumen tException,  " Invalid model type. " ) ;  
		
	
		
			
				                STORM_LOG_THROW ( false ,  storm : : exceptions : : WrongForma tException,  " Invalid model type. " ) ;  
		
	
		
			
				            }             }  
		
	
		
			
				        }         }  
		
	
		
			
				                 
		
	
	
		
			
				
					
						
							 
					
					
						
							 
					
					
				 
				@ -1718,7 +1761,7 @@ namespace storm { 
		
	
		
			
				                        transitionMatrix  + =  deadlockStatesAdd  *  globalIdentity  *  action ;                         transitionMatrix  + =  deadlockStatesAdd  *  globalIdentity  *  action ;  
		
	
		
			
				                    }                     }  
		
	
		
			
				                }  else  {                 }  else  {  
		
	
		
			
				                    STORM_LOG_THROW ( false ,  storm : : exceptions : : InvalidArgumen tException,  " The model contains  "  < <  deadlockStates . getNonZeroCount ( )  < <  "  deadlock states. Please unset the option to not fix deadlocks, if you want to fix them automatically. " ) ;  
		
	
		
			
				                    STORM_LOG_THROW ( false ,  storm : : exceptions : : WrongForma tException,  " The model contains  "  < <  deadlockStates . getNonZeroCount ( )  < <  "  deadlock states. Please unset the option to not fix deadlocks, if you want to fix them automatically. " ) ;  
		
	
		
			
				                }                 }  
		
	
		
			
				            }             }  
		
	
		
			
				            return  deadlockStates ;             return  deadlockStates ;  
		
	
	
		
			
				
					
						
							 
					
					
						
							 
					
					
				 
				@ -1813,8 +1856,7 @@ namespace storm { 
		
	
		
			
				                STORM_LOG_THROW ( false ,  storm : : exceptions : : InvalidArgumentException ,  " Model still contains these undefined constants:  "  < <  boost : : join ( strings ,  " ,  " )  < <  " . " ) ;                 STORM_LOG_THROW ( false ,  storm : : exceptions : : InvalidArgumentException ,  " Model still contains these undefined constants:  "  < <  boost : : join ( strings ,  " ,  " )  < <  " . " ) ;  
		
	
		
			
				            }             }  
		
	
		
			
				                         
		
	
		
			
				            STORM_LOG_THROW ( ! model . usesAssignmentLevels ( ) ,  storm : : exceptions : : InvalidSettingsException ,  " The symbolic JANI model builder currently does not support assignment levels. " ) ;  
		
	
		
			
				            STORM_LOG_THROW ( ! model . reusesActionsInComposition ( ) ,  storm : : exceptions : : InvalidSettingsException ,  " The symbolic JANI model builder currently does not support reusing actions in parallel composition " ) ;  
		
	
		
			
				            STORM_LOG_THROW ( ! model . usesAssignmentLevels ( ) ,  storm : : exceptions : : WrongFormatException ,  " The symbolic JANI model builder currently does not support assignment levels. " ) ;  
		
	
		
			
				
 
		
	
		
			
				            storm : : jani : : Model  preparedModel  =  model ;             storm : : jani : : Model  preparedModel  =  model ;