mirror of
https://github.com/idris-lang/Idris2.git
synced 2025-01-05 15:08:00 +03:00
1d4c84171d
Main change =========== The main change is to the type of function dealing with an untouched segment of the local scope. e.g. ``` weak : {outer, vars : _} -> (ns : List Name) -> tm (outer ++ inner) -> tm (outer ++ ns ++ inner) ``` Instead we now write ``` weak : SizeOf ns -> tm (outer ++ inner) -> tm (outer ++ ns ++ inner) ``` meaning that we do not need the values of `outer`, `inner` and `ns` at runtime. Instead we only demand a `SizeOf ns` which is a `Nat` together with an (erased) proof that `ns` is of that length. Other modifications =================== Quadratic behaviour ------------------- A side effect of this refactor is the removal of two sources of quadratic behaviour. They typically arise in a situation where work is done on a scope of the form ``` outer ++ done ++ ns ++ inner ``` When `ns` is non-empty, some work is performed and then the variable is moved to the pile of things we are `done` with. This leads to recursive calls of the form `f done` -> `f (done ++ [v])` leading to a cost quadratic in the size of `ns`. Now that we only care about `SizeOf done`, the recursive call is (once all the runtime irrelevant content is erased) for the form `f n` -> `f (S n)`! More runtime irrelevance ------------------------ In some places we used to rely on a list of names `vars` being available. However once we only care about the length of `vars`, the fact it is not available is not a limitation. For instance a `SizeOf vars` can be reconstructed from an environment assigning values to `vars` even if `vars` is irrelevant. Indeed the size of the environment is the same as that of `vars`.
124 lines
3.5 KiB
Idris
124 lines
3.5 KiB
Idris
module Data.Stream
|
|
|
|
import Data.List
|
|
|
|
||| The first element of an infinite stream
|
|
public export
|
|
head : Stream a -> a
|
|
head (x::xs) = x
|
|
|
|
||| Drop the first n elements from the stream
|
|
||| @ n how many elements to drop
|
|
public export
|
|
drop : (n : Nat) -> Stream a -> Stream a
|
|
drop Z xs = xs
|
|
drop (S k) (x::xs) = drop k xs
|
|
|
|
||| An infinite stream of repetitions of the same thing
|
|
public export
|
|
repeat : a -> Stream a
|
|
repeat x = x :: repeat x
|
|
|
|
||| Generate an infinite stream by repeatedly applying a function
|
|
||| @ f the function to iterate
|
|
||| @ x the initial value that will be the head of the stream
|
|
public export
|
|
iterate : (f : a -> a) -> (x : a) -> Stream a
|
|
iterate f x = x :: iterate f (f x)
|
|
|
|
||| Get the nth element of a stream
|
|
public export
|
|
index : Nat -> Stream a -> a
|
|
index Z (x::xs) = x
|
|
index (S k) (x::xs) = index k xs
|
|
|
|
||| Combine two streams element-wise using a function.
|
|
|||
|
|
||| @ f the function to combine elements with
|
|
||| @ xs the first stream of elements
|
|
||| @ ys the second stream of elements
|
|
export
|
|
zipWith : (f : a -> b -> c) -> (xs : Stream a) -> (ys : Stream b) -> Stream c
|
|
zipWith f (x::xs) (y::ys) = f x y :: zipWith f xs ys
|
|
|
|
||| Combine three streams by applying a function element-wise along them
|
|
export
|
|
zipWith3 : (a -> b -> c -> d) -> Stream a -> Stream b -> Stream c -> Stream d
|
|
zipWith3 f (x::xs) (y::ys) (z::zs) = f x y z :: zipWith3 f xs ys zs
|
|
|
|
||| Create a stream of pairs from two streams
|
|
export
|
|
zip : Stream a -> Stream b -> Stream (a, b)
|
|
zip = zipWith (\x,y => (x,y))
|
|
|
|
||| Combine three streams into a stream of tuples elementwise
|
|
export
|
|
zip3 : Stream a -> Stream b -> Stream c -> Stream (a, b, c)
|
|
zip3 = zipWith3 (\x,y,z => (x,y,z))
|
|
|
|
||| Create a pair of streams from a stream of pairs
|
|
export
|
|
unzip : Stream (a, b) -> (Stream a, Stream b)
|
|
unzip xs = (map fst xs, map snd xs)
|
|
|
|
||| Split a stream of three-element tuples into three streams
|
|
export
|
|
unzip3 : Stream (a, b, c) -> (Stream a, Stream b, Stream c)
|
|
unzip3 xs = (map (\(x,_,_) => x) xs, map (\(_,x,_) => x) xs, map (\(_,_,x) => x) xs)
|
|
|
|
||| Return the diagonal elements of a stream of streams
|
|
export
|
|
diag : Stream (Stream a) -> Stream a
|
|
diag ((x::xs)::xss) = x :: diag (map tail xss)
|
|
|
|
||| Produce a Stream of left folds of prefixes of the given Stream
|
|
||| @ f the combining function
|
|
||| @ acc the initial value
|
|
||| @ xs the Stream to process
|
|
export
|
|
scanl : (f : a -> b -> a) -> (acc : a) -> (xs : Stream b) -> Stream a
|
|
scanl f acc (x :: xs) = acc :: scanl f (f acc x) xs
|
|
|
|
||| Produce a Stream repeating a sequence
|
|
||| @ xs the sequence to repeat
|
|
||| @ ok proof that the list is non-empty
|
|
export
|
|
cycle : (xs : List a) -> {auto ok : NonEmpty xs} -> Stream a
|
|
cycle {a} (x :: xs) {ok = IsNonEmpty} = x :: cycle' xs
|
|
where cycle' : List a -> Stream a
|
|
cycle' [] = x :: cycle' xs
|
|
cycle' (y :: ys) = y :: cycle' ys
|
|
|
|
public export
|
|
partial
|
|
takeUntil : (n -> Bool) -> Stream n -> List n
|
|
takeUntil p (x :: xs)
|
|
= if p x
|
|
then [x]
|
|
else x :: takeUntil p xs
|
|
|
|
public export
|
|
partial
|
|
takeBefore : (n -> Bool) -> Stream n -> List n
|
|
takeBefore p (x :: xs)
|
|
= if p x
|
|
then []
|
|
else x :: takeBefore p xs
|
|
|
|
export
|
|
Applicative Stream where
|
|
pure = repeat
|
|
(<*>) = zipWith apply
|
|
|
|
export
|
|
Monad Stream where
|
|
s >>= f = diag (map f s)
|
|
|
|
--------------------------------------------------------------------------------
|
|
-- Properties
|
|
--------------------------------------------------------------------------------
|
|
|
|
lengthTake : (1 n : Nat) -> (xs : Stream a) -> length (take n xs) = n
|
|
lengthTake Z _ = Refl
|
|
lengthTake (S n) (x :: xs) = cong S (lengthTake n xs)
|