Commit fc4ab6bd by Alan Mishchenko

Started experiments with a new solver.

parent 0cfc97b9
...@@ -118,15 +118,6 @@ static inline lit clause_read_lit (clause* c) { return (lit)((ABC_PTRUINT_T) ...@@ -118,15 +118,6 @@ static inline lit clause_read_lit (clause* c) { return (lit)((ABC_PTRUINT_T)
static inline int sat_solver_dlevel(sat_solver* s) { return veci_size(&s->trail_lim); } static inline int sat_solver_dlevel(sat_solver* s) { return veci_size(&s->trail_lim); }
static inline vecp* sat_solver_read_wlist(sat_solver* s, lit l) { return &s->wlists[l]; } static inline vecp* sat_solver_read_wlist(sat_solver* s, lit l) { return &s->wlists[l]; }
static inline void vecp_remove(vecp* v, void* e)
{
void** ws = vecp_begin(v);
int j = 0;
for (; ws[j] != e ; j++);
assert(j < vecp_size(v));
for (; j < vecp_size(v)-1; j++) ws[j] = ws[j+1];
vecp_resize(v,vecp_size(v)-1);
}
//================================================================================================= //=================================================================================================
// Variable order functions: // Variable order functions:
......
...@@ -30,7 +30,7 @@ ABC_NAMESPACE_IMPL_START ...@@ -30,7 +30,7 @@ ABC_NAMESPACE_IMPL_START
#define SAT_USE_ANALYZE_FINAL #define SAT_USE_ANALYZE_FINAL
#define SAT_USE_WATCH_ARRAYS //#define SAT_USE_WATCH_ARRAYS
//================================================================================================= //=================================================================================================
// Debug: // Debug:
...@@ -73,38 +73,69 @@ static inline int irand(double* seed, int size) { ...@@ -73,38 +73,69 @@ static inline int irand(double* seed, int size) {
static void sat_solver2_sort(void** array, int size, int(*comp)(const void *, const void *)); static void sat_solver2_sort(void** array, int size, int(*comp)(const void *, const void *));
//================================================================================================= //=================================================================================================
// Variable datatype + minor functions:
static const int var0 = 1;
static const int var1 = 0;
static const int varX = 3;
struct varinfo_t
{
unsigned val : 2;
unsigned pol : 1;
unsigned mark : 1;
unsigned tag : 3;
unsigned lev : 25;
};
static inline int var_value (sat_solver2 * s, int v) { return s->vi[v].val; }
static inline int var_polar (sat_solver2 * s, int v) { return s->vi[v].pol; }
static inline int var_tag (sat_solver2 * s, int v) { return s->vi[v].tag; }
static inline int var_mark (sat_solver2 * s, int v) { return s->vi[v].mark; }
static inline int var_level (sat_solver2 * s, int v) { return s->vi[v].lev; }
static inline void var_set_value (sat_solver2 * s, int v, int val ) { s->vi[v].val = val; }
static inline void var_set_polar (sat_solver2 * s, int v, int pol ) { s->vi[v].pol = pol; }
static inline void var_set_tag (sat_solver2 * s, int v, int tag ) { s->vi[v].tag = tag; }
static inline void var_set_mark (sat_solver2 * s, int v, int mark) { s->vi[v].mark = mark; }
static inline void var_set_level (sat_solver2 * s, int v, int lev ) { s->vi[v].lev = lev; }
//=================================================================================================
// Clause datatype + minor functions: // Clause datatype + minor functions:
struct clause_t struct clause_t // should have odd number of entries!
{ {
int size_learnt; unsigned learnt : 1;
unsigned fMark : 1;
unsigned fPartA : 1;
unsigned fRefed : 1;
unsigned nLits : 28;
unsigned act; unsigned act;
unsigned proof;
#ifndef SAT_USE_WATCH_ARRAYS #ifndef SAT_USE_WATCH_ARRAYS
clause * pNext[2]; int pNext[2];
#endif #endif
lit lits[0]; lit lits[0];
}; };
static inline int clause_size (clause* c) { return c->size_learnt >> 1; } static inline lit* clause_begin (clause* c) { return c->lits; }
static inline int clause_learnt (clause* c) { return c->size_learnt & 1; } static inline int clause_learnt (clause* c) { return c->learnt; }
static inline lit* clause_begin (clause* c) { return c->lits; } static inline int clause_nlits (clause* c) { return c->nLits; }
static inline int clause_size (int nLits) { return sizeof(clause)/4 + nLits + !(nLits & 1); }
static inline clause * get_clause (sat_solver2* s, int c) { return (clause *)(s->pMemArray + c); } static inline clause* get_clause (sat_solver2* s, int c) { return (clause *)(s->pMemArray + c); }
static inline int clause_id (sat_solver2* s, clause* c) { return ((int *)c - s->pMemArray); }
#define sat_solver_foreach_clause( s, c, i ) \
for ( i = 16; (i < s->nMemSize) && (((c) = get_clause(s, i)), 1); i += clause_size(c->nLits) )
#define sat_solver_foreach_learnt( s, c, i ) \
for ( i = s->iLearnt; (i < s->nMemSize) && (((c) = get_clause(s, i)), 1); i += clause_size(c->nLits) )
//================================================================================================= //=================================================================================================
// Simple helpers: // Simple helpers:
static inline int sat_solver2_dlevel(sat_solver2* s) { return veci_size(&s->trail_lim); } static inline int sat_solver2_dlevel(sat_solver2* s) { return veci_size(&s->trail_lim); }
static inline vecp* sat_solver2_read_wlist(sat_solver2* s, lit l) { return &s->wlists[l]; } static inline veci* sat_solver2_read_wlist(sat_solver2* s, lit l) { return &s->wlists[l]; }
static inline void vecp_remove(vecp* v, void* e)
{
void** ws = vecp_begin(v);
int j = 0;
for (; ws[j] != e ; j++);
assert(j < vecp_size(v));
for (; j < vecp_size(v)-1; j++) ws[j] = ws[j+1];
vecp_resize(v,vecp_size(v)-1);
}
//================================================================================================= //=================================================================================================
// Variable order functions: // Variable order functions:
...@@ -147,13 +178,14 @@ static inline int order_select(sat_solver2* s, float random_var_freq) // select ...@@ -147,13 +178,14 @@ static inline int order_select(sat_solver2* s, float random_var_freq) // select
{ {
int* heap = veci_begin(&s->order); int* heap = veci_begin(&s->order);
int* orderpos = s->orderpos; int* orderpos = s->orderpos;
lbool* values = s->assigns; // lbool* values = s->assigns;
// Random decision: // Random decision:
if (drand(&s->random_seed) < random_var_freq){ if (drand(&s->random_seed) < random_var_freq){
int next = irand(&s->random_seed,s->size); int next = irand(&s->random_seed,s->size);
assert(next >= 0 && next < s->size); assert(next >= 0 && next < s->size);
if (values[next] == l_Undef) // if (values[next] == l_Undef)
if (var_value(s, next) == varX)
return next; return next;
} }
...@@ -191,7 +223,8 @@ static inline int order_select(sat_solver2* s, float random_var_freq) // select ...@@ -191,7 +223,8 @@ static inline int order_select(sat_solver2* s, float random_var_freq) // select
} }
//printf( "-%d ", next ); //printf( "-%d ", next );
if (values[next] == l_Undef) // if (values[next] == l_Undef)
if (var_value(s, next) == varX)
return next; return next;
} }
...@@ -228,16 +261,14 @@ static inline void act_var_bump(sat_solver2* s, int v) { ...@@ -228,16 +261,14 @@ static inline void act_var_bump(sat_solver2* s, int v) {
static inline void act_var_decay(sat_solver2* s) { s->var_inc += (s->var_inc >> 4); } static inline void act_var_decay(sat_solver2* s) { s->var_inc += (s->var_inc >> 4); }
static inline void act_clause_rescale(sat_solver2* s) { static inline void act_clause_rescale(sat_solver2* s) {
clause** cs = (clause**)vecp_begin(&s->learnts); clause * c;
int i, clk = clock(); int i, clk = clock();
static int Total = 0; static int Total = 0;
sat_solver_foreach_learnt( s, c, i )
for (i = 0; i < vecp_size(&s->learnts); i++) c->act >>= 14;
cs[i]->act >>= 14;
s->cla_inc >>= 14; s->cla_inc >>= 14;
// assert( s->cla_inc > (1<<10)-1 ); // assert( s->cla_inc > (1<<10)-1 );
s->cla_inc = Abc_MaxInt( s->cla_inc, (1<<10) ); s->cla_inc = Abc_MaxInt( s->cla_inc, (1<<10) );
printf( "Rescaling... Cla inc = %5d Conf = %10d ", s->cla_inc, s->stats.conflicts ); printf( "Rescaling... Cla inc = %5d Conf = %10d ", s->cla_inc, s->stats.conflicts );
Total += clock() - clk; Total += clock() - clk;
Abc_PrintTime( 1, "Time", Total ); Abc_PrintTime( 1, "Time", Total );
...@@ -257,7 +288,7 @@ static inline void act_clause_decay(sat_solver2* s) { s->cla_inc += (s->cla_inc ...@@ -257,7 +288,7 @@ static inline void act_clause_decay(sat_solver2* s) { s->cla_inc += (s->cla_inc
// Clause functions: // Clause functions:
#ifndef SAT_USE_WATCH_ARRAYS #ifndef SAT_USE_WATCH_ARRAYS
static inline void clause_watch_( sat_solver2* s, clause* c, lit Lit ) static inline void clause_watch_front( sat_solver2* s, clause* c, lit Lit )
{ {
if ( c->lits[0] == Lit ) if ( c->lits[0] == Lit )
c->pNext[0] = s->pWatches[lit_neg(Lit)]; c->pNext[0] = s->pWatches[lit_neg(Lit)];
...@@ -266,53 +297,12 @@ static inline void clause_watch_( sat_solver2* s, clause* c, lit Lit ) ...@@ -266,53 +297,12 @@ static inline void clause_watch_( sat_solver2* s, clause* c, lit Lit )
assert( c->lits[1] == Lit ); assert( c->lits[1] == Lit );
c->pNext[1] = s->pWatches[lit_neg(Lit)]; c->pNext[1] = s->pWatches[lit_neg(Lit)];
} }
s->pWatches[lit_neg(Lit)] = c; s->pWatches[lit_neg(Lit)] = clause_id(s, c);
} }
static inline void clause_watch( sat_solver2* s, clause* c, lit Lit )
static inline void clause_watch__( sat_solver2* s, clause* c, lit Lit )
{
clause* pLast = s->pWatches[lit_neg(Lit)];
if ( c->lits[0] == Lit )
c->pNext[0] = NULL;
else
{
assert( c->lits[1] == Lit );
c->pNext[1] = NULL;
}
// add at the tail
if ( pLast == NULL )
{
s->pWatches[lit_neg(Lit)] = c;
return;
}
// find the last one
while ( 1 )
{
if ( pLast->lits[0] == Lit )
{
if ( pLast->pNext[0] == NULL )
{
pLast->pNext[0] = c;
return;
}
pLast = pLast->pNext[0];
}
else
{
assert( pLast->lits[1] == Lit );
if ( pLast->pNext[1] == NULL )
{
pLast->pNext[1] = c;
return;
}
pLast = pLast->pNext[1];
}
}
}
static inline void clause_watch33( sat_solver2* s, clause* c, lit Lit )
{ {
clause** ppList = s->pWatches + lit_neg(Lit),** ppNext; clause * pList;
cla * ppList = s->pWatches + lit_neg(Lit), * ppNext;
if ( c->lits[0] == Lit ) if ( c->lits[0] == Lit )
ppNext = &c->pNext[0]; ppNext = &c->pNext[0];
else else
...@@ -320,53 +310,59 @@ static inline void clause_watch33( sat_solver2* s, clause* c, lit Lit ) ...@@ -320,53 +310,59 @@ static inline void clause_watch33( sat_solver2* s, clause* c, lit Lit )
assert( c->lits[1] == Lit ); assert( c->lits[1] == Lit );
ppNext = &c->pNext[1]; ppNext = &c->pNext[1];
} }
if ( *ppList == NULL ) if ( *ppList == 0 )
{ {
*ppList = c; *ppList = clause_id(s, c);
*ppNext = c; *ppNext = clause_id(s, c);
return; return;
} }
// add at the tail // add at the tail
if ( (*ppList)->lits[0] == Lit ) pList = get_clause( s, *ppList );
if ( pList->lits[0] == Lit )
{ {
assert( (*ppList)->pNext[0] != NULL ); assert( pList->pNext[0] != 0 );
*ppNext = (*ppList)->pNext[0]; *ppNext = pList->pNext[0];
(*ppList)->pNext[0] = c; pList->pNext[0] = clause_id(s, c);
} }
else else
{ {
assert( (*ppList)->lits[1] == Lit ); assert( pList->lits[1] == Lit );
assert( (*ppList)->pNext[1] != NULL ); assert( pList->pNext[1] != 0 );
*ppNext = (*ppList)->pNext[1]; *ppNext = pList->pNext[1];
(*ppList)->pNext[1] = c; pList->pNext[1] = clause_id(s, c);
} }
*ppList = c; *ppList = clause_id(s, c);
} }
/*
static inline void clause_watch( sat_solver2* s, clause* c, lit Lit ) static inline void clause_watch( sat_solver2* s, clause* c, lit Lit )
{ {
clause * pList = s->pWatches[lit_neg(Lit)]; int * ppList = s->pWatches[lit_neg(Lit)];
assert( c->lits[0] == Lit || c->lits[1] == Lit ); assert( c->lits[0] == Lit || c->lits[1] == Lit );
if ( pList == NULL ) if ( *ppList == NULL )
c->pNext[c->lits[1] == Lit] = c; c->pNext[c->lits[1] == Lit] = clause_id(s, c);
else else
{ {
clause * pList = get_clause( s, *ppList );
assert( pList->lits[0] == Lit || pList->lits[1] == Lit ); assert( pList->lits[0] == Lit || pList->lits[1] == Lit );
c->pNext[c->lits[1] == Lit] = pList->pNext[pList->lits[1] == Lit]; c->pNext[c->lits[1] == Lit] = pList->pNext[pList->lits[1] == Lit];
pList->pNext[pList->lits[1] == Lit] = c; pList->pNext[pList->lits[1] == Lit] = clause_id(s, c);
} }
s->pWatches[lit_neg(Lit)] = c; s->pWatches[lit_neg(Lit)] = clause_id(s, c);
} }
*/
#endif #endif
static clause* clause_new(sat_solver2* s, lit* begin, lit* end, int learnt) static int clause_new(sat_solver2* s, lit* begin, lit* end, int learnt)
{ {
clause* c; clause* c;
int i, size = end - begin; int i, Cid, nLits = end - begin;
assert(size > 1); assert(nLits > 1);
assert(begin[0] >= 0);
assert(begin[1] >= 0);
assert(lit_var(begin[0]) < s->size);
assert(lit_var(begin[1]) < s->size);
// add memory if needed // add memory if needed
if ( s->nMemSize + (int)sizeof(clause)/4 + size > s->nMemAlloc ) if ( s->nMemSize + clause_size(nLits) > s->nMemAlloc )
{ {
int nMemAlloc = s->nMemAlloc ? s->nMemAlloc * 2 : (1 << 24); int nMemAlloc = s->nMemAlloc ? s->nMemAlloc * 2 : (1 << 24);
s->pMemArray = ABC_REALLOC( int, s->pMemArray, nMemAlloc ); s->pMemArray = ABC_REALLOC( int, s->pMemArray, nMemAlloc );
...@@ -377,32 +373,33 @@ static clause* clause_new(sat_solver2* s, lit* begin, lit* end, int learnt) ...@@ -377,32 +373,33 @@ static clause* clause_new(sat_solver2* s, lit* begin, lit* end, int learnt)
} }
// create clause // create clause
c = (clause*)(s->pMemArray + s->nMemSize); c = (clause*)(s->pMemArray + s->nMemSize);
s->nMemSize += sizeof(clause)/4 + size; c->learnt = learnt;
c->nLits = nLits;
for (i = 0; i < nLits; i++)
c->lits[i] = begin[i];
// extend storage
Cid = s->nMemSize;
s->nMemSize += clause_size(nLits);
if ( s->nMemSize > s->nMemAlloc ) if ( s->nMemSize > s->nMemAlloc )
printf( "Out of memory!!!\n" ); printf( "Out of memory!!!\n" );
assert( s->nMemSize <= s->nMemAlloc ); assert( s->nMemSize <= s->nMemAlloc );
c->size_learnt = (size << 1) | learnt; assert(((ABC_PTRUINT_T)c & 7) == 0);
assert(((ABC_PTRUINT_T)c & 1) == 0);
c->act = 0;
for (i = 0; i < size; i++)
c->lits[i] = begin[i];
assert(begin[0] >= 0);
assert(begin[0] < s->size*2);
assert(begin[1] >= 0);
assert(begin[1] < s->size*2);
assert(lit_neg(begin[0]) < s->size*2);
assert(lit_neg(begin[1]) < s->size*2);
#ifdef SAT_USE_WATCH_ARRAYS #ifdef SAT_USE_WATCH_ARRAYS
vecp_push(sat_solver2_read_wlist(s,lit_neg(begin[0])),(void*)c); veci_push(sat_solver2_read_wlist(s,lit_neg(begin[0])),clause_id(s,c));
vecp_push(sat_solver2_read_wlist(s,lit_neg(begin[1])),(void*)c); veci_push(sat_solver2_read_wlist(s,lit_neg(begin[1])),clause_id(s,c));
#else #else
clause_watch( s, c, begin[0] ); clause_watch( s, c, begin[0] );
clause_watch( s, c, begin[1] ); clause_watch( s, c, begin[1] );
#endif #endif
return c;
// remember the last one and first learnt
s->iLast = Cid;
if ( learnt && s->iLearnt == -1 )
s->iLearnt = Cid;
return Cid;
} }
...@@ -413,16 +410,16 @@ static void clause_remove(sat_solver2* s, clause* c) ...@@ -413,16 +410,16 @@ static void clause_remove(sat_solver2* s, clause* c)
assert(lit_neg(lits[1]) < s->size*2); assert(lit_neg(lits[1]) < s->size*2);
#ifdef SAT_USE_WATCH_ARRAYS #ifdef SAT_USE_WATCH_ARRAYS
vecp_remove(sat_solver2_read_wlist(s,lit_neg(lits[0])),(void*)c); veci_remove(sat_solver2_read_wlist(s,lit_neg(lits[0])),clause_id(s,c));
vecp_remove(sat_solver2_read_wlist(s,lit_neg(lits[1])),(void*)c); veci_remove(sat_solver2_read_wlist(s,lit_neg(lits[1])),clause_id(s,c));
#endif #endif
if (clause_learnt(c)){ if (clause_learnt(c)){
s->stats.learnts--; s->stats.learnts--;
s->stats.learnts_literals -= clause_size(c); s->stats.learnts_literals -= clause_nlits(c);
}else{ }else{
s->stats.clauses--; s->stats.clauses--;
s->stats.clauses_literals -= clause_size(c); s->stats.clauses_literals -= clause_nlits(c);
} }
} }
...@@ -430,14 +427,15 @@ static void clause_remove(sat_solver2* s, clause* c) ...@@ -430,14 +427,15 @@ static void clause_remove(sat_solver2* s, clause* c)
static lbool clause_simplify(sat_solver2* s, clause* c) static lbool clause_simplify(sat_solver2* s, clause* c)
{ {
lit* lits = clause_begin(c); lit* lits = clause_begin(c);
lbool* values = s->assigns; // lbool* values = s->assigns;
int i; int i;
assert(sat_solver2_dlevel(s) == 0); assert(sat_solver2_dlevel(s) == 0);
for (i = 0; i < clause_size(c); i++){ for (i = 0; i < clause_nlits(c); i++){
lbool sig = !lit_sign(lits[i]); sig += sig - 1; // lbool sig = !lit_sign(lits[i]); sig += sig - 1;
if (values[lit_var(lits[i])] == sig) // if (values[lit_var(lits[i])] == sig)
if (var_value(s, lit_var(lits[i])) == lit_sign(lits[i]))
return l_True; return l_True;
} }
return l_False; return l_False;
...@@ -455,40 +453,32 @@ void sat_solver2_setnvars(sat_solver2* s,int n) ...@@ -455,40 +453,32 @@ void sat_solver2_setnvars(sat_solver2* s,int n)
while (s->cap < n) s->cap = s->cap*2+1; while (s->cap < n) s->cap = s->cap*2+1;
#ifdef SAT_USE_WATCH_ARRAYS #ifdef SAT_USE_WATCH_ARRAYS
s->wlists = ABC_REALLOC(vecp, s->wlists, s->cap*2); s->wlists = ABC_REALLOC(veci, s->wlists, s->cap*2);
#else #else
s->pWatches = ABC_REALLOC(clause*,s->pWatches, s->cap*2); s->pWatches = ABC_REALLOC(cla, s->pWatches, s->cap*2);
#endif #endif
s->vi = ABC_REALLOC(varinfo, s->vi, s->cap);
s->activity = ABC_REALLOC(unsigned, s->activity, s->cap); s->activity = ABC_REALLOC(unsigned, s->activity, s->cap);
s->assigns = ABC_REALLOC(lbool, s->assigns, s->cap); s->orderpos = ABC_REALLOC(int, s->orderpos, s->cap);
s->orderpos = ABC_REALLOC(int, s->orderpos, s->cap); s->reasons = ABC_REALLOC(cla, s->reasons, s->cap);
s->reasons = ABC_REALLOC(clause*,s->reasons, s->cap); s->trail = ABC_REALLOC(lit, s->trail, s->cap);
s->levels = ABC_REALLOC(int, s->levels, s->cap);
s->tags = ABC_REALLOC(lbool, s->tags, s->cap);
s->trail = ABC_REALLOC(lit, s->trail, s->cap);
s->polarity = ABC_REALLOC(char, s->polarity, s->cap);
} }
for (var = s->size; var < n; var++){ for (var = s->size; var < n; var++){
#ifdef SAT_USE_WATCH_ARRAYS #ifdef SAT_USE_WATCH_ARRAYS
vecp_new(&s->wlists[2*var]); veci_new(&s->wlists[2*var]);
vecp_new(&s->wlists[2*var+1]); veci_new(&s->wlists[2*var+1]);
#else #else
s->pWatches[2*var] = NULL; s->pWatches[2*var] = 0;
s->pWatches[2*var+1] = NULL; s->pWatches[2*var+1] = 0;
#endif #endif
s->activity [var] = (1<<10); *((int*)s->vi + var) = 0; s->vi[var].val = varX;
s->assigns [var] = l_Undef; s->activity [var] = (1<<10);
s->orderpos [var] = veci_size(&s->order); s->orderpos [var] = veci_size(&s->order);
s->reasons [var] = (clause*)0; s->reasons [var] = 0;
s->levels [var] = 0;
s->tags [var] = l_Undef;
s->polarity [var] = 0;
/* does not hold because variables enqueued at top level will not be reinserted in the heap // does not hold because variables enqueued at top level will not be reinserted in the heap
assert(veci_size(&s->order) == var); // assert(veci_size(&s->order) == var);
*/
veci_push(&s->order,var); veci_push(&s->order,var);
order_update(s, var); order_update(s, var);
} }
...@@ -497,77 +487,59 @@ void sat_solver2_setnvars(sat_solver2* s,int n) ...@@ -497,77 +487,59 @@ void sat_solver2_setnvars(sat_solver2* s,int n)
} }
static inline int enqueue(sat_solver2* s, lit l, clause* from) static inline int enqueue(sat_solver2* s, lit l, int from)
{ {
lbool* values = s->assigns;
int v = lit_var(l); int v = lit_var(l);
lbool val = values[v];
#ifdef VERBOSEDEBUG #ifdef VERBOSEDEBUG
printf(L_IND"enqueue("L_LIT")\n", L_ind, L_lit(l)); printf(L_IND"enqueue("L_LIT")\n", L_ind, L_lit(l));
#endif #endif
if (var_value(s, v) != varX)
lbool sig = !lit_sign(l); sig += sig - 1; return var_value(s, v) == lit_sign(l);
if (val != l_Undef){ else
return val == sig; { // New fact -- store it.
}else{
// New fact -- store it.
#ifdef VERBOSEDEBUG #ifdef VERBOSEDEBUG
printf(L_IND"bind("L_LIT")\n", L_ind, L_lit(l)); printf(L_IND"bind("L_LIT")\n", L_ind, L_lit(l));
#endif #endif
int* levels = s->levels; var_set_value( s, v, lit_sign(l) );
clause** reasons = s->reasons; var_set_level( s, v, sat_solver2_dlevel(s) );
s->reasons[v] = from;
values [v] = sig;
levels [v] = sat_solver2_dlevel(s);
reasons[v] = from;
s->trail[s->qtail++] = l; s->trail[s->qtail++] = l;
order_assigned(s, v); order_assigned(s, v);
return true; return true;
} }
} }
static inline int assume(sat_solver2* s, lit l)
static inline int assume(sat_solver2* s, lit l){ {
assert(s->qtail == s->qhead); assert(s->qtail == s->qhead);
assert(s->assigns[lit_var(l)] == l_Undef); assert(var_value(s, lit_var(l)) == varX);
#ifdef VERBOSEDEBUG #ifdef VERBOSEDEBUG
printf(L_IND"assume("L_LIT")\n", L_ind, L_lit(l)); printf(L_IND"assume("L_LIT")\n", L_ind, L_lit(l));
#endif #endif
veci_push(&s->trail_lim,s->qtail); veci_push(&s->trail_lim,s->qtail);
return enqueue(s,l,(clause*)0); return enqueue(s,l,0);
} }
static void sat_solver2_canceluntil(sat_solver2* s, int level) { static void sat_solver2_canceluntil(sat_solver2* s, int level) {
lit* trail; lit* trail;
lbool* values;
clause** reasons;
int bound; int bound;
int lastLev; int lastLev;
int c; int c, x;
if (sat_solver2_dlevel(s) <= level) if (sat_solver2_dlevel(s) <= level)
return; return;
trail = s->trail; trail = s->trail;
values = s->assigns;
reasons = s->reasons;
bound = (veci_begin(&s->trail_lim))[level]; bound = (veci_begin(&s->trail_lim))[level];
lastLev = (veci_begin(&s->trail_lim))[veci_size(&s->trail_lim)-1]; lastLev = (veci_begin(&s->trail_lim))[veci_size(&s->trail_lim)-1];
//////////////////////////////////////// for (c = s->qtail-1; c >= bound; c--)
// added to cancel all assignments {
// if ( level == -1 ) x = lit_var(trail[c]);
// bound = 0; var_set_value(s, x, varX);
//////////////////////////////////////// s->reasons[x] = 0;
for (c = s->qtail-1; c >= bound; c--) {
int x = lit_var(trail[c]);
values [x] = l_Undef;
reasons[x] = (clause*)0;
if ( c < lastLev ) if ( c < lastLev )
s->polarity[x] = !lit_sign(trail[c]); var_set_polar(s, x, !lit_sign(trail[c]));
} }
for (c = s->qhead-1; c >= bound; c--) for (c = s->qhead-1; c >= bound; c--)
...@@ -579,16 +551,15 @@ static void sat_solver2_canceluntil(sat_solver2* s, int level) { ...@@ -579,16 +551,15 @@ static void sat_solver2_canceluntil(sat_solver2* s, int level) {
static void sat_solver2_record(sat_solver2* s, veci* cls) static void sat_solver2_record(sat_solver2* s, veci* cls)
{ {
lit* begin = veci_begin(cls); lit* begin = veci_begin(cls);
lit* end = begin + veci_size(cls); lit* end = begin + veci_size(cls);
clause* c = (veci_size(cls) > 1) ? clause_new(s,begin,end,1) : (clause*)0; int c = (veci_size(cls) > 1) ? clause_new(s,begin,end,1) : 0;
enqueue(s,*begin,c); enqueue(s,*begin,c);
assert(veci_size(cls) > 0); assert(veci_size(cls) > 0);
if (c != 0) { if (c) {
vecp_push(&s->learnts,c); act_clause_bump(s,get_clause(s,c));
act_clause_bump(s,c);
s->stats.learnts++; s->stats.learnts++;
s->stats.learnts_literals += veci_size(cls); s->stats.learnts_literals += veci_size(cls);
} }
...@@ -597,15 +568,11 @@ static void sat_solver2_record(sat_solver2* s, veci* cls) ...@@ -597,15 +568,11 @@ static void sat_solver2_record(sat_solver2* s, veci* cls)
static double sat_solver2_progress(sat_solver2* s) static double sat_solver2_progress(sat_solver2* s)
{ {
lbool* values = s->assigns; int i;
int* levels = s->levels; double progress = 0.0, F = 1.0 / s->size;
int i;
double progress = 0;
double F = 1.0 / s->size;
for (i = 0; i < s->size; i++) for (i = 0; i < s->size; i++)
if (values[i] != l_Undef) if (var_value(s, i) != varX)
progress += pow(F, levels[i]); progress += pow(F, var_level(s, i));
return progress / s->size; return progress / s->size;
} }
...@@ -614,48 +581,38 @@ static double sat_solver2_progress(sat_solver2* s) ...@@ -614,48 +581,38 @@ static double sat_solver2_progress(sat_solver2* s)
static int sat_solver2_lit_removable(sat_solver2* s, lit l, int minl) static int sat_solver2_lit_removable(sat_solver2* s, lit l, int minl)
{ {
lbool* tags = s->tags; clause* c;
clause** reasons = s->reasons; lit* lits;
int* levels = s->levels; int i, j, v, top = veci_size(&s->tagged);
int top = veci_size(&s->tagged);
assert(lit_var(l) >= 0 && lit_var(l) < s->size); assert(lit_var(l) >= 0 && lit_var(l) < s->size);
assert(reasons[lit_var(l)] != 0); assert(s->reasons[lit_var(l)] != 0);
veci_resize(&s->stack,0); veci_resize(&s->stack,0);
veci_push(&s->stack,lit_var(l)); veci_push(&s->stack,lit_var(l));
while (veci_size(&s->stack) > 0)
while (veci_size(&s->stack) > 0){ {
clause* c; v = veci_begin(&s->stack)[veci_size(&s->stack)-1];
int v = veci_begin(&s->stack)[veci_size(&s->stack)-1];
assert(v >= 0 && v < s->size); assert(v >= 0 && v < s->size);
veci_resize(&s->stack,veci_size(&s->stack)-1); veci_resize(&s->stack,veci_size(&s->stack)-1);
assert(reasons[v] != 0); assert(s->reasons[v] != 0);
c = reasons[v]; c = get_clause(s, s->reasons[v]);
lits = clause_begin(c);
{ for (i = 1; i < clause_nlits(c); i++){
lit* lits = clause_begin(c);
int i, j;
for (i = 1; i < clause_size(c); i++){
int v = lit_var(lits[i]); int v = lit_var(lits[i]);
if (tags[v] == l_Undef && levels[v] != 0){ if (s->vi[v].tag == 0 && var_level(s,v)){
if (reasons[v] != 0 && ((1 << (levels[v] & 31)) & minl)){ if (s->reasons[v] != 0 && ((1 << (var_level(s,v) & 31)) & minl)){
veci_push(&s->stack,lit_var(lits[i])); veci_push(&s->stack,lit_var(lits[i]));
tags[v] = l_True; var_set_tag(s, v, 1);
veci_push(&s->tagged,v); veci_push(&s->tagged,v);
}else{ }else{
int* tagged = veci_begin(&s->tagged); int* tagged = veci_begin(&s->tagged);
for (j = top; j < veci_size(&s->tagged); j++) for (j = top; j < veci_size(&s->tagged); j++)
tags[tagged[j]] = l_Undef; var_set_tag(s, tagged[j], 0);
veci_resize(&s->tagged,top); veci_resize(&s->tagged,top);
return false; return false;
} }
} }
} }
}
} }
return true; return true;
} }
...@@ -713,19 +670,18 @@ void Solver::analyzeFinal(Clause* confl, bool skip_first) ...@@ -713,19 +670,18 @@ void Solver::analyzeFinal(Clause* confl, bool skip_first)
static void sat_solver2_analyze_final(sat_solver2* s, clause* conf, int skip_first) static void sat_solver2_analyze_final(sat_solver2* s, clause* conf, int skip_first)
{ {
int* tagged;
int i, j, start; int i, j, start;
veci_resize(&s->conf_final,0); veci_resize(&s->conf_final,0);
if ( s->root_level == 0 ) if ( s->root_level == 0 )
return; return;
assert( veci_size(&s->tagged) == 0 ); assert( veci_size(&s->tagged) == 0 );
// assert( s->tags[lit_var(p)] == l_Undef ); for (i = skip_first ? 1 : 0; i < clause_nlits(conf); i++)
// s->tags[lit_var(p)] = l_True;
for (i = skip_first ? 1 : 0; i < clause_size(conf); i++)
{ {
int x = lit_var(clause_begin(conf)[i]); int x = lit_var(clause_begin(conf)[i]);
if (s->levels[x] > 0) if ( var_level(s,x) )
{ {
s->tags[x] = l_True; var_set_tag(s, x, 1);
veci_push(&s->tagged,x); veci_push(&s->tagged,x);
} }
} }
...@@ -733,26 +689,25 @@ static void sat_solver2_analyze_final(sat_solver2* s, clause* conf, int skip_fir ...@@ -733,26 +689,25 @@ static void sat_solver2_analyze_final(sat_solver2* s, clause* conf, int skip_fir
start = (s->root_level >= veci_size(&s->trail_lim))? s->qtail-1 : (veci_begin(&s->trail_lim))[s->root_level]; start = (s->root_level >= veci_size(&s->trail_lim))? s->qtail-1 : (veci_begin(&s->trail_lim))[s->root_level];
for (i = start; i >= (veci_begin(&s->trail_lim))[0]; i--){ for (i = start; i >= (veci_begin(&s->trail_lim))[0]; i--){
int x = lit_var(s->trail[i]); int x = lit_var(s->trail[i]);
if (s->tags[x] == l_True){ if (s->vi[x].tag == 1){
if (s->reasons[x] == NULL){ if (s->reasons[x] == 0){
assert(s->levels[x] > 0); assert( var_level(s,x) );
veci_push(&s->conf_final,lit_neg(s->trail[i])); veci_push(&s->conf_final,lit_neg(s->trail[i]));
}else{ }else{
clause* c = s->reasons[x]; clause* c = get_clause(s, s->reasons[x]);
{ int* lits = clause_begin(c);
int* lits = clause_begin(c); for (j = 1; j < clause_nlits(c); j++)
for (j = 1; j < clause_size(c); j++) if ( var_level(s,lit_var(lits[j])) )
if (s->levels[lit_var(lits[j])] > 0) {
{ var_set_tag(s, lit_var(lits[j]), 1);
s->tags[lit_var(lits[j])] = l_True; veci_push(&s->tagged,lit_var(lits[j]));
veci_push(&s->tagged,lit_var(lits[j])); }
}
}
} }
} }
} }
tagged = veci_begin(&s->tagged);
for (i = 0; i < veci_size(&s->tagged); i++) for (i = 0; i < veci_size(&s->tagged); i++)
s->tags[veci_begin(&s->tagged)[i]] = l_Undef; var_set_tag(s, tagged[i], 0);
veci_resize(&s->tagged,0); veci_resize(&s->tagged,0);
} }
...@@ -762,15 +717,13 @@ static void sat_solver2_analyze_final(sat_solver2* s, clause* conf, int skip_fir ...@@ -762,15 +717,13 @@ static void sat_solver2_analyze_final(sat_solver2* s, clause* conf, int skip_fir
static void sat_solver2_analyze(sat_solver2* s, clause* c, veci* learnt) static void sat_solver2_analyze(sat_solver2* s, clause* c, veci* learnt)
{ {
lit* trail = s->trail; lit* trail = s->trail;
lbool* tags = s->tags;
clause** reasons = s->reasons;
int* levels = s->levels;
int cnt = 0; int cnt = 0;
lit p = lit_Undef; lit p = lit_Undef;
int ind = s->qtail-1; int ind = s->qtail-1;
lit* lits; lit* lits;
int i, j, minl; int i, j, minl;
int* tagged; int* tagged;
assert( veci_size(&s->tagged) == 0 );
veci_push(learnt,lit_Undef); veci_push(learnt,lit_Undef);
...@@ -780,25 +733,24 @@ static void sat_solver2_analyze(sat_solver2* s, clause* c, veci* learnt) ...@@ -780,25 +733,24 @@ static void sat_solver2_analyze(sat_solver2* s, clause* c, veci* learnt)
act_clause_bump(s,c); act_clause_bump(s,c);
lits = clause_begin(c); lits = clause_begin(c);
//printlits(lits,lits+clause_size(c)); printf("\n"); for (j = (p == lit_Undef ? 0 : 1); j < clause_nlits(c); j++){
for (j = (p == lit_Undef ? 0 : 1); j < clause_size(c); j++){
lit q = lits[j]; lit q = lits[j];
assert(lit_var(q) >= 0 && lit_var(q) < s->size); assert(lit_var(q) >= 0 && lit_var(q) < s->size);
if (tags[lit_var(q)] == l_Undef && levels[lit_var(q)] > 0){ if (!var_tag(s, lit_var(q)) && var_level(s,lit_var(q))){
tags[lit_var(q)] = l_True; var_set_tag(s, lit_var(q), 1);
veci_push(&s->tagged,lit_var(q)); veci_push(&s->tagged,lit_var(q));
act_var_bump(s,lit_var(q)); act_var_bump(s,lit_var(q));
if (levels[lit_var(q)] == sat_solver2_dlevel(s)) if (var_level(s,lit_var(q)) == sat_solver2_dlevel(s))
cnt++; cnt++;
else else
veci_push(learnt,q); veci_push(learnt,q);
} }
} }
while (tags[lit_var(trail[ind--])] == l_Undef); while (!var_tag(s, lit_var(trail[ind--])));
p = trail[ind+1]; p = trail[ind+1];
c = reasons[lit_var(p)]; c = get_clause(s, s->reasons[lit_var(p)]);
cnt--; cnt--;
}while (cnt > 0); }while (cnt > 0);
...@@ -807,14 +759,12 @@ static void sat_solver2_analyze(sat_solver2* s, clause* c, veci* learnt) ...@@ -807,14 +759,12 @@ static void sat_solver2_analyze(sat_solver2* s, clause* c, veci* learnt)
lits = veci_begin(learnt); lits = veci_begin(learnt);
minl = 0; minl = 0;
for (i = 1; i < veci_size(learnt); i++){ for (i = 1; i < veci_size(learnt); i++)
int lev = levels[lit_var(lits[i])]; minl |= 1 << (var_level(s,lit_var(lits[i])) & 31);
minl |= 1 << (lev & 31);
}
// simplify (full) // simplify (full)
for (i = j = 1; i < veci_size(learnt); i++){ for (i = j = 1; i < veci_size(learnt); i++){
if (reasons[lit_var(lits[i])] == 0 || !sat_solver2_lit_removable(s,lits[i],minl)) if (s->reasons[lit_var(lits[i])] == 0 || !sat_solver2_lit_removable(s,lits[i],minl))
lits[j++] = lits[i]; lits[j++] = lits[i];
} }
...@@ -826,12 +776,12 @@ static void sat_solver2_analyze(sat_solver2* s, clause* c, veci* learnt) ...@@ -826,12 +776,12 @@ static void sat_solver2_analyze(sat_solver2* s, clause* c, veci* learnt)
// clear tags // clear tags
tagged = veci_begin(&s->tagged); tagged = veci_begin(&s->tagged);
for (i = 0; i < veci_size(&s->tagged); i++) for (i = 0; i < veci_size(&s->tagged); i++)
tags[tagged[i]] = l_Undef; var_set_tag(s, tagged[i], 0);
veci_resize(&s->tagged,0); veci_resize(&s->tagged,0);
#ifdef DEBUG #ifdef DEBUG
for (i = 0; i < s->size; i++) for (i = 0; i < s->size; i++)
assert(tags[i] == l_Undef); assert(!var_tag(s, i));
#endif #endif
#ifdef VERBOSEDEBUG #ifdef VERBOSEDEBUG
...@@ -839,13 +789,12 @@ static void sat_solver2_analyze(sat_solver2* s, clause* c, veci* learnt) ...@@ -839,13 +789,12 @@ static void sat_solver2_analyze(sat_solver2* s, clause* c, veci* learnt)
for (i = 0; i < veci_size(learnt); i++) printf(" "L_LIT, L_lit(lits[i])); for (i = 0; i < veci_size(learnt); i++) printf(" "L_LIT, L_lit(lits[i]));
#endif #endif
if (veci_size(learnt) > 1){ if (veci_size(learnt) > 1){
int max_i = 1;
int max = levels[lit_var(lits[1])];
lit tmp; lit tmp;
int max_i = 1;
int max = var_level(s, lit_var(lits[1]));
for (i = 2; i < veci_size(learnt); i++) for (i = 2; i < veci_size(learnt); i++)
if (levels[lit_var(lits[i])] > max){ if (max < var_level(s, lit_var(lits[i]))) {
max = levels[lit_var(lits[i])]; max = var_level(s, lit_var(lits[i]));
max_i = i; max_i = i;
} }
...@@ -855,7 +804,7 @@ static void sat_solver2_analyze(sat_solver2* s, clause* c, veci* learnt) ...@@ -855,7 +804,7 @@ static void sat_solver2_analyze(sat_solver2* s, clause* c, veci* learnt)
} }
#ifdef VERBOSEDEBUG #ifdef VERBOSEDEBUG
{ {
int lev = veci_size(learnt) > 1 ? levels[lit_var(lits[1])] : 0; int lev = veci_size(learnt) > 1 ? var_level(s, lit_var(lits[1])) : 0;
printf(" } at level %d\n", lev); printf(" } at level %d\n", lev);
} }
#endif #endif
...@@ -863,20 +812,22 @@ static void sat_solver2_analyze(sat_solver2* s, clause* c, veci* learnt) ...@@ -863,20 +812,22 @@ static void sat_solver2_analyze(sat_solver2* s, clause* c, veci* learnt)
#ifndef SAT_USE_WATCH_ARRAYS #ifndef SAT_USE_WATCH_ARRAYS
static inline clause* clause_propagate__( sat_solver2* s, lit Lit ) static inline clause* clause_propagate__front( sat_solver2* s, lit Lit )
{ {
clause ** ppPrev, * pCur, * pTemp; clause * pCur;
cla * ppPrev, pThis, pTemp;
lit LitF = lit_neg(Lit); lit LitF = lit_neg(Lit);
int i, sig, Counter = 0; int i, Counter = 0;
s->stats.propagations++; s->stats.propagations++;
if ( s->pWatches[Lit] == NULL ) if ( s->pWatches[Lit] == 0 )
return NULL; return NULL;
// iterate through the literals // iterate through the literals
ppPrev = s->pWatches + Lit; ppPrev = s->pWatches + Lit;
for ( pCur = *ppPrev; pCur; pCur = *ppPrev ) for ( pThis = *ppPrev; pThis; pThis = *ppPrev )
{ {
Counter++; Counter++;
// make sure the false literal is in the second literal of the clause // make sure the false literal is in the second literal of the clause
pCur = get_clause( s, pThis );
if ( pCur->lits[0] == LitF ) if ( pCur->lits[0] == LitF )
{ {
pCur->lits[0] = pCur->lits[1]; pCur->lits[0] = pCur->lits[1];
...@@ -889,20 +840,22 @@ static inline clause* clause_propagate__( sat_solver2* s, lit Lit ) ...@@ -889,20 +840,22 @@ static inline clause* clause_propagate__( sat_solver2* s, lit Lit )
// if the first literal is true, the clause is satisfied // if the first literal is true, the clause is satisfied
// if ( pCur->lits[0] == s->pAssigns[lit_var(pCur->lits[0])] ) // if ( pCur->lits[0] == s->pAssigns[lit_var(pCur->lits[0])] )
sig = !lit_sign(pCur->lits[0]); sig += sig - 1; // sig = !lit_sign(pCur->lits[0]); sig += sig - 1;
if (s->assigns[lit_var(pCur->lits[0])] == sig) // if (s->assigns[lit_var(pCur->lits[0])] == sig)
if (var_value(s, lit_var(pCur->lits[0])) == lit_sign(pCur->lits[0]))
{ {
ppPrev = &pCur->pNext[1]; ppPrev = &pCur->pNext[1];
continue; continue;
} }
// look for a new literal to watch // look for a new literal to watch
for ( i = 2; i < clause_size(pCur); i++ ) for ( i = 2; i < clause_nlits(pCur); i++ )
{ {
// skip the case when the literal is false // skip the case when the literal is false
// if ( lit_neg(pCur->lits[i]) == p->pAssigns[lit_var(pCur->lits[i])] ) // if ( lit_neg(pCur->lits[i]) == p->pAssigns[lit_var(pCur->lits[i])] )
sig = lit_sign(pCur->lits[i]); sig += sig - 1; // sig = lit_sign(pCur->lits[i]); sig += sig - 1;
if (s->assigns[lit_var(pCur->lits[i])] == sig) // if (s->assigns[lit_var(pCur->lits[i])] == sig)
if (var_value(s, lit_var(pCur->lits[i])) == !lit_sign(pCur->lits[i]))
continue; continue;
// the literal is either true or unassigned - watch it // the literal is either true or unassigned - watch it
pCur->lits[1] = pCur->lits[i]; pCur->lits[1] = pCur->lits[i];
...@@ -913,11 +866,11 @@ static inline clause* clause_propagate__( sat_solver2* s, lit Lit ) ...@@ -913,11 +866,11 @@ static inline clause* clause_propagate__( sat_solver2* s, lit Lit )
clause_watch( s, pCur, pCur->lits[1] ); clause_watch( s, pCur, pCur->lits[1] );
break; break;
} }
if ( i < clause_size(pCur) ) // found new watch if ( i < clause_nlits(pCur) ) // found new watch
continue; continue;
// clause is unit - enqueue new implication // clause is unit - enqueue new implication
if ( enqueue(s, pCur->lits[0], pCur) ) if ( enqueue(s, pCur->lits[0], clause_id(s,pCur)) )
{ {
ppPrev = &pCur->pNext[1]; ppPrev = &pCur->pNext[1];
continue; continue;
...@@ -930,31 +883,32 @@ static inline clause* clause_propagate__( sat_solver2* s, lit Lit ) ...@@ -930,31 +883,32 @@ static inline clause* clause_propagate__( sat_solver2* s, lit Lit )
// printf( "%d ", Counter ); // printf( "%d ", Counter );
return NULL; return NULL;
} }
clause* sat_solver2_propagate_new__( sat_solver2* s ) clause* sat_solver2_propagate_new__front( sat_solver2* s )
{ {
clause* pClause; clause* pClause;
int Lit; lit Lit;
while ( s->qtail - s->qhead > 0 ) while ( s->qtail - s->qhead > 0 )
{ {
Lit = s->trail[s->qhead++]; Lit = s->trail[s->qhead++];
pClause = clause_propagate__( s, Lit ); pClause = clause_propagate__front( s, Lit );
if ( pClause ) if ( pClause )
return pClause; return pClause;
} }
return NULL; return NULL;
} }
static inline clause* clause_propagate( sat_solver2* s, lit Lit, clause** ppHead, clause** ppTail ) static inline clause* clause_propagate( sat_solver2* s, lit Lit, int * ppHead, int * ppTail )
{ {
clause ** ppPrev = ppHead; clause * pCur;
clause * pCur, * pTemp, * pPrev = NULL; cla * ppPrev = ppHead, pThis, pTemp, pPrev = 0;
lit LitF = lit_neg(Lit); lit LitF = lit_neg(Lit);
int i, sig, Counter = 0; int i, Counter = 0;
// iterate through the clauses in the linked list // iterate through the literals
for ( pCur = *ppPrev; pCur; pCur = *ppPrev ) for ( pThis = *ppPrev; pThis; pThis = *ppPrev )
{ {
Counter++; Counter++;
// make sure the false literal is in the second literal of the clause // make sure the false literal is in the second literal of the clause
pCur = get_clause( s, pThis );
if ( pCur->lits[0] == LitF ) if ( pCur->lits[0] == LitF )
{ {
pCur->lits[0] = pCur->lits[1]; pCur->lits[0] = pCur->lits[1];
...@@ -967,29 +921,31 @@ static inline clause* clause_propagate( sat_solver2* s, lit Lit, clause** ppHead ...@@ -967,29 +921,31 @@ static inline clause* clause_propagate( sat_solver2* s, lit Lit, clause** ppHead
// if the first literal is true, the clause is satisfied // if the first literal is true, the clause is satisfied
// if ( pCur->lits[0] == s->pAssigns[lit_var(pCur->lits[0])] ) // if ( pCur->lits[0] == s->pAssigns[lit_var(pCur->lits[0])] )
sig = !lit_sign(pCur->lits[0]); sig += sig - 1; // sig = !lit_sign(pCur->lits[0]); sig += sig - 1;
if (s->assigns[lit_var(pCur->lits[0])] == sig) // if ( s->assigns[lit_var(pCur->lits[0])] == sig )
if (var_value(s, lit_var(pCur->lits[0])) == lit_sign(pCur->lits[0]))
{ {
pPrev = pCur; pPrev = pThis;
ppPrev = &pCur->pNext[1]; ppPrev = &pCur->pNext[1];
continue; continue;
} }
// look for a new literal to watch // look for a new literal to watch
for ( i = 2; i < clause_size(pCur); i++ ) for ( i = 2; i < clause_nlits(pCur); i++ )
{ {
// skip the case when the literal is false // skip the case when the literal is false
// if ( lit_neg(pCur->lits[i]) == p->pAssigns[lit_var(pCur->lits[i])] ) // if ( lit_neg(pCur->lits[i]) == p->pAssigns[lit_var(pCur->lits[i])] )
sig = lit_sign(pCur->lits[i]); sig += sig - 1; // sig = lit_sign(pCur->lits[i]); sig += sig - 1;
if (s->assigns[lit_var(pCur->lits[i])] == sig) // if ( s->assigns[lit_var(pCur->lits[i])] == sig )
if (var_value(s, lit_var(pCur->lits[i])) == !lit_sign(pCur->lits[i]))
continue; continue;
// the literal is either true or unassigned - watch it // the literal is either true or unassigned - watch it
pCur->lits[1] = pCur->lits[i]; pCur->lits[1] = pCur->lits[i];
pCur->lits[i] = LitF; pCur->lits[i] = LitF;
// remove this clause from the watch list of Lit // remove this clause from the watch list of Lit
if ( pCur->pNext[1] == NULL ) if ( pCur->pNext[1] == 0 )
{ {
assert( *ppTail == pCur ); assert( *ppTail == pThis );
*ppTail = pPrev; *ppTail = pPrev;
} }
*ppPrev = pCur->pNext[1]; *ppPrev = pCur->pNext[1];
...@@ -997,13 +953,13 @@ static inline clause* clause_propagate( sat_solver2* s, lit Lit, clause** ppHead ...@@ -997,13 +953,13 @@ static inline clause* clause_propagate( sat_solver2* s, lit Lit, clause** ppHead
clause_watch( s, pCur, pCur->lits[1] ); clause_watch( s, pCur, pCur->lits[1] );
break; break;
} }
if ( i < clause_size(pCur) ) // found new watch if ( i < clause_nlits(pCur) ) // found new watch
continue; continue;
// clause is unit - enqueue new implication // clause is unit - enqueue new implication
if ( enqueue(s, pCur->lits[0], pCur) ) if ( enqueue(s, pCur->lits[0], clause_id(s,pCur)) )
{ {
pPrev = pCur; pPrev = pThis;
ppPrev = &pCur->pNext[1]; ppPrev = &pCur->pNext[1];
continue; continue;
} }
...@@ -1014,71 +970,67 @@ static inline clause* clause_propagate( sat_solver2* s, lit Lit, clause** ppHead ...@@ -1014,71 +970,67 @@ static inline clause* clause_propagate( sat_solver2* s, lit Lit, clause** ppHead
} }
// printf( "%d ", Counter ); // printf( "%d ", Counter );
return NULL; return NULL;
} }
clause* sat_solver2_propagate_new( sat_solver2* s ) clause* sat_solver2_propagate_new( sat_solver2* s )
{ {
clause* pClause,* pHead,* pTail; clause * pClause, * pTailC;
int Lit, Flag; cla pHead, pTail;
lit Lit;
while ( s->qtail - s->qhead > 0 ) while ( s->qtail - s->qhead > 0 )
{ {
s->stats.propagations++; s->stats.propagations++;
Lit = s->trail[s->qhead++]; Lit = s->trail[s->qhead++];
if ( s->pWatches[Lit] == NULL ) if ( s->pWatches[Lit] == 0 )
continue; continue;
// get head and tail // get head and tail
pTail = s->pWatches[Lit]; pTail = s->pWatches[Lit];
pTailC = get_clause( s, pTail );
/* /*
if ( pTail->lits[0] == lit_neg(Lit) ) if ( pTailC->lits[0] == lit_neg(Lit) )
{ {
pHead = pTail->pNext[0]; pHead = pTailC->pNext[0];
pTail->pNext[0] = NULL; pTailC->pNext[0] = NULL;
} }
else else
{ {
assert( pTail->lits[1] == lit_neg(Lit) ); assert( pTailC->lits[1] == lit_neg(Lit) );
pHead = pTail->pNext[1]; pHead = pTailC->pNext[1];
pTail->pNext[1] = NULL; pTailC->pNext[1] = NULL;
} }
*/ */
/* if ( s->stats.propagations == 10 )
Flag = pTail->lits[1] == lit_neg(Lit); {
assert( pTail->lits[0] == lit_neg(Lit) || Flag ); int s = 0;
pHead = pTail->pNext[Flag]; }
pTail->pNext[Flag] = NULL; assert( pTailC->lits[0] == lit_neg(Lit) || pTailC->lits[1] == lit_neg(Lit) );
*/ pHead = pTailC->pNext[pTailC->lits[1] == lit_neg(Lit)];
assert( pTail->lits[0] == lit_neg(Lit) || pTail->lits[1] == lit_neg(Lit) ); pTailC->pNext[pTailC->lits[1] == lit_neg(Lit)] = 0;
pHead = pTail->pNext[pTail->lits[1] == lit_neg(Lit)];
pTail->pNext[pTail->lits[1] == lit_neg(Lit)] = NULL;
// propagate // propagate
pClause = clause_propagate( s, Lit, &pHead, &pTail ); pClause = clause_propagate( s, Lit, &pHead, &pTail );
assert( (pHead == NULL) == (pTail == NULL) ); assert( (pHead == 0) == (pTail == 0) );
// create new list // create new list
s->pWatches[Lit] = pTail; s->pWatches[Lit] = pTail;
/* /*
if ( pTail ) if ( pTail )
{ {
if ( pTail->lits[0] == lit_neg(Lit) ) pTailC = get_clause( s, pTail );
pTail->pNext[0] = pHead; if ( pTailC->lits[0] == lit_neg(Lit) )
pTailC->pNext[0] = pHead;
else else
{ {
assert( pTail->lits[1] == lit_neg(Lit) ); assert( pTailC->lits[1] == lit_neg(Lit) );
pTail->pNext[1] = pHead; pTailC->pNext[1] = pHead;
} }
} }
*/ */
/*
if ( pTail )
{
Flag = pTail->lits[1] == lit_neg(Lit);
assert( pTail->lits[0] == lit_neg(Lit) || Flag );
pTail->pNext[Flag] = pHead;
}
*/
if ( pTail ) if ( pTail )
{ {
assert( pTail->lits[0] == lit_neg(Lit) || pTail->lits[1] == lit_neg(Lit) ); pTailC = get_clause( s, pTail );
pTail->pNext[pTail->lits[1] == lit_neg(Lit)] = pHead; assert( pTailC->lits[0] == lit_neg(Lit) || pTailC->lits[1] == lit_neg(Lit) );
pTailC->pNext[pTailC->lits[1] == lit_neg(Lit)] = pHead;
} }
if ( pClause ) if ( pClause )
return pClause; return pClause;
...@@ -1090,75 +1042,71 @@ clause* sat_solver2_propagate_new( sat_solver2* s ) ...@@ -1090,75 +1042,71 @@ clause* sat_solver2_propagate_new( sat_solver2* s )
clause* sat_solver2_propagate(sat_solver2* s) clause* sat_solver2_propagate(sat_solver2* s)
{ {
lbool* values = s->assigns; clause * c, * confl = NULL;
clause* confl = NULL; veci* ws;
lit* lits; lit* lits, false_lit, p, * stop, * k;
cla* begin,* end, *i, *j;
#ifndef SAT_USE_WATCH_ARRAYS #ifndef SAT_USE_WATCH_ARRAYS
return sat_solver2_propagate_new( s ); return sat_solver2_propagate_new( s );
#endif #endif
while (confl == 0 && s->qtail - s->qhead > 0){ while (confl == 0 && s->qtail - s->qhead > 0){
lit p = s->trail[s->qhead++]; p = s->trail[s->qhead++];
vecp* ws = sat_solver2_read_wlist(s,p); ws = sat_solver2_read_wlist(s,p);
clause **begin = (clause**)vecp_begin(ws); begin = (cla*)veci_begin(ws);
clause **end = begin + vecp_size(ws); end = begin + veci_size(ws);
clause **i, **j;
s->stats.propagations++; s->stats.propagations++;
s->simpdb_props--; s->simpdb_props--;
//printf("checking lit %d: "L_LIT"\n", veci_size(ws), L_lit(p));
for (i = j = begin; i < end; ){ for (i = j = begin; i < end; ){
{ {
lit false_lit; c = get_clause(s,*i);
lbool sig; lits = clause_begin(c);
lits = clause_begin(*i); // Make sure the false literal is data[1]:
false_lit = lit_neg(p);
// Make sure the false literal is data[1]: if (lits[0] == false_lit){
false_lit = lit_neg(p); lits[0] = lits[1];
if (lits[0] == false_lit){ lits[1] = false_lit;
lits[0] = lits[1];
lits[1] = false_lit;
}
assert(lits[1] == false_lit);
//printf("checking clause: "); printlits(lits, lits+clause_size(*i)); printf("\n");
// If 0th watch is true, then clause is already satisfied.
sig = !lit_sign(lits[0]); sig += sig - 1;
if (values[lit_var(lits[0])] == sig){
*j++ = *i;
}else{
// Look for new watch:
lit* stop = lits + clause_size(*i);
lit* k;
for (k = lits + 2; k < stop; k++){
lbool sig = lit_sign(*k); sig += sig - 1;
if (values[lit_var(*k)] != sig){
lits[1] = *k;
*k = false_lit;
vecp_push(sat_solver2_read_wlist(s,lit_neg(lits[1])),*i);
goto next; }
} }
assert(lits[1] == false_lit);
*j++ = *i;
// Clause is unit under assignment: // If 0th watch is true, then clause is already satisfied.
if (!enqueue(s,lits[0], *i)){ // sig = !lit_sign(lits[0]); sig += sig - 1;
confl = *i++; // if (values[lit_var(lits[0])] == sig){
// Copy the remaining watches: if (var_value(s, lit_var(lits[0])) == lit_sign(lits[0]))
// s->stats.inspects2 += end - i; *j++ = *i;
while (i < end) else{
*j++ = *i++; // Look for new watch:
stop = lits + clause_nlits(c);
for (k = lits + 2; k < stop; k++){
// lbool sig = lit_sign(*k); sig += sig - 1;
// if (values[lit_var(*k)] != sig){
if (var_value(s, lit_var(*k)) != !lit_sign(*k)){
lits[1] = *k;
*k = false_lit;
veci_push(sat_solver2_read_wlist(s,lit_neg(lits[1])),*i);
goto next; }
}
*j++ = *i;
// Clause is unit under assignment:
if (!enqueue(s,lits[0], *i)){
confl = get_clause(s,*i++);
// Copy the remaining watches:
// s->stats.inspects2 += end - i;
while (i < end)
*j++ = *i++;
}
} }
} }
next: i++;
} }
next: s->stats.inspects += j - (int*)veci_begin(ws);
i++; veci_resize(ws,j - (int*)veci_begin(ws));
}
s->stats.inspects += j - (clause**)vecp_begin(ws);
vecp_resize(ws,j - (clause**)vecp_begin(ws));
} }
return confl; return confl;
...@@ -1166,49 +1114,45 @@ clause* sat_solver2_propagate(sat_solver2* s) ...@@ -1166,49 +1114,45 @@ clause* sat_solver2_propagate(sat_solver2* s)
static int clause_cmp (const void* x, const void* y) { static int clause_cmp (const void* x, const void* y) {
// return clause_size((clause*)x) > 2 && (clause_size((clause*)y) == 2 || clause_activity((clause*)x) < clause_activity((clause*)y)) ? -1 : 1; } // return clause_nlits((clause*)x) > 2 && (clause_nlits((clause*)y) == 2 || clause_activity((clause*)x) < clause_activity((clause*)y)) ? -1 : 1; }
return clause_size((clause*)x) > 2 && (clause_size((clause*)y) == 2 || ((clause*)x)->act < ((clause*)y)->act) ? -1 : 1; } return clause_nlits((clause*)x) > 2 && (clause_nlits((clause*)y) == 2 || ((clause*)x)->act < ((clause*)y)->act) ? -1 : 1; }
void sat_solver2_reducedb(sat_solver2* s) void sat_solver2_reducedb(sat_solver2* s)
{ {
int Counter = 0; /*
int i, j; vecp Vec, * pVec = &Vec;
unsigned extra_lim; double extra_limD = (double)s->cla_inc / veci_size(&s->learnts); // Remove any clause below this activity
double extra_limD = (double)s->cla_inc / vecp_size(&s->learnts); // Remove any clause below this activity unsigned extra_lim = (extra_limD < 1.0) ? 1 : (int)extra_limD;
clause** learnts = (clause**)vecp_begin(&s->learnts); int i, j, * pBeg, * pEnd;
clause** reasons = s->reasons;
// move clauses into vector
if ( extra_limD < 1.0 ) vecp_new( pVec );
extra_lim = 1; pBeg = (int*)veci_begin(&s->learnts);
else pEnd = pBeg + veci_size(&s->learnts);
extra_lim = (int)extra_limD; while ( pBeg < pEnd )
vecp_push( pVec, get_clause(s,*pBeg++) );
sat_solver2_sort(vecp_begin(&s->learnts), vecp_size(&s->learnts), &clause_cmp); sat_solver2_sort( vecp_begin(pVec), vecp_size(pVec), &clause_cmp );
vecp_delete( pVec );
for (i = j = 0; i < vecp_size(&s->learnts) / 2; i++){
// printf( "%d ", learnts[i]->act ); // compact clauses
// for (i = j = 0; i < vecp_size(&s->learnts) / 4; i++){ pBeg = (int*)veci_begin(&s->learnts);
if (clause_size(learnts[i]) > 2 && reasons[lit_var(*clause_begin(learnts[i]))] != learnts[i]) for (i = j = 0; i < vecp_size(pVec); i++)
clause_remove(s,learnts[i]), Counter++; {
else clause * c = ((clause**)vecp_begin(pVec))[i];
learnts[j++] = learnts[i]; int Cid = clause_id(s,c);
} // printf( "%d ", c->act );
for (; i < vecp_size(&s->learnts); i++){ if (clause_nlits(c) > 2 && s->reasons[lit_var(*clause_begin(c))] != Cid && (i < vecp_size(pVec)/2 || c->act < extra_lim) )
// if (clause_size(learnts[i]) > 2 && reasons[lit_var(*clause_begin(learnts[i]))] != learnts[i] && clause_activity(learnts[i]) < extra_lim) clause_remove(s,c);
if (clause_size(learnts[i]) > 2 && reasons[lit_var(*clause_begin(learnts[i]))] != learnts[i] && learnts[i]->act < extra_lim)
clause_remove(s,learnts[i]), Counter++;
else else
learnts[j++] = learnts[i]; pBeg[j++] = Cid;
} }
printf( "Reduction removed %10d clauses (out of %10d)... Value = %d\n", Counter, vecp_size(&s->learnts), extra_lim ); printf( "Reduction removed %10d clauses (out of %10d)... Value = %d\n", veci_size(&s->learnts) - j, veci_size(&s->learnts), extra_lim );
veci_resize(&s->learnts,j);
//printf("reducedb deleted %d\n", vecp_size(&s->learnts) - j); */
vecp_resize(&s->learnts,j);
} }
static lbool sat_solver2_search(sat_solver2* s, ABC_INT64_T nof_conflicts, ABC_INT64_T * nof_learnts) static lbool sat_solver2_search(sat_solver2* s, ABC_INT64_T nof_conflicts, ABC_INT64_T * nof_learnts)
{ {
int* levels = s->levels;
double var_decay = 0.95; double var_decay = 0.95;
double clause_decay = 0.999; double clause_decay = 0.999;
double random_var_freq = s->fNotUseRandom ? 0.0 : 0.02; double random_var_freq = s->fNotUseRandom ? 0.0 : 0.02;
...@@ -1244,13 +1188,15 @@ static lbool sat_solver2_search(sat_solver2* s, ABC_INT64_T nof_conflicts, ABC_I ...@@ -1244,13 +1188,15 @@ static lbool sat_solver2_search(sat_solver2* s, ABC_INT64_T nof_conflicts, ABC_I
veci_resize(&learnt_clause,0); veci_resize(&learnt_clause,0);
sat_solver2_analyze(s, confl, &learnt_clause); sat_solver2_analyze(s, confl, &learnt_clause);
blevel = veci_size(&learnt_clause) > 1 ? levels[lit_var(veci_begin(&learnt_clause)[1])] : s->root_level; blevel = veci_size(&learnt_clause) > 1 ? var_level(s, lit_var(veci_begin(&learnt_clause)[1])) : s->root_level;
blevel = s->root_level > blevel ? s->root_level : blevel; blevel = s->root_level > blevel ? s->root_level : blevel;
sat_solver2_canceluntil(s,blevel); sat_solver2_canceluntil(s,blevel);
sat_solver2_record(s,&learnt_clause); sat_solver2_record(s,&learnt_clause);
#ifdef SAT_USE_ANALYZE_FINAL #ifdef SAT_USE_ANALYZE_FINAL
// if (learnt_clause.size() == 1) level[var(learnt_clause[0])] = 0; // (this is ugly (but needed for 'analyzeFinal()') -- in future versions, we will backtrack past the 'root_level' and redo the assumptions) // if (learnt_clause.size() == 1) level[var(learnt_clause[0])] = 0;
if ( learnt_clause.size == 1 ) s->levels[lit_var(learnt_clause.ptr[0])] = 0; // (this is ugly (but needed for 'analyzeFinal()') -- in future versions, we will backtrack past the 'root_level' and redo the assumptions)
if ( learnt_clause.size == 1 )
var_set_level( s, lit_var(learnt_clause.ptr[0]), 0 );
#endif #endif
act_var_decay(s); act_var_decay(s);
act_clause_decay(s); act_clause_decay(s);
...@@ -1281,7 +1227,7 @@ static lbool sat_solver2_search(sat_solver2* s, ABC_INT64_T nof_conflicts, ABC_I ...@@ -1281,7 +1227,7 @@ static lbool sat_solver2_search(sat_solver2* s, ABC_INT64_T nof_conflicts, ABC_I
// Simplify the set of problem clauses: // Simplify the set of problem clauses:
// sat_solver2_simplify(s); // sat_solver2_simplify(s);
if (*nof_learnts >= 0 && vecp_size(&s->learnts) - s->qtail >= *nof_learnts) if (*nof_learnts >= 0 && s->stats.learnts - s->qtail >= *nof_learnts)
{ {
// Reduce the set of learnt clauses: // Reduce the set of learnt clauses:
// sat_solver2_reducedb(s); // sat_solver2_reducedb(s);
...@@ -1294,11 +1240,10 @@ static lbool sat_solver2_search(sat_solver2* s, ABC_INT64_T nof_conflicts, ABC_I ...@@ -1294,11 +1240,10 @@ static lbool sat_solver2_search(sat_solver2* s, ABC_INT64_T nof_conflicts, ABC_I
if (next == var_Undef){ if (next == var_Undef){
// Model found: // Model found:
lbool* values = s->assigns;
int i; int i;
veci_resize(&s->model, 0); veci_resize(&s->model, 0);
for (i = 0; i < s->size; i++) for (i = 0; i < s->size; i++)
veci_push(&s->model,(int)values[i]); veci_push( &s->model, var_value(s,i)==var1 ? l_True : l_False );
sat_solver2_canceluntil(s,s->root_level); sat_solver2_canceluntil(s,s->root_level);
veci_delete(&learnt_clause); veci_delete(&learnt_clause);
...@@ -1313,7 +1258,7 @@ static lbool sat_solver2_search(sat_solver2* s, ABC_INT64_T nof_conflicts, ABC_I ...@@ -1313,7 +1258,7 @@ static lbool sat_solver2_search(sat_solver2* s, ABC_INT64_T nof_conflicts, ABC_I
return l_True; return l_True;
} }
if ( s->polarity[next] ) // positive polarity if ( var_polar(s, next) ) // positive polarity
assume(s,toLit(next)); assume(s,toLit(next));
else else
assume(s,lit_neg(toLit(next))); assume(s,lit_neg(toLit(next)));
...@@ -1328,50 +1273,24 @@ static lbool sat_solver2_search(sat_solver2* s, ABC_INT64_T nof_conflicts, ABC_I ...@@ -1328,50 +1273,24 @@ static lbool sat_solver2_search(sat_solver2* s, ABC_INT64_T nof_conflicts, ABC_I
sat_solver2* sat_solver2_new(void) sat_solver2* sat_solver2_new(void)
{ {
sat_solver2* s = (sat_solver2*)ABC_ALLOC( char, sizeof(sat_solver2)); sat_solver2 * s;
memset( s, 0, sizeof(sat_solver2) ); s = (sat_solver2 *)ABC_CALLOC( char, sizeof(sat_solver2) );
// initialize vectors // initialize vectors
vecp_new(&s->clauses);
vecp_new(&s->learnts);
veci_new(&s->order); veci_new(&s->order);
veci_new(&s->trail_lim); veci_new(&s->trail_lim);
veci_new(&s->tagged); veci_new(&s->tagged);
veci_new(&s->stack); veci_new(&s->stack);
veci_new(&s->model); veci_new(&s->model);
// veci_new(&s->act_vars);
veci_new(&s->temp_clause); veci_new(&s->temp_clause);
veci_new(&s->conf_final); veci_new(&s->conf_final);
// initialize other vars // initialize other
s->size = 0; s->iLearnt = -1; // the first learnt clause
s->cap = 0; s->iLast = -1; // the last learnt clause
s->qhead = 0;
s->qtail = 0;
s->cla_inc = (1 << 11); s->cla_inc = (1 << 11);
s->var_inc = (1 << 5); s->var_inc = (1 << 5);
// s->cla_decay = 1;
// s->var_decay = 1;
s->root_level = 0;
s->simpdb_assigns = 0;
s->simpdb_props = 0;
s->random_seed = 91648253; s->random_seed = 91648253;
s->progress_estimate = 0;
s->binary = (clause*)ABC_ALLOC( char, sizeof(clause) + sizeof(lit)*2);
s->binary->size_learnt = (2 << 1);
s->verbosity = 0;
s->stats.starts = 0;
s->stats.decisions = 0;
s->stats.propagations = 0;
s->stats.inspects = 0;
s->stats.conflicts = 0;
s->stats.clauses = 0;
s->stats.clauses_literals = 0;
s->stats.learnts = 0;
s->stats.learnts_literals = 0;
s->stats.max_literals = 0;
s->stats.tot_literals = 0;
return s; return s;
} }
...@@ -1381,8 +1300,6 @@ void sat_solver2_delete(sat_solver2* s) ...@@ -1381,8 +1300,6 @@ void sat_solver2_delete(sat_solver2* s)
ABC_FREE(s->pMemArray); ABC_FREE(s->pMemArray);
// delete vectors // delete vectors
vecp_delete(&s->clauses);
vecp_delete(&s->learnts);
veci_delete(&s->order); veci_delete(&s->order);
veci_delete(&s->trail_lim); veci_delete(&s->trail_lim);
veci_delete(&s->tagged); veci_delete(&s->tagged);
...@@ -1390,24 +1307,20 @@ void sat_solver2_delete(sat_solver2* s) ...@@ -1390,24 +1307,20 @@ void sat_solver2_delete(sat_solver2* s)
veci_delete(&s->model); veci_delete(&s->model);
veci_delete(&s->temp_clause); veci_delete(&s->temp_clause);
veci_delete(&s->conf_final); veci_delete(&s->conf_final);
ABC_FREE(s->binary);
// delete arrays // delete arrays
if (s->assigns != 0){ if (s->vi != 0){
int i; int i;
if ( s->wlists ) if ( s->wlists )
for (i = 0; i < s->size*2; i++) for (i = 0; i < s->size*2; i++)
vecp_delete(&s->wlists[i]); veci_delete(&s->wlists[i]);
ABC_FREE(s->wlists ); ABC_FREE(s->wlists );
ABC_FREE(s->pWatches ); ABC_FREE(s->pWatches );
ABC_FREE(s->vi );
ABC_FREE(s->activity ); ABC_FREE(s->activity );
ABC_FREE(s->assigns );
ABC_FREE(s->orderpos ); ABC_FREE(s->orderpos );
ABC_FREE(s->reasons ); ABC_FREE(s->reasons );
ABC_FREE(s->levels );
ABC_FREE(s->trail ); ABC_FREE(s->trail );
ABC_FREE(s->tags );
ABC_FREE(s->polarity );
} }
ABC_FREE(s); ABC_FREE(s);
} }
...@@ -1415,10 +1328,9 @@ void sat_solver2_delete(sat_solver2* s) ...@@ -1415,10 +1328,9 @@ void sat_solver2_delete(sat_solver2* s)
int sat_solver2_addclause(sat_solver2* s, lit* begin, lit* end) int sat_solver2_addclause(sat_solver2* s, lit* begin, lit* end)
{ {
clause * c; int c;
lit *i,*j; lit *i,*j;
int maxvar; int maxvar;
lbool* values;
lit last; lit last;
veci_resize( &s->temp_clause, 0 ); veci_resize( &s->temp_clause, 0 );
...@@ -1444,16 +1356,18 @@ int sat_solver2_addclause(sat_solver2* s, lit* begin, lit* end) ...@@ -1444,16 +1356,18 @@ int sat_solver2_addclause(sat_solver2* s, lit* begin, lit* end)
// sat_solver2_setnvars(s, lit_var(*(end-1))+1 ); // sat_solver2_setnvars(s, lit_var(*(end-1))+1 );
//printlits(begin,end); printf("\n"); //printlits(begin,end); printf("\n");
values = s->assigns; // values = s->assigns;
// delete duplicates // delete duplicates
last = lit_Undef; last = lit_Undef;
for (i = j = begin; i < end; i++){ for (i = j = begin; i < end; i++){
//printf("lit: "L_LIT", value = %d\n", L_lit(*i), (lit_sign(*i) ? -values[lit_var(*i)] : values[lit_var(*i)])); //printf("lit: "L_LIT", value = %d\n", L_lit(*i), (lit_sign(*i) ? -values[lit_var(*i)] : values[lit_var(*i)]));
lbool sig = !lit_sign(*i); sig += sig - 1; // lbool sig = !lit_sign(*i); sig += sig - 1;
if (*i == lit_neg(last) || sig == values[lit_var(*i)]) // if (*i == lit_neg(last) || sig == values[lit_var(*i)])
if (*i == lit_neg(last) || var_value(s, lit_var(*i)) == lit_sign(*i))
return true; // tautology return true; // tautology
else if (*i != last && values[lit_var(*i)] == l_Undef) // else if (*i != last && values[lit_var(*i)] == l_Undef)
else if (*i != last && var_value(s, lit_var(*i)) == varX)
last = *j++ = *i; last = *j++ = *i;
} }
...@@ -1463,23 +1377,19 @@ int sat_solver2_addclause(sat_solver2* s, lit* begin, lit* end) ...@@ -1463,23 +1377,19 @@ int sat_solver2_addclause(sat_solver2* s, lit* begin, lit* end)
return false; return false;
if (j - begin == 1) // unit clause if (j - begin == 1) // unit clause
return enqueue(s,*begin,(clause*)0); return enqueue(s,*begin,0);
// create new clause // create new clause
c = clause_new(s,begin,j,0); c = clause_new(s,begin,j,0);
if ( c )
vecp_push( &s->clauses,c );
s->stats.clauses++; s->stats.clauses++;
s->stats.clauses_literals += j - begin; s->stats.clauses_literals += j - begin;
return true; return true;
} }
int sat_solver2_simplify(sat_solver2* s) int sat_solver2_simplify(sat_solver2* s)
{ {
clause** reasons;
// int type; // int type;
int Counter = 0; int Counter = 0;
...@@ -1491,21 +1401,21 @@ int sat_solver2_simplify(sat_solver2* s) ...@@ -1491,21 +1401,21 @@ int sat_solver2_simplify(sat_solver2* s)
if (s->qhead == s->simpdb_assigns || s->simpdb_props > 0) if (s->qhead == s->simpdb_assigns || s->simpdb_props > 0)
return true; return true;
reasons = s->reasons;
/* /*
for (type = 0; type < 2; type++){ for (type = 0; type < 2; type++){
vecp* cs = type ? &s->learnts : &s->clauses; veci* cs = type ? &s->learnts : &s->clauses;
clause** cls = (clause**)vecp_begin(cs); int* cls = (int*)veci_begin(cs);
int i, j; int i, j;
for (j = i = 0; i < vecp_size(cs); i++){ for (j = i = 0; i < veci_size(cs); i++){
if (reasons[lit_var(*clause_begin(cls[i]))] != cls[i] && clause * c = get_clause(s,cls[i]);
clause_simplify(s,cls[i]) == l_True) if (s->reasons[lit_var(*clause_begin(c))] != cls[i] &&
clause_remove(s,cls[i]), Counter++; clause_simplify(s,c) == l_True)
clause_remove(s,c), Counter++;
else else
cls[j++] = cls[i]; cls[j++] = cls[i];
} }
vecp_resize(cs,j); veci_resize(cs,j);
} }
*/ */
//printf( "Simplification removed %d clauses (out of %d)...\n", Counter, s->stats.clauses ); //printf( "Simplification removed %d clauses (out of %d)...\n", Counter, s->stats.clauses );
...@@ -1543,7 +1453,7 @@ int sat_solver2_solve(sat_solver2* s, lit* begin, lit* end, ABC_INT64_T nConfLim ...@@ -1543,7 +1453,7 @@ int sat_solver2_solve(sat_solver2* s, lit* begin, lit* end, ABC_INT64_T nConfLim
ABC_INT64_T nof_conflicts; ABC_INT64_T nof_conflicts;
ABC_INT64_T nof_learnts = sat_solver2_nclauses(s) / 3; ABC_INT64_T nof_learnts = sat_solver2_nclauses(s) / 3;
lbool status = l_Undef; lbool status = l_Undef;
lbool* values = s->assigns; // lbool* values = s->assigns;
lit* i; lit* i;
// set the external limits // set the external limits
...@@ -1565,15 +1475,16 @@ int sat_solver2_solve(sat_solver2* s, lit* begin, lit* end, ABC_INT64_T nConfLim ...@@ -1565,15 +1475,16 @@ int sat_solver2_solve(sat_solver2* s, lit* begin, lit* end, ABC_INT64_T nConfLim
//printf("solve: "); printlits(begin, end); printf("\n"); //printf("solve: "); printlits(begin, end); printf("\n");
for (i = begin; i < end; i++){ for (i = begin; i < end; i++){
switch (lit_sign(*i) ? -values[lit_var(*i)] : values[lit_var(*i)]){ // switch (lit_sign(*i) ? -values[lit_var(*i)] : values[lit_var(*i)]){
case 1: // l_True: switch (var_value(s, *i)) {
case var1: // l_True:
break; break;
case 0: // l_Undef case varX: // l_Undef
assume(s, *i); assume(s, *i);
if (sat_solver2_propagate(s) == NULL) if (sat_solver2_propagate(s) == NULL)
break; break;
// fallthrough // fallthrough
case -1: // l_False case var0: // l_False
sat_solver2_canceluntil(s, 0); sat_solver2_canceluntil(s, 0);
return l_False; return l_False;
} }
...@@ -1622,9 +1533,9 @@ int sat_solver2_solve(sat_solver2* s, lit* begin, lit* end, ABC_INT64_T nConfLim ...@@ -1622,9 +1533,9 @@ int sat_solver2_solve(sat_solver2* s, lit* begin, lit* end, ABC_INT64_T nConfLim
lit p = *i; lit p = *i;
assert(lit_var(p) < s->size); assert(lit_var(p) < s->size);
veci_push(&s->trail_lim,s->qtail); veci_push(&s->trail_lim,s->qtail);
if (!enqueue(s,p,(clause*)0)) if (!enqueue(s,p,0))
{ {
clause * r = s->reasons[lit_var(p)]; clause * r = get_clause(s, s->reasons[lit_var(p)]);
if (r != NULL) if (r != NULL)
{ {
clause* confl = r; clause* confl = r;
...@@ -1722,7 +1633,7 @@ int sat_solver2_nvars(sat_solver2* s) ...@@ -1722,7 +1633,7 @@ int sat_solver2_nvars(sat_solver2* s)
int sat_solver2_nclauses(sat_solver2* s) int sat_solver2_nclauses(sat_solver2* s)
{ {
return vecp_size(&s->clauses); return (int)s->stats.clauses;
} }
......
...@@ -77,71 +77,61 @@ extern void * sat_solver2_store_release( sat_solver2 * s ); ...@@ -77,71 +77,61 @@ extern void * sat_solver2_store_release( sat_solver2 * s );
//struct clause_t; //struct clause_t;
//typedef struct clause_t clause; //typedef struct clause_t clause;
struct varinfo_t;
typedef struct varinfo_t varinfo;
struct sat_solver2_t struct sat_solver2_t
{ {
int size; // nof variables int size; // nof variables
int cap; // size of varmaps int cap; // size of varmaps
int qhead; // Head index of queue. int qhead; // Head index of queue.
int qtail; // Tail index of queue. int qtail; // Tail index of queue.
int root_level; // Level of first proper decision.
int simpdb_assigns;// Number of top-level assignments at last 'simplifyDB()'.
int simpdb_props; // Number of propagations before next 'simplifyDB()'.
double random_seed;
double progress_estimate;
int verbosity; // Verbosity level. 0=silent, 1=some progress report, 2=everything
int fNotUseRandom; // do not allow random decisions with a fixed probability
// int fSkipSimplify; // set to one to skip simplification of the clause database
// clauses // clauses
vecp clauses; // List of problem constraints. (contains: clause*) int iLearnt; // the first learnt clause
vecp learnts; // List of learnt clauses. (contains: clause*) int iLast; // the last learnt clause
veci* wlists; // watcher lists (for each literal)
cla* pWatches; // watcher lists (for each literal)
// activities // clause memory
// double var_inc; // Amount to bump next variable with. int * pMemArray;
// double var_decay; // INVERSE decay factor for variable activity: stores 1/decay. int nMemAlloc;
int var_inc; int nMemSize;
// int var_decay;
// float cla_inc; // Amount to bump next clause with.
int cla_inc; // Amount to bump next clause with.
// float cla_decay; // INVERSE decay factor for clause activity: stores 1/decay.
clause** pWatches; // watcher lists (for each literal)
vecp* wlists; //
// double* activity; // A heuristic measurement of the activity of a variable.
unsigned*activity; // A heuristic measurement of the activity of a variable.
lbool* assigns; // Current values of variables.
int* orderpos; // Index in variable order.
clause** reasons; //
int* levels; //
lit* trail;
char* polarity;
clause* binary; // A temporary binary clause
lbool* tags; //
veci tagged; // (contains: var)
veci stack; // (contains: var)
veci order; // Variable order. (heap) (contains: var)
veci trail_lim; // Separator indices for different decision levels in 'trail'. (contains: int)
veci model; // If problem is solved, this vector contains the model (contains: lbool).
veci conf_final; // If problem is unsatisfiable (possibly under assumptions),
// this vector represent the final conflict clause expressed in the assumptions.
int root_level; // Level of first proper decision.
int simpdb_assigns;// Number of top-level assignments at last 'simplifyDB()'.
int simpdb_props; // Number of propagations before next 'simplifyDB()'.
double random_seed;
double progress_estimate;
int verbosity; // Verbosity level. 0=silent, 1=some progress report, 2=everything
stats_t stats;
// activities
int var_inc; // Amount to bump next variable with.
int cla_inc; // Amount to bump next clause with.
unsigned* activity; // A heuristic measurement of the activity of a variable.
// internal state
varinfo * vi; // variable information
cla* reasons;
lit* trail;
int* orderpos; // Index in variable order.
veci tagged; // (contains: var)
veci stack; // (contains: var)
veci order; // Variable order. (heap) (contains: var)
veci trail_lim; // Separator indices for different decision levels in 'trail'. (contains: int)
veci temp_clause; // temporary storage for a CNF clause
veci model; // If problem is solved, this vector contains the model (contains: lbool).
veci conf_final; // If problem is unsatisfiable (possibly under assumptions),
// this vector represent the final conflict clause expressed in the assumptions.
// statistics
stats_t stats;
ABC_INT64_T nConfLimit; // external limit on the number of conflicts ABC_INT64_T nConfLimit; // external limit on the number of conflicts
ABC_INT64_T nInsLimit; // external limit on the number of implications ABC_INT64_T nInsLimit; // external limit on the number of implications
int nRuntimeLimit; // external limit on runtime int nRuntimeLimit; // external limit on runtime
// clause memory
int * pMemArray;
int nMemAlloc;
int nMemSize;
// int fSkipSimplify; // set to one to skip simplification of the clause database
int fNotUseRandom; // do not allow random decisions with a fixed probability
veci temp_clause; // temporary storage for a CNF clause
}; };
static int sat_solver2_var_value( sat_solver2* s, int v ) static int sat_solver2_var_value( sat_solver2* s, int v )
......
...@@ -54,6 +54,15 @@ static inline void veci_push (veci* v, int e) ...@@ -54,6 +54,15 @@ static inline void veci_push (veci* v, int e)
v->cap = newsize; } v->cap = newsize; }
v->ptr[v->size++] = e; v->ptr[v->size++] = e;
} }
static inline void veci_remove(veci* v, int e)
{
int * ws = (int*)veci_begin(v);
int j = 0;
for (; ws[j] != e ; j++);
assert(j < veci_size(v));
for (; j < veci_size(v)-1; j++) ws[j] = ws[j+1];
veci_resize(v,veci_size(v)-1);
}
// vector of 32- or 64-bit pointers // vector of 32- or 64-bit pointers
...@@ -83,6 +92,15 @@ static inline void vecp_push (vecp* v, void* e) ...@@ -83,6 +92,15 @@ static inline void vecp_push (vecp* v, void* e)
v->cap = newsize; } v->cap = newsize; }
v->ptr[v->size++] = e; v->ptr[v->size++] = e;
} }
static inline void vecp_remove(vecp* v, void* e)
{
void** ws = vecp_begin(v);
int j = 0;
for (; ws[j] != e ; j++);
assert(j < vecp_size(v));
for (; j < vecp_size(v)-1; j++) ws[j] = ws[j+1];
vecp_resize(v,vecp_size(v)-1);
}
...@@ -99,6 +117,7 @@ static inline void vecp_push (vecp* v, void* e) ...@@ -99,6 +117,7 @@ static inline void vecp_push (vecp* v, void* e)
#endif #endif
typedef int lit; typedef int lit;
typedef int cla;
typedef char lbool; typedef char lbool;
static const int var_Undef = -1; static const int var_Undef = -1;
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment