aboutsummaryrefslogtreecommitdiffstats
path: root/passes/sat/qbfsat.h
blob: 253cecce4c3284421c46897431bf0cc898dfe385 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
/* -*- c++ -*-
 *  yosys -- Yosys Open SYnthesis Suite
 *
 *  Copyright (C) 2020  Alberto Gonzalez <boqwxp@airmail.cc>
 *
 *  Permission to use, copy, modify, and/or distribute this software for any
 *  purpose with or without fee is hereby granted, provided that the above
 *  copyright notice and this permission notice appear in all copies.
 *
 *  THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
 *  WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
 *  MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
 *  ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
 *  WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
 *  ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
 *  OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
 *
 */

#ifndef QBFSAT_H
#define QBFSAT_H

#include "kernel/yosys.h"
#include <numeric>

YOSYS_NAMESPACE_BEGIN

struct QbfSolveOptions {
	bool specialize = false, specialize_from_file = false, write_solution = false, nocleanup = false;
	bool dump_final_smt2 = false, assume_outputs = false, assume_neg = false, nooptimize = false;
	bool nobisection = false, sat = false, unsat = false, show_smtbmc = false;
	enum Solver{Z3, Yices, CVC4, CVC5} solver = Yices;
	enum OptimizationLevel{O0, O1, O2} oflag = O0;
	dict<std::string, std::string> solver_options;
	int timeout = 0;
	std::string specialize_soln_file = "";
	std::string write_soln_soln_file = "";
	std::string dump_final_smt2_file = "";
	size_t argidx = 0;

	std::string get_solver_name() const {
		if (solver == Solver::Z3)
			return "z3";
		else if (solver == Solver::Yices)
			return "yices";
		else if (solver == Solver::CVC4)
			return "cvc4";
		else if (solver == Solver::CVC5)
			return "cvc5";

		log_cmd_error("unknown solver specified.\n");
		return "";
	}
};

struct QbfSolutionType {
	std::vector<std::string> stdout_lines = {};
	dict<pool<std::string>, std::string> hole_to_value = {};
	double solver_time = 0;
	bool sat = false;
	bool unknown = true; //true if neither 'sat' nor 'unsat'

	dict<std::pair<pool<std::string>, int>, RTLIL::SigBit> get_hole_loc_idx_sigbit_map(RTLIL::Module *module) const {
		dict<std::pair<pool<std::string>, int>, RTLIL::SigBit> hole_loc_idx_to_sigbit;
		pool<RTLIL::SigBit> anyconst_sigbits;
		dict<RTLIL::SigBit, RTLIL::SigBit> anyconst_sigbit_to_wire_sigbit;

		for (auto cell : module->cells()) {
			pool<std::string> cell_src = cell->get_strpool_attribute(ID::src);
			auto pos = hole_to_value.find(cell_src);
			if (pos != hole_to_value.end() && cell->type.in("$anyconst", "$anyseq")) {
				RTLIL::SigSpec port_y = cell->getPort(ID::Y);
				for (int i = GetSize(port_y) - 1; i >= 0; --i) {
					hole_loc_idx_to_sigbit[std::make_pair(pos->first, i)] = port_y[i];
					anyconst_sigbits.insert(port_y[i]);
				}
			}
		}

		for (auto &conn : module->connections()) {
			auto lhs = conn.first;
			auto rhs = conn.second;
			for (auto i = 0; i < GetSize(rhs); ++i) {
				if (anyconst_sigbits[rhs[i]]) {
					auto pos = anyconst_sigbit_to_wire_sigbit.find(rhs[i]);
					if (pos != anyconst_sigbit_to_wire_sigbit.end())
						log_cmd_error("conflicting names for hole $anyconst sigbit %s\n", log_signal(rhs[i]));
					anyconst_sigbit_to_wire_sigbit[rhs[i]] = lhs[i];
				}
			}
		}

		for (auto &it : hole_loc_idx_to_sigbit) {
			auto pos = anyconst_sigbit_to_wire_sigbit.find(it.second);
			if (pos != anyconst_sigbit_to_wire_sigbit.end())
				it.second = pos->second;
		}

		return hole_loc_idx_to_sigbit;
	}

	void dump_model(RTLIL::Module *module) const {
		log("Satisfiable model:\n");
		auto hole_loc_idx_to_sigbit = get_hole_loc_idx_sigbit_map(module);
		for (auto &it : hole_to_value) {
			pool<std::string> hole_loc = it.first;
			std::string hole_value = it.second;

			for (unsigned int i = 0; i < hole_value.size(); ++i) {
				int bit_idx = GetSize(hole_value) - 1 - i;
				auto it = hole_loc_idx_to_sigbit.find(std::make_pair(hole_loc, i));
				log_assert(it != hole_loc_idx_to_sigbit.end());

				RTLIL::SigBit hole_sigbit = it->second;
				log("\t%s = 1'b%c\n", log_signal(hole_sigbit), hole_value[bit_idx]);
			}
		}
	}

	void write_solution(RTLIL::Module *module, const std::string &file) const {
		std::ofstream fout(file.c_str());
		if (!fout)
			log_cmd_error("could not open solution file for writing.\n");

		//There is a question here: How exactly shall we identify holes?
		//There are at least two reasonable options:
		//1. By the source location of the $anyconst cells
		//2. By the name(s) of the wire(s) connected to each SigBit of the $anyconst cell->getPort(ID::Y) SigSpec.
		//
		//Option 1 has the benefit of being very precise.  There is very limited potential for confusion, as long
		//as the source attribute has been set.  However, if the source attribute is not set, this won't work.
		//More importantly, we want to have the ability to port hole assignments to other modules with compatible
		//hole names and widths.  Obviously in those cases source locations of the $anyconst cells will not match.
		//
		//Option 2 has the benefits previously described, but wire names can be changed automatically by 
		//optimization or techmapping passes, especially when (ex/im)porting from BLIF for optimization with ABC.
		//
		//The approach taken here is to allow both options.  We write the assignment information for each bit of
		//the solution on a separate line.  Each line is of one of two forms:
		//
		//location bit name = value
		//location bit name [offset] = value
		//
		//where '[', ']', and '=' are literal symbols, "location" is the $anyconst cell source location attribute,
		//"bit" is the index of the $anyconst cell, "name" is the `wire->name` field of the SigBit corresponding
		//to the current bit of the $anyconst cell->getPort(ID::Y), "offset" is the `offset` field of that same
		//SigBit, and "value", which is either '0' or '1', represents the assignment for that bit.
		auto hole_loc_idx_to_sigbit = get_hole_loc_idx_sigbit_map(module);
		for (auto &x : hole_to_value) {
			std::string src_as_str = std::accumulate(x.first.begin(), x.first.end(), std::string(), [](const std::string &a, const std::string &b){return a + "|" + b;});
			for (auto i = 0; i < GetSize(x.second); ++i)
				fout << src_as_str.c_str() << " " << i << " " << log_signal(hole_loc_idx_to_sigbit[std::make_pair(x.first, i)]) << " = " << x.second[GetSize(x.second) - 1 - i] << std::endl;
		}
	}

	void recover_solution() {
		std::regex sat_regex = YS_REGEX_COMPILE("Status: PASSED");
		std::regex unsat_regex = YS_REGEX_COMPILE("Solver Error.*model is not available");
		std::regex unsat_regex2 = YS_REGEX_COMPILE("Status: FAILED");
		std::regex timeout_regex = YS_REGEX_COMPILE("No solution found! \\(timeout\\)");
		std::regex timeout_regex2 = YS_REGEX_COMPILE("No solution found! \\(interrupted\\)");
		std::regex unknown_regex = YS_REGEX_COMPILE("No solution found! \\(unknown\\)");
		std::regex unknown_regex2 = YS_REGEX_COMPILE("Unexpected EOF response from solver");
		std::regex memout_regex = YS_REGEX_COMPILE("Solver Error:.*error \"out of memory\"");
		std::regex hole_value_regex = YS_REGEX_COMPILE_WITH_SUBS("Value for anyconst in [a-zA-Z0-9_]* \\(([^:]*:[^\\)]*)\\): (.*)");
#ifndef NDEBUG
		std::regex hole_loc_regex = YS_REGEX_COMPILE("[^:]*:[0-9]+.[0-9]+-[0-9]+.[0-9]+");
		std::regex hole_val_regex = YS_REGEX_COMPILE("[0-9]+");
#endif
		std::smatch m;
		bool sat_regex_found = false;
		bool unsat_regex_found = false;
		dict<std::string, bool> hole_value_recovered;
		for (const std::string &x : stdout_lines) {
			if(std::regex_search(x, m, hole_value_regex)) {
				std::string loc = m[1].str();
				std::string val = m[2].str();
#ifndef NDEBUG
				log_assert(std::regex_search(loc, hole_loc_regex));
				log_assert(std::regex_search(val, hole_val_regex));
#endif
				auto locs = split_tokens(loc, "|");
				pool<std::string> loc_pool(locs.begin(), locs.end());
				hole_to_value[loc_pool] = val;
			}
			else if (std::regex_search(x, sat_regex)) {
				sat_regex_found = true;
				sat = true;
				unknown = false;
			}
			else if (std::regex_search(x, unsat_regex)) {
				unsat_regex_found = true;
				sat = false;
				unknown = false;
			}
			else if (std::regex_search(x, memout_regex)) {
				unknown = true;
				log_warning("solver ran out of memory\n");
			}
			else if (std::regex_search(x, timeout_regex)) {
				unknown = true;
				log_warning("solver timed out\n");
			}
			else if (std::regex_search(x, timeout_regex2)) {
				unknown = true;
				log_warning("solver timed out\n");
			}
			else if (std::regex_search(x, unknown_regex)) {
				unknown = true;
				log_warning("solver returned \"unknown\"\n");
			}
			else if (std::regex_search(x, unsat_regex2)) {
				unsat_regex_found = true;
				sat = false;
				unknown = false;
			}
			else if (std::regex_search(x, unknown_regex2)) {
				unknown = true;
			}
		}
		log_assert(!unknown && sat? sat_regex_found : true);
		log_assert(!unknown && !sat? unsat_regex_found : true);
	}
};

void print_proof_failed()
{
	log("\n");
	log("   ______                   ___       ___       _ _            _ _ \n");
	log("  (_____ \\                 / __)     / __)     (_) |          | | |\n");
	log("   _____) )___ ___   ___ _| |__    _| |__ _____ _| | _____  __| | |\n");
	log("  |  ____/ ___) _ \\ / _ (_   __)  (_   __|____ | | || ___ |/ _  |_|\n");
	log("  | |   | |  | |_| | |_| || |       | |  / ___ | | || ____( (_| |_ \n");
	log("  |_|   |_|   \\___/ \\___/ |_|       |_|  \\_____|_|\\_)_____)\\____|_|\n");
	log("\n");
}

void print_qed()
{
	log("\n");
	log("                  /$$$$$$      /$$$$$$$$     /$$$$$$$    \n");
	log("                 /$$__  $$    | $$_____/    | $$__  $$   \n");
	log("                | $$  \\ $$    | $$          | $$  \\ $$   \n");
	log("                | $$  | $$    | $$$$$       | $$  | $$   \n");
	log("                | $$  | $$    | $$__/       | $$  | $$   \n");
	log("                | $$/$$ $$    | $$          | $$  | $$   \n");
	log("                |  $$$$$$/ /$$| $$$$$$$$ /$$| $$$$$$$//$$\n");
	log("                 \\____ $$$|__/|________/|__/|_______/|__/\n");
	log("                       \\__/                              \n");
	log("\n");
}

YOSYS_NAMESPACE_END

#endif