mirror of
https://github.com/GaloisInc/cryptol.git
synced 2025-01-05 15:07:12 +03:00
Merge branch 'writeFile' of github.com:TomMD/cryptol into tommd/writeFile
This commit is contained in:
commit
0d18bc821c
@ -44,10 +44,12 @@ flag server
|
||||
library
|
||||
Default-language:
|
||||
Haskell98
|
||||
|
||||
Build-depends: base >= 4.7 && < 5,
|
||||
base-compat >= 0.6,
|
||||
array >= 0.4,
|
||||
async >= 2.0,
|
||||
bytestring >= 0.10,
|
||||
containers >= 0.5,
|
||||
deepseq >= 1.3,
|
||||
deepseq-generics >= 0.1,
|
||||
|
@ -194,6 +194,7 @@ cmdArgument ct cursor@(l,_) = case ct of
|
||||
ShellArg _ -> completeFilename cursor
|
||||
OptionArg _ -> completeOption cursor
|
||||
NoArg _ -> return (l,[])
|
||||
FileExprArg _ -> completeExpr cursor
|
||||
|
||||
-- | Complete a name from the expression environment.
|
||||
completeExpr :: CompletionFunc REPL
|
||||
|
@ -75,6 +75,8 @@ import qualified Cryptol.Symbolic as Symbolic
|
||||
import Control.DeepSeq
|
||||
import qualified Control.Exception as X
|
||||
import Control.Monad (guard,unless,forM_,when)
|
||||
import qualified Data.ByteString as BS
|
||||
import Data.Bits ((.&.))
|
||||
import Data.Char (isSpace,isPunctuation,isSymbol)
|
||||
import Data.Function (on)
|
||||
import Data.List (intercalate,nub,sortBy,partition)
|
||||
@ -124,6 +126,7 @@ instance Ord CommandDescr where
|
||||
|
||||
data CommandBody
|
||||
= ExprArg (String -> REPL ())
|
||||
| FileExprArg (FilePath -> String -> REPL ())
|
||||
| DeclsArg (String -> REPL ())
|
||||
| ExprTypeArg (String -> REPL ())
|
||||
| FilenameArg (FilePath -> REPL ())
|
||||
@ -186,6 +189,10 @@ commandList =
|
||||
"set the current working directory"
|
||||
, CommandDescr [ ":m", ":module" ] (FilenameArg moduleCmd)
|
||||
"load a module"
|
||||
, CommandDescr [ ":w", ":write" ] (FileExprArg writeFileCmd)
|
||||
"write data of type `fin n => [n][8]` to a file"
|
||||
, CommandDescr [ ":read" ] (FilenameArg readFileCmd)
|
||||
"read data from a file as type `fin n => [n][8]`, binding the value to variable `it`"
|
||||
]
|
||||
|
||||
genHelp :: [CommandDescr] -> [String]
|
||||
@ -533,6 +540,31 @@ typeOfCmd str = do
|
||||
(_,_,names) <- getFocusedEnv
|
||||
rPrint $ runDoc names $ pp re <+> text ":" <+> pp sig
|
||||
|
||||
readFileCmd :: FilePath -> REPL ()
|
||||
readFileCmd fp = do
|
||||
bytes <- replReadFile fp (\err -> rPutStrLn (show err) >> return Nothing)
|
||||
case bytes of
|
||||
Nothing -> return ()
|
||||
Just bs ->
|
||||
do let expr = T.eString (map (toEnum . fromIntegral) (BS.unpack bs))
|
||||
ty = T.tString (BS.length bs)
|
||||
bindItVariable ty expr
|
||||
|
||||
writeFileCmd :: FilePath -> String -> REPL ()
|
||||
writeFileCmd file str = do
|
||||
expr <- replParseExpr str
|
||||
(val,ty) <- replEvalExpr expr
|
||||
if not (tIsByteSeq ty)
|
||||
then rPrint $ text "Can not write expression of types other than [n][8]. Type was: " <+> pp ty
|
||||
else wf file =<< serializeValue val
|
||||
where
|
||||
wf fp bytes = replWriteFile fp bytes (rPutStrLn . show)
|
||||
tIsByteSeq = maybe False (tIsByte . snd) . T.tIsSeq
|
||||
tIsByte x = maybe False (\(n,b) -> T.tIsBit b && T.tIsNum n == Just 8) (T.tIsSeq x)
|
||||
serializeValue (E.VSeq _ vs) = return $ BS.pack $ map (serializeByte . E.fromVWord) vs
|
||||
serializeValue _ = panic "REPL write" ["Impossible: Non-VSeq value of type [n][8]."]
|
||||
serializeByte (E.BV _ v) = fromIntegral (v .&. 0xFF)
|
||||
|
||||
reloadCmd :: REPL ()
|
||||
reloadCmd = do
|
||||
mb <- getLoadedMod
|
||||
@ -857,10 +889,19 @@ replEvalExpr expr =
|
||||
warnDefault ns (x,t) =
|
||||
rPrint $ text "Assuming" <+> ppWithNames ns x <+> text "=" <+> pp t
|
||||
|
||||
|
||||
itIdent :: M.Ident
|
||||
itIdent = M.packIdent "it"
|
||||
|
||||
replWriteFile :: FilePath -> BS.ByteString -> (X.SomeException -> REPL ()) -> REPL ()
|
||||
replWriteFile fp bytes handler =
|
||||
do x <- io $ X.catch (BS.writeFile fp bytes >> return Nothing) (return . Just)
|
||||
maybe (return ()) handler x
|
||||
|
||||
replReadFile :: FilePath -> (X.SomeException -> REPL (Maybe BS.ByteString)) -> REPL (Maybe BS.ByteString)
|
||||
replReadFile fp handler =
|
||||
do x <- io $ X.catch (Right `fmap` BS.readFile fp) (\e -> return $ Left e)
|
||||
either handler (return . Just) x
|
||||
|
||||
-- | Creates a fresh binding of "it" to the expression given, and adds
|
||||
-- it to the current dynamic environment
|
||||
bindItVariable :: T.Type -> T.Expr -> REPL ()
|
||||
@ -974,7 +1015,10 @@ parseCommand findCmd line = do
|
||||
OptionArg body -> Just (Command (body args'))
|
||||
ShellArg body -> Just (Command (body args'))
|
||||
NoArg body -> Just (Command body)
|
||||
|
||||
FileExprArg body -> case words args' of
|
||||
(fp:_:_) -> let expr = drop (length fp + 1) args'
|
||||
in Just (Command (body fp expr))
|
||||
_ -> Nothing
|
||||
[] -> case uncons cmd of
|
||||
Just (':',_) -> Just (Unknown cmd)
|
||||
Just _ -> Just (Command (evalCmd line))
|
||||
|
Loading…
Reference in New Issue
Block a user