clutch.foxtrot.proofmode

From clutch.con_prob_lang Require Import lang notation class_instances tactics.
From clutch.con_prob_lang Require Export wp_tactics spec_tactics.
From clutch.foxtrot Require Import weakestpre primitive_laws derived_laws.
From iris.prelude Require Import options.

(* LHS *)
#[global] Program Instance rel_logic_wptactics_base `{!foxtrotGS Σ} : GwpTacticsBase Σ unit wp.
Next Obligation. intros. by apply wp_value. Qed.
Next Obligation. intros. by apply wp_fupd. Qed.

#[global] Program Instance rel_logic_wptactics_bind `{!foxtrotGS Σ} : GwpTacticsBind Σ unit wp.
Next Obligation. intros. by apply wp_bind. Qed.

#[global] Program Instance rel_logic_wptactics_pure `{!foxtrotGS Σ} : GwpTacticsPure Σ unit true wp.
Next Obligation. intros. by eapply lifting.wp_pure_step_later. Qed.

#[global] Program Instance rel_logic_wptactics_frame_wand `{!foxtrotGS Σ} : GwpTacticsFrameWand Σ unit true wp :=
  Build_GwpTacticsFrameWand _ _ _ _ _.
Next Obligation. iIntros (???????) "H1 H2". by iApply (wp_frame_wand with "[H1][$]"). Qed.

#[global] Program Instance rel_logic_wptactics_heap `{!foxtrotGS Σ} : GwpTacticsHeap Σ unit true wp :=
  Build_GwpTacticsHeap _ _ _ _ (λ l q v, (l ↦{q} v)%I) (λ l q vs, (l ↦∗{q} vs)%I) _ _ _ _.
Next Obligation. intros. by apply wp_alloc. Qed.
Next Obligation. intros. by apply wp_allocN. Qed.
Next Obligation. intros. by apply wp_load. Qed.
Next Obligation. intros. by apply wp_store. Qed.

#[global] Program Instance rel_logic_wptactics_tape `{!foxtrotGS Σ} : GwpTacticsTapes Σ unit true wp :=
  Build_GwpTacticsTapes _ _ _ _ (λ l q N ns, (l N ( N ; ns ))%I) _ _.
Next Obligation. intros. by apply wp_alloc_tape. Qed.
Next Obligation. intros. rewrite (bi.wand_curry (lN(N;ns))). by apply wp_rand_tape. Qed.

#[global] Program Instance rel_logic_wptactics_atomic_concurrency `{!foxtrotGS Σ} : GwpTacticsAtomicConcurrency Σ unit true wp :=
  Build_GwpTacticsAtomicConcurrency _ _ _ _ (λ l q v, (l ↦{q} v)%I) _ _ _ _.
Next Obligation. intros. by apply wp_cmpxchg_fail. Qed.
Next Obligation. intros. by apply wp_cmpxchg_suc. Qed.
Next Obligation. intros. by apply wp_xchg. Qed.
Next Obligation. intros. by apply wp_faa. Qed.

(* RHS *)
#[global] Program Instance rel_logic_tptactics_pure `{!foxtrotGS Σ} : UpdPure pupd.
Next Obligation. intros. epose proof pupd_step_pure _ _ _ _ _ [] _ as H'.
                 simpl in H'. apply H'; last first; done.
Qed.

#[global] Program Instance rel_logic_tptactics_heap `{!foxtrotGS Σ} : UpdHeap pupd.
Next Obligation. iIntros. by iMod (pupd_alloc with "[$]") as (?) "[$$]". Qed.
Next Obligation. iIntros. by iMod (pupd_load with "[$]") as "[$$]". Qed.
Next Obligation. iIntros. by iMod (pupd_store with "[$]") as "[$$]". Qed.

#[global] Program Instance rel_logic_tptactics_concurrency `{!foxtrotGS Σ} : UpdAtomicConcurrency pupd.
Next Obligation. iIntros. by iMod (pupd_cmpxchg_fail with "[$]") as "[$$]". Qed.
Next Obligation. iIntros. by iMod (pupd_cmpxchg_suc with "[$]") as "[$$]". Qed.
Next Obligation. iIntros. by iMod (pupd_xchg with "[$]") as "[$$]". Qed.
Next Obligation. iIntros. by iMod (pupd_faa with "[$]") as "[$$]". Qed.
Next Obligation. iIntros. by iMod (pupd_fork with "[$]") as "[$ [% $]]". Qed.

#[global] Program Instance rel_logic_tptactics_tapes `{!foxtrotGS Σ} : UpdTapes pupd.
Next Obligation. intros ???????->. iIntros. by iMod (pupd_alloc_tape with "[$]"). Qed.

Section foxtrot_test.
  Context `{!foxtrotGS Σ}.

  Local Lemma test_wp_tp_pures j E K:
    {{{ j fill K (#2 + #2 + #2)%E }}} #3 + #3 @ E {{{ RET #6; j fill K #6 }}}.
  Proof.
    iIntros (Ψ) "Hs HΨ".
    tp_pures j.
    wp_pures.
    by iApply "HΨ".
  Qed.
End foxtrot_test.