Andrew Sneap, 17 February 2022

\begin{code}

{-# OPTIONS --safe --without-K #-}

open import MLTT.Spartan renaming (_+_ to _∔_)
open import Dyadics.Type
open import Dyadics.Negation
open import Naturals.Addition renaming (_+_ to _ℕ+_)
open import Naturals.Multiplication renaming (_*_ to _ℕ*_)
open import Integers.Type
open import Integers.Addition renaming (_+_ to _ℤ+_)
open import Integers.Multiplication
open import Integers.Negation renaming (-_ to ℤ-_)
open import Naturals.Exponentiation
open import UF.Base hiding (_≈_)

module Dyadics.Addition where

\end{code}

The usual strategy is applied to define addition of dyadic
rationals. We first define addition of unsimplified dyadics, and then
addition of dyadic rationals is defined by normalising the result of
this addition.

\begin{code}

_+'_ :  ×    ×    × 
(p , a) +' (q , b) = p * pos (2^ b) ℤ+ q * pos (2^ a) , a ℕ+ b

_+_ : ℤ[1/2]  ℤ[1/2]  ℤ[1/2]
(p , _) + (q , _) = normalise-pos (p +' q)

infixl 32 _+'_
infixl 32 _+_

\end{code}

Commutativity is trivial as usual, and follows by commutativity of
addition of integers and natural numbers. To get commutativity of
addition of dyadic rationals, we can simply take the action on paths
of normalise-pos on addition of unsimplified dyadic rationals.

\begin{code}

ℤ[1/2]+'-comm : (p q :  × )  p +' q  q +' p
ℤ[1/2]+'-comm (p , a) (q , b) = ap₂ _,_ I II
 where
  I : p * pos (2^ b) ℤ+ q * pos (2^ a)  q * pos (2^ a) ℤ+ p * pos (2^ b)
  I = ℤ+-comm (p * pos (2^ b)) (q * pos (2^ a))
  II : a ℕ+ b  b ℕ+ a
  II = addition-commutativity a b

ℤ[1/2]+-comm : (p q : ℤ[1/2])  p + q  q + p
ℤ[1/2]+-comm (p , _) (q , _) = ap normalise-pos (ℤ[1/2]+'-comm p q)

\end{code}

For associativity, it is slightly more involved. It's not possible to
take the action on paths as with commutativity, because on each side
we have two addition, meaning we apply normalise-pos on intermediate
calculations. The idea of the associativity proof is as follows:

(p , α) + (q , β) + (r , γ)
 = normalise-pos p + normalise-pos q + normalise-pos r
 = normalise-pos (p +' q) + normalise-pos r
 = normalise-pos (p +' q +' r)
 = normalise-pos (p +' (q +' r))
 = normalise-pos p + normalise (q +' r)
 = normalise-pos p + (normalise pos q + normalise-pos r)
 = (p , α) + ((q , β) + (r , γ))
 ∎

This proof requires that proof that (p , α) = normalise-pos p, which
is proved in the Dyadics.Type file. It also requires that
(normalise-pos p + normalise-pos q) = (normalise-pos (p +' q)).

This proof follows, preceded by a lemma about equivalences on
unsimplified rationals.

\begin{code}

ℤ[1/2]+'-≈' : (p q r :  × )  p ≈' q  (p +' r) ≈' (q +' r)
ℤ[1/2]+'-≈' (p , a) (q , b) (r , c) e = γ
 where
  e' : p * pos (2^ b)  q * pos (2^ a)
  e' = e
  a' = pos (2^ a)
  b' = pos (2^ b)
  c' = pos (2^ c)

  rearrangement₁ : (a : ) (b c d : )
    a * pos (2^ b) * pos (2^ (c ℕ+ d))  a * pos (2^ c) * pos (2^ (b ℕ+ d))
  rearrangement₁ a b c d = a * b'' * pos (2^ (c ℕ+ d))  =⟨ i    
                           a * b'' * pos (2^ c ℕ* 2^ d) =⟨ ii   
                           a * b'' * (c'' * pos (2^ d)) =⟨ iii  
                           a * b'' * c'' * pos (2^ d)   =⟨ iv   
                           a * (b'' * c'') * pos (2^ d) =⟨ v    
                           a * (c'' * b'') * pos (2^ d) =⟨ vi   
                           a * c'' * b'' * pos (2^ d)   =⟨ vii  
                           a * c'' * (b'' * pos (2^ d)) =⟨ viii 
                           a * c'' * pos (2^ b ℕ* 2^ d) =⟨ ix   
                           a * c'' * pos (2^ (b ℕ+ d))  
   where
    b'' = pos (2^ b)
    c'' = pos (2^ c)
    i    = ap  -  a * b'' * pos -) (prod-of-powers 2 c d ⁻¹)
    ii   = ap (a * b'' *_) (pos-multiplication-equiv-to-ℕ (2^ c) (2^ d) ⁻¹)
    iii  = ℤ*-assoc (a * b'') (c'') (pos (2^ d)) ⁻¹
    iv   = ap (_* pos (2^ d)) (ℤ*-assoc a b'' (c''))
    v    = ap  -  a * - * pos (2^ d)) (ℤ*-comm b'' (c''))
    vi   = ap (_* pos (2^ d)) (ℤ*-assoc a (c'') b'' ⁻¹)
    vii  = ℤ*-assoc (a * c'') b'' (pos (2^ d))
    viii = ap (a * c'' *_) (pos-multiplication-equiv-to-ℕ (2^ b) (2^ d))
    ix   = ap  -  a * c'' * pos - ) (prod-of-powers 2 b d)

  I : p * c' * pos (2^ (b ℕ+ c))  q * c' * pos (2^ (a ℕ+ c))
  I = p * c' * pos (2^ (b ℕ+ c))   =⟨ rearrangement₁ p c b c      
      p * b' * pos (2^ (c ℕ+ c))   =⟨ ap (_* pos (2^ (c ℕ+ c))) e 
      q * a' * pos (2^ (c ℕ+ c))   =⟨ rearrangement₁ q a c c      
      q * c' * pos (2^ (a ℕ+ c))   

  γ : (p * c' ℤ+ r * a') * pos (2^ (b ℕ+ c)) -- lhs of unfolded type
     (q * c' ℤ+ r * b') * pos (2^ (a ℕ+ c)) -- rhs of unfolded type
  γ = (p * c' ℤ+ r * a') * pos (2^ (b ℕ+ c))                   =⟨ i   
      p * c' * pos (2^ (b ℕ+ c)) ℤ+ r * a' * pos (2^ (b ℕ+ c)) =⟨ ii  
      p * c' * pos (2^ (b ℕ+ c)) ℤ+ r * b' * pos (2^ (a ℕ+ c)) =⟨ iii 
      q * c' * pos (2^ (a ℕ+ c)) ℤ+ r * b' * pos (2^ (a ℕ+ c)) =⟨ iv  
      (q * c' ℤ+ r * b') * pos (2^ (a ℕ+ c))                   
   where
    i   = distributivity-mult-over-ℤ (p * c') (r * a') (pos (2^ (b ℕ+ c)))
    ii  = ap (p * c' * pos (2^ (b ℕ+ c)) ℤ+_) (rearrangement₁ r a b c)
    iii = ap (_ℤ+ r * b' * pos (2^ (a ℕ+ c))) I
    iv  = distributivity-mult-over-ℤ (q * c') (r * b') (pos (2^ (a ℕ+ c))) ⁻¹

ℤ[1/2]+-normalise-pos : (p q :  × )
  normalise-pos (p +' q)  (normalise-pos p) + (normalise-pos q)
ℤ[1/2]+-normalise-pos p q = ≈'-to-= (p +' q) (p' +' q') γ
 where
  p' = from-ℤ[1/2] (normalise-pos p)
  q' = from-ℤ[1/2] (normalise-pos q)

  I : p ≈' p'
  I = ≈'-normalise-pos p

  II : q ≈' q'
  II = ≈'-normalise-pos q

  III : (p +' q) ≈' (p' +' q)
  III = ℤ[1/2]+'-≈' p p' q I

  IV : (q +' p') ≈' (q' +' p')
  IV = ℤ[1/2]+'-≈' q q' p' II

  V : (p' +' q) ≈' (p' +' q')
  V = transport₂ _≈'_ (ℤ[1/2]+'-comm q p') (ℤ[1/2]+'-comm q' p') IV

  γ : (p +' q) ≈' (p' +' q')
  γ = ≈'-trans (p +' q) (p' +' q) (p' +' q') III V

ℤ[1/2]+'-assoc : (p q r :  × )  (p +' q) +' r  p +' (q +' r)
ℤ[1/2]+'-assoc (p , a) (q , b) (r , c) = to-×-=' (γ₁ , γ₂)
 where
  a' = pos (2^ a)
  b' = pos (2^ b)
  c' = pos (2^ c)

  I : (p * b' ℤ+ q * a') * c'  p * (b' * c') ℤ+ q * c' * a'
  I = (p * b' ℤ+ q * a') * c'      =⟨ i   
      p * b' * c' ℤ+ q * a' * c'   =⟨ ii  
      p * b' * c' ℤ+ q * c' * a'   =⟨ iii 
      p * (b' * c') ℤ+ q * c' * a' 
   where
    i   = distributivity-mult-over-ℤ (p * b') (q * a') c'
    ii  = ap ( p * b' * c' ℤ+_) (ℤ-mult-rearrangement q a' c')
    iii = ap (_ℤ+ q * c' * a') (ℤ*-assoc p b' c')

  γ₁ : (p * b' ℤ+ q * a') * c' ℤ+ r * pos (2^ (a ℕ+ b))
      p * pos (2^ (b ℕ+ c)) ℤ+ (q * c' ℤ+ r * b') * a'
  γ₁ = (p * b' ℤ+ q * a') * c' ℤ+ r * pos (2^ (a ℕ+ b))  =⟨ i    
       (p * b' ℤ+ q * a') * c' ℤ+ r * pos (2^ a ℕ* 2^ b) =⟨ ii   
       (p * b' ℤ+ q * a') * c' ℤ+ r * (a' * b')          =⟨ iii  
       p * (b' * c') ℤ+ q * c' * a' ℤ+ r * (a' * b')     =⟨ iv   
       p * (b' * c') ℤ+ q * c' * a' ℤ+ r * (b' * a')     =⟨ v    
       p * (b' * c') ℤ+ q * c' * a' ℤ+ r * b' * a'       =⟨ vi   
       p * (b' * c') ℤ+ (q * c' * a' ℤ+ r * b' * a')     =⟨ vii  
       p * (b' * c') ℤ+ (q * c' ℤ+ r * b') * a'          =⟨ viii 
       p * pos (2^ b ℕ* 2^ c) ℤ+ (q * c' ℤ+ r * b') * a' =⟨ ix   
       p * pos (2^ (b ℕ+ c)) ℤ+ (q * c' ℤ+ r * b') * a'  
        where
         iₐₚ : 2^ (a ℕ+ b)  2^ a ℕ* 2^ b
         iₐₚ = prod-of-powers 2 a b ⁻¹
         iiₐₚ : pos (2^ a ℕ* 2^ b)  pos (2^ a) * pos (2^ b)
         iiₐₚ = pos-multiplication-equiv-to-ℕ (2^ a) (2^ b) ⁻¹
         vₐₚ : r * (b' * a')  r * b' * a'
         vₐₚ = ℤ*-assoc r b' a' ⁻¹
         viiₐₚ : q * c' * a' ℤ+ r * b' * a'  (q * c' ℤ+ r * b') * a'
         viiₐₚ = distributivity-mult-over-ℤ (q * c') (r * b') a' ⁻¹
         viiiₐₚ : pos (2^ b) * pos (2^ c)  pos (2^ b ℕ* 2^ c)
         viiiₐₚ = pos-multiplication-equiv-to-ℕ (2^ b) (2^ c)
         ixₐₚ : 2^ b ℕ* 2^ c  2 ℕ^ (b ℕ+ c)
         ixₐₚ = prod-of-powers 2 b c

         i    = ap  -  (p * b' ℤ+ q * a') * c' ℤ+ r * pos -) iₐₚ
         ii   = ap  -  (p * b' ℤ+ q * a') * c' ℤ+ r * -) iiₐₚ
         iii  = ap  -  - ℤ+ r * (a' * b')) I
         iv   = ap  -  p * (b' * c') ℤ+ q * c' * a' ℤ+ r * -) (ℤ*-comm a' b')
         v    = ap  -  p * (b' * c') ℤ+ q * c' * a' ℤ+ -) vₐₚ
         vi   = ℤ+-assoc (p * (b' * c')) ((q * c' * a')) (r * b' * a')
         vii  = ap  -  p * (b' * c') ℤ+ -) viiₐₚ
         viii = ap  -  p * - ℤ+ (q * c' ℤ+ r * b') * a') viiiₐₚ
         ix = ap  -  p * pos - ℤ+ (q * c' ℤ+ r * b') * a') ixₐₚ

  γ₂ : a ℕ+ b ℕ+ c  a ℕ+ (b ℕ+ c)
  γ₂ = addition-associativity a b c

ℤ[1/2]+-assoc : (p q r : ℤ[1/2])  p + q + r  p + (q + r)
ℤ[1/2]+-assoc (p , α) (q , β) (r , δ) = γ
 where
  r' : r , δ  normalise-pos r
  r' = ℤ[1/2]-to-normalise-pos (r , δ)
  p' : p , α  normalise-pos p
  p' = ℤ[1/2]-to-normalise-pos (p , α)

  γ : (p , α) + (q , β) + (r , δ)  (p , α) + ((q , β) + (r , δ))
  γ = (p , α) + (q , β) + (r , δ)              =⟨ refl 
      normalise-pos (p +' q) + (r , δ)         =⟨ i    
      normalise-pos (p +' q) + normalise-pos r =⟨ ii   
      normalise-pos ((p +' q) +' r)            =⟨ iii  
      normalise-pos (p +' (q +' r))            =⟨ iv   
      normalise-pos p + normalise-pos (q +' r) =⟨ v    
      (p , α) + normalise-pos (q +' r)         =⟨ refl 
      (p , α) + ((q , β) + (r , δ)) 
   where
    i   = ap  -  normalise-pos (p +' q) + -) r'
    ii  = ℤ[1/2]+-normalise-pos (p +' q) r ⁻¹
    iii = ap normalise-pos (ℤ[1/2]+'-assoc p q r)
    iv  = ℤ[1/2]+-normalise-pos p (q +' r)
    v   = ap (_+ normalise-pos (q +' r)) (p' ⁻¹)

ℤ[1/2]'-zero-left-neutral : (p :  × )  (pos 0 , 0) +' p  p
ℤ[1/2]'-zero-left-neutral (p , a) = ap₂ _,_ γ₁ γ₂
 where
  γ₁ : pos 0 * pos (2^ a) ℤ+ p * pos (2^ 0)  p
  γ₁ = pos 0 * pos (2^ a) ℤ+ p * pos (2^ 0) =⟨ i  
       pos 0 ℤ+ p                           =⟨ ii 
       p                                    
   where
    i  = ap (_ℤ+ p * pos (2^ 0)) (ℤ-zero-left-base (pos (2^ a)))
    ii = ℤ-zero-left-neutral p

  γ₂ : 0 ℕ+ a  a
  γ₂ = zero-left-neutral a

ℤ[1/2]-zero-left-neutral : (q : ℤ[1/2])  0ℤ[1/2] + q  q
ℤ[1/2]-zero-left-neutral (q , α) = γ
 where
  γ : 0ℤ[1/2] + (q , α)  (q , α)
  γ = 0ℤ[1/2] + (q , α)                 =⟨ refl 
      normalise-pos ((pos 0 , 0) +' q)  =⟨ i    
      normalise-pos q                   =⟨ ii   
      (q , α) 
   where
    i  = ap normalise-pos (ℤ[1/2]'-zero-left-neutral q)
    ii = ℤ[1/2]-to-normalise-pos (q , α) ⁻¹

ℤ[1/2]-zero-right-neutral : (q : ℤ[1/2])  q + 0ℤ[1/2]  q
ℤ[1/2]-zero-right-neutral q = γ
 where
  γ : q + 0ℤ[1/2]  q
  γ = q + 0ℤ[1/2] =⟨ ℤ[1/2]+-comm 0ℤ[1/2] q ⁻¹  
      0ℤ[1/2] + q =⟨ ℤ[1/2]-zero-left-neutral q 
      q           

ℤ[1/2]-negation-dist : (p q : ℤ[1/2])  - p + q  (- p) + (- q)
ℤ[1/2]-negation-dist ((p , a) , α) ((q , b) , β) = γ
 where
  a' = pos (2^ a)
  b' = pos (2^ b)

  p' : (p , a) , α  normalise-pos (p , a)
  p' = ℤ[1/2]-to-normalise-pos ((p , a) , α)
  q' : (q , b) , β  normalise-pos (q , b)
  q' = ℤ[1/2]-to-normalise-pos ((q , b) , β)

  γ : - ((p , a) , α) + ((q , b) , β)  (- ((p , a) , α)) + (- ((q , b) , β))
  γ = - ((p , a) , α) + ((q , b) , β)                       =⟨ i    
      - ((p , a) , α) + normalise-pos (q , b)               =⟨ ii   
      - normalise-pos (p , a) + normalise-pos (q , b)       =⟨ iii  
      - normalise-pos ((p , a) +' (q , b))                  =⟨ refl 
      - normalise-pos (p * b' ℤ+ q * a' , a ℕ+ b)           =⟨ iv   
      normalise-pos (ℤ- (p * b' ℤ+ q * a') , a ℕ+ b)        =⟨ v    
      normalise-pos ((ℤ- p * b') ℤ+ (ℤ- q * a') , a ℕ+ b)   =⟨ vi   
      normalise-pos ((ℤ- p) * b' ℤ+ (ℤ- q * a') , a ℕ+ b)   =⟨ vii  
      normalise-pos ((ℤ- p) * b' ℤ+ (ℤ- q) * a' , a ℕ+ b)   =⟨ refl 
      normalise-pos ((ℤ- p , a) +' (ℤ- q , b))              =⟨ viii 
      normalise-pos (ℤ- p , a) + normalise-pos (ℤ- q , b)   =⟨ ix   
      (- normalise-pos (p , a)) + normalise-pos (ℤ- q , b)  =⟨ x    
      (- normalise-pos (p , a)) + (- normalise-pos (q , b)) =⟨ xi   
      (- normalise-pos (p , a)) + (- ((q , b) , β))         =⟨ xii  
      (- ((p , a) , α)) + (- ((q , b) , β))                 
   where
    vₐₚ : ℤ- (p * b' ℤ+ q * a')  (ℤ- p * b') ℤ+ (ℤ- q * a')
    vₐₚ = negation-dist (p * b') (q * a') ⁻¹
    viₐₚ : ℤ- p * b'  (ℤ- p) * b'
    viₐₚ = negation-dist-over-mult' p b' ⁻¹
    viiₐₚ : ℤ- q * a'  (ℤ- q) * a'
    viiₐₚ = negation-dist-over-mult' q a' ⁻¹

    i    = ap  z  - ((p , a) , α) + z) q'
    ii   = ap  z  - z + normalise-pos (q , b)) p'
    iii  = ap -_ (ℤ[1/2]+-normalise-pos (p , a) (q , b) ⁻¹)
    iv   = minus-normalise-pos (p * b' ℤ+ q * a') (a ℕ+ b)
    v    = ap  z  normalise-pos (z , a ℕ+ b)) vₐₚ
    vi   = ap  z  normalise-pos (z ℤ+ (ℤ- q * a') , a ℕ+ b)) viₐₚ
    vii  = ap  z  normalise-pos ((ℤ- p) * b' ℤ+ z , a ℕ+ b)) viiₐₚ
    viii = ℤ[1/2]+-normalise-pos (ℤ- p , a) (ℤ- q , b)
    ix   = ap (_+ normalise-pos (ℤ- q , b)) (minus-normalise-pos p a ⁻¹)
    x    = ap ((- normalise-pos (p , a)) +_) (minus-normalise-pos q b ⁻¹)
    xi   =  ap  z  (- normalise-pos (p , a)) + (- z)) (q' ⁻¹)
    xii  = ap  z  (- z) + (- ((q , b) , β))) (p' ⁻¹)

\end{code}