{-# LANGUAGE CPP #-}
{-# LANGUAGE FlexibleContexts #-}
{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE GADTs #-}
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE TemplateHaskell #-}
{-# LANGUAGE UndecidableInstances #-}
{-# OPTIONS_GHC -fno-warn-orphans #-}
{-# OPTIONS_HADDOCK hide #-}
module Data.Array.Accelerate.Pretty (
PrettyAcc, ExtractAcc,
prettyPreOpenAcc,
prettyPreOpenAfun,
prettyOpenExp,
prettyOpenFun,
Graph,
PrettyGraph(..), Detail(..),
graphDelayedAcc, graphDelayedAfun,
) where
import Data.Array.Accelerate.AST hiding ( Acc, Exp )
import Data.Array.Accelerate.Debug.Internal.Flags
import Data.Array.Accelerate.Error
import Data.Array.Accelerate.Pretty.Graphviz
import Data.Array.Accelerate.Pretty.Print hiding ( Keyword(..) )
import Data.Array.Accelerate.Smart ( Acc, Exp )
import Data.Array.Accelerate.Sugar.Array
import Data.Array.Accelerate.Sugar.Elt
import Data.Array.Accelerate.Trafo
import Data.Array.Accelerate.Trafo.Delayed
import Data.Maybe
import Prettyprinter
import Prettyprinter.Render.String
import Prettyprinter.Render.Terminal
import System.Environment
import System.IO
import System.IO.Unsafe
import qualified Data.Text.Lazy as T
import qualified System.Console.ANSI as Term
import qualified System.Console.Terminal.Size as Term
#if ACCELERATE_DEBUG
import Control.DeepSeq
import Data.Array.Accelerate.Debug.Internal.Stats
#endif
instance Arrays arrs => Show (Acc arrs) where
show :: Acc arrs -> String
show = ShowS
withSimplStats ShowS -> (Acc arrs -> String) -> Acc arrs -> String
forall b c a. (b -> c) -> (a -> b) -> a -> c
. DelayedAcc (ArraysR arrs) -> String
forall a. Show a => a -> String
show (DelayedAcc (ArraysR arrs) -> String)
-> (Acc arrs -> DelayedAcc (ArraysR arrs)) -> Acc arrs -> String
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Acc arrs -> DelayedAcc (ArraysR arrs)
forall arrs. Acc arrs -> DelayedAcc (ArraysR arrs)
convertAcc
instance Afunction (Acc a -> f) => Show (Acc a -> f) where
show :: (Acc a -> f) -> String
show = ShowS
withSimplStats ShowS -> ((Acc a -> f) -> String) -> (Acc a -> f) -> String
forall b c a. (b -> c) -> (a -> b) -> a -> c
. DelayedAfun (ArraysR a -> ArraysFunctionR f) -> String
forall a. Show a => a -> String
show (DelayedAfun (ArraysR a -> ArraysFunctionR f) -> String)
-> ((Acc a -> f) -> DelayedAfun (ArraysR a -> ArraysFunctionR f))
-> (Acc a -> f)
-> String
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Acc a -> f) -> DelayedAfun (ArraysFunctionR (Acc a -> f))
(Acc a -> f) -> DelayedAfun (ArraysR a -> ArraysFunctionR f)
forall f. Afunction f => f -> DelayedAfun (ArraysFunctionR f)
convertAfun
instance Elt e => Show (Exp e) where
show :: Exp e -> String
show = ShowS
withSimplStats ShowS -> (Exp e -> String) -> Exp e -> String
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Exp () (EltR e) -> String
forall a. Show a => a -> String
show (Exp () (EltR e) -> String)
-> (Exp e -> Exp () (EltR e)) -> Exp e -> String
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Exp e -> Exp () (EltR e)
forall e. Exp e -> Exp () (EltR e)
convertExp
instance Function (Exp a -> f) => Show (Exp a -> f) where
show :: (Exp a -> f) -> String
show = ShowS
withSimplStats ShowS -> ((Exp a -> f) -> String) -> (Exp a -> f) -> String
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Fun () (EltR a -> EltFunctionR f) -> String
forall a. Show a => a -> String
show (Fun () (EltR a -> EltFunctionR f) -> String)
-> ((Exp a -> f) -> Fun () (EltR a -> EltFunctionR f))
-> (Exp a -> f)
-> String
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Exp a -> f) -> Fun () (EltFunctionR (Exp a -> f))
(Exp a -> f) -> Fun () (EltR a -> EltFunctionR f)
forall f. Function f => f -> Fun () (EltFunctionR f)
convertFun
instance PrettyEnv aenv => Show (OpenAcc aenv a) where
show :: OpenAcc aenv a -> String
show = Adoc -> String
renderForTerminal (Adoc -> String)
-> (OpenAcc aenv a -> Adoc) -> OpenAcc aenv a -> String
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PrettyConfig OpenAcc
-> Context -> Val aenv -> OpenAcc aenv a -> Adoc
PrettyAcc OpenAcc
prettyOpenAcc PrettyConfig OpenAcc
forall (acc :: * -> * -> *). PrettyConfig acc
configPlain Context
context0 (Adoc -> Val aenv
forall env. PrettyEnv env => Adoc -> Val env
prettyEnv (Char -> Adoc
forall ann. Char -> Doc ann
forall a ann. Pretty a => a -> Doc ann
pretty Char
'a'))
instance PrettyEnv aenv => Show (OpenAfun aenv f) where
show :: OpenAfun aenv f -> String
show = Adoc -> String
renderForTerminal (Adoc -> String)
-> (OpenAfun aenv f -> Adoc) -> OpenAfun aenv f -> String
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PrettyConfig OpenAcc
-> PrettyAcc OpenAcc -> Val aenv -> OpenAfun aenv f -> Adoc
forall (acc :: * -> * -> *) aenv f.
PrettyConfig acc
-> PrettyAcc acc -> Val aenv -> PreOpenAfun acc aenv f -> Adoc
prettyPreOpenAfun PrettyConfig OpenAcc
forall (acc :: * -> * -> *). PrettyConfig acc
configPlain PrettyConfig OpenAcc
-> Context -> Val aenv -> OpenAcc aenv a -> Adoc
PrettyAcc OpenAcc
prettyOpenAcc (Adoc -> Val aenv
forall env. PrettyEnv env => Adoc -> Val env
prettyEnv (Char -> Adoc
forall ann. Char -> Doc ann
forall a ann. Pretty a => a -> Doc ann
pretty Char
'a'))
instance PrettyEnv aenv => Show (DelayedOpenAcc aenv a) where
show :: DelayedOpenAcc aenv a -> String
show = let config :: PrettyConfig DelayedOpenAcc
config = if Bool
shouldPrintHash then PrettyConfig DelayedOpenAcc
configWithHash else PrettyConfig DelayedOpenAcc
forall (acc :: * -> * -> *). PrettyConfig acc
configPlain
in Adoc -> String
renderForTerminal (Adoc -> String)
-> (DelayedOpenAcc aenv a -> Adoc)
-> DelayedOpenAcc aenv a
-> String
forall b c a. (b -> c) -> (a -> b) -> a -> c
. HasCallStack => PrettyAcc DelayedOpenAcc
PrettyConfig DelayedOpenAcc
-> Context -> Val aenv -> DelayedOpenAcc aenv a -> Adoc
PrettyAcc DelayedOpenAcc
prettyDelayedOpenAcc PrettyConfig DelayedOpenAcc
config Context
context0 (Adoc -> Val aenv
forall env. PrettyEnv env => Adoc -> Val env
prettyEnv (Char -> Adoc
forall ann. Char -> Doc ann
forall a ann. Pretty a => a -> Doc ann
pretty Char
'a'))
instance PrettyEnv aenv => Show (DelayedOpenAfun aenv f) where
show :: DelayedOpenAfun aenv f -> String
show = let config :: PrettyConfig DelayedOpenAcc
config = if Bool
shouldPrintHash then PrettyConfig DelayedOpenAcc
configWithHash else PrettyConfig DelayedOpenAcc
forall (acc :: * -> * -> *). PrettyConfig acc
configPlain
in Adoc -> String
renderForTerminal (Adoc -> String)
-> (DelayedOpenAfun aenv f -> Adoc)
-> DelayedOpenAfun aenv f
-> String
forall b c a. (b -> c) -> (a -> b) -> a -> c
. PrettyConfig DelayedOpenAcc
-> PrettyAcc DelayedOpenAcc
-> Val aenv
-> DelayedOpenAfun aenv f
-> Adoc
forall (acc :: * -> * -> *) aenv f.
PrettyConfig acc
-> PrettyAcc acc -> Val aenv -> PreOpenAfun acc aenv f -> Adoc
prettyPreOpenAfun PrettyConfig DelayedOpenAcc
config HasCallStack => PrettyAcc DelayedOpenAcc
PrettyConfig DelayedOpenAcc
-> Context -> Val aenv -> DelayedOpenAcc aenv a -> Adoc
PrettyAcc DelayedOpenAcc
prettyDelayedOpenAcc (Adoc -> Val aenv
forall env. PrettyEnv env => Adoc -> Val env
prettyEnv (Char -> Adoc
forall ann. Char -> Doc ann
forall a ann. Pretty a => a -> Doc ann
pretty Char
'a'))
instance (PrettyEnv env, PrettyEnv aenv) => Show (OpenExp env aenv e) where
show :: OpenExp env aenv e -> String
show = Adoc -> String
renderForTerminal (Adoc -> String)
-> (OpenExp env aenv e -> Adoc) -> OpenExp env aenv e -> String
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Context -> Val env -> Val aenv -> OpenExp env aenv e -> Adoc
forall env aenv t.
Context -> Val env -> Val aenv -> OpenExp env aenv t -> Adoc
prettyOpenExp Context
context0 (Adoc -> Val env
forall env. PrettyEnv env => Adoc -> Val env
prettyEnv (Char -> Adoc
forall ann. Char -> Doc ann
forall a ann. Pretty a => a -> Doc ann
pretty Char
'x')) (Adoc -> Val aenv
forall env. PrettyEnv env => Adoc -> Val env
prettyEnv (Char -> Adoc
forall ann. Char -> Doc ann
forall a ann. Pretty a => a -> Doc ann
pretty Char
'a'))
instance (PrettyEnv env, PrettyEnv aenv) => Show (OpenFun env aenv e) where
show :: OpenFun env aenv e -> String
show = Adoc -> String
renderForTerminal (Adoc -> String)
-> (OpenFun env aenv e -> Adoc) -> OpenFun env aenv e -> String
forall b c a. (b -> c) -> (a -> b) -> a -> c
. Val env -> Val aenv -> OpenFun env aenv e -> Adoc
forall env aenv f.
Val env -> Val aenv -> OpenFun env aenv f -> Adoc
prettyOpenFun (Adoc -> Val env
forall env. PrettyEnv env => Adoc -> Val env
prettyEnv (Char -> Adoc
forall ann. Char -> Doc ann
forall a ann. Pretty a => a -> Doc ann
pretty Char
'x')) (Adoc -> Val aenv
forall env. PrettyEnv env => Adoc -> Val env
prettyEnv (Char -> Adoc
forall ann. Char -> Doc ann
forall a ann. Pretty a => a -> Doc ann
pretty Char
'a'))
renderForTerminal :: Adoc -> String
renderForTerminal :: Adoc -> String
renderForTerminal = SimpleDocStream Keyword -> String
render (SimpleDocStream Keyword -> String)
-> (Adoc -> SimpleDocStream Keyword) -> Adoc -> String
forall b c a. (b -> c) -> (a -> b) -> a -> c
. LayoutOptions -> Adoc -> SimpleDocStream Keyword
forall ann. LayoutOptions -> Doc ann -> SimpleDocStream ann
layoutSmart LayoutOptions
terminalLayoutOptions
where
fancy :: Bool
fancy = Bool
terminalSupportsANSI Bool -> Bool -> Bool
&& Bool
terminalColourAllowed
render :: SimpleDocStream Keyword -> String
render
| Bool
fancy = Text -> String
T.unpack (Text -> String)
-> (SimpleDocStream Keyword -> Text)
-> SimpleDocStream Keyword
-> String
forall b c a. (b -> c) -> (a -> b) -> a -> c
. SimpleDocStream AnsiStyle -> Text
renderLazy (SimpleDocStream AnsiStyle -> Text)
-> (SimpleDocStream Keyword -> SimpleDocStream AnsiStyle)
-> SimpleDocStream Keyword
-> Text
forall b c a. (b -> c) -> (a -> b) -> a -> c
. (Keyword -> AnsiStyle)
-> SimpleDocStream Keyword -> SimpleDocStream AnsiStyle
forall ann ann'.
(ann -> ann') -> SimpleDocStream ann -> SimpleDocStream ann'
reAnnotateS Keyword -> AnsiStyle
ansiKeyword
| Bool
otherwise = SimpleDocStream Keyword -> String
forall ann. SimpleDocStream ann -> String
renderString
{-# NOINLINE terminalColourAllowed #-}
terminalColourAllowed :: Bool
terminalColourAllowed :: Bool
terminalColourAllowed = IO Bool -> Bool
forall a. IO a -> a
unsafePerformIO (IO Bool -> Bool) -> IO Bool -> Bool
forall a b. (a -> b) -> a -> b
$ Maybe String -> Bool
forall a. Maybe a -> Bool
isNothing (Maybe String -> Bool) -> IO (Maybe String) -> IO Bool
forall (f :: * -> *) a b. Functor f => (a -> b) -> f a -> f b
<$> String -> IO (Maybe String)
lookupEnv String
"NO_COLOR"
{-# NOINLINE terminalSupportsANSI #-}
terminalSupportsANSI :: Bool
terminalSupportsANSI :: Bool
terminalSupportsANSI = IO Bool -> Bool
forall a. IO a -> a
unsafePerformIO (IO Bool -> Bool) -> IO Bool -> Bool
forall a b. (a -> b) -> a -> b
$ Handle -> IO Bool
Term.hSupportsANSI Handle
stdout
{-# NOINLINE terminalLayoutOptions #-}
terminalLayoutOptions :: LayoutOptions
terminalLayoutOptions :: LayoutOptions
terminalLayoutOptions
= IO LayoutOptions -> LayoutOptions
forall a. IO a -> a
unsafePerformIO
(IO LayoutOptions -> LayoutOptions)
-> IO LayoutOptions -> LayoutOptions
forall a b. (a -> b) -> a -> b
$ do Maybe (Window Int)
term <- IO (Maybe (Window Int))
forall n. Integral n => IO (Maybe (Window n))
Term.size
LayoutOptions -> IO LayoutOptions
forall a. a -> IO a
forall (m :: * -> *) a. Monad m => a -> m a
return (LayoutOptions -> IO LayoutOptions)
-> LayoutOptions -> IO LayoutOptions
forall a b. (a -> b) -> a -> b
$ case Maybe (Window Int)
term of
Maybe (Window Int)
Nothing -> LayoutOptions
defaultLayoutOptions
Just Window Int
t -> LayoutOptions { layoutPageWidth :: PageWidth
layoutPageWidth = Int -> Double -> PageWidth
AvailablePerLine (Int -> Int -> Int
forall a. Ord a => a -> a -> a
min Int
w Int
120) Double
f }
where
w :: Int
w = Window Int -> Int
forall a. Window a -> a
Term.width Window Int
t
f :: Double
f | Int
w Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
<= Int
80 = Double
1
| Int
w Int -> Int -> Bool
forall a. Ord a => a -> a -> Bool
<= Int
100 = Double
0.9
| Bool
otherwise = Double
0.8
prettyOpenAcc :: PrettyAcc OpenAcc
prettyOpenAcc :: PrettyAcc OpenAcc
prettyOpenAcc PrettyConfig OpenAcc
config Context
context Val aenv
aenv (OpenAcc PreOpenAcc OpenAcc aenv a
pacc) =
PrettyConfig OpenAcc
-> Context
-> PrettyAcc OpenAcc
-> ExtractAcc OpenAcc
-> Val aenv
-> PreOpenAcc OpenAcc aenv a
-> Adoc
forall (acc :: * -> * -> *) aenv arrs.
PrettyConfig acc
-> Context
-> PrettyAcc acc
-> ExtractAcc acc
-> Val aenv
-> PreOpenAcc acc aenv arrs
-> Adoc
prettyPreOpenAcc PrettyConfig OpenAcc
config Context
context PrettyConfig OpenAcc
-> Context -> Val aenv -> OpenAcc aenv a -> Adoc
PrettyAcc OpenAcc
prettyOpenAcc OpenAcc aenv a -> PreOpenAcc OpenAcc aenv a
ExtractAcc OpenAcc
extractOpenAcc Val aenv
aenv PreOpenAcc OpenAcc aenv a
pacc
extractOpenAcc :: OpenAcc aenv a -> PreOpenAcc OpenAcc aenv a
(OpenAcc PreOpenAcc OpenAcc aenv a
pacc) = PreOpenAcc OpenAcc aenv a
pacc
prettyDelayedOpenAcc :: HasCallStack => PrettyAcc DelayedOpenAcc
prettyDelayedOpenAcc :: HasCallStack => PrettyAcc DelayedOpenAcc
prettyDelayedOpenAcc PrettyConfig DelayedOpenAcc
config Context
context Val aenv
aenv (Manifest PreOpenAcc DelayedOpenAcc aenv a
pacc)
= PrettyConfig DelayedOpenAcc
-> Context
-> PrettyAcc DelayedOpenAcc
-> ExtractAcc DelayedOpenAcc
-> Val aenv
-> PreOpenAcc DelayedOpenAcc aenv a
-> Adoc
forall (acc :: * -> * -> *) aenv arrs.
PrettyConfig acc
-> Context
-> PrettyAcc acc
-> ExtractAcc acc
-> Val aenv
-> PreOpenAcc acc aenv arrs
-> Adoc
prettyPreOpenAcc PrettyConfig DelayedOpenAcc
config Context
context HasCallStack => PrettyAcc DelayedOpenAcc
PrettyConfig DelayedOpenAcc
-> Context -> Val aenv -> DelayedOpenAcc aenv a -> Adoc
PrettyAcc DelayedOpenAcc
prettyDelayedOpenAcc DelayedOpenAcc aenv a -> PreOpenAcc DelayedOpenAcc aenv a
forall aenv a.
HasCallStack =>
DelayedOpenAcc aenv a -> PreOpenAcc DelayedOpenAcc aenv a
ExtractAcc DelayedOpenAcc
extractDelayedOpenAcc Val aenv
aenv PreOpenAcc DelayedOpenAcc aenv a
pacc
prettyDelayedOpenAcc PrettyConfig DelayedOpenAcc
_ Context
_ Val aenv
aenv (Delayed ArrayR (Array sh e)
_ Exp aenv sh
sh Fun aenv (sh -> e)
f Fun aenv (Int -> e)
_)
= Adoc -> Adoc
forall ann. Doc ann -> Doc ann
parens
(Adoc -> Adoc) -> Adoc -> Adoc
forall a b. (a -> b) -> a -> b
$ Int -> Adoc -> Adoc
forall ann. Int -> Doc ann -> Doc ann
nest Int
shiftwidth
(Adoc -> Adoc) -> Adoc -> Adoc
forall a b. (a -> b) -> a -> b
$ [Adoc] -> Adoc
forall ann. [Doc ann] -> Doc ann
sep [ Operator -> Adoc
delayed Operator
"delayed"
, Context -> Val () -> Val aenv -> Exp aenv sh -> Adoc
forall env aenv t.
Context -> Val env -> Val aenv -> OpenExp env aenv t -> Adoc
prettyOpenExp Context
app Val ()
Empty Val aenv
aenv Exp aenv sh
sh
, Adoc -> Adoc
forall ann. Doc ann -> Doc ann
parens (Adoc -> Adoc) -> Adoc -> Adoc
forall a b. (a -> b) -> a -> b
$ Val () -> Val aenv -> Fun aenv (sh -> e) -> Adoc
forall env aenv f.
Val env -> Val aenv -> OpenFun env aenv f -> Adoc
prettyOpenFun Val ()
Empty Val aenv
aenv Fun aenv (sh -> e)
f
]
extractDelayedOpenAcc :: HasCallStack => DelayedOpenAcc aenv a -> PreOpenAcc DelayedOpenAcc aenv a
(Manifest PreOpenAcc DelayedOpenAcc aenv a
pacc) = PreOpenAcc DelayedOpenAcc aenv a
pacc
extractDelayedOpenAcc Delayed{} = Format
(PreOpenAcc DelayedOpenAcc aenv (Array sh e))
(PreOpenAcc DelayedOpenAcc aenv a)
-> PreOpenAcc DelayedOpenAcc aenv a
forall r a. HasCallStack => Format r a -> a
internalError Format
(PreOpenAcc DelayedOpenAcc aenv (Array sh e))
(PreOpenAcc DelayedOpenAcc aenv a)
"expected manifest array"
shouldPrintHash :: Bool
shouldPrintHash :: Bool
shouldPrintHash = IO Bool -> Bool
forall a. IO a -> a
unsafePerformIO (IO Bool -> Bool) -> IO Bool -> Bool
forall a b. (a -> b) -> a -> b
$ Flag -> IO Bool
getFlag Flag
verbose
withSimplStats :: String -> String
#ifdef ACCELERATE_DEBUG
withSimplStats x = unsafePerformIO $ do
when dump_simpl_stats $ x `deepseq` dumpSimplStats
return x
#else
withSimplStats :: ShowS
withSimplStats String
x = String
x
#endif