slight cleanup, metacall
This commit is contained in:
parent
32f6fe0291
commit
60ff47250b
|
@ -2,11 +2,15 @@ module Builtins where
|
||||||
|
|
||||||
import Code
|
import Code
|
||||||
( Builtin(..)
|
( Builtin(..)
|
||||||
, BuiltinFn
|
|
||||||
, Cho(..)
|
, Cho(..)
|
||||||
, Datum(..)
|
, Datum(..)
|
||||||
|
, Dereferenced(..)
|
||||||
|
, Heap(..)
|
||||||
, Instr(..)
|
, Instr(..)
|
||||||
, Interp(..)
|
, Interp(..)
|
||||||
|
, InterpFn
|
||||||
|
, InterpFn
|
||||||
|
, derefHeap
|
||||||
, heapStruct
|
, heapStruct
|
||||||
)
|
)
|
||||||
import qualified Compiler as Co
|
import qualified Compiler as Co
|
||||||
|
@ -16,7 +20,7 @@ import Control.Monad.Trans.State.Lazy (get, gets, modify)
|
||||||
import Data.Functor.Identity (runIdentity)
|
import Data.Functor.Identity (runIdentity)
|
||||||
import Data.List (intercalate)
|
import Data.List (intercalate)
|
||||||
import qualified Data.Map as M
|
import qualified Data.Map as M
|
||||||
import Env (PrlgEnv(..), findAtom, findStruct)
|
import Env (PrlgEnv(..), findAtom, findStruct, prlgError)
|
||||||
import qualified IR
|
import qualified IR
|
||||||
import Interpreter (backtrack)
|
import Interpreter (backtrack)
|
||||||
import qualified Operators as O
|
import qualified Operators as O
|
||||||
|
@ -37,7 +41,7 @@ showTerm itos heap = runIdentity . heapStruct atom struct hrec heap
|
||||||
else "_Rec") ++
|
else "_Rec") ++
|
||||||
show hr
|
show hr
|
||||||
|
|
||||||
printLocals :: BuiltinFn
|
printLocals :: InterpFn
|
||||||
printLocals = do
|
printLocals = do
|
||||||
scope <- gets (gvar . cur)
|
scope <- gets (gvar . cur)
|
||||||
heap <- gets (heap . cur)
|
heap <- gets (heap . cur)
|
||||||
|
@ -47,37 +51,39 @@ printLocals = do
|
||||||
(maybe "_" id $ itos M.!? name) ++ " = " ++ showTerm itos heap ref
|
(maybe "_" id $ itos M.!? name) ++ " = " ++ showTerm itos heap ref
|
||||||
return Nothing
|
return Nothing
|
||||||
|
|
||||||
promptRetry :: BuiltinFn
|
promptRetry :: InterpFn
|
||||||
promptRetry = do
|
promptRetry = do
|
||||||
last <- gets (null . cho)
|
last <- gets (null . cho)
|
||||||
if last
|
if last
|
||||||
then return Nothing
|
then return Nothing
|
||||||
else promptRetry'
|
else promptRetry'
|
||||||
|
|
||||||
promptRetry' :: BuiltinFn
|
promptRetry' :: InterpFn
|
||||||
promptRetry' = do
|
promptRetry' = do
|
||||||
x <- lift $ getInputChar "? "
|
x <- lift $ getInputChar "? "
|
||||||
case x of
|
case x of
|
||||||
Just ';' -> backtrack
|
Just ';' -> backtrack
|
||||||
_ -> return Nothing
|
_ -> return Nothing
|
||||||
|
|
||||||
write :: BuiltinFn
|
write :: InterpFn
|
||||||
write = do
|
write
|
||||||
|
--TODO: prlgError on write(Unbound)
|
||||||
|
= do
|
||||||
scope <- gets (hvar . cur)
|
scope <- gets (hvar . cur)
|
||||||
heap <- gets (heap . cur)
|
heap <- gets (heap . cur)
|
||||||
IR.StrTable _ _ itos <- gets strtable
|
IR.StrTable _ _ itos <- gets strtable
|
||||||
lift . outputStr . showTerm itos heap . fst $ scope M.! 0
|
lift . outputStr . showTerm itos heap . fst $ scope M.! 0
|
||||||
return Nothing
|
return Nothing
|
||||||
|
|
||||||
nl :: BuiltinFn
|
nl :: InterpFn
|
||||||
nl = do
|
nl = do
|
||||||
lift $ outputStrLn ""
|
lift $ outputStrLn ""
|
||||||
return Nothing
|
return Nothing
|
||||||
|
|
||||||
writeln :: BuiltinFn
|
writeln :: InterpFn
|
||||||
writeln = write >> nl
|
writeln = write >> nl
|
||||||
|
|
||||||
assertFact :: BuiltinFn
|
assertFact :: InterpFn
|
||||||
assertFact = do
|
assertFact = do
|
||||||
scope <- gets (hvar . cur)
|
scope <- gets (hvar . cur)
|
||||||
heap <- gets (heap . cur)
|
heap <- gets (heap . cur)
|
||||||
|
@ -86,9 +92,9 @@ assertFact = do
|
||||||
Just (U (Struct s):head) -> do
|
Just (U (Struct s):head) -> do
|
||||||
addClause s $ head ++ [NoGoal]
|
addClause s $ head ++ [NoGoal]
|
||||||
return Nothing
|
return Nothing
|
||||||
_ -> backtrack --TODO actually throw
|
_ -> prlgError "assert fact failure"
|
||||||
|
|
||||||
assertClause :: BuiltinFn
|
assertClause :: InterpFn
|
||||||
assertClause = do
|
assertClause = do
|
||||||
scope <- gets (hvar . cur)
|
scope <- gets (hvar . cur)
|
||||||
heap <- gets (heap . cur)
|
heap <- gets (heap . cur)
|
||||||
|
@ -101,11 +107,33 @@ assertClause = do
|
||||||
Co.compileGoal hs ++ Co.seqGoals (Co.compileGoals commaId cut gs)
|
Co.compileGoal hs ++ Co.seqGoals (Co.compileGoals commaId cut gs)
|
||||||
in do addClause s cs
|
in do addClause s cs
|
||||||
return Nothing
|
return Nothing
|
||||||
_ -> backtrack
|
_ -> prlgError "assert clause failure"
|
||||||
|
|
||||||
retractall :: BuiltinFn
|
retractall :: InterpFn
|
||||||
retractall = do
|
retractall = prlgError "no retractall yet"
|
||||||
return Nothing
|
|
||||||
|
call :: InterpFn
|
||||||
|
call = do
|
||||||
|
ref <- gets (fst . (M.! 0) . hvar . cur)
|
||||||
|
heap@(Heap _ hmap) <- gets (heap . cur)
|
||||||
|
let exec base struct arity = do
|
||||||
|
cur <- gets cur
|
||||||
|
modify $ \s ->
|
||||||
|
s
|
||||||
|
{ cur =
|
||||||
|
cur
|
||||||
|
{ gol =
|
||||||
|
[Call, Goal, U struct] ++
|
||||||
|
[U $ hmap M.! (base + i) | i <- [1 .. arity]] ++ gol cur
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return Nothing
|
||||||
|
case derefHeap heap ref of
|
||||||
|
BoundRef addr struct@(Struct IR.Id {IR.arity = arity}) ->
|
||||||
|
exec addr struct arity
|
||||||
|
BoundRef addr (Atom a) ->
|
||||||
|
exec addr (Struct IR.Id {IR.arity = 0, IR.str = a}) 0
|
||||||
|
_ -> prlgError "not callable"
|
||||||
|
|
||||||
{- adding the builtins -}
|
{- adding the builtins -}
|
||||||
addOp op = modify $ \s -> s {ops = op : ops s}
|
addOp op = modify $ \s -> s {ops = op : ops s}
|
||||||
|
@ -148,6 +176,7 @@ addPrelude = do
|
||||||
, [U (LocalRef 0 0), Invoke (bi assertFact)]
|
, [U (LocalRef 0 0), Invoke (bi assertFact)]
|
||||||
]
|
]
|
||||||
addProc "retractall" 1 [[U (LocalRef 0 0), Invoke (bi retractall)]]
|
addProc "retractall" 1 [[U (LocalRef 0 0), Invoke (bi retractall)]]
|
||||||
|
addProc "call" 1 [[U (LocalRef 0 0), Invoke (bi call)]]
|
||||||
{- query tools -}
|
{- query tools -}
|
||||||
addBi0 "print_locals" printLocals
|
addBi0 "print_locals" printLocals
|
||||||
addBi0 "prompt_retry" promptRetry'
|
addBi0 "prompt_retry" promptRetry'
|
||||||
|
|
20
app/Code.hs
20
app/Code.hs
|
@ -65,14 +65,30 @@ data Interp =
|
||||||
|
|
||||||
type PrlgEnv a = StateT Interp (InputT IO) a
|
type PrlgEnv a = StateT Interp (InputT IO) a
|
||||||
|
|
||||||
type BuiltinFn = PrlgEnv (Maybe (Either String Bool))
|
type InterpFn = PrlgEnv (Maybe (Either String Bool))
|
||||||
|
|
||||||
data Builtin =
|
data Builtin =
|
||||||
Builtin BuiltinFn
|
Builtin InterpFn
|
||||||
|
|
||||||
instance Show Builtin where
|
instance Show Builtin where
|
||||||
show _ = "Builtin _"
|
show _ = "Builtin _"
|
||||||
|
|
||||||
|
data Dereferenced
|
||||||
|
= FreeRef Int
|
||||||
|
| BoundRef Int Datum
|
||||||
|
| NoRef
|
||||||
|
|
||||||
|
-- TRICKY: HeapRefs alone must not form a cycle other than the FreeRef case.
|
||||||
|
derefHeap :: Heap -> Int -> Dereferenced
|
||||||
|
derefHeap h@(Heap _ hmap) x =
|
||||||
|
case hmap M.!? x of
|
||||||
|
Just (HeapRef x') ->
|
||||||
|
if x == x'
|
||||||
|
then FreeRef x'
|
||||||
|
else derefHeap h x'
|
||||||
|
Just x' -> BoundRef x x'
|
||||||
|
_ -> NoRef
|
||||||
|
|
||||||
-- TODO are we actually going to use this?
|
-- TODO are we actually going to use this?
|
||||||
codeStruct ::
|
codeStruct ::
|
||||||
Monad m
|
Monad m
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
module Env where
|
module Env where
|
||||||
|
|
||||||
import Code (Interp(..), PrlgEnv)
|
import Code (Interp(..), InterpFn, PrlgEnv)
|
||||||
import Control.Monad.Trans.State.Lazy (gets, modify)
|
import Control.Monad.Trans.State.Lazy (gets, modify)
|
||||||
import qualified IR
|
import qualified IR
|
||||||
|
|
||||||
|
@ -20,3 +20,6 @@ findAtom :: String -> Env.PrlgEnv Int
|
||||||
findAtom = withStrTable . flip IR.strtablize
|
findAtom = withStrTable . flip IR.strtablize
|
||||||
|
|
||||||
type PrlgEnv a = Code.PrlgEnv a
|
type PrlgEnv a = Code.PrlgEnv a
|
||||||
|
|
||||||
|
prlgError :: String -> InterpFn
|
||||||
|
prlgError = pure . pure . Left
|
||||||
|
|
|
@ -16,7 +16,7 @@ data Id =
|
||||||
deriving (Show, Eq, Ord)
|
deriving (Show, Eq, Ord)
|
||||||
|
|
||||||
data PrlgInt
|
data PrlgInt
|
||||||
= CallI Id [PrlgInt]
|
= CallI Id [PrlgInt] --TODO this should be Int
|
||||||
| LiteralI Int
|
| LiteralI Int
|
||||||
| VarI Int Int -- VarI localIndex strTableString
|
| VarI Int Int -- VarI localIndex strTableString
|
||||||
| VoidI
|
| VoidI
|
||||||
|
|
|
@ -6,9 +6,12 @@ import Code
|
||||||
, Cho(..)
|
, Cho(..)
|
||||||
, Code
|
, Code
|
||||||
, Datum(..)
|
, Datum(..)
|
||||||
|
, Dereferenced(..)
|
||||||
, Heap(..)
|
, Heap(..)
|
||||||
, Instr(..)
|
, Instr(..)
|
||||||
, Interp(..)
|
, Interp(..)
|
||||||
|
, InterpFn
|
||||||
|
, derefHeap
|
||||||
, emptyHeap
|
, emptyHeap
|
||||||
, emptyScope
|
, emptyScope
|
||||||
)
|
)
|
||||||
|
@ -43,13 +46,8 @@ prove g = do
|
||||||
Nothing -> loop -- not finished yet
|
Nothing -> loop -- not finished yet
|
||||||
Just x -> return x
|
Just x -> return x
|
||||||
|
|
||||||
data Dereferenced
|
|
||||||
= FreeRef Int
|
|
||||||
| BoundRef Int Datum
|
|
||||||
| NoRef
|
|
||||||
|
|
||||||
{- Simple "fail" backtracking -}
|
{- Simple "fail" backtracking -}
|
||||||
backtrack :: PrlgEnv (Maybe (Either String Bool))
|
backtrack :: InterpFn
|
||||||
backtrack = do
|
backtrack = do
|
||||||
chos <- St.gets cho
|
chos <- St.gets cho
|
||||||
case chos
|
case chos
|
||||||
|
@ -61,7 +59,7 @@ backtrack = do
|
||||||
{- if there's no other choice, answer no -}
|
{- if there's no other choice, answer no -}
|
||||||
_ -> pure . Just $ Right False
|
_ -> pure . Just $ Right False
|
||||||
|
|
||||||
proveStep :: PrlgEnv (Maybe (Either String Bool))
|
proveStep :: InterpFn
|
||||||
proveStep = St.get >>= go
|
proveStep = St.get >>= go
|
||||||
where
|
where
|
||||||
finish = pure . Just
|
finish = pure . Just
|
||||||
|
@ -76,23 +74,14 @@ proveStep = St.get >>= go
|
||||||
Just d -> cont d
|
Just d -> cont d
|
||||||
_ -> ifail $ "no definition: " ++ show fn
|
_ -> ifail $ "no definition: " ++ show fn
|
||||||
{- Unification -}
|
{- Unification -}
|
||||||
go i@Interp {cur = cur@Cho { hed = U h:hs
|
go i@Interp {cur = cur@Cho {hed = U h:hs, gol = U g:gs, heap = heap}} =
|
||||||
, gol = U g:gs
|
unify h g
|
||||||
, heap = heap@(Heap _ hmap)
|
|
||||||
}} = unify h g
|
|
||||||
where
|
where
|
||||||
uok = c i {cur = cur {hed = hs, gol = gs}}
|
uok = c i {cur = cur {hed = hs, gol = gs}}
|
||||||
setHeap r x =
|
setHeap r x =
|
||||||
c i {cur = cur {hed = hs, gol = gs, heap = writeHeap r x heap}}
|
c i {cur = cur {hed = hs, gol = gs, heap = writeHeap r x heap}}
|
||||||
{- heap tools -}
|
{- heap tools -}
|
||||||
deref x =
|
deref = derefHeap heap
|
||||||
case hmap M.!? x of
|
|
||||||
Just (HeapRef x') ->
|
|
||||||
if x == x'
|
|
||||||
then FreeRef x'
|
|
||||||
else deref x'
|
|
||||||
Just x' -> BoundRef x x'
|
|
||||||
_ -> NoRef
|
|
||||||
writeHeap addr x (Heap nxt m) = Heap nxt (M.adjust (const x) addr m)
|
writeHeap addr x (Heap nxt m) = Heap nxt (M.adjust (const x) addr m)
|
||||||
newHeapVar h = head <$> newHeapVars 1 h
|
newHeapVar h = head <$> newHeapVars 1 h
|
||||||
newHeapVars n (Heap nxt m) =
|
newHeapVars n (Heap nxt m) =
|
||||||
|
|
Loading…
Reference in a new issue