10h vacuum cleaner sound
This commit is contained in:
parent
71992db7d0
commit
2f07d89043
171
app/Builtins.hs
171
app/Builtins.hs
|
@ -46,9 +46,9 @@ printLocals = do
|
||||||
scope <- gets (gvar . cur)
|
scope <- gets (gvar . cur)
|
||||||
heap <- gets (heap . cur)
|
heap <- gets (heap . cur)
|
||||||
IR.StrTable _ _ itos <- gets strtable
|
IR.StrTable _ _ itos <- gets strtable
|
||||||
flip traverse (M.elems scope) $ \(ref, name) ->
|
flip traverse (M.assocs scope) $ \(local, ref) ->
|
||||||
lift . outputStrLn $
|
lift . outputStrLn $
|
||||||
(maybe "_" id $ itos M.!? name) ++ " = " ++ showTerm itos heap ref
|
"_Local" ++ show local ++ " = " ++ showTerm itos heap ref
|
||||||
return Nothing
|
return Nothing
|
||||||
|
|
||||||
promptRetry :: InterpFn
|
promptRetry :: InterpFn
|
||||||
|
@ -65,15 +65,22 @@ promptRetry' = do
|
||||||
Just ';' -> backtrack
|
Just ';' -> backtrack
|
||||||
_ -> return Nothing
|
_ -> return Nothing
|
||||||
|
|
||||||
write :: InterpFn
|
withArgs :: [Int] -> ([Int] -> InterpFn) -> InterpFn
|
||||||
write
|
withArgs as f = do
|
||||||
--TODO: prlgError on write(Unbound)
|
|
||||||
= do
|
|
||||||
scope <- gets (hvar . cur)
|
scope <- gets (hvar . cur)
|
||||||
heap <- gets (heap . cur)
|
if all (`M.member` scope) as
|
||||||
IR.StrTable _ _ itos <- gets strtable
|
then f $ map (scope M.!) as
|
||||||
lift . outputStr . showTerm itos heap . fst $ scope M.! 0
|
else prlgError "arguments not bound"
|
||||||
return Nothing
|
|
||||||
|
write' :: InterpFn -> InterpFn
|
||||||
|
write' c =
|
||||||
|
withArgs [0] $ \[arg] -> do
|
||||||
|
heap <- gets (heap . cur)
|
||||||
|
IR.StrTable _ _ itos <- gets strtable
|
||||||
|
lift . outputStr $ showTerm itos heap arg
|
||||||
|
c --this now allows error fallthrough but we might like EitherT
|
||||||
|
|
||||||
|
write = write' $ return Nothing
|
||||||
|
|
||||||
nl :: InterpFn
|
nl :: InterpFn
|
||||||
nl = do
|
nl = do
|
||||||
|
@ -81,59 +88,71 @@ nl = do
|
||||||
return Nothing
|
return Nothing
|
||||||
|
|
||||||
writeln :: InterpFn
|
writeln :: InterpFn
|
||||||
writeln = write >> nl
|
writeln = write' nl
|
||||||
|
|
||||||
assertFact :: InterpFn
|
assertFact :: InterpFn
|
||||||
assertFact = do
|
assertFact =
|
||||||
scope <- gets (hvar . cur)
|
withArgs [0] $ \[arg] -> do
|
||||||
heap <- gets (heap . cur)
|
heap <- gets (heap . cur)
|
||||||
case Co.compileGoal . Co.squashVars <$>
|
case Co.compileGoal . Co.squashVars <$>
|
||||||
Co.heapStructPrlgInt Nothing heap (fst $ scope M.! 0) of
|
Co.heapStructPrlgInt Nothing heap arg of
|
||||||
Just (U (Struct s):head) -> do
|
Just (U (Struct s):head) -> do
|
||||||
addClause s $ head ++ [NoGoal]
|
addClause s $ head ++ [NoGoal]
|
||||||
return Nothing
|
return Nothing
|
||||||
_ -> prlgError "assert fact failure"
|
_ -> prlgError "assert fact failure"
|
||||||
|
|
||||||
assertClause :: InterpFn
|
assertClause :: InterpFn
|
||||||
assertClause = do
|
assertClause =
|
||||||
scope <- gets (hvar . cur)
|
withArgs [0, 1] $ \args -> do
|
||||||
heap <- gets (heap . cur)
|
scope <- gets (hvar . cur)
|
||||||
commaId <- findStruct "," 2
|
heap <- gets (heap . cur)
|
||||||
cut <- findAtom "!"
|
comma <- findAtom ","
|
||||||
case Co.squashVars . IR.CallI (IR.Id 0 0) <$>
|
cut <- findAtom "!"
|
||||||
traverse (Co.heapStructPrlgInt Nothing heap . fst . (M.!) scope) [0, 1] of
|
case Co.squashVars . IR.CallI 0 <$>
|
||||||
Just (IR.CallI (IR.Id 0 0) [hs, gs]) ->
|
traverse (Co.heapStructPrlgInt Nothing heap) args of
|
||||||
let (U (Struct s):cs) =
|
Just (IR.CallI 0 [hs, gs]) ->
|
||||||
Co.compileGoal hs ++ Co.seqGoals (Co.compileGoals commaId cut gs)
|
let (U (Struct s):cs) =
|
||||||
in do addClause s cs
|
Co.compileGoal hs ++ Co.seqGoals (Co.compileGoals comma cut gs)
|
||||||
return Nothing
|
in do addClause s cs
|
||||||
_ -> prlgError "assert clause failure"
|
return Nothing
|
||||||
|
_ -> prlgError "assert clause failure"
|
||||||
|
|
||||||
retractall :: InterpFn
|
retractall :: InterpFn
|
||||||
retractall = prlgError "no retractall yet"
|
retractall =
|
||||||
|
withArgs [0] $ \[arg] -> do
|
||||||
|
heap <- gets (heap . cur)
|
||||||
|
case derefHeap heap arg of
|
||||||
|
BoundRef _ (Atom a) -> dropProcedure (IR.Id {IR.arity = 0, IR.str = a})
|
||||||
|
BoundRef _ (Struct id) -> dropProcedure id
|
||||||
|
_ -> prlgError "retractall needs a struct"
|
||||||
|
|
||||||
call :: InterpFn
|
call :: InterpFn
|
||||||
call = do
|
call =
|
||||||
ref <- gets (fst . (M.! 0) . hvar . cur)
|
withArgs [0] $ \[arg] -> do
|
||||||
heap@(Heap _ hmap) <- gets (heap . cur)
|
heap@(Heap _ hmap) <- gets (heap . cur)
|
||||||
let exec base struct arity = do
|
let exec base struct arity = do
|
||||||
cur <- gets cur
|
cur <- gets cur
|
||||||
modify $ \s ->
|
modify $ \s ->
|
||||||
s
|
s
|
||||||
{ cur =
|
{ cur =
|
||||||
cur
|
cur
|
||||||
{ gol =
|
{ gol =
|
||||||
[Call, Goal, U struct] ++
|
[Call, Goal, U struct] ++
|
||||||
[U $ hmap M.! (base + i) | i <- [1 .. arity]] ++ gol cur
|
[U $ hmap M.! (base + i) | i <- [1 .. arity]] ++ gol cur
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return Nothing
|
return Nothing
|
||||||
case derefHeap heap ref of
|
case derefHeap heap arg of
|
||||||
BoundRef addr struct@(Struct IR.Id {IR.arity = arity}) ->
|
BoundRef addr struct@(Struct IR.Id {IR.arity = arity}) ->
|
||||||
exec addr struct arity
|
exec addr struct arity
|
||||||
BoundRef addr (Atom a) ->
|
BoundRef addr (Atom a) ->
|
||||||
exec addr (Struct IR.Id {IR.arity = 0, IR.str = a}) 0
|
exec addr (Struct IR.Id {IR.arity = 0, IR.str = a}) 0
|
||||||
_ -> prlgError "not callable"
|
_ -> prlgError "not callable"
|
||||||
|
|
||||||
|
struct :: InterpFn
|
||||||
|
struct = do
|
||||||
|
scope <- gets (hvar . cur)
|
||||||
|
prlgError "not yet"
|
||||||
|
|
||||||
{- adding the builtins -}
|
{- adding the builtins -}
|
||||||
addOp op = modify $ \s -> s {ops = op : ops s}
|
addOp op = modify $ \s -> s {ops = op : ops s}
|
||||||
|
@ -145,20 +164,29 @@ addClause struct code =
|
||||||
addProcedure struct heads =
|
addProcedure struct heads =
|
||||||
modify $ \s -> s {defs = M.insert struct heads $ defs s}
|
modify $ \s -> s {defs = M.insert struct heads $ defs s}
|
||||||
|
|
||||||
|
dropProcedure struct = do
|
||||||
|
d <- gets defs
|
||||||
|
if struct `M.member` d
|
||||||
|
then do
|
||||||
|
modify $ \s -> s {defs = M.delete struct d}
|
||||||
|
return Nothing
|
||||||
|
else prlgError "no such definition" -- this should backtrack?
|
||||||
|
|
||||||
addProc n a c = do
|
addProc n a c = do
|
||||||
sym <- findStruct n a
|
sym <- findStruct n a
|
||||||
addProcedure sym c
|
addProcedure sym c
|
||||||
|
|
||||||
addBi0 n b = addProc n 0 [[Invoke $ bi b]]
|
addBi n i b =
|
||||||
|
addProc n i [[U (LocalRef $ r - 1) | r <- [1 .. i]] ++ [Invoke $ bi b]]
|
||||||
|
|
||||||
addPrelude :: PrlgEnv ()
|
addPrelude :: PrlgEnv ()
|
||||||
addPrelude = do
|
addPrelude = do
|
||||||
pure undefined
|
pure undefined
|
||||||
{- primitives -}
|
{- primitives -}
|
||||||
addBi0 "true" (pure Nothing)
|
addBi "true" 0 (pure Nothing)
|
||||||
addBi0 "fail" backtrack
|
addBi "fail" 0 backtrack
|
||||||
addOp $ O.xfx "=" 700
|
addOp $ O.xfx "=" 700
|
||||||
addProc "=" 2 [[U (LocalRef 0 0), U (LocalRef 0 0), NoGoal]]
|
addProc "=" 2 [[U (LocalRef 0), U (LocalRef 0), NoGoal]]
|
||||||
{- clauses -}
|
{- clauses -}
|
||||||
addOp $ O.xfy "," 1000
|
addOp $ O.xfy "," 1000
|
||||||
addOp $ O.xfx ":-" 1200
|
addOp $ O.xfx ":-" 1200
|
||||||
|
@ -168,22 +196,23 @@ addPrelude = do
|
||||||
"assert"
|
"assert"
|
||||||
1
|
1
|
||||||
[ [ U (Struct horn2)
|
[ [ U (Struct horn2)
|
||||||
, U (LocalRef 0 0)
|
, U (LocalRef 0)
|
||||||
, U (LocalRef 1 0)
|
, U (LocalRef 1)
|
||||||
, Cut
|
, Cut
|
||||||
, Invoke (bi assertClause)
|
, Invoke (bi assertClause)
|
||||||
]
|
]
|
||||||
, [U (LocalRef 0 0), Invoke (bi assertFact)]
|
, [U (LocalRef 0), Invoke (bi assertFact)]
|
||||||
]
|
]
|
||||||
addProc "retractall" 1 [[U (LocalRef 0 0), Invoke (bi retractall)]]
|
addBi "retractall" 1 retractall
|
||||||
addProc "call" 1 [[U (LocalRef 0 0), Invoke (bi call)]]
|
addBi "call" 1 call
|
||||||
|
addBi "struct" 3 struct
|
||||||
{- query tools -}
|
{- query tools -}
|
||||||
addBi0 "print_locals" printLocals
|
addBi "print_locals" 0 printLocals
|
||||||
addBi0 "prompt_retry" promptRetry'
|
addBi "prompt_retry" 0 promptRetry'
|
||||||
addBi0 "query" (printLocals >> promptRetry)
|
addBi "query" 0 (printLocals >> promptRetry)
|
||||||
{- IO -}
|
{- IO -}
|
||||||
addProc "write" 1 [[U (LocalRef 0 0), Invoke (bi write)]]
|
addBi "write" 1 write
|
||||||
addProc "writeln" 1 [[U (LocalRef 0 0), Invoke (bi writeln)]]
|
addBi "writeln" 1 writeln
|
||||||
addBi0 "nl" nl
|
addBi "nl" 0 nl
|
||||||
{- debug -}
|
{- debug -}
|
||||||
addBi0 "interpreter_trace" (get >>= liftIO . print >> pure Nothing)
|
addBi "interpreter_trace" 0 (get >>= liftIO . print >> pure Nothing)
|
||||||
|
|
|
@ -12,7 +12,7 @@ data Datum
|
||||||
= Atom Int -- unifies a constant
|
= Atom Int -- unifies a constant
|
||||||
| Struct Id -- unifies a structure with arity
|
| Struct Id -- unifies a structure with arity
|
||||||
| VoidRef -- unifies with anything
|
| VoidRef -- unifies with anything
|
||||||
| LocalRef Int Int -- code-local variable idx (should never occur on heap)
|
| LocalRef Int -- code-local variable idx (should never occur on heap)
|
||||||
| HeapRef Int -- something further on the heap
|
| HeapRef Int -- something further on the heap
|
||||||
deriving (Show, Eq, Ord)
|
deriving (Show, Eq, Ord)
|
||||||
|
|
||||||
|
@ -36,7 +36,7 @@ data Heap =
|
||||||
|
|
||||||
emptyHeap = Heap 1 M.empty
|
emptyHeap = Heap 1 M.empty
|
||||||
|
|
||||||
type Scope = M.Map Int (Int, Int)
|
type Scope = M.Map Int Int
|
||||||
|
|
||||||
emptyScope :: Scope
|
emptyScope :: Scope
|
||||||
emptyScope = M.empty
|
emptyScope = M.empty
|
||||||
|
@ -103,7 +103,7 @@ codeStruct ::
|
||||||
codeStruct atom struct local rec end heap = go
|
codeStruct atom struct local rec end heap = go
|
||||||
where
|
where
|
||||||
go [] = ([], ) <$> end
|
go [] = ([], ) <$> end
|
||||||
go (U lr@(LocalRef _ _):cs) = do
|
go (U lr@(LocalRef _):cs) = do
|
||||||
x <- local lr
|
x <- local lr
|
||||||
case x of
|
case x of
|
||||||
Left ref -> (cs, ) <$> heapStruct atom struct rec heap ref
|
Left ref -> (cs, ) <$> heapStruct atom struct rec heap ref
|
||||||
|
|
|
@ -10,8 +10,7 @@ desugarPrlg :: Int -> PrlgInt -> PrlgInt
|
||||||
desugarPrlg list = go
|
desugarPrlg list = go
|
||||||
where
|
where
|
||||||
go (CallI id ps) = CallI id $ map go ps
|
go (CallI id ps) = CallI id $ map go ps
|
||||||
go (ListI (x:xs) t) =
|
go (ListI (x:xs) t) = CallI list [go x, go (ListI xs t)]
|
||||||
CallI Id {str = list, arity = 2} [go x, go (ListI xs t)]
|
|
||||||
go (ListI [] Nothing) = LiteralI list
|
go (ListI [] Nothing) = LiteralI list
|
||||||
go (ListI [] (Just x)) = go x
|
go (ListI [] (Just x)) = go x
|
||||||
go x = x
|
go x = x
|
||||||
|
@ -29,7 +28,7 @@ varOccurs _ = M.empty
|
||||||
variablizePrlg :: Int -> StrTable -> PrlgInt -> PrlgInt
|
variablizePrlg :: Int -> StrTable -> PrlgInt -> PrlgInt
|
||||||
variablizePrlg void (StrTable _ _ itos) = go
|
variablizePrlg void (StrTable _ _ itos) = go
|
||||||
where
|
where
|
||||||
go (CallI id ps) = CallI id $ map go ps
|
go (CallI i ps) = CallI i $ map go ps
|
||||||
go (LiteralI i)
|
go (LiteralI i)
|
||||||
| i == void = VoidI
|
| i == void = VoidI
|
||||||
| varname (itos M.! i) = VarI i i
|
| varname (itos M.! i) = VarI i i
|
||||||
|
@ -38,7 +37,7 @@ variablizePrlg void (StrTable _ _ itos) = go
|
||||||
renumVars :: (Int -> Maybe PrlgInt) -> PrlgInt -> PrlgInt
|
renumVars :: (Int -> Maybe PrlgInt) -> PrlgInt -> PrlgInt
|
||||||
renumVars rename = go
|
renumVars rename = go
|
||||||
where
|
where
|
||||||
go (CallI id ps) = CallI id $ map go ps
|
go (CallI i ps) = CallI i $ map go ps
|
||||||
go (VarI idx i)
|
go (VarI idx i)
|
||||||
| Just new <- rename idx = new
|
| Just new <- rename idx = new
|
||||||
go x = x
|
go x = x
|
||||||
|
@ -52,13 +51,13 @@ squashVars x =
|
||||||
[(idx, VarI idx' 0) | ((idx, n), idx') <- zip occurs [1 ..], n > 1]
|
[(idx, VarI idx' 0) | ((idx, n), idx') <- zip occurs [1 ..], n > 1]
|
||||||
in renumVars (m' M.!?) x
|
in renumVars (m' M.!?) x
|
||||||
|
|
||||||
compileGoals :: Id -> Int -> PrlgInt -> [Code]
|
compileGoals :: Int -> Int -> PrlgInt -> [Code]
|
||||||
compileGoals andop cut = go'
|
compileGoals andop cut = go'
|
||||||
where
|
where
|
||||||
go' = go . struct2goal
|
go' = go . struct2goal
|
||||||
go p@(CallI x args)
|
go p@(CallI x args@[_, _])
|
||||||
| x == andop = concatMap go' args
|
| x == andop = concatMap go' args
|
||||||
go p@(CallI (Id x 0) [])
|
go p@(CallI x [])
|
||||||
| x == cut = [[Cut]]
|
| x == cut = [[Cut]]
|
||||||
go x = [compileGoal x]
|
go x = [compileGoal x]
|
||||||
|
|
||||||
|
@ -66,9 +65,10 @@ compileGoal :: PrlgInt -> Code
|
||||||
compileGoal = compileArg . struct2goal
|
compileGoal = compileArg . struct2goal
|
||||||
|
|
||||||
compileArg :: PrlgInt -> Code
|
compileArg :: PrlgInt -> Code
|
||||||
compileArg (CallI s args) = U (Struct s) : concatMap compileArg args
|
compileArg (CallI i args) =
|
||||||
|
U (Struct Id {str = i, arity = length args}) : concatMap compileArg args
|
||||||
compileArg (LiteralI s) = [U (Atom s)]
|
compileArg (LiteralI s) = [U (Atom s)]
|
||||||
compileArg (VarI x s) = [U (LocalRef x s)]
|
compileArg (VarI x _) = [U (LocalRef x)]
|
||||||
compileArg (VoidI) = [U VoidRef]
|
compileArg (VoidI) = [U VoidRef]
|
||||||
|
|
||||||
seqGoals :: [Code] -> Code
|
seqGoals :: [Code] -> Code
|
||||||
|
@ -84,16 +84,16 @@ heapStructPrlgInt heaperr heap ref = heapStruct atom struct hrec heap ref
|
||||||
where
|
where
|
||||||
atom (Atom s) = pure $ LiteralI s
|
atom (Atom s) = pure $ LiteralI s
|
||||||
atom VoidRef = pure $ VoidI
|
atom VoidRef = pure $ VoidI
|
||||||
struct (Struct s) args = pure $ CallI s args
|
struct (Struct s) args = pure $ CallI (str s) args
|
||||||
hrec (HeapRef r) ref
|
hrec (HeapRef r) ref
|
||||||
| r == ref = pure $ VarI r 0
|
| r == ref = pure $ VarI r 0
|
||||||
| otherwise = heaperr
|
| otherwise = heaperr
|
||||||
|
|
||||||
-- TODO check if this is used
|
-- TODO check if this is used
|
||||||
goal2struct :: PrlgInt -> PrlgInt
|
goal2struct :: PrlgInt -> PrlgInt
|
||||||
goal2struct (CallI (Id s 0) []) = LiteralI s
|
goal2struct (CallI s []) = LiteralI s
|
||||||
goal2struct x = x
|
goal2struct x = x
|
||||||
|
|
||||||
struct2goal :: PrlgInt -> PrlgInt
|
struct2goal :: PrlgInt -> PrlgInt
|
||||||
struct2goal (LiteralI s) = CallI (Id s 0) []
|
struct2goal (LiteralI s) = CallI s []
|
||||||
struct2goal x = x
|
struct2goal x = x
|
||||||
|
|
|
@ -8,7 +8,7 @@ import Control.Monad.Trans.Class (lift)
|
||||||
import Control.Monad.Trans.State.Lazy (evalStateT, gets)
|
import Control.Monad.Trans.State.Lazy (evalStateT, gets)
|
||||||
import Data.Foldable (traverse_)
|
import Data.Foldable (traverse_)
|
||||||
import qualified Data.Map as M
|
import qualified Data.Map as M
|
||||||
import Env (PrlgEnv, findAtom, findStruct, withStrTable)
|
import Env (PrlgEnv, findAtom, withStrTable)
|
||||||
import qualified IR
|
import qualified IR
|
||||||
import qualified Interpreter as I
|
import qualified Interpreter as I
|
||||||
import qualified Parser as P
|
import qualified Parser as P
|
||||||
|
@ -55,9 +55,9 @@ interpret = (>> return True) . lex
|
||||||
C.variablizePrlg underscore st $ C.desugarPrlg list prlgi)
|
C.variablizePrlg underscore st $ C.desugarPrlg list prlgi)
|
||||||
compile prlgv
|
compile prlgv
|
||||||
compile prlgv = do
|
compile prlgv = do
|
||||||
commaId <- findStruct "," 2
|
comma <- findAtom ","
|
||||||
cut <- findAtom "!"
|
cut <- findAtom "!"
|
||||||
let code = C.seqGoals $ C.compileGoals commaId cut prlgv
|
let code = C.seqGoals $ C.compileGoals comma cut prlgv
|
||||||
execute code
|
execute code
|
||||||
execute code = do
|
execute code = do
|
||||||
res <- I.prove code
|
res <- I.prove code
|
||||||
|
|
|
@ -17,7 +17,7 @@ data Id =
|
||||||
deriving (Show, Eq, Ord)
|
deriving (Show, Eq, Ord)
|
||||||
|
|
||||||
data PrlgInt
|
data PrlgInt
|
||||||
= CallI Id [PrlgInt] --TODO this should be Int
|
= CallI Int [PrlgInt]
|
||||||
| LiteralI Int
|
| LiteralI Int
|
||||||
| ListI [PrlgInt] (Maybe PrlgInt) -- only exists before desugaring
|
| ListI [PrlgInt] (Maybe PrlgInt) -- only exists before desugaring
|
||||||
| VarI Int Int -- VarI localIndex strTableString
|
| VarI Int Int -- VarI localIndex strTableString
|
||||||
|
@ -41,7 +41,7 @@ internPrlg = go
|
||||||
go t (LiteralS str) = LiteralI <$> strtablize t str
|
go t (LiteralS str) = LiteralI <$> strtablize t str
|
||||||
go t (CallS str ps) =
|
go t (CallS str ps) =
|
||||||
let (t', i) = strtablize t str
|
let (t', i) = strtablize t str
|
||||||
in CallI (Id i $ length ps) <$> mapAccumL go t' ps
|
in CallI i <$> mapAccumL go t' ps
|
||||||
go t (ListS elems Nothing) = flip ListI Nothing <$> mapAccumL go t elems
|
go t (ListS elems Nothing) = flip ListI Nothing <$> mapAccumL go t elems
|
||||||
go t (ListS elems (Just tail)) =
|
go t (ListS elems (Just tail)) =
|
||||||
let (t', tail') = go t tail
|
let (t', tail') = go t tail
|
||||||
|
|
|
@ -89,10 +89,10 @@ proveStep = St.get >>= go
|
||||||
in ( Heap (nxt + n) $
|
in ( Heap (nxt + n) $
|
||||||
foldr (uncurry M.insert) m [(a, HeapRef a) | a <- addrs]
|
foldr (uncurry M.insert) m [(a, HeapRef a) | a <- addrs]
|
||||||
, addrs)
|
, addrs)
|
||||||
allocLocal (LocalRef reg ident) scope cont
|
allocLocal (LocalRef reg) scope cont
|
||||||
| Just (addr, _) <- scope M.!? reg = cont scope heap addr
|
| Just addr <- scope M.!? reg = cont scope heap addr
|
||||||
| (heap', addr) <- newHeapVar heap =
|
| (heap', addr) <- newHeapVar heap =
|
||||||
cont (M.insert reg (addr, ident) scope) heap' addr
|
cont (M.insert reg addr scope) heap' addr
|
||||||
newHeapStruct addr s@(Struct Id {arity = arity}) cont =
|
newHeapStruct addr s@(Struct Id {arity = arity}) cont =
|
||||||
let (Heap nxt' m', addrs) = newHeapVars (arity + 1) heap
|
let (Heap nxt' m', addrs) = newHeapVars (arity + 1) heap
|
||||||
m'' =
|
m'' =
|
||||||
|
@ -113,10 +113,10 @@ proveStep = St.get >>= go
|
||||||
unify (Struct Id {arity = a}) VoidRef =
|
unify (Struct Id {arity = a}) VoidRef =
|
||||||
c i {cur = cur {hed = hs, gol = replicate a (U VoidRef) ++ gs}}
|
c i {cur = cur {hed = hs, gol = replicate a (U VoidRef) ++ gs}}
|
||||||
{- handle local refs; first ignore their combination with voids to save memory -}
|
{- handle local refs; first ignore their combination with voids to save memory -}
|
||||||
unify (LocalRef _ _) VoidRef = uok
|
unify (LocalRef _) VoidRef = uok -- TRICKY: builtins need to check if locals actually exist because of this
|
||||||
unify VoidRef (LocalRef _ _) = uok
|
unify VoidRef (LocalRef _) = uok
|
||||||
{- allocate heap for LocalRefs and retry with HeapRefs -}
|
{- allocate heap for LocalRefs and retry with HeapRefs -}
|
||||||
unify lr@(LocalRef _ _) _ =
|
unify lr@(LocalRef _) _ =
|
||||||
allocLocal lr (hvar cur) $ \hvar' heap' addr ->
|
allocLocal lr (hvar cur) $ \hvar' heap' addr ->
|
||||||
c
|
c
|
||||||
i
|
i
|
||||||
|
@ -124,7 +124,7 @@ proveStep = St.get >>= go
|
||||||
cur
|
cur
|
||||||
{hed = U (HeapRef addr) : hs, hvar = hvar', heap = heap'}
|
{hed = U (HeapRef addr) : hs, hvar = hvar', heap = heap'}
|
||||||
}
|
}
|
||||||
unify _ lr@(LocalRef _ _) =
|
unify _ lr@(LocalRef _) =
|
||||||
allocLocal lr (gvar cur) $ \gvar' heap' addr ->
|
allocLocal lr (gvar cur) $ \gvar' heap' addr ->
|
||||||
c
|
c
|
||||||
i
|
i
|
||||||
|
|
|
@ -172,29 +172,27 @@ literal = Literal . unTok <$> free (satisfy isNormalTok <* notFollowedBy lParen)
|
||||||
|
|
||||||
call = do
|
call = do
|
||||||
fn <- unTok <$> satisfy isNormalTok -- not free
|
fn <- unTok <$> satisfy isNormalTok -- not free
|
||||||
Seq inner <- free parens
|
(Call fn [] <$ try emptyParens) <|> do
|
||||||
return $ Call fn $ splitOn [Literal ","] inner
|
Seq inner <- free parens
|
||||||
|
return $ Call fn $ splitOn [Literal ","] inner
|
||||||
|
|
||||||
parens = Seq <$> (free lParen *> some seqItem <* free rParen)
|
parens = Seq <$> (free lParen *> some seqItem <* free rParen)
|
||||||
|
|
||||||
|
emptyParens = Literal "()" <$ (free lParen >> free rParen)
|
||||||
|
|
||||||
list = do
|
list = do
|
||||||
free lBracket
|
free lBracket
|
||||||
choice
|
(List [] Nothing <$ free rBracket) <|> do
|
||||||
[ List [] Nothing <$ free rBracket
|
items <- splitOn [Literal ","] <$> some seqItem
|
||||||
, do items <- splitOn [Literal ","] <$> some seqItem
|
(List items Nothing <$ free rBracket) <|>
|
||||||
choice
|
(List items . Just <$> (free listTail *> some seqItem <* free rBracket))
|
||||||
[ List items Nothing <$ free rBracket
|
|
||||||
, List items . Just <$>
|
|
||||||
(free listTail *> some seqItem <* free rBracket)
|
|
||||||
]
|
|
||||||
]
|
|
||||||
|
|
||||||
seqItem = choice [try call, literal, parens, list]
|
seqItem = choice [try call, literal, try emptyParens, parens, list]
|
||||||
|
|
||||||
simpleTok :: String -> Parser ()
|
simpleTok :: String -> Parser ()
|
||||||
simpleTok s = void $ single (Tok s)
|
simpleTok s = void $ single (Tok s)
|
||||||
|
|
||||||
comma = simpleTok "."
|
period = simpleTok "."
|
||||||
|
|
||||||
lParen = simpleTok "("
|
lParen = simpleTok "("
|
||||||
|
|
||||||
|
@ -207,7 +205,7 @@ listTail = simpleTok "|"
|
||||||
rBracket = simpleTok "]"
|
rBracket = simpleTok "]"
|
||||||
|
|
||||||
clause :: Parser PAST
|
clause :: Parser PAST
|
||||||
clause = Seq <$> some (free seqItem) <* free comma
|
clause = Seq <$> some (free seqItem) <* free period
|
||||||
|
|
||||||
parsePrlg :: Parser [PAST]
|
parsePrlg :: Parser [PAST]
|
||||||
parsePrlg = ws *> many clause <* eof
|
parsePrlg = ws *> many clause <* eof
|
||||||
|
|
Loading…
Reference in a new issue