----------------------------------------------------------------------------- -- | -- Module : Hadrian.Builder.Ar -- Copyright : (c) Andrey Mokhov 2014-2017 -- License : MIT (see the file LICENSE) -- Maintainer : andrey.mokhov@gmail.com -- Stability : experimental -- -- Support for invoking the archiving utility @ar@. We take care not to exceed -- the limit on command line length, which differs across supported operating -- systems (see 'cmdLineLengthLimit'). We need to handle @ar@ in a special way -- because we sometimes archive __a lot__ of files (in the Cabal library, for -- example, command line length can reach 2MB!). To work around the limit on the -- command line length we pass the list of files to be archived via a temporary -- file (see 'runAr'), or alternatively, we split the argument list into chunks -- and call @ar@ multiple times, e.g. when passing arguments via a temporary -- file is not supported (see 'runArWithoutTempFile'). ----------------------------------------------------------------------------- module Hadrian.Builder.Ar (ArMode (..), runAr, runArWithoutTempFile) where import Control.Monad import Development.Shake import Development.Shake.Classes import GHC.Generics import Hadrian.Utilities -- | We support packing and unpacking archives with @ar@. data ArMode = Pack | Unpack deriving (Eq, Generic, Show) instance Binary ArMode instance Hashable ArMode instance NFData ArMode -- | Invoke @ar@ given a path to it and a list of arguments. The list of files -- to be archived is passed via a temporary response file. Passing arguments -- via a response file is not supported by some versions of @ar@, in which -- case you should use 'runArWithoutTempFile' instead. runAr :: FilePath -- ^ path to @ar@ -> [String] -- ^ other arguments -> [FilePath] -- ^ input file paths -> Action () runAr arPath flagArgs fileArgs = withTempFile $ \tmp -> do writeFile' tmp $ unwords fileArgs cmd [arPath] flagArgs ('@' : tmp) -- | Invoke @ar@ given a path to it and a list of arguments. Note that @ar@ -- will be called multiple times if the list of files to be archived is too -- long and doesn't fit into the command line length limit. This function is -- typically much slower than 'runAr'. runArWithoutTempFile :: FilePath -- ^ path to @ar@ -> [String] -- ^ other arguments -> [FilePath] -- ^ input file paths -> Action () runArWithoutTempFile arPath flagArgs fileArgs = forM_ (chunksOfSize cmdLineLengthLimit fileArgs) $ \argsChunk -> unit . cmd [arPath] $ flagArgs ++ argsChunk