proofmode.v 6.65 KB
Newer Older
Robbert Krebbers's avatar
Robbert Krebbers committed
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
From iris.proofmode Require Import coq_tactics.
From iris.proofmode Require Export weakestpre.
From iris.heap_lang Require Export wp_tactics heap.
Import uPred.

Ltac strip_later ::= iNext.

Section heap.
Context {Σ : gFunctors} `{heapG Σ}.
Implicit Types N : namespace.
Implicit Types P Q : iPropG heap_lang Σ.
Implicit Types Φ : val  iPropG heap_lang Σ.
Implicit Types Δ : envs (iResR heap_lang (globalF Σ)).

Global Instance sep_destruct_mapsto l q v :
  SepDestruct false (l {q} v) (l {q/2} v) (l {q/2} v).
Proof. by rewrite /SepDestruct heap_mapsto_op_split. Qed.

Lemma tac_wp_alloc Δ Δ' N E j e v Φ :
  to_val e = Some v  
  Δ  heap_ctx N  nclose N  E 
  StripLaterEnvs Δ Δ' 
  ( l,  Δ'',
24
25
    envs_app false (Esnoc Enil j (l  v)) Δ' = Some Δ'' 
    Δ''  Φ (LitV (LitLoc l))) 
Robbert Krebbers's avatar
Robbert Krebbers committed
26
27
  Δ  WP Alloc e @ E {{ Φ }}.
Proof.
28
29
  intros ???? HΔ. rewrite -wp_alloc // -always_and_sep_l.
  apply and_intro; first done.
Robbert Krebbers's avatar
Robbert Krebbers committed
30
31
32
33
34
35
36
37
38
39
  rewrite strip_later_env_sound; apply later_mono, forall_intro=> l.
  destruct (HΔ l) as (Δ''&?&HΔ'). rewrite envs_app_sound //; simpl.
  by rewrite right_id HΔ'.
Qed.

Lemma tac_wp_load Δ Δ' N E i l q v Φ :
  Δ  heap_ctx N  nclose N  E 
  StripLaterEnvs Δ Δ' 
  envs_lookup i Δ' = Some (false, l {q} v)%I 
  Δ'  Φ v 
40
  Δ  WP Load (Lit (LitLoc l)) @ E {{ Φ }}.
Robbert Krebbers's avatar
Robbert Krebbers committed
41
Proof.
Ralf Jung's avatar
Ralf Jung committed
42
  intros. rewrite -wp_load // -always_and_sep_l. apply and_intro; first done.
Robbert Krebbers's avatar
Robbert Krebbers committed
43
44
45
46
47
48
49
50
51
52
  rewrite strip_later_env_sound -later_sep envs_lookup_split //; simpl.
  by apply later_mono, sep_mono_r, wand_mono.
Qed.

Lemma tac_wp_store Δ Δ' Δ'' N E i l v e v' Φ :
  to_val e = Some v' 
  Δ  heap_ctx N  nclose N  E 
  StripLaterEnvs Δ Δ' 
  envs_lookup i Δ' = Some (false, l  v)%I 
  envs_simple_replace i false (Esnoc Enil i (l  v')) Δ' = Some Δ'' 
53
  Δ''  Φ (LitV LitUnit)  Δ  WP Store (Lit (LitLoc l)) e @ E {{ Φ }}.
Robbert Krebbers's avatar
Robbert Krebbers committed
54
Proof.
55
  intros. rewrite -wp_store // -always_and_sep_l. apply and_intro; first done.
Robbert Krebbers's avatar
Robbert Krebbers committed
56
57
58
59
60
61
62
63
64
65
  rewrite strip_later_env_sound -later_sep envs_simple_replace_sound //; simpl.
  rewrite right_id. by apply later_mono, sep_mono_r, wand_mono.
Qed.

Lemma tac_wp_cas_fail Δ Δ' N E i l q v e1 v1 e2 v2 Φ :
  to_val e1 = Some v1  to_val e2 = Some v2 
  Δ  heap_ctx N  nclose N  E 
  StripLaterEnvs Δ Δ' 
  envs_lookup i Δ' = Some (false, l {q} v)%I  v  v1 
  Δ'  Φ (LitV (LitBool false)) 
66
  Δ  WP CAS (Lit (LitLoc l)) e1 e2 @ E {{ Φ }}.
Robbert Krebbers's avatar
Robbert Krebbers committed
67
Proof.
68
  intros. rewrite -wp_cas_fail // -always_and_sep_l. apply and_intro; first done.
Robbert Krebbers's avatar
Robbert Krebbers committed
69
70
71
72
  rewrite strip_later_env_sound -later_sep envs_lookup_split //; simpl.
  by apply later_mono, sep_mono_r, wand_mono.
Qed.

73
Lemma tac_wp_cas_suc Δ Δ' Δ'' N E i l v e1 v1 e2 v2 Φ :
Robbert Krebbers's avatar
Robbert Krebbers committed
74
75
76
  to_val e1 = Some v1  to_val e2 = Some v2 
  Δ  heap_ctx N  nclose N  E 
  StripLaterEnvs Δ Δ' 
77
  envs_lookup i Δ' = Some (false, l  v)%I  v = v1 
Robbert Krebbers's avatar
Robbert Krebbers committed
78
  envs_simple_replace i false (Esnoc Enil i (l  v2)) Δ' = Some Δ'' 
79
  Δ''  Φ (LitV (LitBool true))  Δ  WP CAS (Lit (LitLoc l)) e1 e2 @ E {{ Φ }}.
Robbert Krebbers's avatar
Robbert Krebbers committed
80
Proof.
81
82
  intros; subst.
  rewrite -wp_cas_suc // -always_and_sep_l. apply and_intro; first done.
Robbert Krebbers's avatar
Robbert Krebbers committed
83
84
85
86
87
88
  rewrite strip_later_env_sound -later_sep envs_simple_replace_sound //; simpl.
  rewrite right_id. by apply later_mono, sep_mono_r, wand_mono.
Qed.
End heap.

Tactic Notation "wp_apply" open_constr(lem) :=
89
  lazymatch goal with
Robbert Krebbers's avatar
Robbert Krebbers committed
90
91
  | |- _  wp ?E ?e ?Q => reshape_expr e ltac:(fun K e' =>
    wp_bind K; iApply lem; try iNext)
92
  | _ => fail "wp_apply: not a 'wp'"
Robbert Krebbers's avatar
Robbert Krebbers committed
93
94
95
  end.

Tactic Notation "wp_alloc" ident(l) "as" constr(H) :=
96
  lazymatch goal with
Robbert Krebbers's avatar
Robbert Krebbers committed
97
98
99
100
101
102
103
104
105
106
107
  | |- _  wp ?E ?e ?Q => reshape_expr e ltac:(fun K e' =>
    match eval hnf in e' with
    | Alloc ?e =>
       wp_bind K; eapply tac_wp_alloc with _ _ H _;
         [wp_done || fail 2 "wp_alloc:" e "not a value"
         |iAssumption || fail 2 "wp_alloc: cannot find heap_ctx"
         |done || eauto with ndisj
         |apply _
         |intros l; eexists; split;
           [env_cbv; reflexivity || fail 2 "wp_alloc:" H "not fresh"
           |wp_finish]]
108
109
    end) || fail "wp_alloc: cannot find 'Alloc' in" e
  | _ => fail "wp_alloc: not a 'wp'"
Robbert Krebbers's avatar
Robbert Krebbers committed
110
  end.
111
112
Tactic Notation "wp_alloc" ident(l) :=
  let H := iFresh in wp_alloc l as H.
Robbert Krebbers's avatar
Robbert Krebbers committed
113
114

Tactic Notation "wp_load" :=
115
  lazymatch goal with
Robbert Krebbers's avatar
Robbert Krebbers committed
116
117
  | |- _  wp ?E ?e ?Q => reshape_expr e ltac:(fun K e' =>
    match eval hnf in e' with
118
    | Load ?l =>
Robbert Krebbers's avatar
Robbert Krebbers committed
119
120
121
122
123
124
       wp_bind K; eapply tac_wp_load;
         [iAssumption || fail 2 "wp_load: cannot find heap_ctx"
         |done || eauto with ndisj
         |apply _
         |iAssumptionCore || fail 2 "wp_cas_fail: cannot find" l "↦ ?"
         |wp_finish]
125
126
    end) || fail "wp_load: cannot find 'Load' in" e
  | _ => fail "wp_load: not a 'wp'"
Robbert Krebbers's avatar
Robbert Krebbers committed
127
128
129
  end.

Tactic Notation "wp_store" :=
130
  lazymatch goal with
Robbert Krebbers's avatar
Robbert Krebbers committed
131
132
133
134
135
136
137
138
139
140
141
  | |- _  wp ?E ?e ?Q => reshape_expr e ltac:(fun K e' =>
    match eval hnf in e' with
    | Store ?l ?e =>
       wp_bind K; eapply tac_wp_store;
         [wp_done || fail 2 "wp_store:" e "not a value"
         |iAssumption || fail 2 "wp_store: cannot find heap_ctx"
         |done || eauto with ndisj
         |apply _
         |iAssumptionCore || fail 2 "wp_store: cannot find" l "↦ ?"
         |env_cbv; reflexivity
         |wp_finish]
142
143
    end) || fail "wp_store: cannot find 'Store' in" e
  | _ => fail "wp_store: not a 'wp'"
Robbert Krebbers's avatar
Robbert Krebbers committed
144
145
146
  end.

Tactic Notation "wp_cas_fail" :=
147
  lazymatch goal with
Robbert Krebbers's avatar
Robbert Krebbers committed
148
149
  | |- _  wp ?E ?e ?Q => reshape_expr e ltac:(fun K e' =>
    match eval hnf in e' with
150
    | CAS ?l ?e1 ?e2 =>
Robbert Krebbers's avatar
Robbert Krebbers committed
151
152
153
154
155
156
157
       wp_bind K; eapply tac_wp_cas_fail;
         [wp_done || fail 2 "wp_cas_fail:" e1 "not a value"
         |wp_done || fail 2 "wp_cas_fail:" e2 "not a value"
         |iAssumption || fail 2 "wp_cas_fail: cannot find heap_ctx"
         |done || eauto with ndisj
         |apply _
         |iAssumptionCore || fail 2 "wp_cas_fail: cannot find" l "↦ ?"
158
         |try congruence
Robbert Krebbers's avatar
Robbert Krebbers committed
159
         |wp_finish]
160
161
    end) || fail "wp_cas_fail: cannot find 'CAS' in" e
  | _ => fail "wp_cas_fail: not a 'wp'"
Robbert Krebbers's avatar
Robbert Krebbers committed
162
163
164
  end.

Tactic Notation "wp_cas_suc" :=
165
  lazymatch goal with
Robbert Krebbers's avatar
Robbert Krebbers committed
166
167
  | |- _  wp ?E ?e ?Q => reshape_expr e ltac:(fun K e' =>
    match eval hnf in e' with
168
    | CAS ?l ?e1 ?e2 =>
Robbert Krebbers's avatar
Robbert Krebbers committed
169
170
171
172
173
174
175
       wp_bind K; eapply tac_wp_cas_suc;
         [wp_done || fail 2 "wp_cas_suc:" e1 "not a value"
         |wp_done || fail 2 "wp_cas_suc:" e1 "not a value"
         |iAssumption || fail 2 "wp_cas_suc: cannot find heap_ctx"
         |done || eauto with ndisj
         |apply _
         |iAssumptionCore || fail 2 "wp_cas_suc: cannot find" l "↦ ?"
176
         |try congruence
Robbert Krebbers's avatar
Robbert Krebbers committed
177
178
         |env_cbv; reflexivity
         |wp_finish]
179
180
    end) || fail "wp_cas_suc: cannot find 'CAS' in" e
  | _ => fail "wp_cas_suc: not a 'wp'"
Robbert Krebbers's avatar
Robbert Krebbers committed
181
  end.