slight cleanup, metacall

This commit is contained in:
Mirek Kratochvil 2022-12-14 19:47:41 +01:00
parent 32f6fe0291
commit 60ff47250b
5 changed files with 76 additions and 39 deletions

View file

@ -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'

View file

@ -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

View file

@ -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

View file

@ -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

View file

@ -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) =