2020-09-19 15:22:54 +03:00
|
|
|
module Data.String.Iterator
|
|
|
|
|
2020-09-19 23:46:46 +03:00
|
|
|
import public Data.List.Lazy
|
|
|
|
|
2020-09-19 22:54:34 +03:00
|
|
|
%default total
|
2020-09-19 15:22:54 +03:00
|
|
|
|
2020-09-20 10:56:58 +03:00
|
|
|
-- Backend-dependent string iteration type.
|
2020-09-19 22:54:34 +03:00
|
|
|
export
|
|
|
|
data StringIterator : Type where [external]
|
2020-09-19 15:22:54 +03:00
|
|
|
|
2020-09-20 10:56:58 +03:00
|
|
|
-- This function is private
|
|
|
|
-- to avoid subverting the linearity guarantees of withString.
|
2020-09-19 22:54:34 +03:00
|
|
|
%foreign
|
|
|
|
"scheme:blodwen-string-iterator-new"
|
2020-09-20 10:56:58 +03:00
|
|
|
private
|
2020-09-19 15:22:54 +03:00
|
|
|
fromString : String -> StringIterator
|
|
|
|
|
2020-09-20 10:56:58 +03:00
|
|
|
-- This function uses a linear string iterator
|
|
|
|
-- so that backends can use mutating iterators.
|
|
|
|
export
|
|
|
|
withString : String -> ((1 it : StringIterator) -> a) -> a
|
|
|
|
withString str f = f (fromString str)
|
|
|
|
|
|
|
|
-- We use a custom data type instead of Maybe (Char, StringIterator)
|
|
|
|
-- to remove one level of pointer indirection
|
|
|
|
-- in every iteration of something that's likely to be a hot loop,
|
|
|
|
-- and to avoid one allocation per character.
|
|
|
|
--
|
|
|
|
-- The Char field of Character is unrestricted for flexibility.
|
|
|
|
public export
|
|
|
|
data UnconsResult : Type where
|
|
|
|
EOF : UnconsResult
|
|
|
|
Character : (c : Char) -> (1 it : StringIterator) -> UnconsResult
|
|
|
|
|
|
|
|
-- We pass the whole string to the uncons function
|
|
|
|
-- to avoid yet another allocation per character
|
|
|
|
-- because for many backends, StringIterator can be simply an integer
|
|
|
|
-- (e.g. byte offset into an UTF-8 string).
|
2020-09-19 22:54:34 +03:00
|
|
|
%foreign
|
|
|
|
"scheme:blodwen-string-iterator-next"
|
2020-09-19 15:22:54 +03:00
|
|
|
export
|
2020-09-20 10:56:58 +03:00
|
|
|
uncons : String -> (1 it : StringIterator) -> UnconsResult
|
|
|
|
|
|
|
|
export
|
|
|
|
foldl : (accTy -> Char -> accTy) -> accTy -> String -> accTy
|
|
|
|
foldl op acc str = withString str (loop acc)
|
|
|
|
where
|
|
|
|
loop : accTy -> (1 it : StringIterator) -> accTy
|
|
|
|
loop acc it =
|
|
|
|
case uncons str it of
|
|
|
|
EOF => acc
|
|
|
|
Character c it' => loop (acc `op` c) (assert_smaller it it')
|
2020-09-19 15:22:54 +03:00
|
|
|
|
2020-09-19 23:46:46 +03:00
|
|
|
export
|
|
|
|
unpack : String -> LazyList Char
|
2020-09-20 10:56:58 +03:00
|
|
|
unpack str = withString str unpack'
|
|
|
|
where
|
|
|
|
unpack' : (1 it : StringIterator) -> LazyList Char
|
|
|
|
unpack' it =
|
|
|
|
case uncons str it of
|
|
|
|
EOF => []
|
|
|
|
Character c it' => c :: Delay (unpack' $ assert_smaller it it')
|