{-# LANGUAGE BangPatterns, CPP #-} module System.Log.FastLogger ( -- * Creating a logger set BufSize , defaultBufSize , LoggerSet , newLoggerSet , renewLoggerSet -- * Removing a logger set , rmLoggerSet -- * Log messages , LogStr , ToLogStr(..) , fromLogStr , logStrLength -- * Writing a log message , pushLogStr -- * Flushing buffered log messages , flushLogStr -- * File rotation , module System.Log.FastLogger.File ) where import Control.Applicative ((<$>)) import Control.Concurrent (getNumCapabilities, myThreadId, threadCapability, takeMVar) import Control.Monad (when, replicateM) import Data.Array (Array, listArray, (!)) import Data.Maybe (isJust) import GHC.IO.Device (close) import GHC.IO.FD (FD(..), openFile, stdout) import GHC.IO.IOMode (IOMode(..)) import System.Log.FastLogger.File import System.Log.FastLogger.IO import System.Log.FastLogger.IORef import System.Log.FastLogger.LogStr import System.Log.FastLogger.Logger ---------------------------------------------------------------- -- | Opening a log file. logOpen :: FilePath -> IO FD logOpen file = fst <$> openFile file AppendMode False ---------------------------------------------------------------- -- | A set of loggers. -- The number of loggers is the capabilities of GHC RTS. -- You can specify it with \"+RTS -N\\". -- A buffer is prepared for each capability. data LoggerSet = LoggerSet (Maybe FilePath) (IORef FD) (Array Int Logger) -- | Creating a new 'LoggerSet'. -- If 'Nothing' is specified to the second argument, -- stdout is used. newLoggerSet :: BufSize -> Maybe FilePath -> IO LoggerSet newLoggerSet size mfile = do fd <- case mfile of Nothing -> return stdout Just file -> logOpen file n <- getNumCapabilities loggers <- replicateM n $ newLogger size let arr = listArray (0,n-1) loggers fref <- newIORef fd return $ LoggerSet mfile fref arr -- | Writing a log message to the corresponding buffer. pushLogStr :: LoggerSet -> LogStr -> IO () pushLogStr (LoggerSet _ fref arr) logmsg = do (i, _) <- myThreadId >>= threadCapability let logger = arr ! i fd <- readIORef fref pushLog fd logger logmsg -- | Flushing log messages in buffers. flushLogStr :: LoggerSet -> IO () flushLogStr (LoggerSet _ fref arr) = do n <- getNumCapabilities fd <- readIORef fref mapM_ (flushIt fd) [0..n-1] where flushIt fd i = flushLog fd (arr ! i) -- | Renewing the internal file information in 'LoggerSet'. renewLoggerSet :: LoggerSet -> IO () renewLoggerSet (LoggerSet Nothing _ _) = return () renewLoggerSet (LoggerSet (Just file) fref _) = do newfd <- logOpen file oldfd <- atomicModifyIORef' fref (\fd -> (newfd, fd)) close oldfd -- | Flushing the buffers, closing the internal file information -- and freeing the buffers. rmLoggerSet :: LoggerSet -> IO () rmLoggerSet (LoggerSet mfile fref arr) = do n <- getNumCapabilities fd <- readIORef fref let nums = [0..n-1] mapM_ (flushIt fd) nums mapM_ freeIt nums when (isJust mfile) $ close fd where flushIt fd i = flushLog fd (arr ! i) freeIt i = do let (Logger mbuf _ _) = arr ! i takeMVar mbuf >>= freeBuffer