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.
		
		
		
		
		
			
		
			
				
					
					
						
							579 lines
						
					
					
						
							17 KiB
						
					
					
				
			
		
		
		
			
			
			
				
					
				
				
					
				
			
		
		
	
	
							579 lines
						
					
					
						
							17 KiB
						
					
					
				| /**CFile*********************************************************************** | |
|  | |
|   FileName    [cuddAddAbs.c] | |
|  | |
|   PackageName [cudd] | |
|  | |
|   Synopsis    [Quantification functions for ADDs.] | |
|  | |
|   Description [External procedures included in this module: | |
| 		<ul> | |
| 		<li> Cudd_addExistAbstract() | |
| 		<li> Cudd_addUnivAbstract() | |
| 		<li> Cudd_addOrAbstract() | |
| 		</ul> | |
| 	Internal procedures included in this module: | |
| 		<ul> | |
| 		<li> cuddAddExistAbstractRecur() | |
| 		<li> cuddAddUnivAbstractRecur() | |
| 		<li> cuddAddOrAbstractRecur() | |
| 		</ul> | |
| 	Static procedures included in this module: | |
| 		<ul> | |
| 		<li> addCheckPositiveCube() | |
| 		</ul>] | |
|  | |
|   Author      [Fabio Somenzi] | |
|  | |
|   Copyright   [Copyright (c) 1995-2012, 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.] | |
|  | |
| ******************************************************************************/ | |
| 
 | |
| #include "util.h" | |
| #include "cuddInt.h" | |
|  | |
| /*---------------------------------------------------------------------------*/ | |
| /* Constant declarations                                                     */ | |
| /*---------------------------------------------------------------------------*/ | |
| 
 | |
| 
 | |
| /*---------------------------------------------------------------------------*/ | |
| /* Stucture declarations                                                     */ | |
| /*---------------------------------------------------------------------------*/ | |
| 
 | |
| 
 | |
| /*---------------------------------------------------------------------------*/ | |
| /* Type declarations                                                         */ | |
| /*---------------------------------------------------------------------------*/ | |
| 
 | |
| 
 | |
| /*---------------------------------------------------------------------------*/ | |
| /* Variable declarations                                                     */ | |
| /*---------------------------------------------------------------------------*/ | |
| 
 | |
| #ifndef lint | |
| static char rcsid[] DD_UNUSED = "$Id: cuddAddAbs.c,v 1.16 2012/02/05 01:07:18 fabio Exp $"; | |
| #endif | |
|  | |
| static	DdNode	*two; | |
| 
 | |
| /*---------------------------------------------------------------------------*/ | |
| /* Macro declarations                                                        */ | |
| /*---------------------------------------------------------------------------*/ | |
| 
 | |
| 
 | |
| /**AutomaticStart*************************************************************/ | |
| 
 | |
| /*---------------------------------------------------------------------------*/ | |
| /* Static function prototypes                                                */ | |
| /*---------------------------------------------------------------------------*/ | |
| 
 | |
| static int addCheckPositiveCube (DdManager *manager, DdNode *cube); | |
| 
 | |
| /**AutomaticEnd***************************************************************/ | |
| 
 | |
| 
 | |
| /*---------------------------------------------------------------------------*/ | |
| /* Definition of exported functions                                          */ | |
| /*---------------------------------------------------------------------------*/ | |
| 
 | |
| /**Function******************************************************************** | |
|  | |
|   Synopsis    [Existentially Abstracts all the variables in cube from f.] | |
|  | |
|   Description [Abstracts all the variables in cube from f by summing | |
|   over all possible values taken by the variables. Returns the | |
|   abstracted ADD.] | |
|  | |
|   SideEffects [None] | |
|  | |
|   SeeAlso     [Cudd_addUnivAbstract Cudd_bddExistAbstract | |
|   Cudd_addOrAbstract] | |
|  | |
| ******************************************************************************/ | |
| DdNode * | |
| Cudd_addExistAbstract( | |
|   DdManager * manager, | |
|   DdNode * f, | |
|   DdNode * cube) | |
| { | |
|     DdNode *res; | |
| 
 | |
|     two = cuddUniqueConst(manager,(CUDD_VALUE_TYPE) 2); | |
|     if (two == NULL) return(NULL); | |
|     cuddRef(two); | |
| 
 | |
|     if (addCheckPositiveCube(manager, cube) == 0) { | |
|         (void) fprintf(manager->err,"Error: Can only abstract cubes"); | |
|         return(NULL); | |
|     } | |
| 
 | |
|     do { | |
| 	manager->reordered = 0; | |
| 	res = cuddAddExistAbstractRecur(manager, f, cube); | |
|     } while (manager->reordered == 1); | |
| 
 | |
|     if (res == NULL) { | |
| 	Cudd_RecursiveDeref(manager,two); | |
| 	return(NULL); | |
|     } | |
|     cuddRef(res); | |
|     Cudd_RecursiveDeref(manager,two); | |
|     cuddDeref(res); | |
| 
 | |
|     return(res); | |
| 
 | |
| } /* end of Cudd_addExistAbstract */ | |
| 
 | |
| 
 | |
| /**Function******************************************************************** | |
|  | |
|   Synopsis    [Universally Abstracts all the variables in cube from f.] | |
|  | |
|   Description [Abstracts all the variables in cube from f by taking | |
|   the product over all possible values taken by the variable. Returns | |
|   the abstracted ADD if successful; NULL otherwise.] | |
|  | |
|   SideEffects [None] | |
|  | |
|   SeeAlso     [Cudd_addExistAbstract Cudd_bddUnivAbstract | |
|   Cudd_addOrAbstract] | |
|  | |
| ******************************************************************************/ | |
| DdNode * | |
| Cudd_addUnivAbstract( | |
|   DdManager * manager, | |
|   DdNode * f, | |
|   DdNode * cube) | |
| { | |
|     DdNode		*res; | |
| 
 | |
|     if (addCheckPositiveCube(manager, cube) == 0) { | |
| 	(void) fprintf(manager->err,"Error:  Can only abstract cubes"); | |
| 	return(NULL); | |
|     } | |
| 
 | |
|     do { | |
| 	manager->reordered = 0; | |
| 	res = cuddAddUnivAbstractRecur(manager, f, cube); | |
|     } while (manager->reordered == 1); | |
| 
 | |
|     return(res); | |
| 
 | |
| } /* end of Cudd_addUnivAbstract */ | |
| 
 | |
| 
 | |
| /**Function******************************************************************** | |
|  | |
|   Synopsis    [Disjunctively abstracts all the variables in cube from the | |
|   0-1 ADD f.] | |
|  | |
|   Description [Abstracts all the variables in cube from the 0-1 ADD f | |
|   by taking the disjunction over all possible values taken by the | |
|   variables.  Returns the abstracted ADD if successful; NULL | |
|   otherwise.] | |
|  | |
|   SideEffects [None] | |
|  | |
|   SeeAlso     [Cudd_addUnivAbstract Cudd_addExistAbstract] | |
|  | |
| ******************************************************************************/ | |
| DdNode * | |
| Cudd_addOrAbstract( | |
|   DdManager * manager, | |
|   DdNode * f, | |
|   DdNode * cube) | |
| { | |
|     DdNode *res; | |
| 
 | |
|     if (addCheckPositiveCube(manager, cube) == 0) { | |
|         (void) fprintf(manager->err,"Error: Can only abstract cubes"); | |
|         return(NULL); | |
|     } | |
| 
 | |
|     do { | |
| 	manager->reordered = 0; | |
| 	res = cuddAddOrAbstractRecur(manager, f, cube); | |
|     } while (manager->reordered == 1); | |
|     return(res); | |
| 
 | |
| } /* end of Cudd_addOrAbstract */ | |
| 
 | |
| 
 | |
| /*---------------------------------------------------------------------------*/ | |
| /* Definition of internal functions                                          */ | |
| /*---------------------------------------------------------------------------*/ | |
| 
 | |
| 
 | |
| /**Function******************************************************************** | |
|  | |
|   Synopsis    [Performs the recursive step of Cudd_addExistAbstract.] | |
|  | |
|   Description [Performs the recursive step of Cudd_addExistAbstract. | |
|   Returns the ADD obtained by abstracting the variables of cube from f, | |
|   if successful; NULL otherwise.] | |
|  | |
|   SideEffects [None] | |
|  | |
|   SeeAlso     [] | |
|  | |
| ******************************************************************************/ | |
| DdNode * | |
| cuddAddExistAbstractRecur( | |
|   DdManager * manager, | |
|   DdNode * f, | |
|   DdNode * cube) | |
| { | |
|     DdNode	*T, *E, *res, *res1, *res2, *zero; | |
| 
 | |
|     statLine(manager); | |
|     zero = DD_ZERO(manager); | |
| 
 | |
|     /* Cube is guaranteed to be a cube at this point. */	 | |
|     if (f == zero || cuddIsConstant(cube)) {   | |
|         return(f); | |
|     } | |
| 
 | |
|     /* Abstract a variable that does not appear in f => multiply by 2. */ | |
|     if (cuddI(manager,f->index) > cuddI(manager,cube->index)) { | |
| 	res1 = cuddAddExistAbstractRecur(manager, f, cuddT(cube)); | |
| 	if (res1 == NULL) return(NULL); | |
| 	cuddRef(res1); | |
| 	/* Use the "internal" procedure to be alerted in case of | |
| 	** dynamic reordering. If dynamic reordering occurs, we | |
| 	** have to abort the entire abstraction. | |
| 	*/ | |
| 	res = cuddAddApplyRecur(manager,Cudd_addTimes,res1,two); | |
| 	if (res == NULL) { | |
| 	    Cudd_RecursiveDeref(manager,res1); | |
| 	    return(NULL); | |
| 	} | |
| 	cuddRef(res); | |
| 	Cudd_RecursiveDeref(manager,res1); | |
| 	cuddDeref(res); | |
|         return(res); | |
|     } | |
| 
 | |
|     if ((res = cuddCacheLookup2(manager, Cudd_addExistAbstract, f, cube)) != NULL) { | |
| 	return(res); | |
|     } | |
| 
 | |
|     T = cuddT(f); | |
|     E = cuddE(f); | |
| 
 | |
|     /* If the two indices are the same, so are their levels. */ | |
|     if (f->index == cube->index) { | |
| 	res1 = cuddAddExistAbstractRecur(manager, T, cuddT(cube)); | |
| 	if (res1 == NULL) return(NULL); | |
|         cuddRef(res1); | |
| 	res2 = cuddAddExistAbstractRecur(manager, E, cuddT(cube)); | |
| 	if (res2 == NULL) { | |
| 	    Cudd_RecursiveDeref(manager,res1); | |
| 	    return(NULL); | |
| 	} | |
|         cuddRef(res2); | |
| 	res = cuddAddApplyRecur(manager, Cudd_addPlus, res1, res2); | |
| 	if (res == NULL) { | |
| 	    Cudd_RecursiveDeref(manager,res1); | |
| 	    Cudd_RecursiveDeref(manager,res2); | |
| 	    return(NULL); | |
| 	} | |
| 	cuddRef(res); | |
| 	Cudd_RecursiveDeref(manager,res1); | |
| 	Cudd_RecursiveDeref(manager,res2); | |
| 	cuddCacheInsert2(manager, Cudd_addExistAbstract, f, cube, res); | |
| 	cuddDeref(res); | |
|         return(res); | |
|     } else { /* if (cuddI(manager,f->index) < cuddI(manager,cube->index)) */ | |
| 	res1 = cuddAddExistAbstractRecur(manager, T, cube); | |
| 	if (res1 == NULL) return(NULL); | |
|         cuddRef(res1); | |
| 	res2 = cuddAddExistAbstractRecur(manager, E, cube); | |
| 	if (res2 == NULL) { | |
| 	    Cudd_RecursiveDeref(manager,res1); | |
| 	    return(NULL); | |
| 	} | |
|         cuddRef(res2); | |
| 	res = (res1 == res2) ? res1 : | |
| 	    cuddUniqueInter(manager, (int) f->index, res1, res2); | |
| 	if (res == NULL) { | |
| 	    Cudd_RecursiveDeref(manager,res1); | |
| 	    Cudd_RecursiveDeref(manager,res2); | |
| 	    return(NULL); | |
| 	} | |
| 	cuddDeref(res1); | |
| 	cuddDeref(res2); | |
| 	cuddCacheInsert2(manager, Cudd_addExistAbstract, f, cube, res); | |
|         return(res); | |
|     }	     | |
| 
 | |
| } /* end of cuddAddExistAbstractRecur */ | |
| 
 | |
| 
 | |
| /**Function******************************************************************** | |
|  | |
|   Synopsis    [Performs the recursive step of Cudd_addUnivAbstract.] | |
|  | |
|   Description [Performs the recursive step of Cudd_addUnivAbstract. | |
|   Returns the ADD obtained by abstracting the variables of cube from f, | |
|   if successful; NULL otherwise.] | |
|  | |
|   SideEffects [None] | |
|  | |
|   SeeAlso     [] | |
|  | |
| ******************************************************************************/ | |
| DdNode * | |
| cuddAddUnivAbstractRecur( | |
|   DdManager * manager, | |
|   DdNode * f, | |
|   DdNode * cube) | |
| { | |
|     DdNode	*T, *E, *res, *res1, *res2, *one, *zero; | |
| 
 | |
|     statLine(manager); | |
|     one = DD_ONE(manager); | |
|     zero = DD_ZERO(manager); | |
| 
 | |
|     /* Cube is guaranteed to be a cube at this point. | |
|     ** zero and one are the only constatnts c such that c*c=c. | |
|     */ | |
|     if (f == zero || f == one || cube == one) {   | |
| 	return(f); | |
|     } | |
| 
 | |
|     /* Abstract a variable that does not appear in f. */ | |
|     if (cuddI(manager,f->index) > cuddI(manager,cube->index)) { | |
| 	res1 = cuddAddUnivAbstractRecur(manager, f, cuddT(cube)); | |
| 	if (res1 == NULL) return(NULL); | |
| 	cuddRef(res1); | |
| 	/* Use the "internal" procedure to be alerted in case of | |
| 	** dynamic reordering. If dynamic reordering occurs, we | |
| 	** have to abort the entire abstraction. | |
| 	*/ | |
| 	res = cuddAddApplyRecur(manager, Cudd_addTimes, res1, res1); | |
| 	if (res == NULL) { | |
| 	    Cudd_RecursiveDeref(manager,res1); | |
| 	    return(NULL); | |
| 	} | |
| 	cuddRef(res); | |
| 	Cudd_RecursiveDeref(manager,res1); | |
| 	cuddDeref(res); | |
| 	return(res); | |
|     } | |
| 
 | |
|     if ((res = cuddCacheLookup2(manager, Cudd_addUnivAbstract, f, cube)) != NULL) { | |
| 	return(res); | |
|     } | |
| 
 | |
|     T = cuddT(f); | |
|     E = cuddE(f); | |
| 
 | |
|     /* If the two indices are the same, so are their levels. */ | |
|     if (f->index == cube->index) { | |
| 	res1 = cuddAddUnivAbstractRecur(manager, T, cuddT(cube)); | |
| 	if (res1 == NULL) return(NULL); | |
|         cuddRef(res1); | |
| 	res2 = cuddAddUnivAbstractRecur(manager, E, cuddT(cube)); | |
| 	if (res2 == NULL) { | |
| 	    Cudd_RecursiveDeref(manager,res1); | |
| 	    return(NULL); | |
| 	} | |
|         cuddRef(res2); | |
| 	res = cuddAddApplyRecur(manager, Cudd_addTimes, res1, res2); | |
| 	if (res == NULL) { | |
| 	    Cudd_RecursiveDeref(manager,res1); | |
| 	    Cudd_RecursiveDeref(manager,res2); | |
| 	    return(NULL); | |
| 	} | |
| 	cuddRef(res); | |
| 	Cudd_RecursiveDeref(manager,res1); | |
| 	Cudd_RecursiveDeref(manager,res2); | |
| 	cuddCacheInsert2(manager, Cudd_addUnivAbstract, f, cube, res); | |
| 	cuddDeref(res); | |
|         return(res); | |
|     } else { /* if (cuddI(manager,f->index) < cuddI(manager,cube->index)) */ | |
| 	res1 = cuddAddUnivAbstractRecur(manager, T, cube); | |
| 	if (res1 == NULL) return(NULL); | |
|         cuddRef(res1); | |
| 	res2 = cuddAddUnivAbstractRecur(manager, E, cube); | |
| 	if (res2 == NULL) { | |
| 	    Cudd_RecursiveDeref(manager,res1); | |
| 	    return(NULL); | |
| 	} | |
|         cuddRef(res2); | |
| 	res = (res1 == res2) ? res1 : | |
| 	    cuddUniqueInter(manager, (int) f->index, res1, res2); | |
| 	if (res == NULL) { | |
| 	    Cudd_RecursiveDeref(manager,res1); | |
| 	    Cudd_RecursiveDeref(manager,res2); | |
| 	    return(NULL); | |
| 	} | |
| 	cuddDeref(res1); | |
| 	cuddDeref(res2); | |
| 	cuddCacheInsert2(manager, Cudd_addUnivAbstract, f, cube, res); | |
|         return(res); | |
|     } | |
| 
 | |
| } /* end of cuddAddUnivAbstractRecur */ | |
| 
 | |
| 
 | |
| /**Function******************************************************************** | |
|  | |
|   Synopsis    [Performs the recursive step of Cudd_addOrAbstract.] | |
|  | |
|   Description [Performs the recursive step of Cudd_addOrAbstract. | |
|   Returns the ADD obtained by abstracting the variables of cube from f, | |
|   if successful; NULL otherwise.] | |
|  | |
|   SideEffects [None] | |
|  | |
|   SeeAlso     [] | |
|  | |
| ******************************************************************************/ | |
| DdNode * | |
| cuddAddOrAbstractRecur( | |
|   DdManager * manager, | |
|   DdNode * f, | |
|   DdNode * cube) | |
| { | |
|     DdNode	*T, *E, *res, *res1, *res2, *one; | |
| 
 | |
|     statLine(manager); | |
|     one = DD_ONE(manager); | |
| 
 | |
|     /* Cube is guaranteed to be a cube at this point. */ | |
|     if (cuddIsConstant(f) || cube == one) {   | |
| 	return(f); | |
|     } | |
| 
 | |
|     /* Abstract a variable that does not appear in f. */ | |
|     if (cuddI(manager,f->index) > cuddI(manager,cube->index)) { | |
| 	res = cuddAddOrAbstractRecur(manager, f, cuddT(cube)); | |
| 	return(res); | |
|     } | |
| 
 | |
|     if ((res = cuddCacheLookup2(manager, Cudd_addOrAbstract, f, cube)) != NULL) { | |
| 	return(res); | |
|     } | |
| 
 | |
|     T = cuddT(f); | |
|     E = cuddE(f); | |
| 
 | |
|     /* If the two indices are the same, so are their levels. */ | |
|     if (f->index == cube->index) { | |
| 	res1 = cuddAddOrAbstractRecur(manager, T, cuddT(cube)); | |
| 	if (res1 == NULL) return(NULL); | |
|         cuddRef(res1); | |
| 	if (res1 != one) { | |
| 	    res2 = cuddAddOrAbstractRecur(manager, E, cuddT(cube)); | |
| 	    if (res2 == NULL) { | |
| 		Cudd_RecursiveDeref(manager,res1); | |
| 		return(NULL); | |
| 	    } | |
| 	    cuddRef(res2); | |
| 	    res = cuddAddApplyRecur(manager, Cudd_addOr, res1, res2); | |
| 	    if (res == NULL) { | |
| 		Cudd_RecursiveDeref(manager,res1); | |
| 		Cudd_RecursiveDeref(manager,res2); | |
| 		return(NULL); | |
| 	    } | |
| 	    cuddRef(res); | |
| 	    Cudd_RecursiveDeref(manager,res1); | |
| 	    Cudd_RecursiveDeref(manager,res2); | |
| 	} else { | |
| 	    res = res1; | |
| 	} | |
| 	cuddCacheInsert2(manager, Cudd_addOrAbstract, f, cube, res); | |
| 	cuddDeref(res); | |
|         return(res); | |
|     } else { /* if (cuddI(manager,f->index) < cuddI(manager,cube->index)) */ | |
| 	res1 = cuddAddOrAbstractRecur(manager, T, cube); | |
| 	if (res1 == NULL) return(NULL); | |
|         cuddRef(res1); | |
| 	res2 = cuddAddOrAbstractRecur(manager, E, cube); | |
| 	if (res2 == NULL) { | |
| 	    Cudd_RecursiveDeref(manager,res1); | |
| 	    return(NULL); | |
| 	} | |
|         cuddRef(res2); | |
| 	res = (res1 == res2) ? res1 : | |
| 	    cuddUniqueInter(manager, (int) f->index, res1, res2); | |
| 	if (res == NULL) { | |
| 	    Cudd_RecursiveDeref(manager,res1); | |
| 	    Cudd_RecursiveDeref(manager,res2); | |
| 	    return(NULL); | |
| 	} | |
| 	cuddDeref(res1); | |
| 	cuddDeref(res2); | |
| 	cuddCacheInsert2(manager, Cudd_addOrAbstract, f, cube, res); | |
|         return(res); | |
|     } | |
| 
 | |
| } /* end of cuddAddOrAbstractRecur */ | |
| 
 | |
| 
 | |
| 
 | |
| /*---------------------------------------------------------------------------*/ | |
| /* Definition of static functions                                            */ | |
| /*---------------------------------------------------------------------------*/ | |
| 
 | |
| 
 | |
| /**Function******************************************************************** | |
|  | |
|   Synopsis    [Checks whether cube is an ADD representing the product | |
|   of positive literals.] | |
|  | |
|   Description [Checks whether cube is an ADD representing the product of | |
|   positive literals. Returns 1 in case of success; 0 otherwise.] | |
|  | |
|   SideEffects [None] | |
|  | |
|   SeeAlso     [] | |
|  | |
| ******************************************************************************/ | |
| static int | |
| addCheckPositiveCube( | |
|   DdManager * manager, | |
|   DdNode * cube) | |
| { | |
|     if (Cudd_IsComplement(cube)) return(0); | |
|     if (cube == DD_ONE(manager)) return(1); | |
|     if (cuddIsConstant(cube)) return(0); | |
|     if (cuddE(cube) == DD_ZERO(manager)) { | |
|         return(addCheckPositiveCube(manager, cuddT(cube))); | |
|     } | |
|     return(0); | |
| 
 | |
| } /* end of addCheckPositiveCube */ | |
| 
 |