{-# LANGUAGE BangPatterns #-} module Hhp.Find ( Symbol , SymMdlDb , findSymbol , getSymMdlDb , lookupSym ) where import GHC (Ghc, DynFlags, Module, ModuleInfo) import qualified GHC as G import Module (Module(..)) import Outputable (ppr) import PackageConfig (PackageConfig, exposedModules, packageConfigId) import Packages (listPackageConfigMap) import Control.DeepSeq (force) import Data.Function (on) import Data.List (groupBy, sort) import Data.Map.Strict (Map) import qualified Data.Map.Strict as M import Data.Maybe (fromMaybe) import Hhp.Doc (showOneLine, styleUnqualified) import Hhp.GHCApi import Hhp.Gap (getModuleName) import Hhp.Types -- | Type of key for `SymMdlDb`. type Symbol = String -- | Database from 'Symbol' to modules. newtype SymMdlDb = SymMdlDb (Map Symbol [ModuleString]) -- | Finding modules to which the symbol belong. findSymbol :: Options -> Cradle -> Symbol -> IO String findSymbol opt cradle sym = withGHC' $ do initializeFlagsWithCradle opt cradle lookupSym opt sym <$> getSymMdlDb -- | Creating 'SymMdlDb'. getSymMdlDb :: Ghc SymMdlDb getSymMdlDb = do sm <- G.getSessionDynFlags >>= browseAll let !sms = force $ map tieup $ groupBy ((==) `on` fst) $ sort sm !m = force $ M.fromList sms return (SymMdlDb m) where tieup x = (head (map fst x), map snd x) -- | Looking up 'SymMdlDb' with 'Symbol' to find modules. lookupSym :: Options -> Symbol -> SymMdlDb -> String lookupSym opt sym (SymMdlDb db) = convert opt $ fromMaybe [] (M.lookup sym db) ---------------------------------------------------------------- -- | Browsing all functions in all system/user modules. browseAll :: DynFlags -> Ghc [(String,String)] browseAll dflag = do let ms = packageModules dflag is <- mapM G.getModuleInfo ms return $ concatMap (toNameModule dflag) (zip ms is) toNameModule :: DynFlags -> (Module, Maybe ModuleInfo) -> [(String,String)] toNameModule _ (_,Nothing) = [] toNameModule dflag (m,Just inf) = map (\name -> (toStr name, mdl)) names where mdl = G.moduleNameString (G.moduleName m) names = G.modInfoExports inf toStr = showOneLine dflag (styleUnqualified dflag) . ppr packageModules :: DynFlags -> [Module] packageModules dflag = concatMap fromPackageConfig $ listPackageConfigMap dflag fromPackageConfig :: PackageConfig -> [Module] fromPackageConfig pkgcnf = modules where uid = packageConfigId pkgcnf -- check me moduleNames = map getModuleName $ exposedModules pkgcnf modules = map (Module uid) moduleNames