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.
 
 
 
 
 
 

1117 lines
32 KiB

/**CFile**********************************************************************
FileName [dddmpStoreBdd.c]
PackageName [dddmp]
Synopsis [Functions to write BDDs to file.]
Description [Functions to write BDDs to file.
BDDs are represended on file either in text or binary format under the
following rules. A file contains a forest of BDDs (a vector of
Boolean functions). BDD nodes are numbered with contiguous numbers,
from 1 to NNodes (total number of nodes on a file). 0 is not used to
allow negative node indexes for complemented edges. A file contains
a header, including information about variables and roots to BDD
functions, followed by the list of nodes. BDD nodes are listed
according to their numbering, and in the present implementation
numbering follows a post-order strategy, in such a way that a node
is never listed before its Then/Else children.
]
Author [Gianpiero Cabodi and Stefano Quer]
Copyright [
Copyright (c) 2004 by Politecnico di Torino.
All Rights Reserved. This software is for educational purposes only.
Permission is given to academic institutions to use, copy, and modify
this software and its documentation provided that this introductory
message is not removed, that this software and its documentation is
used for the institutions' internal research and educational purposes,
and that no monies are exchanged. No guarantee is expressed or implied
by the distribution of this code.
Send bug-reports and/or questions to:
{gianpiero.cabodi,stefano.quer}@polito.it.
]
******************************************************************************/
#include "dddmpInt.h"
/*---------------------------------------------------------------------------*/
/* Stucture declarations */
/*---------------------------------------------------------------------------*/
/*---------------------------------------------------------------------------*/
/* Type declarations */
/*---------------------------------------------------------------------------*/
/*---------------------------------------------------------------------------*/
/* Variable declarations */
/*---------------------------------------------------------------------------*/
/*---------------------------------------------------------------------------*/
/* Macro declarations */
/*---------------------------------------------------------------------------*/
/**AutomaticStart*************************************************************/
/*---------------------------------------------------------------------------*/
/* Static function prototypes */
/*---------------------------------------------------------------------------*/
static int NodeStoreRecurBdd(DdManager *ddMgr, DdNode *f, int mode, int *supportids, char **varnames, int *outids, FILE *fp);
static int NodeTextStoreBdd(DdManager *ddMgr, DdNode *f, int mode, int *supportids, char **varnames, int *outids, FILE *fp, int idf, int vf, int idT, int idE);
static int NodeBinaryStoreBdd(DdManager *ddMgr, DdNode *f, int mode, int *supportids, char **varnames, int *outids, FILE *fp, int idf, int vf, int idT, int idE, int vT, int vE, DdNode *T, DdNode *E);
/**AutomaticEnd***************************************************************/
/*---------------------------------------------------------------------------*/
/* Definition of exported functions */
/*---------------------------------------------------------------------------*/
/**Function********************************************************************
Synopsis [Writes a dump file representing the argument BDD.]
Description [Dumps the argument BDD to file. Dumping is done through
Dddmp_cuddBddArrayStore. A dummy array of 1 BDD root is
used for this purpose.
]
SideEffects [Nodes are temporarily removed from unique hash. They are
re-linked after the store operation in a modified order.
]
SeeAlso [Dddmp_cuddBddLoad Dddmp_cuddBddArrayLoad]
******************************************************************************/
int
Dddmp_cuddBddStore (
DdManager *ddMgr /* IN: DD Manager */,
char *ddname /* IN: DD name (or NULL) */,
DdNode *f /* IN: BDD root to be stored */,
char **varnames /* IN: array of variable names (or NULL) */,
int *auxids /* IN: array of converted var ids */,
int mode /* IN: storing mode selector */,
Dddmp_VarInfoType varinfo /* IN: extra info for variables in text mode */,
char *fname /* IN: File name */,
FILE *fp /* IN: File pointer to the store file */
)
{
int retValue;
DdNode *tmpArray[1];
tmpArray[0] = f;
retValue = Dddmp_cuddBddArrayStore (ddMgr,ddname,1,tmpArray,NULL,
varnames, auxids, mode, varinfo, fname, fp);
return (retValue);
}
/**Function********************************************************************
Synopsis [Writes a dump file representing the argument Array of BDDs.]
Description [Dumps the argument array of BDDs to file. Dumping is either
in text or binary form. BDDs are stored to the fp (already
open) file if not NULL. Otherwise the file whose name is
fname is opened in write mode. The header has the same format
for both textual and binary dump. Names are allowed for input
variables (vnames) and for represented functions (rnames).
For sake of generality and because of dynamic variable
ordering both variable IDs and permuted IDs are included.
New IDs are also supported (auxids). Variables are identified
with incremental numbers. according with their positiom in
the support set. In text mode, an extra info may be added,
chosen among the following options: name, ID, PermID, or an
auxiliary id. Since conversion from DD pointers to integers
is required, DD nodes are temporarily removed from the unique
hash table. This allows the use of the next field to store
node IDs.
]
SideEffects [Nodes are temporarily removed from the unique hash
table. They are re-linked after the store operation in a
modified order.
]
SeeAlso [Dddmp_cuddBddStore, Dddmp_cuddBddLoad,
Dddmp_cuddBddArrayLoad
]
******************************************************************************/
int
Dddmp_cuddBddArrayStore (
DdManager *ddMgr /* IN: DD Manager */,
char *ddname /* IN: dd name (or NULL) */,
int nRoots /* IN: number of output BDD roots to be stored */,
DdNode **f /* IN: array of BDD roots to be stored */,
char **rootnames /* IN: array of root names (or NULL) */,
char **varnames /* IN: array of variable names (or NULL) */,
int *auxids /* IN: array of converted var IDs */,
int mode /* IN: storing mode selector */,
Dddmp_VarInfoType varinfo /* IN: extra info for variables in text mode */,
char *fname /* IN: File name */,
FILE *fp /* IN: File pointer to the store file */
)
{
int retValue;
#ifdef DDDMP_DEBUG
#ifndef __alpha__
int retValueBis;
retValueBis = Cudd_DebugCheck (ddMgr);
if (retValueBis == 1) {
fprintf (stderr, "Inconsistency Found During BDD Store.\n");
fflush (stderr);
} else {
if (retValueBis == CUDD_OUT_OF_MEM) {
fprintf (stderr, "Out of Memory During BDD Store.\n");
fflush (stderr);
}
}
#endif
#endif
retValue = DddmpCuddBddArrayStore (DDDMP_BDD, ddMgr, ddname, nRoots, f,
rootnames, varnames, auxids, mode, varinfo, fname, fp);
#ifdef DDDMP_DEBUG
#ifndef __alpha__
retValueBis = Cudd_DebugCheck (ddMgr);
if (retValueBis == 1) {
fprintf (stderr, "Inconsistency Found During BDD Store.\n");
fflush (stderr);
} else {
if (retValueBis == CUDD_OUT_OF_MEM) {
fprintf (stderr, "Out of Memory During BDD Store.\n");
fflush (stderr);
}
}
#endif
#endif
return (retValue);
}
/*---------------------------------------------------------------------------*/
/* Definition of internal functions */
/*---------------------------------------------------------------------------*/
/**Function********************************************************************
Synopsis [Writes a dump file representing the argument Array of
BDDs.
]
Description [Dumps the argument array of BDDs to file.
Internal function doing inner steps of store for BDDs.
]
SideEffects [Nodes are temporarily removed from the unique hash
table. They are re-linked after the store operation in a
modified order.
]
SeeAlso [Dddmp_cuddBddStore, Dddmp_cuddBddLoad,
Dddmp_cuddBddArrayLoad
]
******************************************************************************/
int
DddmpCuddBddArrayStore (
Dddmp_DecompType ddType /* IN: Selects the decomp type BDD */,
DdManager *ddMgr /* IN: DD Manager */,
char *ddname /* IN: DD name (or NULL) */,
int nRoots /* IN: number of output BDD roots to be stored */,
DdNode **f /* IN: array of DD roots to be stored */,
char **rootnames /* IN: array of root names (or NULL) */,
char **varnames /* IN: array of variable names (or NULL) */,
int *auxids /* IN: array of converted var IDs */,
int mode /* IN: storing mode selector */,
Dddmp_VarInfoType varinfo /* IN: extra info for variables in text mode */,
char *fname /* IN: File name */,
FILE *fp /* IN: File pointer to the store file */
)
{
DdNode *support = NULL;
DdNode *scan;
int *ids = NULL;
int *permids = NULL;
int *invpermids = NULL;
int *supportids = NULL;
int *outids = NULL;
char **outvarnames = NULL;
int nVars = ddMgr->size;
int nnodes;
int retValue;
int i, var;
int fileToClose = 0;
/*
* Check DD Type
*/
Dddmp_CheckAndGotoLabel (ddType==DDDMP_ADD,
"Error writing to file: ADD Type.", failure);
/*
* Check if File needs to be opened in the proper mode.
*/
if (fp == NULL) {
fp = fopen (fname, "w");
Dddmp_CheckAndGotoLabel (fp==NULL, "Error opening file.",
failure);
fileToClose = 1;
}
/*
* Force binary mode if automatic.
*/
switch (mode) {
case DDDMP_MODE_TEXT:
case DDDMP_MODE_BINARY:
break;
case DDDMP_MODE_DEFAULT:
mode = DDDMP_MODE_BINARY;
break;
default:
mode = DDDMP_MODE_BINARY;
break;
}
/*
* Alloc vectors for variable IDs, perm IDs and support IDs.
* +1 to include a slot for terminals.
*/
ids = DDDMP_ALLOC (int, nVars);
Dddmp_CheckAndGotoLabel (ids==NULL, "Error allocating memory.", failure);
permids = DDDMP_ALLOC (int, nVars);
Dddmp_CheckAndGotoLabel (permids==NULL, "Error allocating memory.", failure);
invpermids = DDDMP_ALLOC (int, nVars);
Dddmp_CheckAndGotoLabel (invpermids==NULL, "Error allocating memory.",
failure);
supportids = DDDMP_ALLOC (int, nVars+1);
Dddmp_CheckAndGotoLabel (supportids==NULL, "Error allocating memory.",
failure);
for (i=0; i<nVars; i++) {
ids[i] = permids[i] = invpermids[i] = supportids[i] = (-1);
}
/* StQ */
supportids[nVars] = -1;
/*
* Take the union of the supports of each output function.
* skip NULL functions.
* Set permids and invpermids of support variables to the proper values.
*/
for (i=0; i<nRoots; i++) {
if (f[i] == NULL) {
continue;
}
support = Cudd_Support (ddMgr, f[i]);
Dddmp_CheckAndGotoLabel (support==NULL, "NULL support returned.",
failure);
cuddRef (support);
scan = support;
while (!cuddIsConstant(scan)) {
ids[scan->index] = scan->index;
permids[scan->index] = ddMgr->perm[scan->index];
invpermids[ddMgr->perm[scan->index]] = scan->index;
scan = cuddT (scan);
}
Cudd_RecursiveDeref (ddMgr, support);
}
/* so that we do not try to free it in case of failure */
support = NULL;
/*
* Set supportids to incremental (shrinked) values following the ordering.
*/
for (i=0, var=0; i<nVars; i++) {
if (invpermids[i] >= 0) {
supportids[invpermids[i]] = var++;
}
}
/* set a dummy id for terminal nodes */
supportids[nVars] = var;
/*
* Select conversion array for extra var info
*/
switch (mode) {
case DDDMP_MODE_TEXT:
switch (varinfo) {
case DDDMP_VARIDS:
outids = ids;
break;
case DDDMP_VARPERMIDS:
outids = permids;
break;
case DDDMP_VARAUXIDS:
outids = auxids;
break;
case DDDMP_VARNAMES:
outvarnames = varnames;
break;
case DDDMP_VARDEFAULT:
break;
}
break;
case DDDMP_MODE_BINARY:
outids = NULL;
break;
}
/*
* Number dd nodes and count them (numbering is from 1 to nnodes)
*/
nnodes = DddmpNumberBddNodes (ddMgr, f, nRoots);
/*
* Start Header
*/
#ifdef DDDMP_VERSION
retValue = fprintf (fp, ".ver %s\n", DDDMP_VERSION);
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
#endif
retValue = fprintf (fp, ".mode %c\n", mode);
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
if (mode == DDDMP_MODE_TEXT) {
retValue = fprintf (fp, ".varinfo %d\n", varinfo);
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
}
if (ddname != NULL) {
retValue = fprintf (fp, ".dd %s\n",ddname);
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
}
retValue = fprintf (fp, ".nnodes %d\n", nnodes);
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
retValue = fprintf (fp, ".nvars %d\n", nVars);
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
retValue = fprintf (fp, ".nsuppvars %d\n", var);
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
/*------------ Write the Var Names by scanning the ids array -------------*/
if (varnames != NULL) {
retValue = fprintf (fp, ".suppvarnames");
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
for (i=0; i<nVars; i++) {
if (ids[i] >= 0) {
if (varnames[ids[i]] == NULL) {
(void) fprintf (stderr,
"DdStore Warning: null variable name. DUMMY%d generated\n", i);
fflush (stderr);
varnames[ids[i]] = DDDMP_ALLOC (char, 10);
Dddmp_CheckAndGotoLabel (varnames[ids[i]] == NULL,
"Error allocating memory.", failure);
sprintf (varnames[ids[i]], "DUMMY%d", i);
}
retValue = fprintf (fp, " %s", varnames[ids[i]]);
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
}
}
retValue = fprintf (fp, "\n");
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
}
/*--------- Write the Var SUPPORT Names by scanning the ids array ---------*/
if (varnames != NULL) {
retValue = fprintf (fp, ".orderedvarnames");
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
for (i=0; i<nVars; i++) {
if (varnames[ddMgr->invperm[i]] == NULL) {
(void) fprintf (stderr,
"DdStore Warning: null variable name. DUMMY%d generated\n", i);
fflush (stderr);
varnames[ddMgr->invperm[i]] = DDDMP_ALLOC (char, 10);
Dddmp_CheckAndGotoLabel (varnames[ddMgr->invperm[i]] == NULL,
"Error allocating memory.", failure);
sprintf (varnames[ddMgr->invperm[i]], "DUMMY%d", i);
}
retValue = fprintf (fp, " %s", varnames[ddMgr->invperm[i]]);
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
}
retValue = fprintf (fp, "\n");
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
}
/*------------ Write the var ids by scanning the ids array ---------------*/
retValue = fprintf (fp, ".ids");
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
for (i=0; i<nVars; i++) {
if (ids[i] >= 0) {
retValue = fprintf (fp, " %d", i);
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
}
}
retValue = fprintf (fp, "\n");
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
/*
* Write the var permids by scanning the permids array.
*/
retValue = fprintf (fp, ".permids");
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
for (i = 0; i < nVars; i++) {
if (permids[i] >= 0) {
retValue = fprintf (fp, " %d", permids[i]);
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
}
}
retValue = fprintf (fp, "\n");
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
if (auxids != NULL) {
/*
* Write the var auxids by scanning the ids array.
*/
retValue = fprintf (fp, ".auxids");
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
for (i = 0; i < nVars; i++) {
if (ids[i] >= 0) {
retValue = fprintf (fp, " %d", auxids[i]);
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
}
}
retValue = fprintf (fp, "\n");
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
}
/*
* Write the roots info.
*/
retValue = fprintf (fp, ".nroots %d\n", nRoots);
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
if (rootnames != NULL) {
/*
* Write the root names.
*/
retValue = fprintf (fp, ".rootnames");
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
for (i = 0; i < nRoots; i++) {
if (rootnames[i] == NULL) {
(void) fprintf (stderr,
"DdStore Warning: null variable name. ROOT%d generated\n",i);
fflush (stderr);
rootnames[i] = DDDMP_ALLOC(char,10);
Dddmp_CheckAndGotoLabel (rootnames[i]==NULL,
"Error writing to file.", failure);
sprintf(rootnames[ids[i]], "ROOT%d",i);
}
retValue = fprintf (fp, " %s", rootnames[i]);
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
}
retValue = fprintf (fp, "\n");
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
}
retValue = fprintf (fp, ".rootids");
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
/*
* Write BDD indexes of function roots.
* Use negative integers for complemented edges.
*/
for (i = 0; i < nRoots; i++) {
if (f[i] == NULL) {
(void) fprintf (stderr, "DdStore Warning: %d-th root is NULL\n",i);
fflush (stderr);
retValue = fprintf (fp, " 0");
}
if (Cudd_IsComplement(f[i])) {
retValue = fprintf (fp, " -%d",
DddmpReadNodeIndexBdd (Cudd_Regular (f[i])));
} else {
retValue = fprintf (fp, " %d",
DddmpReadNodeIndexBdd (Cudd_Regular (f[i])));
}
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
}
retValue = fprintf (fp, "\n");
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
retValue = fprintf (fp, ".nodes\n");
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
/*
* END HEADER
*/
/*
* Call the function that really gets the job done.
*/
for (i = 0; i < nRoots; i++) {
if (f[i] != NULL) {
retValue = NodeStoreRecurBdd (ddMgr, Cudd_Regular(f[i]),
mode, supportids, outvarnames, outids, fp);
Dddmp_CheckAndGotoLabel (retValue==DDDMP_FAILURE,
"Error writing to file.", failure);
}
}
/*
* Write trailer and return.
*/
retValue = fprintf (fp, ".end\n");
Dddmp_CheckAndGotoLabel (retValue==EOF, "Error writing to file.",
failure);
if (fileToClose) {
fclose (fp);
}
DddmpUnnumberBddNodes (ddMgr, f, nRoots);
DDDMP_FREE (ids);
DDDMP_FREE (permids);
DDDMP_FREE (invpermids);
DDDMP_FREE (supportids);
return (DDDMP_SUCCESS);
failure:
if (ids != NULL) {
DDDMP_FREE (ids);
}
if (permids != NULL) {
DDDMP_FREE (permids);
}
if (invpermids != NULL) {
DDDMP_FREE (invpermids);
}
if (supportids != NULL) {
DDDMP_FREE (supportids);
}
if (support != NULL) {
Cudd_RecursiveDeref (ddMgr, support);
}
return (DDDMP_FAILURE);
}
/*---------------------------------------------------------------------------*/
/* Definition of static functions */
/*---------------------------------------------------------------------------*/
/**Function********************************************************************
Synopsis [Performs the recursive step of Dddmp_bddStore.]
Description [Stores a node to file in either test or binary mode.<l>
In text mode a node is represented (on a text line basis) as
<UL>
<LI> node-index \[var-extrainfo\] var-index Then-index Else-index
</UL>
where all indexes are integer numbers and var-extrainfo
(optional redundant field) is either an integer or a string
(variable name). Node-index is redundant (due to the node
ordering) but we keep it for readability.<p>
In binary mode nodes are represented as a sequence of bytes,
representing var-index, Then-index, and Else-index in an
optimized way. Only the first byte (code) is mandatory.
Integer indexes are represented in absolute or relative mode,
where relative means offset wrt. a Then/Else node info.
Suppose Var(NodeId), Then(NodeId) and Else(NodeId) represent
infos about a given node.<p>
The generic "NodeId" node is stored as
<UL>
<LI> code-byte
<LI> \[var-info\]
<LI> \[Then-info\]
<LI> \[Else-info\]
</UL>
where code-byte contains bit fields
<UL>
<LI>Unused : 1 bit
<LI>Variable: 2 bits, one of the following codes
<UL>
<LI>DDDMP_ABSOLUTE_ID var-info = Var(NodeId) follows
<LI>DDDMP_RELATIVE_ID Var(NodeId) is represented in relative form as
var-info = Min(Var(Then(NodeId)),Var(Else(NodeId))) -Var(NodeId)
<LI>DDDMP_RELATIVE_1 No var-info follows, because
Var(NodeId) = Min(Var(Then(NodeId)),Var(Else(NodeId)))-1
<LI>DDDMP_TERMINAL Node is a terminal, no var info required
</UL>
<LI>T : 2 bits, with codes similar to V
<UL>
<LI>DDDMP_ABSOLUTE_ID Then-info = Then(NodeId) follows
<LI>DDDMP_RELATIVE_ID Then(NodeId) is represented in relative form as
Then-info = Nodeid-Then(NodeId)
<LI>DDDMP_RELATIVE_1 No info on Then(NodeId) follows, because
Then(NodeId) = NodeId-1
<LI>DDDMP_TERMINAL Then Node is a terminal, no info required (for BDDs)
</UL>
<LI>Ecompl : 1 bit, if 1 means complemented edge
<LI>E : 2 bits, with codes and meanings as for the Then edge
</UL>
var-info, Then-info, Else-info (if required) are represented as unsigned
integer values on a sufficient set of bytes (MSByte first).
]
SideEffects [None]
SeeAlso []
******************************************************************************/
static int
NodeStoreRecurBdd (
DdManager *ddMgr /* IN: DD Manager */,
DdNode *f /* IN: DD node to be stored */,
int mode /* IN: store mode */,
int *supportids /* IN: internal ids for variables */,
char **varnames /* IN: names of variables: to be stored with nodes */,
int *outids /* IN: output ids for variables */,
FILE *fp /* IN: store file */
)
{
DdNode *T = NULL;
DdNode *E = NULL;
int idf = (-1);
int idT = (-1);
int idE = (-1);
int vf = (-1);
int vT = (-1);
int vE = (-1);
int retValue;
int nVars;
nVars = ddMgr->size;
T = E = NULL;
idf = idT = idE = (-1);
#ifdef DDDMP_DEBUG
assert(!Cudd_IsComplement(f));
assert(f!=NULL);
assert(supportids!=NULL);
#endif
/* If already visited, nothing to do. */
if (DddmpVisitedBdd (f)) {
return (DDDMP_SUCCESS);
}
/* Mark node as visited. */
DddmpSetVisitedBdd (f);
if (Cudd_IsConstant(f)) {
/* Check for special case: don't recur */
idf = DddmpReadNodeIndexBdd (f);
} else {
#ifdef DDDMP_DEBUG
/* BDDs! Only one constant supported */
assert (!cuddIsConstant(f));
#endif
/*
* Recursive call for Then edge
*/
T = cuddT(f);
#ifdef DDDMP_DEBUG
/* ROBDDs! No complemented Then edge */
assert (!Cudd_IsComplement(T));
#endif
/* recur */
retValue = NodeStoreRecurBdd (ddMgr, T, mode, supportids, varnames, outids,
fp);
if (retValue != DDDMP_SUCCESS) {
return (retValue);
}
/*
* Recursive call for Else edge
*/
E = Cudd_Regular (cuddE (f));
retValue = NodeStoreRecurBdd (ddMgr, E, mode, supportids, varnames, outids,
fp);
if (retValue != DDDMP_SUCCESS) {
return (retValue);
}
/*
* Obtain nodeids and variable ids of f, T, E
*/
idf = DddmpReadNodeIndexBdd (f);
vf = f->index;
idT = DddmpReadNodeIndexBdd (T);
if (Cudd_IsConstant(T)) {
vT = nVars;
} else {
vT = T->index;
}
idE = DddmpReadNodeIndexBdd (E);
if (Cudd_IsConstant(E)) {
vE = nVars;
} else {
vE = E->index;
}
}
switch (mode) {
case DDDMP_MODE_TEXT:
retValue = NodeTextStoreBdd (ddMgr, f, mode, supportids, varnames,
outids, fp, idf, vf, idT, idE);
break;
case DDDMP_MODE_BINARY:
retValue = NodeBinaryStoreBdd (ddMgr, f, mode, supportids, varnames,
outids, fp, idf, vf, idT, idE, vT, vE, T, E);
break;
default:
return (DDDMP_FAILURE);
}
return (retValue);
}
/**Function********************************************************************
Synopsis [Store One Single Node in Text Format.]
Description [Store 1 0 0 for the terminal node.
Store id, left child pointer, right pointer for all the other nodes.
]
SideEffects [None]
SeeAlso [NodeBinaryStoreBdd]
******************************************************************************/
static int
NodeTextStoreBdd (
DdManager *ddMgr /* IN: DD Manager */,
DdNode *f /* IN: DD node to be stored */,
int mode /* IN: store mode */,
int *supportids /* IN: internal ids for variables */,
char **varnames /* IN: names of variables: to be stored with nodes */,
int *outids /* IN: output ids for variables */,
FILE *fp /* IN: Store file */,
int idf /* IN: index of the current node */,
int vf /* IN: variable of the current node */,
int idT /* IN: index of the Then node */,
int idE /* IN: index of the Else node */
)
{
int retValue = EOF;
(void) mode; /* avoid warning */
/*
* Check for Constant
*/
if (Cudd_IsConstant(f)) {
if (f == Cudd_ReadOne(ddMgr)) {
if ((varnames != NULL) || (outids != NULL)) {
retValue = fprintf (fp, "%d T 1 0 0\n", idf);
} else {
retValue = fprintf (fp, "%d 1 0 0\n", idf);
}
if (retValue == EOF) {
return (DDDMP_FAILURE);
} else {
return (DDDMP_SUCCESS);
}
}
if (f == Cudd_ReadZero(ddMgr)) {
if ((varnames != NULL) || (outids != NULL)) {
retValue = fprintf (fp, "%d T 0 0 0\n", idf);
} else {
retValue = fprintf (fp, "%d 0 0 0\n", idf);
}
if (retValue == EOF) {
return (DDDMP_FAILURE);
} else {
return (DDDMP_SUCCESS);
}
}
/*
* A constant node different from 1: an ADD constant
*/
Dddmp_CheckAndReturn (((varnames!=NULL)||(outids!=NULL)),
"Error writing to file: ADD Type.");
if (retValue == EOF) {
return (DDDMP_FAILURE);
} else {
return (DDDMP_SUCCESS);
}
}
/*
* ... Not A Constant
*/
if (Cudd_IsComplement (cuddE(f))) {
idE = -idE;
}
if (varnames != NULL) {
retValue = fprintf (fp, "%d %s %d %d %d\n",
idf, varnames[vf], supportids[vf], idT, idE);
if (retValue == EOF) {
return (DDDMP_FAILURE);
} else {
return (DDDMP_SUCCESS);
}
}
if (outids != NULL) {
retValue = fprintf (fp, "%d %d %d %d %d\n",
idf, outids[vf], supportids[vf], idT, idE);
if (retValue == EOF) {
return (DDDMP_FAILURE);
} else {
return (DDDMP_SUCCESS);
}
}
retValue = fprintf (fp, "%d %d %d %d\n",
idf, supportids[vf], idT, idE);
if (retValue == EOF) {
return (DDDMP_FAILURE);
} else {
return (DDDMP_SUCCESS);
}
}
/**Function********************************************************************
Synopsis [Store One Single Node in Binary Format.]
Description [Store 1 0 0 for the terminal node.
Store id, left child pointer, right pointer for all the other nodes.
Store every information as coded binary values.]
SideEffects [None]
SeeAlso [NodeTextStoreBdd]
******************************************************************************/
static int
NodeBinaryStoreBdd (
DdManager *ddMgr /* IN: DD Manager */,
DdNode *f /* IN: DD node to be stored */,
int mode /* IN: store mode */,
int *supportids /* IN: internal ids for variables */,
char **varnames /* IN: names of variables: to be stored with nodes */,
int *outids /* IN: output ids for variables */,
FILE *fp /* IN: store file */,
int idf /* IN: index of the node */,
int vf /* IN: variable of the node */,
int idT /* IN: index of the Then node */,
int idE /* IN: index of the Else node */,
int vT /* IN: variable of the Then node */,
int vE /* IN: variable of the Else node */,
DdNode *T /* IN: Then node */,
DdNode *E /* IN: Else node */
)
{
int retValue, diff, var;
struct binary_dd_code code;
(void) mode; /* avoid warning */
(void) varnames; /* avoid warning */
(void) outids; /* avoid warning */
/*
* Check for Constant
*/
/* only integer ids used, varnames ignored */
/* Terminal one is coded as DDDMP_TERMINAL, all other fields are 0 */
if (Cudd_IsConstant(f)) {
code.Unused = 0;
code.V = DDDMP_TERMINAL;
code.T = 0;
code.E = 0;
code.Ecompl = 0;
retValue = DddmpWriteCode (fp,code);
if (retValue == EOF) {
return (DDDMP_FAILURE);
} else {
return (DDDMP_SUCCESS);
}
}
/*
* Non terminal: output variable id
*/
var = supportids[vf];
diff = (supportids[vT]<supportids[vE]) ?
(supportids[vT]-var) : (supportids[vE]-var);
code.V = DDDMP_ABSOLUTE_ID;
if (diff <= var) {
if (diff == 1) {
code.V = DDDMP_RELATIVE_1;
} else {
code.V = DDDMP_RELATIVE_ID;
var = diff;
}
}
if (T == DD_ONE(ddMgr)) {
code.T = DDDMP_TERMINAL;
} else {
/* compute displacement */
diff = idf - idT;
code.T = DDDMP_ABSOLUTE_ID;
if (diff <= idT) {
if (diff == 1) {
code.T = DDDMP_RELATIVE_1;
} else {
code.T = DDDMP_RELATIVE_ID;
idT = diff;
}
}
}
if (E == DD_ONE(ddMgr)) {
code.E = DDDMP_TERMINAL;
} else {
/* compute displacement */
diff = idf - idE;
code.E = DDDMP_ABSOLUTE_ID;
if (diff <= idE) {
if (diff == 1) {
code.E = DDDMP_RELATIVE_1;
} else {
code.E = DDDMP_RELATIVE_ID;
idE = diff;
}
}
}
if (Cudd_IsComplement(cuddE(f))) {
code.Ecompl = 1;
} else {
code.Ecompl = 0;
}
retValue = DddmpWriteCode (fp,code);
if (retValue == EOF) {
return (DDDMP_FAILURE);
}
if ((code.V == DDDMP_ABSOLUTE_ID) || (code.V == DDDMP_RELATIVE_ID)) {
retValue = DddmpWriteInt (fp, var);
if (retValue == EOF) {
return (DDDMP_FAILURE);
}
}
if ((code.T == DDDMP_ABSOLUTE_ID) || (code.T == DDDMP_RELATIVE_ID)) {
retValue = DddmpWriteInt(fp,idT);
if (retValue == EOF) {
return (DDDMP_FAILURE);
}
}
if ((code.E == DDDMP_ABSOLUTE_ID) || (code.E == DDDMP_RELATIVE_ID)) {
retValue = DddmpWriteInt(fp,idE);
if (retValue == EOF) {
return (DDDMP_FAILURE);
}
}
return (retValue);
}