⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 zverify_bf.cpp

📁 命题逻辑的求解器,2004年SAT竞赛第一名的求解器
💻 CPP
📖 第 1 页 / 共 2 页
字号:
/********************************************************************* Copyright 2000-2004, Princeton University.  All rights reserved.  By using this software the USER indicates that he or she has read,  understood and will comply with the following: --- Princeton University hereby grants USER nonexclusive permission  to use, copy and/or modify this software for internal, noncommercial, research purposes only. Any distribution, including commercial sale  or license, of this software, copies of the software, its associated  documentation and/or modifications of either is strictly prohibited  without the prior consent of Princeton University.  Title to copyright to this software and its associated documentation shall at all times  remain with Princeton University.  Appropriate copyright notice shall  be placed on all software copies, and a complete copy of this notice  shall be included in all copies of the associated documentation.   No right is  granted to use in advertising, publicity or otherwise  any trademark, service mark, or the name of Princeton University.  --- This software and any associated documentation is provided "as is"  PRINCETON UNIVERSITY MAKES NO REPRESENTATIONS OR WARRANTIES, EXPRESS  OR IMPLIED, INCLUDING THOSE OF MERCHANTABILITY OR FITNESS FOR A  PARTICULAR PURPOSE, OR THAT  USE OF THE SOFTWARE, MODIFICATIONS, OR  ASSOCIATED DOCUMENTATION WILL NOT INFRINGE ANY PATENTS, COPYRIGHTS,  TRADEMARKS OR OTHER INTELLECTUAL PROPERTY RIGHTS OF A THIRD PARTY.   Princeton University shall not be liable under any circumstances for  any direct, indirect, special, incidental, or consequential damages  with respect to any claim by USER or any third party on account of  or arising from the use, or inability to use, this software or its  associated documentation, even if Princeton University has been advised of the possibility of those damages.*********************************************************************/#include <sys/time.h>#include <sys/resource.h>#include <unistd.h>#include <cstdio>#include <cstdlib>#include <vector>#include <set>#include <iostream>#include <fstream>#include <assert.h>#include "mystl_hash.h"using namespace std;const int MAX_BUFF_SIZE = 1024* 1024 * 4;const int CLS_COUNT_ARRAY_SIZE = 1024 * 1024;const int WORD_LEN = 64000;#define UNKNOWN 2#define MEM_LIMIT 800000int _peak_mem;//================================================================================double get_cpu_time(){    double res;    struct rusage usage;    getrusage(RUSAGE_SELF, &usage);    res = usage.ru_utime.tv_usec + usage.ru_stime.tv_usec;    res *= 1e-6;    res += usage.ru_utime.tv_sec + usage.ru_stime.tv_sec; 	    return res;}void get_line(ifstream &fs, vector<char> &buf){    buf.clear();    buf.reserve(4096);    while(!fs.eof()) {        char ch = fs.get();        if(ch == '\n' || ch == '\377')	    break;        if(ch == '\r') 	    continue;        buf.push_back(ch);    }    buf.push_back('\0');    return;}int get_token (char * & lp, char * token){    char * wp = token;    while (*lp && ((*lp == ' ') || (*lp == '\t'))) {	lp++;    }    while (*lp && (*lp != ' ') && (*lp != '\t') && (*lp != '\n')) {	*(wp++) = *(lp++);    }    *wp = '\0';                                 // terminate string    return wp - token; }int get_mem_usage(void) {    FILE * fp;    char buffer[128];    char token[128];    char filename[128];    int pid = getpid();    sprintf(filename, "/proc/%i/status", pid);    if ( (fp = fopen (filename, "r")) == NULL) {	cerr << "Can't open Proc file, are you sure you are using Linux?" << endl;	exit(1);    }    while(!feof(fp)) {	fgets(buffer, 1024, fp);	char * ptr = buffer;	get_token(ptr, token);	if (strcmp(token, "VmSize:")==0) {	    get_token(ptr, token);	    fclose(fp);	    return atoi(token);	}    }    cerr << "Error in get memeory usage" << endl;    exit(1);    return 0;}void check_mem_out(void){    int mem = get_mem_usage();    if (mem > MEM_LIMIT) {	cerr << "Mem out" << endl;	exit(1);    }    if (mem > _peak_mem)	_peak_mem = mem;}int my_a2i(char * str){    int result = 0;    bool neg = false;    if (str[0] == '-') {	neg = true;	++ str;    }    else if (str[0] == '+')	++ str;    for (unsigned i=0, l = strlen(str); i< l; ++i) {	int d = str[i] - '0';	if (d < 0 || d > 9) {	    cerr << "Abort: Unable to change " << str << " into a number " << endl;	    exit(1);	}	result = result * 10 + d;    }    if (neg) 	result = -result;    return result;}//================================================================================class CClause {public:        int id;    vector<int> literals;    int num_fanouts;    int num_used;public:    CClause(void) {	num_fanouts = 0;	num_used = 0;    }    ~CClause(){}};class CVariable{public:    short value;    short in_clause_phase;     int antecedent;    int level ;    bool is_needed;    CVariable(void) { 	value = UNKNOWN; 	is_needed = false;	antecedent = -1; 	in_clause_phase = UNKNOWN;	level = -1;    }};struct cmp_var_level {    bool operator () (CVariable * v1, CVariable * v2)	{	    if (v1->level > v2->level)		return true;	    else if (v1->level < v2->level)		return false;	    else if ( (int)v1 > (int)v2)		return true;	    return false;	}};class CDatabase {private:    bool _output_core;    char * _counter_filename;    fstream _counter_stream;    ifstream _trace_stream;    FILE * _tmp_rsource_fp;        int	_num_init_clauses;    int _current_clause_count;    int _max_clause_count;    int _init_clause_count;    int _current_clause_id;    int _max_clause_id;    hash_map<int, CClause * > _clauses;    vector<CVariable> _variables;    int _conf_id;    vector<int> _conf_clause;    vector<int> _empty_r_source;public:    CDatabase(void) {	_counter_filename = NULL;	_output_core = false;	_tmp_rsource_fp = NULL;    }    ~CDatabase() {	if (_counter_filename != NULL)	    remove(_counter_filename);     }    int & num_init_clauses(void) { return _num_init_clauses; }    vector<CVariable> & variables(void) {return _variables; }    hash_map<int, CClause *> & clauses(void)  {return  _clauses; }    void init(char * cnf_file, char * trace_file);    void read_cnf(char * cnf_file);    void first_pass(void);    void second_pass(void);    bool real_verify(void);    bool verify(void);    void add_clause_by_literals(int cl_id, vector<int> & lits);    void add_clause_by_resolvents(int cl_id, vector<int> & resolvents);    void set_clause_fanout(int cl_id, int n);    void set_var_number(int nvar);    int recursive_find_level(int vid);    void set_init_cls_number (int n) {	_num_init_clauses = n;    }    int lit_value(int svar) { 	assert (_variables[svar>>1].value != UNKNOWN);	return _variables[svar>>1].value ^ (svar&0x1);    }        FILE * reverse_file(FILE * fp_in);    void print_file(FILE * fp);    void calculate_unsat_core(void);    void set_output_core(bool foo) {	_output_core = foo;	if (foo == true) {	    assert (_tmp_rsource_fp == NULL);	    _tmp_rsource_fp = tmpfile();	    if (_tmp_rsource_fp == NULL) {		cerr << "Can't Open Temp File " << endl;		exit(1);	    }	}    }       void dump(void);    };bool CDatabase::verify(void) {    cout << "Current Mem Usage\t\t\t" << get_mem_usage() << endl;    cout << "Begin First Pass, calculating fanout... " << endl;    first_pass();    cout << "Finished First Pass, Mem Usage \t\t" << get_mem_usage() << endl;    cout << "Init Num. Clause\t\t\t" << _num_init_clauses << endl;    cout << "Total Num. Clauses\t\t\t" << _max_clause_id << endl;    second_pass();    cout << "Finished Second Pass, Mem Usage \t" << get_mem_usage() << endl;    cout << "Init Clause Count\t\t\t" << _init_clause_count << endl;    cout << "Current Clause Count\t\t\t" << _current_clause_count << endl;    assert ((unsigned)_current_clause_count == _clauses.size());    cout << "Max Clause Count\t\t\t" << _max_clause_count << endl;    int r = real_verify();        cout << "Final Mem Usage \t\t\t" << get_mem_usage() << endl;    return r;}void CDatabase::set_clause_fanout(int cl_id, int n){    assert (_clauses.find(cl_id) != _clauses.end());    CClause * cl = _clauses[cl_id];    cl->num_fanouts = n;    if (n==0) { // no fanout, don't need it anyway	if (!(_output_core && cl_id < num_init_clauses())) { 	    delete cl;	    _clauses.erase(cl_id);	    -- _current_clause_count;	}    }}void CDatabase::add_clause_by_literals(int cl_id, vector<int> & lits){    CClause * cl = new CClause;    cl->id = cl_id;    cl->literals = lits;    _clauses[cl_id] = cl;    ++ _current_clause_count;    if (_current_clause_count > _max_clause_count)	_max_clause_count = _current_clause_count;    if (_current_clause_id % 10 == 0)	check_mem_out();}void CDatabase::add_clause_by_resolvents(int cl_id, vector<int> & resolvents){    vector<int> literals;    int cl1_id = resolvents[0];    assert (cl1_id < cl_id);    assert (_clauses.find(cl1_id) != _clauses.end());    CClause & cl1 = *_clauses[cl1_id];    for (unsigned i=0; i< cl1.literals.size(); ++i) {	int lit = cl1.literals[i];	int vid = (lit>>0x1);	int sign = (lit&0x1);	assert (_variables[vid].in_clause_phase == UNKNOWN);	_variables[vid].in_clause_phase = sign;	literals.push_back(lit);    }            for (unsigned i=1; i< resolvents.size(); ++i) {	int distance = 0;	int cl1_id = resolvents[i];	assert (cl1_id < cl_id);	assert (_clauses.find(cl1_id) != _clauses.end());	CClause & cl1 = *_clauses[cl1_id];		for (unsigned j=0; j< cl1.literals.size(); ++j) {	    int lit = cl1.literals[j];	    int vid = (lit>>0x1);	    int sign = (lit&0x1);	    if (_variables[vid].in_clause_phase == UNKNOWN) {		_variables[vid].in_clause_phase = sign;		literals.push_back(lit);	    }	    else if (_variables[vid].in_clause_phase != sign) {				//distance 1 literal		++distance;		_variables[vid].in_clause_phase = UNKNOWN;	    }	}	if (distance != 1) {	    cerr << "Resolve between two clauses with distance larger than 1" << endl;	    cerr << "The resulting clause is " << cl_id << endl;	    cerr << "Starting clause is " << resolvents[0] << endl;	    cerr << "One of the clause involved is " << cl1_id << endl;	    exit(1);	}    }    for (unsigned i=0; i< resolvents.size(); ++i) {	int key = resolvents[i];	CClause * cl = _clauses[key];	++ cl->num_used;	if (cl->num_used == cl->num_fanouts) {	    if (!(_output_core && key < num_init_clauses())) { 		delete cl;		_clauses.erase(key);		-- _current_clause_count;	    }	}    }    CClause * cl = new CClause;    for (unsigned i=0; i< literals.size(); ++i) {	int lit = literals[i];	int vid = (lit>>0x1);	int sign = (lit&0x1);	if (_variables[vid].in_clause_phase == UNKNOWN) 	    continue;	assert (_variables[vid].in_clause_phase == sign);	_variables[vid].in_clause_phase = UNKNOWN;	cl->literals.push_back(lit);    }    cl->id = cl_id;    _clauses[cl_id] = cl;    ++ _current_clause_count;    if (_current_clause_count > _max_clause_count) 	_max_clause_count = _current_clause_count;    if (_current_clause_id % 10 == 0)	check_mem_out();}void CDatabase::init(char * cnf_file, char * trace_file){    _num_init_clauses = 0;    _current_clause_count = 0;    _max_clause_count = 0;    _conf_id	= -1;    _current_clause_id = 0;    _counter_filename = strdup(tmpnam(NULL));    cout << "Temp file is " << _counter_filename << endl;    _counter_stream.open(_counter_filename, ios::in | ios::out | ios::trunc);	    if (!_counter_stream.is_open()) {	cerr << "Abort: Can't open a temp file " << endl;	exit(1);    }    _trace_stream.open(trace_file, fstream::in);    if (!_trace_stream.is_open()) {	cerr << "Abort: Can't open the trace file " << endl;	exit(1);    }    read_cnf(cnf_file);}void CDatabase::read_cnf (char * filename){    ifstream in_file(filename);    if (!in_file) {	cerr << "Can't open input CNF file " << filename << endl;	exit(1);    }    vector<char> buffer;    vector<int> literals;    bool header_encountered = false;    char token[WORD_LEN];	    while (!in_file.eof()) {	get_line(in_file, buffer);	char * ptr = &(*buffer.begin());	if (get_token(ptr, token)) {	    if (strcmp(token, "c")==0)		continue;	    else if (strcmp(token, "p")==0) {		assert (literals.empty());		assert (header_encountered == false);		get_token(ptr, token);		if (strcmp(token, "cnf") != 0) {		    cerr << "Format Error, p cnf NumVar NumCls " << endl;		    exit(1);		}		get_token(ptr, token);		int nvar = my_a2i(token);		set_var_number(nvar);		get_token(ptr, token);		int ncls = my_a2i(token);		set_init_cls_number(ncls);		header_encountered = true;		continue;	    }	    else {		int lit = my_a2i(token);		if (lit != 0) {		    if (lit > 0) 			literals.push_back(lit + lit);		    else 			literals.push_back(1- lit - lit);		}		else {		    add_clause_by_literals(_current_clause_id, literals);		    ++ _current_clause_id;		    literals.clear();		}	    }	}	while (get_token(ptr, token)) {	    int lit = my_a2i(token);	    if (lit != 0) {		if (lit > 0) 		    literals.push_back(lit + lit);		else 		    literals.push_back(1- lit - lit);	    }	    else {		add_clause_by_literals(_current_clause_id, literals);		++ _current_clause_id;		literals.clear();	    }	}    }    if (!literals.empty()) {	cerr << "Trailing numbers without termination " << endl;	exit(1);    }    if (clauses().size() != (unsigned)num_init_clauses()) 	cerr << "WARNING : Clause count inconsistant with the header " << endl;

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -