-
Notifications
You must be signed in to change notification settings - Fork 0
/
bmc.py
75 lines (63 loc) · 2.84 KB
/
bmc.py
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
from z3 import *
class BMC:
def __init__(self, primary_inputs, literals, primes, init, trans, post, pv2next, primes_inp, filename):
'''
:param primary_inputs:
:param literals: Boolean Variables
:param primes: The Post Condition Variable
:param init: The initial State
:param trans: Transition Function
:param post: The Safety Property
'''
self.primary_inputs = primary_inputs
self.init = init
self.filename = filename
self.trans = trans
#self.literals = literals
self.literals = literals + primary_inputs
self.items = self.primary_inputs + self.literals
self.inp_prime = primes_inp
self.lMap = {str(l): l for l in self.items}
self.post = post
self.frames = list()
#self.primes = primes
self.primes = primes + [Bool(str(pi)+'_prime') for pi in primary_inputs]
#self.primeMap = [(literals[i], primes[i]) for i in range(len(literals))]
self.primeMap = [(self.literals[i], self.primes[i]) for i in range(len(self.literals))]
self.pv2next = pv2next
self.initprime = substitute(self.init.cube(), self.primeMap)
self.inp_map = [(primary_inputs[i], primes_inp[i]) for i in range(len(primes_inp))]
self.vardict = dict()
#entend literals and prime
for item in self.inp_map:
literals.append(item[0])
primes.append(item[1])
def vardef(self, n:str):
if n in self.vardict:
return self.vardict[n]
v = Bool(n)
self.vardict[n] = v
return v
def setup(self):
self.slv = Solver()
initmap = [(self.literals[i], self.vardef(str(self.literals[i])+"_0")) for i in range(len(self.literals))]
self.slv.add(substitute(self.init.cube(), initmap))
self.cnt = 0
def get_map(self, idx):
curr_map = [(self.literals[i], self.vardef(str(self.literals[i])+"_"+str(idx))) for i in range(len(self.literals))]
next_map = [(self.primes[i], self.vardef(str(self.literals[i])+"_"+str(idx+1))) for i in range(len(self.literals))]
return curr_map + next_map
def unroll(self):
idx = self.cnt
var_map = self.get_map(idx)
self.slv.add(substitute(self.trans.cube(), var_map) )
# Constrain the input variables as well
# input_map = [(self.primary_inputs[i], self.vardef(str(self.primary_inputs[i])+"_"+str(idx))) for i in range(len(self.primary_inputs))]
# self.slv.add(substitute(And([pi == pi_prime for pi, pi_prime in self.inp_map]), input_map))
self.cnt += 1
def add(self, constraint):
idx = self.cnt
var_map = self.get_map(idx)
self.slv.add(substitute(constraint, var_map))
def check(self):
return self.slv.check()