(* Modification of the file by Dominique. *)

(**************************************************************)
(*   Copyright Dominique Larchey-Wendling *                 *)
(*                                                            *)
(*                             * Affiliation LORIA -- CNRS  *)
(**************************************************************)
(*      This file is distributed under the terms of the       *)
(*         CeCILL v2 FREER SOFTWARE LICENSE AGREEMENT         *)
(**************************************************************)

Require Import List Arith Wellfounded.
Import ListNotations.

Set Implicit Arguments.

Section measure_rect.

  Variable (X : Type) (m : X ) (P : X Type).

  Hypothesis F : x, ( y, m y < m x P y) P x.

  Definition measure_rect x : P x.
  Proof.
    cut (Acc (fun x y m x < m y) x); [ revert x | ].
    + refine (
        fix loop x Dx := @F x (fun y Dy loop y _)
      ).
      apply (Acc_inv Dx), Dy.
    + apply wf_inverse_image with (f := m), lt_wf.
  Qed.


End measure_rect.

Tactic Notation "induction" "on" hyp(x) "as" ident(IH) "with" "measure" uconstr(f) :=
  pattern x; revert x; apply measure_rect with (m := fun x f); intros x IH.

Section lex_wf.

  Variable (X : Type) (R : X X Prop) (Rwf : well_founded R).

  Reserved Notation "l '<lex' m" (at level 70). (* for lexicographic product *)

  Inductive lex : list X list X Prop :=
    | lex_skip x l m : l <lex m x::l <lex x::m
    | lex_cons x y l m : length l = length m R x y x::l <lex y::m
  where "l <lex m" := (lex l m).

  Fact lex_length l m : l <lex m length l = length m.
  Proof. induction 1; simpl; auto. Qed.

  Fact lex_cons_inv l m :
          l <lex m
        match m with
            | [] False
            | y::m
            match l with
              | [] False
              | x::l x = y lex l m
                      R x y length l = length m
            end
          end.
  Proof. inversion 1; auto. Qed.

  Theorem lex_wf : well_founded lex.
  Proof.
    intros m; induction on m as IHm with measure (length m).
    destruct m as [ | y m ].
    + constructor; intros l Hl; apply lex_cons_inv in Hl; easy.
    + revert m IHm.
      induction y as [ y IHy' ] using (well_founded_induction Rwf).
      intros m IHm.
      assert (Acc lex m) as Hm.
      1: apply IHm; simpl; auto.
      assert ( x l, R x y length l = length m Acc lex (x::l)) as IHy.
      1: { intros x l Hx Hl; apply IHy'; auto.
           intros; apply IHm.
           simpl in *; rewrite Hl; auto. }
      clear IHy' IHm.
      revert Hm IHy.
      induction 1 as [ m Hm IHm ]; intros IHy.
      constructor; intros l Hl; apply lex_cons_inv in Hl.
      destruct l as [ | x l ]; try tauto.
      destruct Hl as [ ( & Hl) | (Hx & Hl) ].
      * apply IHm; auto.
        apply lex_length in Hl as ; auto.
      * apply IHy; auto.
  Qed.


End lex_wf.