diff options
author | Edward Z. Yang <ezyang@cs.stanford.edu> | 2015-10-10 12:01:14 -0700 |
---|---|---|
committer | Edward Z. Yang <ezyang@cs.stanford.edu> | 2016-10-08 00:20:34 -0700 |
commit | 00b530d5402aaa37e4085ecdcae0ae54454736c1 (patch) | |
tree | 2d2963db4abdbcba9c12aea13a26e29e718e4778 /libraries/ghc-boot | |
parent | 887485a45ae55e81b26b6412b6f9dcf6a497f044 (diff) | |
download | haskell-00b530d5402aaa37e4085ecdcae0ae54454736c1.tar.gz |
The Backpack patch.
Summary:
This patch implements Backpack for GHC. It's a big patch but I've tried quite
hard to keep things, by-in-large, self-contained.
The user facing specification for Backpack can be found at:
https://github.com/ezyang/ghc-proposals/blob/backpack/proposals/0000-backpack.rst
A guide to the implementation can be found at:
https://github.com/ezyang/ghc-proposals/blob/backpack-impl/proposals/0000-backpack-impl.rst
Has a submodule update for Cabal, as well as a submodule update
for filepath to handle more strict checking of cabal-version.
Signed-off-by: Edward Z. Yang <ezyang@cs.stanford.edu>
Test Plan: validate
Reviewers: simonpj, austin, simonmar, bgamari, goldfire
Subscribers: thomie, mpickering
Differential Revision: https://phabricator.haskell.org/D1482
Diffstat (limited to 'libraries/ghc-boot')
-rw-r--r-- | libraries/ghc-boot/GHC/PackageDb.hs | 134 |
1 files changed, 101 insertions, 33 deletions
diff --git a/libraries/ghc-boot/GHC/PackageDb.hs b/libraries/ghc-boot/GHC/PackageDb.hs index 26bf67f98d..2e51af0dcb 100644 --- a/libraries/ghc-boot/GHC/PackageDb.hs +++ b/libraries/ghc-boot/GHC/PackageDb.hs @@ -39,8 +39,9 @@ module GHC.PackageDb ( InstalledPackageInfo(..), DbModule(..), + DbUnitId(..), BinaryStringRep(..), - DbModuleRep(..), + DbUnitIdModuleRep(..), emptyInstalledPackageInfo, readPackageDbForGhc, readPackageDbForGhcPkg, @@ -67,14 +68,15 @@ import System.Directory -- | This is a subset of Cabal's 'InstalledPackageInfo', with just the bits -- that GHC is interested in. -- -data InstalledPackageInfo srcpkgid srcpkgname unitid modulename mod +data InstalledPackageInfo compid srcpkgid srcpkgname instunitid unitid modulename mod = InstalledPackageInfo { - unitId :: unitid, + unitId :: instunitid, + instantiatedWith :: [(modulename, mod)], sourcePackageId :: srcpkgid, packageName :: srcpkgname, packageVersion :: Version, abiHash :: String, - depends :: [unitid], + depends :: [instunitid], importDirs :: [FilePath], hsLibraries :: [String], extraLibraries :: [String], @@ -97,37 +99,62 @@ data InstalledPackageInfo srcpkgid srcpkgname unitid modulename mod -- | A convenience constraint synonym for common constraints over parameters -- to 'InstalledPackageInfo'. -type RepInstalledPackageInfo srcpkgid srcpkgname unitid modulename mod = +type RepInstalledPackageInfo compid srcpkgid srcpkgname instunitid unitid modulename mod = (BinaryStringRep srcpkgid, BinaryStringRep srcpkgname, - BinaryStringRep unitid, BinaryStringRep modulename, - DbModuleRep unitid modulename mod) + BinaryStringRep modulename, BinaryStringRep compid, + BinaryStringRep instunitid, + DbUnitIdModuleRep compid unitid modulename mod) --- | A type-class for the types which can be converted into 'DbModule'. +-- | A type-class for the types which can be converted into 'DbModule'/'DbUnitId'. +-- There is only one type class because these types are mutually recursive. -- NB: The functional dependency helps out type inference in cases -- where types would be ambiguous. -class DbModuleRep unitid modulename mod - | mod -> unitid, unitid -> mod, mod -> modulename where - fromDbModule :: DbModule unitid modulename -> mod - toDbModule :: mod -> DbModule unitid modulename +class DbUnitIdModuleRep compid unitid modulename mod + | mod -> unitid, unitid -> mod, mod -> modulename, unitid -> compid where + fromDbModule :: DbModule compid unitid modulename mod -> mod + toDbModule :: mod -> DbModule compid unitid modulename mod + fromDbUnitId :: DbUnitId compid unitid modulename mod -> unitid + toDbUnitId :: unitid -> DbUnitId compid unitid modulename mod -- | @ghc-boot@'s copy of 'Module', i.e. what is serialized to the database. --- Use 'DbModuleRep' to convert it into an actual 'Module'. -data DbModule unitid modulename +-- Use 'DbUnitIdModuleRep' to convert it into an actual 'Module'. +-- It has phantom type parameters as this is the most convenient way +-- to avoid undecidable instances. +data DbModule compid unitid modulename mod = DbModule { dbModuleUnitId :: unitid, dbModuleName :: modulename } + | DbModuleVar { + dbModuleVarName :: modulename + } + deriving (Eq, Show) + +-- | @ghc-boot@'s copy of 'UnitId', i.e. what is serialized to the database. +-- Use 'DbUnitIdModuleRep' to convert it into an actual 'UnitId'. +-- It has phantom type parameters as this is the most convenient way +-- to avoid undecidable instances. +data DbUnitId compid unitid modulename mod + = DbUnitId { + dbUnitIdComponentId :: compid, + dbUnitIdInsts :: [(modulename, mod)] + } + | DbHashedUnitId { + dbUnitIdComponentId :: compid, + dbUnitIdHash :: Maybe BS.ByteString + } deriving (Eq, Show) class BinaryStringRep a where fromStringRep :: BS.ByteString -> a toStringRep :: a -> BS.ByteString -emptyInstalledPackageInfo :: RepInstalledPackageInfo a b c d e - => InstalledPackageInfo a b c d e +emptyInstalledPackageInfo :: RepInstalledPackageInfo a b c d e f g + => InstalledPackageInfo a b c d e f g emptyInstalledPackageInfo = InstalledPackageInfo { unitId = fromStringRep BS.empty, + instantiatedWith = [], sourcePackageId = fromStringRep BS.empty, packageName = fromStringRep BS.empty, packageVersion = Version [] [], @@ -154,8 +181,8 @@ emptyInstalledPackageInfo = -- | Read the part of the package DB that GHC is interested in. -- -readPackageDbForGhc :: RepInstalledPackageInfo a b c d e => - FilePath -> IO [InstalledPackageInfo a b c d e] +readPackageDbForGhc :: RepInstalledPackageInfo a b c d e f g => + FilePath -> IO [InstalledPackageInfo a b c d e f g] readPackageDbForGhc file = decodeFromFile file getDbForGhc where @@ -187,8 +214,8 @@ readPackageDbForGhcPkg file = -- | Write the whole of the package DB, both parts. -- -writePackageDb :: (Binary pkgs, RepInstalledPackageInfo a b c d e) => - FilePath -> [InstalledPackageInfo a b c d e] -> pkgs -> IO () +writePackageDb :: (Binary pkgs, RepInstalledPackageInfo a b c d e f g) => + FilePath -> [InstalledPackageInfo a b c d e f g] -> pkgs -> IO () writePackageDb file ghcPkgs ghcPkgPart = writeFileAtomic file (runPut putDbForGhcPkg) where @@ -274,10 +301,10 @@ writeFileAtomic targetPath content = do hClose handle renameFile tmpPath targetPath) -instance (RepInstalledPackageInfo a b c d e) => - Binary (InstalledPackageInfo a b c d e) where +instance (RepInstalledPackageInfo a b c d e f g) => + Binary (InstalledPackageInfo a b c d e f g) where put (InstalledPackageInfo - unitId sourcePackageId + unitId instantiatedWith sourcePackageId packageName packageVersion abiHash depends importDirs hsLibraries extraLibraries extraGHCiLibraries libraryDirs @@ -291,6 +318,8 @@ instance (RepInstalledPackageInfo a b c d e) => put (toStringRep packageName) put packageVersion put (toStringRep unitId) + put (map (\(mod_name, mod) -> (toStringRep mod_name, toDbModule mod)) + instantiatedWith) put abiHash put (map toStringRep depends) put importDirs @@ -306,7 +335,7 @@ instance (RepInstalledPackageInfo a b c d e) => put includeDirs put haddockInterfaces put haddockHTMLs - put (map (\(mod_name, mod) -> (toStringRep mod_name, fmap toDbModule mod)) + put (map (\(mod_name, mb_mod) -> (toStringRep mod_name, fmap toDbModule mb_mod)) exposedModules) put (map toStringRep hiddenModules) put exposed @@ -317,6 +346,7 @@ instance (RepInstalledPackageInfo a b c d e) => packageName <- get packageVersion <- get unitId <- get + instantiatedWith <- get abiHash <- get depends <- get importDirs <- get @@ -338,6 +368,8 @@ instance (RepInstalledPackageInfo a b c d e) => trusted <- get return (InstalledPackageInfo (fromStringRep unitId) + (map (\(mod_name, mod) -> (fromStringRep mod_name, fromDbModule mod)) + instantiatedWith) (fromStringRep sourcePackageId) (fromStringRep packageName) packageVersion abiHash @@ -348,19 +380,55 @@ instance (RepInstalledPackageInfo a b c d e) => ldOptions ccOptions includes includeDirs haddockInterfaces haddockHTMLs - (map (\(mod_name, mod) -> - (fromStringRep mod_name, fmap fromDbModule mod)) + (map (\(mod_name, mb_mod) -> + (fromStringRep mod_name, fmap fromDbModule mb_mod)) exposedModules) (map fromStringRep hiddenModules) exposed trusted) -instance (BinaryStringRep a, BinaryStringRep b) => - Binary (DbModule a b) where +instance (BinaryStringRep modulename, BinaryStringRep compid, + DbUnitIdModuleRep compid unitid modulename mod) => + Binary (DbModule compid unitid modulename mod) where put (DbModule dbModuleUnitId dbModuleName) = do - put (toStringRep dbModuleUnitId) + putWord8 0 + put (toDbUnitId dbModuleUnitId) put (toStringRep dbModuleName) + put (DbModuleVar dbModuleVarName) = do + putWord8 1 + put (toStringRep dbModuleVarName) + get = do + b <- getWord8 + case b of + 0 -> do dbModuleUnitId <- get + dbModuleName <- get + return (DbModule (fromDbUnitId dbModuleUnitId) + (fromStringRep dbModuleName)) + _ -> do dbModuleVarName <- get + return (DbModuleVar (fromStringRep dbModuleVarName)) + +instance (BinaryStringRep modulename, BinaryStringRep compid, + DbUnitIdModuleRep compid unitid modulename mod) => + Binary (DbUnitId compid unitid modulename mod) where + put (DbHashedUnitId cid hash) = do + putWord8 0 + put (toStringRep cid) + put hash + put (DbUnitId dbUnitIdComponentId dbUnitIdInsts) = do + putWord8 1 + put (toStringRep dbUnitIdComponentId) + put (map (\(mod_name, mod) -> (toStringRep mod_name, toDbModule mod)) dbUnitIdInsts) get = do - dbModuleUnitId <- get - dbModuleName <- get - return (DbModule (fromStringRep dbModuleUnitId) - (fromStringRep dbModuleName)) + b <- getWord8 + case b of + 0 -> do + cid <- get + hash <- get + return (DbHashedUnitId (fromStringRep cid) hash) + _ -> do + dbUnitIdComponentId <- get + dbUnitIdInsts <- get + return (DbUnitId + (fromStringRep dbUnitIdComponentId) + (map (\(mod_name, mod) -> ( fromStringRep mod_name + , fromDbModule mod)) + dbUnitIdInsts)) |