You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							346 lines
						
					
					
						
							9.5 KiB
						
					
					
				
			
		
		
		
			
			
			
				
					
				
				
					
				
			
		
		
	
	
							346 lines
						
					
					
						
							9.5 KiB
						
					
					
				| /** | |
|   @file | |
|  | |
|   @ingroup cudd | |
|  | |
|   @brief Boolean equation solver and related functions. | |
|  | |
|   @author Balakrishna Kumthekar | |
|  | |
|   @copyright@parblock | |
|   Copyright (c) 1995-2015, Regents of the University of Colorado | |
|  | |
|   All rights reserved. | |
|  | |
|   Redistribution and use in source and binary forms, with or without | |
|   modification, are permitted provided that the following conditions | |
|   are met: | |
|  | |
|   Redistributions of source code must retain the above copyright | |
|   notice, this list of conditions and the following disclaimer. | |
|  | |
|   Redistributions in binary form must reproduce the above copyright | |
|   notice, this list of conditions and the following disclaimer in the | |
|   documentation and/or other materials provided with the distribution. | |
|  | |
|   Neither the name of the University of Colorado nor the names of its | |
|   contributors may be used to endorse or promote products derived from | |
|   this software without specific prior written permission. | |
|  | |
|   THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | |
|   "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | |
|   LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS | |
|   FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE | |
|   COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, | |
|   INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, | |
|   BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; | |
|   LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | |
|   CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT | |
|   LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN | |
|   ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE | |
|   POSSIBILITY OF SUCH DAMAGE. | |
|   @endparblock | |
|  | |
| */ | |
| 
 | |
| #include "util.h" | |
| #include "cuddInt.h" | |
|  | |
| /*---------------------------------------------------------------------------*/ | |
| /* Constant declarations                                                     */ | |
| /*---------------------------------------------------------------------------*/ | |
| 
 | |
| 
 | |
| /*---------------------------------------------------------------------------*/ | |
| /* Type declarations                                                         */ | |
| /*---------------------------------------------------------------------------*/ | |
| 
 | |
| 
 | |
| /*---------------------------------------------------------------------------*/ | |
| /* Structure declarations                                                    */ | |
| /*---------------------------------------------------------------------------*/ | |
| 
 | |
| 
 | |
| /*---------------------------------------------------------------------------*/ | |
| /* Variable declarations                                                     */ | |
| /*---------------------------------------------------------------------------*/ | |
| 
 | |
| 
 | |
| /*---------------------------------------------------------------------------*/ | |
| /* Macro declarations                                                        */ | |
| /*---------------------------------------------------------------------------*/ | |
| 
 | |
| /** \cond */ | |
| 
 | |
| /*---------------------------------------------------------------------------*/ | |
| /* Static function prototypes                                                */ | |
| /*---------------------------------------------------------------------------*/ | |
| 
 | |
| /** \endcond */ | |
| 
 | |
| 
 | |
| /*---------------------------------------------------------------------------*/ | |
| /* Definition of exported functions                                          */ | |
| /*---------------------------------------------------------------------------*/ | |
| 
 | |
| 
 | |
| /** | |
|   @brief Implements the solution of F(x,y) = 0. | |
|  | |
|   @details The return value is the consistency condition. The y | |
|   variables are the unknowns and the remaining variables are the | |
|   parameters.  Cudd_SolveEqn allocates an array and fills it with the | |
|   indices of the unknowns. This array is used by Cudd_VerifySol. | |
|  | |
|   @return the consistency condition if successful; NULL otherwise. | |
|  | |
|   @sideeffect The solution is returned in G; the indices of the y | |
|   variables are returned in yIndex. | |
|  | |
|   @see Cudd_VerifySol | |
|  | |
| */ | |
| DdNode * | |
| Cudd_SolveEqn( | |
|   DdManager *  bdd /**< CUDD manager */, | |
|   DdNode * F /**< the left-hand side of the equation */, | |
|   DdNode * Y /**< the cube of the y variables */, | |
|   DdNode ** G /**< the array of solutions (return parameter) */, | |
|   int ** yIndex /**< index of y variables */, | |
|   int  n /**< numbers of unknowns */) | |
| { | |
|     DdNode *res; | |
|     int *temp; | |
| 
 | |
|     *yIndex = temp = ALLOC(int, n); | |
|     if (temp == NULL) { | |
| 	bdd->errorCode = CUDD_MEMORY_OUT; | |
| 	(void) fprintf(bdd->out, | |
| 		       "Cudd_SolveEqn: Out of memory for yIndex\n"); | |
| 	return(NULL); | |
|     } | |
| 
 | |
|     do { | |
| 	bdd->reordered = 0; | |
| 	res = cuddSolveEqnRecur(bdd, F, Y, G, n, temp, 0); | |
|     } while (bdd->reordered == 1); | |
|     if (bdd->errorCode == CUDD_TIMEOUT_EXPIRED && bdd->timeoutHandler) { | |
|         bdd->timeoutHandler(bdd, bdd->tohArg); | |
|     } | |
| 
 | |
|     return(res); | |
| 
 | |
| } /* end of Cudd_SolveEqn */ | |
| 
 | |
| 
 | |
| /** | |
|   @brief Checks the solution of F(x,y) = 0. | |
|  | |
|   @details This procedure substitutes the solution components for the | |
|   unknowns of F and returns the resulting %BDD for F. | |
|  | |
|   @sideeffect Frees the memory pointed by yIndex. | |
|  | |
|   @see Cudd_SolveEqn | |
|  | |
| */ | |
| DdNode * | |
| Cudd_VerifySol( | |
|   DdManager *  bdd /**< CUDD manager */, | |
|   DdNode * F /**< the left-hand side of the equation */, | |
|   DdNode ** G /**< the array of solutions */, | |
|   int * yIndex /**< index of y variables */, | |
|   int  n /**< numbers of unknowns */) | |
| { | |
|     DdNode *res; | |
| 
 | |
|     do { | |
| 	bdd->reordered = 0; | |
| 	res = cuddVerifySol(bdd, F, G, yIndex, n); | |
|     } while (bdd->reordered == 1); | |
| 
 | |
|     FREE(yIndex); | |
|     if (bdd->errorCode == CUDD_TIMEOUT_EXPIRED && bdd->timeoutHandler) { | |
|         bdd->timeoutHandler(bdd, bdd->tohArg); | |
|     } | |
| 
 | |
|     return(res); | |
| 
 | |
| } /* end of Cudd_VerifySol */ | |
| 
 | |
| 
 | |
| /*---------------------------------------------------------------------------*/ | |
| /* Definition of internal functions                                          */ | |
| /*---------------------------------------------------------------------------*/ | |
| 
 | |
| 
 | |
| /** | |
|   @brief Implements the recursive step of Cudd_SolveEqn. | |
|  | |
|   @return NULL if the intermediate solution blows up or reordering | |
|   occurs. | |
|  | |
|   @sideeffect The parametric solutions are stored in the array G. | |
|  | |
|   @see Cudd_SolveEqn, Cudd_VerifySol | |
|  | |
| */ | |
| DdNode * | |
| cuddSolveEqnRecur( | |
|   DdManager * bdd /**< CUDD manager */, | |
|   DdNode * F /**< the left-hand side of the equation */, | |
|   DdNode * Y /**< the cube of remaining y variables */, | |
|   DdNode ** G /**< the array of solutions */, | |
|   int  n /**< number of unknowns */, | |
|   int * yIndex /**< array holding the y variable indices */, | |
|   int  i /**< level of recursion */) | |
| { | |
|     DdNode *Fn, *Fm1, *Fv, *Fvbar, *T, *w, *nextY, *one; | |
|     DdNodePtr *variables; | |
| 
 | |
|     int j; | |
| 
 | |
|     statLine(bdd); | |
|     variables = bdd->vars; | |
|     one = DD_ONE(bdd); | |
| 
 | |
|     /* Base condition. */ | |
|     if (Y == one) { | |
| 	return F; | |
|     } | |
| 
 | |
|     /* Cofactor of Y. */ | |
|     yIndex[i] = Y->index; | |
|     nextY = Cudd_T(Y); | |
| 
 | |
|     /* Universal abstraction of F with respect to the top variable index. */ | |
|     Fm1 = cuddBddExistAbstractRecur(bdd, Cudd_Not(F), variables[yIndex[i]]); | |
|     if (Fm1) { | |
| 	Fm1 = Cudd_Not(Fm1); | |
| 	cuddRef(Fm1); | |
|     } else { | |
| 	return(NULL); | |
|     } | |
| 
 | |
|     Fn = cuddSolveEqnRecur(bdd, Fm1, nextY, G, n, yIndex, i+1); | |
|     if (Fn) { | |
| 	cuddRef(Fn); | |
|     } else { | |
| 	Cudd_RecursiveDeref(bdd, Fm1); | |
| 	return(NULL); | |
|     } | |
| 
 | |
|     Fv = cuddCofactorRecur(bdd, F, variables[yIndex[i]]); | |
|     if (Fv) { | |
| 	cuddRef(Fv); | |
|     } else { | |
| 	Cudd_RecursiveDeref(bdd, Fm1); | |
| 	Cudd_RecursiveDeref(bdd, Fn); | |
| 	return(NULL); | |
|     } | |
| 
 | |
|     Fvbar = cuddCofactorRecur(bdd, F, Cudd_Not(variables[yIndex[i]])); | |
|     if (Fvbar) { | |
| 	cuddRef(Fvbar); | |
|     } else { | |
| 	Cudd_RecursiveDeref(bdd, Fm1); | |
| 	Cudd_RecursiveDeref(bdd, Fn); | |
| 	Cudd_RecursiveDeref(bdd, Fv); | |
| 	return(NULL); | |
|     } | |
| 
 | |
|     /* Build i-th component of the solution. */ | |
|     w = cuddBddIteRecur(bdd, variables[yIndex[i]], Cudd_Not(Fv), Fvbar); | |
|     if (w) { | |
| 	cuddRef(w); | |
|     } else { | |
| 	Cudd_RecursiveDeref(bdd, Fm1); | |
| 	Cudd_RecursiveDeref(bdd, Fn); | |
| 	Cudd_RecursiveDeref(bdd, Fv); | |
| 	Cudd_RecursiveDeref(bdd, Fvbar); | |
| 	return(NULL); | |
|     } | |
| 
 | |
|     T = cuddBddRestrictRecur(bdd, w, Cudd_Not(Fm1)); | |
|     if(T) { | |
| 	cuddRef(T); | |
|     } else { | |
| 	Cudd_RecursiveDeref(bdd, Fm1); | |
| 	Cudd_RecursiveDeref(bdd, Fn); | |
| 	Cudd_RecursiveDeref(bdd, Fv); | |
| 	Cudd_RecursiveDeref(bdd, Fvbar); | |
| 	Cudd_RecursiveDeref(bdd, w); | |
| 	return(NULL); | |
|     } | |
| 
 | |
|     Cudd_RecursiveDeref(bdd,Fm1); | |
|     Cudd_RecursiveDeref(bdd,w); | |
|     Cudd_RecursiveDeref(bdd,Fv); | |
|     Cudd_RecursiveDeref(bdd,Fvbar); | |
| 
 | |
|     /* Substitute components of solution already found into solution. */ | |
|     for (j = n-1; j > i; j--) { | |
| 	w = cuddBddComposeRecur(bdd,T, G[j], variables[yIndex[j]]); | |
| 	if(w) { | |
| 	    cuddRef(w); | |
| 	} else { | |
| 	    Cudd_RecursiveDeref(bdd, Fn); | |
| 	    Cudd_RecursiveDeref(bdd, T); | |
| 	    return(NULL); | |
| 	} | |
| 	Cudd_RecursiveDeref(bdd,T); | |
| 	T = w; | |
|     } | |
|     G[i] = T; | |
| 
 | |
|     Cudd_Deref(Fn); | |
| 
 | |
|     return(Fn); | |
| 
 | |
| } /* end of cuddSolveEqnRecur */ | |
| 
 | |
| 
 | |
| /** | |
|   @brief Implements the recursive step of Cudd_VerifySol.  | |
|  | |
|   @sideeffect none | |
|  | |
|   @see Cudd_VerifySol | |
|  | |
| */ | |
| DdNode * | |
| cuddVerifySol( | |
|   DdManager * bdd /**< CUDD manager */, | |
|   DdNode * F /**< the left-hand side of the equation */, | |
|   DdNode ** G /**< the array of solutions */, | |
|   int * yIndex /**< array holding the y variable indices */, | |
|   int  n /**< number of unknowns */) | |
| { | |
|     DdNode *w, *R; | |
| 
 | |
|     int j; | |
| 
 | |
|     R = F; | |
|     cuddRef(R); | |
|     for(j = n - 1; j >= 0; j--) { | |
| 	 w = Cudd_bddCompose(bdd, R, G[j], yIndex[j]); | |
| 	if (w) { | |
| 	    cuddRef(w); | |
| 	} else { | |
| 	    return(NULL);  | |
| 	} | |
| 	Cudd_RecursiveDeref(bdd,R); | |
| 	R = w; | |
|     } | |
| 
 | |
|     cuddDeref(R); | |
| 
 | |
|     return(R); | |
| 
 | |
| } /* end of cuddVerifySol */ | |
| 
 | |
| 
 | |
| /*---------------------------------------------------------------------------*/ | |
| /* Definition of static functions                                            */ | |
| /*---------------------------------------------------------------------------*/ | |
| 
 |