aboutsummaryrefslogtreecommitdiff
path: root/Data/Matrix/Core.agda
diff options
context:
space:
mode:
authorJacques Comeaux <jacquesrcomeaux@protonmail.com>2026-04-02 20:27:02 -0500
committerJacques Comeaux <jacquesrcomeaux@protonmail.com>2026-04-02 20:27:02 -0500
commit6a35dcbbf1b3859b012e5f8546e8cb353898bde2 (patch)
tree2df5a1357e482917db0216583cb8060305d16265 /Data/Matrix/Core.agda
parent29dacb01350879a1f94ca100aafc058298bcb8a1 (diff)
Reorganize matrix code
Diffstat (limited to 'Data/Matrix/Core.agda')
-rw-r--r--Data/Matrix/Core.agda266
1 files changed, 266 insertions, 0 deletions
diff --git a/Data/Matrix/Core.agda b/Data/Matrix/Core.agda
new file mode 100644
index 0000000..d97cb20
--- /dev/null
+++ b/Data/Matrix/Core.agda
@@ -0,0 +1,266 @@
+{-# OPTIONS --without-K --safe #-}
+
+open import Level using (Level; _⊔_)
+open import Relation.Binary using (Setoid; Rel; IsEquivalence)
+
+module Data.Matrix.Core {c ℓ : Level} (S : Setoid c ℓ) where
+
+import Data.Vec.Relation.Binary.Equality.Setoid as PW-≈
+import Data.Vec.Relation.Binary.Pointwise.Inductive as PW
+import Relation.Binary.Reasoning.Setoid as ≈-Reasoning
+
+open import Data.Matrix.Vec using (transpose)
+open import Data.Nat using (ℕ; _+_)
+open import Data.Vec as Vec using (Vec; map; zipWith; head; tail; replicate)
+open import Data.Vec.Properties using (map-cong; map-id)
+open import Data.Vector.Core S using (Vector; Vectorₛ; _++_; ⟨⟩; ⟨⟩-!; _≊_)
+open import Data.Vector.Vec using (zipWith-map; replicate-++)
+open import Function using (id)
+open import Relation.Binary.PropositionalEquality as ≡ using (_≡_; module ≡-Reasoning)
+
+open Setoid S
+open ℕ
+open Vec.Vec
+
+private
+ variable
+ n m p : ℕ
+ A B C : ℕ
+
+private
+
+ module PW-≊ {n} = PW-≈ (Vectorₛ n)
+
+opaque
+
+ -- Matrices over a setoid
+ Matrix : Rel ℕ c
+ Matrix n m = Vec (Vector n) m
+
+ -- Pointwise equality of matrices
+ _≋_ : Rel (Matrix n m) (c ⊔ ℓ)
+ _≋_ {n} {m} A B = A PW-≊.≋ B
+
+ -- Pointwise equivalence is an equivalence relation
+ ≋-isEquiv : IsEquivalence (_≋_ {n} {m})
+ ≋-isEquiv {n} {m} = PW-≊.≋-isEquivalence m
+
+ mapRows : (Vector n → Vector m) → Matrix n p → Matrix m p
+ mapRows = map
+
+ _∥_ : Matrix A C → Matrix B C → Matrix (A + B) C
+ _∥_ M N = zipWith _++_ M N
+
+ infixr 7 _∥_
+
+ _≑_ : Matrix A B → Matrix A C → Matrix A (B + C)
+ _≑_ M N = M Vec.++ N
+
+ infixr 6 _≑_
+
+ _∷ᵥ_ : Vector A → Matrix A B → Matrix A (suc B)
+ _∷ᵥ_ V M = V Vec.∷ M
+
+ infixr 5 _∷ᵥ_
+
+ opaque
+
+ unfolding Vector
+
+ _∷ₕ_ : Vector B → Matrix A B → Matrix (suc A) B
+ _∷ₕ_ V M = zipWith _∷_ V M
+
+ infixr 5 _∷ₕ_
+
+ ∷ₕ-cong : {V V′ : Vector B} {M M′ : Matrix A B} → V ≊ V′ → M ≋ M′ → V ∷ₕ M ≋ V′ ∷ₕ M′
+ ∷ₕ-cong PW.[] PW.[] = PW.[]
+ ∷ₕ-cong (≈x PW.∷ ≊V) (≊M₀ PW.∷ ≋M) = (≈x PW.∷ ≊M₀) PW.∷ (∷ₕ-cong ≊V ≋M)
+
+ headₕ : Matrix (suc A) B → Vector B
+ headₕ M = map Vec.head M
+
+ tailₕ : Matrix (suc A) B → Matrix A B
+ tailₕ M = map Vec.tail M
+
+ head-∷-tailₕ : (M : Matrix (suc A) B) → headₕ M ∷ₕ tailₕ M ≡ M
+ head-∷-tailₕ M = begin
+ zipWith _∷_ (map Vec.head M) (map Vec.tail M) ≡⟨ zipWith-map head tail _∷_ M ⟩
+ map (λ x → head x ∷ tail x) M ≡⟨ map-cong (λ { (_ ∷ _) → ≡.refl }) M ⟩
+ map id M ≡⟨ map-id M ⟩
+ M ∎
+ where
+ open ≡-Reasoning
+
+ []ᵥ : Matrix 0 B
+ []ᵥ = replicate _ []
+
+ []ᵥ-! : (E : Matrix 0 B) → E ≡ []ᵥ
+ []ᵥ-! [] = ≡.refl
+ []ᵥ-! ([] ∷ E) = ≡.cong ([] ∷_) ([]ᵥ-! E)
+
+ []ᵥ-≑ : []ᵥ {A} ≑ []ᵥ {B} ≡ []ᵥ
+ []ᵥ-≑ {A} {B} = replicate-++ A B []
+
+ []ᵥ-∥ : (M : Matrix A B) → []ᵥ ∥ M ≡ M
+ []ᵥ-∥ [] = ≡.refl
+ []ᵥ-∥ (M₀ ∷ M) = ≡.cong (M₀ ∷_) ([]ᵥ-∥ M)
+
+ ∷ₕ-∥ : (V : Vector C) (M : Matrix A C) (N : Matrix B C) → V ∷ₕ (M ∥ N) ≡ (V ∷ₕ M) ∥ N
+ ∷ₕ-∥ [] [] [] = ≡.refl
+ ∷ₕ-∥ (x ∷ V) (M₀ ∷ M) (N₀ ∷ N) = ≡.cong ((x ∷ M₀ ++ N₀) ∷_) (∷ₕ-∥ V M N)
+
+ ∷ₕ-≑ : (V : Vector A) (W : Vector B) (M : Matrix C A) (N : Matrix C B) → (V ++ W) ∷ₕ (M ≑ N) ≡ (V ∷ₕ M) ≑ (W ∷ₕ N)
+ ∷ₕ-≑ [] W [] N = ≡.refl
+ ∷ₕ-≑ (x ∷ V) W (M₀ ∷ M) N = ≡.cong ((x ∷ M₀) ∷_) (∷ₕ-≑ V W M N)
+
+ headᵥ : Matrix A (suc B) → Vector A
+ headᵥ (V ∷ _) = V
+
+ tailᵥ : Matrix A (suc B) → Matrix A B
+ tailᵥ (_ ∷ M) = M
+
+ head-∷-tailᵥ : (M : Matrix A (suc B)) → headᵥ M ∷ᵥ tailᵥ M ≡ M
+ head-∷-tailᵥ (_ ∷ _) = ≡.refl
+
+ []ₕ : Matrix A 0
+ []ₕ = []
+
+ []ₕ-! : (E : Matrix A 0) → E ≡ []ₕ
+ []ₕ-! [] = ≡.refl
+
+ []ₕ-≑ : (M : Matrix A B) → []ₕ ≑ M ≡ M
+ []ₕ-≑ _ = ≡.refl
+
+ ∷ᵥ-≑ : (V : Vector A) (M : Matrix A B) (N : Matrix A C) → V ∷ᵥ (M ≑ N) ≡ (V ∷ᵥ M) ≑ N
+ ∷ᵥ-≑ V M N = ≡.refl
+
+infix 4 _≋_
+
+module ≋ {n} {m} = IsEquivalence (≋-isEquiv {n} {m})
+
+Matrixₛ : ℕ → ℕ → Setoid c (c ⊔ ℓ)
+Matrixₛ n m = record
+ { Carrier = Matrix n m
+ ; _≈_ = _≋_ {n} {m}
+ ; isEquivalence = ≋-isEquiv
+ }
+
+opaque
+
+ unfolding Vector
+
+ head′ : Vector (suc A) → Carrier
+ head′ = head
+
+ head-cong : {V V′ : Vector (suc A)} → V ≊ V′ → head′ V ≈ head′ V′
+ head-cong (≈x PW.∷ _) = ≈x
+
+ tail′ : Vector (suc A) → Vector A
+ tail′ = tail
+
+ tail-cong : {V V′ : Vector (suc A)} → V ≊ V′ → tail′ V ≊ tail′ V′
+ tail-cong (_ PW.∷ ≊V) = ≊V
+
+opaque
+
+ unfolding headₕ head′
+
+ ≋headₕ : {M M′ : Matrix (suc A) B} → M ≋ M′ → headₕ M ≊ headₕ M′
+ ≋headₕ M≋M′ = PW.map⁺ head-cong M≋M′
+
+ ≋tailₕ : {M M′ : Matrix (suc A) B} → M ≋ M′ → tailₕ M ≋ tailₕ M′
+ ≋tailₕ M≋M′ = PW.map⁺ tail-cong M≋M′
+
+opaque
+ unfolding _≋_ _∥_ []ᵥ _∷ₕ_
+ ∥-cong : {M M′ : Matrix A C} {N N′ : Matrix B C} → M ≋ M′ → N ≋ N′ → M ∥ N ≋ M′ ∥ N′
+ ∥-cong {zero} {C} {B} {M} {M′} {N} {N′} ≋M ≋N
+ rewrite []ᵥ-! M
+ rewrite []ᵥ-! M′ = begin
+ ([]ᵥ ∥ N) ≡⟨ []ᵥ-∥ N ⟩
+ N ≈⟨ ≋N ⟩
+ N′ ≡⟨ []ᵥ-∥ N′ ⟨
+ ([]ᵥ ∥ N′) ∎
+ where
+ open ≈-Reasoning (Matrixₛ _ _)
+ ∥-cong {suc A} {C} {B} {M} {M′} {N} {N′} ≋M ≋N
+ rewrite ≡.sym (head-∷-tailₕ M)
+ using M₀ ← headₕ M
+ using M- ← tailₕ M
+ rewrite ≡.sym (head-∷-tailₕ M′)
+ using M₀′ ← headₕ M′
+ using M-′ ← tailₕ M′ = begin
+ (M₀ ∷ₕ M-) ∥ N ≡⟨ ∷ₕ-∥ M₀ M- N ⟨
+ M₀ ∷ₕ M- ∥ N ≈⟨ ∷ₕ-cong ≊M₀ (∥-cong ≋M- ≋N) ⟩
+ M₀′ ∷ₕ M-′ ∥ N′ ≡⟨ ∷ₕ-∥ M₀′ M-′ N′ ⟩
+ (M₀′ ∷ₕ M-′) ∥ N′ ∎
+ where
+ ≊M₀ : M₀ ≊ M₀′
+ ≊M₀ = begin
+ headₕ M ≡⟨ ≡.cong headₕ (head-∷-tailₕ M) ⟨
+ headₕ (M₀ ∷ₕ M-) ≈⟨ ≋headₕ ≋M ⟩
+ headₕ (M₀′ ∷ₕ M-′) ≡⟨ ≡.cong headₕ (head-∷-tailₕ M′) ⟩
+ headₕ M′ ∎
+ where
+ open ≈-Reasoning (Vectorₛ _)
+ ≋M- : M- ≋ M-′
+ ≋M- = begin
+ tailₕ M ≡⟨ ≡.cong tailₕ (head-∷-tailₕ M) ⟨
+ tailₕ (M₀ ∷ₕ M-) ≈⟨ ≋tailₕ ≋M ⟩
+ tailₕ (M₀′ ∷ₕ M-′) ≡⟨ ≡.cong tailₕ (head-∷-tailₕ M′) ⟩
+ tailₕ M′ ∎
+ where
+ open ≈-Reasoning (Matrixₛ _ _)
+ open ≈-Reasoning (Matrixₛ _ _)
+
+opaque
+ unfolding _≑_
+ ≑-cong : {M M′ : Matrix A B} {N N′ : Matrix A C} → M ≋ M′ → N ≋ N′ → M ≑ N ≋ M′ ≑ N′
+ ≑-cong PW.[] ≋N = ≋N
+ ≑-cong (M₀≊M₀′ PW.∷ ≋M) ≋N = M₀≊M₀′ PW.∷ ≑-cong ≋M ≋N
+
+opaque
+
+ unfolding Matrix
+
+ _ᵀ : Matrix n m → Matrix m n
+ _ᵀ [] = []ᵥ
+ _ᵀ (M₀ ∷ M) = M₀ ∷ₕ M ᵀ
+
+ infix 10 _ᵀ
+
+ -ᵀ-cong : {M₁ M₂ : Matrix n m} → M₁ ≋ M₂ → M₁ ᵀ ≋ M₂ ᵀ
+ -ᵀ-cong PW.[] = ≋.refl
+ -ᵀ-cong (≊M₀ PW.∷ ≋M) = ∷ₕ-cong ≊M₀ (-ᵀ-cong ≋M)
+
+ opaque
+
+ unfolding []ᵥ []ₕ
+
+ []ᵥ-ᵀ : []ᵥ ᵀ ≡ []ₕ {A}
+ []ᵥ-ᵀ {zero} = ≡.refl
+ []ᵥ-ᵀ {suc A} = ≡.cong (zipWith _∷_ []) ([]ᵥ-ᵀ)
+
+ opaque
+
+ unfolding _∷ₕ_ Vector
+
+ ∷ₕ-ᵀ : (V : Vector A) (M : Matrix B A) → (V ∷ₕ M) ᵀ ≡ V ∷ᵥ M ᵀ
+ ∷ₕ-ᵀ [] [] = ≡.refl
+ ∷ₕ-ᵀ (x ∷ V) (M₀ ∷ M) = ≡.cong ((x ∷ M₀) ∷ₕ_) (∷ₕ-ᵀ V M)
+
+ ∷ᵥ-ᵀ : (V : Vector B) (M : Matrix B A) → (V ∷ᵥ M) ᵀ ≡ V ∷ₕ M ᵀ
+ ∷ᵥ-ᵀ V M = ≡.refl
+
+ opaque
+
+ _ᵀᵀ : (M : Matrix n m) → M ᵀ ᵀ ≡ M
+ _ᵀᵀ [] = []ᵥ-ᵀ
+ _ᵀᵀ (M₀ ∷ M) = begin
+ (M₀ ∷ₕ M ᵀ) ᵀ ≡⟨ ∷ₕ-ᵀ M₀ (M ᵀ) ⟩
+ M₀ ∷ᵥ M ᵀ ᵀ ≡⟨ ≡.cong (M₀ ∷ᵥ_) (M ᵀᵀ) ⟩
+ M₀ ∷ᵥ M ∎
+ where
+ open ≡-Reasoning
+
+ infix 10 _ᵀᵀ