{-# LANGUAGE CPP #-} import Data.Maybe(fromJust, isJust, fromMaybe, maybeToList) import Distribution.Simple.Compiler import Distribution.Simple.LocalBuildInfo import Distribution.PackageDescription import Distribution.Simple import Distribution.System import Distribution.Simple.Setup import Distribution.Simple.Utils import Distribution.Verbosity import Distribution.Text ( display ) import Control.Monad import Data.List import Distribution.Simple.Program ( Program(..), ConfiguredProgram(..), programPath , requireProgram, requireProgramVersion , rawSystemProgramConf, rawSystemProgram , greencardProgram, cpphsProgram, hsc2hsProgram, c2hsProgram , happyProgram, alexProgram, ghcProgram, gccProgram, requireProgram, arProgram) import Distribution.Simple.Program.Db import Distribution.Simple.PreProcess import Distribution.Simple.Register ( generateRegistrationInfo, registerPackage ) import System.IO.Unsafe (unsafePerformIO) import System.IO.Error import qualified Distribution.Simple.Program.Ar as Ar import qualified Distribution.ModuleName as ModuleName import Distribution.Simple.BuildPaths import System.Directory(getCurrentDirectory, getDirectoryContents, withCurrentDirectory, doesDirectoryExist) import System.FilePath ( (), (<.>), takeExtension, combine, takeBaseName, takeDirectory) import qualified Distribution.Simple.GHC as GHC import qualified Distribution.Simple.JHC as JHC import qualified Distribution.Simple.LHC as LHC import qualified Distribution.Simple.UHC as UHC import qualified Distribution.Simple.PackageIndex as PackageIndex import Distribution.PackageDescription as PD import Distribution.InstalledPackageInfo (ldOptions, extraGHCiLibraries, showInstalledPackageInfo, libraryDynDirs, libraryDirs) import System.Environment (getEnv, setEnv) main :: IO () main = defaultMainWithHooks autoconfUserHooks { preConf = myPreConf, buildHook = myBuildHook, cleanHook = myCleanHook, copyHook = copyCBindingsAndBundledExecutables, regHook = registerHook } fltkSource = "fltk-1.3.4-1" runMake :: IO () runMake = case buildOS of Windows -> rawSystemExit normal "make" [] os | os `elem` [FreeBSD, OpenBSD, NetBSD, DragonFly] -> rawSystemExit normal "gmake" [] _ -> rawSystemExit normal "make" [] buildFltk :: IO () buildFltk = do rawSystemExit normal "tar" ["-zxf", fltkSource ++ "-source.tar.gz"] projectRoot <- getCurrentDirectory let fltkDir = projectRoot fltkSource fltkFlags = [ "--enable-gl", "--enable-shared", "--enable-localjpeg", "--enable-localzlib", "--enable-localpng"] withCurrentDirectory fltkDir ( case buildOS of OSX -> do rawSystemExit normal (fltkDir "configure") fltkFlags runMake updateEnv "DYLD_LIBRARY_PATH" fltkDir updateEnv "DYLD_LIBRARY_PATH" (fltkDir "lib") updateEnv "DYLD_LIBRARY_PATH" (fltkDir "src") updateEnv "LIBRARY_PATH" (fltkDir "lib") updateEnv "PATH" (fltkDir "fluid") updateEnv "PATH" fltkDir Windows -> do rawSystemExit normal "sh" ([(fltkDir "configure")] ++ fltkFlags) runMake let path = windowsFriendlyPaths fltkDir updateEnv "INCLUDE_PATH" path updateEnv "LD_LIBRARY_PATH" (path "lib") updateEnv "LD_LIBRARY_PATH" (path "src") updateEnv "LIBRARY_PATH" (path "lib") updateEnv "PATH" (path "fluid") updateEnv "PATH" path _ -> do rawSystemExit normal (fltkDir "configure") fltkFlags runMake updateEnv "INCLUDE_PATH" fltkDir updateEnv "LD_LIBRARY_PATH" (fltkDir "lib") updateEnv "LD_LIBRARY_PATH" (fltkDir "src") updateEnv "LIBRARY_PATH" (fltkDir "lib") updateEnv "PATH" (fltkDir "fluid") updateEnv "PATH" fltkDir (getEnv "PATH") >>= putStrLn ) myPreConf :: Args -> ConfigFlags -> IO HookedBuildInfo myPreConf args flags = do if (flagIsSet (FlagName "bundled") flags) then putStrLn "Building bundled FLTK" >> buildFltk else return () putStrLn "Running autoconf ..." case buildOS of Windows -> rawSystemExit normal "sh" ["autoconf"] _ -> rawSystemExit normal "autoconf" [] preConf autoconfUserHooks args flags fltkcdir :: FilePath fltkcdir = unsafePerformIO $ do d <- getCurrentDirectory return (d "c-lib") fltkclib = "fltkc" flagIsSet :: PD.FlagName -> ConfigFlags -> Bool flagIsSet flag configFlags = maybe False id (lookup flag (configConfigurationsFlags configFlags)) addFltkcDir :: PackageDescription -> PackageDescription addFltkcDir pkg_descr = pkg_descr { library = fmap (\l' -> l' { libBuildInfo = (libBuildInfo l') { extraLibDirs = (extraLibDirs (libBuildInfo l')) ++ [fltkcdir] } } ) (library pkg_descr) } updateEnv :: String -> String -> IO () updateEnv env value = do old <- tryIOError (getEnv env) setEnv env ((either (const value) (\old' -> value ++ (case buildOS of Windows -> ";" _ -> ":") ++ old' ) old)) replaceAllInfixes :: String -> String -> String -> String replaceAllInfixes needle subString haystack = concat (unfoldr (\str -> if (null str) then Nothing else if (needle `isPrefixOf` str) then Just (subString, drop (length needle) str) else Just ([head str], tail str)) haystack) cygpath o p = let removeTrailingNewline = head . lines in removeTrailingNewline<$>(rawSystemStdout normal "cygpath" [o, p]) windowsFriendlyPaths s = if (isPrefixOf "/" s) then (unsafePerformIO (cygpath "-m" "/")) ++ (drop 1 s) else s myBuildHook pkg_descr local_bld_info user_hooks bld_flags = do case buildOS of Windows -> if (flagIsSet (FlagName "bundled") (configFlags local_bld_info)) then do projectRoot <- getCurrentDirectory let fltkDir = projectRoot fltkSource updateEnv "PATH" (windowsFriendlyPaths fltkDir) else return () _ -> return () let compileC = putStrLn "==Compiling C bindings==" >> runMake cdirexists <- doesDirectoryExist fltkcdir if cdirexists then do clibraries <- getDirectoryContents fltkcdir when (null $ filter (Data.List.isInfixOf "fltkc") clibraries) compileC else compileC case buildOS of Windows -> let rewritePaths :: PackageDescription -> IO PackageDescription rewritePaths pkg_descr = let lib = library pkg_descr ld_incdirs :: Maybe ([String], [String]) ld_incdirs = fmap (\lib' -> (PD.ldOptions (libBuildInfo lib'), includeDirs (libBuildInfo lib'))) lib stdCppGccPthreadPaths ghcPath = [ ((takeDirectory . takeDirectory) ghcPath) "mingw" "lib" "gcc" "x86_64-w64-mingw32" "5.2.0" , ((takeDirectory . takeDirectory) ghcPath) "mingw" "x86_64-w64-mingw32" "lib" ] in do ghcPath <- rawSystemStdout normal "sh" ["-c", "which ghc"] cppGccPthreadPaths <- mapM (cygpath "-w") (stdCppGccPthreadPaths ghcPath) let fixedLib = maybe Nothing (\(ld, incDirs) -> fmap (\l' -> l' { libBuildInfo = (libBuildInfo l') { PD.ldOptions = fmap windowsFriendlyPaths ld, includeDirs = fmap windowsFriendlyPaths incDirs, extraLibDirs = cppGccPthreadPaths ++ (extraLibDirs (libBuildInfo l')), extraLibs = ["stdc++", "gcc", "pthread"] } }) lib ) ld_incdirs return (pkg_descr {library = fixedLib}) in do fixedPkgDescr <- rewritePaths pkg_descr >>= return . addFltkcDir buildHook autoconfUserHooks fixedPkgDescr local_bld_info user_hooks bld_flags Linux -> do updateEnv "LIBRARY_PATH" fltkcdir buildHook autoconfUserHooks pkg_descr local_bld_info user_hooks bld_flags _ -> do updateEnv "DYLD_LIBRARY_PATH" fltkcdir updateEnv "LIBRARY_PATH" fltkcdir buildHook autoconfUserHooks pkg_descr local_bld_info user_hooks bld_flags copyCBindingsAndBundledExecutables :: PackageDescription -> LocalBuildInfo -> UserHooks -> CopyFlags -> IO () copyCBindingsAndBundledExecutables pkg_descr lbi uhs flags = do (copyHook autoconfUserHooks) pkg_descr lbi uhs flags let installDirs = absoluteInstallDirs pkg_descr lbi . fromFlag . copyDest $ flags libPref = libdir installDirs rawSystemExit (fromFlag $ copyVerbosity flags) "cp" [fltkcdir "libfltkc.a" , libPref] case buildOS of OSX -> rawSystemExit (fromFlag $ copyVerbosity flags) "cp" ["c-lib" "libfltkc-dyn.dylib", libPref] _ -> rawSystemExit (fromFlag $ copyVerbosity flags) "cp" ["c-lib" "libfltkc-dyn.so", libPref] if (flagIsSet (FlagName "bundled") (configFlags lbi)) then do projectRoot <- getCurrentDirectory let binPref = bindir installDirs let fltkDir = projectRoot fltkSource rawSystemExit (fromFlag $ copyVerbosity flags) "cp" [(fltkDir "fluid" "fluid"), binPref] rawSystemExit (fromFlag $ copyVerbosity flags) "cp" [(fltkDir "fltk-config"), binPref] else return () myCleanHook pd x uh cf = do case buildOS of os | os `elem` [FreeBSD, OpenBSD, NetBSD, DragonFly] -> rawSystemExit normal "gmake" ["clean"] _ -> rawSystemExit normal "make" ["clean"] cleanHook autoconfUserHooks pd x uh cf -- Based on code in "Gtk2HsSetup.hs" from "gtk" package registerHook pkg_descr localbuildinfo _ flags = if hasLibs pkg_descr then register pkg_descr localbuildinfo flags else setupMessage verbosity "Package contains no library to register:" (packageId pkg_descr) where verbosity = fromFlag (regVerbosity flags) register :: PackageDescription -> LocalBuildInfo -> RegisterFlags -> IO () register pkg@PackageDescription { library = Just lib } lbi regFlags = do let clbi = getComponentLocalBuildInfo lbi CLibName installedPkgInfoRaw' <- generateRegistrationInfo verbosity pkg lib lbi clbi inplace False distPref packageDb let installedPkgInfoRaw = case buildOS of Windows -> installedPkgInfoRaw' { Distribution.InstalledPackageInfo.ldOptions = fmap windowsFriendlyPaths (Distribution.InstalledPackageInfo.ldOptions installedPkgInfoRaw') } _ -> installedPkgInfoRaw' let installedPkgInfo = installedPkgInfoRaw { libraryDynDirs = (libraryDynDirs installedPkgInfoRaw) ++ (libraryDirs installedPkgInfoRaw), -- this is what this whole register code is all about extraGHCiLibraries = case buildOS of Windows -> ["libfltkc-dyn"] _ -> ["fltkc-dyn"] } -- Three different modes: case () of _ | modeGenerateRegFile -> writeRegistrationFile installedPkgInfo | modeGenerateRegScript -> die "Generate Reg Script not supported" | otherwise -> registerPackage verbosity (compiler lbi) (withPrograms lbi) False {- multiinstance -} packageDbs installedPkgInfo where modeGenerateRegFile = isJust (flagToMaybe (regGenPkgConf regFlags)) regFile = fromMaybe (display (packageId pkg) <.> "conf") (fromFlag (regGenPkgConf regFlags)) modeGenerateRegScript = fromFlag (regGenScript regFlags) inplace = fromFlag (regInPlace regFlags) packageDbs = nub $ withPackageDB lbi ++ maybeToList (flagToMaybe (regPackageDB regFlags)) packageDb = registrationPackageDB packageDbs distPref = fromFlag (regDistPref regFlags) verbosity = fromFlag (regVerbosity regFlags) writeRegistrationFile installedPkgInfo = do notice verbosity ("Creating package registration file: " ++ regFile) writeUTF8File regFile (showInstalledPackageInfo installedPkgInfo)