Compare commits
10 commits
336feaeba0
...
17138aabda
Author | SHA1 | Date | |
---|---|---|---|
|
17138aabda | ||
|
535598bd84 | ||
|
2156869837 | ||
|
4ce2abdd59 | ||
|
038bc63b45 | ||
|
768fb71200 | ||
|
452cd49496 | ||
|
f61d6a0179 | ||
|
98c40f4bf8 | ||
|
45c3f81891 |
253
app/Builtins.hs
253
app/Builtins.hs
|
@ -2,9 +2,10 @@ module Builtins where
|
||||||
|
|
||||||
import Paths_prlg
|
import Paths_prlg
|
||||||
|
|
||||||
import Code (Builtin(..), Code, Datum(..), Instr(..), InterpFn)
|
import Code (Builtin(..), Code, Datum(..), Id(..), Instr(..), InterpFn)
|
||||||
import CodeLens
|
import CodeLens
|
||||||
import qualified Compiler as Co
|
import qualified Compiler as Co
|
||||||
|
import Constant
|
||||||
import Control.Exception (IOException, catch)
|
import Control.Exception (IOException, catch)
|
||||||
import Control.Monad.IO.Class (liftIO)
|
import Control.Monad.IO.Class (liftIO)
|
||||||
import Control.Monad.Trans.Class (lift)
|
import Control.Monad.Trans.Class (lift)
|
||||||
|
@ -16,7 +17,7 @@ import qualified Data.Map as M
|
||||||
import Data.Maybe (fromJust)
|
import Data.Maybe (fromJust)
|
||||||
import Env (PrlgEnv(..), findAtom, findStruct, prlgError)
|
import Env (PrlgEnv(..), findAtom, findStruct, prlgError)
|
||||||
import Heap (Dereferenced(..), derefHeap, heapStruct, newHeapVars, writeHeap)
|
import Heap (Dereferenced(..), derefHeap, heapStruct, newHeapVars, writeHeap)
|
||||||
import qualified IR
|
import IR (PrlgInt(..), StrTable(..))
|
||||||
import Interpreter (backtrack)
|
import Interpreter (backtrack)
|
||||||
import Lens.Micro.Mtl
|
import Lens.Micro.Mtl
|
||||||
import Load (processInput)
|
import Load (processInput)
|
||||||
|
@ -29,10 +30,11 @@ continue = pure Nothing
|
||||||
|
|
||||||
showTerm itos heap = runIdentity . heapStruct atom struct hrec heap
|
showTerm itos heap = runIdentity . heapStruct atom struct hrec heap
|
||||||
where
|
where
|
||||||
atom (Atom a) = pure $ "'" ++ itos M.! a ++ "'"
|
atom (C (Atom a)) = pure $ "'" ++ itos M.! a ++ "'"
|
||||||
atom (Number n) = pure (show n)
|
atom (C (Number n)) = pure (show n)
|
||||||
|
atom (C (Str str)) = pure (show str)
|
||||||
atom VoidRef = pure "_"
|
atom VoidRef = pure "_"
|
||||||
struct (Struct (IR.Id h _)) args =
|
struct (Struct (Id h _)) args =
|
||||||
pure $ "'" ++ itos M.! h ++ "'(" ++ intercalate "," args ++ ")"
|
pure $ "'" ++ itos M.! h ++ "'(" ++ intercalate "," args ++ ")"
|
||||||
hrec (HeapRef hr) ref =
|
hrec (HeapRef hr) ref =
|
||||||
pure $
|
pure $
|
||||||
|
@ -45,7 +47,7 @@ printLocals :: InterpFn
|
||||||
printLocals = do
|
printLocals = do
|
||||||
scope <- use (cur . gvar)
|
scope <- use (cur . gvar)
|
||||||
heap <- use (cur . heap)
|
heap <- use (cur . heap)
|
||||||
IR.StrTable _ _ itos <- use strtable
|
StrTable _ _ itos <- use strtable
|
||||||
flip traverse (M.assocs scope) $ \(local, ref) ->
|
flip traverse (M.assocs scope) $ \(local, ref) ->
|
||||||
lift . outputStrLn $ "_Local" ++ show local ++ " = " ++
|
lift . outputStrLn $ "_Local" ++ show local ++ " = " ++
|
||||||
showTerm itos heap ref
|
showTerm itos heap ref
|
||||||
|
@ -76,7 +78,7 @@ write' :: InterpFn -> InterpFn
|
||||||
write' c =
|
write' c =
|
||||||
withArgs [0] $ \[arg] -> do
|
withArgs [0] $ \[arg] -> do
|
||||||
heap <- use (cur . heap)
|
heap <- use (cur . heap)
|
||||||
IR.StrTable _ _ itos <- use strtable
|
StrTable _ _ itos <- use strtable
|
||||||
lift . outputStr $ showTerm itos heap arg
|
lift . outputStr $ showTerm itos heap arg
|
||||||
c --this now allows error fallthrough but we might like EitherT
|
c --this now allows error fallthrough but we might like EitherT
|
||||||
|
|
||||||
|
@ -90,32 +92,31 @@ nl = do
|
||||||
writeln :: InterpFn
|
writeln :: InterpFn
|
||||||
writeln = write' nl
|
writeln = write' nl
|
||||||
|
|
||||||
assertFact :: (Code -> IR.Id -> PrlgEnv ()) -> InterpFn
|
assertFact :: (Code -> Id -> PrlgEnv ()) -> InterpFn
|
||||||
assertFact addClause =
|
assertFact addClause =
|
||||||
withArgs [0] $ \[arg] -> do
|
withArgs [0] $ \[arg] -> do
|
||||||
heap <- use (cur . heap)
|
heap <- use (cur . heap)
|
||||||
case Co.compileGoal . Co.squashVars <$>
|
case Co.heapStructPrlgInt Nothing heap arg of
|
||||||
Co.heapStructPrlgInt Nothing heap arg of
|
Just x -> do
|
||||||
Just (U (Struct s):head) -> do
|
case Co.compileGoal $ Co.squashVars x of
|
||||||
addClause (head ++ [Done]) s
|
Right (U (Struct s):head) -> do
|
||||||
continue
|
addClause (head ++ [Done]) s
|
||||||
|
continue
|
||||||
|
Left err -> prlgError err
|
||||||
_ -> prlgError "assert fact failure"
|
_ -> prlgError "assert fact failure"
|
||||||
|
|
||||||
assertRule :: (Code -> IR.Id -> PrlgEnv ()) -> InterpFn
|
assertRule :: (Code -> Id -> PrlgEnv ()) -> InterpFn
|
||||||
assertRule addClause =
|
assertRule addClause =
|
||||||
withArgs [0, 1] $ \args -> do
|
withArgs [0, 1] $ \args -> do
|
||||||
scope <- use (cur . hvar)
|
scope <- use (cur . hvar)
|
||||||
heap <- use (cur . heap)
|
heap <- use (cur . heap)
|
||||||
comma <- findAtom ","
|
[comma, semi, cut] <- traverse findAtom [",", ";", "!"]
|
||||||
semi <- findAtom ";"
|
|
||||||
cut <- findAtom "!"
|
|
||||||
case Co.squashVars . IR.CallI 0 <$>
|
case Co.squashVars . IR.CallI 0 <$>
|
||||||
traverse (Co.heapStructPrlgInt Nothing heap) args of
|
traverse (Co.heapStructPrlgInt Nothing heap) args of
|
||||||
Just (IR.CallI 0 [hs, gs]) ->
|
Just (IR.CallI 0 [hs, gs]) ->
|
||||||
let (U (Struct s):cs) =
|
case (++) <$> Co.compileGoal hs <*> Co.compileGoals comma semi cut gs of
|
||||||
Co.compileGoal hs ++ Co.compileGoals comma semi cut gs
|
Right (U (Struct s):cs) -> addClause cs s >> continue
|
||||||
in do addClause cs s
|
Left err -> prlgError err
|
||||||
continue
|
|
||||||
_ -> prlgError "assert clause failure"
|
_ -> prlgError "assert clause failure"
|
||||||
|
|
||||||
retractall :: InterpFn
|
retractall :: InterpFn
|
||||||
|
@ -123,37 +124,29 @@ retractall =
|
||||||
withArgs [0] $ \[arg] -> do
|
withArgs [0] $ \[arg] -> do
|
||||||
heap <- use (cur . heap)
|
heap <- use (cur . heap)
|
||||||
case derefHeap heap arg of
|
case derefHeap heap arg of
|
||||||
BoundRef _ (Atom a) ->
|
BoundRef _ (C (Atom a)) ->
|
||||||
dropProcedure (IR.Id {IR.arity = 0, IR.str = a}) >> continue
|
dropProcedure (Id {arity = 0, str = a}) >> continue
|
||||||
BoundRef _ (Struct id) -> dropProcedure id >> continue
|
BoundRef _ (Struct id) -> dropProcedure id >> continue
|
||||||
_ -> prlgError "retractall needs a struct"
|
_ -> prlgError "retractall needs a struct"
|
||||||
|
|
||||||
exec' :: (Code -> Code) -> InterpFn
|
call :: InterpFn
|
||||||
exec' fgol =
|
call =
|
||||||
withArgs [0] $ \[arg] -> do
|
withArgs [0] $ \[arg] -> do
|
||||||
heap <- use (cur . heap)
|
heap <- use (cur . heap)
|
||||||
case Co.squashVars <$> Co.heapStructPrlgInt Nothing heap arg of
|
case derefHeap heap arg of
|
||||||
Just gs -> do
|
BoundRef _ (C (Atom a)) -> do
|
||||||
comma <- findAtom ","
|
cur . hed .= [U (Struct (Id {str = a, arity = 0})), Done]
|
||||||
semi <- findAtom ";"
|
|
||||||
cut <- findAtom "!"
|
|
||||||
zoom cur $ do
|
|
||||||
hvar .= M.empty
|
|
||||||
hed .= Co.compileGoals comma semi cut gs
|
|
||||||
gol %= fgol
|
|
||||||
continue
|
continue
|
||||||
_ -> prlgError "bad goal"
|
BoundRef addr s@(Struct Id {arity = arity}) -> do
|
||||||
|
cur . hed .= [U s] ++ [U (HeapRef $ addr + i) | i <- [1 .. arity]] ++
|
||||||
call :: InterpFn
|
[Done]
|
||||||
call = exec' id
|
continue
|
||||||
|
_ -> prlgError "bad call"
|
||||||
exec :: InterpFn
|
|
||||||
exec = exec' (const [Done])
|
|
||||||
|
|
||||||
stop :: InterpFn
|
stop :: InterpFn
|
||||||
stop =
|
stop =
|
||||||
withArgs [0] $ \[arg] -> do
|
withArgs [0] $ \[arg] -> do
|
||||||
IR.StrTable _ _ itos <- use strtable
|
StrTable _ _ itos <- use strtable
|
||||||
heap <- use (cur . heap)
|
heap <- use (cur . heap)
|
||||||
prlgError $ "stop: " ++ showTerm itos heap arg
|
prlgError $ "stop: " ++ showTerm itos heap arg
|
||||||
|
|
||||||
|
@ -163,17 +156,18 @@ struct = do
|
||||||
heap <- use (cur . heap)
|
heap <- use (cur . heap)
|
||||||
scope <- use (cur . hvar)
|
scope <- use (cur . hvar)
|
||||||
case derefHeap heap <$> scope M.!? 0 of
|
case derefHeap heap <$> scope M.!? 0 of
|
||||||
Just (BoundRef addr (Struct IR.Id {IR.arity = arity, IR.str = str})) ->
|
Just (BoundRef _ (Struct Id {arity = arity, str = str})) ->
|
||||||
structUnify arity str
|
structUnify arity str
|
||||||
|
Just (BoundRef _ _) -> backtrack
|
||||||
_ -> structAssemble
|
_ -> structAssemble
|
||||||
|
|
||||||
heapListLength listAtom heap ref = go 0 ref ref (id, fromJust . step)
|
heapListLength listAtom heap ref = go 0 ref ref (id, fromJust . step)
|
||||||
where
|
where
|
||||||
nil r
|
nil r
|
||||||
| BoundRef _ str <- derefHeap heap r = str == Atom listAtom
|
| BoundRef _ str <- derefHeap heap r = str == C (Atom listAtom)
|
||||||
| otherwise = False
|
| otherwise = False
|
||||||
step r
|
step r
|
||||||
| BoundRef addr (Struct IR.Id {IR.arity = 2, IR.str = listAtom'}) <-
|
| BoundRef addr (Struct Id {arity = 2, str = listAtom'}) <-
|
||||||
derefHeap heap r
|
derefHeap heap r
|
||||||
, listAtom == listAtom' = Just (addr + 2)
|
, listAtom == listAtom' = Just (addr + 2)
|
||||||
| otherwise = Nothing
|
| otherwise = Nothing
|
||||||
|
@ -190,7 +184,7 @@ structAssemble = do
|
||||||
heap <- use (cur . heap)
|
heap <- use (cur . heap)
|
||||||
scope <- use (cur . hvar)
|
scope <- use (cur . hvar)
|
||||||
case derefHeap heap <$> scope M.!? 1 of
|
case derefHeap heap <$> scope M.!? 1 of
|
||||||
Just (BoundRef addr (Atom str)) -> do
|
Just (BoundRef addr (C (Atom str))) -> do
|
||||||
listAtom <- findAtom "[]"
|
listAtom <- findAtom "[]"
|
||||||
case scope M.!? 2 >>= heapListLength listAtom heap of
|
case scope M.!? 2 >>= heapListLength listAtom heap of
|
||||||
Just arity -> structUnify arity str
|
Just arity -> structUnify arity str
|
||||||
|
@ -203,17 +197,17 @@ structUnify arity str = do
|
||||||
listAtom <- findAtom "[]"
|
listAtom <- findAtom "[]"
|
||||||
pvars <- newHeapVars arity
|
pvars <- newHeapVars arity
|
||||||
let hcode = map U $ maybe VoidRef HeapRef . (scope M.!?) <$> [0 .. 2]
|
let hcode = map U $ maybe VoidRef HeapRef . (scope M.!?) <$> [0 .. 2]
|
||||||
structData =
|
structData = Struct Id {arity = arity, str = str} : map HeapRef pvars
|
||||||
Struct IR.Id {IR.arity = arity, IR.str = str} : map HeapRef pvars
|
|
||||||
paramsData =
|
paramsData =
|
||||||
concatMap
|
concatMap
|
||||||
(\pv -> [Struct IR.Id {IR.arity = 2, IR.str = listAtom}, HeapRef pv])
|
(\pv -> [Struct Id {arity = 2, str = listAtom}, HeapRef pv])
|
||||||
pvars ++
|
pvars ++
|
||||||
[Atom listAtom]
|
[C $ Atom listAtom]
|
||||||
gcode = map U $ structData ++ [Atom str] ++ paramsData
|
gcode = map U $ structData ++ [C $ Atom str] ++ paramsData
|
||||||
zoom cur $ do
|
zoom cur $ do
|
||||||
gol %= (gcode ++)
|
gol %= (gcode ++)
|
||||||
hed %= (hcode ++)
|
hed %= (hcode ++)
|
||||||
|
unis += 3
|
||||||
continue
|
continue
|
||||||
|
|
||||||
{- terms -}
|
{- terms -}
|
||||||
|
@ -226,12 +220,29 @@ var = do
|
||||||
Just (FreeRef _) -> continue
|
Just (FreeRef _) -> continue
|
||||||
_ -> backtrack
|
_ -> backtrack
|
||||||
|
|
||||||
|
atom :: InterpFn
|
||||||
|
atom = do
|
||||||
|
heap <- use (cur . heap)
|
||||||
|
scope <- use (cur . hvar)
|
||||||
|
case derefHeap heap <$> scope M.!? 0 of
|
||||||
|
Just (BoundRef _ (C (Atom _))) -> continue
|
||||||
|
_ -> backtrack
|
||||||
|
|
||||||
number :: InterpFn
|
number :: InterpFn
|
||||||
number = do
|
number = do
|
||||||
heap <- use (cur . heap)
|
heap <- use (cur . heap)
|
||||||
scope <- use (cur . hvar)
|
scope <- use (cur . hvar)
|
||||||
case derefHeap heap <$> scope M.!? 0 of
|
case derefHeap heap <$> scope M.!? 0 of
|
||||||
Just (BoundRef _ (Number _)) -> continue
|
Just (BoundRef _ (C (Number _))) -> continue
|
||||||
|
_ -> backtrack
|
||||||
|
|
||||||
|
string :: InterpFn
|
||||||
|
string = do
|
||||||
|
heap <- use (cur . heap)
|
||||||
|
scope <- use (cur . hvar)
|
||||||
|
--TODO unify with number/var/...
|
||||||
|
case derefHeap heap <$> scope M.!? 0 of
|
||||||
|
Just (BoundRef _ (C (Str _))) -> continue
|
||||||
_ -> backtrack
|
_ -> backtrack
|
||||||
|
|
||||||
sameTerm :: InterpFn
|
sameTerm :: InterpFn
|
||||||
|
@ -260,9 +271,9 @@ op :: InterpFn
|
||||||
op =
|
op =
|
||||||
withArgs [0, 1, 2] $ \args -> do
|
withArgs [0, 1, 2] $ \args -> do
|
||||||
heap <- use (cur . heap)
|
heap <- use (cur . heap)
|
||||||
IR.StrTable _ _ itos <- use strtable
|
StrTable _ _ itos <- use strtable
|
||||||
case map (derefHeap heap) args of
|
case map (derefHeap heap) args of
|
||||||
[BoundRef _ (Number prio), BoundRef _ (Atom fixityAtom), BoundRef _ (Atom opatom)]
|
[BoundRef _ (C (Number prio)), BoundRef _ (C (Atom fixityAtom)), BoundRef _ (C (Atom opatom))]
|
||||||
| Just op <-
|
| Just op <-
|
||||||
(,) <$> itos M.!? opatom <*>
|
(,) <$> itos M.!? opatom <*>
|
||||||
(O.Op prio <$> ((itos M.!? fixityAtom) >>= O.fixity)) -> do
|
(O.Op prio <$> ((itos M.!? fixityAtom) >>= O.fixity)) -> do
|
||||||
|
@ -274,9 +285,9 @@ deop :: InterpFn
|
||||||
deop =
|
deop =
|
||||||
withArgs [0] $ \[arg] -> do
|
withArgs [0] $ \[arg] -> do
|
||||||
heap <- use (cur . heap)
|
heap <- use (cur . heap)
|
||||||
IR.StrTable _ _ itos <- use strtable
|
StrTable _ _ itos <- use strtable
|
||||||
case derefHeap heap arg of
|
case derefHeap heap arg of
|
||||||
BoundRef _ (Atom opatom)
|
BoundRef _ (C (Atom opatom))
|
||||||
| Just op <- itos M.!? opatom -> do
|
| Just op <- itos M.!? opatom -> do
|
||||||
ops %= filter ((/= op) . fst)
|
ops %= filter ((/= op) . fst)
|
||||||
continue
|
continue
|
||||||
|
@ -326,7 +337,8 @@ intBinary op =
|
||||||
withArgs [0, 1] $ \[arg1, arg2] -> do
|
withArgs [0, 1] $ \[arg1, arg2] -> do
|
||||||
heap <- use (cur . heap)
|
heap <- use (cur . heap)
|
||||||
case derefHeap heap <$> [arg1, arg2] of
|
case derefHeap heap <$> [arg1, arg2] of
|
||||||
[BoundRef _ (Number n1), BoundRef _ (Number n2)] -> putInt (op n1 n2) 2
|
[BoundRef _ (C (Number n1)), BoundRef _ (C (Number n2))] ->
|
||||||
|
putInt (op n1 n2) 2
|
||||||
_ -> prlgError "int binary needs numbers"
|
_ -> prlgError "int binary needs numbers"
|
||||||
|
|
||||||
intBinPred :: (Int -> Int -> Bool) -> InterpFn
|
intBinPred :: (Int -> Int -> Bool) -> InterpFn
|
||||||
|
@ -334,7 +346,7 @@ intBinPred op =
|
||||||
withArgs [0, 1] $ \args -> do
|
withArgs [0, 1] $ \args -> do
|
||||||
heap <- use (cur . heap)
|
heap <- use (cur . heap)
|
||||||
case derefHeap heap <$> args of
|
case derefHeap heap <$> args of
|
||||||
[BoundRef _ (Number n1), BoundRef _ (Number n2)] ->
|
[BoundRef _ (C (Number n1)), BoundRef _ (C (Number n2))] ->
|
||||||
if op n1 n2
|
if op n1 n2
|
||||||
then continue
|
then continue
|
||||||
else backtrack
|
else backtrack
|
||||||
|
@ -345,7 +357,7 @@ intUnary op =
|
||||||
withArgs [0] $ \[arg] -> do
|
withArgs [0] $ \[arg] -> do
|
||||||
heap <- use (cur . heap)
|
heap <- use (cur . heap)
|
||||||
case derefHeap heap arg of
|
case derefHeap heap arg of
|
||||||
BoundRef _ (Number n) -> putInt (op n) 1
|
BoundRef _ (C (Number n)) -> putInt (op n) 1
|
||||||
_ -> prlgError "int unary needs number"
|
_ -> prlgError "int unary needs number"
|
||||||
|
|
||||||
intUnPred :: (Int -> Bool) -> InterpFn
|
intUnPred :: (Int -> Bool) -> InterpFn
|
||||||
|
@ -353,7 +365,7 @@ intUnPred op =
|
||||||
withArgs [0] $ \[arg] -> do
|
withArgs [0] $ \[arg] -> do
|
||||||
heap <- use (cur . heap)
|
heap <- use (cur . heap)
|
||||||
case derefHeap heap arg of
|
case derefHeap heap arg of
|
||||||
BoundRef _ (Number n) ->
|
BoundRef _ (C (Number n)) ->
|
||||||
if op n
|
if op n
|
||||||
then continue
|
then continue
|
||||||
else backtrack
|
else backtrack
|
||||||
|
@ -366,28 +378,28 @@ putInt val sc = do
|
||||||
Nothing -> continue
|
Nothing -> continue
|
||||||
Just a ->
|
Just a ->
|
||||||
case derefHeap heap a of
|
case derefHeap heap a of
|
||||||
BoundRef _ (Number val')
|
BoundRef _ (C (Number val'))
|
||||||
| val == val' -> continue
|
| val == val' -> continue
|
||||||
FreeRef a' -> writeHeap a' (Number val) >> continue
|
FreeRef a' -> writeHeap a' (C (Number val)) >> continue
|
||||||
_ -> backtrack
|
_ -> backtrack
|
||||||
|
|
||||||
{- adding the builtins -}
|
{- adding the builtins -}
|
||||||
addOp :: (String, O.Op) -> PrlgEnv ()
|
addOp :: (String, O.Op) -> PrlgEnv ()
|
||||||
addOp op = ops %= (op :)
|
addOp op = ops %= (op :)
|
||||||
|
|
||||||
modDef :: ([Code] -> Maybe [Code]) -> IR.Id -> PrlgEnv ()
|
modDef :: ([Code] -> Maybe [Code]) -> Id -> PrlgEnv ()
|
||||||
modDef fn struct = defs %= M.alter (maybe (fn []) fn) struct
|
modDef fn struct = defs %= M.alter (maybe (fn []) fn) struct
|
||||||
|
|
||||||
addClauseA :: Code -> IR.Id -> PrlgEnv ()
|
addClauseA :: Code -> Id -> PrlgEnv ()
|
||||||
addClauseA code = modDef $ Just . (code :)
|
addClauseA code = modDef $ Just . (code :)
|
||||||
|
|
||||||
addClauseZ :: Code -> IR.Id -> PrlgEnv ()
|
addClauseZ :: Code -> Id -> PrlgEnv ()
|
||||||
addClauseZ code = modDef $ Just . (++ [code])
|
addClauseZ code = modDef $ Just . (++ [code])
|
||||||
|
|
||||||
addProcedure :: [Code] -> IR.Id -> PrlgEnv ()
|
addProcedure :: [Code] -> Id -> PrlgEnv ()
|
||||||
addProcedure heads = modDef $ Just . const heads
|
addProcedure heads = modDef $ Just . const heads
|
||||||
|
|
||||||
dropProcedure :: IR.Id -> PrlgEnv ()
|
dropProcedure :: Id -> PrlgEnv ()
|
||||||
dropProcedure = modDef $ const Nothing
|
dropProcedure = modDef $ const Nothing
|
||||||
|
|
||||||
addProc :: [Code] -> String -> Int -> PrlgEnv ()
|
addProc :: [Code] -> String -> Int -> PrlgEnv ()
|
||||||
|
@ -413,9 +425,9 @@ load :: Bool -> InterpFn
|
||||||
load queryMode =
|
load queryMode =
|
||||||
withArgs [0] $ \[arg] -> do
|
withArgs [0] $ \[arg] -> do
|
||||||
heap <- use (cur . heap)
|
heap <- use (cur . heap)
|
||||||
IR.StrTable _ _ itos <- use strtable --TODO the argument here should preferably be a string, right?
|
StrTable _ _ itos <- use strtable --TODO the argument here should preferably be a string, right?
|
||||||
case derefHeap heap arg of
|
case derefHeap heap arg of
|
||||||
BoundRef _ (Atom a) -> do
|
BoundRef _ (C (Atom a)) -> do
|
||||||
let fn = itos M.! a
|
let fn = itos M.! a
|
||||||
doLoad queryMode (itos M.! a)
|
doLoad queryMode (itos M.! a)
|
||||||
_ -> prlgError "load needs an atom"
|
_ -> prlgError "load needs an atom"
|
||||||
|
@ -425,7 +437,7 @@ addPrelude :: PrlgEnv ()
|
||||||
addPrelude = do
|
addPrelude = do
|
||||||
pure undefined
|
pure undefined
|
||||||
{- absolute primitives -}
|
{- absolute primitives -}
|
||||||
addBi (pure Nothing) "true" 0
|
addProc [[Done]] "true" 0
|
||||||
addBi backtrack "fail" 0
|
addBi backtrack "fail" 0
|
||||||
addBi stop "stop" 1
|
addBi stop "stop" 1
|
||||||
addOp $ O.xfx "=" 700
|
addOp $ O.xfx "=" 700
|
||||||
|
@ -435,27 +447,58 @@ addPrelude = do
|
||||||
addOp $ O.xfy ";" 1100
|
addOp $ O.xfy ";" 1100
|
||||||
addOp $ O.xfx ":-" 1200
|
addOp $ O.xfx ":-" 1200
|
||||||
addOp $ O.fx ":-" 1200
|
addOp $ O.fx ":-" 1200
|
||||||
horn1 <- findStruct ":-" 1
|
do [horn1, horn2] <- traverse (findStruct ":-") [1, 2]
|
||||||
horn2 <- findStruct ":-" 2
|
doCall <- U . Struct <$> findStruct "call" 1
|
||||||
let assertCode ac =
|
let assertCode ac =
|
||||||
[ [ U (Struct horn2)
|
[ [ U (Struct horn2)
|
||||||
, U (LocalRef 0)
|
, U (LocalRef 0)
|
||||||
, U (LocalRef 1)
|
, U (LocalRef 1)
|
||||||
, Cut
|
, Cut
|
||||||
, Invoke . bi $ assertRule ac
|
, Invoke . bi $ assertRule ac
|
||||||
]
|
]
|
||||||
, [U (Struct horn1), U (LocalRef 0), Cut, Invoke $ bi exec]
|
, [ U (Struct horn1)
|
||||||
, [U (LocalRef 0), Invoke . bi $ assertFact ac]
|
, U (LocalRef 0)
|
||||||
]
|
, Cut
|
||||||
in do addProc (assertCode addClauseA) "asserta" 1
|
, doCall
|
||||||
addProc (assertCode addClauseZ) "assertz" 1
|
, U (LocalRef 0)
|
||||||
addProc (assertCode addClauseZ) "assert" 1
|
, Done
|
||||||
|
]
|
||||||
|
, [U (LocalRef 0), Invoke . bi $ assertFact ac]
|
||||||
|
]
|
||||||
|
addProc (assertCode addClauseA) "asserta" 1
|
||||||
|
addProc (assertCode addClauseZ) "assertz" 1
|
||||||
|
addProc (assertCode addClauseZ) "assert" 1
|
||||||
addBi retractall "retractall" 1
|
addBi retractall "retractall" 1
|
||||||
addBi call "call" 1
|
do [comma, semi] <- traverse (flip findStruct 2) [",", ";"]
|
||||||
|
doCall <- U . Struct <$> findStruct "call" 1
|
||||||
|
addProc
|
||||||
|
[ [ U (Struct comma)
|
||||||
|
, U (LocalRef 0)
|
||||||
|
, U (LocalRef 1)
|
||||||
|
, Cut
|
||||||
|
, doCall
|
||||||
|
, U (LocalRef 0)
|
||||||
|
, doCall
|
||||||
|
, U (LocalRef 1)
|
||||||
|
, Done
|
||||||
|
]
|
||||||
|
, [ U (Struct semi)
|
||||||
|
, U (LocalRef 0)
|
||||||
|
, U (LocalRef 1)
|
||||||
|
, Cut
|
||||||
|
, Choices [[doCall, U (LocalRef 0)], [doCall, U (LocalRef 1)]]
|
||||||
|
, Done
|
||||||
|
]
|
||||||
|
, [U (LocalRef 0), Invoke $ bi call]
|
||||||
|
]
|
||||||
|
"call"
|
||||||
|
1
|
||||||
{- terms -}
|
{- terms -}
|
||||||
addBi struct "struct" 3
|
addBi struct "struct" 3
|
||||||
addBi var "var" 1
|
addBi var "var" 1
|
||||||
|
addBi atom "atom" 1
|
||||||
addBi number "number" 1
|
addBi number "number" 1
|
||||||
|
addBi string "string" 1
|
||||||
addBi sameTerm "same_term" 2
|
addBi sameTerm "same_term" 2
|
||||||
addBi currentPredicate "current_predicate" 1
|
addBi currentPredicate "current_predicate" 1
|
||||||
{- code loading -}
|
{- code loading -}
|
||||||
|
@ -489,30 +532,30 @@ addPrelude = do
|
||||||
]
|
]
|
||||||
("expand_" ++ q)
|
("expand_" ++ q)
|
||||||
2
|
2
|
||||||
expandCode "load"
|
in do expandCode "load"
|
||||||
expandCode "query"
|
expandCode "query"
|
||||||
{- int primops -}
|
{- int primops -}
|
||||||
let add2IntOp nm op = addBi (intBinary op) ("int2_" ++ nm) 3
|
let add2IntOp nm op = addBi (intBinary op) ("int2_" ++ nm) 3
|
||||||
add1IntOp nm op = addBi (intUnary op) ("int1_" ++ nm) 2
|
add1IntOp nm op = addBi (intUnary op) ("int1_" ++ nm) 2
|
||||||
add2IntPred nm op = addBi (intBinPred op) ("int2p_" ++ nm) 2
|
add2IntPred nm op = addBi (intBinPred op) ("int2p_" ++ nm) 2
|
||||||
add1IntPred nm op = addBi (intUnPred op) ("int1p_" ++ nm) 1
|
add1IntPred nm op = addBi (intUnPred op) ("int1p_" ++ nm) 1
|
||||||
add2IntOp "add" (+)
|
in do add2IntOp "add" (+)
|
||||||
add2IntOp "sub" (-)
|
add2IntOp "sub" (-)
|
||||||
add1IntOp "neg" negate
|
add1IntOp "neg" negate
|
||||||
add1IntOp "abs" abs
|
add1IntOp "abs" abs
|
||||||
add2IntOp "mul" (*)
|
add2IntOp "mul" (*)
|
||||||
add2IntOp "div" div
|
add2IntOp "div" div
|
||||||
add2IntOp "mod" mod
|
add2IntOp "mod" mod
|
||||||
add2IntOp "bitand" (.&.)
|
add2IntOp "bitand" (.&.)
|
||||||
add2IntOp "bitor" (.|.)
|
add2IntOp "bitor" (.|.)
|
||||||
add2IntOp "bitxor" xor
|
add2IntOp "bitxor" xor
|
||||||
add2IntOp "shl" shiftL
|
add2IntOp "shl" shiftL
|
||||||
add2IntOp "shr" shiftR
|
add2IntOp "shr" shiftR
|
||||||
add1IntPred "zero" (== 0)
|
add1IntPred "zero" (== 0)
|
||||||
add2IntPred "eq" (==)
|
add2IntPred "eq" (==)
|
||||||
add2IntPred "lt" (<)
|
add2IntPred "lt" (<)
|
||||||
add2IntPred "leq" (<=)
|
add2IntPred "leq" (<=)
|
||||||
add2IntPred "neq" (/=)
|
add2IntPred "neq" (/=)
|
||||||
{- query tools -}
|
{- query tools -}
|
||||||
addBi printLocals "print_locals" 0
|
addBi printLocals "print_locals" 0
|
||||||
addBi promptRetry' "prompt_retry" 0
|
addBi promptRetry' "prompt_retry" 0
|
||||||
|
|
26
app/Code.hs
26
app/Code.hs
|
@ -2,24 +2,31 @@
|
||||||
|
|
||||||
module Code where
|
module Code where
|
||||||
|
|
||||||
|
import Constant
|
||||||
import Control.Monad.Trans.State.Lazy (StateT)
|
import Control.Monad.Trans.State.Lazy (StateT)
|
||||||
import qualified Data.Map as M
|
import qualified Data.Map as M
|
||||||
import IR (Id(..), StrTable)
|
import IR (StrTable)
|
||||||
import Operators (Ops)
|
import Operators (Ops)
|
||||||
import Parser (PAST)
|
import Parser (PAST)
|
||||||
import System.Console.Haskeline (InputT)
|
import System.Console.Haskeline (InputT)
|
||||||
|
|
||||||
|
data Id =
|
||||||
|
Id
|
||||||
|
{ str :: !Int
|
||||||
|
, arity :: !Int
|
||||||
|
}
|
||||||
|
deriving (Show, Eq, Ord)
|
||||||
|
|
||||||
data Datum
|
data Datum
|
||||||
= Atom Int -- unifies a symbolic constant
|
= C !Constant -- unifies a constant
|
||||||
| Number Int -- unifies a numeric 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 -- 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)
|
||||||
|
|
||||||
data Instr
|
data Instr
|
||||||
= U Datum -- unify/resolve something
|
= U !Datum -- unify/resolve something
|
||||||
| Invoke Builtin -- give control to a builtin (invoked from head)
|
| Invoke Builtin -- give control to a builtin (invoked from head)
|
||||||
| Done -- all done, can return
|
| Done -- all done, can return
|
||||||
| Cut -- remove choicepoints of the current goal
|
| Cut -- remove choicepoints of the current goal
|
||||||
|
@ -31,7 +38,7 @@ type Code = [Instr]
|
||||||
type Defs = M.Map Id [Code]
|
type Defs = M.Map Id [Code]
|
||||||
|
|
||||||
data Heap =
|
data Heap =
|
||||||
Heap Int (M.Map Int Datum)
|
Heap !Int (M.Map Int Datum)
|
||||||
deriving (Show)
|
deriving (Show)
|
||||||
|
|
||||||
emptyHeap = Heap 1 M.empty
|
emptyHeap = Heap 1 M.empty
|
||||||
|
@ -51,7 +58,8 @@ data Cho =
|
||||||
, _retcut :: Bool -- cut after this goal succeeds
|
, _retcut :: Bool -- cut after this goal succeeds
|
||||||
, _heap :: Heap -- a snapshot of the heap (there's no trail; we rely on CoW copies in other choicepoints)
|
, _heap :: Heap -- a snapshot of the heap (there's no trail; we rely on CoW copies in other choicepoints)
|
||||||
, _stk :: [(Code, Scope, [Cho], Bool)] -- remaining goals together with their vars, cuts, and ret-cut flag
|
, _stk :: [(Code, Scope, [Cho], Bool)] -- remaining goals together with their vars, cuts, and ret-cut flag
|
||||||
, _cut :: [Cho] -- snapshot of choicepoints before entering
|
, _cut :: [Cho] -- snapshot of choicepoints before entering the goal
|
||||||
|
, _hcut :: [Cho] -- save of choicepoints just before starting to match head
|
||||||
}
|
}
|
||||||
deriving (Show)
|
deriving (Show)
|
||||||
|
|
||||||
|
|
|
@ -1,20 +1,13 @@
|
||||||
module Compiler where
|
module Compiler where
|
||||||
|
|
||||||
|
import Constant
|
||||||
|
import Control.Monad
|
||||||
import Data.Char (isUpper)
|
import Data.Char (isUpper)
|
||||||
import qualified Data.Map as M
|
import qualified Data.Map as M
|
||||||
|
|
||||||
import Code (Code, Datum(..), Heap, Instr(..))
|
import Code (Code, Datum(..), Heap, Id(..), Instr(..))
|
||||||
import Heap (heapStruct)
|
import Heap (heapStruct)
|
||||||
import IR (Id(..), PrlgInt(..), StrTable(..))
|
import IR (PrlgInt(..), StrTable(..))
|
||||||
|
|
||||||
desugarPrlg :: Int -> PrlgInt -> PrlgInt
|
|
||||||
desugarPrlg list = go
|
|
||||||
where
|
|
||||||
go (CallI id ps) = CallI id $ map go ps
|
|
||||||
go (ListI (x:xs) t) = CallI list [go x, go (ListI xs t)]
|
|
||||||
go (ListI [] Nothing) = AtomI list
|
|
||||||
go (ListI [] (Just x)) = go x
|
|
||||||
go x = x
|
|
||||||
|
|
||||||
varname :: String -> Bool
|
varname :: String -> Bool
|
||||||
varname ('_':_) = True
|
varname ('_':_) = True
|
||||||
|
@ -23,24 +16,24 @@ varname _ = False
|
||||||
|
|
||||||
varOccurs :: PrlgInt -> M.Map Int Int
|
varOccurs :: PrlgInt -> M.Map Int Int
|
||||||
varOccurs (CallI _ xs) = M.unionsWith (+) $ map varOccurs xs
|
varOccurs (CallI _ xs) = M.unionsWith (+) $ map varOccurs xs
|
||||||
varOccurs (VarI idx _) = M.singleton idx 1
|
varOccurs (VarI idx) = M.singleton idx 1
|
||||||
varOccurs _ = M.empty
|
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 i ps) = CallI i $ map go ps
|
go (CallI i ps) = CallI i $ map go ps
|
||||||
go (AtomI i)
|
go o@(ConstI (Atom i))
|
||||||
| i == void = VoidI
|
| i == void = VoidI
|
||||||
| varname (itos M.! i) = VarI i i
|
| varname (itos M.! i) = VarI i
|
||||||
| otherwise = AtomI i
|
| otherwise = o
|
||||||
go x = x
|
go x = x
|
||||||
|
|
||||||
renumVars :: (Int -> Maybe PrlgInt) -> PrlgInt -> PrlgInt
|
renumVars :: (Int -> Maybe PrlgInt) -> PrlgInt -> PrlgInt
|
||||||
renumVars rename = go
|
renumVars rename = go
|
||||||
where
|
where
|
||||||
go (CallI i ps) = CallI i $ map go ps
|
go (CallI i ps) = CallI i $ map go ps
|
||||||
go (VarI idx i)
|
go (VarI idx)
|
||||||
| Just new <- rename idx = new
|
| Just new <- rename idx = new
|
||||||
go x = x
|
go x = x
|
||||||
|
|
||||||
|
@ -50,38 +43,37 @@ squashVars x =
|
||||||
m' =
|
m' =
|
||||||
M.fromList $
|
M.fromList $
|
||||||
[(idx, VoidI) | (idx, n) <- occurs, n <= 1] ++
|
[(idx, VoidI) | (idx, n) <- occurs, n <= 1] ++
|
||||||
[(idx, VarI idx' 0) | ((idx, n), idx') <- zip occurs [1 ..], n > 1]
|
[(idx, VarI idx') | ((idx, n), idx') <- zip occurs [1 ..], n > 1]
|
||||||
in renumVars (m' M.!?) x
|
in renumVars (m' M.!?) x
|
||||||
|
|
||||||
squashChoices :: [Code] -> Code
|
squashChoices :: [Code] -> Either String Code
|
||||||
squashChoices = out . concatMap go
|
squashChoices = out . concatMap go
|
||||||
where
|
where
|
||||||
go [Choices cs] = cs
|
go [Choices cs] = cs
|
||||||
go x = [x]
|
go x = [x]
|
||||||
out [] = error "choice compilation"
|
out [] = Left "goal compilation has no choices?"
|
||||||
out [x] = x
|
out [x] = pure x
|
||||||
out xs = [Choices xs]
|
out xs = pure [Choices xs]
|
||||||
|
|
||||||
compileGoals :: Int -> Int -> Int -> PrlgInt -> Code
|
compileGoals :: Int -> Int -> Int -> PrlgInt -> Either String Code
|
||||||
compileGoals andop orop cut = (++ [Done]) . go'
|
compileGoals andop orop cut = fmap (++ [Done]) . go'
|
||||||
where
|
where
|
||||||
go' = go . struct2goal
|
go' = struct2goal >=> go
|
||||||
go p@(CallI x args@[_, _])
|
go p@(CallI x args@[_, _])
|
||||||
| x == andop = concatMap go' args
|
| x == andop = concat <$> traverse go' args
|
||||||
| x == orop = squashChoices $ map go' args
|
| x == orop = traverse go' args >>= squashChoices
|
||||||
go p@(CallI x [])
|
go p@(CallI x [])
|
||||||
| x == cut = [Cut]
|
| x == cut = pure [Cut]
|
||||||
go x = compileGoal x
|
go x = compileGoal x
|
||||||
|
|
||||||
compileGoal :: PrlgInt -> Code
|
compileGoal :: PrlgInt -> Either String Code
|
||||||
compileGoal = compileArg . struct2goal
|
compileGoal = fmap compileArg . struct2goal
|
||||||
|
|
||||||
compileArg :: PrlgInt -> Code
|
compileArg :: PrlgInt -> Code
|
||||||
compileArg (CallI i args) =
|
compileArg (CallI i args) =
|
||||||
U (Struct Id {str = i, arity = length args}) : concatMap compileArg args
|
U (Struct Id {str = i, arity = length args}) : concatMap compileArg args
|
||||||
compileArg (AtomI s) = [U (Atom s)]
|
compileArg (ConstI c) = [U (C c)]
|
||||||
compileArg (NumI s) = [U (Number s)]
|
compileArg (VarI x) = [U (LocalRef x)]
|
||||||
compileArg (VarI x _) = [U (LocalRef x)]
|
|
||||||
compileArg (VoidI) = [U VoidRef]
|
compileArg (VoidI) = [U VoidRef]
|
||||||
|
|
||||||
seqGoals :: [Code] -> Code
|
seqGoals :: [Code] -> Code
|
||||||
|
@ -90,20 +82,14 @@ seqGoals = (++ [Done]) . concat
|
||||||
heapStructPrlgInt :: Monad m => m PrlgInt -> Heap -> Int -> m PrlgInt
|
heapStructPrlgInt :: Monad m => m PrlgInt -> Heap -> Int -> m PrlgInt
|
||||||
heapStructPrlgInt heaperr heap ref = heapStruct atom struct hrec heap ref
|
heapStructPrlgInt heaperr heap ref = heapStruct atom struct hrec heap ref
|
||||||
where
|
where
|
||||||
atom (Atom s) = pure $ AtomI s
|
atom (C c) = pure (ConstI c)
|
||||||
atom (Number n) = pure $ NumI n
|
|
||||||
atom VoidRef = pure $ VoidI
|
atom VoidRef = pure $ VoidI
|
||||||
struct (Struct s) args = pure $ CallI (str 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
|
||||||
| otherwise = heaperr
|
| otherwise = heaperr
|
||||||
|
|
||||||
-- TODO check if this is used
|
struct2goal :: PrlgInt -> Either String PrlgInt
|
||||||
goal2struct :: PrlgInt -> PrlgInt
|
struct2goal (ConstI (Atom s)) = pure $ CallI s []
|
||||||
goal2struct (CallI s []) = AtomI s
|
struct2goal call@(CallI _ _) = pure call
|
||||||
goal2struct x = x
|
struct2goal x = Left $ "cannot compile goal: " ++ show x
|
||||||
|
|
||||||
struct2goal :: PrlgInt -> PrlgInt
|
|
||||||
struct2goal (AtomI s) = CallI s []
|
|
||||||
struct2goal call@(CallI _ _) = call
|
|
||||||
struct2goal _ = error "TODO."
|
|
||||||
|
|
7
app/Constant.hs
Normal file
7
app/Constant.hs
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
module Constant where
|
||||||
|
|
||||||
|
data Constant
|
||||||
|
= Atom Int
|
||||||
|
| Number Int
|
||||||
|
| Str String
|
||||||
|
deriving (Show, Eq, Ord)
|
12
app/Env.hs
12
app/Env.hs
|
@ -1,23 +1,23 @@
|
||||||
module Env where
|
module Env where
|
||||||
|
|
||||||
import Code (InterpFn, PrlgEnv)
|
import Code (Id(..), InterpFn, PrlgEnv)
|
||||||
import CodeLens
|
import CodeLens
|
||||||
import qualified IR
|
import IR (StrTable, strtablize)
|
||||||
import Lens.Micro.Mtl
|
import Lens.Micro.Mtl
|
||||||
|
|
||||||
withStrTable :: (IR.StrTable -> (IR.StrTable, a)) -> Env.PrlgEnv a
|
withStrTable :: (StrTable -> (StrTable, a)) -> Env.PrlgEnv a
|
||||||
withStrTable f = do
|
withStrTable f = do
|
||||||
(st', x) <- f <$> use strtable
|
(st', x) <- f <$> use strtable
|
||||||
strtable .= st'
|
strtable .= st'
|
||||||
return x
|
return x
|
||||||
|
|
||||||
findStruct :: String -> Int -> Env.PrlgEnv IR.Id
|
findStruct :: String -> Int -> Env.PrlgEnv Id
|
||||||
findStruct str arity = do
|
findStruct str arity = do
|
||||||
stri <- findAtom str
|
stri <- findAtom str
|
||||||
return IR.Id {IR.str = stri, IR.arity = arity}
|
return Id {str = stri, arity = arity}
|
||||||
|
|
||||||
findAtom :: String -> Env.PrlgEnv Int
|
findAtom :: String -> Env.PrlgEnv Int
|
||||||
findAtom = withStrTable . flip IR.strtablize
|
findAtom = withStrTable . flip strtablize
|
||||||
|
|
||||||
type PrlgEnv a = Code.PrlgEnv a
|
type PrlgEnv a = Code.PrlgEnv a
|
||||||
|
|
||||||
|
|
|
@ -33,8 +33,8 @@ handleError m = do
|
||||||
|
|
||||||
processCmd precompileHook ast' = do
|
processCmd precompileHook ast' = do
|
||||||
ast <- shunt ast'
|
ast <- shunt ast'
|
||||||
code <- lift $ intern ast >>= precompileHook >>= compile
|
source <- lift $ intern ast >>= precompileHook
|
||||||
lift (I.prove code) >>= except
|
compile source >>= lift . I.prove >>= except
|
||||||
|
|
||||||
interpreterStart :: PrlgEnv ()
|
interpreterStart :: PrlgEnv ()
|
||||||
interpreterStart = do
|
interpreterStart = do
|
||||||
|
|
|
@ -4,7 +4,6 @@ import Code
|
||||||
import CodeLens
|
import CodeLens
|
||||||
import Data.Foldable (traverse_)
|
import Data.Foldable (traverse_)
|
||||||
import qualified Data.Map as M
|
import qualified Data.Map as M
|
||||||
import IR (Id(..))
|
|
||||||
import Lens.Micro.Mtl
|
import Lens.Micro.Mtl
|
||||||
|
|
||||||
data Dereferenced
|
data Dereferenced
|
||||||
|
|
35
app/IR.hs
35
app/IR.hs
|
@ -1,28 +1,15 @@
|
||||||
module IR where
|
module IR where
|
||||||
|
|
||||||
|
import Constant
|
||||||
import Data.Char (isNumber)
|
import Data.Char (isNumber)
|
||||||
import Data.List (mapAccumL)
|
import Data.List (mapAccumL)
|
||||||
import qualified Data.Map as M
|
import qualified Data.Map as M
|
||||||
|
import Parser (Lexeme(..), PrlgStr(..))
|
||||||
data PrlgStr
|
|
||||||
= CallS String [PrlgStr]
|
|
||||||
| LiteralS String
|
|
||||||
| ListS [PrlgStr] (Maybe PrlgStr)
|
|
||||||
deriving (Show)
|
|
||||||
|
|
||||||
data Id =
|
|
||||||
Id
|
|
||||||
{ str :: Int
|
|
||||||
, arity :: Int
|
|
||||||
}
|
|
||||||
deriving (Show, Eq, Ord)
|
|
||||||
|
|
||||||
data PrlgInt
|
data PrlgInt
|
||||||
= CallI Int [PrlgInt]
|
= CallI Int [PrlgInt]
|
||||||
| AtomI Int
|
| ConstI Constant
|
||||||
| NumI Int
|
| VarI Int -- VarI localIndex strTableString
|
||||||
| ListI [PrlgInt] (Maybe PrlgInt) -- only exists before desugaring
|
|
||||||
| VarI Int Int -- VarI localIndex strTableString
|
|
||||||
| VoidI
|
| VoidI
|
||||||
deriving (Show)
|
deriving (Show)
|
||||||
|
|
||||||
|
@ -37,16 +24,16 @@ strtablize t@(StrTable nxt fwd rev) str =
|
||||||
Just i -> (t, i)
|
Just i -> (t, i)
|
||||||
_ -> (StrTable (nxt + 1) (M.insert str nxt fwd) (M.insert nxt str rev), nxt)
|
_ -> (StrTable (nxt + 1) (M.insert str nxt fwd) (M.insert nxt str rev), nxt)
|
||||||
|
|
||||||
|
internLexeme t (Tok str)
|
||||||
|
| all isNumber str = (t, ConstI . Number $ read str)
|
||||||
|
| otherwise = ConstI . Atom <$> strtablize t str
|
||||||
|
internLexeme t (QTok str _) = ConstI . Atom <$> strtablize t str
|
||||||
|
internLexeme t (DQTok str _) = (t, ConstI $ Str str)
|
||||||
|
|
||||||
internPrlg :: StrTable -> PrlgStr -> (StrTable, PrlgInt)
|
internPrlg :: StrTable -> PrlgStr -> (StrTable, PrlgInt)
|
||||||
internPrlg = go
|
internPrlg = go
|
||||||
where
|
where
|
||||||
go t (LiteralS str)
|
go t (LiteralS lex) = internLexeme t lex
|
||||||
| all isNumber str = (t, NumI $ read str)
|
|
||||||
| otherwise = AtomI <$> 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 i <$> 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 (Just tail)) =
|
|
||||||
let (t', tail') = go t tail
|
|
||||||
in flip ListI (Just tail') <$> mapAccumL go t' elems
|
|
||||||
|
|
|
@ -8,6 +8,7 @@ import Code
|
||||||
, Cho(..)
|
, Cho(..)
|
||||||
, Code
|
, Code
|
||||||
, Datum(..)
|
, Datum(..)
|
||||||
|
, Id(..)
|
||||||
, Instr(..)
|
, Instr(..)
|
||||||
, InterpFn
|
, InterpFn
|
||||||
, emptyHeap
|
, emptyHeap
|
||||||
|
@ -18,7 +19,7 @@ import Control.Monad (when)
|
||||||
import qualified Data.Map as M
|
import qualified Data.Map as M
|
||||||
import Env (PrlgEnv)
|
import Env (PrlgEnv)
|
||||||
import Heap
|
import Heap
|
||||||
import IR (Id(..), StrTable(..))
|
import IR (StrTable(..))
|
||||||
import Lens.Micro
|
import Lens.Micro
|
||||||
import Lens.Micro.Mtl
|
import Lens.Micro.Mtl
|
||||||
|
|
||||||
|
@ -35,6 +36,7 @@ prove g = do
|
||||||
, _heap = emptyHeap
|
, _heap = emptyHeap
|
||||||
, _stk = []
|
, _stk = []
|
||||||
, _cut = []
|
, _cut = []
|
||||||
|
, _hcut = []
|
||||||
}
|
}
|
||||||
cho .= []
|
cho .= []
|
||||||
loop
|
loop
|
||||||
|
@ -54,10 +56,14 @@ proveStep = do
|
||||||
import Control.Monad.Trans.Class (lift)
|
import Control.Monad.Trans.Class (lift)
|
||||||
import System.Console.Haskeline
|
import System.Console.Haskeline
|
||||||
g <- use (cur . gol)
|
g <- use (cur . gol)
|
||||||
|
cho <- use cho
|
||||||
|
cut <- use (cur . cut)
|
||||||
lift $ do
|
lift $ do
|
||||||
outputStrLn $ "STEP (unis="++show u++")"
|
outputStrLn $ "STEP (unis="++show u++")"
|
||||||
outputStrLn $ "head = "++ show h
|
outputStrLn $ "head = "++ show h
|
||||||
outputStrLn $ "goal = "++ show g
|
outputStrLn $ "goal = "++ show g
|
||||||
|
outputStrLn $ "cut = " ++ show cut
|
||||||
|
outputStrLn $ "cho = " ++ show cho
|
||||||
-}
|
-}
|
||||||
case (u, h) of
|
case (u, h) of
|
||||||
(0, []) -> goalStep
|
(0, []) -> goalStep
|
||||||
|
@ -92,7 +98,7 @@ headStep h = do
|
||||||
case (h, g) of
|
case (h, g) of
|
||||||
([Done], _) -> succeedHead
|
([Done], _) -> succeedHead
|
||||||
(Cut:_, _) -> cutHead
|
(Cut:_, _) -> cutHead
|
||||||
(Invoke (Builtin bf):_, _) -> advanceHead >> bf
|
(Invoke (Builtin bf):_, _) -> cur . hed .= [Done] >> bf
|
||||||
(_, [Done]) -> tailCall
|
(_, [Done]) -> tailCall
|
||||||
(_, [Cut, Done]) -> tailCut
|
(_, [Cut, Done]) -> tailCut
|
||||||
(_, _) -> pushCall
|
(_, _) -> pushCall
|
||||||
|
@ -134,11 +140,13 @@ retCut = do
|
||||||
doCut
|
doCut
|
||||||
cur . retcut .= False
|
cur . retcut .= False
|
||||||
|
|
||||||
cutHead = doCut >> advanceHead
|
cutHead = do
|
||||||
|
use (cur . hcut) >>= assign cho
|
||||||
|
advanceHead
|
||||||
|
|
||||||
cutGoal = doCut >> advance
|
cutGoal = doCut >> advance
|
||||||
|
|
||||||
openGoal :: IR.Id -> InterpFn
|
openGoal :: Id -> InterpFn
|
||||||
openGoal fn = do
|
openGoal fn = do
|
||||||
def <- (M.!? fn) <$> use defs
|
def <- (M.!? fn) <$> use defs
|
||||||
case def of
|
case def of
|
||||||
|
@ -147,7 +155,8 @@ openGoal fn = do
|
||||||
cur . hvar .= emptyScope
|
cur . hvar .= emptyScope
|
||||||
cur . unis .= arity fn
|
cur . unis .= arity fn
|
||||||
cc <- use cur
|
cc <- use cur
|
||||||
let (newcur:newcho) = [cc & hed .~ h | h <- hs]
|
oldcho <- use cho
|
||||||
|
let (newcur:newcho) = [cc & hcut .~ oldcho & hed .~ h | h <- hs]
|
||||||
cur .= newcur
|
cur .= newcur
|
||||||
cho %= (newcho ++)
|
cho %= (newcho ++)
|
||||||
continue
|
continue
|
||||||
|
@ -162,12 +171,15 @@ pushCall = do
|
||||||
ngol <- use (cur . hed)
|
ngol <- use (cur . hed)
|
||||||
ngvar <- use (cur . hvar)
|
ngvar <- use (cur . hvar)
|
||||||
scut <- use (cur . cut)
|
scut <- use (cur . cut)
|
||||||
|
ncut <- use (cur . hcut)
|
||||||
sretcut <- use (cur . retcut)
|
sretcut <- use (cur . retcut)
|
||||||
cur . stk %= ((sgol, sgvar, scut, sretcut) :)
|
cur . stk %= ((sgol, sgvar, scut, sretcut) :)
|
||||||
cur . gol .= ngol
|
cur . gol .= ngol
|
||||||
cur . gvar .= ngvar
|
cur . gvar .= ngvar
|
||||||
|
cur . cut .= ncut
|
||||||
cur . hed .= []
|
cur . hed .= []
|
||||||
cur . hvar .= emptyScope
|
cur . hvar .= emptyScope
|
||||||
|
cur . hcut .= []
|
||||||
cur . retcut .= False
|
cur . retcut .= False
|
||||||
continue
|
continue
|
||||||
|
|
||||||
|
@ -179,6 +191,7 @@ tailCall = do
|
||||||
cur . gvar .= ngvar
|
cur . gvar .= ngvar
|
||||||
cur . hed .= []
|
cur . hed .= []
|
||||||
cur . hvar .= emptyScope
|
cur . hvar .= emptyScope
|
||||||
|
cur . hcut .= []
|
||||||
continue
|
continue
|
||||||
|
|
||||||
tailCut :: InterpFn
|
tailCut :: InterpFn
|
||||||
|
@ -189,8 +202,9 @@ tailCut = do
|
||||||
|
|
||||||
succeedHead :: InterpFn
|
succeedHead :: InterpFn
|
||||||
succeedHead = do
|
succeedHead = do
|
||||||
cur . hvar .= emptyScope
|
|
||||||
cur . hed .= []
|
cur . hed .= []
|
||||||
|
cur . hvar .= emptyScope
|
||||||
|
cur . hcut .= []
|
||||||
continue
|
continue
|
||||||
|
|
||||||
succeedGoal :: InterpFn
|
succeedGoal :: InterpFn
|
||||||
|
@ -231,13 +245,10 @@ uOK = uNext >> continue
|
||||||
|
|
||||||
unify :: Datum -> Datum -> InterpFn
|
unify :: Datum -> Datum -> InterpFn
|
||||||
unify VoidRef VoidRef = uOK
|
unify VoidRef VoidRef = uOK
|
||||||
unify (Atom _) VoidRef = uOK
|
unify (C _) VoidRef = uOK
|
||||||
unify VoidRef (Atom _) = uOK
|
unify VoidRef (C _) = uOK
|
||||||
unify (Atom a) (Atom b)
|
unify (C a) (C b)
|
||||||
| a == b = uOK
|
| a == b = uOK
|
||||||
unify (Number _) VoidRef = uOK
|
|
||||||
unify VoidRef (Number _) = uOK
|
|
||||||
unify (Number a) (Number b)
|
|
||||||
| a == b = uOK
|
| a == b = uOK
|
||||||
unify (Struct a) VoidRef = do
|
unify (Struct a) VoidRef = do
|
||||||
uNext
|
uNext
|
||||||
|
|
22
app/Load.hs
22
app/Load.hs
|
@ -1,6 +1,6 @@
|
||||||
module Load where
|
module Load where
|
||||||
|
|
||||||
import Code (Code, PrlgEnv)
|
import Code (Code, Id(..), PrlgEnv)
|
||||||
import CodeLens
|
import CodeLens
|
||||||
import qualified Compiler as C
|
import qualified Compiler as C
|
||||||
import Control.Monad.Trans.Class (lift)
|
import Control.Monad.Trans.Class (lift)
|
||||||
|
@ -21,27 +21,23 @@ tokenize fn = left MP.errorBundlePretty . MP.parse P.lexPrlg fn
|
||||||
parse :: String -> [P.Lexeme] -> Either String [P.PAST]
|
parse :: String -> [P.Lexeme] -> Either String [P.PAST]
|
||||||
parse fn = left MP.errorBundlePretty . MP.parse P.parsePrlg fn
|
parse fn = left MP.errorBundlePretty . MP.parse P.parsePrlg fn
|
||||||
|
|
||||||
shunt :: P.PAST -> ExceptT String PrlgEnv IR.PrlgStr
|
shunt :: P.PAST -> ExceptT String PrlgEnv P.PrlgStr
|
||||||
shunt past = do
|
shunt past = do
|
||||||
ops <- lift $ use ops
|
ops <- lift $ use ops
|
||||||
except . left (\err -> "operator resolution: " ++ err ++ "\n") $
|
except . left (\err -> "operator resolution: " ++ err ++ "\n") $
|
||||||
P.shuntPrlg ops past
|
P.shuntPrlg ops past
|
||||||
|
|
||||||
intern :: IR.PrlgStr -> PrlgEnv IR.PrlgInt
|
intern :: P.PrlgStr -> PrlgEnv IR.PrlgInt
|
||||||
intern prlgs = do
|
intern prlgs = do
|
||||||
prlgi <- withStrTable $ \st -> IR.internPrlg st prlgs
|
prlgi <- withStrTable $ \st -> IR.internPrlg st prlgs
|
||||||
underscore <- findAtom "_"
|
underscore <- findAtom "_"
|
||||||
list <- findAtom "[]"
|
|
||||||
withStrTable $ \st ->
|
withStrTable $ \st ->
|
||||||
( st
|
(st, C.squashVars $ C.variablizePrlg underscore st prlgi)
|
||||||
, C.squashVars $ C.variablizePrlg underscore st $ C.desugarPrlg list prlgi)
|
|
||||||
|
|
||||||
compile :: IR.PrlgInt -> PrlgEnv Code
|
compile :: IR.PrlgInt -> ExceptT String PrlgEnv Code
|
||||||
compile prlgv = do
|
compile prlgv = do
|
||||||
comma <- findAtom ","
|
[comma, semi, cut] <- lift $ traverse findAtom [",", ";", "!"]
|
||||||
semi <- findAtom ";"
|
except $ C.compileGoals comma semi cut prlgv
|
||||||
cut <- findAtom "!"
|
|
||||||
return $ C.compileGoals comma semi cut prlgv
|
|
||||||
|
|
||||||
expansion ::
|
expansion ::
|
||||||
(Int -> IR.PrlgInt -> IR.PrlgInt)
|
(Int -> IR.PrlgInt -> IR.PrlgInt)
|
||||||
|
@ -58,9 +54,7 @@ expansion noexpand expander output x = do
|
||||||
if expand
|
if expand
|
||||||
then IR.CallI
|
then IR.CallI
|
||||||
comma
|
comma
|
||||||
[ IR.CallI (IR.str es) [x, IR.VarI (-1) 0]
|
[IR.CallI (str es) [x, IR.VarI (-1)], IR.CallI o [IR.VarI (-1)]]
|
||||||
, IR.CallI o [IR.VarI (-1) 0]
|
|
||||||
]
|
|
||||||
else noexpand o x
|
else noexpand o x
|
||||||
|
|
||||||
queryExpansion = expansion (\_ -> id) "expand_query" "call"
|
queryExpansion = expansion (\_ -> id) "expand_query" "call"
|
||||||
|
|
|
@ -5,7 +5,8 @@ module Parser
|
||||||
, parsePrlg
|
, parsePrlg
|
||||||
, shuntPrlg
|
, shuntPrlg
|
||||||
, PAST
|
, PAST
|
||||||
, Lexeme
|
, Lexeme(..)
|
||||||
|
, PrlgStr(..)
|
||||||
) where
|
) where
|
||||||
|
|
||||||
import Control.Monad (void)
|
import Control.Monad (void)
|
||||||
|
@ -42,14 +43,13 @@ import Text.Megaparsec
|
||||||
)
|
)
|
||||||
import Text.Megaparsec.Char (string)
|
import Text.Megaparsec.Char (string)
|
||||||
|
|
||||||
import IR (PrlgStr(..))
|
|
||||||
import Operators (ArgKind(..), Fixity(..), Op(..), Ops)
|
import Operators (ArgKind(..), Fixity(..), Op(..), Ops)
|
||||||
|
|
||||||
singleToks = ",;|()[]{}!"
|
singleToks = ",;|()[]{}!"
|
||||||
|
|
||||||
identParts = "_"
|
identParts = "_"
|
||||||
|
|
||||||
notOpToks = "\'" ++ identParts
|
notOpToks = "'\"" ++ identParts
|
||||||
|
|
||||||
isOperatorlike x =
|
isOperatorlike x =
|
||||||
(isSymbol x || isPunctuation x) && not (x `elem` singleToks ++ notOpToks)
|
(isSymbol x || isPunctuation x) && not (x `elem` singleToks ++ notOpToks)
|
||||||
|
@ -64,6 +64,7 @@ data Lexeme
|
||||||
= Blank String
|
= Blank String
|
||||||
| Tok String
|
| Tok String
|
||||||
| QTok String String -- unquoted quoted
|
| QTok String String -- unquoted quoted
|
||||||
|
| DQTok String String -- unquoted quoted
|
||||||
deriving (Show, Eq, Ord)
|
deriving (Show, Eq, Ord)
|
||||||
|
|
||||||
blank :: Lexer Lexeme
|
blank :: Lexer Lexeme
|
||||||
|
@ -86,14 +87,15 @@ qtok = do
|
||||||
z <- string "'"
|
z <- string "'"
|
||||||
return $ QTok y (x ++ y ++ z)
|
return $ QTok y (x ++ y ++ z)
|
||||||
|
|
||||||
cmt :: Lexer Lexeme
|
dqtok :: Lexer Lexeme
|
||||||
cmt =
|
dqtok = do
|
||||||
Blank . concat <$>
|
x <- string "\""
|
||||||
sequence
|
y <- many $ satisfy (/= '\"') -- TODO actual escaping
|
||||||
[string "%", many $ satisfy (/= '\n'), choice [string "\n", "" <$ eof]]
|
z <- string "\""
|
||||||
|
return $ DQTok y (x ++ y ++ z)
|
||||||
|
|
||||||
lexeme :: Lexer Lexeme
|
lexeme :: Lexer Lexeme
|
||||||
lexeme = choice [blank, tok, qtok, cmt]
|
lexeme = choice [blank, tok, qtok, dqtok]
|
||||||
|
|
||||||
lexPrlg :: Lexer [Lexeme]
|
lexPrlg :: Lexer [Lexeme]
|
||||||
lexPrlg = many lexeme <* (many blank >> eof)
|
lexPrlg = many lexeme <* (many blank >> eof)
|
||||||
|
@ -101,6 +103,7 @@ lexPrlg = many lexeme <* (many blank >> eof)
|
||||||
showTok (Blank x) = x
|
showTok (Blank x) = x
|
||||||
showTok (Tok x) = x
|
showTok (Tok x) = x
|
||||||
showTok (QTok _ x) = x
|
showTok (QTok _ x) = x
|
||||||
|
showTok (DQTok _ x) = x
|
||||||
|
|
||||||
instance VisualStream [Lexeme] where
|
instance VisualStream [Lexeme] where
|
||||||
showTokens _ (a :| b) = concatMap showTok (a : b)
|
showTokens _ (a :| b) = concatMap showTok (a : b)
|
||||||
|
@ -146,9 +149,9 @@ instance TraversableStream [Lexeme] where
|
||||||
|
|
||||||
data PAST
|
data PAST
|
||||||
= Call String [[PAST]]
|
= Call String [[PAST]]
|
||||||
| Seq [PAST]
|
| Group [PAST]
|
||||||
| List [[PAST]] (Maybe [PAST])
|
| List [[PAST]] (Maybe [PAST])
|
||||||
| Literal String
|
| Literal Lexeme
|
||||||
deriving (Show, Eq)
|
deriving (Show, Eq)
|
||||||
|
|
||||||
type Parser = Parsec Void [Lexeme]
|
type Parser = Parsec Void [Lexeme]
|
||||||
|
@ -165,43 +168,46 @@ isNormalTokStr = (`notElem` [".", "[", "]", "(", "|", ")", "{", "}"])
|
||||||
isNormalTok :: Lexeme -> Bool
|
isNormalTok :: Lexeme -> Bool
|
||||||
isNormalTok (Tok x) = isNormalTokStr x
|
isNormalTok (Tok x) = isNormalTokStr x
|
||||||
isNormalTok (QTok _ _) = True
|
isNormalTok (QTok _ _) = True
|
||||||
|
isNormalTok (DQTok _ _) = True
|
||||||
isNormalTok _ = False
|
isNormalTok _ = False
|
||||||
|
|
||||||
isCallTok :: Lexeme -> Bool
|
isCallTok :: Lexeme -> Bool
|
||||||
isCallTok (Tok x) =
|
isCallTok (Tok x) =
|
||||||
all (\c -> not (isSymbol c) && not (isPunctuation c) || c `elem` identParts) x
|
all (\c -> not (isSymbol c) && not (isPunctuation c) || c `elem` identParts) x
|
||||||
isCallTok _ = True
|
isCallTok (QTok _ _) = True
|
||||||
|
isCallTok _ = False
|
||||||
|
|
||||||
unTok (Tok t) = t
|
unTok (Tok t) = t
|
||||||
unTok (QTok t _) = t
|
unTok (QTok t _) = t
|
||||||
|
unTok (DQTok t _) = t
|
||||||
|
|
||||||
literal :: Parser PAST
|
literal :: Parser PAST
|
||||||
literal =
|
literal =
|
||||||
Literal . unTok <$>
|
Literal <$>
|
||||||
free
|
free
|
||||||
(choice
|
(choice
|
||||||
[ satisfy isCallTok <* notFollowedBy (choice [lParen, lBrace])
|
[ satisfy isCallTok <* notFollowedBy (choice [lParen, lBrace])
|
||||||
, satisfy (\x -> not (isCallTok x) && isNormalTok x)
|
, satisfy (\x -> not (isCallTok x) && isNormalTok x)
|
||||||
])
|
])
|
||||||
|
|
||||||
makeParams (Seq inner) = splitOn [Literal ","] inner
|
makeParams (Group inner) = splitOn [Literal (Tok ",")] inner
|
||||||
|
|
||||||
call eb contents fmod = do
|
call eb contents fmod = do
|
||||||
fn <- fmod . unTok <$> satisfy isCallTok -- not free
|
fn <- fmod . unTok <$> satisfy isCallTok -- not free
|
||||||
(Call fn [] <$ try eb) <|> (Call fn . makeParams <$> free contents)
|
(Call fn [] <$ try eb) <|> (Call fn . makeParams <$> free contents)
|
||||||
|
|
||||||
parens = Seq <$> (free lParen *> some seqItem <* free rParen)
|
parens = Group <$> (free lParen *> some seqItem <* free rParen)
|
||||||
|
|
||||||
braces = Seq <$> (free lBrace *> some seqItem <* free rBrace)
|
braces = Group <$> (free lBrace *> some seqItem <* free rBrace)
|
||||||
|
|
||||||
emptyParens = Literal "()" <$ (free lParen >> free rParen)
|
emptyParens = Literal (QTok "()" "()") <$ (free lParen >> free rParen)
|
||||||
|
|
||||||
emptyBraces = Literal "{}" <$ (free lBrace >> free rBrace)
|
emptyBraces = Literal (QTok "{}" "{}") <$ (free lBrace >> free rBrace)
|
||||||
|
|
||||||
list = do
|
list = do
|
||||||
free lBracket
|
free lBracket
|
||||||
(List [] Nothing <$ free rBracket) <|> do
|
(List [] Nothing <$ free rBracket) <|> do
|
||||||
items <- splitOn [Literal ","] <$> some seqItem
|
items <- splitOn [Literal (Tok ",")] <$> some seqItem
|
||||||
(List items Nothing <$ free rBracket) <|>
|
(List items Nothing <$ free rBracket) <|>
|
||||||
(List items . Just <$> (free listTail *> some seqItem <* free rBracket))
|
(List items . Just <$> (free listTail *> some seqItem <* free rBracket))
|
||||||
|
|
||||||
|
@ -237,7 +243,7 @@ lBrace = simpleTok "{"
|
||||||
rBrace = simpleTok "}"
|
rBrace = simpleTok "}"
|
||||||
|
|
||||||
clause :: Parser PAST
|
clause :: Parser PAST
|
||||||
clause = Seq <$> some (free seqItem) <* free period
|
clause = Group <$> some (free seqItem) <* free period
|
||||||
|
|
||||||
parsePrlg :: Parser [PAST]
|
parsePrlg :: Parser [PAST]
|
||||||
parsePrlg = ws *> many clause <* eof
|
parsePrlg = ws *> many clause <* eof
|
||||||
|
@ -246,6 +252,11 @@ type ShuntError = String
|
||||||
|
|
||||||
type ShuntResult = Either ShuntError PrlgStr
|
type ShuntResult = Either ShuntError PrlgStr
|
||||||
|
|
||||||
|
data PrlgStr
|
||||||
|
= CallS String [PrlgStr]
|
||||||
|
| LiteralS Lexeme
|
||||||
|
deriving (Show)
|
||||||
|
|
||||||
err :: ShuntError -> Either ShuntError a
|
err :: ShuntError -> Either ShuntError a
|
||||||
err = Left
|
err = Left
|
||||||
|
|
||||||
|
@ -254,8 +265,10 @@ shuntPrlg ot = shuntPrlg' (("", Op 0 $ Infix Y X) : ot)
|
||||||
|
|
||||||
shuntPrlg' :: Ops -> PAST -> ShuntResult
|
shuntPrlg' :: Ops -> PAST -> ShuntResult
|
||||||
shuntPrlg' ot (List hs t) =
|
shuntPrlg' ot (List hs t) =
|
||||||
ListS <$> traverse (shunt ot) hs <*> traverse (shunt ot) t
|
foldr (\x y -> CallS "[]" [x, y]) <$>
|
||||||
shuntPrlg' ot (Seq ss) = shunt ot ss
|
(maybe (LiteralS $ Tok "[]") id <$> traverse (shunt ot) t) <*>
|
||||||
|
traverse (shunt ot) hs
|
||||||
|
shuntPrlg' ot (Group ss) = shunt ot ss
|
||||||
shuntPrlg' ot (Literal s) = pure (LiteralS s)
|
shuntPrlg' ot (Literal s) = pure (LiteralS s)
|
||||||
shuntPrlg' ot (Call fn ss) = CallS fn <$> traverse (shunt ot) ss
|
shuntPrlg' ot (Call fn ss) = CallS fn <$> traverse (shunt ot) ss
|
||||||
|
|
||||||
|
@ -296,13 +309,13 @@ shunt optable = start
|
||||||
(ops', vs') <- pushInfix ops vs x
|
(ops', vs') <- pushInfix ops vs x
|
||||||
wo ops' vs' xs
|
wo ops' vs' xs
|
||||||
, do getOperand x
|
, do getOperand x
|
||||||
ho ops vs (Literal "" : xs') -- app (see below)
|
ho ops vs (Literal (Tok "") : xs') -- app (see below)
|
||||||
, do getPrefix x
|
, do getPrefix x
|
||||||
ho ops vs (Literal "" : xs') -- also app!
|
ho ops vs (Literal (Tok "") : xs') -- also app!
|
||||||
, err "expected infix or suffix operator"
|
, err "expected infix or suffix operator"
|
||||||
]
|
]
|
||||||
{- incoming non-literal operand; there must be an app in between -}
|
{- incoming non-literal operand; there must be an app in between -}
|
||||||
ho ops vs xs@(_:_) = ho ops vs (Literal "" : xs)
|
ho ops vs xs@(_:_) = ho ops vs (Literal (Tok "") : xs)
|
||||||
{- the last operand was last, pop until finished -}
|
{- the last operand was last, pop until finished -}
|
||||||
ho [] [res] [] = pure res
|
ho [] [res] [] = pure res
|
||||||
ho ops vs [] = do
|
ho ops vs [] = do
|
||||||
|
@ -319,11 +332,14 @@ shunt optable = start
|
||||||
{- Operator checks -}
|
{- Operator checks -}
|
||||||
uniq [x] = pure x
|
uniq [x] = pure x
|
||||||
uniq _ = err "ambiguous operator"
|
uniq _ = err "ambiguous operator"
|
||||||
getPrefix x = uniq [op | (s, op@(Op _ (Prefix _))) <- optable, s == x]
|
getPrefix t =
|
||||||
getSuffix x = uniq [op | (s, op@(Op _ (Suffix _))) <- optable, s == x]
|
uniq [op | Tok x <- [t], (s, op@(Op _ (Prefix _))) <- optable, s == x]
|
||||||
getInfix x = uniq [op | (s, op@(Op _ (Infix _ _))) <- optable, s == x]
|
getSuffix t =
|
||||||
getOperand x
|
uniq [op | Tok x <- [t], (s, op@(Op _ (Suffix _))) <- optable, s == x]
|
||||||
| null [op | (s, op) <- optable, s == x] = pure ()
|
getInfix t =
|
||||||
|
uniq [op | Tok x <- [t], (s, op@(Op _ (Infix _ _))) <- optable, s == x]
|
||||||
|
getOperand t
|
||||||
|
| null [op | Tok x <- [t], (s, op) <- optable, s == x] = pure ()
|
||||||
| otherwise = err "expected an operand"
|
| otherwise = err "expected an operand"
|
||||||
{- actual pushery -}
|
{- actual pushery -}
|
||||||
canPush :: Ops -> Op -> Either ShuntError Bool
|
canPush :: Ops -> Op -> Either ShuntError Bool
|
||||||
|
@ -371,7 +387,7 @@ shunt optable = start
|
||||||
shunt1 ops vs x op = do
|
shunt1 ops vs x op = do
|
||||||
cp <- canPush ops op
|
cp <- canPush ops op
|
||||||
if cp
|
if cp
|
||||||
then pure ((x, op) : ops, vs)
|
then pure ((unTok x, op) : ops, vs)
|
||||||
else do
|
else do
|
||||||
(ops', vs') <- pop ops vs
|
(ops', vs') <- pop ops vs
|
||||||
shunt1 ops' vs' x op --prefix would behave differently here but that's impossible by canPush
|
shunt1 ops' vs' x op --prefix would behave differently here but that's impossible by canPush
|
||||||
|
|
|
@ -1,9 +1,13 @@
|
||||||
member(X, [X|_]).
|
member(X, [X|_]).
|
||||||
member(X, [_|T]) :- member(X,T).
|
member(X, [_|T]) :- member(X,T).
|
||||||
|
|
||||||
append([], X, X).
|
append([], [], []).
|
||||||
|
append([], [H|T], [H|T]).
|
||||||
append([X|T], Y, [X|TY]) :- append(T,Y,TY).
|
append([X|T], Y, [X|TY]) :- append(T,Y,TY).
|
||||||
|
|
||||||
|
list([]).
|
||||||
|
list([_|_]).
|
||||||
|
|
||||||
:- op(700, xfx, is),
|
:- op(700, xfx, is),
|
||||||
op(700, xfx, <),
|
op(700, xfx, <),
|
||||||
op(700, xfx, =<),
|
op(700, xfx, =<),
|
||||||
|
@ -45,17 +49,44 @@ Ax > Bx :- A is Ax, B is Bx, int2p_lt(B,A).
|
||||||
Ax >= Bx :- A is Ax, B is Bx, int2p_leq(B,A).
|
Ax >= Bx :- A is Ax, B is Bx, int2p_leq(B,A).
|
||||||
zero(Ax) :- A is Ax, int1p_zero(A).
|
zero(Ax) :- A is Ax, int1p_zero(A).
|
||||||
|
|
||||||
gcd(X,Y,R) :- writeln(a), fail.
|
gcd(X,Y,R) :- Y > X, !, gcd(Y,X,R).
|
||||||
gcd(X,Y,R) :- writeln(b), zero(Y), !, R=X.
|
gcd(X,Y,R) :- zero(Y), !, R=X.
|
||||||
gcd(X,Y,R) :- writeln(c), X1 is X mod Y, gcd(Y,X1,R).
|
gcd(X,Y,R) :- X1 is X mod Y, gcd(Y,X1,R).
|
||||||
gcd(X,Y,R) :- writeln(a), Y > X, writeln(wat), !, gcd(Y,X,R).
|
|
||||||
gcd(X,Y,R) :- writeln(b), zero(Y), !, R=X.
|
|
||||||
gcd(X,Y,R) :- writeln(c), X1 is X mod Y, gcd(Y,X1,R).
|
|
||||||
|
|
||||||
test(X) :- writeln(there), zero(X), fail.
|
lcm(X,Y,R) :- gcd(X,Y,GCD), R is X*(Y/GCD).
|
||||||
test(X) :- writeln(here).
|
|
||||||
|
|
||||||
test :- writeln(a), a=a, !, fail.
|
:- op(1200, xfx, -->).
|
||||||
test :- writeln(b).
|
|
||||||
|
|
||||||
xxx :- test.
|
sequence([], SameState, SameState) :- !.
|
||||||
|
sequence(Tokens, ParsedList, Rest) :-
|
||||||
|
append(Tokens, Rest, ParsedList).
|
||||||
|
|
||||||
|
load_expansion((X, L) --> Y, Xp :- (Yp, Lp)) :- !,
|
||||||
|
expand_phrasecall(X, Xp, S0, S),
|
||||||
|
expand_phrase(Y, Yp, S0, S),
|
||||||
|
expand_phrase(L, Lp, S, _).
|
||||||
|
|
||||||
|
load_expansion(X --> Y, Xp :- Yp) :- !,
|
||||||
|
expand_phrasecall(X, Xp, S0, S),
|
||||||
|
expand_phrase(Y, Yp, S0, S).
|
||||||
|
|
||||||
|
expand_phrase((A, B), (Ap, Bp), S0, S) :- !,
|
||||||
|
expand_phrase(A, Ap, S0, S1),
|
||||||
|
expand_phrase(B, Bp, S1, S).
|
||||||
|
expand_phrase((A; B), (Ap; Bp), S0, S) :- !,
|
||||||
|
expand_phrase(A, Ap, S0, S),
|
||||||
|
expand_phrase(B, Bp, S0, S).
|
||||||
|
expand_phrase(L, sequence(L, S0, S), S0, S) :- list(L), !.
|
||||||
|
expand_phrase({X}, X, S, S) :- !.
|
||||||
|
expand_phrase(!, !, S, S) :- !.
|
||||||
|
expand_phrase(X, Xp, S0, S) :- expand_phrasecall(X, Xp, S0, S).
|
||||||
|
|
||||||
|
expand_phrasecall(X, Xp, S0, S) :-
|
||||||
|
atom(X), !,
|
||||||
|
struct(Xp, X, [S0, S]).
|
||||||
|
expand_phrasecall(X, Xp, S0, S) :- !,
|
||||||
|
struct(X, Id, Args),
|
||||||
|
append(Args, [S0, S], Args1),
|
||||||
|
struct(Xp, Id, Args1).
|
||||||
|
|
||||||
|
phrase(X, S0, S) :- expand_phrasecall(X, Xp, S0, S), call(Xp).
|
||||||
|
|
30
prlg.cabal
30
prlg.cabal
|
@ -21,18 +21,40 @@ maintainer: exa.exa@gmail.com
|
||||||
-- category:
|
-- category:
|
||||||
extra-source-files: CHANGELOG.md
|
extra-source-files: CHANGELOG.md
|
||||||
|
|
||||||
data-dir: inst
|
data-dir: inst
|
||||||
data-files: *.pl
|
data-files: *.pl
|
||||||
|
|
||||||
executable prlg
|
executable prlg
|
||||||
main-is: Main.hs
|
main-is: Main.hs
|
||||||
|
|
||||||
-- Modules included in this executable, other than Main.
|
-- Modules included in this executable, other than Main.
|
||||||
other-modules: Interpreter, Compiler, Parser, Frontend, IR, Operators, Code, Builtins, Env, Load, CodeLens, Heap, Paths_prlg
|
other-modules: Paths_prlg,
|
||||||
|
Builtins,
|
||||||
|
Code,
|
||||||
|
CodeLens,
|
||||||
|
Compiler,
|
||||||
|
Constant,
|
||||||
|
Env,
|
||||||
|
Frontend,
|
||||||
|
Heap,
|
||||||
|
Interpreter,
|
||||||
|
IR,
|
||||||
|
Load,
|
||||||
|
Operators,
|
||||||
|
Parser
|
||||||
|
|
||||||
-- LANGUAGE extensions used by modules in this package.
|
-- LANGUAGE extensions used by modules in this package.
|
||||||
-- other-extensions:
|
-- other-extensions:
|
||||||
build-depends: base >=4.14, containers, megaparsec, haskeline, split, transformers, microlens, microlens-th, microlens-mtl
|
build-depends: base >=4.14,
|
||||||
|
containers,
|
||||||
|
haskeline,
|
||||||
|
megaparsec,
|
||||||
|
microlens,
|
||||||
|
microlens-mtl,
|
||||||
|
microlens-th,
|
||||||
|
split,
|
||||||
|
transformers
|
||||||
|
|
||||||
hs-source-dirs: app
|
hs-source-dirs: app
|
||||||
default-language: Haskell2010
|
default-language: Haskell2010
|
||||||
ghc-options: -Wunused-imports
|
ghc-options: -Wunused-imports
|
||||||
|
|
Loading…
Reference in a new issue