12

我正在尝试使用 GHC API 动态编译和加载 Haskell 模块。我知道 API 从一个版本到另一个版本波动很大,所以我专门谈论 GHC 7.6.*。

我尝试在 MacOS 和 Linux 上运行相同的代码。在这两种情况下,插件模块都可以正常编译,但在加载时会出现以下错误:Cannot add module Plugin to context: not interpreted

该问题与此问题类似,其中模块仅在主机程序的同一运行中编译时才会加载。

-- Host.hs: compile with ghc-7.6.*
-- $ ghc -package ghc -package ghc-paths Host.hs
-- Needs Plugin.hs in the same directory.
module Main where

import GHC
import GHC.Paths ( libdir )
import DynFlags
import Unsafe.Coerce

main :: IO ()
main =
    defaultErrorHandler defaultFatalMessager defaultFlushOut $ do
      result <- runGhc (Just libdir) $ do
        dflags <- getSessionDynFlags
        setSessionDynFlags dflags
        target <- guessTarget "Plugin.hs" Nothing
        setTargets [target]
        r <- load LoadAllTargets
        case r of
          Failed    -> error "Compilation failed"
          Succeeded -> do
            setContext [IIModule (mkModuleName "Plugin")]
            result <- compileExpr ("Plugin.getInt")
            let result' = unsafeCoerce result :: Int
            return result'
      print result

和插件:

-- Plugin.hs
module Plugin where

getInt :: Int
getInt = 33
4

1 回答 1

14

问题是您正在使用IIModule. 这表明您希望将模块及其中的所有内容(包括未导出的内容)带入上下文。:load它与 GHCi 中的星号基本相同。正如您所注意到的,这仅适用于解释代码,因为它让您“查看”模块。

但这不是你在这里需要的。您想要的是像使用:moduleimport声明一样加载它,它适用于已编译的模块。为此,您使用IIDeclwhich 可以使用以下方式进行导入声明simpleImportDecl

setContext [IIDecl $ simpleImportDecl (mkModuleName "Plugin")]
于 2013-05-29T06:15:39.710 回答