module Main (main) where import Data.List import Control.Error import Control.Arrow import Control.Applicative import System.Environment (getArgs) import Language.Haskell.Parser import Language.Haskell.Syntax import Data.Text.Lazy.Builder (toLazyText) import qualified Data.Text.Lazy.IO as TL import qualified Data.Map as M import Records import MustacheTemplates data Type = TInt | TUTCTime | TDouble | TString | TText | TLText deriving (Show, Ord, Eq) data RType = RPure Type | RIOUnit | RIO Type deriving (Show, Eq) importPrefix :: HsImportDecl -> String importPrefix (HsImportDecl {importQualified = True, importAs = Just (Module m)}) = m importPrefix (HsImportDecl {importQualified = True, importAs = Nothing, importModule = (Module m)}) = m importPrefix _ = "" doParse :: String -> (String, [(String, Module)], [HsDecl]) doParse moduleSrc = (mod, importEnv, decls) where importEnv = map (importPrefix &&& importModule) imports ParseOk (HsModule _ (Module mod) _ imports decls) = parseModule moduleSrc dataDeclNamed :: HsName -> HsDecl -> Bool dataDeclNamed n (HsDataDecl _ _ name _ _ _) | n == name = True dataDeclNamed _ _ = False extractTypeFromBangType :: HsBangType -> HsType extractTypeFromBangType (HsBangedTy t) = t extractTypeFromBangType (HsUnBangedTy t) = t findType :: [(String, Module)] -> Module -> [(Module, Type)] -> Either String Type findType i (Module m) q = headErr ("No " ++ show q ++ " imports find, but one used as type.") matches where matches = mapMaybe (`lookup` q) $ filter (`elem` moduls) ns moduls = map fst q ns = map snd $ filter ((==m).fst) i findTextType :: [(String, Module)] -> Module -> Either String Type findTextType i m = findType i m [ (Module "Data.Text", TText), (Module "Data.Text.Lazy", TLText) ] findUTCTimeType :: [(String, Module)] -> Module -> Either String Type findUTCTimeType i m = findType i m [ (Module "Data.Time", TUTCTime), (Module "Data.Time.Clock", TUTCTime) ] resolveType :: [(String, Module)] -> HsType -> Either String Type resolveType _ (HsTyCon (UnQual (HsIdent "Int"))) = Right TInt resolveType _ (HsTyCon (UnQual (HsIdent "Double"))) = Right TDouble resolveType _ (HsTyCon (UnQual (HsIdent "String"))) = Right TString resolveType i (HsTyCon (UnQual (HsIdent "Text"))) = findTextType i (Module "") resolveType i (HsTyCon (Qual m (HsIdent "Text"))) = findTextType i m resolveType i (HsTyCon (UnQual (HsIdent "UTCTime"))) = findUTCTimeType i (Module "") resolveType i (HsTyCon (Qual m (HsIdent "UTCTime"))) = findUTCTimeType i m resolveType _ t = Left $ "Type not yet supported: " ++ show t flattenSum :: HsConDecl -> Either String (String, [HsType]) flattenSum (HsConDecl _ (HsIdent n) ts) = Right (n, map extractTypeFromBangType ts) flattenSum (HsRecDecl _ (HsIdent n) ts) = Right (n, map (extractTypeFromBangType.snd) ts) flattenSum x = Left $ "Cannot process sum type constructor: " ++ show x getSignals :: String -> [(String, Module)] -> [HsDecl] -> Either String [(String, [Type])] getSignals dataName i decls = signalDecl >>= signalsFrom >>= mapM signalConstr where signalConstr (n, ts) = (,) <$> pure n <*> mapM (resolveType i) ts signalsFrom (HsDataDecl _ _ _ _ cons _) = mapM flattenSum cons signalsFrom x = Left $ "Could not process signal near: " ++ show x signalDecl = note "No sum type named \"SignalToUI\" found." $ find (dataDeclNamed (HsIdent dataName)) decls mapQtType :: Type -> String mapQtType TInt = "int" mapQtType TUTCTime = "QDateTime" mapQtType TDouble = "double" mapQtType TString = "QString" mapQtType TText = "QString" mapQtType TLText = "QString" mapLowCType :: Type -> String mapLowCType TInt = "int" mapLowCType TUTCTime = "unsigned int" -- Not time_t because of Qt mapLowCType TDouble = "double" mapLowCType TString = "char *" mapLowCType TText = "char *" mapLowCType TLText = "char *" destroyLowCArg :: Type -> String destroyLowCArg TInt = "" destroyLowCArg TUTCTime = "" destroyLowCArg TDouble = "" destroyLowCArg TString = "free" destroyLowCArg TText = "free" destroyLowCArg TLText = "free" mapCType :: Type -> String mapCType TInt = "CInt" mapCType TUTCTime = "CUInt" -- Not CTime because of Qt mapCType TDouble = "CDouble" mapCType TString = "CString" mapCType TText = "CString" mapCType TLText = "CString" wrapTypeFromC :: Type -> String wrapTypeFromC TInt = "fmap (fromIntegral :: CInt -> Int)" wrapTypeFromC TUTCTime = "fmap (posixSecondsToUTCTime . realToFrac :: CUInt -> UTCTime)" wrapTypeFromC TDouble = "fmap (realToFrac :: CDouble -> Double)" wrapTypeFromC TString = "fmap (Text.unpack . Text.decodeUtf8) . (ByteString.packCString =<<)" wrapTypeFromC TText = "fmap Text.decodeUtf8 . (ByteString.packCString =<<)" wrapTypeFromC TLText = "fmap (LText.fromStrict . Text.decodeUtf8) . (ByteString.packCString =<<)" wrapTypeToC :: Type -> String -> String wrapTypeToC TInt arg = "(flip ($) (fromIntegral " ++ arg ++ "))" wrapTypeToC TUTCTime arg = "(flip ($) (fromInteger $ floor $ utcTimeToPOSIXSeconds " ++ arg ++ "))" wrapTypeToC TDouble arg = "(flip ($) (realToFrac " ++ arg ++ "))" wrapTypeToC TString arg = "(ByteString.useAsCString (Text.encodeUtf8 $ Text.pack " ++ arg ++ "))" wrapTypeToC TText arg = "(ByteString.useAsCString (Text.encodeUtf8 " ++ arg ++ "))" wrapTypeToC TLText arg = "(ByteString.useAsCString (Text.encodeUtf8 $ LText.toStrict " ++ arg ++ "))" wrapTypeFromQt :: Type -> String -> String wrapTypeFromQt TUTCTime arg = "(" ++ arg ++ ").toTime_t()" wrapTypeFromQt TString arg = "qstrdup((" ++ arg ++ ").toUtf8().constData())" wrapTypeFromQt TText arg = "qstrdup((" ++ arg ++ ").toUtf8().constData())" wrapTypeFromQt TLText arg = "qstrdup((" ++ arg ++ ").toUtf8().constData())" wrapTypeFromQt _ arg = "(" ++ arg ++ ")" wrapTypeToQt :: Type -> String -> String wrapTypeToQt TUTCTime arg = "(QDateTime::fromTime_t(" ++ arg ++ "))" wrapTypeToQt TString arg = "(QString::fromUtf8(" ++ arg ++ "))" wrapTypeToQt TText arg = "(QString::fromUtf8(" ++ arg ++ "))" wrapTypeToQt TLText arg = "(QString::fromUtf8(" ++ arg ++ "))" wrapTypeToQt _ arg = "(" ++ arg ++ ")" defTypeToC :: Type -> String defTypeToC TInt = "(flip ($) 0)" defTypeToC TUTCTime = "(flip ($) 0)" defTypeToC TDouble = "(flip ($) 0)" defTypeToC TString = "(flip ($) nullPtr)" defTypeToC TText = "(flip ($) nullPtr)" defTypeToC TLText = "(flip ($) nullPtr)" sigCWrap' :: [(Int, Type)] -> [(Int, Type)] -> ([(Int, Int)], [String]) sigCWrap' [] _ = ([], []) sigCWrap' ((ei,e):es) ts | null sameType = -- No more of this type in this signal second (("(.) " ++ defTypeToC e) :) (sigCWrap' es ts) | otherwise = -- Use up one of this type let (idxs, r) = sigCWrap' es (tail sameType ++ rest) in ((i, ei) : idxs, ("(.) " ++ wrapTypeToC e ("arg" ++ show i)) : r) where (i, _) = head sameType (sameType, rest) = partition ((==e).snd) ts sigCWrap :: [Type] -> [(Int, Type)] -> ([(Int, Int)], String) sigCWrap es ts = (,) idxs $ case wraps of [] -> "" [x] -> drop 4 x (x:xs) -> intercalate " $ (.) $ " (reverse xs) ++ " $ " ++ x where (idxs, wraps) = second reverse $ sigCWrap' (zip [0..] es) ts templateSignal :: [Type] -> Integer -> (String, [Type]) -> Signal templateSignal signalTypes i (name, ts) = Signal { sigfirst = i == 1, -- This must be set to the initial eventid provided by main signame = name, sigargs = zipWith (\i t -> SignalArg { sigargfirst = i == 0, siganame = "arg" ++ show i, qtsigargtype = mapQtType t, csigargtype = mapLowCType t, sigargfromc = wrapTypeFromC t, sigargfromqt = wrapTypeFromQt t ("arg" ++ show i), sigargdestroy = destroyLowCArg t, sigargsigtypename = let Just x = lookup i idxs in "arg" ++ show x }) [0..] ts, sigevent = i, sigcwrap = cwrap } where (idxs, cwrap) = sigCWrap signalTypes $ zip [0..] ts parseArgs :: [String] -> Either String [String] parseArgs [a,b,c] = return [a,b,c] parseArgs _ = Left "Usage: haskades HaskadesBinding.hsc haskades_run.cpp haskades_run.h < Types.hs" main :: IO () main = runScript $ do [hsout, cppout, headerPath] <- hoistEither . parseArgs =<< scriptIO getArgs (mod, importEnv, decls) <- fmap doParse (scriptIO getContents) toUI <- hoistEither $ getSignals "SignalToUI" importEnv decls let toUItypes = concatMap (\(t,c) -> replicate c t) $ M.toAscList $ M.unionsWith max $ map (M.fromListWith (+) . (`zip` [1,1..]) . snd) toUI let toUItmpl = zipWith (templateSignal toUItypes) [1..] toUI let toUItypeTmpl = zipWith (\i t -> SignalType { csigtype = mapCType t, qtsigtype = mapQtType t, qtwrapsig = wrapTypeToQt t ("arg" ++ show i), lowcsigtype = mapLowCType t, sigtypename = "arg" ++ show i }) [(0::Int)..] toUItypes fromUI <- hoistEither $ getSignals "SignalFromUI" importEnv decls let fromUItmpl = zipWith (templateSignal []) [1..] fromUI let template = Template { headerPath = headerPath, modul = mod, toUItypes = toUItypeTmpl, toUI = toUItmpl, fromUI = fromUItmpl } scriptIO $ TL.writeFile hsout $ toLazyText $ haskadesBindinghsc id template scriptIO $ TL.writeFile cppout $ toLazyText $ haskades_runcpp id template scriptIO $ TL.writeFile headerPath $ toLazyText $ haskades_runh id template