{-# LANGUAGE CPP #-} {-# LANGUAGE FlexibleContexts #-} {-# LANGUAGE FlexibleInstances #-} {-# LANGUAGE FunctionalDependencies #-} {-# LANGUAGE MultiParamTypeClasses #-} {-# LANGUAGE PatternSynonyms #-} {-# LANGUAGE RankNTypes #-} {-# LANGUAGE TypeApplications #-} {-# LANGUAGE UndecidableInstances #-} -- | Thin compatibility layer around GHC -- -- This should be the only module with GHC-specific CPP directives, and the -- rest of the plugin should not import from any GHC modules directly. module Data.Record.Internal.GHC.Shim ( -- * Miscellaneous importDecl , conPat , mkFunBind , HsModule , LHsModule , LRdrName , pattern GHC.HsModule , putLogMsg -- * Extensions , HasDefaultExt(..) -- * Generalized @forall@ #if __GLASGOW_HASKELL__ >= 900 , HsTyVarBndr , LHsTyVarBndr #endif , hsFunTy , userTyVar , kindedTyVar , hsTyVarLName , setDefaultSpecificity -- * New functionality , compareHs , InheritLoc(..) -- * Re-exports -- The whole-sale module exports are not ideal for preserving compatibility -- across ghc versions, but we'll deal with this on a case by case basis. #if __GLASGOW_HASKELL__ < 900 , module Bag , module BasicTypes , module ErrUtils , module GHC , module GhcPlugins , module HscMain , module NameCache , module TcEvidence #else , module GHC.Data.Bag , module GHC.Driver.Main , module GHC.Hs , module GHC.Plugins , module GHC.Tc.Types.Evidence , module GHC.Types.Basic , module GHC.Types.Name.Cache , module GHC.Utils.Error #endif ) where import Data.List.NonEmpty (NonEmpty(..)) import Data.Generics (Data, GenericQ, cast, toConstr, gzipWithQ) import qualified Data.List.NonEmpty as NE #if __GLASGOW_HASKELL__ < 900 import Bag (listToBag, emptyBag) import BasicTypes (SourceText(NoSourceText)) import ConLike (ConLike) import ErrUtils (mkErrMsg, mkWarnMsg) import GHC hiding (AnnKeywordId(..), HsModule, exprType, typeKind, mkFunBind) import GhcPlugins hiding ((<>), getHscEnv, putLogMsg) import HscMain (getHscEnv) import NameCache (NameCache(nsUniqs)) import PatSyn (PatSyn) import TcEvidence (HsWrapper(WpHole)) import qualified GHC import qualified GhcPlugins as GHC #else import GHC.Core.Class (Class) import GHC.Core.ConLike (ConLike) import GHC.Core.PatSyn (PatSyn) import GHC.Data.Bag (listToBag, emptyBag) import GHC.Driver.Main (getHscEnv) import GHC.Hs hiding (LHsTyVarBndr, HsTyVarBndr, HsModule, mkFunBind) import GHC.Parser.Annotation (IsUnicodeSyntax(NormalSyntax)) import GHC.Plugins hiding ((<>), getHscEnv, putLogMsg) import GHC.Tc.Types.Evidence (HsWrapper(WpHole)) import GHC.Types.Basic (SourceText(NoSourceText)) import GHC.Types.Name.Cache (NameCache(nsUniqs)) import GHC.Utils.Error (Severity(SevError, SevWarning), mkErrMsg, mkWarnMsg) import qualified GHC.Hs as GHC import qualified GHC.Plugins as GHC #endif {------------------------------------------------------------------------------- Miscellaneous -------------------------------------------------------------------------------} -- | Optionally @qualified@ import declaration importDecl :: ModuleName -> Bool -> LImportDecl GhcPs importDecl name qualified = noLoc $ ImportDecl { ideclExt = defExt , ideclSourceSrc = NoSourceText , ideclName = noLoc name , ideclPkgQual = Nothing , ideclSafe = False , ideclImplicit = False , ideclAs = Nothing , ideclHiding = Nothing #if __GLASGOW_HASKELL__ < 810 , ideclQualified = qualified #else , ideclQualified = if qualified then QualifiedPre else NotQualified #endif #if __GLASGOW_HASKELL__ < 900 , ideclSource = False #else , ideclSource = NotBoot #endif } conPat :: Located RdrName -> HsConPatDetails GhcPs -> Pat GhcPs #if __GLASGOW_HASKELL__ < 900 conPat x y = ConPatIn x y #else conPat x y = ConPat noExtField x y #endif mkFunBind :: Located RdrName -> [LMatch GhcPs (LHsExpr GhcPs)] -> HsBind GhcPs #if __GLASGOW_HASKELL__ < 810 mkFunBind = GHC.mkFunBind #else mkFunBind = GHC.mkFunBind Generated #endif #if __GLASGOW_HASKELL__ < 900 type HsModule = GHC.HsModule GhcPs #else type HsModule = GHC.HsModule #endif type LHsModule = Located HsModule type LRdrName = Located RdrName putLogMsg :: DynFlags -> WarnReason -> Severity -> SrcSpan -> SDoc -> IO () #if __GLASGOW_HASKELL__ < 900 putLogMsg flags reason sev srcspan = GHC.putLogMsg flags reason sev srcspan (defaultErrStyle flags) #else putLogMsg = GHC.putLogMsg #endif {------------------------------------------------------------------------------- Extensions -------------------------------------------------------------------------------} class HasDefaultExt a where defExt :: a #if __GLASGOW_HASKELL__ < 810 instance HasDefaultExt NoExt where defExt = noExt #else instance HasDefaultExt NoExtField where defExt = noExtField #endif #if __GLASGOW_HASKELL__ >= 900 instance HasDefaultExt LayoutInfo where defExt = NoLayoutInfo #endif {------------------------------------------------------------------------------- Generalized @forall@ in 9.0 -------------------------------------------------------------------------------} #if __GLASGOW_HASKELL__ >= 900 type HsTyVarBndr pass = GHC.HsTyVarBndr () pass type LHsTyVarBndr pass = GHC.LHsTyVarBndr () pass #endif hsFunTy :: XFunTy pass -> LHsType pass -> LHsType pass -> HsType pass #if __GLASGOW_HASKELL__ < 900 hsFunTy = HsFunTy #else hsFunTy ext = HsFunTy ext (HsUnrestrictedArrow NormalSyntax) #endif userTyVar :: XUserTyVar pass -> Located (IdP pass) -> HsTyVarBndr pass #if __GLASGOW_HASKELL__ < 900 userTyVar = UserTyVar #else userTyVar ext = UserTyVar ext () #endif kindedTyVar :: XKindedTyVar pass -> Located (IdP pass) -> LHsKind pass -> HsTyVarBndr pass #if __GLASGOW_HASKELL__ < 900 kindedTyVar = KindedTyVar #else kindedTyVar ext = KindedTyVar ext () #endif -- | Like 'hsTyVarName', but don't throw away the location information hsTyVarLName :: HsTyVarBndr GhcPs -> LRdrName #if __GLASGOW_HASKELL__ < 900 hsTyVarLName (UserTyVar _ n ) = n hsTyVarLName (KindedTyVar _ n _) = n hsTyVarLName _ = panic "hsTyVarLName" #else hsTyVarLName (UserTyVar _ _ n ) = n hsTyVarLName (KindedTyVar _ _ n _) = n #endif #if __GLASGOW_HASKELL__ < 900 setDefaultSpecificity :: LHsTyVarBndr pass -> GHC.LHsTyVarBndr pass setDefaultSpecificity = id #else setDefaultSpecificity :: LHsTyVarBndr pass -> GHC.LHsTyVarBndr Specificity pass setDefaultSpecificity (L l v) = L l $ case v of UserTyVar ext () name -> UserTyVar ext SpecifiedSpec name KindedTyVar ext () name kind -> KindedTyVar ext SpecifiedSpec name kind XTyVarBndr ext -> XTyVarBndr ext #endif {------------------------------------------------------------------------------- New functionality -------------------------------------------------------------------------------} -- | Generic comparison for (parts of) the AST -- -- NOTE: Not all abstract types are given special treatment here; in particular, -- types only used in type-checked code ignored. To extend/audit this function, -- grep the @ghc@ source for @abstractConstr@. Without further extensions, -- all values of these types are considered equal. -- -- NOTE: Although @ghc@ declares the constructor of @Bag@ as abstract as well, -- we don't actually need a special case here: the constructors will be -- considered equal, but 'gfoldl' will traverse the /elements/ of the @Bag@ -- nonetheless, which is precisely what we want. compareHs' :: GenericQ (GenericQ Bool) compareHs' x y | (Just x', Just y') <- (cast x, cast y) = (==) @ConLike x' y' | (Just x', Just y') <- (cast x, cast y) = (==) @PatSyn x' y' | (Just x', Just y') <- (cast x, cast y) = (==) @Class x' y' | (Just x', Just y') <- (cast x, cast y) = (==) @DataCon x' y' | (Just x', Just y') <- (cast x, cast y) = (==) @FastString x' y' | (Just x', Just y') <- (cast x, cast y) = (==) @Module x' y' | (Just x', Just y') <- (cast x, cast y) = (==) @ModuleName x' y' | (Just x', Just y') <- (cast x, cast y) = (==) @Name x' y' | (Just x', Just y') <- (cast x, cast y) = (==) @OccName x' y' | (Just x', Just y') <- (cast x, cast y) = (==) @TyCon x' y' | (Just x', Just y') <- (cast x, cast y) = (==) @UnitId x' y' | (Just x', Just y') <- (cast x, cast y) = (==) @Var x' y' #if __GLASGOW_HASKELL__ >= 900 | (Just x', Just y') <- (cast x, cast y) = (==) @Unit x' y' #endif | (Just x', Just y') <- (cast x, cast y) = ignr @RealSrcSpan x' y' | (Just x', Just y') <- (cast x, cast y) = ignr @SrcSpan x' y' | otherwise = (toConstr x == toConstr y) && and (gzipWithQ compareHs' x y) where ignr :: a -> a -> Bool ignr _ _ = True -- | Compare two (parts) of a Haskell source tree for equality -- -- The trees are compared for literal equality, but 'SrcSpan's are ignored. compareHs :: Data a => a -> a -> Bool compareHs x y = compareHs' x y {------------------------------------------------------------------------------- Working with locations -------------------------------------------------------------------------------} class InheritLoc a b lb | lb -> b where inheritLoc :: a -> b -> lb instance InheritLoc (Located a) b (Located b) where inheritLoc (L l _) = L l instance InheritLoc a b lb => InheritLoc (NonEmpty a) b lb where inheritLoc = inheritLoc . NE.head -- | The instance for @[]@ is not ideal: we use 'noLoc' if the list is empty -- -- For the use cases in this library, this is acceptable: typically these are -- lists with elements for the record fields, and having slightly poorer error -- messages for highly unusual "empty large" records is fine. instance InheritLoc a b (Located b) => InheritLoc [a] b (Located b) where inheritLoc (a:_) = inheritLoc a inheritLoc [] = noLoc #if __GLASGOW_HASKELL__ < 810 -- In 8.8, 'LPat' is a synonym for 'Pat' instance InheritLoc a (Pat p) (LPat p) where inheritLoc _ = id #endif