mirror of
https://github.com/anoma/juvix.git
synced 2025-01-05 22:46:08 +03:00
3b0cde27bb
* Remove input file fields from command opts * [cli] Make version and help commands * Fix on reviews * Fixes for dealing with global options inside subcmds * Fix minijuvix emacs mode and add some instance to GlobalOpts * Remove unrelated code * Propagate globals opts in each cmd parser * Add initial shell tests * Add test-shell to makefile and CI * Fix CI: adding .local/bin to PATH * Fixing CI * Installing shelltest just before running it * Install app for shell testing * Hide global flags after cmd. Fix shell tests accordingly. * Fixing CI * Shell test only run on ubuntu for now
178 lines
5.4 KiB
Haskell
178 lines
5.4 KiB
Haskell
module Commands.Compile where
|
|
|
|
import Data.ByteString qualified as BS
|
|
import Data.FileEmbed qualified as FE
|
|
import Data.Text.IO qualified as TIO
|
|
import MiniJuvix.Prelude hiding (Doc)
|
|
import Options.Applicative
|
|
import System.Environment
|
|
import System.Process qualified as P
|
|
|
|
minijuvixBuildDir :: FilePath
|
|
minijuvixBuildDir = ".minijuvix-build"
|
|
|
|
data CompileTarget = TargetC | TargetWasm
|
|
deriving stock (Show)
|
|
|
|
data CompileRuntime = RuntimeStandalone | RuntimeLibC
|
|
deriving stock (Show)
|
|
|
|
data CompileOptions = CompileOptions
|
|
{ _compileTarget :: CompileTarget,
|
|
_compileRuntime :: CompileRuntime,
|
|
_compileOutputFile :: Maybe FilePath
|
|
}
|
|
|
|
makeLenses ''CompileOptions
|
|
|
|
parseCompile :: Parser CompileOptions
|
|
parseCompile = do
|
|
_compileTarget <-
|
|
option
|
|
(eitherReader parseTarget)
|
|
( long "target"
|
|
<> short 't'
|
|
<> metavar "TARGET"
|
|
<> value TargetWasm
|
|
<> showDefaultWith targetShow
|
|
<> help "select a target: wasm, c"
|
|
)
|
|
|
|
_compileRuntime <-
|
|
option
|
|
(eitherReader parseRuntime)
|
|
( long "runtime"
|
|
<> short 'r'
|
|
<> metavar "RUNTIME"
|
|
<> value RuntimeStandalone
|
|
<> showDefaultWith runtimeShow
|
|
<> help "select a runtime: standalone, libc"
|
|
)
|
|
|
|
_compileOutputFile <-
|
|
optional $
|
|
option
|
|
str
|
|
( long "output"
|
|
<> short 'o'
|
|
<> metavar "OUTPUT_FILE"
|
|
<> help "Path to output file"
|
|
<> action "file"
|
|
)
|
|
pure CompileOptions {..}
|
|
where
|
|
parseTarget :: String -> Either String CompileTarget
|
|
parseTarget = \case
|
|
"wasm" -> Right TargetWasm
|
|
"c" -> Right TargetC
|
|
s -> Left $ "unrecognised target: " <> s
|
|
|
|
targetShow :: CompileTarget -> String
|
|
targetShow = \case
|
|
TargetC -> "c"
|
|
TargetWasm -> "wasm"
|
|
|
|
parseRuntime :: String -> Either String CompileRuntime
|
|
parseRuntime = \case
|
|
"standalone" -> Right RuntimeStandalone
|
|
"libc" -> Right RuntimeLibC
|
|
s -> Left $ "unrecognised runtime: " <> s
|
|
|
|
runtimeShow :: CompileRuntime -> String
|
|
runtimeShow = \case
|
|
RuntimeStandalone -> "standalone"
|
|
RuntimeLibC -> "libc"
|
|
|
|
inputCFile :: FilePath -> FilePath -> FilePath
|
|
inputCFile projRoot compileInputFile =
|
|
projRoot </> minijuvixBuildDir </> outputMiniCFile
|
|
where
|
|
outputMiniCFile :: FilePath
|
|
outputMiniCFile = takeBaseName compileInputFile <> ".c"
|
|
|
|
runCompile :: FilePath -> FilePath -> CompileOptions -> Text -> IO (Either Text ())
|
|
runCompile projRoot compileInputFile o minic = do
|
|
createDirectoryIfMissing True (projRoot </> minijuvixBuildDir)
|
|
TIO.writeFile (inputCFile projRoot compileInputFile) minic
|
|
prepareRuntime projRoot o
|
|
case o ^. compileTarget of
|
|
TargetWasm -> clangCompile projRoot compileInputFile o
|
|
TargetC -> return (Right ())
|
|
|
|
prepareRuntime :: FilePath -> CompileOptions -> IO ()
|
|
prepareRuntime projRoot o = do
|
|
mapM_ writeRuntime runtimeProjectDir
|
|
where
|
|
standaloneRuntimeDir :: [(FilePath, BS.ByteString)]
|
|
standaloneRuntimeDir = $(FE.makeRelativeToProject "minic-runtime/standalone" >>= FE.embedDir)
|
|
|
|
libCRuntimeDir :: [(FilePath, BS.ByteString)]
|
|
libCRuntimeDir = $(FE.makeRelativeToProject "minic-runtime/libc" >>= FE.embedDir)
|
|
|
|
runtimeProjectDir :: [(FilePath, BS.ByteString)]
|
|
runtimeProjectDir = case o ^. compileRuntime of
|
|
RuntimeStandalone -> standaloneRuntimeDir
|
|
RuntimeLibC -> libCRuntimeDir
|
|
|
|
writeRuntime :: (FilePath, BS.ByteString) -> IO ()
|
|
writeRuntime (filePath, contents) =
|
|
BS.writeFile (projRoot </> minijuvixBuildDir </> takeFileName filePath) contents
|
|
|
|
clangCompile :: FilePath -> FilePath -> CompileOptions -> IO (Either Text ())
|
|
clangCompile projRoot compileInputFile o = do
|
|
v <- sysrootEnvVar
|
|
case v of
|
|
Left s -> return (Left s)
|
|
Right sysrootPath -> withSysrootPath sysrootPath
|
|
where
|
|
sysrootEnvVar :: IO (Either Text String)
|
|
sysrootEnvVar =
|
|
maybeToEither "Missing environment variable WASI_SYSROOT_PATH"
|
|
<$> lookupEnv "WASI_SYSROOT_PATH"
|
|
|
|
withSysrootPath :: String -> IO (Either Text ())
|
|
withSysrootPath sysrootPath = runClang clangArgs
|
|
where
|
|
clangArgs :: [String]
|
|
clangArgs = case o ^. compileRuntime of
|
|
RuntimeStandalone -> standaloneArgs sysrootPath outputFile inputFile
|
|
RuntimeLibC -> libcArgs sysrootPath outputFile inputFile
|
|
|
|
outputFile :: FilePath
|
|
outputFile = fromMaybe (takeBaseName compileInputFile <> ".wasm") (o ^. compileOutputFile)
|
|
|
|
inputFile :: FilePath
|
|
inputFile = inputCFile projRoot compileInputFile
|
|
|
|
standaloneArgs :: FilePath -> FilePath -> FilePath -> [String]
|
|
standaloneArgs sysrootPath wasmOutputFile inputFile =
|
|
commonArgs sysrootPath wasmOutputFile
|
|
<> [minijuvixBuildDir </> "walloc.c", inputFile]
|
|
|
|
libcArgs :: FilePath -> FilePath -> FilePath -> [String]
|
|
libcArgs sysrootPath wasmOutputFile inputFile =
|
|
commonArgs sysrootPath wasmOutputFile
|
|
<> ["-lc", inputFile]
|
|
|
|
commonArgs :: FilePath -> FilePath -> [String]
|
|
commonArgs sysrootPath wasmOutputFile =
|
|
[ "-nodefaultlibs",
|
|
"-Oz",
|
|
"-I",
|
|
minijuvixBuildDir,
|
|
"--target=wasm32-wasi",
|
|
"--sysroot",
|
|
sysrootPath,
|
|
"-o",
|
|
wasmOutputFile
|
|
]
|
|
|
|
runClang ::
|
|
[String] ->
|
|
IO (Either Text ())
|
|
runClang args = do
|
|
(exitCode, _, err) <- P.readProcessWithExitCode "clang" args ""
|
|
case exitCode of
|
|
ExitSuccess -> return (Right ())
|
|
_ -> return (Left (pack err))
|