#!/usr/bin/runhaskell module Main where import Control.Monad import Data.Char import Data.List import Data.Maybe import qualified Distribution.InstalledPackageInfo as I import qualified Distribution.ModuleName as ModuleName import Distribution.PackageDescription import Distribution.Simple import Distribution.Simple.BuildPaths import Distribution.Simple.Compiler import Distribution.Simple.LocalBuildInfo import Distribution.Simple.Program import Distribution.Simple.Program.Ld import Distribution.Simple.Register import Distribution.Simple.Setup import Distribution.Simple.Utils import Distribution.Text import Distribution.Types.CondTree import Distribution.Types.LocalBuildInfo import Distribution.Verbosity import System.Environment import System.FilePath main :: IO () main = do -- If system uses qtchooser(1) then encourage it to choose Qt 5 env <- getEnvironment case lookup "QT_SELECT" env of Nothing -> setEnv "QT_SELECT" "5" _ -> return () -- Chain standard setup defaultMainWithHooks simpleUserHooks { confHook = confWithQt, buildHook = buildWithQt, copyHook = copyWithQt, instHook = instWithQt, regHook = regWithQt} getCustomStr :: String -> PackageDescription -> String getCustomStr name pkgDesc = fromMaybe "" $ do lib <- library pkgDesc lookup name $ customFieldsBI $ libBuildInfo lib getCustomFlag :: String -> PackageDescription -> Bool getCustomFlag name pkgDesc = fromMaybe False . simpleParse $ getCustomStr name pkgDesc xForceGHCiLib, xMocHeaders, xFrameworkDirs, xSeparateCbits :: String xForceGHCiLib = "x-force-ghci-lib" xMocHeaders = "x-moc-headers" xFrameworkDirs = "x-framework-dirs" xSeparateCbits = "x-separate-cbits" confWithQt :: (GenericPackageDescription, HookedBuildInfo) -> ConfigFlags -> IO LocalBuildInfo confWithQt (gpd,hbi) flags = do let verb = fromFlag $ configVerbosity flags mocPath <- (fmap . fmap) fst $ programFindLocation mocProgram verb defaultProgramSearchPath cppPath <- (fmap . fmap) fst $ findProgramOnSearchPath verb defaultProgramSearchPath "cpp" let mapLibBI = fmap $ mapCondTree (mapBI $ substPaths mocPath cppPath) id id gpd' = gpd { condLibrary = mapLibBI $ condLibrary gpd, condExecutables = mapAllBI mocPath cppPath $ condExecutables gpd, condTestSuites = mapAllBI mocPath cppPath $ condTestSuites gpd, condBenchmarks = mapAllBI mocPath cppPath $ condBenchmarks gpd} lbi <- confHook simpleUserHooks (gpd',hbi) flags -- Find Qt moc program and store in database (_,_,db') <- requireProgramVersion verb mocProgram qtVersionRange (withPrograms lbi) -- Force enable GHCi workaround library if flag set and not using shared libs let forceGHCiLib = (getCustomFlag xForceGHCiLib $ localPkgDescr lbi) && (not $ withSharedLib lbi) -- Update LocalBuildInfo return lbi {withPrograms = db', withGHCiLib = withGHCiLib lbi || forceGHCiLib} mapAllBI :: (HasBuildInfo a) => Maybe FilePath -> Maybe FilePath -> [(x, CondTree c v a)] -> [(x, CondTree c v a)] mapAllBI mocPath cppPath = mapSnd $ mapCondTree (mapBI $ substPaths mocPath cppPath) id id substPaths :: Maybe FilePath -> Maybe FilePath -> BuildInfo -> BuildInfo substPaths mocPath cppPath build = let escapeStr = init . tail . show toRoot = escapeStr . takeDirectory . takeDirectory . fromMaybe "" in read . replace "/QT_ROOT" (toRoot mocPath) . replace "/SYS_ROOT" (toRoot cppPath) . replace "-hide-option-" "-" $ show build buildWithQt :: PackageDescription -> LocalBuildInfo -> UserHooks -> BuildFlags -> IO () buildWithQt pkgDesc lbi hooks flags = do let verb = fromFlag $ buildVerbosity flags libs' <- maybeMapM (\lib -> fmap (\lib' -> lib {libBuildInfo = lib'}) $ fixQtBuild verb lbi $ libBuildInfo lib) $ library pkgDesc let pkgDesc' = pkgDesc {library = libs'} lbi' = if (needsGHCiFix pkgDesc lbi) then lbi {withGHCiLib = False, splitObjs = False} else lbi buildHook simpleUserHooks pkgDesc' lbi' hooks flags case libs' of Just lib -> when (needsGHCiFix pkgDesc lbi) $ buildGHCiFix verb pkgDesc lbi lib Nothing -> return () fixQtBuild :: Verbosity -> LocalBuildInfo -> BuildInfo -> IO BuildInfo fixQtBuild verb lbi build = do let moc = fromJust $ lookupProgram mocProgram $ withPrograms lbi option name = words $ fromMaybe "" $ lookup name $ customFieldsBI build incs = option xMocHeaders bDir = buildDir lbi cpps = map (\inc -> bDir (takeDirectory inc) ("moc_" ++ (takeBaseName inc) ++ ".cpp")) incs args = map ("-I"++) (includeDirs build) ++ map ("-F"++) (option xFrameworkDirs) -- Run moc on each of the header files containing QObject subclasses mapM_ (\(i,o) -> do createDirectoryIfMissingVerbose verb True (takeDirectory o) runProgram verb moc $ [i,"-o",o] ++ args) $ zip incs cpps -- Add the moc generated source files to be compiled return build {cSources = cpps ++ cSources build, ccOptions = "-fPIC" : ccOptions build} needsGHCiFix :: PackageDescription -> LocalBuildInfo -> Bool needsGHCiFix pkgDesc lbi = withGHCiLib lbi && getCustomFlag xSeparateCbits pkgDesc mkGHCiFixLibPkgId :: PackageDescription -> PackageIdentifier mkGHCiFixLibPkgId pkgDesc = let pid = packageId pkgDesc name = unPackageName $ pkgName pid in pid {pkgName = mkPackageName $ "cbits-" ++ name} mkGHCiFixLibName :: PackageDescription -> String mkGHCiFixLibName pkgDesc = ("lib" ++ display (mkGHCiFixLibPkgId pkgDesc)) <.> dllExtension mkGHCiFixLibRefName :: PackageDescription -> String mkGHCiFixLibRefName pkgDesc = prefix ++ display (mkGHCiFixLibPkgId pkgDesc) where prefix = if dllExtension == "dll" then "lib" else "" buildGHCiFix :: Verbosity -> PackageDescription -> LocalBuildInfo -> Library -> IO () buildGHCiFix verb pkgDesc lbi lib = let bDir = buildDir lbi clbis = componentNameCLBIs lbi CLibName in flip mapM_ clbis $ \clbi -> do let ms = map ModuleName.toFilePath $ allLibModules lib clbi hsObjs = map ((bDir ) . (<.> "o")) ms lname = getHSLibraryName $ componentUnitId clbi stubObjs <- fmap catMaybes $ mapM (findFileWithExtension ["o"] [bDir]) $ map (++ "_stub") ms (ld,_) <- requireProgram verb ldProgram (withPrograms lbi) combineObjectFiles verb ld (bDir lname <.> "o") (stubObjs ++ hsObjs) (ghc,_) <- requireProgram verb ghcProgram (withPrograms lbi) let bi = libBuildInfo lib runProgram verb ghc ( ["-shared","-o",bDir (mkGHCiFixLibName pkgDesc)] ++ (ldOptions bi) ++ (map ("-l" ++) $ extraLibs bi) ++ (map ("-L" ++) $ extraLibDirs bi) ++ (map ((bDir ) . flip replaceExtension objExtension) $ cSources bi)) return () mocProgram :: Program mocProgram = Program { programName = "moc", programFindLocation = \verb search -> fmap msum $ mapM (findProgramOnSearchPath verb search) ["moc-qt5", "moc"], programFindVersion = \verb path -> do (oLine,eLine,_) <- rawSystemStdInOut verb path ["-v"] Nothing Nothing Nothing False return $ msum (map (\(p,l) -> findSubseq (stripPrefix p) l) [("(Qt ",eLine), ("moc-qt5 ",oLine), ("moc ",oLine)]) >>= simpleParse . takeWhile (\c -> isDigit c || c == '.'), programPostConf = \_ c -> return c } qtVersionRange :: VersionRange qtVersionRange = intersectVersionRanges (orLaterVersion $ mkVersion [5,0]) (earlierVersion $ mkVersion [6,0]) copyWithQt :: PackageDescription -> LocalBuildInfo -> UserHooks -> CopyFlags -> IO () copyWithQt pkgDesc lbi hooks flags = do copyHook simpleUserHooks pkgDesc lbi hooks flags let verb = fromFlag $ copyVerbosity flags dest = fromFlag $ copyDest flags bDir = buildDir lbi instDirs = absoluteInstallDirs pkgDesc lbi dest file = mkGHCiFixLibName pkgDesc when (needsGHCiFix pkgDesc lbi) $ do installOrdinaryFile verb (bDir file) (dynlibdir instDirs file) -- Stack looks in the non-dyn lib directory installOrdinaryFile verb (bDir file) (libdir instDirs file) regWithQt :: PackageDescription -> LocalBuildInfo -> UserHooks -> RegisterFlags -> IO () regWithQt pkg@PackageDescription { library = Just lib } lbi _ flags = do let verb = fromFlag $ regVerbosity flags inplace = fromFlag $ regInPlace flags dist = fromFlag $ regDistPref flags reloc = relocatable lbi pkgDb = withPackageDB lbi clbis = componentNameCLBIs lbi CLibName regDb <- fmap registrationPackageDB $ absolutePackageDBPaths pkgDb flip mapM_ clbis $ \clbi -> do instPkgInfo <- generateRegistrationInfo verb pkg lib lbi clbi inplace reloc dist regDb let instPkgInfo' = instPkgInfo { -- Add extra library for GHCi workaround I.extraGHCiLibraries = (if needsGHCiFix pkg lbi then [mkGHCiFixLibRefName pkg] else []) ++ I.extraGHCiLibraries instPkgInfo, -- Add directories to framework search path I.frameworkDirs = words (getCustomStr xFrameworkDirs pkg) ++ I.frameworkDirs instPkgInfo} case flagToMaybe $ regGenPkgConf flags of Just regFile -> do writeUTF8File (fromMaybe (display (packageId pkg) <.> "conf") regFile) $ I.showInstalledPackageInfo instPkgInfo' _ | fromFlag (regGenScript flags) -> die' verb "Registration scripts are not implemented." | otherwise -> let comp = compiler lbi progs = withPrograms lbi opts = defaultRegisterOptions in registerPackage verb comp progs pkgDb instPkgInfo' opts regWithQt pkgDesc _ _ flags = setupMessage (fromFlag $ regVerbosity flags) "Package contains no library to register:" (packageId pkgDesc) instWithQt :: PackageDescription -> LocalBuildInfo -> UserHooks -> InstallFlags -> IO () instWithQt pkgDesc lbi hooks flags = do let copyFlags = defaultCopyFlags { copyDistPref = installDistPref flags, copyVerbosity = installVerbosity flags } regFlags = defaultRegisterFlags { regDistPref = installDistPref flags, regInPlace = installInPlace flags, regPackageDB = installPackageDB flags, regVerbosity = installVerbosity flags } copyWithQt pkgDesc lbi hooks copyFlags when (hasLibs pkgDesc) $ regWithQt pkgDesc lbi hooks regFlags class HasBuildInfo a where mapBI :: (BuildInfo -> BuildInfo) -> a -> a instance HasBuildInfo Library where mapBI f x = x {libBuildInfo = f $ libBuildInfo x} instance HasBuildInfo Executable where mapBI f x = x {buildInfo = f $ buildInfo x} instance HasBuildInfo TestSuite where mapBI f x = x {testBuildInfo = f $ testBuildInfo x} instance HasBuildInfo Benchmark where mapBI f x = x {benchmarkBuildInfo = f $ benchmarkBuildInfo x} maybeMapM :: (Monad m) => (a -> m b) -> (Maybe a) -> m (Maybe b) maybeMapM f = maybe (return Nothing) $ liftM Just . f mapSnd :: (a -> a) -> [(x, a)] -> [(x, a)] mapSnd f = map (\(x,y) -> (x,f y)) findSubseq :: ([a] -> Maybe b) -> [a] -> Maybe b findSubseq f [] = f [] findSubseq f xs@(_:ys) = case f xs of Nothing -> findSubseq f ys Just r -> Just r replace :: (Eq a) => [a] -> [a] -> [a] -> [a] replace [] _ xs = xs replace _ _ [] = [] replace src dst xs@(x:xs') = case stripPrefix src xs of Just xs'' -> dst ++ replace src dst xs'' Nothing -> x : replace src dst xs'