{-# LANGUAGE CPP #-}
{-# LANGUAGE RecordWildCards #-}
module Overloaded.Plugin (plugin) where
import Control.Applicative ((<|>))
import Control.Monad (foldM, when)
import Control.Monad.IO.Class (MonadIO (..))
import Data.List (foldl', intercalate)
import Data.List.Split (splitOn)
import Data.Maybe (catMaybes)
import qualified Data.Generics as SYB
import qualified ErrUtils as Err
import qualified Finder
import qualified GhcPlugins as GHC
import HsSyn
import qualified IfaceEnv
import SrcLoc
import qualified TcRnMonad as TcM
import qualified TcRnTypes
plugin :: GHC.Plugin
plugin = GHC.defaultPlugin
{ GHC.renamedResultAction = pluginImpl
, GHC.pluginRecompile = GHC.purePlugin
}
pluginImpl
:: [GHC.CommandLineOption]
-> TcRnTypes.TcGblEnv
-> HsGroup GhcRn
-> TcRnTypes.TcM (TcRnTypes.TcGblEnv, HsGroup GhcRn)
pluginImpl args' env gr = do
dflags <- GHC.getDynFlags
topEnv <- TcM.getTopEnv
debug $ show args
debug $ GHC.showPpr dflags gr
names <- getNames dflags topEnv
opts@Options {..} <- parseArgs dflags args
when (opts == defaultOptions) $
warn dflags noSrcSpan $ GHC.text "No Overloaded features enabled"
let transformNoOp :: LHsExpr GhcRn -> Maybe (LHsExpr GhcRn)
transformNoOp _ = Nothing
trStr <- case optStrings of
NoStr -> return transformNoOp
Str Nothing -> return $ transformStrings names
Sym Nothing -> return $ transformSymbols names
Str (Just vn) -> do
n <- lookupVarName dflags topEnv vn
return $ transformStrings $ names { fromStringName = n }
Sym (Just vn) -> do
n <- lookupVarName dflags topEnv vn
return $ transformSymbols $ names { fromSymbolName = n }
trNum <- case optNumerals of
NoNum -> return transformNoOp
IsNum Nothing -> return $ transformNumerals names
IsNat Nothing -> return $ transformNaturals names
IsNum (Just vn) -> do
n <- lookupVarName dflags topEnv vn
return $ transformNumerals $ names { fromNumeralName = n }
IsNat (Just vn) -> do
n <- lookupVarName dflags topEnv vn
return $ transformNaturals $ names { fromNaturalName = n }
trChr <- case optChars of
Off -> return transformNoOp
On Nothing -> return $ transformChars names
On (Just vn) -> do
n <- lookupVarName dflags topEnv vn
return $ transformChars $ names { fromCharName = n }
trLists <- case optLists of
Off -> return transformNoOp
On Nothing -> return $ transformLists names
On (Just (V2 xn yn)) -> do
x <- lookupVarName dflags topEnv xn
y <- lookupVarName dflags topEnv yn
return $ transformLists $ names { nilName = x, consName = y }
trIf <- case optIf of
Off -> return transformNoOp
On Nothing -> return $ transformIf names
On (Just vn) -> do
n <- lookupVarName dflags topEnv vn
return $ transformIf $ names { ifteName = n }
trLabel <- case optLabels of
Off -> return transformNoOp
On Nothing -> return $ transformLabels names
On (Just vn) -> do
n <- lookupVarName dflags topEnv vn
return $ transformLabels $ names { fromLabelName = n }
let tr = trStr /\ trNum /\ trChr /\ trLists /\ trIf /\ trLabel
gr' <- transform dflags tr gr
return (env, gr')
where
args = concatMap (splitOn ":") args'
(/\) :: (a -> Maybe b) -> (a -> Maybe b) -> a -> Maybe b
f /\ g = \x -> f x <|> g x
infixr 9 /\
parseArgs :: GHC.DynFlags -> [String] -> TcRnTypes.TcM Options
parseArgs dflags = foldM go0 defaultOptions where
go0 opts arg = do
(arg', vns) <- elaborateArg arg
go opts arg' vns
go opts "Strings" vns = do
when (isSym $ optStrings opts) $ warn dflags noSrcSpan $
GHC.text "Overloaded:Strings and Overloaded:Symbols enabled"
GHC.$$
GHC.text "picking Overloaded:Strings"
mvn <- oneName "Strings" vns
return $ opts { optStrings = Str mvn }
go opts "Symbols" vns = do
when (isStr $ optStrings opts) $ warn dflags noSrcSpan $
GHC.text "Overloaded:Strings and Overloaded:Symbols enabled"
GHC.$$
GHC.text "picking Overloaded:Symbols"
mvn <- oneName "Symbols" vns
return $ opts { optStrings = Sym mvn }
go opts "Numerals" vns = do
when (isNat $ optNumerals opts) $ warn dflags noSrcSpan $
GHC.text "Overloaded:Numerals and Overloaded:Naturals enabled"
GHC.$$
GHC.text "picking Overloaded:Numerals"
mvn <- oneName "Numerals" vns
return $ opts { optNumerals = IsNum mvn }
go opts "Naturals" vns = do
when (isNum $ optNumerals opts) $ warn dflags noSrcSpan $
GHC.text "Overloaded:Numerals and Overloaded:Naturals enabled"
GHC.$$
GHC.text "picking Overloaded:Naturals"
mvn <- oneName "Naturals" vns
return $ opts { optNumerals = IsNat mvn }
go opts "Chars" vns = do
mvn <- oneName "Chars" vns
return $ opts { optChars = On mvn }
go opts "Lists" vns = do
mvn <- twoNames "Lists" vns
return $ opts { optLists = On mvn }
go opts "If" vns = do
mvn <- oneName "If" vns
return $ opts { optIf = On mvn }
go opts "Labels" vns = do
mvn <- oneName "Symbols" vns
return $ opts { optLabels = On mvn }
go opts s _ = do
warn dflags noSrcSpan $ GHC.text $ "Unknown Overloaded option " ++ show s
return opts
oneName arg vns = case vns of
[] -> return Nothing
[vn] -> return (Just vn)
(vn:_) -> do
warn dflags noSrcSpan $ GHC.text $ "Multiple desugaring names specified for " ++ arg
return (Just vn)
twoNames arg vns = case vns of
[] -> return Nothing
[_] -> do
warn dflags noSrcSpan $ GHC.text $ "Only single desugaring name specified for " ++ arg
return Nothing
[x,y] -> return (Just (V2 x y))
(x:y:_) -> do
warn dflags noSrcSpan $ GHC.text $ "Over two names specified for " ++ arg
return (Just (V2 x y))
elaborateArg :: String -> TcRnTypes.TcM (String, [VarName])
elaborateArg s = case splitOn "=" s of
[] -> return ("", [])
(pfx:xs) -> do
vns <- traverse parseVarName xs
return (pfx, catMaybes vns)
parseVarName :: String -> TcRnTypes.TcM (Maybe VarName)
parseVarName "" = return Nothing
parseVarName xs = do
let ps = splitOn "." xs
return (Just (VN (intercalate "." $ init ps) (last ps)))
data Options = Options
{ optStrings :: StrSym
, optNumerals :: NumNat
, optChars :: OnOff VarName
, optLists :: OnOff (V2 VarName)
, optIf :: OnOff VarName
, optLabels :: OnOff VarName
}
deriving (Eq, Show)
defaultOptions :: Options
defaultOptions = Options
{ optStrings = NoStr
, optNumerals = NoNum
, optChars = Off
, optLists = Off
, optIf = Off
, optLabels = Off
}
data StrSym
= NoStr
| Str (Maybe VarName)
| Sym (Maybe VarName)
deriving (Eq, Show)
isSym :: StrSym -> Bool
isSym (Sym _) = True
isSym _ = False
isStr :: StrSym -> Bool
isStr (Str _) = True
isStr _ = False
data NumNat
= NoNum
| IsNum (Maybe VarName)
| IsNat (Maybe VarName)
deriving (Eq, Show)
isNum :: NumNat -> Bool
isNum (IsNum _) = True
isNum _ = False
isNat :: NumNat -> Bool
isNat (IsNat _) = True
isNat _ = False
data OnOff a
= Off
| On (Maybe a)
deriving (Eq, Show)
transformStrings :: Names -> LHsExpr GhcRn -> Maybe (LHsExpr GhcRn)
transformStrings Names {..} e@(L l (HsLit _ (HsString _ _fs))) =
Just $ hsApps l (hsVar l fromStringName) [e]
transformStrings _ _ = Nothing
transformSymbols :: Names -> LHsExpr GhcRn -> Maybe (LHsExpr GhcRn)
transformSymbols Names {..} (L l (HsLit _ (HsString _ fs))) = do
let name' = hsVar l fromSymbolName
let inner = hsTyApp l name' (HsTyLit noExt (HsStrTy GHC.NoSourceText fs))
Just inner
transformSymbols _ _ = Nothing
transformNumerals :: Names -> LHsExpr GhcRn -> Maybe (LHsExpr GhcRn)
transformNumerals Names {..} (L l (HsOverLit _ (OverLit _ (HsIntegral (GHC.IL _ n i)) _)))
| not n, i >= 0 = do
let name' = hsVar l fromNumeralName
let inner = hsTyApp l name' (HsTyLit noExt (HsNumTy GHC.NoSourceText i))
Just inner
transformNumerals _ _ = Nothing
transformNaturals :: Names -> LHsExpr GhcRn -> Maybe (LHsExpr GhcRn)
transformNaturals Names {..} e@(L l (HsOverLit _ (OverLit _ (HsIntegral (GHC.IL _ n i)) _)))
| not n, i >= 0 = do
Just $ hsApps l (hsVar l fromNaturalName) [e]
transformNaturals _ _ = Nothing
transformChars :: Names -> LHsExpr GhcRn -> Maybe (LHsExpr GhcRn)
transformChars Names {..} e@(L l (HsLit _ (HsChar _ _))) =
Just $ hsApps l (hsVar l fromCharName) [e]
transformChars _ _ = Nothing
transformLists :: Names -> LHsExpr GhcRn -> Maybe (LHsExpr GhcRn)
transformLists Names {..} (L l (ExplicitList _ Nothing xs)) =
Just $ foldr cons' nil' xs
where
cons' :: LHsExpr GhcRn -> LHsExpr GhcRn -> LHsExpr GhcRn
cons' y ys = hsApps l (hsVar l consName) [y, ys]
nil' :: LHsExpr GhcRn
nil' = hsVar l nilName
transformLists _ _ = Nothing
transformIf :: Names -> LHsExpr GhcRn -> Maybe (LHsExpr GhcRn)
transformIf Names {..} (L l (HsIf _ _ co th el)) = Just val4 where
val4 = L l $ HsApp noExt val3 el
val3 = L l $ HsApp noExt val2 th
val2 = L l $ HsApp noExt val1 co
val1 = L l $ HsVar noExt $ L l ifteName
transformIf _ _ = Nothing
transformLabels :: Names -> LHsExpr GhcRn -> Maybe (LHsExpr GhcRn)
transformLabels Names {..} (L l (HsOverLabel _ Nothing fs)) = do
let name' = hsVar l fromLabelName
let inner = hsTyApp l name' (HsTyLit noExt (HsStrTy GHC.NoSourceText fs))
Just inner
transformLabels _ _ = Nothing
transform
:: GHC.DynFlags
-> (LHsExpr GhcRn -> Maybe (LHsExpr GhcRn))
-> HsGroup GhcRn
-> TcRnTypes.TcM (HsGroup GhcRn)
transform _dflags f = SYB.everywhereM (SYB.mkM transform') where
transform' :: LHsExpr GhcRn -> TcRnTypes.TcM (LHsExpr GhcRn)
transform' e = do
return $ case f e of
Just e' -> e'
Nothing -> e
hsVar :: SrcSpan -> GHC.Name -> LHsExpr GhcRn
hsVar l n = L l (HsVar noExt (L l n))
hsApps :: SrcSpan -> LHsExpr GhcRn -> [LHsExpr GhcRn] -> LHsExpr GhcRn
hsApps l = foldl' app where
app :: LHsExpr GhcRn -> LHsExpr GhcRn -> LHsExpr GhcRn
app f x = L l (HsApp noExt f x)
hsTyApp :: SrcSpan -> LHsExpr GhcRn -> HsType GhcRn -> LHsExpr GhcRn
#if MIN_VERSION_ghc(8,8,0)
hsTyApp l x ty = L l $ HsAppType noExt x (HsWC [] (L l ty))
#else
hsTyApp l x ty = L l $ HsAppType (HsWC [] (L l ty)) x
#endif
dataStringMN :: GHC.ModuleName
dataStringMN = GHC.mkModuleName "Data.String"
overloadedCharsMN :: GHC.ModuleName
overloadedCharsMN = GHC.mkModuleName "Overloaded.Chars"
overloadedSymbolsMN :: GHC.ModuleName
overloadedSymbolsMN = GHC.mkModuleName "Overloaded.Symbols"
overloadedNaturalsMN :: GHC.ModuleName
overloadedNaturalsMN = GHC.mkModuleName "Overloaded.Naturals"
overloadedNumeralsMN :: GHC.ModuleName
overloadedNumeralsMN = GHC.mkModuleName "Overloaded.Numerals"
overloadedListsMN :: GHC.ModuleName
overloadedListsMN = GHC.mkModuleName "Overloaded.Lists"
overloadedIfMN :: GHC.ModuleName
overloadedIfMN = GHC.mkModuleName "Overloaded.If"
ghcOverloadedLabelsMN :: GHC.ModuleName
ghcOverloadedLabelsMN = GHC.mkModuleName "GHC.OverloadedLabels"
data Names = Names
{ fromStringName :: GHC.Name
, fromSymbolName :: GHC.Name
, fromNumeralName :: GHC.Name
, fromNaturalName :: GHC.Name
, fromCharName :: GHC.Name
, nilName :: GHC.Name
, consName :: GHC.Name
, ifteName :: GHC.Name
, fromLabelName :: GHC.Name
}
getNames :: GHC.DynFlags -> GHC.HscEnv -> TcRnTypes.TcM Names
getNames dflags env = do
fromStringName <- lookupName dflags env dataStringMN "fromString"
fromSymbolName <- lookupName dflags env overloadedSymbolsMN "fromSymbol"
fromNumeralName <- lookupName dflags env overloadedNumeralsMN "fromNumeral"
fromNaturalName <- lookupName dflags env overloadedNaturalsMN "fromNatural"
fromCharName <- lookupName dflags env overloadedCharsMN "fromChar"
nilName <- lookupName dflags env overloadedListsMN "nil"
consName <- lookupName dflags env overloadedListsMN "cons"
ifteName <- lookupName dflags env overloadedIfMN "ifte"
fromLabelName <- lookupName dflags env ghcOverloadedLabelsMN "fromLabel"
return Names {..}
lookupName :: GHC.DynFlags -> GHC.HscEnv -> GHC.ModuleName -> String -> TcM.TcM GHC.Name
lookupName dflags env mn vn = do
res <- liftIO $ Finder.findImportedModule env mn Nothing
case res of
GHC.Found _ md -> IfaceEnv.lookupOrig md (GHC.mkVarOcc vn)
_ -> do
liftIO $ GHC.putLogMsg dflags GHC.NoReason Err.SevError noSrcSpan (GHC.defaultErrStyle dflags) $
GHC.text "Cannot find module" GHC.<+> GHC.ppr mn
fail "panic!"
data VarName = VN String String
deriving (Eq, Show)
lookupVarName :: GHC.DynFlags -> GHC.HscEnv -> VarName -> TcM.TcM GHC.Name
lookupVarName dflags env (VN vn mn) = lookupName dflags env (GHC.mkModuleName vn) mn
warn :: MonadIO m => GHC.DynFlags -> SrcSpan -> GHC.SDoc -> m ()
warn dflags l doc =
liftIO $ GHC.putLogMsg dflags GHC.NoReason Err.SevWarning l (GHC.defaultErrStyle dflags) doc
debug :: MonadIO m => String -> m ()
debug _ = pure ()
data V2 a = V2 a a
deriving (Eq, Show)