aboutsummaryrefslogtreecommitdiff
path: root/src/Language/Mili/Analyzer.hs
diff options
context:
space:
mode:
authorMarvin Borner2024-11-13 16:18:01 +0100
committerMarvin Borner2024-11-13 21:53:37 +0100
commitf60b209eae598160f6cf160415e08ae72658cd32 (patch)
tree3f8db41ec8b0e378b2aa3e2d1c8f827b981cbb32 /src/Language/Mili/Analyzer.hs
Initial structure
Diffstat (limited to 'src/Language/Mili/Analyzer.hs')
-rw-r--r--src/Language/Mili/Analyzer.hs76
1 files changed, 76 insertions, 0 deletions
diff --git a/src/Language/Mili/Analyzer.hs b/src/Language/Mili/Analyzer.hs
new file mode 100644
index 0000000..a85e990
--- /dev/null
+++ b/src/Language/Mili/Analyzer.hs
@@ -0,0 +1,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