diff options
Diffstat (limited to 'src/Reducer.hs')
-rw-r--r-- | src/Reducer.hs | 117 |
1 files changed, 9 insertions, 108 deletions
diff --git a/src/Reducer.hs b/src/Reducer.hs index 820cc04..3be8306 100644 --- a/src/Reducer.hs +++ b/src/Reducer.hs @@ -1,117 +1,18 @@ --- MIT License, Copyright (c) 2023 Marvin Borner --- based on the RKNL abstract machine +-- MIT License, Copyright (c) 2024 Marvin Borner module Reducer ( reduce , unsafeReduce ) where -import Control.Concurrent.MVar -import Data.List ( elemIndex ) -import Data.Map.Strict ( Map ) -import qualified Data.Map.Strict as Map -import Data.Maybe ( fromMaybe ) import Helper -import System.IO.Unsafe ( unsafePerformIO ) -- TODO: AAH +import qualified Reducer.ION as ION +import qualified Reducer.RKNL as RKNL -type Store = Map Int Box -type Stack = [Redex] +reduce :: EvalConf -> Expression -> IO Expression +reduce conf e = case _reducer conf of + "RKNL" -> RKNL.reduce e + "ION" -> pure $ ION.reduce e + _ -> error "Invalid reducer" -newtype NameGen = NameGen Int -data BoxValue = Todo Redex | Done Redex | Empty -newtype Box = Box (MVar BoxValue) -data Rvar = Num Int | Hole -data Redex = Rabs Int Redex | Rapp Redex Redex | Rvar Rvar | Rclosure Redex Store | Rcache Box Redex -data Conf = Econf NameGen Redex Store Stack | Cconf NameGen Stack Redex | End - -nextName :: NameGen -> (Int, NameGen) -nextName (NameGen x) = (x, NameGen $ x + 1) - -toRedex :: Expression -> Redex -toRedex = convertWorker (NameGen 1) [] - where - convertWorker g ns (Abstraction e) = - let (v, g') = nextName g - t = convertWorker g' (v : ns) e - in Rabs v t - convertWorker g ns (Application l r) = - let lhs = convertWorker g ns l - rhs = convertWorker g ns r - in Rapp lhs rhs - convertWorker _ ns (Bruijn i) = - Rvar $ Num (if i < 0 || i >= length ns then i else ns !! i) - convertWorker g ns (Unquote e) = convertWorker g ns e - convertWorker _ _ _ = invalidProgramState - -fromRedex :: Redex -> Expression -fromRedex = convertWorker [] - where - convertWorker es (Rabs n e) = Abstraction $ convertWorker (n : es) e - convertWorker es (Rapp l r) = - let lhs = convertWorker es l - rhs = convertWorker es r - in Application lhs rhs - convertWorker es (Rvar (Num n)) = Bruijn $ fromMaybe n (elemIndex n es) - convertWorker _ _ = invalidProgramState - -transition :: Conf -> IO Conf -transition (Econf g (Rapp u v) e s) = - pure $ Econf g u e (Rapp (Rvar Hole) (Rclosure v e) : s) -transition (Econf g (Rabs x t) e s) = do - box <- newMVar Empty - pure $ Cconf g s (Rcache (Box box) (Rclosure (Rabs x t) e)) -transition (Econf g (Rvar (Num x)) e s) = do - def <- newMVar $ Done $ Rvar $ Num x - let b@(Box m) = Map.findWithDefault (Box def) x e - rd <- readMVar m - case rd of - Todo (Rclosure v e') -> pure $ Econf g v e' (Rcache b (Rvar Hole) : s) - Done t -> pure $ Cconf g s t - _ -> invalidProgramState -transition (Cconf g ((Rcache (Box m) (Rvar Hole)) : s) t) = do - modifyMVar_ m (\_ -> pure $ Done t) - pure $ Cconf g s t -transition (Cconf g ((Rapp (Rvar Hole) ve) : s) (Rcache _ (Rclosure (Rabs x t) e))) - = do - box <- newMVar (Todo ve) - pure $ Econf g t (Map.insert x (Box box) e) s -transition (Cconf g s (Rcache (Box m) (Rclosure (Rabs x t) e))) = do - rd <- readMVar m - case rd of - Done v -> pure $ Cconf g s v - Empty -> do - let (x1, g') = nextName g - box <- newMVar $ Done $ Rvar $ Num x1 - pure $ Econf g' - t - (Map.insert x (Box box) e) - (Rabs x1 (Rvar Hole) : Rcache (Box m) (Rvar Hole) : s) - Todo _ -> invalidProgramState -transition (Cconf g ((Rapp (Rvar Hole) (Rclosure v e)) : s) t) = - pure $ Econf g v e (Rapp t (Rvar Hole) : s) -transition (Cconf g ((Rapp t (Rvar Hole)) : s) v) = pure $ Cconf g s (Rapp t v) -transition (Cconf g ((Rabs x1 (Rvar Hole)) : s) v) = - pure $ Cconf g s (Rabs x1 v) -transition (Cconf _ [] _) = pure End -transition _ = invalidProgramState - -forEachState :: Conf -> (Conf -> IO Conf) -> IO Conf -forEachState conf trans = trans conf >>= \case - End -> pure conf - next -> forEachState next trans - --- TODO: NameGen is arbitrary to not conflict with toRedex -loadTerm :: Redex -> Conf -loadTerm t = Econf (NameGen 1000000) t Map.empty [] - -reduce :: Expression -> IO Expression -reduce e = do - let redex = toRedex e - forEachState (loadTerm redex) transition >>= \case - Cconf _ [] v -> pure $ fromRedex v - _ -> invalidProgramState - --- TODO: AAAAAAAAAAAAAAAAH remove this --- (probably not thaaat bad) -{-# NOINLINE unsafeReduce #-} unsafeReduce :: Expression -> Expression -unsafeReduce = unsafePerformIO . reduce +unsafeReduce = RKNL.unsafeReduce |