{-# LANGUAGE ScopedTypeVariables, PatternGuards, TupleSections #-}
module Language.Fortran.Analysis.Renaming
( analyseRenames, analyseRenamesWithModuleMap, rename, unrename, ModuleMap )
where
import Language.Fortran.AST hiding (fromList)
import Language.Fortran.Intrinsics
import Language.Fortran.Analysis
import Language.Fortran.ParserMonad (FortranVersion(..))
import Prelude hiding (lookup)
import Data.Maybe (mapMaybe, maybe, fromMaybe)
import qualified Data.List as L
import Data.Map (insert, empty, lookup, Map)
import qualified Data.Map.Strict as M
import Control.Monad (void)
import Control.Monad.State.Strict
import Data.Generics.Uniplate.Data
import Data.Data
import Data.Functor.Identity (Identity)
type ModuleMap = Map ProgramUnitName ModEnv
type Renamer a = State RenameState a
data RenameState = RenameState { langVersion :: FortranVersion
, intrinsics :: IntrinsicsTable
, scopeStack :: [String]
, uniqNums :: [Int]
, environ :: [ModEnv]
, moduleMap :: ModuleMap }
deriving (Show, Eq)
type RenamerFunc t = t -> Renamer t
analyseRenames :: Data a => ProgramFile (Analysis a) -> ProgramFile (Analysis a)
analyseRenames (ProgramFile mi pus) = cleanupUseRenames $ ProgramFile mi pus'
where
(Just pus', _) = runRenamer (renameSubPUs (Just pus)) (renameState0 (miVersion mi))
analyseRenamesWithModuleMap :: Data a => ModuleMap -> ProgramFile (Analysis a) -> ProgramFile (Analysis a)
analyseRenamesWithModuleMap mmap (ProgramFile mi pus) = cleanupUseRenames $ ProgramFile mi pus'
where
(Just pus', _) = runRenamer (renameSubPUs (Just pus)) (renameState0 (miVersion mi)) { moduleMap = mmap }
rename :: Data a => ProgramFile (Analysis a) -> ProgramFile (Analysis a)
rename pf = trPU fPU (trE fE pf)
where
trE :: Data a => (Expression a -> Expression a) -> ProgramFile a -> ProgramFile a
trE = transformBi
fE :: Data a => Expression (Analysis a) -> Expression (Analysis a)
fE (ExpValue a s (ValVariable v)) = ExpValue a s . ValVariable $ fromMaybe v (uniqueName a)
fE (ExpValue a s (ValIntrinsic v)) = ExpValue a s . ValIntrinsic $ fromMaybe v (uniqueName a)
fE x = x
trPU :: Data a => (ProgramUnit a -> ProgramUnit a) -> ProgramFile a -> ProgramFile a
trPU = transformBi
fPU :: Data a => ProgramUnit (Analysis a) -> ProgramUnit (Analysis a)
fPU (PUFunction a s ty r n args res b subs) =
PUFunction a s ty r (fromMaybe n (uniqueName a)) args res b subs
fPU (PUSubroutine a s r n args b subs) =
PUSubroutine a s r (fromMaybe n (uniqueName a)) args b subs
fPU x = x
unrename :: Data a => ProgramFile (Analysis a) -> ProgramFile (Analysis a)
unrename = trPU fPU . trE fE
where
trE :: Data a => (Expression (Analysis a) -> Expression (Analysis a)) -> ProgramFile (Analysis a) -> ProgramFile (Analysis a)
trE = transformBi
fE :: Data a => Expression (Analysis a) -> Expression (Analysis a)
fE e@(ExpValue a s (ValVariable _)) = ExpValue a s (ValVariable (srcName e))
fE e@(ExpValue a s (ValIntrinsic _)) = ExpValue a s (ValIntrinsic (srcName e))
fE e = e
trPU :: Data a => (ProgramUnit (Analysis a) -> ProgramUnit (Analysis a)) -> ProgramFile (Analysis a) -> ProgramFile (Analysis a)
trPU = transformBi
fPU :: Data a => ProgramUnit (Analysis a) -> ProgramUnit (Analysis a)
fPU (PUFunction a s ty r _ args res b subs)
| Just srcN <- sourceName a = PUFunction a s ty r srcN args res b subs
fPU (PUSubroutine a s r _ args b subs)
| Just srcN <- sourceName a = PUSubroutine a s r srcN args b subs
fPU pu = pu
programUnit :: Data a => RenamerFunc (ProgramUnit (Analysis a))
programUnit (PUModule a s name blocks m_contains) = do
env0 <- initialEnv blocks
pushScope name env0
blocks1 <- mapM renameModDecls blocks
blocks2 <- mapM renameUseSt blocks1
m_contains' <- renameSubPUs m_contains
blocks3 <- mapM renameBlock blocks2
env <- getEnv
addModEnv name env
let a' = a { moduleEnv = Just env }
popScope
return (PUModule a' s name blocks3 m_contains')
programUnit (PUFunction a s ty rec name args res blocks m_contains) = do
~(Just name') <- getFromEnv name
(blocks1, _) <- returnBlocksEnv blocks name
blocks2 <- mapM renameEntryPointResultDecl blocks1
res' <- mapM renameGenericDecls res
args' <- mapM renameGenericDecls args
blocks3 <- mapM renameDeclDecls blocks2
m_contains' <- renameSubPUs m_contains
blocks4 <- mapM renameBlock blocks3
let env = M.singleton name (name', NTSubprogram)
let a' = a { moduleEnv = Just env }
popScope
let pu' = PUFunction a' s ty rec name args' res' blocks4 m_contains'
return . setSourceName name . setUniqueName name' $ pu'
programUnit (PUSubroutine a s rec name args blocks m_contains) = do
~(Just name') <- getFromEnv name
(blocks1, _) <- returnBlocksEnv blocks name
args' <- mapM renameGenericDecls args
blocks2 <- mapM renameDeclDecls blocks1
m_contains' <- renameSubPUs m_contains
blocks3 <- mapM renameBlock blocks2
let env = M.singleton name (name', NTSubprogram)
let a' = a { moduleEnv = Just env }
popScope
let pu' = PUSubroutine a' s rec name args' blocks3 m_contains'
return . setSourceName name . setUniqueName name' $ pu'
programUnit (PUMain a s n blocks m_contains) = do
env0 <- initialEnv blocks
pushScope (fromMaybe "_main" n) env0
blocks' <- mapM renameDeclDecls blocks
m_contains' <- renameSubPUs m_contains
blocks'' <- mapM renameBlock blocks'
popScope
return (PUMain a s n blocks'' m_contains')
programUnit pu = return pu
returnBlocksEnv :: Data a => [Block (Analysis a)]
-> String
-> StateT RenameState Identity ([Block (Analysis a)], ModEnv)
returnBlocksEnv bs n = do
bs1 <- mapM renameEntryPointDecl bs
e0 <- initialEnv bs1
pushScope n e0
return (bs1, e0)
declarator :: forall a. Data a => RenamerFunc (Declarator (Analysis a))
declarator (DeclVariable a s e1 me2 me3) = do
e1' <- renameExpDecl e1
me2' <- transformBiM (renameExp :: RenamerFunc (Expression (Analysis a))) me2
me3' <- transformBiM (renameExp :: RenamerFunc (Expression (Analysis a))) me3
return $ DeclVariable a s e1' me2' me3'
declarator (DeclArray a s e1 ddAList me2 me3) = do
e1' <- renameExpDecl e1
ddAList' <- transformBiM (renameExp :: RenamerFunc (Expression (Analysis a))) ddAList
me2' <- transformBiM (renameExp :: RenamerFunc (Expression (Analysis a))) me2
me3' <- transformBiM (renameExp :: RenamerFunc (Expression (Analysis a))) me3
return $ DeclArray a s e1' ddAList' me2' me3'
expression :: Data a => RenamerFunc (Expression (Analysis a))
expression = renameExp
renameState0 :: FortranVersion -> RenameState
renameState0 v = RenameState { langVersion = v
, intrinsics = getVersionIntrinsics v
, scopeStack = []
, uniqNums = [1..]
, environ = [empty]
, moduleMap = empty }
runRenamer :: State a b -> a -> (b, a)
runRenamer = runState
getUniqNum :: Renamer Int
getUniqNum = do
uniqNum <- gets (head . uniqNums)
modify $ \ s -> s { uniqNums = drop 1 (uniqNums s) }
return uniqNum
uniquify :: String -> String -> Renamer String
uniquify scope var = do
n <- getUniqNum
return $ scope ++ "_" ++ var ++ show n
isUseStatement :: Block a -> Bool
isUseStatement (BlStatement _ _ _ (StUse _ _ (ExpValue _ _ (ValVariable _)) _ _ _)) = True
isUseStatement _ = False
initialEnv :: forall a. Data a => [Block (Analysis a)] -> Renamer ModEnv
initialEnv blocks = do
let uses = filter isUseStatement blocks
mMap <- gets moduleMap
modEnv <- fmap M.unions . forM uses $ \ use -> case use of
(BlStatement _ _ _ (StUse _ _ (ExpValue _ _ (ValVariable m)) _ _ Nothing)) ->
return $ fromMaybe empty (Named m `lookup` mMap)
(BlStatement _ _ _ (StUse _ _ (ExpValue _ _ (ValVariable m)) _ _ (Just onlyAList)))
| only <- aStrip onlyAList -> do
let env = fromMaybe empty (Named m `lookup` mMap)
let localNamePairs = flip mapMaybe only $ \ r -> case r of
UseID _ _ v@(ExpValue _ _ ValVariable{}) -> Just (varName v, varName v)
UseRename _ _ u v -> Just (varName u, varName v)
_ -> Nothing
let re = M.fromList [ (local, info) | (local, orig) <- localNamePairs
, Just info <- [M.lookup orig env] ]
return re
_ -> return empty
let global = fromMaybe M.empty $ M.lookup NamelessMain mMap
let common = M.fromList [ (v, (v', NTVariable))
| CommonGroup _ _ me1 alist <- universeBi blocks :: [CommonGroup (Analysis a)]
, let prefix = case me1 of Just e1 -> srcName e1; _ -> ""
, e@(ExpValue _ _ ValVariable{}) <- universeBi (aStrip alist) :: [Expression (Analysis a)]
, let v = srcName e
, let v' = prefix ++ "_" ++ v ++ "_common" ]
return $ M.unions [modEnv, global, common]
getScopes :: Renamer String
getScopes = gets (L.intercalate "_" . reverse . scopeStack)
pushScope :: String -> ModEnv -> Renamer ()
pushScope name env0 = modify $ \ s -> s { scopeStack = name : scopeStack s
, environ = env0 : environ s }
popScope :: Renamer ()
popScope = modify $ \ s -> s { scopeStack = drop 1 $ scopeStack s
, environ = drop 1 $ environ s }
addModEnv :: String -> ModEnv -> Renamer ()
addModEnv name env = modify $ \ s -> s { moduleMap = insert (Named name) env (moduleMap s) }
getEnv :: Renamer ModEnv
getEnv = gets (head . environ)
getEnvs :: Renamer ModEnv
getEnvs = M.unionsWith (curry fst) `fmap` gets environ
getFromEnv :: String -> Renamer (Maybe String)
getFromEnv v = ((fst `fmap`) . lookup v) `fmap` getEnv
getFromEnvs :: String -> Renamer (Maybe String)
getFromEnvs = fmap (fmap fst) . getFromEnvsWithType
getFromEnvsWithType :: String -> Renamer (Maybe (String, NameType))
getFromEnvsWithType v = do
envs <- getEnvs
case lookup v envs of
Just (v', nt) -> return $ Just (v', nt)
Nothing -> do
itab <- gets intrinsics
case getIntrinsicReturnType v itab of
Nothing -> return Nothing
Just _ -> (Just . (,NTIntrinsic)) `fmap` addUnique v NTIntrinsic
getFromEnvsIfSubprogram :: String -> Renamer (Maybe String)
getFromEnvsIfSubprogram v = do
mEntry <- getFromEnvsWithType v
case mEntry of
Just (v', NTSubprogram) -> return $ Just v'
Just (_, NTVariable) -> getFromEnv v
_ -> return Nothing
addToEnv :: String -> String -> NameType -> Renamer ()
addToEnv v v' nt = modify $ \ s -> s { environ = insert v (v', nt) (head (environ s)) : drop 1 (environ s) }
addUnique :: String -> NameType -> Renamer String
addUnique v nt = do
v' <- flip uniquify v =<< getScopes
addToEnv v v' nt
return v'
addUnique_ :: String -> NameType -> Renamer ()
addUnique_ v nt = void (addUnique v nt)
maybeAddUnique :: String -> NameType -> Renamer String
maybeAddUnique v nt = maybe (addUnique v nt) return =<< getFromEnvsIfSubprogram v
setUniqueName, setSourceName :: (Annotated f, Data a) => String -> f (Analysis a) -> f (Analysis a)
setUniqueName un x
| a@Analysis { uniqueName = Nothing } <- getAnnotation x = setAnnotation (a { uniqueName = Just un }) x
| otherwise = x
setSourceName sn x
| a@Analysis { sourceName = Nothing } <- getAnnotation x = setAnnotation (a { sourceName = Just sn }) x
| otherwise = x
renameSubPUs :: Data a => RenamerFunc (Maybe [ProgramUnit (Analysis a)])
renameSubPUs Nothing = return Nothing
renameSubPUs (Just pus) = skimProgramUnits pus >> Just <$> mapM programUnit pus
skimProgramUnits :: Data a => [ProgramUnit (Analysis a)] -> Renamer ()
skimProgramUnits pus = forM_ pus $ \ pu -> case pu of
PUModule _ _ name _ _ -> addToEnv name name NTSubprogram
PUFunction _ _ _ _ name _ _ _ _ -> addUnique_ name NTSubprogram
PUSubroutine _ _ _ name _ _ _ -> addUnique_ name NTSubprogram
PUMain _ _ (Just name) _ _ -> addToEnv name name NTSubprogram
_ -> return ()
renameGenericDecls :: (Data a, Data (f (Analysis a))) => RenamerFunc (f (Analysis a))
renameGenericDecls = trans renameExpDecl
where
trans :: (Data a, Data (f (Analysis a))) => RenamerFunc (Expression (Analysis a)) -> RenamerFunc (f (Analysis a))
trans = transformBiM
renameExpDecl :: Data a => RenamerFunc (Expression (Analysis a))
renameExpDecl e@(ExpValue _ _ (ValVariable v)) = flip setUniqueName (setSourceName v e) `fmap` maybeAddUnique v NTVariable
renameExpDecl e@(ExpValue _ _ (ValIntrinsic v)) = flip setUniqueName (setSourceName v e) `fmap` addUnique v NTIntrinsic
renameExpDecl e = return e
renameInterfaces :: (Data a, Data (f (Analysis a))) => RenamerFunc (f (Analysis a))
renameInterfaces = trans interface
where
trans :: (Data a, Data (f (Analysis a))) => RenamerFunc (Block (Analysis a)) -> RenamerFunc (f (Analysis a))
trans = transformBiM
interface :: Data a => RenamerFunc (Block (Analysis a))
interface (BlInterface a s (Just e@(ExpValue _ _ (ValVariable v))) abst pus bs) = do
e' <- flip setUniqueName (setSourceName v e) `fmap` maybeAddUnique v NTSubprogram
pure $ BlInterface a s (Just e') abst pus bs
interface b = pure b
renameModDecls :: (Data a, Data (f (Analysis a))) => RenamerFunc (f (Analysis a))
renameModDecls = renameDeclDecls <=< renameInterfaces
renameDeclDecls :: (Data a, Data (f (Analysis a))) => RenamerFunc (f (Analysis a))
renameDeclDecls = trans declarator
where
trans :: (Data a, Data (f (Analysis a))) => RenamerFunc (Declarator (Analysis a)) -> RenamerFunc (f (Analysis a))
trans = transformBiM
renameEntryPointDecl :: Data a => RenamerFunc (Block (Analysis a))
renameEntryPointDecl (BlStatement a s l (StEntry a' s' v mArgs mRes)) = do
v' <- renameExpDecl v
return (BlStatement a s l (StEntry a' s' v' mArgs mRes))
renameEntryPointDecl b = return b
renameEntryPointResultDecl :: Data a => RenamerFunc (Block (Analysis a))
renameEntryPointResultDecl (BlStatement a s l (StEntry a' s' v mArgs (Just res))) = do
res' <- renameExpDecl res
return (BlStatement a s l (StEntry a' s' v mArgs (Just res')))
renameEntryPointResultDecl b = return b
renameExp :: Data a => RenamerFunc (Expression (Analysis a))
renameExp e@(ExpValue _ _ (ValVariable v)) = maybe e (`setUniqueName` setSourceName v e) `fmap` getFromEnvs v
renameExp e@(ExpValue _ _ (ValIntrinsic v)) = flip setUniqueName (setSourceName v e) `fmap` addUnique v NTIntrinsic
renameExp e = return e
renameBlock :: Data a => RenamerFunc (Block (Analysis a))
renameBlock = trans expression
where
trans :: Data a => RenamerFunc (Expression a) -> RenamerFunc (Block a)
trans = transformBiM
renameUseSt :: Data a => RenamerFunc (Block (Analysis a))
renameUseSt (BlStatement a s l st@StUse{}) = BlStatement a s l <$> trans expression st
where
trans :: Data a => RenamerFunc (Expression a) -> RenamerFunc (Statement a)
trans = transformBiM
renameUseSt b = return b
cleanupUseRenames :: forall a. Data a => ProgramFile (Analysis a) -> ProgramFile (Analysis a)
cleanupUseRenames = transformBi (\ u -> case u :: Use (Analysis a) of
UseRename a s e1 e2@(ExpValue _ _ (ValVariable v)) -> UseRename a s e1 $ setUniqueName (varName e1) (setSourceName v e2)
_ -> u)