atpg-ls/ls.cpp

510 lines
14 KiB
C++
Raw Permalink Normal View History

2023-02-16 18:51:31 +08:00
#include "circuit.h"
2023-02-12 18:14:12 +08:00
2023-02-19 19:42:50 +08:00
#include <queue>
#include <unordered_set>
2023-02-21 19:07:40 +08:00
#include <unordered_map>
2023-02-19 19:42:50 +08:00
#include <algorithm>
#include "assert.h"
2023-03-11 07:02:02 +00:00
#include <chrono>
2023-03-17 05:37:29 +00:00
#include "clause.h"
2023-03-21 08:14:37 +00:00
#include "CCAnr/ccanr.h"
2023-03-17 05:37:29 +00:00
2023-03-13 05:44:49 +00:00
bool Circuit::local_search() {
2023-03-11 07:02:02 +00:00
2023-03-22 02:37:08 +00:00
//printf("[start ls]\n");
2023-03-13 05:44:49 +00:00
ls_reset_data();
2023-02-19 19:42:50 +08:00
2023-03-13 05:44:49 +00:00
ls_init_stems();
2023-02-12 18:14:12 +08:00
2023-03-13 05:44:49 +00:00
ls_init_weight();
2023-02-20 13:08:25 +08:00
2023-03-21 17:14:32 +08:00
CCAnr::module_reset();
2023-03-21 08:14:37 +00:00
2023-03-13 05:44:49 +00:00
ls_random_circuit();
2023-02-20 13:08:25 +08:00
2023-03-22 02:37:08 +00:00
//printf("stem-cnt: %d\n", stem_total_cnt);
2023-03-13 05:44:49 +00:00
for(int i=0; i<MAX_STEPS; i++) {
2023-02-21 19:07:40 +08:00
2023-03-11 07:02:02 +00:00
auto start = std::chrono::system_clock::now();
2023-03-21 08:14:37 +00:00
//printf("[FLIP] stem: %lld, fault:%lld, stem_cnt: %lld, fault_cnt:%lld, fpl_score: %lld citcuit-score: %lld\n", stem_total_cost, fault_total_weight, stems.size() - stem_total_cnt, fault_total_cnt, fault_propagate_score, ls_circuit_score());
2023-03-11 07:02:02 +00:00
2023-03-22 02:37:08 +00:00
int id;
double p = 0.8;
// if(rand() % 10000 <= p * 10000) {
// id = CCAnr::module_pick_var();
// } else {
// }
id = ls_pick();
2023-03-13 05:44:49 +00:00
2023-03-21 17:14:32 +08:00
CCAnr::module_flip_var(id);
2023-03-22 02:37:08 +00:00
ls_flip_var(id);
2023-03-13 05:44:49 +00:00
if(stem_total_cnt == stems.size()) {
printf("FIND SOLUTION!\n");
2023-03-13 05:44:49 +00:00
break;
2023-02-21 19:07:40 +08:00
}
2023-03-21 08:14:37 +00:00
//assert(is_valid_circuit());
2023-02-21 19:07:40 +08:00
2023-03-13 05:44:49 +00:00
auto end = std::chrono::system_clock::now();
std::chrono::duration<double> elapsed_seconds = end - start;
update_cnt++;
update_time += elapsed_seconds.count();
2023-02-21 19:07:40 +08:00
2023-03-13 05:44:49 +00:00
//printf("[UP] flip: %lld, stem: %lld, fault:%lld. flip_cnt: %lld, stem_cnt: %lld, fault_cnt:%lld\n", flip_total_weight, stem_total_weight, fault_total_weight, flip_total_cnt, stem_total_cnt, fault_total_cnt);
}
2023-02-21 19:07:40 +08:00
2023-03-13 05:44:49 +00:00
ls_statistics();
2023-02-21 19:07:40 +08:00
2023-03-13 05:44:49 +00:00
return true;
}
2023-02-21 19:07:40 +08:00
2023-03-13 05:44:49 +00:00
void Circuit::ls_statistics() {
2023-02-21 19:07:40 +08:00
static int pattern = 0;
2023-03-13 05:44:49 +00:00
int last_undetect = global_fault_undetected_count;
2023-03-11 07:02:02 +00:00
2023-03-13 05:44:49 +00:00
for(Gate* g : gates) {
if(g->fault_detected[0] && !g->global_fault_detected[0]) {
global_fault_undetected_count--;
g->global_fault_detected[0] = 1;
}
if(g->fault_detected[1] && !g->global_fault_detected[1]) {
global_fault_undetected_count--;
g->global_fault_detected[1] = 1;
}
}
2023-02-21 19:07:40 +08:00
if(global_fault_undetected_count - last_undetect < 0) {
pattern++;
}
2023-03-13 10:54:51 +00:00
printf("coverage: %.2f%% undected_fault: %d delta: %d total-pt:%d\n",
(gates.size() * 2.0 - global_fault_undetected_count) / (gates.size() * 2.0) * 100,
global_fault_undetected_count, global_fault_undetected_count - last_undetect, pattern);
2023-03-13 10:54:51 +00:00
printf("flip-cnt: %d flip-time: %.3fs update-cnt: %d update-time: %.3fs\n", flip_cnt, flip_time, update_cnt, update_time);
printf("time-per-update: %.2fms\n", update_time / update_cnt * 1000);
2023-03-13 05:44:49 +00:00
}
2023-02-21 19:07:40 +08:00
2023-03-13 05:44:49 +00:00
void Circuit::ls_update_weight() {
2023-02-21 19:07:40 +08:00
2023-03-13 05:44:49 +00:00
if(rand() % 10000 <= SP * 10000) {
2023-03-17 05:37:29 +00:00
for(auto& clause : ClauseLS::satisfied_clauses) {
if(clause->weight - CLAUSE_FALSIFIED_INC < 1) continue;
clause->weight -= CLAUSE_FALSIFIED_INC;
}
2023-02-21 19:07:40 +08:00
2023-03-17 05:37:29 +00:00
for(Gate* g : gates) {
2023-03-13 05:44:49 +00:00
if(g->fault_detected[0] && g->fault_weight[0] - FAULT_INC >= 1) {
g->fault_weight[0] -= FAULT_INC;
fault_propagate_score -= FAULT_INC * (g->fault_propagate_length[0]);
2023-02-21 19:07:40 +08:00
}
2023-03-13 05:44:49 +00:00
if(g->fault_detected[1] && g->fault_weight[1] - FAULT_INC >= 1) {
g->fault_weight[1] -= FAULT_INC;
fault_propagate_score -= FAULT_INC * (g->fault_propagate_length[1]);
}
}
} else {
2023-03-17 05:37:29 +00:00
for(auto& clause : ClauseLS::falsified_clauses) {
if(clause->weight + CLAUSE_FALSIFIED_INC > CLAUSE_FALSIFIED_MAX) continue;
clause->weight += CLAUSE_FALSIFIED_INC;
clause->total_cost += CLAUSE_FALSIFIED_INC;
}
2023-02-21 19:07:40 +08:00
2023-03-17 05:37:29 +00:00
for(Gate* g : gates) {
2023-03-13 05:44:49 +00:00
if(!g->fault_detected[0] && g->fault_weight[0] > 0 && (g->fault_weight[0] + FAULT_INC < FAULT_WEIGHT_MAX)) {
g->fault_weight[0] += FAULT_INC;
fault_propagate_score += FAULT_INC * (g->fault_propagate_length[0]);
2023-02-21 19:07:40 +08:00
}
2023-03-13 05:44:49 +00:00
if(!g->fault_detected[1] && g->fault_weight[1] > 0 && (g->fault_weight[1] + FAULT_INC < FAULT_WEIGHT_MAX)) {
g->fault_weight[1] += FAULT_INC;
fault_propagate_score += FAULT_INC * (g->fault_propagate_length[1]);
2023-02-21 19:07:40 +08:00
}
2023-03-13 05:44:49 +00:00
}
}
}
2023-02-21 19:07:40 +08:00
2023-03-17 05:37:29 +00:00
int Circuit::ls_pick() {
int var = -1;
2023-03-13 05:44:49 +00:00
ll max_score = 0;
2023-02-12 18:14:12 +08:00
2023-03-17 05:37:29 +00:00
for(int i=0; i<SAMPLING_COUNT; i++) {
int t_var = rand() % ClauseLS::num_vars + 1;
if(!ClauseLS::CC[t_var]) continue;
2023-02-21 19:07:40 +08:00
2023-03-17 05:37:29 +00:00
ll t_score = ls_pick_score(t_var);
2023-03-13 10:54:51 +00:00
2023-03-13 05:44:49 +00:00
if(t_score > max_score) {
max_score = t_score;
2023-03-17 05:37:29 +00:00
var = t_var;
2023-02-23 11:00:24 +08:00
}
}
2023-02-21 19:07:40 +08:00
2023-03-17 05:37:29 +00:00
if(var == -1) {
ls_update_weight();
2023-02-21 19:07:40 +08:00
2023-03-21 08:14:37 +00:00
//printf("[UP] stem: %lld, fault:%lld, stem_cnt: %lld, fault_cnt:%lld, fpl_score: %lld citcuit-score: %lld\n", stem_total_cost, fault_total_weight, stems.size() - stem_total_cnt, fault_total_cnt, fault_propagate_score, ls_circuit_score());
2023-02-23 11:00:24 +08:00
2023-03-17 05:37:29 +00:00
auto it = std::next(ClauseLS::falsified_clauses.begin(), rand() % ClauseLS::falsified_clauses.size());
2023-02-21 19:07:40 +08:00
2023-03-17 05:37:29 +00:00
auto& lits = (*it)->lits;
2023-03-13 05:44:49 +00:00
2023-03-17 05:37:29 +00:00
var = abs(lits[rand()%lits.size()]);
2023-03-13 05:44:49 +00:00
}
2023-03-21 08:14:37 +00:00
int var_score = ls_pick_score(var);
2023-03-22 02:37:08 +00:00
assert(var != -1);;
2023-03-21 08:36:53 +00:00
2023-03-22 02:37:08 +00:00
return var;
2023-02-23 11:00:24 +08:00
}
2023-03-13 05:44:49 +00:00
void Circuit::ls_init_stems() {
stems.clear();
2023-02-23 05:42:34 +00:00
2023-03-13 05:44:49 +00:00
for(Gate* g : gates) {
if(g->pi || g->po) {
g->stem = true;
}
if(!g->global_fault_detected[0] || !g->global_fault_detected[0]) {
g->stem = true;
}
2023-02-23 05:42:34 +00:00
2023-03-13 05:44:49 +00:00
if(g->fan_outs.size() >= 2) {
g->stem = true;
2023-02-23 05:42:34 +00:00
}
2023-02-21 19:07:40 +08:00
2023-03-13 05:44:49 +00:00
if(g->stem) {
stems.push_back(g);
}
}
2023-02-23 11:00:24 +08:00
2023-03-13 05:44:49 +00:00
for(Gate *g : gates) {
if(g->pi) continue;
std::queue<Gate*> q;
std::unordered_map<Gate*, bool> used;
q.push(g);
while(!q.empty()) {
Gate* now = q.front();
q.pop();
for(Gate* in : now->fan_ins) {
if(in->stem) {
g->pre_stems.push_back(in);
} else if(!used[in]) {
used[in] = true;
q.push(in);
}
2023-02-21 19:07:40 +08:00
}
2023-03-13 05:44:49 +00:00
}
}
2023-02-21 19:07:40 +08:00
2023-03-13 05:44:49 +00:00
for(Gate *g : gates) {
if(g->po) continue;
std::queue<Gate*> q;
std::unordered_map<Gate*, bool> used;
q.push(g);
while(!q.empty()) {
Gate* now = q.front();
q.pop();
for(Gate* out : now->fan_outs) {
if(out->stem) {
g->suc_stems.push_back(out);
} else if(!used[out]) {
used[out] = true;
q.push(out);
}
2023-02-21 19:07:40 +08:00
}
}
}
}
2023-03-17 05:37:29 +00:00
void Circuit::ls_flip_var(int var) {
ClauseLS::flip(var);
if(id2gate.count(var) && id2gate[var]->stem) {
ls_flip_stem(id2gate[var]);
}
}
ll Circuit::ls_pick_score(int var) {
2023-02-21 19:07:40 +08:00
2023-03-13 05:44:49 +00:00
ll old_score = ls_circuit_score();
2023-02-23 05:42:34 +00:00
2023-03-17 05:37:29 +00:00
ls_flip_var(var);
2023-02-20 14:41:19 +08:00
2023-03-13 05:44:49 +00:00
ll new_score = ls_circuit_score();
2023-02-21 19:07:40 +08:00
2023-03-17 05:37:29 +00:00
ls_flip_var(var);
2023-03-13 10:54:51 +00:00
2023-03-22 02:37:08 +00:00
assert(old_score == ls_circuit_score());
2023-03-13 10:54:51 +00:00
2023-02-21 19:07:40 +08:00
return new_score - old_score;
}
2023-03-13 05:44:49 +00:00
ll Circuit::ls_circuit_score() {
2023-03-21 08:14:37 +00:00
ll score = - Clause::total_cost + fault_propagate_score + fault_total_weight;
//ll score = - Clause::total_cost;
2023-02-20 14:41:19 +08:00
return score;
}
2023-03-13 05:44:49 +00:00
void Circuit::ls_init_weight() {
2023-02-20 13:08:25 +08:00
for(Gate* s : stems) {
2023-03-13 05:44:49 +00:00
s->stem_weight = 1;
stem_total_cost += s->stem_weight;
2023-02-20 13:08:25 +08:00
}
2023-03-11 07:02:02 +00:00
2023-03-13 05:44:49 +00:00
for(Gate* g : gates) {
g->fault_weight[0] = !g->global_fault_detected[0];
g->fault_weight[1] = !g->global_fault_detected[1];
2023-02-20 13:08:25 +08:00
}
}
2023-03-13 05:44:49 +00:00
void Circuit::ls_random_circuit() {
2023-02-23 11:00:24 +08:00
// random init clauseLS
for(int i=1; i<=ClauseLS::num_vars; i++) {
2023-03-21 08:36:53 +00:00
int new_v = CCAnr::module_cur_soln()[i];
if(new_v != ClauseLS::lit_value[i]) {
2023-03-22 02:37:08 +00:00
//ls_flip_var(i);
ClauseLS::flip(i);
}
}
2023-03-13 05:44:49 +00:00
// init assignment
2023-02-19 19:42:50 +08:00
for(Gate* s : stems) {
2023-03-17 05:37:29 +00:00
s->value = ClauseLS::lit_value[s->id];
2023-02-19 19:42:50 +08:00
}
2023-03-13 05:44:49 +00:00
// recal value by topo
for(Gate *g : gates) {
if(g->stem) {
g->stem_satisfied = (g->recal_value() == g->value);
if(g->stem_satisfied) {
stem_total_cost -= g->stem_weight;
stem_total_cnt++;
}
} else {
g->value = g->recal_value();
}
2023-02-12 18:14:12 +08:00
}
2023-02-20 13:08:25 +08:00
2023-03-13 05:44:49 +00:00
// recal fault by rtopo
for(Gate* g : rtopo_gates) {
g->recal_fault(g->fault_detected);
if(g->fault_detected[0]) {
fault_total_weight += g->fault_weight[0];
fault_total_cnt++;
}
if(g->fault_detected[1]) {
fault_total_weight += g->fault_weight[1];
fault_total_cnt++;
}
2023-02-20 13:08:25 +08:00
2023-03-13 05:44:49 +00:00
g->propagate = (g->fault_detected[0] || g->fault_detected[1]);
2023-02-21 19:07:40 +08:00
2023-03-13 05:44:49 +00:00
g->recal_propagate_len(g->fault_propagate_length);
2023-02-20 13:08:25 +08:00
2023-03-13 05:44:49 +00:00
fault_propagate_score += g->fault_weight[0] * g->fault_propagate_length[0];
fault_propagate_score += g->fault_weight[1] * g->fault_propagate_length[1];
2023-02-21 19:07:40 +08:00
}
2023-03-22 02:37:08 +00:00
assert(is_valid_circuit());
2023-03-13 05:44:49 +00:00
}
2023-03-09 13:49:18 +08:00
2023-03-13 05:44:49 +00:00
void Circuit::ls_reset_data() {
2023-03-09 02:51:10 +00:00
2023-03-13 05:44:49 +00:00
stems.clear();
2023-03-06 11:05:30 +00:00
2023-03-13 05:44:49 +00:00
fault_propagate_score = 0;
2023-02-21 19:07:40 +08:00
2023-03-13 05:44:49 +00:00
stem_total_cost = 0;
2023-02-21 19:07:40 +08:00
stem_total_cnt = 0;
fault_total_weight = 0;
fault_total_cnt = 0;
2023-03-06 11:05:30 +00:00
for(Gate *g : gates) {
2023-03-13 05:44:49 +00:00
g->CC = 1;
g->stem = 0;
g->propagate = 0;
g->stem_weight = 0;
g->stem_satisfied = 0;
2023-02-24 16:01:09 +08:00
2023-03-13 05:44:49 +00:00
g->fault_weight[0] = g->fault_weight[1] = 0;
g->fault_detected[0] = g->fault_detected[1] = 0;
2023-02-24 16:01:09 +08:00
2023-03-13 05:44:49 +00:00
g->fault_propagate_length[0] = 0;
g->fault_propagate_length[1] = 0;
2023-02-24 16:01:09 +08:00
}
2023-03-13 05:44:49 +00:00
}
2023-02-24 16:01:09 +08:00
2023-03-17 05:37:29 +00:00
void Circuit::ls_flip_stem(Gate* stem) {
2023-03-13 10:54:51 +00:00
stem->value = !stem->value;
2023-03-13 05:44:49 +00:00
// update CC
stem->CC = 0;
for(Gate* pre : stem->pre_stems) {
pre->CC = 1;
2023-02-24 16:01:09 +08:00
}
2023-03-13 05:44:49 +00:00
for(Gate* suc : stem->suc_stems) {
suc->CC = 1;
}
2023-02-24 16:01:09 +08:00
2023-03-13 05:44:49 +00:00
// update value
bool new_stem_satisfied = (stem->recal_value() == stem->value);
2023-03-06 11:05:30 +00:00
2023-03-13 05:44:49 +00:00
if(new_stem_satisfied && !stem->stem_satisfied){
stem->stem_satisfied = true;
stem_total_cost -= stem->stem_weight;
stem_total_cnt += 1;
}
2023-03-03 07:33:14 +00:00
2023-03-13 05:44:49 +00:00
if(!new_stem_satisfied && stem->stem_satisfied){
stem->stem_satisfied = false;
stem_total_cost += stem->stem_weight;
stem_total_cnt -= 1;
}
2023-03-03 07:33:14 +00:00
2023-03-13 05:44:49 +00:00
// update po fault
if(stem->po) {
stem->propagate = true;
if(!stem->fault_detected[!stem->value]) {
fault_total_weight += stem->fault_weight[!stem->value];
fault_total_cnt += 1;
stem->fault_detected[!stem->value] = true;
2023-02-24 16:01:09 +08:00
}
2023-03-13 05:44:49 +00:00
if(stem->fault_detected[stem->value]) {
fault_total_weight -= stem->fault_weight[stem->value];
2023-02-24 16:01:09 +08:00
fault_total_cnt -= 1;
2023-03-13 05:44:49 +00:00
stem->fault_detected[stem->value] = false;
2023-02-24 16:01:09 +08:00
}
}
2023-03-13 05:44:49 +00:00
static std::queue<Gate*> q1;
static std::unordered_map<Gate*, int> used1;
static std::queue<Gate*> q2;
static std::unordered_map<Gate*, int> used2;
used1.clear();
used2.clear();
2023-02-24 16:01:09 +08:00
2023-03-13 05:44:49 +00:00
q1.push(stem);
2023-02-24 16:01:09 +08:00
2023-03-13 05:44:49 +00:00
while(!q1.empty()) {
Gate* g = q1.front();
q1.pop();
used1[g] = false;
for(Gate* suc : g->suc_stems) {
if(!used2[suc]) {
used2[suc] = true;
q2.push(suc);
2023-02-24 16:01:09 +08:00
}
}
2023-03-13 05:44:49 +00:00
for(Gate* out : g->fan_outs) {
if(out->stem) {
Gate* stem = out;
bool new_stem_satisfied = (stem->recal_value() == stem->value);
if(new_stem_satisfied && !stem->stem_satisfied){
stem->stem_satisfied = true;
stem_total_cost -= stem->stem_weight;
stem_total_cnt += 1;
}
2023-03-06 11:05:30 +00:00
2023-03-13 05:44:49 +00:00
if(!new_stem_satisfied && stem->stem_satisfied){
stem->stem_satisfied = false;
stem_total_cost += stem->stem_weight;
stem_total_cnt -= 1;
2023-03-06 11:05:30 +00:00
}
2023-03-13 05:44:49 +00:00
continue;
2023-03-06 11:05:30 +00:00
}
2023-03-13 05:44:49 +00:00
int new_value = out->recal_value();
if(new_value == out->value) continue;
out->value = new_value;
if(!used1[out]) {
used1[out] = true;
q1.push(out);
}
}
}
2023-03-06 11:05:30 +00:00
2023-03-13 10:54:51 +00:00
q2.push(stem);
used2[stem] = true;
2023-03-13 05:44:49 +00:00
while(!q2.empty()) {
Gate *g = q2.front();
q2.pop();
2023-02-24 16:01:09 +08:00
2023-03-13 05:44:49 +00:00
used2[g] = false;
2023-02-24 16:01:09 +08:00
2023-03-13 05:44:49 +00:00
for(Gate* in : g->fan_ins) {
2023-03-06 11:05:30 +00:00
2023-03-13 05:44:49 +00:00
bool update = false;
2023-03-11 07:02:02 +00:00
2023-03-13 05:44:49 +00:00
bool fd[2];
in->recal_fault(fd);
2023-03-06 11:05:30 +00:00
2023-03-13 05:44:49 +00:00
if(fd[0] != in->fault_detected[0]) {
update = true;
2023-03-13 10:54:51 +00:00
if(fd[0]) {
2023-03-13 05:44:49 +00:00
fault_total_weight += in->fault_weight[0];
2023-02-24 16:01:09 +08:00
fault_total_cnt += 1;
} else {
2023-03-13 10:54:51 +00:00
fault_total_weight -= in->fault_weight[0];
2023-02-24 16:01:09 +08:00
fault_total_cnt -= 1;
}
2023-03-13 05:44:49 +00:00
in->fault_detected[0] = fd[0];
2023-02-24 16:01:09 +08:00
}
2023-03-13 05:44:49 +00:00
if(fd[1] != in->fault_detected[1]) {
update = true;
2023-03-13 10:54:51 +00:00
if(fd[1]) {
2023-03-13 05:44:49 +00:00
fault_total_weight += in->fault_weight[1];
2023-02-24 16:01:09 +08:00
fault_total_cnt += 1;
} else {
2023-03-13 05:44:49 +00:00
fault_total_weight -= in->fault_weight[1];
2023-02-24 16:01:09 +08:00
fault_total_cnt -= 1;
}
2023-03-13 05:44:49 +00:00
in->fault_detected[1] = fd[1];
}
in->propagate = (in->fault_detected[0] || in->fault_detected[1]);
int fpl[2];
in->recal_propagate_len(fpl);
if(fpl[0] != in->fault_propagate_length[0] || fpl[1] != in->fault_propagate_length[1]) {
update = true;
fault_propagate_score += in->fault_weight[0] * (fpl[0] - in->fault_propagate_length[0]);
fault_propagate_score += in->fault_weight[1] * (fpl[1] - in->fault_propagate_length[1]);
in->fault_propagate_length[0] = fpl[0];
in->fault_propagate_length[1] = fpl[1];
2023-02-24 16:01:09 +08:00
}
2023-03-13 10:54:51 +00:00
if(update && !used2[in]) {
2023-03-13 05:44:49 +00:00
used2[in] = true;
q2.push(in);
2023-02-24 16:01:09 +08:00
}
}
}
2023-02-12 18:14:12 +08:00
}