Warning
This post was published 111 days ago. The information described in this article may have changed.
This write up is a report to share our insights from learning Formal Verification and reviewing how it can be used to formally verify zk circuits.
Machine assisted property verification of formal propositions.
Comparison with Informal Verification (regular math proofs):
Traditionally used for software programs.
Prove that:
Also has been used for Smart Contracts. Example:
Can be used to verify zk circuits
Examples:
Two categories of tools to do formal verification:
We made a study group and followed the “Software Foundations, Volume 1: Logical Foundations” book to learn Coq
The language is very minimal and only provides a very small set of primitives. Primitive types and operations are defined via libraries.
Source of these examples is “Software Foundations, Volume 1: Logical Foundations”.
Natural numbers:
Inductive nat : Type := | O | S (n : nat).
> A natural
is 0
or next(n)
Examples:
O -> 0
S O -> next(0) -> 1 + 0 -> 1
S S S O -> next(next(next(0))) -> 1 + 1 + 1 + 0 -> 3
Natural addition:
Fixpoint plus (n : nat) (m : nat) : nat := match n with | O => m | S n' => S (plus n' m) end.
> To do plus n m
:
> if n
is 0
, the result is m
> if n
is 1+n'
, the result is 1 + (plus n' m)
Examples:
plus 0 5 -> 5
plus 2 3 -> 1 + (plus 1 3) -> 1 + 1 + (plus 0 3) -> 1 + 1 + 3 -> 5
Basic proof:
Theorem add_0_r : forall n:nat, n + 0 = n. Proof. intros n. induction n as [| n' IHn']. - (* n = 0 *) reflexivity. - (* n = S n' *) simpl. rewrite -> IHn'. reflexivity. Qed.
> Prove that for all n
, n + 0 = n
> Induction by n
> Base case:
> - n = 0
. Then we have 0 + 0 = 0
.
> - By definition of plus
this is True.
> Inductive case:
> - Inductive hypothesis: n' + 0 = n'
with n = 1 + n'
> - We need to prove 1 + n' + 0 = 1 + n'
> - Which is the same as 1 + (n' + 0) = 1 + n'
> - We rewrite our goal using the hypothesis equality and get:
> - 1 + n' = 1 + n'
> - Which is True.
Checking the add_0_r
proof interactively.
Consider the proposition . We know it to be true and we can use it in classical proofs:
But this principle can’t be derived in Coq because it follows Constructive Logic:
Nevertheless we can still use it as an unproven Lemma in Coq :)
Informal Theorem and Proof:
> Source: MIT 18.211: COMBINATORIAL ANALYSIS, FELIX GOTTI
The same Theorem and Proof in Coq:
Lemma in_split : forall (X:Type) (x:X) (l:list X), In x l -> exists l1 l2, l = l1 ++ x :: l2. Proof. intros X x l H. induction l as [|x' l' IHl']. - destruct H. - inversion H. + exists []. exists l'. simpl. rewrite H0. reflexivity. + apply IHl' in H0. destruct H0 as [l1 [l2 H1]]. exists (x' :: l1). exists l2. simpl. f_equal. apply H1. Qed. Inductive repeats {X:Type} : list X -> Prop := | Repeats x l (H: In x l) : repeats (x :: l) | RepeatsRm x l (H: repeats l) : repeats (x :: l) . (** Now, here's a way to formalize the pigeonhole principle. Suppose list [l2] represents a list of pigeonhole labels, and list [l1] represents the labels assigned to a list of items. If there are more items than labels, at least two items must have the same label -- i.e., list [l1] must contain repeats. This proof is much easier if you use the [excluded_middle] hypothesis to show that [In] is decidable, i.e., [forall x l, (In x l) \/ ~ (In x l)]. However, it is also possible to make the proof go through _without_ assuming that [In] is decidable; if you manage to do this, you will not need the [excluded_middle] hypothesis. *) Theorem pigeonhole_principle: excluded_middle -> forall (X:Type) (l1 l2:list X), (forall x, In x l1 -> In x l2) -> length l2 < length l1 -> repeats l1. Proof. intros EM X l1. induction l1 as [|x l1' IHl1']. - (* l1 = [] *) intros l2 H1 H2. destruct l2. + inversion H2. + inversion H2. - (* l1 = x :: l1' *) simpl. intros l2 Hi Hl. unfold excluded_middle in EM. destruct (EM (In x l1')). + (* [In x l1'] means that the repeated element in [l1] is [x] *) apply Repeats. apply H. + (* [~ In x l1'] means that the repeated element in [l1] is not [x] *) apply RepeatsRm. assert (Hix : In x l2). { apply Hi. left. reflexivity. } apply in_split in Hix. destruct Hix as [l21 [l22 El2]]. apply (IHl1' (l21 ++ l22)). * intros x0 Hix0. assert (Hix02 : In x0 l2). { apply Hi. right. apply Hix0. } rewrite El2 in Hix02. rewrite In_app_iff in Hix02. rewrite In_app_iff. destruct Hix02 as [Hix0l21 | Hix0l22]. ** left. apply Hix0l21. ** right. simpl in Hix0l22. destruct Hix0l22. *** rewrite H0 in *. destruct (H Hix0). *** apply H0. * unfold lt in *. apply le_trans with (length l2). ** rewrite app_length. rewrite El2. rewrite app_length. simpl. rewrite plus_n_Sm. apply le_n. ** apply Sn_le_Sm__n_le_m. apply Hl. Qed.
> Source: Software Foundations exercise solution by Soowon. I wasn’t able to complete this one :(
> Source: > Compositional Formal Verification of Zero-Knowledge Circuits > by Alessandro Coglio and Eric McCarthy and Eric Smith and Collin Chin and Pranav Gaddamadugu and Michel Dellepere > Cryptology {ePrint} Archive, Paper 2023/1278 > https://eprint.iacr.org/2023/1278.pdf
IsZero
proofSource code from Veriside/Coda circomlib Formal Verification project: https://github.com/Veridise/Coda/blob/96e5c94e1a27d25466f24beee31c010810273e33/BigInt/src/CircomLib/Comparators.v#L38
Original circom:
template IsZero() { signal input in; signal output out; signal inv; inv <-- in!=0 ? 1/in : 0; out <== -in*inv +1; in*out === 0; }
Extracted constraints:
Definition cons (_in out: F) := exists _inv, out = 1 - _in * _inv /\ _in * out = 0.
> Exists inv
such that
> out = 1 - in * inv
AND
> in * out = 0
Manually written spec:
Record t : Type := { _in: F; out: F; _cons: cons _in out }. Definition spec (c: t) : Prop := if (c.(_in) = 0)? then c.(out) = 1 else c.(out) = 0.
> If in == 0
then out = 1
else out = 0
Soundness proof:
Theorem soundness: forall (c: t), spec c. Proof. unwrap_C. intros. destruct c as [x y [x_inv H]]. unfold spec, cons in *. simpl. intuition. destruct (dec (x = 0)); fqsatz. Qed.
> Prove that for all input-output pairs that follow the constraints, the spec is satisfied.
> The proof involves proving the case where in = 0
and in != 0
.
Not all components are as easy to prove; take a look at the soundness proof of LessThan
: https://github.com/Veridise/Coda/blob/96e5c94e1a27d25466f24beee31c010810273e33/BigInt/src/CircomLib/Comparators.v#L161
We have learned how to write formal proofs with Coq.
We have an intuition of how to use Coq to verify zk circuits.
The next (reasonable) step would be to formally verify an existing PSE project.
truthixify 2025-01-12 👍 1 👎
Can I join the group?
@CPerezz
Can we post this on PSE Blog too? And tweet about it from PSE account @
ed255 ? Happy to help setting this up
I’m not sure, originally I thought it was not necessary because currently the blog has a low frequency of posts and they look like important announcements / articles; whereas this is just a regular report of a short-timed work. If we want to increase the frequency of blog posting and include more regular reports of our work, then yes we could post this in the PSE blog.
@truthixify
Can I join the group?
This exploration concluded by the end of 2024 so there’s no longer an active group. Nevertheless you can always follow the book by yourself. If you’d like to do that, feel free to reach us for questions or discussions about the contents and exercises :)