mirror of
https://github.com/idris-lang/Idris2.git
synced 2025-01-05 06:37:23 +03:00
139 lines
4.8 KiB
Idris
139 lines
4.8 KiB
Idris
||| This module implements a relation between a natural number and a list.
|
|
||| The relation witnesses the fact the number is the length of the list.
|
|
|||
|
|
||| It is meant to be used in a runtime-irrelevant fashion in computations
|
|
||| manipulating data indexed over lists where the computation actually only
|
|
||| depends on the length of said lists.
|
|
|||
|
|
||| Instead of writing:
|
|
||| ```idris example
|
|
||| f0 : (xs : List a) -> P xs
|
|
||| ```
|
|
|||
|
|
||| We would write either of:
|
|
||| ```idris example
|
|
||| f1 : (n : Nat) -> (0 _ : HasLength xs n) -> P xs
|
|
||| f2 : (n : Subset n (HasLength xs)) -> P xs
|
|
||| ```
|
|
|||
|
|
||| See `sucR` for an example where the update to the runtime-relevant Nat is O(1)
|
|
||| but the udpate to the list (were we to keep it around) an O(n) traversal.
|
|
|
|
module Data.List.HasLength
|
|
|
|
import Data.DPair
|
|
import Data.List
|
|
|
|
%default total
|
|
|
|
------------------------------------------------------------------------
|
|
-- Type
|
|
|
|
||| Ensure that the list's length is the provided natural number
|
|
public export
|
|
data HasLength : List a -> Nat -> Type where
|
|
Z : HasLength [] Z
|
|
S : HasLength xs n -> HasLength (x :: xs) (S n)
|
|
|
|
------------------------------------------------------------------------
|
|
-- Properties
|
|
|
|
||| The length is unique
|
|
export
|
|
hasLengthUnique : HasLength xs m -> HasLength xs n -> m === n
|
|
hasLengthUnique Z Z = Refl
|
|
hasLengthUnique (S p) (S q) = cong S (hasLengthUnique p q)
|
|
|
|
||| This specification corresponds to the length function
|
|
export
|
|
hasLength : (xs : List a) -> HasLength xs (length xs)
|
|
hasLength [] = Z
|
|
hasLength (_ :: xs) = S (hasLength xs)
|
|
|
|
export
|
|
map : (f : a -> b) -> HasLength xs n -> HasLength (map f xs) n
|
|
map f Z = Z
|
|
map f (S n) = S (map f n)
|
|
|
|
||| @sucR demonstrates that snoc only increases the lenght by one
|
|
||| So performing this operation while carrying the list around would cost O(n)
|
|
||| but relying on n together with an erased HasLength proof instead is O(1)
|
|
export
|
|
sucR : HasLength xs n -> HasLength (snoc xs x) (S n)
|
|
sucR Z = S Z
|
|
sucR (S n) = S (sucR n)
|
|
|
|
------------------------------------------------------------------------
|
|
-- Views
|
|
|
|
namespace SubsetView
|
|
|
|
||| We provide this view as a convenient way to perform nested pattern-matching
|
|
||| on values of type `Subset Nat (HasLength xs)`. Functions using this view will
|
|
||| be seen as terminating as long as the index list `xs` is left untouched.
|
|
||| See e.g. listTerminating below for such a function.
|
|
public export
|
|
data View : (xs : List a) -> Subset Nat (HasLength xs) -> Type where
|
|
Z : View [] (Element Z Z)
|
|
S : (p : Subset Nat (HasLength xs)) -> View (x :: xs) (Element (S (fst p)) (S (snd p)))
|
|
|
|
||| This auxiliary function gets around the limitation of the check ensuring that
|
|
||| we do not match on runtime-irrelevant data to produce runtime-relevant data.
|
|
viewZ : (0 p : HasLength xs Z) -> View xs (Element Z p)
|
|
viewZ Z = Z
|
|
|
|
||| This auxiliary function gets around the limitation of the check ensuring that
|
|
||| we do not match on runtime-irrelevant data to produce runtime-relevant data.
|
|
viewS : (n : Nat) -> (0 p : HasLength xs (S n)) -> View xs (Element (S n) p)
|
|
viewS n (S p) = S (Element n p)
|
|
|
|
||| Proof that the view covers all possible cases.
|
|
export
|
|
view : (p : Subset Nat (HasLength xs)) -> View xs p
|
|
view (Element Z p) = viewZ p
|
|
view (Element (S n) p) = viewS n p
|
|
|
|
namespace CurriedView
|
|
|
|
||| We provide this view as a convenient way to perform nested pattern-matching
|
|
||| on pairs of values of type `n : Nat` and `HasLength xs n`. If transformations
|
|
||| to the list between recursive calls (e.g. mapping over the list) that prevent
|
|
||| it from being a valid termination metric, it is best to take the Nat argument
|
|
||| separately from the HasLength proof and the Subset view is not as useful anymore.
|
|
||| See e.g. natTerminating below for (a contrived example of) such a function.
|
|
public export
|
|
data View : (xs : List a) -> (n : Nat) -> HasLength xs n -> Type where
|
|
Z : View [] Z Z
|
|
S : (n : Nat) -> (0 p : HasLength xs n) -> View (x :: xs) (S n) (S p)
|
|
|
|
||| Proof that the view covers all possible cases.
|
|
export
|
|
view : (n : Nat) -> (0 p : HasLength xs n) -> View xs n p
|
|
view Z Z = Z
|
|
view (S n) (S p) = S n p
|
|
|
|
------------------------------------------------------------------------
|
|
-- Examples
|
|
|
|
-- /!\ Do NOT re-export these examples
|
|
|
|
listTerminating : (p : Subset Nat (HasLength xs)) -> HasLength (xs ++ [x]) (S (fst p))
|
|
listTerminating p = case view p of
|
|
Z => S Z
|
|
S p => S (listTerminating p)
|
|
|
|
data P : List Nat -> Type where
|
|
PNil : P []
|
|
PCon : P (map f xs) -> P (x :: xs)
|
|
|
|
covering
|
|
notListTerminating : (p : Subset Nat (HasLength xs)) -> P xs
|
|
notListTerminating p = case view p of
|
|
Z => PNil
|
|
S p => PCon (notListTerminating {xs = map id (tail xs)} ({ snd $= map id } p))
|
|
|
|
natTerminating : (n : Nat) -> (0 p : HasLength xs n) -> P xs
|
|
natTerminating n p = case view n p of
|
|
Z => PNil
|
|
S n p => PCon (natTerminating n (map id p))
|