Library iris.algebra.numbers

From iris.algebra Require Export cmra local_updates proofmode_classes.
From iris.prelude Require Import options.

Natural numbers with add as the operation.

Section nat.
  Local Instance nat_valid_instance : Valid nat := λ x, True.
  Local Instance nat_validN_instance : ValidN nat := λ n x, True.
  Local Instance nat_pcore_instance : PCore nat := λ x, Some 0.
  Local Instance nat_op_instance : Op nat := plus.
  Definition nat_op x y : x y = x + y := eq_refl.
  Lemma nat_included (x y : nat) : x y x y.
  Proof. by rewrite Nat.le_sum. Qed.
  Lemma nat_ra_mixin : RAMixin nat.
  Proof.
    apply ra_total_mixin; try by eauto.
    - solve_proper.
    - intros x y z. apply Nat.add_assoc.
    - intros x y. apply Nat.add_comm.
    - by 0.
  Qed.
  Canonical Structure natR : cmra := discreteR nat nat_ra_mixin.

  Global Instance nat_cmra_discrete : CmraDiscrete natR.
  Proof. apply discrete_cmra_discrete. Qed.

  Local Instance nat_unit_instance : Unit nat := 0.
  Lemma nat_ucmra_mixin : UcmraMixin nat.
  Proof. split; apply _ || done. Qed.
  Canonical Structure natUR : ucmra := Ucmra nat nat_ucmra_mixin.

  Global Instance nat_cancelable (x : nat) : Cancelable x.
  Proof. by intros ???? ?%Nat.add_cancel_l. Qed.

  Lemma nat_local_update (x y x' y' : nat) :
    
Morally x - y = x' - y': the difference between auth and frag must stay the same with this update. Written using + due to underflow.
    x + y' = x' + y (x,y) ¬l~> (x',y').
  Proof.
    intros ??; apply local_update_unital_discretez _.
    compute -[minus plus]; lia.
  Qed.

  Global Instance nat_is_op (n1 n2 : nat) : IsOp (n1 + n2) n1 n2.
  Proof. done. Qed.
End nat.

Natural numbers with max as the operation.

Record max_nat := MaxNat { max_nat_car : nat }.
Add Printing Constructor max_nat.

Canonical Structure max_natO := leibnizO max_nat.

Section max_nat.
  Local Instance max_nat_unit_instance : Unit max_nat := MaxNat 0.
  Local Instance max_nat_valid_instance : Valid max_nat := λ x, True.
  Local Instance max_nat_validN_instance : ValidN max_nat := λ n x, True.
  Local Instance max_nat_pcore_instance : PCore max_nat := Some.
  Local Instance max_nat_op_instance : Op max_nat := λ n m, MaxNat (max_nat_car n `max` max_nat_car m).
  Definition max_nat_op x y : MaxNat x MaxNat y = MaxNat (x `max` y) := eq_refl.

  Lemma max_nat_included x y : x y max_nat_car x max_nat_car y.
  Proof.
    split.
    - intros [z ->]. simpl. lia.
    - y. rewrite /op /max_nat_op_instance. rewrite Nat.max_r; last lia. by destruct y.
  Qed.
  Lemma max_nat_ra_mixin : RAMixin max_nat.
  Proof.
    apply ra_total_mixin; apply _ || eauto.
    - intros [x] [y] [z]. repeat rewrite max_nat_op. by rewrite Nat.max_assoc.
    - intros [x] [y]. by rewrite max_nat_op Nat.max_comm.
    - intros [x]. by rewrite max_nat_op Nat.max_id.
  Qed.
  Canonical Structure max_natR : cmra := discreteR max_nat max_nat_ra_mixin.

  Global Instance max_nat_cmra_discrete : CmraDiscrete max_natR.
  Proof. apply discrete_cmra_discrete. Qed.

  Lemma max_nat_ucmra_mixin : UcmraMixin max_nat.
  Proof. split; try apply _ || done. intros [x]. done. Qed.
  Canonical Structure max_natUR : ucmra := Ucmra max_nat max_nat_ucmra_mixin.

  Global Instance max_nat_core_id (x : max_nat) : CoreId x.
  Proof. by constructor. Qed.

  Lemma max_nat_local_update (x y x' : max_nat) :
    max_nat_car x max_nat_car x' (x,y) ¬l~> (x',x').
  Proof.
    move: x y x' ⇒ [x] [y] [y'] /= ?.
    rewrite local_update_unital_discrete⇒ [[z]] _.
    rewrite 2!max_nat_op. intros [= ?].
    split; first done. apply f_equal. lia.
  Qed.

  Global Instance : IdemP (=@{max_nat}) (⋅).
  Proof. intros [x]. rewrite max_nat_op. apply f_equal. lia. Qed.

  Global Instance max_nat_is_op (x y : nat) :
    IsOp (MaxNat (x `max` y)) (MaxNat x) (MaxNat y).
  Proof. done. Qed.
End max_nat.

Natural numbers with min as the operation.

Record min_nat := MinNat { min_nat_car : nat }.
Add Printing Constructor min_nat.

Canonical Structure min_natO := leibnizO min_nat.

Section min_nat.
  Local Instance min_nat_valid_instance : Valid min_nat := λ x, True.
  Local Instance min_nat_validN_instance : ValidN min_nat := λ n x, True.
  Local Instance min_nat_pcore_instance : PCore min_nat := Some.
  Local Instance min_nat_op_instance : Op min_nat := λ n m, MinNat (min_nat_car n `min` min_nat_car m).
  Definition min_nat_op_min x y : MinNat x MinNat y = MinNat (x `min` y) := eq_refl.

  Lemma min_nat_included (x y : min_nat) : x y min_nat_car y min_nat_car x.
  Proof.
    split.
    - intros [z ->]. simpl. lia.
    - y. rewrite /op /min_nat_op_instance. rewrite Nat.min_r; last lia. by destruct y.
  Qed.
  Lemma min_nat_ra_mixin : RAMixin min_nat.
  Proof.
    apply ra_total_mixin; apply _ || eauto.
    - intros [x] [y] [z]. repeat rewrite min_nat_op_min. by rewrite Nat.min_assoc.
    - intros [x] [y]. by rewrite min_nat_op_min Nat.min_comm.
    - intros [x]. by rewrite min_nat_op_min Nat.min_id.
  Qed.
  Canonical Structure min_natR : cmra := discreteR min_nat min_nat_ra_mixin.

  Global Instance min_nat_cmra_discrete : CmraDiscrete min_natR.
  Proof. apply discrete_cmra_discrete. Qed.

  Global Instance min_nat_core_id (x : min_nat) : CoreId x.
  Proof. by constructor. Qed.

  Lemma min_nat_local_update (x y x' : min_nat) :
    min_nat_car x' min_nat_car x (x,y) ¬l~> (x',x').
  Proof.
    move: x y x' ⇒ [x] [y] [x'] /= ?.
    rewrite local_update_discrete. move⇒ [[z]|] _ /=; last done.
    rewrite 2!min_nat_op_min. intros [= ?].
    split; first done. apply f_equal. lia.
  Qed.

  Global Instance : LeftAbsorb (=) (MinNat 0) (⋅).
  Proof. done. Qed.

  Global Instance : RightAbsorb (=) (MinNat 0) (⋅).
  Proof. intros [x]. by rewrite min_nat_op_min Nat.min_0_r. Qed.

  Global Instance : IdemP (=@{min_nat}) (⋅).
  Proof. intros [x]. rewrite min_nat_op_min. apply f_equal. lia. Qed.

  Global Instance min_nat_is_op (x y : nat) :
    IsOp (MinNat (x `min` y)) (MinNat x) (MinNat y).
  Proof. done. Qed.
End min_nat.

Positive integers with Pos.add as the operation.

Section positive.
  Local Instance pos_valid_instance : Valid positive := λ x, True.
  Local Instance pos_validN_instance : ValidN positive := λ n x, True.
  Local Instance pos_pcore_instance : PCore positive := λ x, None.
  Local Instance pos_op_instance : Op positive := Pos.add.
  Definition pos_op_add x y : x y = (x + y)%positive := eq_refl.
  Lemma pos_included (x y : positive) : x y (x < y)%positive.
  Proof. by rewrite Pos.lt_sum. Qed.
  Lemma pos_ra_mixin : RAMixin positive.
  Proof.
    split; try by eauto.
    - by intros ??? →.
    - intros ???. apply Pos.add_assoc.
    - intros ??. apply Pos.add_comm.
  Qed.
  Canonical Structure positiveR : cmra := discreteR positive pos_ra_mixin.

  Global Instance pos_cmra_discrete : CmraDiscrete positiveR.
  Proof. apply discrete_cmra_discrete. Qed.

  Global Instance pos_cancelable (x : positive) : Cancelable x.
  Proof. intros n y z ??. by eapply Pos.add_reg_l, leibniz_equiv. Qed.
  Global Instance pos_id_free (x : positive) : IdFree x.
  Proof.
    intros y ??. apply (Pos.add_no_neutral x y). rewrite Pos.add_comm.
    by apply leibniz_equiv.
  Qed.

  Global Instance pos_is_op (x y : positive) : IsOp (x + y)%positive x y.
  Proof. done. Qed.
End positive.

Integers (positive and negative) with Z.add as the operation.

Section Z.
  Local Open Scope Z_scope.
  Local Instance Z_valid_instance : Valid Z := λ x, True.
  Local Instance Z_validN_instance : ValidN Z := λ n x, True.
  Local Instance Z_pcore_instance : PCore Z := λ x, Some 0.
  Local Instance Z_op_instance : Op Z := Z.add.
  Definition Z_op x y : x y = x + y := eq_refl.
  Lemma Z_ra_mixin : RAMixin Z.
  Proof.
    apply ra_total_mixin; try by eauto.
    - solve_proper.
    - intros x y z. apply Z.add_assoc.
    - intros x y. apply Z.add_comm.
    - by 0.
  Qed.
  Canonical Structure ZR : cmra := discreteR Z Z_ra_mixin.

  Global Instance Z_cmra_discrete : CmraDiscrete ZR.
  Proof. apply discrete_cmra_discrete. Qed.

  Local Instance Z_unit_instance : Unit Z := 0.
  Lemma Z_ucmra_mixin : UcmraMixin Z.
  Proof. split; apply _ || done. Qed.
  Canonical Structure ZUR : ucmra := Ucmra Z Z_ucmra_mixin.

  Global Instance Z_cancelable (x : Z) : Cancelable x.
  Proof. by intros ???? ?%Z.add_cancel_l. Qed.

The difference between auth and frag must stay the same with this update.
  Lemma Z_local_update (x y x' y' : Z) :
    x - y = x' - y' (x,y) ¬l~> (x',y').
  Proof.
    intros. rewrite local_update_unital_discretez _.
    compute -[Z.sub Z.add]; lia.
  Qed.

  Global Instance Z_is_op (n1 n2 : Z) : IsOp (n1 + n2) n1 n2.
  Proof. done. Qed.
End Z.

Integers (positive and negative) with Z.max as the operation.

Record max_Z := MaxZ { max_Z_car : Z }.
Add Printing Constructor max_Z.

Canonical Structure max_ZO := leibnizO max_Z.

Section max_Z.
  Local Open Scope Z_scope.

  Local Instance max_Z_unit_instance : Unit max_Z := MaxZ 0.
  Local Instance max_Z_valid_instance : Valid max_Z := λ x, True.
  Local Instance max_Z_validN_instance : ValidN max_Z := λ n x, True.
  Local Instance max_Z_pcore_instance : PCore max_Z := Some.
  Local Instance max_Z_op_instance : Op max_Z := λ n m,
    MaxZ (max_Z_car n `max` max_Z_car m).
  Definition max_Z_op x y : MaxZ x MaxZ y = MaxZ (x `max` y) := eq_refl.

  Lemma max_Z_included x y : x y max_Z_car x max_Z_car y.
  Proof.
    split.
    - intros [z ->]. simpl. lia.
    - y. rewrite /op /max_Z_op_instance. rewrite Z.max_r; last lia.
      by destruct y.
  Qed.
  Lemma max_Z_ra_mixin : RAMixin max_Z.
  Proof.
    apply ra_total_mixin; apply _ || eauto.
    - intros [x] [y] [z]. repeat rewrite max_Z_op. by rewrite Z.max_assoc.
    - intros [x] [y]. by rewrite max_Z_op Z.max_comm.
    - intros [x]. by rewrite max_Z_op Z.max_id.
  Qed.
  Canonical Structure max_ZR : cmra := discreteR max_Z max_Z_ra_mixin.

  Global Instance max_Z_cmra_total : CmraTotal max_ZR.
  Proof. intros x. eauto. Qed.

  Global Instance max_Z_cmra_discrete : CmraDiscrete max_ZR.
  Proof. apply discrete_cmra_discrete. Qed.

  Global Instance max_Z_core_id (x : max_Z) : CoreId x.
  Proof. by constructor. Qed.

  Lemma max_Z_local_update (x y x' : max_Z) :
    max_Z_car x max_Z_car x' (x,y) ¬l~> (x',x').
  Proof.
    move: x y x' ⇒ [x] [y] [y'] /= ?.
    rewrite local_update_discrete⇒ [[[z]|]] //= _ [?].
    split; first done. rewrite max_Z_op. f_equal. lia.
  Qed.

  Global Instance : IdemP (=@{max_Z}) (⋅).
  Proof. intros [x]. rewrite max_Z_op. apply f_equal. lia. Qed.

  Global Instance max_Z_is_op (x y : nat) :
    IsOp (MaxZ (x `max` y)) (MaxZ x) (MaxZ y).
  Proof. done. Qed.
End max_Z.