blob: a85e990b3ba6e9e5e54d8e3a52e18ca001ae458b (
plain) (
blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
|
module Language.Mili.Analyzer
( linearity
) where
import Data.HashMap.Strict ( HashMap )
import qualified Data.HashMap.Strict as M
import Data.List ( intercalate
, isSuffixOf
)
import Data.Mili ( Term(..) )
-- | A trace specifies the exact path to any subterm
type Trace = [Breadcrumb]
data Breadcrumb = South -- | Down into abstraction
| West -- | Left hand side of application
| East -- | Right hand side of application
deriving (Eq, Show)
-- | Map abstractions to a hashmap such that de Bruijn levels correspond to traces
-- | of abstractions at that level
traceAbs :: Term -> HashMap Int [Trace]
traceAbs = go 0 []
where
go n t (Abs m) =
M.unionWith (++) (go (n + 1) (South : t) m) (M.singleton n [t])
go n t (App a b) = M.unionWith (++) (go n (West : t) a) (go n (East : t) b)
go _ _ (Lvl _ ) = M.empty
-- | Map de Bruijn levels to a hashmap such that de Bruijn levels correspond to
-- | traces of de Bruijn levels at that level
traceLvl :: Term -> HashMap Int [Trace]
traceLvl = go []
where
go t (Abs m ) = go (South : t) m
go t (App a b) = M.unionWith (++) (go (West : t) a) (go (East : t) b)
go t (Lvl l ) = M.singleton l [t]
-- | Unify two two mapped traces to find levels at which the traces are no suffixes
-- | Level traces not being a suffix of abstraction traces implies nonlinearity
-- TODO: Proof?
unifyTraces :: HashMap Int [Trace] -> HashMap Int [Trace] -> HashMap Int [Bool]
unifyTraces traceA traceB = M.fromList $ map zipMap allKeys
where
allKeys = M.keys (traceA `M.union` traceB)
left key = M.lookupDefault [[]] key traceA
right key = M.lookupDefault [[]] key traceB
zipMap key
| length (left key) == length (right key)
= (key, zipWith isSuffixOf (left key) (right key))
| otherwise
= (key, [False])
linearityError
:: HashMap Int [Trace] -> HashMap Int [Trace] -> HashMap Int [Bool] -> String
linearityError absTrace lvlTrace = pretty . filter (any not . snd) . M.toList
where
pretty = intercalate "\n" . map
(\(depth, _) ->
"Linearity divergence in depth "
<> show depth
<> "\n\t"
<> show (M.lookupDefault [[]] depth absTrace)
<> "\nvs\n\t"
<> show (M.lookupDefault [[]] depth lvlTrace)
)
linearity :: Term -> Either String Term
linearity t =
let absTrace = traceAbs t
lvlTrace = traceLvl t
unified = unifyTraces absTrace lvlTrace
isLinear = all (all id) unified
in if isLinear
then Right t
else Left $ linearityError absTrace lvlTrace unified
|