improve cudd with bdd printing -> nodes, and true=1 false=0

This commit is contained in:
Vítor Santos Costa 2015-03-04 09:50:15 +00:00
parent 3d9006db32
commit 4386c42d02
2 changed files with 109 additions and 26 deletions

View File

@ -28,8 +28,9 @@ The following predicates construct a BDD:
mtbdd_eval/2, mtbdd_eval/2,
bdd_tree/2, bdd_tree/2,
bdd_size/2, bdd_size/2,
bdd_print/2, bdd_print/2,
bdd_to_probability_sum_product/2, bdd_print/3,
bdd_to_probability_sum_product/2,
bdd_to_probability_sum_product/3, bdd_to_probability_sum_product/3,
bdd_close/1, bdd_close/1,
mtbdd_close/1]). mtbdd_close/1]).
@ -55,7 +56,7 @@ may include:
+ Logical Variables: + Logical Variables:
a leaf-node can be a logical variable. a leaf-node can be a logical variable.
+ `0` and `1` + `0` and `1`
@ -359,6 +360,22 @@ bdd_print(cudd(M,Top,_Vars, _), File) :-
bdd_print(add(M,Top,_Vars, _), File) :- bdd_print(add(M,Top,_Vars, _), File) :-
cudd_print(M, Top, File). cudd_print(M, Top, File).
bdd_print(cudd(M,Top, Vars, _), File, Names) :-
Vars =.. [_|LVars],
%trace,
maplist( fetch_name(Names), LVars, Ss),
cudd_print(M, Top, File, Ss).
bdd_print(add(M,Top, Vars, _), File, Names) :-
Vars =.. [_|LVars],
maplist( fetch_name(Names), LVars, Ss),
cudd_print(M, Top, File, Ss).
fetch_name([S=V1|_], V2, SN) :- V1 == V2, !,
( atom(S) -> SN = S ; format(atom(SN), '~w', [S]) ).
fetch_name([_|Y], V, S) :- !,
fetch_name(Y, V, S).
fetch_name([], V, V).
mtbdd_close(add(M,_,_Vars,_)) :- mtbdd_close(add(M,_,_Vars,_)) :-
cudd_die(M). cudd_die(M).

View File

@ -1,5 +1,5 @@
/** /**
@defgroup CUDD CUDD Interface @defgroup CUDD CUDD Interface
@ingroup BDDs @ingroup BDDs
@brief Interface to the CUDD Library @brief Interface to the CUDD Library
@ -67,6 +67,7 @@ static YAP_Functor FunctorDollarVar,
FunctorNand, FunctorNand,
FunctorNor, FunctorNor,
FunctorTimes, FunctorTimes,
FunctorImplies,
FunctorPlus, FunctorPlus,
FunctorMinus, FunctorMinus,
FunctorTimes4, FunctorTimes4,
@ -75,7 +76,7 @@ static YAP_Functor FunctorDollarVar,
FunctorOutPos, FunctorOutPos,
FunctorOutNeg; FunctorOutNeg;
static YAP_Term TermMinusOne, TermPlusOne; static YAP_Term TermMinusOne, TermZero, TermPlusOne, TermTrue, TermFalse;
void init_cudd(void); void init_cudd(void);
@ -254,6 +255,15 @@ term_to_cudd(DdManager *manager, YAP_Term t)
YAP_Error(DOMAIN_ERROR_OUT_OF_RANGE, t, "unsupported number in CUDD"); YAP_Error(DOMAIN_ERROR_OUT_OF_RANGE, t, "unsupported number in CUDD");
return NULL; return NULL;
} }
} else if (YAP_IsAtomTerm(t)) {
if (t == TermFalse)
return Cudd_ReadLogicZero(manager);
else if (t == TermTrue)
return Cudd_ReadOne(manager);
else {
YAP_Error(DOMAIN_ERROR_OUT_OF_RANGE, t, "unsupported atom %s in CUDD", YAP_AtomName(YAP_AtomOfTerm(t)));
return NULL;
}
} else if (YAP_IsVarTerm(t)) { } else if (YAP_IsVarTerm(t)) {
YAP_Error(INSTANTIATION_ERROR, t, "unsupported unbound term in CUDD"); YAP_Error(INSTANTIATION_ERROR, t, "unsupported unbound term in CUDD");
return NULL; return NULL;
@ -293,6 +303,16 @@ add_times(DdManager *manager, DdNode *x1, DdNode *x2)
return tmp; return tmp;
} }
static DdNode *
add_implies(DdManager *manager, DdNode *x1, DdNode *x2)
{
DdNode *tmp;
tmp = Cudd_addApply(manager,Cudd_addLeq,x1,x2);
Cudd_Ref(tmp);
return tmp;
}
static DdNode * static DdNode *
add_plus(DdManager *manager, DdNode *x1, DdNode *x2) add_plus(DdManager *manager, DdNode *x1, DdNode *x2)
{ {
@ -379,6 +399,14 @@ term_to_add(DdManager *manager, YAP_Term t)
DdNode *x2 = term_to_add(manager, YAP_ArgOfTerm(2, t)); DdNode *x2 = term_to_add(manager, YAP_ArgOfTerm(2, t));
DdNode *tmp = add_minus(manager, x1, x2); DdNode *tmp = add_minus(manager, x1, x2);
Cudd_RecursiveDeref(manager,x1);
Cudd_RecursiveDeref(manager,x2);
return tmp;
} else if (f == FunctorImplies) {
DdNode *x1 = term_to_add(manager, YAP_ArgOfTerm(1, t));
DdNode *x2 = term_to_add(manager, YAP_ArgOfTerm(2, t));
DdNode *tmp = add_implies(manager, x1, x2);
Cudd_RecursiveDeref(manager,x1); Cudd_RecursiveDeref(manager,x1);
Cudd_RecursiveDeref(manager,x2); Cudd_RecursiveDeref(manager,x2);
return tmp; return tmp;
@ -836,6 +864,40 @@ p_cudd_print(void)
return TRUE; return TRUE;
} }
static YAP_Bool
p_cudd_print_with_names(void)
{
DdManager *manager = (DdManager *)YAP_IntOfTerm(YAP_ARG1);
DdNode *n0 = (DdNode *)YAP_IntOfTerm(YAP_ARG2);
const char *s = YAP_AtomName(YAP_AtomOfTerm(YAP_ARG3));
const char **namesp;
YAP_Term names = YAP_ARG4;
FILE *f;
YAP_Int len;
if (!strcmp(s, "user_output")) f = stdout;
else if (!strcmp(s, "user_error")) f = stderr;
else if (!strcmp(s, "user")) f = stdout;
else f = fopen(s, "w");
if ((len = YAP_ListLength(names)) < 0)
return FALSE;
if ((namesp = malloc(sizeof(const char *)*len)) == NULL)
return FALSE;
while (YAP_IsPairTerm(names)) {
YAP_Int i = 0;
YAP_Term hd = YAP_HeadOfTerm( names);
char * s = YAP_AtomName(YAP_AtomOfTerm(hd));
const char *ns = malloc(strlen(s)+1);
strncpy(ns, s, strlen(s)+1);
namesp[i++] = ns;
names = YAP_TailOfTerm( names);
}
Cudd_DumpDot(manager, 1, &n0, (char **)namesp, NULL, f);
free( namesp );
if (f != stdout && f != stderr)
fclose(f);
return TRUE;
}
static YAP_Bool static YAP_Bool
p_cudd_die(void) p_cudd_die(void)
{ {
@ -872,6 +934,7 @@ init_cudd(void)
FunctorMinus = YAP_MkFunctor(YAP_LookupAtom("-"), 2); FunctorMinus = YAP_MkFunctor(YAP_LookupAtom("-"), 2);
FunctorTimes4 = YAP_MkFunctor(YAP_LookupAtom("*"), 4); FunctorTimes4 = YAP_MkFunctor(YAP_LookupAtom("*"), 4);
FunctorPlus4 = YAP_MkFunctor(YAP_LookupAtom("+"), 4); FunctorPlus4 = YAP_MkFunctor(YAP_LookupAtom("+"), 4);
FunctorImplies = YAP_MkFunctor(YAP_LookupAtom("->"), 2);
FunctorNot = YAP_MkFunctor(YAP_LookupAtom("not"), 1); FunctorNot = YAP_MkFunctor(YAP_LookupAtom("not"), 1);
FunctorMinus1 = YAP_MkFunctor(YAP_LookupAtom("-"), 1); FunctorMinus1 = YAP_MkFunctor(YAP_LookupAtom("-"), 1);
FunctorOutPos = YAP_MkFunctor(YAP_LookupAtom("pp"), 4); FunctorOutPos = YAP_MkFunctor(YAP_LookupAtom("pp"), 4);
@ -879,7 +942,9 @@ init_cudd(void)
FunctorOutAdd = YAP_MkFunctor(YAP_LookupAtom("add"), 4); FunctorOutAdd = YAP_MkFunctor(YAP_LookupAtom("add"), 4);
FunctorCudd = YAP_MkFunctor(YAP_LookupAtom("cudd"), 1); FunctorCudd = YAP_MkFunctor(YAP_LookupAtom("cudd"), 1);
TermMinusOne = YAP_MkIntTerm(-1); TermMinusOne = YAP_MkIntTerm(-1);
TermPlusOne = YAP_MkIntTerm(-1); TermPlusOne = YAP_MkIntTerm(+1);
TermFalse = YAP_MkAtomTerm(YAP_LookupAtom("false"));
TermTrue = YAP_MkAtomTerm(YAP_LookupAtom("true"));
YAP_UserCPredicate("term_to_cudd", p_term_to_cudd, 3); YAP_UserCPredicate("term_to_cudd", p_term_to_cudd, 3);
YAP_UserCPredicate("term_to_add", p_term_to_add, 4); YAP_UserCPredicate("term_to_add", p_term_to_add, 4);
YAP_UserCPredicate("cudd_eval", p_eval_cudd, 4); YAP_UserCPredicate("cudd_eval", p_eval_cudd, 4);
@ -891,6 +956,7 @@ init_cudd(void)
YAP_UserCPredicate("cudd_die", p_cudd_die, 1); YAP_UserCPredicate("cudd_die", p_cudd_die, 1);
YAP_UserCPredicate("cudd_release_node", p_cudd_release_node, 2); YAP_UserCPredicate("cudd_release_node", p_cudd_release_node, 2);
YAP_UserCPredicate("cudd_print", p_cudd_print, 3); YAP_UserCPredicate("cudd_print", p_cudd_print, 3);
YAP_UserCPredicate("cudd_print", p_cudd_print_with_names, 4);
} }
/** /**