diff options
| author | Patrick Günther <info@paddybu.de> | 2022-04-16 09:46:55 +0200 | 
|---|---|---|
| committer | Patrick Günther <info@paddybu.de> | 2022-04-16 09:46:55 +0200 | 
| commit | d1b9edff080b0de64c5445c2ecfa833cd60d18af (patch) | |
| tree | 316f71d2c8114fdd638d96db1d4bba39b87af42f | |
| parent | 727478f5b8916d8d98ae4208d4f6a80abb4fafc7 (diff) | |
| download | xmobar-d1b9edff080b0de64c5445c2ecfa833cd60d18af.tar.gz xmobar-d1b9edff080b0de64c5445c2ecfa833cd60d18af.tar.bz2 | |
Fixed unit of battery power consumption
| -rw-r--r-- | src/Xmobar/Plugins/Monitors/Batt/Linux.hs | 78 | 
1 files changed, 52 insertions, 26 deletions
| diff --git a/src/Xmobar/Plugins/Monitors/Batt/Linux.hs b/src/Xmobar/Plugins/Monitors/Batt/Linux.hs index 454e7e9..362c529 100644 --- a/src/Xmobar/Plugins/Monitors/Batt/Linux.hs +++ b/src/Xmobar/Plugins/Monitors/Batt/Linux.hs @@ -23,7 +23,7 @@ import Xmobar.Plugins.Monitors.Batt.Common (BattOpts(..)  import Control.Monad (unless)  import Control.Exception (SomeException, handle)  import System.FilePath ((</>)) -import System.IO (IOMode(ReadMode), hGetLine, withFile) +import System.IO (IOMode(ReadMode), hGetLine, withFile, Handle)  import System.Posix.Files (fileExist)  import Data.List (sort, sortBy, group)  import Data.Maybe (fromMaybe) @@ -34,9 +34,10 @@ data Files = Files    { fFull :: String    , fNow :: String    , fVoltage :: String -  , fCurrent :: String +  , fVoltageMin :: String +  , fCurrent :: Maybe String +  , fPower :: Maybe String    , fStatus :: String -  , isCurrent :: Bool    } | NoFiles deriving Eq  data Battery = Battery @@ -57,45 +58,70 @@ batteryFiles :: String -> IO Files  batteryFiles bat =    do is_charge <- exists "charge_now"       is_energy <- if is_charge then return False else exists "energy_now" -     is_power <- exists "power_now"       plain <- exists (if is_charge then "charge_full" else "energy_full") -     let cf = if is_power then "power_now" else "current_now" +     has_power <- exists powerNowPath +     has_current <- exists currentNowPath +     let pf = if has_power then Just powerNowPath else Nothing +         cf = if has_current then Just currentNowPath else Nothing           sf = if plain then "" else "_design"       return $ case (is_charge, is_energy) of -       (True, _) -> files "charge" cf sf is_power -       (_, True) -> files "energy" cf sf is_power +       (True, _) -> files "charge" cf pf sf  +       (_, True) -> files "energy" cf pf sf         _ -> NoFiles    where prefix = sysDir </> bat +        justPrefix a = Just $ prefix </> a          exists = safeFileExist prefix -        files ch cf sf ip = Files { fFull = prefix </> ch ++ "_full" ++ sf +        files ch cf pf sf = Files { fFull = prefix </> ch ++ "_full" ++ sf                                    , fNow = prefix </> ch ++ "_now" -                                  , fCurrent = prefix </> cf +                                  , fCurrent = maybe Nothing justPrefix cf +                                  , fPower = maybe Nothing justPrefix pf                                    , fVoltage = prefix </> "voltage_now" -                                  , fStatus = prefix </> "status" -                                  , isCurrent = not ip} +                                  , fVoltageMin = prefix </> "voltage_min_design" +                                  , fStatus = prefix </> "status"} +        currentNowPath = "current_now" +        powerNowPath = "power_now"  haveAc :: FilePath -> IO Bool  haveAc f = -  handle onError $ withFile (sysDir </> f) ReadMode (fmap (== "1") . hGetLine) -  where onError = const (return False) :: SomeException -> IO Bool +  handle (onError False) $ withFile (sysDir </> f) ReadMode (fmap (== "1") . hGetLine) + +readBatPower :: Float -> Files -> IO Float + +readBatPower sc (Files {fPower = Just p}) =  +    do valp <- grabNumber p +       return $ valp / sc + +readBatPower sc (Files {fPower = Nothing, fCurrent = Just c, fVoltage = v}) =  +    do valv <- grabNumber v +       valc <- grabNumber c +       return $ valc * valv / (sc * sc) + +readBatPower _ _ = do return 999  readBattery :: Float -> Files -> IO Battery  readBattery _ NoFiles = return $ Battery 0 0 0 "Unknown"  readBattery sc files = -    do a <- grab $ fFull files -       b <- grab $ fNow files -       d <- grab $ fCurrent files -       s <- grabs $ fStatus files -       let sc' = if isCurrent files then sc / 10 else sc -           a' = max a b -- sometimes the reported max charge is lower than -       return $ Battery (3600 * a' / sc') -- wattseconds -                        (3600 * b / sc') -- wattseconds -                        (abs d / sc') -- watts +    do a <- grabNumber $ fFull files +       b <- grabNumber $ fNow files +       p <- readBatPower sc files +       s <- grabString $ fStatus files +       let a' = max a b -- sometimes the reported max charge is lower than +       return $ Battery (3600 * a' / sc) -- wattseconds +                        (3600 * b / sc) -- wattseconds +                        (abs p) -- watts                          s -- string: Discharging/Charging/Full -    where grab f = handle onError $ withFile f ReadMode (fmap read . hGetLine) -          onError = const (return (-1)) :: SomeException -> IO Float -          grabs f = handle onError' $ withFile f ReadMode hGetLine -          onError' = const (return "Unknown") :: SomeException -> IO String + +grabNumber :: (Num a, Read a) => FilePath -> IO a +grabNumber = grabFile (-1) (fmap read . hGetLine) + +grabString :: FilePath -> IO String +grabString = grabFile "Unknown" hGetLine + +grabFile :: a -> (Handle -> IO a) -> FilePath -> IO a +grabFile returnOnError readMode f = handle (onFileError returnOnError) $ withFile f ReadMode readMode + +onFileError :: a -> SomeException -> IO a +onFileError returnOnError = const (return returnOnError)   -- sortOn is only available starting at ghc 7.10  sortOn :: Ord b => (a -> b) -> [a] -> [a] | 
