如何为名称中包含非字母字符的库定义MIN_VERSION_宏?

lef*_*out 5 versioning haskell ghc cabal c-preprocessor

Cabal或最近的GHC本身预先定义了CPP宏,允许检查使用哪个库的版本.它很容易在简单的例子中使用

#if MIN_VERSION_base(4,0,0)
... code that works with base-4 ...
#else
... code that works with base-3 ...
#endif
Run Code Online (Sandbox Code Playgroud)

但是,CPP宏名称比Cabal包名称更受限制,因此以下不起作用:

#if !MIN_VERSION_quickcheck-instances(0,3,17)
instance Arbitrary SBS.ShortByteString where arbitrary = fmap SBS.pack arbitrary
#endif
Run Code Online (Sandbox Code Playgroud)
     error: missing binary operator before token "("
     #if !MIN_VERSION_quickcheck-instances(0,3,17)
     ^
   |
18 | #if !MIN_VERSION_quickcheck-instances(0,3,17)
   | ^
`gcc' failed in phase `C pre-processor'. (Exit code: 1)

似乎没有正确记录如何处理这样的库名称.

Wil*_*sem 5

简短回答:连字符 ( -) 替换为下划线 ( _)。

据我所知,宏是在文件generatePackageVersionMacros的函数DriverPipeline.hs中生成的:

generatePackageVersionMacros :: [PackageConfig] -> String
generatePackageVersionMacros pkgs = concat
  -- Do not add any C-style comments. See Trac #3389.
  [ generateMacros "" pkgname version
  | pkg <- pkgs
  , let version = packageVersion pkg
        pkgname = map fixchar (packageNameString pkg)
  ]

fixchar :: Char -> Char
fixchar '-' = '_'
fixchar c = c

generateMacros :: String -> String -> Version -> String
generateMacros prefix name version =
  concat
  ["#define ", prefix, "VERSION_",name," ",show (showVersion version),"\n"
  ,"#define MIN_", prefix, "VERSION_",name,"(major1,major2,minor) (\\\n"
  ,"  (major1) <  ",major1," || \\\n"
  ,"  (major1) == ",major1," && (major2) <  ",major2," || \\\n"
  ,"  (major1) == ",major1," && (major2) == ",major2," && (minor) <= ",minor,")"
  ,"\n\n"
  ]
  where
(major1:major2:minor:_) = map show (versionBranch version ++ repeat 0)
Run Code Online (Sandbox Code Playgroud)

正如您所看到的,通过对其packageNameString执行 a 来修复包的。map fixchar连字符 ( -) 替换为下划线 ( _)。因此,该宏可以写成:

#if !MIN_VERSION_quickcheck_instances(0,3,17)
instance Arbitrary SBS.ShortByteString where arbitrary = fmap SBS.pack arbitrary
#endif
Run Code Online (Sandbox Code Playgroud)