{-

Technical results about elementary transformations

-}
{-# OPTIONS --safe #-}
module Cubical.Algebra.Matrix.Elementaries where

open import Cubical.Foundations.Prelude
open import Cubical.Foundations.Function

open import Cubical.Data.Nat hiding (_+_ ; _·_)
open import Cubical.Data.Nat.Order
open import Cubical.Data.Empty as Empty
open import Cubical.Data.Bool
open import Cubical.Data.FinData renaming (znots to znotsFin ; snotz to snotzFin)

open import Cubical.Relation.Nullary

open import Cubical.Tactics.CommRingSolver.Reflection
open import Cubical.Algebra.Ring.BigOps
open import Cubical.Algebra.CommRing

open import Cubical.Algebra.Matrix
open import Cubical.Algebra.Matrix.CommRingCoefficient
open import Cubical.Algebra.Matrix.RowTransformation

private
  variable
     : Level
    A : Type 
    m n k l : 

module ElemTransformation (𝓡 : CommRing ) where

  private
    R = 𝓡 .fst

  open CommRingStr     (𝓡 .snd) renaming ( is-set to isSetR )
  open KroneckerDelta  (CommRing→Ring 𝓡)
  open Sum             (CommRing→Ring 𝓡)

  open Coefficient           𝓡
  open LinearTransformation  𝓡

  open SimRel
  open Sim

  open isLinear
  open isLinear2×2

  -- Swapping the first row with another

  swapMat : Mat 2 2
  swapMat zero zero = 0r
  swapMat zero one  = 1r
  swapMat one  zero = 1r
  swapMat one  one  = 0r

  uniSwapMat : swapMat  swapMat  𝟙
  uniSwapMat t zero zero =
    (mul2 swapMat swapMat zero zero  helper) t
    where helper : 0r · 0r + 1r · 1r  1r
          helper = solve 𝓡
  uniSwapMat t zero one  =
    (mul2 swapMat swapMat zero one   helper) t
    where helper : 0r · 1r + 1r · 0r  0r
          helper = solve 𝓡
  uniSwapMat t one  zero =
    (mul2 swapMat swapMat one  zero  helper) t
    where helper : 1r · 0r + 0r · 1r  0r
          helper = solve 𝓡
  uniSwapMat t one  one  =
    (mul2 swapMat swapMat one  one   helper) t
    where helper : 1r · 1r + 0r · 0r  1r
          helper = solve 𝓡

  isInvSwapMat2 : isInv swapMat
  isInvSwapMat2 .fst = swapMat
  isInvSwapMat2 .snd .fst = uniSwapMat
  isInvSwapMat2 .snd .snd = uniSwapMat

  swapRow2 : Mat 2 n  Mat 2 n
  swapRow2 M zero = M one
  swapRow2 M one  = M zero

  isLinear2×2SwapRow2 : isLinear2×2 {n = n} swapRow2
  isLinear2×2SwapRow2 .transMat _ = swapMat
  isLinear2×2SwapRow2 .transEq M t zero j =
    ((mul2 swapMat M zero j)  helper _ _) (~ t)
    where helper : (a b : R)  0r · a + 1r · b  b
          helper = solve 𝓡
  isLinear2×2SwapRow2 .transEq M t one  j =
    ((mul2 swapMat M one  j)  helper _ _) (~ t)
    where helper : (a b : R)  1r · a + 0r · b  a
          helper = solve 𝓡

  swapRow : (i₀ : Fin m)(M : Mat (suc m) n)  Mat (suc m) n
  swapRow i M zero = M (suc i)
  swapRow zero M one  = M zero
  swapRow zero M (suc (suc i)) = M (suc (suc i))
  swapRow (suc i₀) M one   = M one
  swapRow (suc i₀) M (suc (suc i)) = swapRow i₀ (takeRowsᶜ M) (suc i)

  swapRowMat : (i₀ : Fin m)  Mat (suc m) (suc m)
  swapRowMat = trans2RowsMat swapMat

  swapRowEq : (i₀ : Fin m)(M : Mat (suc m) n)  swapRow i₀ M  swapRowMat i₀  M
  swapRowEq {m = suc m} zero M t zero j =
      (helper1 _ _
      t  0r · M zero j + ∑Mulr1 _  i  M (suc i) j) zero (~ t))
      t  0r · M zero j +   i  helper2 (δ i zero) (M (suc i) j) t))) t
    where helper1 : (a b : R)  b  0r · a + b
          helper1 = solve 𝓡
          helper2 : (a b : R)  b · a  (1r · a) · b
          helper2 = solve 𝓡
  swapRowEq {m = suc m} zero M t one  j =
      (helper _
      t  (1r · 1r) · M zero j + ∑Mul0r  i  M (suc i) j) (~ t))) t
    where helper : (a : R)  a  (1r · 1r) · a + 0r
          helper = solve 𝓡
  swapRowEq {m = suc m} zero M t (suc (suc i)) j =
      (helper _ _
      t  (1r · 0r) · M zero j + ∑Mul1r _  l  M (suc l) j) (suc i) (~ t))) t
    where helper : (a b : R)  b  (1r · 0r) · a + b
          helper = solve 𝓡
  swapRowEq {m = suc m} (suc i₀) M t zero j =
      (helper1 _ _
      t  0r · M zero j + ∑Mul1r _  i  M (suc i) j) (suc i₀) (~ t))
      t  0r · M zero j +   l  helper2 (δ (suc i₀) l) (M (suc l) j) t))) t
    where helper1 : (a b : R)  b  0r · a + b
          helper1 = solve 𝓡
          helper2 : (a b : R)  a · b  (1r · a) · b
          helper2 = solve 𝓡
  swapRowEq {m = suc m} (suc i₀) M t one  j =
        (helper _ _ --helper1 _ _
        t  (1r · 0r) · M zero j + ∑Mul1r _  i  M (suc i) j) zero (~ t))) t
    where helper : (a b : R)  b  (1r · 0r) · a + b
          helper = solve 𝓡
  swapRowEq {m = suc m} (suc i₀) M t (suc (suc i)) j =
     ((λ t  swapRowEq i₀ (takeRowsᶜ M) t (suc i) j)
     helper _ (M one j) _) t
    where helper : (a b c : R)  a + c  a + (0r · b + c)
          helper = solve 𝓡

  isLinearSwapRow : (i : Fin m)  isLinear (swapRow {n = n} i)
  isLinearSwapRow i .transMat _ = swapRowMat i
  isLinearSwapRow i .transEq    = swapRowEq  i

  isInvSwapMat : (i : Fin m)(M : Mat (suc m) (suc n))  isInv (isLinearSwapRow i .transMat M)
  isInvSwapMat i _ = isInvTrans2RowsMat _ i isInvSwapMat2

  -- Similarity defined by swapping

  record SwapFirstRow (i : Fin m)(M : Mat (suc m) (suc n)) : Type  where
    field
      sim : Sim M
      swapEq : (j : Fin (suc n))  M (suc i) j  sim .result zero j

  record SwapFirstCol (j : Fin n)(M : Mat (suc m) (suc n)) : Type  where
    field
      sim : Sim M
      swapEq : (i : Fin (suc m))  M i (suc j)  sim .result i zero

  record SwapPivot (i : Fin (suc m))(j : Fin (suc n))(M : Mat (suc m) (suc n)) : Type  where
    field
      sim : Sim M
      swapEq : M i j  sim .result zero zero

  open SwapFirstRow
  open SwapFirstCol
  open SwapPivot

  swapFirstRow : (i : Fin m)(M : Mat (suc m) (suc n))  SwapFirstRow i M
  swapFirstRow i M .sim .result    = swapRow i M
  swapFirstRow i M .sim .simrel .transMatL = isLinearSwapRow i .transMat M
  swapFirstRow i M .sim .simrel .transMatR = 𝟙
  swapFirstRow i M .sim .simrel .transEq     = isLinearSwapRow i .transEq _  sym (⋆rUnit _)
  swapFirstRow i M .sim .simrel .isInvTransL = isInvSwapMat i M
  swapFirstRow i M .sim .simrel .isInvTransR = isInv𝟙
  swapFirstRow i M .swapEq j = refl

  swapFirstCol : (j : Fin n)(M : Mat (suc m) (suc n))  SwapFirstCol j M
  swapFirstCol j M .sim .result    = (swapRow j (M ))
  swapFirstCol j M .sim .simrel .transMatL = 𝟙
  swapFirstCol j M .sim .simrel .transMatR = (isLinearSwapRow j .transMat (M ))
  swapFirstCol j M .sim .simrel .transEq =
    let P = isLinearSwapRow j .transMat (M ) in
       t  (isLinearSwapRow j .transEq (M ) t))
     compᵗ P (M )
      t  idemᵗ M t  P )
      t  ⋆lUnit M (~ t)  P )
  swapFirstCol j M .sim .simrel .isInvTransL = isInv𝟙
  swapFirstCol j M .sim .simrel .isInvTransR =
    isInvᵗ {M = isLinearSwapRow j .transMat (M )} (isInvSwapMat j (M ))
  swapFirstCol j M .swapEq i t = M i (suc j)

  swapPivot : (i : Fin (suc m))(j : Fin (suc n))(M : Mat (suc m) (suc n))  SwapPivot i j M
  swapPivot zero zero M .sim    = idSim M
  swapPivot zero zero M .swapEq = refl
  swapPivot (suc i) zero M .sim    = swapFirstRow i M .sim
  swapPivot (suc i) zero M .swapEq = refl
  swapPivot zero (suc j) M .sim    = swapFirstCol j M .sim
  swapPivot zero (suc j) M .swapEq = refl
  swapPivot (suc i) (suc j) M .sim = compSim (swapFirstRow i M .sim) (swapFirstCol j _ .sim)
  swapPivot (suc i) (suc j) M .swapEq = refl


  -- Add the first row to another

  addMat : Mat 2 2
  addMat zero zero = 1r
  addMat zero one  = 0r
  addMat one  zero = 1r
  addMat one  one  = 1r

  subMat : Mat 2 2
  subMat zero zero = 1r
  subMat zero one  = 0r
  subMat one  zero = - 1r
  subMat one  one  = 1r

  add⋆subMat : addMat  subMat  𝟙
  add⋆subMat t zero zero =
    (mul2 addMat subMat zero zero  helper) t
    where helper : 1r · 1r + 0r · - 1r  1r
          helper = solve 𝓡
  add⋆subMat t zero one  =
    (mul2 addMat subMat zero one   helper) t
    where helper : 1r · 0r + 0r · 1r  0r
          helper = solve 𝓡
  add⋆subMat t one  zero =
    (mul2 addMat subMat one  zero  helper) t
    where helper : 1r · 1r + 1r · - 1r  0r
          helper = solve 𝓡
  add⋆subMat t one  one  =
    (mul2 addMat subMat one  one   helper) t
    where helper : 1r · 0r + 1r · 1r  1r
          helper = solve 𝓡

  sub⋆addMat : subMat  addMat  𝟙
  sub⋆addMat t zero zero =
    (mul2 subMat addMat  zero zero  helper) t
    where helper : 1r · 1r + 0r · 1r  1r
          helper = solve 𝓡
  sub⋆addMat t zero one  =
    (mul2 subMat addMat  zero one   helper) t
    where helper : 1r · 0r + 0r · 1r  0r
          helper = solve 𝓡
  sub⋆addMat t one  zero =
    (mul2 subMat addMat  one  zero  helper) t
    where helper : - 1r · 1r + 1r · 1r  0r
          helper = solve 𝓡
  sub⋆addMat t one  one  =
    (mul2 subMat addMat  one  one   helper) t
    where helper : - 1r · 0r + 1r · 1r  1r
          helper = solve 𝓡

  isInvAddMat2 : isInv addMat
  isInvAddMat2 .fst = subMat
  isInvAddMat2 .snd .fst = add⋆subMat
  isInvAddMat2 .snd .snd = sub⋆addMat

  addRow2 : Mat 2 n  Mat 2 n
  addRow2 M zero  = M zero
  addRow2 M one j = M zero j + M one j

  isLinear2AddRow2 : isLinear2×2 {n = n} addRow2
  isLinear2AddRow2 .transMat _ = addMat
  isLinear2AddRow2 .transEq M t zero j =
    ((mul2 addMat M zero j)  helper _ _) (~ t)
    where helper : (a b : R)  1r · a + 0r · b  a
          helper = solve 𝓡
  isLinear2AddRow2 .transEq M t one  j =
    ((mul2 addMat M one  j)  helper _ _) (~ t)
    where helper : (a b : R)  1r · a + 1r · b  a + b
          helper = solve 𝓡

  -- Add the first row to all other rows

  addRows : Mat (suc m) n  Mat (suc m) n
  addRows M zero = M zero
  addRows M (suc i) j = M zero j + M (suc i) j

  private
    firstRowStayInvariant : (M : Mat (suc m) n)  M zero  transRows addRow2 M zero
    firstRowStayInvariant = invRow₀ _  _  refl)

  addRowsEq : (M : Mat (suc m) n)  transRows addRow2 M  addRows M
  addRowsEq M t zero = firstRowStayInvariant M (~ t)
  addRowsEq M t one j = M zero j + M one j
  addRowsEq M t (suc (suc i)) j = takeCombShufRows addRow2  N  addRowsEq N t) M (suc (suc i)) j

  isLinearAddRows : isLinear (addRows {m = m} {n = suc n})
  isLinearAddRows =
    let isLinear = isLinearTransRows _ isLinear2AddRow2 _
    in  record
          { transMat = isLinear .transMat
          ; transEq  =  M  sym (addRowsEq M)  isLinear .transEq M) }

  isInvAddRows : (M : Mat (suc m) (suc n))  isInv (isLinearAddRows .transMat M)
  isInvAddRows = isInvTransRows _ _  _  isInvAddMat2)

-- Similarity defined by adding rows

  record AddFirstRow (M : Mat (suc m) (suc n)) : Type  where
    field
      sim : Sim M
      inv₀  : M zero  sim .result zero
      addEq : (i : Fin m)(j : Fin (suc n))  M zero j + M (suc i) j  sim .result (suc i) j

  open AddFirstRow

  addFirstRow : (M : Mat (suc m) (suc n))  AddFirstRow M
  addFirstRow M .sim .result    = addRows M
  addFirstRow M .sim .simrel .transMatL = isLinearAddRows .transMat M
  addFirstRow M .sim .simrel .transMatR = 𝟙
  addFirstRow M .sim .simrel .transEq     = isLinearAddRows .transEq _  sym (⋆rUnit _)
  addFirstRow M .sim .simrel .isInvTransL = isInvAddRows M
  addFirstRow M .sim .simrel .isInvTransR = isInv𝟙
  addFirstRow M .inv₀      = refl
  addFirstRow M .addEq i j = refl