forked from mit-plv/fiat-crypto
-
Notifications
You must be signed in to change notification settings - Fork 1
/
XZProofs.v
414 lines (376 loc) · 20.7 KB
/
XZProofs.v
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
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
Require Import Crypto.Algebra.Field.
Require Import Crypto.Algebra.ScalarMult.
Require Import Crypto.Util.Sum Crypto.Util.Prod Crypto.Util.LetIn.
Require Import Crypto.Util.Decidable.
Require Import Crypto.Util.Tuple.
Require Import Crypto.Util.ZUtil.Notations.
Require Import Crypto.Util.ZUtil.Tactics.LtbToLt.
Require Import Crypto.Util.ZUtil.Shift.
Require Import Crypto.Util.ZUtil.Peano.
Require Import Crypto.Util.Tactics.SetoidSubst.
Require Import Crypto.Util.Tactics.SpecializeBy.
Require Import Crypto.Util.Tactics.DestructHead.
Require Import Crypto.Util.Tactics.BreakMatch.
Require Import Crypto.Util.Tactics.UniquePose.
Require Import Crypto.Spec.MontgomeryCurve Crypto.Curves.Montgomery.Affine.
Require Import Crypto.Curves.Montgomery.AffineInstances.
Require Import Crypto.Curves.Montgomery.XZ BinPos.
From Coq Require Import Morphisms.
From Coq Require Import ZArith.
From Coq Require Import Lia.
Module M.
Section MontgomeryCurve.
Context {F Feq Fzero Fone Fopp Fadd Fsub Fmul Finv Fdiv}
{field:@Algebra.Hierarchy.field F Feq Fzero Fone Fopp Fadd Fsub Fmul Finv Fdiv}
{Feq_dec:Decidable.DecidableRel Feq}
{char_ge_28:@Ring.char_ge F Feq Fzero Fone Fopp Fadd Fsub Fmul 28}.
Lemma Private_char_ge_3: @Ring.char_ge F Feq Fzero Fone Fopp Fadd Fsub Fmul 3.
Proof. clear -char_ge_28; eapply Algebra.Hierarchy.char_ge_weaken; eauto; vm_decide. Qed.
Context {char_ge_3 : @Ring.char_ge F Feq Fzero Fone Fopp Fadd Fsub Fmul 3}. (* appears in statement *)
Local Instance char_ge_5: @Ring.char_ge F Feq Fzero Fone Fopp Fadd Fsub Fmul 5.
Proof. clear -char_ge_28; eapply Algebra.Hierarchy.char_ge_weaken; eauto; vm_decide. Qed.
Local Instance char_ge_12: @Ring.char_ge F Feq Fzero Fone Fopp Fadd Fsub Fmul 12.
Proof. clear -char_ge_28; eapply Algebra.Hierarchy.char_ge_weaken; eauto; vm_decide. Qed.
Local Infix "=" := Feq : type_scope. Local Notation "a <> b" := (not (a = b)) : type_scope.
Local Infix "+" := Fadd. Local Infix "*" := Fmul.
Local Infix "-" := Fsub. Local Infix "/" := Fdiv.
Local Notation "0" := Fzero. Local Notation "1" := Fone.
Local Notation "'∞'" := (inr tt) : core_scope.
Local Notation "( x , y )" := (inl (pair x y)).
Context {a b: F} {b_nonzero:b <> 0}.
Context {a24:F} {a24_correct:(1+1+1+1)*a24 = a-(1+1)}.
Local Notation Madd := (M.add(a:=a)(b_nonzero:=b_nonzero)(char_ge_3:=char_ge_3)).
Local Notation Mopp := (M.opp(a:=a)(b_nonzero:=b_nonzero)).
Local Notation Mpoint := (@M.point F Feq Fadd Fmul a b).
Local Notation X0 := (M.X0(Fzero:=Fzero)(Feq:=Feq)(Fadd:=Fadd)(Fmul:=Fmul)(a:=a)(b:=b)).
Local Notation to_xz := (M.to_xz(Fzero:=Fzero)(Fone:=Fone)(Feq:=Feq)(Fadd:=Fadd)(Fmul:=Fmul)(a:=a)(b:=b)).
Local Notation xzladderstep := (M.xzladderstep(a24:=a24)(Fadd:=Fadd)(Fsub:=Fsub)(Fmul:=Fmul)).
(** This tactic is to solve goals that fsatz also solves, but to
do so faster in some cases common in this file. It may also
work around deficiencies in the Coq 8.6 (released) version of
[nsatz]; it has some heuristics for clearing hypotheses and
running [exfalso], and then tries [fsatz] *)
Local Ltac maybefast :=
try lazymatch goal with
| [ fld : Hierarchy.field (T:=?F) (eq:=?Feq), Feq_dec : DecidableRel ?Feq |- _ ]
=> lazymatch goal with
| [ H : ?x * 1 = ?y * ?z, H' : ?x <> 0, H'' : ?z = 0 |- _ ]
=> clear -H H' H'' fld Feq_dec; exfalso; fsatz
| [ H : ?x * 0 = 1 * ?y, H' : ?y <> 0 |- _ ]
=> clear -H H' fld Feq_dec; exfalso; fsatz
| _
=> match goal with
| [ H : ?b * ?lhs = ?rhs, H' : ?b * ?lhs' = ?rhs', Heq : ?x = ?y, Hb : ?b <> 0 |- _ ]
=> exfalso;
repeat match goal with H : Ring.char_ge _ |- _ => revert H end;
let rhs := match (eval pattern x in rhs) with ?f _ => f end in
let rhs' := match (eval pattern y in rhs') with ?f _ => f end in
unify rhs rhs';
match goal with
| [ H'' : ?x = ?Fopp ?x, H''' : ?x <> ?Fopp (?Fopp ?y) |- _ ]
=> let lhs := match (eval pattern x in lhs) with ?f _ => f end in
let lhs' := match (eval pattern y in lhs') with ?f _ => f end in
unify lhs lhs';
clear -H H' Heq H'' H''' Hb fld Feq_dec; intros
| [ H'' : ?x <> ?Fopp ?y, H''' : ?x <> ?Fopp (?Fopp ?y) |- _ ]
=> let lhs := match (eval pattern x in lhs) with ?f _ => f end in
let lhs' := match (eval pattern y in lhs') with ?f _ => f end in
unify lhs lhs';
clear -H H' Heq H'' H''' Hb fld Feq_dec; intros
end;
fsatz
| [ H : ?x * (?y * ?z) = 0 |- _ ]
=> exfalso;
repeat match goal with
| [ H : ?x * 1 = ?y * ?z |- _ ]
=> is_var x; is_var y; is_var z;
revert H
end;
generalize fld;
let lhs := match type of H with ?lhs = _ => lhs end in
repeat match goal with
| [ x : F |- _ ] => lazymatch type of H with
| context[x] => fail
| _ => clear dependent x
end
end;
intros _; intros;
fsatz
end
end
end.
(* TODO(#152): deduplicate between curves files *)
Ltac prept_step :=
match goal with
| _ => progress intros
| _ => progress autounfold with points_as_coordinates in *
| _ => progress destruct_head'_True
| _ => progress destruct_head'_unit
| _ => progress destruct_head'_prod
| _ => progress destruct_head'_sig
| _ => progress destruct_head'_bool
| _ => progress destruct_head'_sum
| _ => progress destruct_head'_and
| _ => progress destruct_head'_or
| H: context[dec ?P] |- _ => destruct (dec P)
| |- context[dec ?P] => destruct (dec P)
| |- ?P => lazymatch type of P with Prop => split end
end.
Ltac prept := repeat prept_step.
Ltac t_fast := prept; trivial; try contradiction.
Ltac t := t_fast; maybefast; fsatz.
Create HintDb points_as_coordinates discriminated.
Hint Unfold Proper respectful Reflexive Symmetric Transitive : points_as_coordinates.
Hint Unfold Let_In : points_as_coordinates.
Hint Unfold fst snd proj1_sig : points_as_coordinates.
Hint Unfold fieldwise fieldwise' : points_as_coordinates.
Hint Unfold M.add M.opp M.point M.coordinates M.xzladderstep M.donnaladderstep M.boringladderstep M.to_xz : points_as_coordinates.
Lemma donnaladderstep_same x1 Q Q' :
fieldwise (n:=2) (fieldwise (n:=2) Feq)
(xzladderstep x1 Q Q')
(M.donnaladderstep(a24:=a24)(Fadd:=Fadd)(Fsub:=Fsub)(Fmul:=Fmul) x1 Q Q').
Proof. t. Qed.
Lemma boringladderstep_same (ap2d4:F) (ap2d4_correct:(1+1+1+1)*a24 = a+1+1) x1 Q Q' :
fieldwise (n:=2) (fieldwise (n:=2) Feq)
(xzladderstep x1 Q Q')
(M.boringladderstep(ap2d4:=ap2d4)(Fadd:=Fadd)(Fsub:=Fsub)(Fmul:=Fmul) x1 Q Q').
Proof. t. Qed.
Definition projective (P:F*F) :=
if dec (snd P = 0) then fst P <> 0 else True.
Definition eq (P Q:F*F) := fst P * snd Q = fst Q * snd P.
Definition ladder_invariant x1 Q Q' :=
match M.coordinates (Madd Q (Mopp Q')) with
| ∞ => False (* Q <> Q' *)
| (x,y) => x = x1 /\ x1 <> 0 (* Q-Q' <> (0, 0) *)
end.
Hint Unfold projective eq ladder_invariant : points_as_coordinates.
Lemma to_xz_add_coordinates (x1:F) (xz x'z':F*F)
(Hxz:projective xz) (Hz'z':projective x'z')
(Q Q':Mpoint)
(HQ:eq xz (to_xz Q)) (HQ':eq x'z' (to_xz Q'))
(HI:ladder_invariant x1 Q Q')
: projective (snd (xzladderstep x1 xz x'z'))
/\ eq (fst (xzladderstep x1 xz x'z')) (to_xz (Madd Q Q ))
/\ eq (snd (xzladderstep x1 xz x'z')) (to_xz (Madd Q Q')).
Proof. Time t_fast. Time par: abstract (maybefast; fsatz). Time Qed.
(* 12 ;; 24 ;; 3 *)
Context {a2m4_nonsquare:forall r, r*r <> a*a - (1+1+1+1)}.
Lemma projective_fst_xzladderstep x1 Q (HQ:projective Q) Q'
: projective (fst (xzladderstep x1 Q Q')).
Proof.
specialize (a2m4_nonsquare (fst Q/snd Q - fst Q/snd Q)).
destruct (dec (snd Q = 0)); t.
Qed.
Local Definition a2m4_nz : a*a - (1+1+1+1) <> 0.
Proof. specialize (a2m4_nonsquare 0). fsatz. Qed.
Lemma difference_preserved Q Q' :
MontgomeryCurve.M.eq
(Madd (Madd Q Q) (Mopp (Madd Q Q')))
(Madd Q (Mopp Q')).
Proof.
pose proof (let (_, h, _, _) := AffineInstances.M.MontgomeryWeierstrassIsomorphism b_nonzero (a:=a) a2m4_nz in h) as commutative_group.
rewrite Group.inv_op.
rewrite <-Hierarchy.associative.
rewrite Group.cancel_left.
rewrite Hierarchy.commutative.
rewrite <-Hierarchy.associative.
rewrite Hierarchy.left_inverse.
rewrite Hierarchy.right_identity.
reflexivity.
Qed.
Lemma ladder_invariant_preserved x1 Q Q' (H:ladder_invariant x1 Q Q')
: ladder_invariant x1 (Madd Q Q) (Madd Q Q').
Proof.
cbv [ladder_invariant] in *.
pose proof difference_preserved Q Q' as Hrw.
(* FIXME: what we actually want to do here is to rewrite with in
match argument with
[sumwise (fieldwise (n:=2) Feq) (fun _ _ => True)] *)
cbv [MontgomeryCurve.M.eq] in *; break_match; break_match_hyps;
destruct_head' @and; repeat split; subst;
try solve [intuition congruence].
congruence (* congruence failed, idk WHY *)
|| (match goal with
[H:?f1 = ?x1, G:?f = ?f1 |- ?f = ?x1] => rewrite G; exact H
end).
Qed.
Lemma ladder_invariant_swap x1 Q Q' (H:ladder_invariant x1 Q Q')
: ladder_invariant x1 Q' Q.
Proof. t. Qed.
Lemma to_xz_add x1 xz x'z' Q Q'
(Hxz : projective xz) (Hx'z' : projective x'z')
(HQ : eq xz (to_xz Q)) (HQ' : eq x'z' (to_xz Q'))
(HI : ladder_invariant x1 Q Q')
: projective (fst (xzladderstep x1 xz x'z'))
/\ projective (snd (xzladderstep x1 xz x'z'))
/\ eq (fst (xzladderstep x1 xz x'z')) (to_xz (Madd Q Q ))
/\ eq (snd (xzladderstep x1 xz x'z')) (to_xz (Madd Q Q'))
/\ ladder_invariant x1 (Madd Q Q) (Madd Q Q').
Proof.
destruct (to_xz_add_coordinates x1 xz x'z' Hxz Hx'z' Q Q' HQ HQ' HI) as [? [? ?]].
eauto 99 using projective_fst_xzladderstep, ladder_invariant_preserved.
Qed.
Definition to_x (xz:F*F) : F :=
if dec (snd xz = 0) then 0 else fst xz / snd xz.
Hint Unfold to_x : points_as_coordinates.
Lemma to_x_to_xz Q : to_x (to_xz Q) = X0 Q.
Proof. t. Qed.
Lemma proper_to_x_projective xz x'z'
(Hxz:projective xz) (Hx'z':projective x'z')
(H:eq xz x'z') : Feq (to_x xz) (to_x x'z').
Proof. destruct (dec (snd xz = 0)), (dec(snd x'z' = 0)); t. Qed.
Lemma to_x_zero x : to_x (pair x 0) = 0.
Proof. t. Qed.
Hint Unfold MontgomeryCurve.M.eq : points_as_coordinates.
Global Instance Proper_to_xz : Proper (MontgomeryCurve.M.eq ==> eq) to_xz.
Proof. t. Qed.
Global Instance Reflexive_eq : Reflexive eq.
Proof. t. Qed.
Global Instance Symmetric_eq : Symmetric eq.
Proof. t. Qed.
Lemma transitive_eq {p} q {r} : projective q -> eq p q -> eq q r -> eq p r.
Proof. t. Qed.
Lemma projective_to_xz Q : projective (to_xz Q).
Proof. t. Qed.
Global Instance Proper_ladder_invariant : Proper (Feq ==> MontgomeryCurve.M.eq ==> MontgomeryCurve.M.eq ==> iff) ladder_invariant.
Proof. t. Qed.
Local Notation montladder := (M.montladder(a24:=a24)(Fadd:=Fadd)(Fsub:=Fsub)(Fmul:=Fmul)(Fzero:=Fzero)(Fone:=Fone)(Finv:=Finv)).
Local Notation scalarmult := (@ScalarMult.scalarmult_ref Mpoint Madd M.zero Mopp).
Import Crypto.Util.Loops.
Import BinInt.
Lemma to_x_inv00 (HFinv:Finv 0 = 0) x z : to_x (pair x z) = x * Finv z.
Proof. t_fast; setoid_subst_rel Feq; rewrite ?HFinv in *; fsatz. Qed.
Lemma Z_shiftr_testbit_1 n i: Logic.eq (n>>i)%Z (Z.div2 (n >> i) + Z.div2 (n >> i) + Z.b2z (Z.testbit n i))%Z.
Proof. rewrite ?Z.testbit_odd, ?Z.add_diag, <-?Z.div2_odd; reflexivity. Qed.
Context (HFinv : Finv 0 = 0) (scalarbits : Z) (Hscalarbits : (0 <= scalarbits)%Z).
(* We prove montladder correct by considering the zero and non-zero case
separately. *)
Lemma montladder_correct_0 n x (Hx : Feq x 0) : montladder scalarbits (Z.testbit n) x = 0.
Proof.
cbv beta delta [M.montladder].
(* [while.by_invariant] expects a goal like [?P (while _ _ _ _)], make it so: *)
lazymatch goal with |- context [while ?t ?b ?l ?ii] => pattern (while t b l ii) end.
eapply (while.by_invariant_fuel
(fun '(x2, z2, x3, z3, swap, i) =>
(i < scalarbits)%Z /\
z2 = 0 /\
if dec (Logic.eq i (Z.pred scalarbits)) then x3 = 0 else z3 = 0)
(fun s => Z.to_nat (Z.succ (snd s)))).
{ split.
(* invariant holds in the beginning *)
{ cbn; split; [lia|split;[reflexivity|try t]]. }
{ (* fuel <= measure *) cbn. rewrite Z.succ_pred. reflexivity. } }
{ intros [ [ [ [ [x2 z2] x3] z3] swap] i] [Hi [Hz2 Hx3z3]].
destruct (i >=? 0)%Z eqn:Hbranch; (* did the loop continue? *)
rewrite Z.geb_ge_iff in Hbranch.
{ split. (* if loop continued, invariant is preserved *)
{ destruct (dec (Logic.eq i (Z.pred scalarbits))).
{ (* first loop iteration *)
cbv -[xzladderstep xorb Z.testbit Z.pred dec Z.lt];
destruct (xorb swap (Z.testbit n i));
split; [lia|t|lia|t]. }
{ (* subsequent loop iterations *)
cbv -[xzladderstep xorb Z.testbit Z.pred dec Z.lt].
destruct (xorb swap (Z.testbit n i));
(split; [lia| split; [t| break_match;[lia|t]]]). } }
{ (* measure decreases *)
cbv [Let_In]; break_match; cbn; rewrite Z.succ_pred; apply Znat.Z2Nat.inj_lt; lia. } }
{ (* if loop exited, invariant implies postcondition *)
cbv [M.cswap]; break_match; break_match_hyps; setoid_subst_rel Feq; fsatz. } }
Qed.
Lemma montladder_correct_nz n P (Hnz : X0 P <> 0)
: montladder scalarbits (Z.testbit n) (X0 P) = X0 (scalarmult (n mod 2^scalarbits) P).
Proof.
pose proof (let (_, h, _, _) := AffineInstances.M.MontgomeryWeierstrassIsomorphism b_nonzero (a:=a) a2m4_nz in h) as commutative_group.
cbv beta delta [M.montladder].
(* [while.by_invariant] expects a goal like [?P (while _ _ _ _)], make it so: *)
lazymatch goal with |- context [while ?t ?b ?l ?i] => pattern (while t b l i) end.
eapply (while.by_invariant_fuel
(fun '(x2, z2, x3, z3, swap, i) =>
(-1 <= i < scalarbits)%Z /\
projective (pair x2 z2) /\
projective (pair x3 z3) /\
let q := if (swap:bool) then (pair x3 z3) else (pair x2 z2) in
let q' := if (swap:bool) then (pair x2 z2) else (pair x3 z3) in
let r := ((n mod 2^scalarbits) >> Z.succ i)%Z in
eq q (to_xz (scalarmult r P)) /\
eq q' (to_xz (scalarmult (Z.succ r) P)) /\
ladder_invariant (X0 P) (scalarmult r P) (scalarmult (Z.succ r) P))
(fun s => Z.to_nat (Z.succ (snd s))) (* decreasing measure *) ).
{ split; cbn.
{ (* invariant holds in the beginning *)
rewrite ?Z.succ_pred, ?Z.lt_pow_2_shiftr, <-?Z.one_succ by (apply Z.mod_pos_bound; lia).
repeat split; [lia|lia|t..]. }
{ (* sufficient fuel *) rewrite Z.succ_pred. reflexivity. } }
{ intros [ [ [ [ [x2 z2] x3] z3] swap] i] [Hi [Hx2z2 [Hx3z3 [Hq [Hq' Hladder]]]]].
destruct (i >=? 0)%Z eqn:Hbranch; (* did the loop continue? *)
rewrite Z.geb_ge_iff in Hbranch.
split.
{ (* if loop continued, invariant is preserved *)
remember (Z.testbit n i) as bit eqn:Hbit; symmetry in Hbit.
erewrite <-(Z.mod_pow2_bits_low _ scalarbits) in Hbit by tauto.
set (n mod 2 ^ scalarbits)%Z as n' in *.
let group _ := ltac:(repeat rewrite ?scalarmult_add_l, ?scalarmult_0_l, ?scalarmult_1_l, ?Hierarchy.left_identity, ?Hierarchy.right_identity, ?Hierarchy.associative, ?(Hierarchy.commutative _ P); reflexivity) in
destruct bit in *;
destruct swap eqn:Hswap in *;
repeat match goal with
| _ => solve [ congruence | assumption | lia ]
| _ => progress cbv beta delta [Let_In]
| _ => progress cbn [xorb Z.b2z]
| _ => progress autorewrite with cancel_pair in *
| _ => rewrite <-!@surjective_pairing
| _ => progress inversion_prod
| _ => progress subst
| _ => break_match_hyps_step ltac:(fun _ => idtac)
| _ => break_match_step ltac:(fun _ => idtac)
| H:projective ?xz, G:projective ?x'z', HQ: eq ?xz (to_xz ?Q), HQ': eq ?x'z' (to_xz ?Q')
|- context [xzladderstep ?p ?xz ?x'z']
=> let pf := constr:(to_xz_add p xz x'z' _ _ H G HQ HQ' ltac:(auto using ladder_invariant_swap)) in
unique pose proof (proj1 pf); destruct (proj2 pf) as [? [? [? ?]]] (* because there is no unique destruct *)
| _ => progress rewrite ?Z.succ_pred, ?Z.shiftr_succ, <-?Z.div2_spec, <-?Z.add_1_r in *
| |- context [scalarmult (n'>>i) ] => rewrite (Z_shiftr_testbit_1 n' i), Hbit; cbn [Z.b2z]
| |- context [scalarmult (n'>>i+1) ] => rewrite (Z_shiftr_testbit_1 n' i), Hbit; cbn [Z.b2z]
| |- ?P => match type of P with Prop => split end
| H: eq (?f _) (to_xz ?LHS) |- eq (?f _) (to_xz ?RHS)
=> eapply (transitive_eq (to_xz LHS) ltac:(auto using projective_to_xz) H); f_equiv; group ()
| H: ladder_invariant _ _ _ |- ladder_invariant _ _ _
=> refine (proj1 (Proper_ladder_invariant _ _ (reflexivity _) _ _ _ _ _ _) H); group ()
| H: ladder_invariant _ _ _ |- ladder_invariant _ _ _
=> refine (proj1 (Proper_ladder_invariant _ _ (reflexivity _) _ _ _ _ _ _) (ladder_invariant_swap _ _ _ H)); group ()
| |- ?P => match type of P with Prop => split end
end. }
{ (* measure decreases *)
cbv [Let_In]; break_match; cbn; rewrite Z.succ_pred; apply Znat.Z2Nat.inj_lt; lia. }
{ (* if loop exited, invariant implies postcondition *)
replace i with ((-(1))%Z) in * by lia; clear Hi Hbranch.
rewrite Z.succ_m1, Z.shiftr_0_r in *.
cbv [M.cswap];
destruct swap eqn:Hswap; rewrite <-!to_x_inv00 by assumption;
etransitivity; try eapply to_x_to_xz; f_equal;
eauto using projective_to_xz, proper_to_x_projective. } }
Qed.
(* Using montladder_correct_0 in the combined correctness theorem requires
additionally showing that the right-hand-side is 0. This comes from there
being two points such that to_x gives 0: infinity and (0, 0). *)
Lemma X0_opp_0 (P : M.point) (H : X0 P = 0) : X0 (Mopp P) = 0.
Proof. t. Qed.
Lemma X0_add_0 (P Q : M.point) (HP : X0 P = 0) (HQ : X0 Q = 0) : X0(Madd P Q) = 0.
Proof. cbv [X0] in *; t. Qed.
Lemma X0_scalarmult_0 (n : Z) (P : M.point) (H : X0 P = 0) : X0 (scalarmult n P) = 0.
Proof.
clear dependent scalarbits.
induction n using Z.peano_rect_strong; try t.
{ unfold scalarmult_ref. rewrite Z.peano_rect_succ by lia. fold (scalarmult n P).
auto using X0_add_0. }
{ unfold scalarmult_ref. rewrite Z.peano_rect_pred by lia. fold (scalarmult n P).
auto using X0_add_0, X0_opp_0. }
Qed.
(* Combine the two cases together. *)
Lemma montladder_correct n P :
montladder scalarbits (Z.testbit n) (X0 P) = X0 (scalarmult (n mod 2^scalarbits) P).
Proof.
destruct (dec (X0 P = 0)) as [Hz|?]; auto using montladder_correct_nz .
rewrite montladder_correct_0, X0_scalarmult_0 by trivial; reflexivity.
Qed.
End MontgomeryCurve.
End M.