Skip to content

Commit

Permalink
Add specs for Language.R*
Browse files Browse the repository at this point in the history
  • Loading branch information
facundominguez committed Jan 10, 2024
1 parent afa690b commit aa5333b
Show file tree
Hide file tree
Showing 5 changed files with 133 additions and 83 deletions.
7 changes: 4 additions & 3 deletions inline-r/inline-r.cabal
Original file line number Diff line number Diff line change
Expand Up @@ -59,18 +59,19 @@ library
Foreign.R.Internal
Foreign.R.Parse
Foreign.R.Type
Foreign.R.Type.Singletons
-- H.Prelude
-- H.Prelude.Interactive
-- Language.R
Language.R
-- Language.R.Debug
Language.R.GC
Language.R.Globals
Language.R.HExp
Language.R.Instance
-- Language.R.Internal
Language.R.Internal
Language.R.Internal.FunWrappers
Language.R.Internal.FunWrappers.TH
-- Language.R.Literal
Language.R.Literal
-- Language.R.Matcher
-- Language.R.QQ
if !os(windows)
Expand Down
58 changes: 31 additions & 27 deletions inline-r/src/Language/R.hs
Original file line number Diff line number Diff line change
Expand Up @@ -36,12 +36,7 @@ import qualified Data.Vector.SEXP as Vector
import Control.Monad.R.Class
import Foreign.R
( SEXP
, SomeSEXP(..)
, typeOf
, asTypeOf
, cast
, unSomeSEXP
, unsafeCoerce
)
import qualified Foreign.R as R
import qualified Foreign.R.Parse as R
Expand Down Expand Up @@ -73,88 +68,96 @@ import Prelude
-- the dependency hierarchy.

-- | Parse and then evaluate expression.
parseEval :: ByteString -> IO (SomeSEXP V)
parseEval :: ByteString -> IO (SEXP V)
parseEval txt = useAsCString txt $ \ctxt ->
R.withProtected (R.mkString ctxt) $ \rtxt ->
alloca $ \status -> do
R.withProtected (R.parseVector rtxt 1 status (R.release nilValue)) $ \exprs -> do
rc <- fromIntegral <$> peek status
unless (R.PARSE_OK == toEnum rc) $
runRegion $ throwRMessage $ "Parse error in: " ++ C8.unpack txt
SomeSEXP expr <- peek $ castPtr $ R.unsafeSEXPToVectorPtr exprs
expr <- peek $ castPtr $ R.unsafeSEXPToVectorPtr exprs
runRegion $ do
SomeSEXP val <- eval expr
return $ SomeSEXP (R.release val)
val <- eval expr
return (R.release val)

-- | Parse file and perform some actions on parsed file.
--
-- This function uses continuation because this is an easy way to make
-- operations GC-safe.
parseFile :: FilePath -> (SEXP s 'R.Expr -> IO a) -> IO a
{-@ parseFile :: FilePath -> (SEXP s Foreign.R.Type.Expr -> IO a) -> IO a @-}
parseFile :: FilePath -> (SEXP s -> IO a) -> IO a
{-# DEPRECATED parseFile "Use [r| parse(file=\"path/to/file\") |] instead." #-}
parseFile fl f = do
withCString fl $ \cfl ->
R.withProtected (R.mkString cfl) $ \rfl ->
r1 (C8.pack "parse") rfl >>= \(R.SomeSEXP s) ->
return (R.unsafeCoerce s) `R.withProtected` f
r1 (C8.pack "parse") rfl >>= \s ->
return s `R.withProtected` f

{-@ parseText :: String -> Bool -> IO (R.SEXP V Foreign.R.Type.Expr) @-}
parseText
:: String -- ^ Text to parse
-> Bool -- ^ Whether to annotate the AST with source locations.
-> IO (R.SEXP V 'R.Expr)
-> IO (R.SEXP V)
{-# DEPRECATED parseText "Use [r| parse(text=...) |] instead." #-}
parseText txt b = do
s <- parseEval $ C8.pack $
"parse(text=" ++ show txt ++ ", keep.source=" ++ keep ++ ")"
return $ (sing :: R.SSEXPTYPE 'R.Expr) `R.cast` s
return $ R.Expr `R.checkSEXPTYPE` s
where
keep | b = "TRUE"
| otherwise = "FALSE"

-- | Internalize a symbol name.
install :: MonadR m => String -> m (SEXP V 'R.Symbol)
{-@ install :: String -> m (SEXP V Foreign.R.Type.Symbol) @-}
install :: MonadR m => String -> m (SEXP V)
install = io . installIO

{-# DEPRECATED string, strings "Use mkSEXP instead" #-}

-- | Create an R character string from a Haskell string.
string :: String -> IO (SEXP V 'R.Char)
{-@ string :: String -> IO (SEXP V Foreign.R.Type.Char) @-}
string :: String -> IO (SEXP V)
string str = withCString str R.mkChar

-- | Create an R string vector from a Haskell string.
strings :: String -> IO (SEXP V 'R.String)
{-@ strings :: String -> IO (SEXP V Foreign.R.Type.String) @-}
strings :: String -> IO (SEXP V)
strings str = withCString str R.mkString

-- | Evaluate a (sequence of) expression(s) in the given environment, returning the
-- value of the last.
evalEnv :: MonadR m => SEXP s a -> SEXP s 'R.Env -> m (SomeSEXP (Region m))
evalEnv (hexp -> Language.R.HExp.Expr _ v) rho = acquireSome =<< do
{-@ assume evalEnv :: SEXP s a -> TSEXP s Foreign.R.Type.Env -> m (SEXP (Region m)) @-}
{-@ ignore evalEnv @-}
evalEnv :: MonadR m => SEXP s -> SEXP s -> m (SEXP (Region m))
evalEnv (hexp -> Language.R.HExp.Expr _ v) rho = acquire =<< do
io $ alloca $ \p -> do
mapM_ (\(SomeSEXP s) -> void $ R.protect s) (Vector.toList v)
x <- Prelude.last <$> forM (Vector.toList v) (\(SomeSEXP s) -> do
mapM_ (\s -> void $ R.protect s) (Vector.toList v)
x <- Prelude.last <$> forM (Vector.toList v) (\s -> do
z <- R.tryEvalSilent s (R.release rho) p
e <- peek p
when (e /= 0) $ runRegion $ throwR rho
return z)
R.unprotect (Vector.length v)
return x
evalEnv x rho = acquireSome =<< do
evalEnv x rho = acquire =<< do
io $ alloca $ \p -> R.withProtected (return (R.release x)) $ \_ -> do
v <- R.tryEvalSilent x rho p
e <- peek p
when (e /= 0) $ runRegion $ throwR rho
return v

-- | Evaluate a (sequence of) expression(s) in the global environment.
eval :: MonadR m => SEXP s a -> m (SomeSEXP (Region m))
eval :: MonadR m => SEXP s -> m (SEXP (Region m))
eval x = evalEnv x (R.release globalEnv)

-- | Silent version of 'eval' function that discards it's result.
eval_ :: MonadR m => SEXP s a -> m ()
eval_ :: MonadR m => SEXP s -> m ()
eval_ = void . eval

-- | Throw an R error as an exception.
throwR :: MonadR m => R.SEXP s 'R.Env -- ^ Environment in which to find error.
{-@ throwR :: TSEXP s Foreign.R.Type.Env -> m a @-}
throwR :: MonadR m => R.SEXP s -- ^ Environment in which to find error.
-> m a
throwR env = getErrorMessage env >>= io . throwIO . R.RError

Expand All @@ -173,12 +176,13 @@ throwRMessage :: MonadR m => String -> m a
throwRMessage = io . throwIO . R.RError

-- | Read last error message.
getErrorMessage :: MonadR m => R.SEXP s 'R.Env -> m String
{-@ getErrorMessage :: TSEXP s Foreign.R.Type.Env -> m String @-}
getErrorMessage :: MonadR m => R.SEXP s -> m String
getErrorMessage e = io $ do
R.withProtected (withCString "geterrmessage" ((R.install >=> R.lang1))) $ \f -> do
R.withProtected (return (R.release e)) $ \env -> do
peekCString
=<< R.char
=<< peek
=<< R.string . R.cast (sing :: R.SSEXPTYPE 'R.String)
=<< R.string . checkSEXPTYPE R.String
=<< R.eval f env
7 changes: 4 additions & 3 deletions inline-r/src/Language/R/Internal.hs
Original file line number Diff line number Diff line change
Expand Up @@ -19,17 +19,18 @@ inVoid = id
{-# INLINE inVoid #-}

-- | Call a pure unary R function of the given name in the global environment.
r1 :: ByteString -> SEXP s a -> IO (SomeSEXP V)
r1 :: ByteString -> SEXP s -> IO (SEXP V)
r1 fn a =
useAsCString fn $ \cfn -> R.install cfn >>= \f ->
R.withProtected (R.lang2 f (R.release a)) (unsafeRunRegion . inVoid . eval)

-- | Call a pure binary R function. See 'r1' for additional comments.
r2 :: ByteString -> SEXP s a -> SEXP s b -> IO (SomeSEXP V)
r2 :: ByteString -> SEXP s -> SEXP s -> IO (SEXP V)
r2 fn a b =
useAsCString fn $ \cfn -> R.install cfn >>= \f ->
R.withProtected (R.lang3 f (R.release a) (R.release b)) (unsafeRunRegion . inVoid . eval)

-- | Internalize a symbol name.
installIO :: String -> IO (SEXP V 'R.Symbol)
{-@ installIO :: String -> IO (TSEXP V Foreign.R.Type.Symbol) @-}
installIO :: String -> IO (SEXP V)
installIO str = withCString str R.install
9 changes: 4 additions & 5 deletions inline-r/src/Language/R/Internal.hs-boot
Original file line number Diff line number Diff line change
Expand Up @@ -4,9 +4,8 @@ module Language.R.Internal where

import Control.Memory.Region
import Data.ByteString (ByteString)
import Foreign.R (SEXP, SomeSEXP(..))
import qualified Foreign.R.Type as R
import Foreign.R (SEXP)

r1 :: ByteString -> SEXP s a -> IO (SomeSEXP V)
r2 :: ByteString -> SEXP s a -> SEXP s b -> IO (SomeSEXP V)
installIO :: String -> IO (SEXP V 'R.Symbol)
r1 :: ByteString -> SEXP s -> IO (SEXP V)
r2 :: ByteString -> SEXP s -> SEXP s -> IO (SEXP V)
installIO :: String -> IO (SEXP V)
Loading

0 comments on commit aa5333b

Please sign in to comment.