使用GHC API评估Haskell语句/表达式

对于我正在编写的工具(http://hackage.haskell.org/package/explore),我需要一种在运行时读取haskell函数定义的方法,将它们应用于我的工具中的值并检索其应用程序的结果。

任何人都可以给我一个使用GHC(6.10.4或6.12.1)API的非常基本的例子吗?

在运行时从文件中读取的示例函数定义:

f x = 10**(4/1102*x - 1)

预期的节目输出

--mapM_ print $ map f [428, 410, 389]
3.577165388142748
3.077536885227335
2.5821307011665815

!! UPDATE!
我发布了一个快速答案,但它在执行目录中创建了一个目标文件,任何提示以避免这种情况并避免所有文件IO是最受欢迎的。 我还想看到一个在内存中执行所有操作的版本:例如,用户在GUI中提供函数定义,编译/评估不会创建任何目标文件。


使用提示。 这是GHC API的一个类GHCi封装,不是很难使用。

如果你想要一个使用它的例子,我在我的酸奶项目中使用它。


改编自:http://www.bluishcoder.co.nz/2008/11/dynamic-compilation-and-loading-of.html

f.hs:

module Func (Func.f) where

f :: Double -> Double
f x = 10**(4/1102*x - 1)

main.hs:

import GHC
import GHC.Paths
import DynFlags
import Unsafe.Coerce

import Control.Monad

main :: IO ()
main =
    defaultErrorHandler defaultDynFlags $ do
      func <- runGhc (Just libdir) $ do
        dflags <- getSessionDynFlags
        setSessionDynFlags dflags
        target <- guessTarget "f.hs" Nothing
        addTarget target
        r <- load LoadAllTargets
        case r of
          Failed -> error "Compilation failed"
          Succeeded -> do
            m <- findModule (mkModuleName "Func") Nothing
            setContext [] [m]
            value <- compileExpr ("Func.f")
            do let value' = (unsafeCoerce value) :: Double -> Double
               return value'
      let f = func
      mapM_ print $ map f [428, 410, 389]
      return ()

获得API的好工作。 我可以告诉你一些关于代码生成器如何工作的信息。

GHC使用系统汇编程序来创建.o文件。 如果没有可供GHC自行清理的选项,那么您应该使用http://hackage.haskell.org/trac/ghc/newticket?type=上的错误跟踪器针对API提交功能请求。功能+请求。 为了提交申请,您需要注册一个帐户。

使用标准代码生成器,您将无法完全避免文件I / O,这仅仅是因为GHC将创建可重定位目标代码的工作委托给汇编程序。 有一个基于LLVM的实验性后端,可能能够完成内存中的所有操作,但如果它在6.13之前的任何版本中都可用,我会感到惊讶。 不过值得一提的是GHC开发者的名单。

链接地址: http://www.djcxy.com/p/43417.html

上一篇: Evaluation of Haskell Statements/Expressions using GHC API

下一篇: Profiling a partially evaluated program