FSM n
represents a function BitStream → ⋯ → BitStream → BitStream
,
where n
is the number of BitStream
arguments,
as a finite state machine.
- α : Type
The arity of the (finite) type
α
determines how many bits the internal carry state of this FSM has - i : FinEnum self.α
- dec_eq : DecidableEq self.α
- initCarry : self.α → Bool
nextBitCirc
is a family of Boolean circuits, which may refer to the current input bits and the current state bits as free variables in the circuit.nextBitCirc none
computes the current output bit.nextBitCirc (some a)
, computes the one bit of the new state that corresponds toa : α
.
Instances For
p.nextBit state in
computes both the next state bits and the output bit,
where state
are the current state bits, and in
are the current input bits.
Equations
Instances For
p.carry in i
computes the internal carry state at step i
, given input streams in
Equations
Instances For
eval p
morally gives the function BitStream → ... → BitStream
represented by FSM p
Equations
- p.eval x n = (p.nextBit (p.carry x n) fun (i : arity) => x i n).2
Instances For
eval'
is an alternative definition of eval
Equations
- One or more equations did not get rendered due to their size.
Instances For
p.changeInitCarry c
yields an FSM with c
as the initial state
Equations
- p.changeInitCarry c = AutoStructs.FSM.mk p.α c p.nextBitCirc
Instances For
unfolds the definition of eval
p.changeVars f
changes the arity of an FSM
.
The function f
determines how the new input bits map to the input expected by p
Equations
- p.changeVars changeVars = AutoStructs.FSM.mk p.α p.initCarry fun (a : Option p.α) => (p.nextBitCirc a).map (Sum.map id changeVars)
Instances For
Given an FSM p
of arity n
,
a family of n
FSMs qᵢ
of posibly different arities mᵢ
,
and given yet another arity m
such that mᵢ ≤ m
for all i
,
we can compose p
with qᵢ
yielding a single FSM of arity m
,
such that each FSM qᵢ
computes the i
th bit that is fed to the FSM p
.
Equations
- One or more equations did not get rendered due to their size.
Instances For
Evaluating a composed fsm is equivalent to composing the evaluations of the constituent FSMs
Equations
- AutoStructs.FSM.and = AutoStructs.FSM.mk Empty Empty.elim fun (a : Option Empty) => a.elim ((Circuit.var true (Sum.inr true)).and (Circuit.var true (Sum.inr false))) Empty.elim
Instances For
Equations
- AutoStructs.FSM.or = AutoStructs.FSM.mk Empty Empty.elim fun (a : Option Empty) => a.elim ((Circuit.var true (Sum.inr true)).or (Circuit.var true (Sum.inr false))) Empty.elim
Instances For
Equations
- AutoStructs.FSM.xor = AutoStructs.FSM.mk Empty Empty.elim fun (a : Option Empty) => a.elim ((Circuit.var true (Sum.inr true)).xor (Circuit.var true (Sum.inr false))) Empty.elim
Instances For
Equations
- One or more equations did not get rendered due to their size.
Instances For
The internal carry state of the add
FSM agrees with
the carry bit of addition as implemented on bitstreams
We don't really need subtraction or negation FSMs, given that we can reduce both those operations to just addition and bitwise complement
Equations
- One or more equations did not get rendered due to their size.
Instances For
Equations
- One or more equations did not get rendered due to their size.
Instances For
Equations
- AutoStructs.FSM.not = AutoStructs.FSM.mk Empty Empty.elim fun (x : Option Empty) => Circuit.var false (Sum.inr ())
Instances For
Equations
- AutoStructs.FSM.zero = AutoStructs.FSM.mk Empty Empty.elim fun (x : Option Empty) => Circuit.fals
Instances For
Equations
- AutoStructs.FSM.one = AutoStructs.FSM.mk Unit (fun (x : Unit) => true) fun (a : Option Unit) => match a with | some PUnit.unit => Circuit.fals | none => Circuit.var true (Sum.inl ())
Instances For
Equations
- AutoStructs.FSM.negOne = AutoStructs.FSM.mk Empty Empty.elim fun (x : Option Empty) => Circuit.tru
Instances For
Equations
- One or more equations did not get rendered due to their size.
Instances For
Equations
- AutoStructs.FSM.var n = AutoStructs.FSM.mk Empty Empty.elim fun (x : Option Empty) => Circuit.var true (Sum.inr (Fin.last n))
Instances For
Equations
- One or more equations did not get rendered due to their size.
Instances For
Equations
- One or more equations did not get rendered due to their size.
Instances For
Equations
- AutoStructs.FSM.repeatBit = AutoStructs.FSM.mk Unit (fun (x : Unit) => false) fun (x : Option Unit) => (Circuit.var true (Sum.inl ())).or (Circuit.var true (Sum.inr ()))
Instances For
- α : Type
- i : FinEnum self.α
- dec_eq : DecidableEq self.α
- initCarry : self.α → Bool
- good : t.evalFinStream = self.eval
Instances For
Equations
Instances For
Equations
- One or more equations did not get rendered due to their size.
Instances For
Equations
- One or more equations did not get rendered due to their size.
Instances For
Equations
- AutoStructs.termEvalEqFSM (AutoStructs.Term.var n) = { toFSM := AutoStructs.FSM.var n, good := ⋯ }
- AutoStructs.termEvalEqFSM AutoStructs.Term.zero = { toFSM := AutoStructs.FSM.zero, good := AutoStructs.termEvalEqFSM.proof_15 }
- AutoStructs.termEvalEqFSM AutoStructs.Term.one = { toFSM := AutoStructs.FSM.one, good := AutoStructs.termEvalEqFSM.proof_16 }
- AutoStructs.termEvalEqFSM AutoStructs.Term.negOne = { toFSM := AutoStructs.FSM.negOne, good := AutoStructs.termEvalEqFSM.proof_17 }
- AutoStructs.termEvalEqFSM (t₁.and t₂) = { toFSM := AutoStructs.composeBinary AutoStructs.FSM.and (AutoStructs.termEvalEqFSM t₁) (AutoStructs.termEvalEqFSM t₂), good := ⋯ }
- AutoStructs.termEvalEqFSM (t₁.or t₂) = { toFSM := AutoStructs.composeBinary AutoStructs.FSM.or (AutoStructs.termEvalEqFSM t₁) (AutoStructs.termEvalEqFSM t₂), good := ⋯ }
- AutoStructs.termEvalEqFSM (t₁.xor t₂) = { toFSM := AutoStructs.composeBinary AutoStructs.FSM.xor (AutoStructs.termEvalEqFSM t₁) (AutoStructs.termEvalEqFSM t₂), good := ⋯ }
- AutoStructs.termEvalEqFSM t.not = { toFSM := id (AutoStructs.composeUnary AutoStructs.FSM.not (AutoStructs.termEvalEqFSM t)), good := ⋯ }
- AutoStructs.termEvalEqFSM (t₁.add t₂) = { toFSM := AutoStructs.composeBinary AutoStructs.FSM.add (AutoStructs.termEvalEqFSM t₁) (AutoStructs.termEvalEqFSM t₂), good := ⋯ }
- AutoStructs.termEvalEqFSM (t₁.sub t₂) = { toFSM := AutoStructs.composeBinary AutoStructs.FSM.sub (AutoStructs.termEvalEqFSM t₁) (AutoStructs.termEvalEqFSM t₂), good := ⋯ }
- AutoStructs.termEvalEqFSM t.neg = { toFSM := id (AutoStructs.composeUnary AutoStructs.FSM.neg (AutoStructs.termEvalEqFSM t)), good := ⋯ }
- AutoStructs.termEvalEqFSM t.incr = { toFSM := id (AutoStructs.composeUnary AutoStructs.FSM.incr (AutoStructs.termEvalEqFSM t)), good := ⋯ }
- AutoStructs.termEvalEqFSM t.decr = { toFSM := id (AutoStructs.composeUnary AutoStructs.FSM.decr (AutoStructs.termEvalEqFSM t)), good := ⋯ }
Instances For
FSM that implement bitwise-and. Since we use 0
as the good state,
we keep the invariant that if both inputs are good and our state is 0
, then we produce a 0
.
If not, we produce an infinite sequence of 1
.
Equations
- One or more equations did not get rendered due to their size.
Instances For
FSM that implement bitwise-or. Since we use 0
as the good state,
we keep the invariant that if either inputs is 0
then our state is 0
.
If not, we produce a 1
.
Equations
- One or more equations did not get rendered due to their size.
Instances For
FSM that implement logical not.
we keep the invariant that if the input ever fails and becomes a 1
, then we produce a 0
.
IF not, we produce an infinite sequence of 1
.
EDIT: Aha, this doesn't work!
We need NFA to DFA here (as the presburger book does),
where we must produce an infinite sequence of0
iff the input can ever become a 1
.
But here, since we phrase things directly in terms of producing sequences, it's a bit less clear
what we should do :)
- Alternatively, we need to be able to decide
eventually always zero
. - Alternatively, we push negations inside, and decide
⬝ ≠ ⬝
and⬝ ≰ ⬝
.
- falseAfter: ℕ → AutoStructs.Result
- trueFor: ℕ → AutoStructs.Result
- trueForall: AutoStructs.Result
Instances For
Equations
- AutoStructs.instReprResult = { reprPrec := AutoStructs.reprResult✝ }
Equations
- AutoStructs.card_compl c = (Finset.filter (fun (a : α → Bool) => c.eval a = false) Finset.univ).card
Instances For
Equations
- One or more equations did not get rendered due to their size.
Instances For
Equations
- AutoStructs.decideIfZeros p = AutoStructs.decideIfZerosAux p (p.nextBitCirc none).fst
Instances For
The fragment of predicate logic that we support in bv_automata
.
Currently, we support equality, conjunction, disjunction, and negation.
This can be expanded to also support arithmetic constraints such as unsigned-less-than.
- eq: (t1 t2 : AutoStructs.Term) → AutoStructs.Predicate (max t1.arity t2.arity)
- and: {n m : ℕ} → AutoStructs.Predicate n → AutoStructs.Predicate m → AutoStructs.Predicate (max n m)
- or: {n m : ℕ} → AutoStructs.Predicate n → AutoStructs.Predicate m → AutoStructs.Predicate (max n m)