Contents:

Library ConCert.Examples.Dexter2.Dexter2CPMMCorrect

Dexter 2 CPMM contract

This file contains an implementation of the Dexter2 CPMM contract https://gitlab.com/dexter2tz/dexter2tz/-/blob/1cec9d9333eba756603d6cd90ea9c70d482a5d3d/dexter.mligo In addition this file contains proof of functional correctness w.r.t the informal specification https://gitlab.com/dexter2tz/dexter2tz/-/blob/1cec9d9333eba756603d6cd90ea9c70d482a5d3d/docs/informal-spec/dexter2-cpmm.md
This contract is an implementation of a Constant Product Market Maker (CPMM). When paired with a FA1.2 or FA2 token contract and a Dexter2 liquidity contract, this contract serves as a decentralized exchange allowing users to trade between XTZ and tokens. Additionally, users can also add or withdraw funds from the exchanges trading reserves. Traders pay a 0.3
From ConCert.Utils Require Import Automation.
From ConCert.Utils Require Import Extras.
From ConCert.Utils Require Import RecordUpdate.
From ConCert.Execution Require Import Blockchain.
From ConCert.Execution Require Import Serializable.
From ConCert.Execution Require Import InterContractCommunication.
From ConCert.Execution Require Import ContractCommon.
From ConCert.Execution Require Import ResultMonad.
From ConCert.Examples.FA2 Require Import FA2Token.
From ConCert.Examples.FA2 Require Import FA2LegacyInterface.
From ConCert.Examples.Dexter2 Require Import Dexter2FA12.
From ConCert.Examples.Dexter2 Require Dexter2FA12Correct.
From ConCert.Examples.Dexter2 Require Import Dexter2CPMM. Import DEX2.
From Coq Require Import ZArith_base.
From Coq Require Import List. Import ListNotations.
From Coq Require Import Lia.

Properties

Section Theories.
  Existing Instance BaseTypes.
  Open Scope N_scope.
  Global Arguments amount_to_N /.

  (* Tactics and facts about helper functions (omitted) *)

Set baker correct

set_baker only changes freezeBaker in state
  Lemma set_baker_state_eq : forall prev_state new_state chain ctx new_acts param,
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (SetBaker param))) = Ok (new_state, new_acts) ->
      prev_state<| freezeBaker := param.(freezeBaker_) |> = new_state.
  Proof.
    intros * receive_some.
    contract_simpl.
  Qed.

  Lemma set_baker_freeze_baker_correct : forall prev_state new_state chain ctx new_acts param,
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (SetBaker param))) = Ok (new_state, new_acts) ->
      new_state.(freezeBaker) = param.(freezeBaker_).
  Proof.
    intros * receive_some.
    apply set_baker_state_eq in receive_some.
    now subst.
  Qed.

set_baker produces no new_acts
  Lemma set_baker_new_acts_correct : forall chain ctx prev_state param new_state new_acts,
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (SetBaker param))) = Ok (new_state, new_acts) ->
      new_acts = set_delegate_call param.(baker).
  Proof.
    intros * receive_some.
    contract_simpl.
  Qed.

If the requirements are met then receive on set_baker msg must succeed and if receive on set_baker msg succeeds then requirements must hold
  Lemma set_baker_is_some : forall prev_state chain ctx param,
    (ctx_amount ctx <= 0)%Z /\
    prev_state.(selfIsUpdatingTokenPool) = false /\
    ctx.(ctx_from) = prev_state.(manager) /\
    prev_state.(freezeBaker) = false
    <->
    exists new_state new_acts, receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (SetBaker param))) = Ok (new_state, new_acts).
  Proof.
    split;
      intros;
      destruct_hyps;
      contract_simpl;
      propify;
      try easy;
      now destruct_address_eq.
  Qed.

Set manager correct

set_manager only changes manager in state
  Lemma set_manager_state_eq : forall prev_state new_state chain ctx new_acts new_manager,
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (SetManager new_manager))) = Ok (new_state, new_acts) ->
      prev_state<| manager := new_manager |> = new_state.
  Proof.
    intros * receive_some.
    contract_simpl.
  Qed.

  Lemma set_manager_manager_correct : forall prev_state new_state chain ctx new_acts new_manager,
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (SetManager new_manager))) = Ok (new_state, new_acts) ->
      new_state.(manager) = new_manager.
  Proof.
    intros * receive_some.
    apply set_manager_state_eq in receive_some.
    now subst.
  Qed.

set_manager produces no new_acts
  Lemma set_manager_new_acts_correct : forall chain ctx prev_state new_manager new_state new_acts,
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (SetManager new_manager))) = Ok (new_state, new_acts) ->
      new_acts = [].
  Proof.
    intros * receive_some.
    contract_simpl.
  Qed.

If the requirements are met then receive on set_manager msg must succeed and if receive on set_manager msg succeeds then requirements must hold
  Lemma set_manager_is_some : forall prev_state chain ctx new_manager,
    (ctx_amount ctx <= 0)%Z /\
    prev_state.(selfIsUpdatingTokenPool) = false /\
    ctx.(ctx_from) = prev_state.(manager)
    <->
    exists new_state new_acts, receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (SetManager new_manager))) = Ok (new_state, new_acts).
  Proof.
    split;
      intros;
      destruct_hyps;
      contract_simpl;
      propify;
      try easy;
      now destruct_address_eq.
  Qed.

Set liquidity address correct

set_lqt_address only changes lqtAddress in state
  Lemma set_lqt_address_state_eq : forall prev_state new_state chain ctx new_acts new_lqt_address,
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (SetLqtAddress new_lqt_address))) = Ok (new_state, new_acts) ->
      prev_state<| lqtAddress := new_lqt_address |> = new_state.
  Proof.
    intros * receive_some.
    contract_simpl.
  Qed.

  Lemma set_lqt_address_correct : forall prev_state new_state chain ctx new_acts new_lqt_address,
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (SetLqtAddress new_lqt_address))) = Ok (new_state, new_acts) ->
      new_state.(lqtAddress) = new_lqt_address.
  Proof.
    intros * receive_some.
    apply set_lqt_address_state_eq in receive_some.
    now subst.
  Qed.

set_lqt_address produces no new_acts
  Lemma set_lqt_address_new_acts_correct : forall chain ctx prev_state new_lqt_address new_state new_acts,
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (SetLqtAddress new_lqt_address))) = Ok (new_state, new_acts) ->
      new_acts = [].
  Proof.
    intros * receive_some.
    contract_simpl.
  Qed.

If the requirements are met then receive on set_lqt_address msg must succeed and if receive on set_lqt_address msg succeeds then requirements must hold
  Lemma set_lqt_address_is_some : forall prev_state chain ctx new_lqt_address,
    (ctx_amount ctx <= 0)%Z /\
    prev_state.(selfIsUpdatingTokenPool) = false /\
    ctx.(ctx_from) = prev_state.(manager) /\
    prev_state.(lqtAddress) = null_address
    <->
    exists new_state new_acts, receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (SetLqtAddress new_lqt_address))) = Ok (new_state, new_acts).
  Proof.
    split;
      intros;
      destruct_hyps;
      contract_simpl;
      propify;
      try easy;
      now destruct_address_eq.
  Qed.

Default entrypoint correct

default_ only changes xtzPool in state
  Lemma default_state_eq : forall prev_state new_state chain ctx new_acts,
    receive_cpmm chain ctx prev_state None = Ok (new_state, new_acts) ->
      prev_state<| xtzPool := prev_state.(xtzPool) + amount_to_N ctx.(ctx_amount) |> = new_state.
  Proof.
    intros * receive_some.
    contract_simpl.
  Qed.

  Lemma default_correct : forall prev_state new_state chain ctx new_acts,
    receive_cpmm chain ctx prev_state None = Ok (new_state, new_acts) ->
      new_state.(xtzPool) = prev_state.(xtzPool) + amount_to_N ctx.(ctx_amount).
  Proof.
    intros * receive_some.
    apply default_state_eq in receive_some.
    now subst.
  Qed.

default_ produces no new_acts
  Lemma default_new_acts_correct : forall chain ctx prev_state new_state new_acts,
    receive_cpmm chain ctx prev_state None = Ok (new_state, new_acts) ->
      new_acts = [].
  Proof.
    intros * receive_some.
    contract_simpl.
  Qed.

If the requirements are met then receive on None msg must succeed and if receive on None msg succeeds then requirements must hold
  Lemma default_entrypoint_is_some : forall prev_state chain ctx,
    prev_state.(selfIsUpdatingTokenPool) = false
    <->
    exists new_state new_acts, receive_cpmm chain ctx prev_state None = Ok (new_state, new_acts).
  Proof.
    split;
      intros;
      destruct_hyps;
      now contract_simpl.
  Qed.

Update token pool correct

update_token_pool only changes selfIsUpdatingTokenPool in state
  Lemma update_token_pool_state_eq : forall prev_state new_state chain ctx new_acts,
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg UpdateTokenPool)) = Ok (new_state, new_acts) ->
      prev_state<| selfIsUpdatingTokenPool := true |> = new_state.
  Proof.
    intros * receive_some.
    contract_simpl.
  Qed.

  Lemma update_token_pool_correct : forall prev_state new_state chain ctx new_acts,
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg UpdateTokenPool)) = Ok (new_state, new_acts) ->
      new_state.(selfIsUpdatingTokenPool) = true.
  Proof.
    intros * receive_some.
    apply update_token_pool_state_eq in receive_some.
    now subst.
  Qed.

update_token_pool produces a call act with amount = 0, calling the token contract with a balance of request
  Lemma update_token_pool_new_acts_correct : forall chain ctx prev_state new_state new_acts,
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg UpdateTokenPool)) = Ok (new_state, new_acts) ->
      new_acts = [
        act_call prev_state.(tokenAddress) 0%Z (serialize
          (msg_balance_of (Build_balance_of_param
            ([Build_balance_of_request ctx.(ctx_contract_address) prev_state.(tokenId)])
            (FA2LegacyInterface.Build_callback _ None ctx.(ctx_contract_address)))))
      ].
  Proof.
    intros * receive_some.
    now contract_simpl.
  Qed.

If the requirements are met then receive on update_token_pool msg must succeed and if receive on update_token_pool msg succeeds then requirements must hold
  Lemma update_token_pool_is_some : forall prev_state chain ctx,
    (ctx_amount ctx <= 0)%Z /\
    prev_state.(selfIsUpdatingTokenPool) = false /\
    ctx.(ctx_from) = ctx.(ctx_origin)
    <->
    exists new_state new_acts, receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg UpdateTokenPool)) = Ok (new_state, new_acts).
  Proof.
    split;
      intros;
      destruct_hyps;
      contract_simpl;
      propify;
      try easy;
      now destruct_address_eq.
  Qed.

  Tactic Notation "invert_responses_Some" :=
    match goal with
    | [H : match ?rs with
           | [] => _
           | _ => _
           end = Ok _ |- _] =>
        match type of rs with
        | list balance_of_response =>
            destruct rs; inversion H; clear H
        | _ => fail "No match on list of balance_of_response"
        end
    end.

Update token pool internal correct

update_token_pool_internal only changes selfIsUpdatingTokenPool and tokenPool in state
  Lemma update_token_pool_internal_state_eq : forall prev_state new_state chain ctx new_acts responses,
    receive_cpmm chain ctx prev_state (Some (FA2Token.receive_balance_of_param responses)) = Ok (new_state, new_acts) ->
      prev_state<| selfIsUpdatingTokenPool := false |>
                <| tokenPool := match responses with
                                | [] => 0
                                | response :: t => response.(balance)
                                end |> = new_state.
  Proof.
    intros * receive_some.
    contract_simpl.
    now invert_responses_Some.
  Qed.

  Lemma update_token_pool_internal_update_correct : forall prev_state new_state chain ctx new_acts responses,
    receive_cpmm chain ctx prev_state (Some (FA2Token.receive_balance_of_param responses)) = Ok (new_state, new_acts) ->
      new_state.(selfIsUpdatingTokenPool) = false.
  Proof.
    intros * receive_some.
    apply update_token_pool_internal_state_eq in receive_some.
    now subst.
  Qed.

update_token_pool_internal produces no new actions
  Lemma update_token_pool_internal_new_acts_correct : forall chain ctx prev_state new_state new_acts responses,
    receive_cpmm chain ctx prev_state (Some (FA2Token.receive_balance_of_param responses)) = Ok (new_state, new_acts) ->
      new_acts = [].
  Proof.
    intros * receive_some.
    contract_simpl.
  Qed.

If the requirements are met then receive on update_token_pool_internal msg must succeed and if receive on update_token_pool_internal msg succeeds then requirements must hold
  Lemma update_token_pool_internal_is_some : forall prev_state chain ctx responses,
    (ctx_amount ctx <= 0)%Z /\
    prev_state.(selfIsUpdatingTokenPool) = true /\
    ctx.(ctx_from) = prev_state.(tokenAddress) /\
    responses <> []
    <->
    exists new_state new_acts, receive_cpmm chain ctx prev_state (Some (FA2Token.receive_balance_of_param responses)) = Ok (new_state, new_acts).
  Proof.
    split;
      intros;
      destruct_hyps;
      contract_simpl; eauto;
      destruct responses;
      propify;
      destruct_or_hyps;
      try easy;
      now destruct_address_eq.
  Qed.

Add liquidity correct

add_liquidity only changes lqtTotal, tokenPool and xtzPool in state
  Lemma add_liquidity_state_eq : forall prev_state new_state chain ctx new_acts param,
    let lqt_minted := amount_to_N ctx.(ctx_amount) * prev_state.(lqtTotal) / prev_state.(xtzPool) in
    let tokens_deposited := ceildiv_ (amount_to_N ctx.(ctx_amount) * prev_state.(tokenPool)) prev_state.(xtzPool) in
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (AddLiquidity param))) = Ok (new_state, new_acts) ->
      prev_state<| lqtTotal := prev_state.(lqtTotal) + lqt_minted |>
                <| tokenPool := prev_state.(tokenPool) + tokens_deposited |>
                <| xtzPool := prev_state.(xtzPool) + amount_to_N ctx.(ctx_amount) |> = new_state.
  Proof.
    intros * receive_some.
    contract_simpl.
    now math_convert.
  Qed.

  Lemma add_liquidity_correct : forall prev_state new_state chain ctx new_acts param,
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (AddLiquidity param))) = Ok (new_state, new_acts) ->
      new_state.(lqtTotal) = prev_state.(lqtTotal) + amount_to_N ctx.(ctx_amount) * prev_state.(lqtTotal) / prev_state.(xtzPool) /\
      new_state.(tokenPool) = prev_state.(tokenPool) + ceildiv_ (amount_to_N ctx.(ctx_amount) * prev_state.(tokenPool)) prev_state.(xtzPool) /\
      new_state.(xtzPool) = prev_state.(xtzPool) + amount_to_N ctx.(ctx_amount).
  Proof.
    intros * receive_some.
    apply add_liquidity_state_eq in receive_some.
    now subst.
  Qed.

In the informal specification it is stated that tokens should be trasnferred from owner, but in the implementation it is trasnferred from the sender. For this we assume that the implementation is correct over the informal specification since that is what other formalizations seem to have assumed
  Lemma add_liquidity_new_acts_correct : forall chain ctx prev_state new_state new_acts param,
    let lqt_minted := amount_to_N ctx.(ctx_amount) * prev_state.(lqtTotal) / prev_state.(xtzPool) in
    let tokens_deposited := ceildiv_ (amount_to_N ctx.(ctx_amount) * prev_state.(tokenPool)) prev_state.(xtzPool) in
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (AddLiquidity param))) = Ok (new_state, new_acts) ->
      new_acts =
      [
        (act_call prev_state.(tokenAddress) 0%Z
          (serialize (FA2Token.msg_transfer
          [build_transfer ctx.(ctx_from) [build_transfer_destination ctx.(ctx_contract_address) prev_state.(tokenId) tokens_deposited] None])));
        (act_call prev_state.(lqtAddress) 0%Z
          (serialize (Dexter2FA12.msg_mint_or_burn {| target := param.(owner); quantity := Z.of_N lqt_minted|})))
      ].
  Proof.
    intros * receive_some.
    contract_simpl.
    now math_convert.
  Qed.

If the requirements are met then receive on add_liquidity msg must succeed and if receive on add_liquidity msg succeeds then requirements must hold
  Lemma add_liquidity_is_some : forall prev_state chain ctx param,
    let lqt_minted := amount_to_N ctx.(ctx_amount) * prev_state.(lqtTotal) / prev_state.(xtzPool) in
    let tokens_deposited := ceildiv_ (amount_to_N ctx.(ctx_amount) * prev_state.(tokenPool)) prev_state.(xtzPool) in
    prev_state.(selfIsUpdatingTokenPool) = false /\
    (current_slot chain < param.(add_deadline))%nat /\
    tokens_deposited <= param.(maxTokensDeposited) /\
    param.(minLqtMinted) <= lqt_minted /\
    prev_state.(xtzPool) <> 0 /\
    prev_state.(lqtAddress) <> null_address
    <->
    exists new_state new_acts, receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (AddLiquidity param))) = Ok (new_state, new_acts).
  Proof.
    split;
      intros;
      destruct_hyps;
      contract_simpl;
      propify;
      math_convert;
      unfold amount_to_N in *;
      try easy;
      try now destruct_address_eq.

  Qed.

Remove liquidity correct

remove_liquidity only changes lqtTotal, tokenPool and xtzPool in state
  Lemma remove_liquidity_state_eq : forall prev_state new_state chain ctx new_acts param,
    let xtz_withdrawn := (param.(lqtBurned) * prev_state.(xtzPool)) / prev_state.(lqtTotal) in
    let tokens_withdrawn := (param.(lqtBurned) * prev_state.(tokenPool)) / prev_state.(lqtTotal) in
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (RemoveLiquidity param))) = Ok (new_state, new_acts) ->
      prev_state<| lqtTotal := prev_state.(lqtTotal) - param.(lqtBurned) |>
                <| tokenPool := prev_state.(tokenPool) - tokens_withdrawn |>
                <| xtzPool := prev_state.(xtzPool) - xtz_withdrawn |> = new_state.
  Proof.
    intros * receive_some.
    contract_simpl.
    now math_convert; cbv.
  Qed.

  Lemma remove_liquidity_correct : forall prev_state new_state chain ctx new_acts param,
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (RemoveLiquidity param))) = Ok (new_state, new_acts) ->
      new_state.(lqtTotal) = prev_state.(lqtTotal) - param.(lqtBurned) /\
      new_state.(tokenPool) = prev_state.(tokenPool) - (param.(lqtBurned) * prev_state.(tokenPool)) / prev_state.(lqtTotal) /\
      new_state.(xtzPool) = prev_state.(xtzPool) - (param.(lqtBurned) * prev_state.(xtzPool)) / prev_state.(lqtTotal).
  Proof.
    intros * receive_some.
    apply remove_liquidity_state_eq in receive_some.
    now subst.
  Qed.

remove_liquidity should produce three acts
  • A call action to LQT contract burning lqtBurned from sender
  • A call action to token contract transferring tokens_withdrawn from this contract to liquidity_to
  • A transfer action transferring xtz_withdrawn from this contract to liquidity_to
  Lemma remove_liquidity_new_acts_correct : forall chain ctx prev_state new_state new_acts param,
    let xtz_withdrawn := (param.(lqtBurned) * prev_state.(xtzPool)) / prev_state.(lqtTotal) in
    let tokens_withdrawn := (param.(lqtBurned) * prev_state.(tokenPool)) / prev_state.(lqtTotal) in
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (RemoveLiquidity param))) = Ok (new_state, new_acts) ->
      new_acts =
      [
        (act_call prev_state.(lqtAddress) 0%Z
          (serialize (Dexter2FA12.msg_mint_or_burn {| target := ctx.(ctx_from); quantity := - Z.of_N param.(lqtBurned)|})));
        (act_call prev_state.(tokenAddress) 0%Z
          (serialize (FA2Token.msg_transfer
          [build_transfer ctx.(ctx_contract_address) [build_transfer_destination param.(liquidity_to) prev_state.(tokenId) tokens_withdrawn] None])));
        (act_transfer param.(liquidity_to) (N_to_amount xtz_withdrawn))
      ].
  Proof.
    intros * receive_some.
    contract_simpl.
    math_convert.
    unfold xtz_transfer in *.
    destruct_match in *; try congruence.
    match goal with
      [ H : Ok _ = Ok _ |- _ ] => now inversion H
    end.
  Qed.

If the requirements are met then receive on remove_liquidity msg must succeed and if receive on remove_liquidity msg succeeds then requirements must hold
  Lemma remove_liquidity_is_some : forall prev_state chain ctx param,
    let xtz_withdrawn := (param.(lqtBurned) * prev_state.(xtzPool)) / prev_state.(lqtTotal) in
    let tokens_withdrawn := (param.(lqtBurned) * prev_state.(tokenPool)) / prev_state.(lqtTotal) in
    prev_state.(selfIsUpdatingTokenPool) = false /\
    (current_slot chain < param.(remove_deadline))%nat /\
    (ctx.(ctx_amount) <= 0)%Z /\
    prev_state.(lqtTotal) <> 0 /\
    param.(minXtzWithdrawn) <= xtz_withdrawn /\
    param.(minTokensWithdrawn) <= tokens_withdrawn /\
    tokens_withdrawn <= prev_state.(tokenPool) /\
    xtz_withdrawn <= prev_state.(xtzPool) /\
    param.(lqtBurned) <= prev_state.(lqtTotal) /\
    address_is_contract param.(liquidity_to) = false /\
    prev_state.(lqtAddress) <> null_address
    <->
    exists new_state new_acts, receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (RemoveLiquidity param))) = Ok (new_state, new_acts).
  Proof.
    split;
      intros;
      destruct_hyps;
      contract_simpl;
      propify;
      math_convert;
      try easy;
      try now destruct_address_eq.
    all :
      unfold xtz_transfer in *;
      destruct_match in *;
      now destruct_address_eq.
  Qed.

XTZ to token correct

xtz_to_token only changes tokenPool and xtzPool in state
  Lemma xtz_to_token_state_eq : forall prev_state new_state chain ctx new_acts param,
    let tokens_bought := ((amount_to_N ctx.(ctx_amount)) * 997 * prev_state.(tokenPool)) /
                            (prev_state.(xtzPool) * 1000 + ((amount_to_N ctx.(ctx_amount)) * 997)) in
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (XtzToToken param))) = Ok (new_state, new_acts) ->
      prev_state<| tokenPool := prev_state.(tokenPool) - tokens_bought |>
                <| xtzPool := prev_state.(xtzPool) + amount_to_N ctx.(ctx_amount) |> = new_state.
  Proof.
    intros * receive_some.
    contract_simpl.
    now math_convert.
  Qed.

  Lemma xtz_to_token_correct : forall prev_state new_state chain ctx new_acts param,
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (XtzToToken param))) = Ok (new_state, new_acts) ->
      new_state.(tokenPool) = prev_state.(tokenPool) - (((amount_to_N ctx.(ctx_amount)) * 997 * prev_state.(tokenPool)) /
                            (prev_state.(xtzPool) * 1000 + ((amount_to_N ctx.(ctx_amount)) * 997))) /\
      new_state.(xtzPool) = prev_state.(xtzPool) + amount_to_N ctx.(ctx_amount).
  Proof.
    intros * receive_some.
    apply xtz_to_token_state_eq in receive_some.
    now subst.
  Qed.

xtz_to_token should produce one action
  • A call action to token contract transferring tokens_bought from this contract to tokens_to
  Lemma xtz_to_token_new_acts_correct : forall chain ctx prev_state new_state new_acts param,
    let tokens_bought := ((amount_to_N ctx.(ctx_amount)) * 997 * prev_state.(tokenPool)) /
                            (prev_state.(xtzPool) * 1000 + ((amount_to_N ctx.(ctx_amount)) * 997)) in
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (XtzToToken param))) = Ok (new_state, new_acts) ->
      new_acts =
      [
        (act_call prev_state.(tokenAddress) 0%Z
          (serialize (FA2Token.msg_transfer
          [build_transfer ctx.(ctx_contract_address) [build_transfer_destination param.(tokens_to) prev_state.(tokenId) tokens_bought] None])))
      ].
  Proof.
    intros * receive_some.
    contract_simpl.
    now math_convert.
  Qed.

If the requirements are met then receive on xtz_to_token msg must succeed and if receive on xtz_to_token msg succeeds then requirements must hold
  Lemma xtz_to_token_is_some : forall prev_state chain ctx param,
    let tokens_bought := ((amount_to_N ctx.(ctx_amount)) * 997 * prev_state.(tokenPool)) /
                            (prev_state.(xtzPool) * 1000 + ((amount_to_N ctx.(ctx_amount)) * 997)) in
    prev_state.(selfIsUpdatingTokenPool) = false /\
    (current_slot chain < param.(xtt_deadline))%nat /\
    (prev_state.(xtzPool) <> 0 \/ (0 < ctx.(ctx_amount))%Z) /\
    param.(minTokensBought) <= tokens_bought /\
    tokens_bought <= prev_state.(tokenPool)
    <->
    exists new_state new_acts, receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (XtzToToken param))) = Ok (new_state, new_acts).
  Proof.
    split;
      intros;
      destruct_hyps;
      contract_simpl;
      propify;
      math_convert;
      unfold amount_to_N in *;
      try easy;
      now destruct_address_eq.
  Qed.

Token to xtz correct

token_to_xtz only changes tokenPool and xtzPool in state
  Lemma token_to_xtz_state_eq : forall prev_state new_state chain ctx new_acts param,
    let xtz_bought := (param.(tokensSold) * 997 * prev_state.(xtzPool)) /
                            (prev_state.(tokenPool) * 1000 + (param.(tokensSold) * 997)) in
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (TokenToXtz param))) = Ok (new_state, new_acts) ->
      prev_state<| tokenPool := prev_state.(tokenPool) + param.(tokensSold) |>
                <| xtzPool := prev_state.(xtzPool) - xtz_bought |> = new_state.
  Proof.
    intros * receive_some.
    contract_simpl.
    now math_convert.
  Qed.

  Lemma token_to_xtz_correct : forall prev_state new_state chain ctx new_acts param,
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (TokenToXtz param))) = Ok (new_state, new_acts) ->
      new_state.(tokenPool) = prev_state.(tokenPool) + param.(tokensSold) /\
      new_state.(xtzPool) = prev_state.(xtzPool) - ((param.(tokensSold) * 997 * prev_state.(xtzPool)) /
                            (prev_state.(tokenPool) * 1000 + (param.(tokensSold) * 997))).
  Proof.
    intros * receive_some.
    apply token_to_xtz_state_eq in receive_some.
    now subst.
  Qed.

token_to_xtz should produce two actions
  • A call action to token contract transferring tokens_sold from sender to this contract
  • A transfer action transferring xtz_bought from this contract to xtz_to
  Lemma token_to_xtz_new_acts_correct : forall chain ctx prev_state new_state new_acts param,
    let xtz_bought := (param.(tokensSold) * 997 * prev_state.(xtzPool)) /
                            (prev_state.(tokenPool) * 1000 + (param.(tokensSold) * 997)) in
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (TokenToXtz param))) = Ok (new_state, new_acts) ->
      new_acts =
      [
        (act_call prev_state.(tokenAddress) 0%Z
          (serialize (FA2Token.msg_transfer
          [build_transfer ctx.(ctx_from) [build_transfer_destination ctx.(ctx_contract_address) prev_state.(tokenId) param.(tokensSold)] None])));
        (act_transfer param.(xtz_to) (N_to_amount xtz_bought))
      ].
  Proof.
    intros * receive_some.
    contract_simpl.
    math_convert.
    unfold xtz_transfer in *.
    destruct_match in *; try congruence.
    match goal with
      [H : Ok _ = Ok _ |- _] => now inversion H
    end.
  Qed.

If the requirements are met then receive on token_to_xtz msg must succeed and if receive on token_to_xtz msg succeeds then requirements must hold
  Lemma token_to_xtz_is_some : forall prev_state chain ctx param,
    let xtz_bought := (param.(tokensSold) * 997 * prev_state.(xtzPool)) /
                            (prev_state.(tokenPool) * 1000 + (param.(tokensSold) * 997)) in
    prev_state.(selfIsUpdatingTokenPool) = false /\
    (current_slot chain < param.(ttx_deadline))%nat /\
    (ctx.(ctx_amount) <= 0)%Z /\
    param.(minXtzBought) <= xtz_bought /\
    (prev_state.(tokenPool) <> 0 \/ param.(tokensSold) <> 0) /\
    address_is_contract param.(xtz_to) = false /\
    xtz_bought <= prev_state.(xtzPool)
    <->
    exists new_state new_acts, receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (TokenToXtz param))) = Ok (new_state, new_acts).
  Proof.
    split;
      intros;
      destruct_hyps;
      contract_simpl;
      propify;
      math_convert;
      try easy;
      try now destruct_address_eq.
    all :
      unfold xtz_transfer in *;
      destruct_match in *;
      now destruct_address_eq.
  Qed.

Token to token correct

token_to_token only changes tokenPool and xtzPool in state
  Lemma token_to_token_state_eq : forall prev_state new_state chain ctx new_acts param,
    let xtz_bought := (param.(tokensSold_) * 997 * prev_state.(xtzPool)) /
                            (prev_state.(tokenPool) * 1000 + (param.(tokensSold_) * 997)) in
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (TokenToToken param))) = Ok (new_state, new_acts) ->
      prev_state<| tokenPool := prev_state.(tokenPool) + param.(tokensSold_) |>
                <| xtzPool := prev_state.(xtzPool) - xtz_bought |> = new_state.
  Proof.
    intros * receive_some.
    contract_simpl.
    now math_convert.
  Qed.

  Lemma token_to_token_correct : forall prev_state new_state chain ctx new_acts param,
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (TokenToToken param))) = Ok (new_state, new_acts) ->
      new_state.(tokenPool) = prev_state.(tokenPool) + param.(tokensSold_) /\
      new_state.(xtzPool) = prev_state.(xtzPool) - ((param.(tokensSold_) * 997 * prev_state.(xtzPool)) /
                            (prev_state.(tokenPool) * 1000 + (param.(tokensSold_) * 997))).
  Proof.
    intros * receive_some.
    apply token_to_token_state_eq in receive_some.
    now subst.
  Qed.

token_to_token should produce two actions
  Lemma token_to_token_new_acts_correct : forall chain ctx prev_state new_state new_acts param,
    let xtz_bought := (param.(tokensSold_) * 997 * prev_state.(xtzPool)) /
                            (prev_state.(tokenPool) * 1000 + (param.(tokensSold_) * 997)) in
    receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (TokenToToken param))) = Ok (new_state, new_acts) ->
      new_acts =
      [
        (act_call prev_state.(tokenAddress) 0%Z
          (serialize (FA2Token.msg_transfer
          [build_transfer ctx.(ctx_from) [build_transfer_destination ctx.(ctx_contract_address) prev_state.(tokenId) param.(tokensSold_)] None])));
        (act_call param.(outputDexterContract) (N_to_amount xtz_bought)
          (serialize ((FA2Token.other_msg (XtzToToken
          {| tokens_to := param.(to_);
             minTokensBought := param.(minTokensBought_);
             xtt_deadline := param.(ttt_deadline) |})))))
      ].
  Proof.
    intros * receive_some.
    contract_simpl.
    now math_convert.
  Qed.

If the requirements are met then receive on token_to_token msg must succeed and if receive on token_to_token msg succeeds then requirements must hold
  Lemma token_to_token_is_some : forall prev_state chain ctx param,
    let xtz_bought := (param.(tokensSold_) * 997 * prev_state.(xtzPool)) /
                            (prev_state.(tokenPool) * 1000 + (param.(tokensSold_) * 997)) in
    prev_state.(selfIsUpdatingTokenPool) = false /\
    (current_slot chain < param.(ttt_deadline))%nat /\
    (ctx.(ctx_amount) <= 0)%Z /\
    xtz_bought <= prev_state.(xtzPool) /\
    (prev_state.(tokenPool) <> 0 \/ param.(tokensSold_) <> 0)
    <->
    exists new_state new_acts, receive_cpmm chain ctx prev_state (Some (FA2Token.other_msg (TokenToToken param))) = Ok (new_state, new_acts).
  Proof.
    split;
      intros;
      destruct_hyps;
      contract_simpl;
      propify;
      math_convert;
      try easy;
      now destruct_address_eq.
  Qed.

Init correct

  Lemma init_state_eq : forall chain ctx setup state,
    init_cpmm chain ctx setup = Ok state ->
      state = {|
        tokenPool := 0;
        xtzPool := 0;
        lqtTotal := setup.(lqtTotal_);
        selfIsUpdatingTokenPool := false;
        freezeBaker := false;
        manager := setup.(manager_);
        tokenAddress := setup.(tokenAddress_);
        lqtAddress := null_address;
        tokenId := setup.(tokenId_)
      |}.
  Proof.
    intros * init_some.
    now contract_simpl.
  Qed.

  Lemma init_correct : forall chain ctx setup state,
    init_cpmm chain ctx setup = Ok state ->
      tokenPool state = 0 /\
      xtzPool state = 0 /\
      lqtTotal state = setup.(lqtTotal_) /\
      selfIsUpdatingTokenPool state = false /\
      freezeBaker state = false /\
      manager state = setup.(manager_) /\
      tokenAddress state = setup.(tokenAddress_) /\
      lqtAddress state = null_address /\
      tokenId state = setup.(tokenId_).
  Proof.
    intros * init_some.
    apply init_state_eq in init_some.
    now subst.
  Qed.

Initialization should always succeed
  Lemma init_is_some : forall chain ctx setup,
    exists state, init_cpmm chain ctx setup = state.
  Proof.
    eauto.
  Qed.


Outgoing acts facts

If contract emits self calls then they are for the XtzToToken entrypoint or default entrypoint
  Lemma self_calls' bstate caddr :
    reachable bstate ->
    env_contracts bstate caddr = Some (contract : WeakContract) ->
    exists cstate,
      contract_state bstate caddr = Some cstate /\
      (tokenAddress cstate <> caddr ->
      Forall (fun act_body =>
        match act_body with
        | act_transfer to _ => True
        | act_call to _ msg => to = caddr ->
            (exists p, msg = serialize (FA2Token.other_msg (XtzToToken p))) \/
            (exists p, msg = serialize (msg_mint_or_burn p))
        | _ => False
        end) (outgoing_acts bstate caddr)).
  Proof.
    contract_induction; intros; cbn in *; auto.
    - now apply list.Forall_cons in IH as [_ IH].
    - destruct_message;
        rewrite_acts_correct;
        rewrite_state_eq;
        try (apply Forall_app; split);
        try apply IH; auto;
        rewrite ?list.Forall_cons, ?list.Forall_nil;
        try easy.
    - destruct_message;
        rewrite_acts_correct;
        rewrite_state_eq;
        try (apply Forall_app; split);
        apply list.Forall_cons in IH as [? IH];
        try apply IH; auto;
        rewrite ?list.Forall_cons, ?list.Forall_nil;
        try easy.
    - now rewrite <- perm.
    - solve_facts.
  Qed.

  Local Open Scope Z_scope.
  Lemma call_amount_zero bstate caddr :
    reachable bstate ->
    env_contracts bstate caddr = Some (contract : WeakContract) ->
      Forall (fun act_body =>
        match act_body with
        | act_transfer _ amount => True
        | act_call _ amount msg => amount = 0 \/ exists p, msg = serialize (FA2Token.other_msg (XtzToToken p))
        | act_deploy amount _ _ => amount = 0
        end) (outgoing_acts bstate caddr).
  Proof.
    intros.
    apply (lift_outgoing_acts_prop contract); auto.
    intros * receive_some.
    cbn in receive_some.
    destruct_message;
      rewrite_acts_correct;
      rewrite ?list.Forall_cons, list.Forall_nil;
      easy.
  Qed.

  Lemma no_contract_deployment bstate caddr :
    reachable bstate ->
    env_contracts bstate caddr = Some (contract : WeakContract) ->
      Forall (fun act_body =>
        match act_body with
        | act_deploy amount _ _ => False
        | _ => True
        end) (outgoing_acts bstate caddr).
  Proof.
    intros.
    apply (lift_outgoing_acts_prop contract); auto.
    intros * receive_some.
    cbn in receive_some.
    destruct_message;
      rewrite_acts_correct; auto.
  Qed.

Contract balance facts

  Lemma contract_balance_correct' : forall bstate caddr (trace : ChainTrace empty_state bstate),
    let effective_balance := (env_account_balances bstate caddr - (sumZ (fun act => act_body_amount act) (outgoing_acts bstate caddr)))%Z in
    env_contracts bstate caddr = Some (contract : WeakContract) ->
    exists cstate deploy_info,
      contract_state bstate caddr = Some cstate /\
      deployment_info Setup trace caddr = Some deploy_info /\
       (deploy_info.(deployment_amount) = 0%Z -> Z.of_N (xtzPool cstate) = effective_balance).
  Proof.
    intros.
    subst effective_balance.
    contract_induction; intros; auto.
    - cbn in *.
      apply init_correct in init_some as (_ & ? & _).
      lia.
    - cbn in IH.
      lia.
    - instantiate (CallFacts := fun _ ctx _ _ _ =>
        (0 <= ctx_amount ctx)%Z).
      unfold CallFacts in facts.
      cbn in receive_some.
      destruct_message;
        rewrite_acts_correct;
        rewrite_state_eq;
        rewrite_receive_is_some;
        unfold N_to_amount in *;
        try match goal with
        | H : (?x <= ?y)%Z, G : (?y <= ?x)%Z |- _ => apply Z.le_antisymm in H; auto; rewrite H in *
        end;
        cbn;
        try lia.
    - unfold CallFacts in facts.
      cbn in receive_some.
      destruct head;
        auto;
        cbn in IH;
        destruct action_facts as (? & ? & ?);
        subst;
      destruct_message;
        rewrite_acts_correct;
        rewrite_state_eq;
        rewrite_receive_is_some;
        unfold N_to_amount in *;
        try match goal with
        | H : (?x <= ?y)%Z, G : (?y <= ?x)%Z |- _ => apply Z.le_antisymm in H; auto; rewrite H in *
        end;
        cbn;
        try lia.
    - now erewrite sumZ_permutation in IH by eauto.
    - solve_facts.
      now apply Z.ge_le.
  Qed.

  Definition no_transfers (queue : list ActionBody) :=
    Forall (fun act_body =>
      match act_body with
      | act_transfer to _ => False
      | act_call to _ msg => forall p, msg <> serialize (FA2Token.other_msg (XtzToToken p))
      | _ => True
      end) queue.

  Lemma contract_balance_correct : forall bstate caddr (trace : ChainTrace empty_state bstate),
    env_contracts bstate caddr = Some (contract : WeakContract) ->
    exists cstate deploy_info,
      contract_state bstate caddr = Some cstate /\
      deployment_info Setup trace caddr = Some deploy_info /\
       (deploy_info.(deployment_amount) = 0%Z ->
        no_transfers (outgoing_acts bstate caddr) ->
          Z.of_N (xtzPool cstate) = env_account_balances bstate caddr).
  Proof.
    intros * deployed.
    eapply contract_balance_correct' in deployed as balance_correct.
    destruct balance_correct as (cstate & deploy_info & deployed_state & ? & balance_correct).
    eapply call_amount_zero in deployed as amount_zero; try now constructor.
    do 2 eexists.
    destruct_and_split; eauto.
    intros.
    rewrite balance_correct by auto.
    clear balance_correct.
    rename H1 into no_transfer.
    unfold no_transfers in no_transfer.
    assert (sum_zero : sumZ (fun act : ActionBody => act_body_amount act) (outgoing_acts bstate caddr) = 0%Z).
    - induction outgoing_acts; auto.
      apply list.Forall_cons in no_transfer as (no_transfer & no_transfers).
      apply list.Forall_cons in amount_zero as (amount_zero & amounts_zero).
      destruct a; auto; cbn; rewrite IHl by auto;
        clear IHl no_transfers amounts_zero.
      + now destruct amount_zero as [-> | []].
      + now subst.
    - lia.
  Qed.

  Lemma xtz_pool_bound : forall bstate caddr (trace : ChainTrace empty_state bstate),
    let effective_balance := (env_account_balances bstate caddr - (sumZ (fun act => act_body_amount act) (outgoing_acts bstate caddr)))%Z in
    env_contracts bstate caddr = Some (contract : WeakContract) ->
    exists cstate,
      contract_state bstate caddr = Some cstate /\
       Z.of_N (xtzPool cstate) <= effective_balance.
  Proof.
    intros.
    subst effective_balance.
    contract_induction; intros; auto.
    - instantiate (DeployFacts := fun _ ctx =>
        (0 <= ctx_amount ctx)%Z).
      unfold DeployFacts in facts.
      cbn in *.
      apply init_correct in init_some as (_ & -> & _).
      lia.
    - cbn in IH.
      lia.
    - instantiate (CallFacts := fun _ ctx _ _ _ =>
        (0 <= ctx_amount ctx)%Z).
      unfold CallFacts in facts.
      cbn in receive_some.
      destruct_message;
        rewrite_acts_correct;
        rewrite_state_eq;
        rewrite_receive_is_some;
        unfold N_to_amount in *;
        try match goal with
        | H : (?x <= ?y)%Z, G : (?y <= ?x)%Z |- _ => apply Z.le_antisymm in H; auto; rewrite H in *
        end;
        cbn;
        try lia.
    - unfold CallFacts in facts.
      cbn in receive_some.
      destruct head;
        auto;
        cbn in IH;
        destruct action_facts as (? & ? & ?);
        subst;
      destruct_message;
        rewrite_acts_correct;
        rewrite_state_eq;
        rewrite_receive_is_some;
        unfold N_to_amount in *;
        try match goal with
        | H : (?x <= ?y)%Z, G : (?y <= ?x)%Z |- _ => apply Z.le_antisymm in H; auto; rewrite H in *
        end;
        cbn;
        try lia.
    - now erewrite sumZ_permutation in IH by eauto.
    - solve_facts.
      + cbn.
        lia.
      + now apply Z.ge_le.
  Qed.

  Lemma transfer_bound bstate caddr :
    let transfered_balance := sumZ (fun act => act_body_amount act) (outgoing_acts bstate caddr) in
    reachable bstate ->
    env_contracts bstate caddr = Some (contract : WeakContract) ->
    exists cstate : State,
      contract_state bstate caddr = Some cstate /\
      transfered_balance <= env_account_balances bstate caddr.
  Proof.
    intros.
    subst transfered_balance.
    contract_induction; intros; cbn in *; auto.
    - instantiate (DeployFacts := fun _ ctx =>
        (0 <= ctx_amount ctx)%Z).
      auto.
    - lia.
    - instantiate (CallFacts := fun _ ctx state out_acts _ =>
        (0 <= ctx_amount ctx)%Z /\
        Z.of_N (xtzPool state) <= ctx_contract_balance ctx- sumZ (fun act => act_body_amount act) out_acts).
      destruct facts.
      destruct_message;
        rewrite_acts_correct;
        rewrite_receive_is_some;
        unfold N_to_amount in *;
        try match goal with
        | H : (?x <= ?y)%Z, G : (?y <= ?x)%Z |- _ => apply Z.le_antisymm in H; auto; rewrite H in *
        end;
        cbn;
        try lia.
    - destruct facts.
      destruct head;
        auto;
        cbn in IH;
        destruct action_facts as (? & ? & ?);
        subst.
      + contract_simpl.
        cbn in *.
        lia.
      + destruct_message;
          rewrite_acts_correct;
          rewrite_receive_is_some;
          unfold N_to_amount in *;
          try match goal with
          | H : (?x <= ?y)%Z, G : (?y <= ?x)%Z |- _ => apply Z.le_antisymm in H; auto; rewrite H in *
          end;
          cbn in *;
          try lia.
    - now rewrite <- perm.
    - solve_facts.
      + cbn.
        lia.
      + split.
        * now apply Z.ge_le.
        * specialize (account_balance_nonnegative bstate_from to_addr) as ?H.
          edestruct xtz_pool_bound as (? & deployed_state' & ?); eauto.
          cbn in *.
          rewrite deployed_state in deployed_state'.
          rewrite deployed_state in deployed_state0.
          rewrite deployed_state0 in deployed_state'.
          inversion deployed_state'.
          subst.
          destruct_address_eq; try easy; lia.
  Qed.

Total supply correct

  Definition mintedOrBurnedTokens_acts (act_body : ActionBody) : Z :=
    match act_body with
    | act_call _ _ msg_serialized =>
      match @deserialize Dexter2FA12.Msg D2LqtSInstances.msg_serializable msg_serialized with
      | Some msg => mintedOrBurnedTokens (Some msg)
      | _ => 0
      end
    | _ => 0
    end.

  Definition mintedOrBurnedTokens_tx (tx : Tx) : Z :=
    match tx.(tx_body) with
    | tx_call (Some msg_serialized) =>
      match @deserialize Dexter2FA12.Msg D2LqtSInstances.msg_serializable msg_serialized with
      | Some msg => mintedOrBurnedTokens (Some msg)
      | _ => 0
      end
    | _ => 0
    end.

  Lemma deserialize_balance_of_ne_mint_or_burn : forall n m,
    @deserialize Dexter2FA12.Msg D2LqtSInstances.msg_serializable (serialize (FA2Token.msg_balance_of n)) <>
    Some (Dexter2FA12.msg_mint_or_burn m).
  Proof.
    intros.
    Transparent serialize deserialize.
    cbn.
    rewrite !Nat2Z.id.
    destruct (Z.of_nat 3 <? 0); auto.
    cbn.
    destruct_match; try discriminate.
    destruct p.
    now destruct_match.
    Opaque serialize deserialize.
  Qed.

  Lemma forall_filter_cons : forall P (Q : Action -> ActionBody) R x l,
    Forall P (map Q (filter R (x :: l))) -> Forall P (map Q (filter R l)).
  Proof.
    intros * forall_l. cbn in forall_l.
    destruct_match in forall_l; auto.
    now eapply Forall_inv_tail.
  Qed.

  Lemma outgoing_acts_no_mint_before_set_lqt_addr : forall bstate caddr (trace : ChainTrace empty_state bstate),
    env_contracts bstate caddr = Some (contract : WeakContract) ->
    exists cstate,
      contract_state bstate caddr = Some cstate /\
        (cstate.(lqtAddress) = null_address ->
          Forall (fun act_body =>
            match act_body with
            | act_call _ _ msg =>
              match @deserialize Dexter2FA12.Msg D2LqtSInstances.msg_serializable msg with
              | Some (msg_mint_or_burn _) => False
              | _ => True
              end
            | _ => True
            end) (outgoing_acts bstate caddr)).
  Proof.
    intros * trace deployed.
    trace_induction; cbn in *.
    - (* Step block *)
      destruct IH as (state & deployed_state & IH).
      rewrite outgoing_acts_after_block_nil; eauto.
      eapply contract_addr_format; eauto.
      now constructor.
    - (* Action transfer *)
      destruct IH as (state & deployed_state & sum_eq).
      eexists.
      split; eauto.
      intros lqtAddr.
      subst.
      apply sum_eq in lqtAddr.
      clear sum_eq.
      unfold outgoing_acts in *.
      rewrite queue_prev, queue_new in *.
      now apply forall_filter_cons in lqtAddr.
    - (* Action deploy *)
      destruct (address_eqb_spec caddr to_addr) as [<-|].
      + (* Deploy this contract *)
        inversion deployed.
        subst. clear deployed IH.
        apply wc_init_strong in init_some as (setup' & state' & _ & deployed_state' & init_some).
        subst.
        rewrite deserialize_serialize.
        eexists.
        split; eauto.
        intros lqtAddr.
        rewrite outgoing_acts_after_deploy_nil; auto.
        rewrite queue_new.
        apply undeployed_contract_no_out_queue in not_deployed; auto.
        * rewrite queue_prev in not_deployed.
          now apply list.Forall_cons in not_deployed.
        * now constructor.
      + (* Deploy other contract *)
        destruct IH as (state' & deployed_state' & sum_eq); auto.
        eexists.
        split; eauto.
        subst.
        intros lqtAddr.
        apply sum_eq in lqtAddr.
        clear sum_eq.
        unfold outgoing_acts in *.
        rewrite queue_prev, queue_new in *.
        now apply forall_filter_cons in lqtAddr.
    - (* Action call *)
      destruct IH as (state & deployed_state' & sum_eq).
      destruct (address_eqb_spec caddr to_addr) as [<-|].
      + (* Call this contract *)
        rewrite deployed in deployed0.
        inversion deployed0.
        subst. clear deployed0.
        apply wc_receive_strong in receive_some as
          (prev_state' & msg' & new_state' & serialize_prev_state & _ & serialize_new_state & receive_some).
        cbn in receive_some.
        rewrite <- serialize_new_state, deserialize_serialize.
        rewrite deployed_state, serialize_prev_state in deployed_state'.
        inversion deployed_state'. subst.
        clear deployed_state' serialize_prev_state.
        eexists.
        split; eauto.
        intros lqtAddr.
        destruct_message;
          rewrite_acts_correct;
          rewrite_state_eq;
          try (apply sum_eq in lqtAddr as lqtAddr'; clear sum_eq);
          unfold outgoing_acts in *;
          rewrite queue_prev, queue_new in *;
          try apply forall_filter_cons in lqtAddr';
          auto;
          cbn;
          rewrite ?address_eq_refl;
          cbn;
          rewrite ?list.Forall_cons;
          repeat split; try easy;
          rewrite_receive_is_some;
          try easy.
        * subst.
          now eapply forall_filter_cons.
        * destruct_match eqn:match_deser; auto.
          destruct m; auto.
          now apply deserialize_balance_of_ne_mint_or_burn in match_deser.
      + (* Call other contract *)
        eexists.
        split; eauto.
        intros lqtAddr.
        apply sum_eq in lqtAddr.
        clear sum_eq.
        unfold outgoing_acts in *.
        rewrite queue_prev, queue_new in *.
        apply forall_filter_cons in lqtAddr.
        subst.
        rewrite filter_app, map_app, Forall_app.
        split; auto.
        rewrite Extras.filter_map.
        cbn.
        rewrite address_eq_ne, filter_false by auto.
        now cbn.
    - (* Invalid action *)
      destruct IH as (state & deployed_state & sum_eq).
      eexists.
      split; eauto.
      intros lqtAddr.
      unfold outgoing_acts in *.
      rewrite queue_prev, <- queue_new in *.
      apply sum_eq in lqtAddr.
      clear sum_eq.
      now apply forall_filter_cons in lqtAddr.
    - (* Permutation *)
      destruct IH as (state & deployed_state & sum_eq).
      eexists.
      split; eauto.
      intros lqtAddr.
      apply sum_eq in lqtAddr.
      clear sum_eq.
      eapply Permutation_filter in perm.
      eapply Permutation.Permutation_map in perm.
      eapply forall_respects_permutation; eauto.
  Qed.

  Lemma outgoing_acts_all_mint_same_dest : forall bstate caddr (trace : ChainTrace empty_state bstate),
    env_contracts bstate caddr = Some (contract : WeakContract) ->
    exists cstate,
      contract_state bstate caddr = Some cstate /\
        Forall (fun act_body =>
          match act_body with
          | act_call to _ msg_serialized =>
            match @deserialize Dexter2FA12.Msg D2LqtSInstances.msg_serializable msg_serialized with
            | Some (msg_mint_or_burn _) => to = lqtAddress cstate
            | _ => True
            end
          | _ => True
          end
        ) (outgoing_acts bstate caddr).
  Proof.
    contract_induction;
      intros; auto.
    - cbn.
      now apply list.Forall_cons in IH as [].
    - instantiate (CallFacts := fun _ _ state out_acts _ =>
        state.(lqtAddress) = null_address ->
          Forall (fun act_body =>
            match act_body with
            | act_call _ _ msg =>
              match @deserialize Dexter2FA12.Msg D2LqtSInstances.msg_serializable msg with
              | Some (msg_mint_or_burn _) => False
              | _ => True
              end
            | _ => True
            end) out_acts).
      unfold CallFacts in facts.
      cbn in receive_some.
      destruct_message;
        rewrite_acts_correct;
        rewrite_state_eq;
        apply Forall_app;
        split; auto;
        rewrite ?list.Forall_cons, ?list.Forall_nil;
        try easy.
      + cbn.
        repeat split; auto.
        now rewrite deserialize_serialize.
      + cbn.
        repeat split; auto.
        now rewrite deserialize_serialize.
      + rewrite_receive_is_some.
        apply facts in H2.
        apply All_Forall.In_Forall.
        intros act act_in.
        eapply Forall_forall in H2; eauto.
        destruct act; auto.
        destruct_match; auto.
        destruct m; auto.
      + cbn.
        destruct_match eqn:contradiction; auto.
        destruct m; auto.
        now apply deserialize_balance_of_ne_mint_or_burn in contradiction.
    - apply list.Forall_cons in IH as [].
      unfold CallFacts in facts.
      cbn in receive_some.
      destruct_message;
        rewrite_acts_correct;
        rewrite_state_eq;
        apply Forall_app;
        split; auto;
        rewrite ?list.Forall_cons, ?list.Forall_nil;
        try easy.
      + cbn.
        repeat split; auto.
        now rewrite deserialize_serialize.
      + cbn.
        repeat split; auto.
        now rewrite deserialize_serialize.
      + rewrite_receive_is_some.
        apply facts, Forall_inv_tail in H4.
        apply All_Forall.In_Forall.
        intros act act_in.
        eapply Forall_forall in H4; eauto.
        destruct act; auto.
        destruct_match; auto.
        destruct m; auto.
      + cbn.
        destruct_match eqn:contradiction; auto.
        destruct m; auto.
        now apply deserialize_balance_of_ne_mint_or_burn in contradiction.
    - now rewrite <- perm.
    - solve_facts.
      rewrite deployed in deployed0.
      inversion deployed0.
      subst.
      clear deployed0.
      edestruct outgoing_acts_no_mint_before_set_lqt_addr as (state & state_deployed & ?); eauto.
      rewrite deployed_state0 in state_deployed.
      now inversion state_deployed.
  Qed.

  Lemma outgoing_acts_sum_filter_eq : forall bstate caddr,
    reachable bstate ->
    env_contracts bstate caddr = Some (contract : WeakContract) ->
    exists cstate,
      contract_state bstate caddr = Some cstate /\
        sumZ mintedOrBurnedTokens_acts (filter (actTo cstate.(lqtAddress)) (outgoing_acts bstate caddr)) =
        sumZ mintedOrBurnedTokens_acts (outgoing_acts bstate caddr).
  Proof.
    intros * [trace] deployed.
    apply outgoing_acts_all_mint_same_dest in deployed as mint_or_burn_to_lqt_addr; auto.
    destruct mint_or_burn_to_lqt_addr as (cstate & deployed_state & mint_or_burn_to_lqt_addr).
    exists cstate.
    split; auto.
    clear trace deployed deployed_state.
    induction outgoing_acts.
    - reflexivity.
    - apply list.Forall_cons in mint_or_burn_to_lqt_addr as [mint_or_burn_to_lqt_addr IH%IHl].
      clear IHl.
      cbn.
      rewrite <- IH. clear IH.
      destruct a eqn:H; cbn; destruct_address_eq; auto.
      destruct_match; auto.
      now destruct_match.
  Qed.

  Lemma outgoing_txs_no_mint_before_set_lqt_addr : forall bstate caddr (trace : ChainTrace empty_state bstate),
    env_contracts bstate caddr = Some (contract : WeakContract) ->
    exists cstate,
      contract_state bstate caddr = Some cstate /\
        (cstate.(lqtAddress) = null_address ->
          Forall (fun tx =>
            match tx.(tx_body) with
            | tx_call (Some msg) =>
              match @deserialize Dexter2FA12.Msg D2LqtSInstances.msg_serializable msg with
              | Some (msg_mint_or_burn _) => False
              | _ => True
              end
            | _ => True
            end) (outgoing_txs trace caddr)).
  Proof.
    intros * deployed.
    remember empty_state.
    induction trace.
    - now subst.
    - subst.
      apply deployed_contract_state_typed in deployed as deployed_state.
      + destruct deployed_state as (state & deployed_state).
        exists state.
        split; auto.
        assert (reach : reachable to) by (constructor; now econstructor).
        destruct_chain_step.
        * (* Step block *)
          intros.
          destruct IHtrace as (state' & deployed_state' & sum_eq);
              try rewrite_environment_equiv; auto.
          cbn in *.
          rewrite deployed_state in deployed_state'.
          inversion deployed_state'.
          now subst.
        * destruct_action_eval.
         -- (* Action transfer *)
            destruct IHtrace as (state' & deployed_state' & sum_eq);
              try rewrite_environment_equiv; auto.
            cbn in *.
            rewrite deployed_state in deployed_state'.
            inversion deployed_state'.
            intros lqtAddr.
            subst.
            destruct_address_eq; auto.
            apply sum_eq in lqtAddr.
            apply Forall_cons; auto.
            now inversion lqtAddr.
         -- (* Action deploy *)
            rewrite_environment_equiv.
            cbn in *.
            intros lqtAddr.
            destruct (address_eqb_spec caddr to_addr) as [<-|]; auto.
          --- (* Deploy contract *)
              specialize undeployed_contract_no_out_txs as H; auto.
              unfold outgoing_txs in H.
              destruct_address_eq;
                try apply Forall_cons; cbn; auto;
                rewrite H; auto.
          --- (* Deploy other contract *)
              destruct IHtrace as (state' & deployed_state' & sum_eq); auto.
              rewrite deployed_state in deployed_state'.
              inversion deployed_state'.
              subst.
              destruct_address_eq; auto.
              apply sum_eq in lqtAddr.
              apply Forall_cons; auto.
              now inversion lqtAddr.
         -- (* Action call *)
            destruct IHtrace as (state' & deployed_state' & sum_eq);
              try rewrite_environment_equiv; auto.
            cbn in *.
            intros lqtAddr.
            destruct (address_eqb_spec caddr to_addr) as [<-|]; auto.
          --- (* Call contract *)
              rewrite deployed in deployed0.
              inversion deployed0.
              subst.
              apply wc_receive_strong in receive_some as
                (prev_state' & msg' & new_state' & serialize_prev_state & msg_ser & serialize_new_state & receive_some).
              cbn in receive_some.
              rewrite <- serialize_new_state, deserialize_serialize in deployed_state.
              inversion deployed_state.
              rewrite deployed_state0, serialize_prev_state in deployed_state'.
              inversion deployed_state'.
              subst.
              clear deployed0 deployed_state deployed_state'.
              assert (lqt_addr_preserved : lqtAddress state' = lqtAddress state \/ exists a, msg' = Some (FA2Token.other_msg (SetLqtAddress a))).
              { destruct_message; now rewrite_state_eq. }
              destruct (address_eqb_spec caddr from_addr) as [<-|]; auto.
          ---- (* Self call *)
                rewrite address_eq_refl.
                edestruct outgoing_acts_no_mint_before_set_lqt_addr as (cstate & deployed_state' & out_acts_forall); eauto.
                cbn in deployed_state'.
                rewrite deployed_state0, serialize_prev_state in deployed_state'.
                inversion deployed_state'.
                subst. clear deployed_state'.
                unfold outgoing_acts in out_acts_forall.
                rewrite queue_prev in out_acts_forall.
                cbn in out_acts_forall.
                rewrite address_eq_refl in out_acts_forall.
                cbn in out_acts_forall.
                apply Forall_cons. cbn.
                { destruct lqt_addr_preserved as [lqt_addr_preserved | lqt_addr_preserved]; auto.
                  - rewrite <- lqt_addr_preserved in lqtAddr.
                    apply Forall_inv in out_acts_forall; auto.
                    destruct msg; auto.
                  - destruct lqt_addr_preserved as [? lqt_addr_preserved].
                    rewrite lqt_addr_preserved in receive_some.
                    rewrite_receive_is_some.
                    apply Forall_inv in out_acts_forall; auto.
                    destruct msg; auto.
                }
                destruct lqt_addr_preserved as [lqt_addr_preserved | lqt_addr_preserved];
                  try now rewrite <- lqt_addr_preserved in lqtAddr.
                destruct lqt_addr_preserved as [? lqt_addr_preserved].
                rewrite lqt_addr_preserved in receive_some.
                now rewrite_receive_is_some.
          ---- (* Call by other contract *)
              rewrite address_eq_ne by auto.
              destruct lqt_addr_preserved as [lqt_addr_preserved | lqt_addr_preserved];
                try now rewrite <- lqt_addr_preserved in lqtAddr.
              destruct lqt_addr_preserved as [? lqt_addr_preserved].
              rewrite lqt_addr_preserved in receive_some.
              now rewrite_receive_is_some.
          --- (* Call other contract *)
              rewrite deployed_state in deployed_state'.
              inversion deployed_state'.
              subst. clear deployed_state'.
              destruct_address_eq; auto.
              subst.
              apply Forall_cons; auto.
              edestruct outgoing_acts_no_mint_before_set_lqt_addr as (cstate & deployed_state' & out_acts_forall); eauto.
              cbn in deployed_state'.
              rewrite deployed_state in deployed_state'.
              inversion deployed_state'.
              subst. clear deployed_state'.
              unfold outgoing_acts in out_acts_forall.
              rewrite queue_prev in out_acts_forall.
              cbn in out_acts_forall.
              rewrite address_eq_refl in out_acts_forall.
              cbn in out_acts_forall.
              apply Forall_inv in out_acts_forall; auto.
              destruct msg; auto.
        * (* Invalid action *)
          intros lqtAddr.
          destruct IHtrace as (state' & deployed_state' & sum_eq);
            try rewrite_environment_equiv; auto.
          cbn in *.
          rewrite deployed_state in deployed_state'.
          inversion deployed_state'.
          now subst.
        * (* Permutation *)
          intros lqtAddr.
          destruct IHtrace as (state' & deployed_state' & sum_eq);
            rewrite env_eq in *; auto.
          cbn in *.
          rewrite deployed_state in deployed_state'.
          inversion deployed_state'.
          now subst.
    + constructor.
      now econstructor.
  Qed.

  Lemma outgoing_txs_all_mint_same_dest : forall bstate caddr (trace : ChainTrace empty_state bstate),
    env_contracts bstate caddr = Some (contract : WeakContract) ->
    exists cstate,
      contract_state bstate caddr = Some cstate /\
        Forall (fun tx =>
          match tx.(tx_body) with
          | tx_call (Some msg) =>
            match @deserialize Dexter2FA12.Msg D2LqtSInstances.msg_serializable msg with
            | Some (msg_mint_or_burn _) => tx.(tx_to) = lqtAddress cstate
            | _ => True
            end
          | _ => True
          end
        ) (outgoing_txs trace caddr).
  Proof.
    intros * deployed.
    remember empty_state.
    induction trace.
    - now subst.
    - subst.
      apply deployed_contract_state_typed in deployed as deployed_state.
      + destruct deployed_state as (state & deployed_state).
        exists state.
        split; auto.
        destruct_chain_step.
        * (* Step block *)
          rewrite_environment_equiv.
          destruct IHtrace as (state' & deployed_state' & sum_eq); auto.
          cbn in *.
          rewrite deployed_state in deployed_state'.
          now inversion deployed_state'.
        * destruct_action_eval.
         -- (* Action transfer *)
            destruct IHtrace as (state' & deployed_state' & sum_eq);
              try rewrite_environment_equiv; auto.
            cbn in *.
            rewrite deployed_state in deployed_state'.
            inversion deployed_state'.
            clear deployed_state'.
            subst.
            destruct_address_eq; auto.
            apply list.Forall_cons.
            now split.
         -- (* Action deploy *)
            rewrite_environment_equiv.
            cbn in *.
            destruct (address_eqb_spec caddr to_addr) as [<-|]; auto.
          --- (* Deploy contract *)
              eapply undeployed_contract_no_out_txs in not_deployed; auto.
              unfold outgoing_txs in not_deployed.
              rewrite not_deployed.
              destruct_address_eq; auto.
              now apply list.Forall_cons.
          --- (* Deploy other contract *)
              destruct IHtrace as (state' & deployed_state' & sum_eq); auto.
              rewrite deployed_state in deployed_state'.
              inversion deployed_state'.
              clear deployed_state'.
              subst.
              destruct_address_eq; auto.
              now apply list.Forall_cons.
         -- (* Action call *)
            destruct IHtrace as (state' & deployed_state' & sum_eq);
              try rewrite_environment_equiv; auto.
            cbn in *.
            destruct (address_eqb_spec caddr to_addr) as [<-|]; auto.
          --- (* Call contract *)
              rewrite deployed in deployed0.
              inversion deployed0.
              subst.
              apply wc_receive_strong in receive_some as
                (prev_state' & msg' & new_state' & serialize_prev_state & msg_ser & serialize_new_state & receive_some).
              cbn in receive_some.
              rewrite <- serialize_new_state, deserialize_serialize in deployed_state.
              inversion deployed_state.
              rewrite deployed_state0, serialize_prev_state in deployed_state'.
              inversion deployed_state'.
              subst.
              clear deployed0 deployed_state deployed_state'.
              assert (lqt_addr_preserved : lqtAddress state' = lqtAddress state \/ exists a, msg' = Some (FA2Token.other_msg (SetLqtAddress a))).
              { destruct_message; now rewrite_state_eq. }
              destruct (address_eqb_spec caddr from_addr) as [<-|]; auto.
          ---- (* Self call *)
                rewrite address_eq_refl.
                apply Forall_cons. cbn.
                { destruct lqt_addr_preserved as [<- | lqt_addr_preserved]; auto.
                  - edestruct outgoing_acts_all_mint_same_dest as (cstate & deployed_state' & out_acts_forall); eauto.
                    cbn in deployed_state'.
                    rewrite deployed_state0, serialize_prev_state in deployed_state'.
                    inversion deployed_state'.
                    subst. clear deployed_state'.
                    unfold outgoing_acts in out_acts_forall.
                    rewrite queue_prev in out_acts_forall.
                    cbn in out_acts_forall.
                    rewrite address_eq_refl in out_acts_forall.
                    cbn in out_acts_forall.
                    apply Forall_inv in out_acts_forall.
                    now destruct msg.
                  - edestruct outgoing_acts_no_mint_before_set_lqt_addr as (cstate & deployed_state' & out_acts_forall); eauto.
                    cbn in deployed_state'.
                    rewrite deployed_state0, serialize_prev_state in deployed_state'.
                    inversion deployed_state'.
                    subst. clear deployed_state'.
                    unfold outgoing_acts in out_acts_forall.
                    rewrite queue_prev in out_acts_forall.
                    cbn in out_acts_forall.
                    rewrite address_eq_refl in out_acts_forall.
                    cbn in out_acts_forall.
                    destruct lqt_addr_preserved as [? lqt_addr_preserved].
                    rewrite lqt_addr_preserved in receive_some.
                    rewrite_receive_is_some.
                    apply Forall_inv in out_acts_forall; auto.
                    destruct msg; auto.
                    destruct_match; auto.
                    now destruct m.
                }
                destruct lqt_addr_preserved as [<- | lqt_addr_preserved]; auto.
                destruct lqt_addr_preserved as [? lqt_addr_preserved].
                rewrite lqt_addr_preserved in receive_some.
                rewrite_receive_is_some.
                edestruct outgoing_txs_no_mint_before_set_lqt_addr as (cstate & deployed_state' & out_acts_forall); eauto.
                cbn in deployed_state'.
                rewrite deployed_state0, serialize_prev_state in deployed_state'.
                inversion deployed_state'.
                subst. clear deployed_state'.
                apply out_acts_forall in H2.
                clear out_acts_forall.
                apply All_Forall.In_Forall.
                intros act act_in.
                eapply Forall_forall in H2; eauto.
                destruct_match; auto.
                destruct msg0; auto.
                destruct_match; auto.
                now destruct_match.
          ---- (* Call by other contract *)
              rewrite address_eq_ne by auto.
              destruct lqt_addr_preserved as [<- | lqt_addr_preserved]; auto.
              destruct lqt_addr_preserved as [? lqt_addr_preserved].
              rewrite lqt_addr_preserved in receive_some.
              rewrite_receive_is_some.
              edestruct outgoing_txs_no_mint_before_set_lqt_addr as (cstate & deployed_state' & out_acts_forall); eauto.
              cbn in deployed_state'.
              rewrite deployed_state0, serialize_prev_state in deployed_state'.
              inversion deployed_state'.
              subst. clear deployed_state'.
              apply out_acts_forall in H2.
              clear out_acts_forall.
              apply All_Forall.In_Forall.
              intros act act_in.
              eapply Forall_forall in H2; eauto.
              destruct_match; auto.
              destruct msg0; auto.
              destruct_match; auto.
              now destruct_match.
          --- (* Call other contract *)
              rewrite deployed_state in deployed_state'.
              inversion deployed_state'.
              clear deployed_state'.
              subst.
              destruct_address_eq; auto.
              apply list.Forall_cons.
              split; auto.
              cbn.
              edestruct outgoing_acts_all_mint_same_dest as (cstate & deployed_state' & out_acts_forall); eauto.
              cbn in deployed_state'.
              rewrite deployed_state in deployed_state'.
              inversion deployed_state'.
              subst. clear deployed_state'.
              unfold outgoing_acts in out_acts_forall.
              rewrite queue_prev in out_acts_forall.
              cbn in out_acts_forall.
              rewrite address_eq_refl in out_acts_forall.
              cbn in out_acts_forall.
              apply Forall_inv in out_acts_forall.
              now destruct msg.
        * (* Invalid action *)
          destruct IHtrace as (state' & deployed_state' & sum_eq);
            try rewrite_environment_equiv; auto.
          cbn in *.
          rewrite deployed_state in deployed_state'.
          now inversion deployed_state'.
        * (* Permutation *)
          destruct IHtrace as (state' & deployed_state' & sum_eq);
            rewrite env_eq in *; auto.
          cbn in *.
          rewrite deployed_state in deployed_state'.
          now inversion deployed_state'.
    + constructor.
      now econstructor.
  Qed.

  Lemma outgoing_txs_sum_filter_eq : forall bstate caddr (trace : ChainTrace empty_state bstate),
    env_contracts bstate caddr = Some (contract : WeakContract) ->
    exists cstate,
      contract_state bstate caddr = Some cstate /\
        sumZ mintedOrBurnedTokens_tx (filter (txCallTo cstate.(lqtAddress)) (outgoing_txs trace caddr)) =
        sumZ mintedOrBurnedTokens_tx (outgoing_txs trace caddr).
  Proof.
    intros * deployed.
    apply (outgoing_txs_all_mint_same_dest _ _ trace) in deployed as mint_or_burn_to_lqt_addr; auto.
    destruct mint_or_burn_to_lqt_addr as (cstate & deployed_state & mint_or_burn_to_lqt_addr).
    exists cstate.
    split; auto.
    clear deployed deployed_state.
    induction (outgoing_txs trace caddr).
    - reflexivity.
    - apply list.Forall_cons in mint_or_burn_to_lqt_addr as [mint_or_burn_to_lqt_addr IH%IHl].
      clear IHl.
      cbn.
      rewrite <- IH. clear IH.
      destruct a eqn:H.
      destruct tx_body eqn:H1;
        auto.
      destruct_match eqn:H2; auto.
      cbn in *.
      do 3 (destruct_match; auto).
      now destruct_address_eq.
  Qed.

lqtTotal is equal to the initial tokens + minted tokens - burned tokens
  Lemma lqt_total_correct' : forall bstate caddr (trace : ChainTrace empty_state bstate),
    env_contracts bstate caddr = Some (contract : WeakContract) ->
    exists cstate depinfo,
      contract_state bstate caddr = Some cstate /\
      deployment_info Setup trace caddr = Some depinfo /\
      let initial_tokens := lqtTotal_ (deployment_setup depinfo) in
      Z.of_N cstate.(lqtTotal) = (Z.of_N initial_tokens) + sumZ mintedOrBurnedTokens_acts (outgoing_acts bstate caddr)
        + sumZ mintedOrBurnedTokens_tx (outgoing_txs trace caddr).
  Proof.
    contract_induction;
      intros; auto.
    - cbn in *.
      now apply init_correct in init_some.
    - rewrite IH.
      cbn.
      rewrite <- 3!Z.add_assoc.
      rewrite Z.add_cancel_l.
      rewrite Z.add_shuffle3.
      rewrite Z.add_cancel_l.
      rewrite Z.add_cancel_r.
      unfold mintedOrBurnedTokens_tx.
      destruct out_act.
      + now destruct tx_act_match as [_ [_ [-> | ->]]].
      + now destruct tx_act_match as [_ [_ ->]].
      + now destruct tx_act_match as [_ ->].
    - cbn in receive_some.
      destruct_message;
        try (now contract_simpl);
        rewrite_acts_correct;
        rewrite_state_eq; auto;
        rewrite_receive_is_some;
        cbn;
        try rewrite deserialize_serialize;
        destruct_match eqn:msg_deserialized;
          try now inversion msg_deserialized;
          cbn;
          try lia.
      destruct_match; try lia.
      now apply deserialize_balance_of_ne_mint_or_burn in msg_deserialized.
    - destruct head;
        auto;
        cbn in IH;
        destruct action_facts as (? & ? & ?);
        subst.
      + now contract_simpl.
      + cbn in receive_some.
        destruct_message;
          try (now contract_simpl);
          rewrite_acts_correct;
          rewrite_state_eq; auto;
          rewrite_receive_is_some;
          cbn;
          try rewrite deserialize_serialize;
          destruct_match eqn:msg_deserialized;
            try now inversion msg_deserialized;
            cbn;
            try lia.
        * destruct_match; try lia.
          now apply deserialize_balance_of_ne_mint_or_burn in msg_deserialized.
        * destruct_match eqn:msg_deserialized0;
            try now inversion msg_deserialized0.
    - now rewrite <- perm.
    - solve_facts.
  Qed.

lqtTotal is equal to the initial tokens + minted tokens - burned tokens
  Lemma lqt_total_correct : forall bstate caddr (trace : ChainTrace empty_state bstate),
    env_contracts bstate caddr = Some (contract : WeakContract) ->
    exists cstate depinfo,
      contract_state bstate caddr = Some cstate /\
      deployment_info Setup trace caddr = Some depinfo /\
      let initial_tokens := lqtTotal_ (deployment_setup depinfo) in
      Z.of_N cstate.(lqtTotal) = (Z.of_N initial_tokens) + sumZ mintedOrBurnedTokens_acts (filter (actTo cstate.(lqtAddress)) (outgoing_acts bstate caddr))
        + sumZ mintedOrBurnedTokens_tx (filter (txCallTo cstate.(lqtAddress)) (outgoing_txs trace caddr)).
  Proof.
    intros * deployed.
    eapply lqt_total_correct' in deployed as lqt_correct.
    apply outgoing_acts_sum_filter_eq in deployed as act_filter_eq; try easy.
    destruct lqt_correct as (cstate & depinfo & deployed_state & deployment_info & lqt_correct).
    destruct act_filter_eq as (cstate' & deployed_state' & act_filter_eq).
    rewrite deployed_state in deployed_state'.
    inversion deployed_state'.
    clear deployed_state'. subst.
    do 2 eexists.
    destruct_and_split; eauto.
    intros.
    rewrite act_filter_eq.
    apply (outgoing_txs_sum_filter_eq _ _ trace) in deployed as tx_filter_eq.
    destruct tx_filter_eq as (cstate & deployed_state' & tx_filter_eq).
    rewrite deployed_state in deployed_state'.
    inversion deployed_state'.
    clear deployed_state'. subst.
    rewrite tx_filter_eq.
    apply lqt_correct.
  Qed.
  Local Close Scope Z_scope.

  Lemma mintedOrBurnedTokens_call_eq_tx : forall call_info addr,
    (fun callInfo => mintedOrBurnedTokens callInfo.(call_msg)) call_info =
    (fun callInfo => mintedOrBurnedTokens_tx (contract_call_info_to_tx addr callInfo)) call_info.
  Proof.
    intros.
    destruct call_info.
    destruct call_msg.
    - cbn.
      now rewrite deserialize_serialize.
    - reflexivity.
  Qed.

  Lemma deserialize_lqt_token_msg_right_inverse : forall x (y : Dexter2FA12.Msg),
    (forall x' (y' : Address), deserialize x' = Some y' -> x' = serialize y') ->
    deserialize x = Some y ->
    x = serialize y.
  Proof.
    intros * address_right_inverse deser_some.
    Transparent deserialize serialize.
    cbn in *.
    Local Hint Resolve deserialize_nat_right_inverse
                       deserialize_N_right_inverse
                       deserialize_int_right_inverse
                       deserialize_unit_right_inverse
                       deserialize_serialized_value_right_inverse : deser.
    repeat (try match goal with
    | H : match _ with Some _ => _ | None => _ end = Some _ |- _ = _ => let H2 := fresh "H" in destruct_match eqn:H2 in H; [| discriminate]
    | H : match ?x with 0%nat => _ | S _ => _ end = Some _ |- _ = _ => destruct x; [| try discriminate]
    | H : (let (_, _) := ?p in _) = Some _ |- _ = _ => destruct p
    | H : Some _ = Some _ |- _ = _ => inversion_clear H
    | H : extract_ser_value _ _ = @Some (interp_type ser_unit) ?i |- _ = _ => apply deserialize_unit_right_inverse in H as ->; destruct i
    | H : @deserialize_product _ _ _ _ _ = Some _ |- _ = _ => apply deserialize_product_right_inverse in H as ->; try clear H
    | |- forall _ _, _ -> _ => intros * deser_some; cbn in *
    end; auto with deser).
    Opaque deserialize serialize.
  Qed.

lqtTotal/total_supply invariant

  Section LqtPoolCorrect.

    Arguments lqt_contract {_ _ _ _} _.
    Arguments lqt_total_supply_correct {_ _ _ _} _.

lqtTotal of the main contract is equal to total_supply of the liquidity token We define the statement for the liquidity token interface contract interface. LqtTokenInterface. That is, for any contract with correct signature satisfying an additional correctness property, namely total_supply is equal to the initial tokens + minted tokens - burned tokens
We prove that the invariant hold for any contract satisfying the interface
    Lemma lqt_pool_correct_interface :
      forall (i_lqt_contract : LqtTokenInterface), (* for any correct liquidity token *)
         (forall x (y : Address), deserialize x = Some y -> x = serialize y) -> (* a technical condition for serialization *)
        lqtTotal_total_supply_invariant i_lqt_contract.
    Proof.
      intros ? ? ? ? ? ? deployed_main deployed_lqt.
      apply (lqt_total_correct _ _ trace) in deployed_main as main_correct.
      destruct main_correct as (state_main & depinfo_main & deployed_state_main & deploy_info_main & main_correct).
      apply (lqt_total_supply_correct _ _ _ trace) in deployed_lqt as lqt_correct.
      destruct lqt_correct as (state_lqt & depinfo_lqt & inc_calls_lqt & deployed_state_lqt & deploy_info_lqt & inc_acts_lqt & lqt_correct).
      specialize incomming_eq_outgoing as incoming_eq.
      edestruct incoming_eq as (? & inc_acts_lqt' & calls_eq);
        [| apply deployed_main | apply deployed_lqt |].
      - intros. eapply deserialize_lqt_token_msg_right_inverse; auto.
      - setoid_rewrite inc_acts_lqt in inc_acts_lqt'.
        inversion inc_acts_lqt'.
        subst. clear inc_acts_lqt'.
        do 4 eexists.
        repeat split; eauto.
        cbn.
        intros addr_main_eq addr_lqt_eq init_pool_eq no_waiting_mint_acts.
        apply N2Z.inj.
        rewrite main_correct, lqt_correct, init_pool_eq, no_waiting_mint_acts, addr_main_eq, addr_lqt_eq.
        rewrite Z.add_0_r, Z.add_cancel_l.
        rewrite calls_eq, sumZ_map.
        apply sumZ_eq.
        intros.
        now rewrite <- mintedOrBurnedTokens_call_eq_tx.
    Qed.
  End LqtPoolCorrect.

Now, we prove that the concrete implementation of the liquidity token satisfies the inter-contract invariant