1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
|
module ToolOptions
( Options(..), defaultOptions, processOptions
, whenStatus, printWhenStatus, printWhenIntermediate, printWhenAll
)
where
import Control.Monad ( when, unless )
import Numeric ( readNat )
import System.Console.GetOpt
import System.CurryPath ( stripCurrySuffix )
import System.Process ( exitWith )
data Options = Options
{ optVerb :: Int
, optHelp :: Bool
, optTime :: Bool
, optShow :: Bool
, optShowResOpts :: Bool
, optShowStats :: Bool
, optOutput :: String
}
defaultOptions :: Options
defaultOptions = Options 1 False False False False False ""
processOptions :: String -> [String] -> IO (Options,[String])
processOptions banner argv = do
let (funopts, args, opterrors) = getOpt Permute options argv
opts = foldl (flip id) defaultOptions funopts
unless (null opterrors)
(putStr (unlines opterrors) >> printUsage >> exitWith 1)
when (optHelp opts) (printUsage >> exitWith 0)
return (opts, map stripCurrySuffix args)
where
printUsage = putStrLn (banner ++ "\n" ++ usageText)
usageText :: String
usageText =
usageInfo ("Usage: curry-anaresinfo [options] <module names>\n") options
options :: [OptDescr (Options -> Options)]
options =
[ Option "h?" ["help"]
(NoArg (\opts -> opts { optHelp = True }))
"print help and exit"
, Option "q" ["quiet"]
(NoArg (\opts -> opts { optVerb = 0 }))
"run quietly (no output, only exit code)"
, Option "v" ["verbosity"]
(OptArg (maybe (checkVerb 2) (safeReadNat checkVerb)) "<n>")
"verbosity level:\n0: quiet (same as `-q')\n1: show status messages (default)\n2: show intermediate results (same as `-v')\n3: show all details"
, Option "o" ["output"]
(ReqArg (\n opts -> opts { optOutput = n }) "<f>")
"store analysis results as data term in <f>"
, Option "s" ["show"]
(NoArg (\opts -> opts { optShow = True }))
"show analysis results"
, Option "" ["resops"]
(NoArg (\opts -> opts { optShowResOpts = True }))
"show all possibly residuating operations"
, Option "" ["stats"]
(NoArg (\opts -> opts { optShowStats = True }))
"show statistics in CSV format"
, Option "t" ["time"]
(NoArg (\opts -> opts { optTime = True }))
"show total analysis time"
]
where
safeReadNat opttrans s opts = case readNat s of
[(n,"")] -> opttrans n opts
_ -> error "Illegal number argument (try `-h' for help)"
checkVerb n opts = if n>=0 && n<4
then opts { optVerb = n }
else error "Illegal verbosity level (try `-h' for help)"
whenStatus :: Options -> IO () -> IO ()
whenStatus opts = when (optVerb opts > 0)
printWhenStatus :: Options -> String -> IO ()
printWhenStatus opts s = whenStatus opts (putStrLn s)
printWhenIntermediate :: Options -> String -> IO ()
printWhenIntermediate opts s =
when (optVerb opts > 1) (putStrLn s)
printWhenAll :: Options -> String -> IO ()
printWhenAll opts s =
when (optVerb opts > 2) (putStrLn s)
|