Panic.lhs 5.64 KB
Newer Older
1
%
2
% (c) The GRASP Project, Glasgow University, 1992-2000
3
4
5
6
7
8
9
10
11
%
\section{Panic error messages}

Defines basic funtions for printing error messages.

It's hard to put these functions anywhere else without causing
some unnecessary loops in the module dependency graph.

\begin{code}
12
13
14
module Panic  
   ( 
     GhcException(..), ghcError, progName, 
15
     pgmError,
sof's avatar
sof committed
16
     panic, panic#, assertPanic, trace,
17
     showException, showGhcException, tryMost,
18
     installSignalHandlers, 
19

20
     catchJust, tryJust, ioErrors, throwTo,
21
   ) where
22

23
#include "HsVersions.h"
24
#include "../includes/ghcconfig.h"
25

26
import Config
27
import FastTypes
28

29
#ifndef mingw32_HOST_OS
sof's avatar
sof committed
30
# if __GLASGOW_HASKELL__ > 504
31
import System.Posix.Signals
sof's avatar
sof committed
32
# else
33
import Posix		( Handler(Catch), installHandler, sigINT, sigQUIT )
34
# endif /* GHC > 504 */
35
36

import CONCURRENT	( myThreadId )
37
#endif /* mingw32_HOST_OS */
38
39
40
41
42

# if __GLASGOW_HASKELL__ < 500
import EXCEPTION        ( raiseInThread )
# else
import EXCEPTION	( throwTo )
43
# endif /* GHC < 500 */
44

45
46
47
48
#if __GLASGOW_HASKELL__ > 408
import EXCEPTION	( catchJust, tryJust, ioErrors )
#endif

49
import DYNAMIC
50
import qualified EXCEPTION as Exception
51
52
import TRACE		( trace )
import UNSAFE_IO	( unsafePerformIO )
53
54

import System
55
56
\end{code}

57
58
59
60
GHC's own exception type.

\begin{code}
ghcError :: GhcException -> a
61
ghcError e = Exception.throwDyn e
62

63
64
65
66
67
68
69
70
71
-- error messages all take the form
--
--	<location>: <error>
--
-- If the location is on the command line, or in GHC itself, then 
-- <location>="ghc".  All of the error types below correspond to 
-- a <location> of "ghc", except for ProgramError (where the string is
-- assumed to contain a location already, so we don't print one).

72
data GhcException
sof's avatar
sof committed
73
74
  = PhaseFailed String		-- name of phase 
  		ExitCode	-- an external phase (eg. cpp) failed
75
  | Interrupted			-- someone pressed ^C
76
  | UsageError String		-- prints the short usage msg after the error
77
  | CmdLineError String		-- cmdline prob, but doesn't print usage
78
  | Panic String		-- the `impossible' happened
79
80
  | InstallationError String	-- an installation problem
  | ProgramError String		-- error in the user's code, probably
81
82
83
84
85
86
87
  deriving Eq

progName = unsafePerformIO (getProgName)
{-# NOINLINE progName #-}

short_usage = "Usage: For basic information, try the `--help' option."
   
88
showException :: Exception.Exception -> String
89
-- Show expected dynamic exceptions specially
90
91
92
showException (Exception.DynException d) | Just e <- fromDynamic d 
					 = show (e::GhcException)
showException other_exn	       	 	 = show other_exn
93

94
instance Show GhcException where
95
96
  showsPrec _ e@(ProgramError _) = showGhcException e
  showsPrec _ e = showString progName . showString ": " . showGhcException e
97

98
showGhcException (UsageError str)
99
   = showString str . showChar '\n' . showString short_usage
100
showGhcException (PhaseFailed phase code)
sof's avatar
sof committed
101
102
103
104
105
106
107
108
   = showString "phase `" . showString phase . 
     showString "' failed (exitcode = " . shows int_code . 
     showString ")"
  where
    int_code = 
      case code of
        ExitSuccess   -> (0::Int)
	ExitFailure x -> x
109
110
111
112
113
114
115
showGhcException (CmdLineError str)
   = showString str
showGhcException (ProgramError str)
   = showString str
showGhcException (InstallationError str)
   = showString str
showGhcException (Interrupted)
116
   = showString "interrupted"
117
showGhcException (Panic s)
118
119
   = showString ("panic! (the `impossible' happened, GHC version "
		 ++ cProjectVersion ++ "):\n\t"
120
121
	         ++ s ++ "\n\n"
	         ++ "Please report it as a compiler bug "
122
123
	         ++ "to glasgow-haskell-bugs@haskell.org,\n"
		 ++ "or http://sourceforge.net/projects/ghc/.\n\n")
124

ralf's avatar
ralf committed
125
#if __GLASGOW_HASKELL__ < 603
ralf's avatar
ralf committed
126
127
128
129
130
myMkTyConApp = mkAppTy
#else 
myMkTyConApp = mkTyConApp
#endif

131
132
133
ghcExceptionTc = mkTyCon "GhcException"
{-# NOINLINE ghcExceptionTc #-}
instance Typeable GhcException where
ralf's avatar
ralf committed
134
  typeOf _ = myMkTyConApp ghcExceptionTc []
135
136
137
138
\end{code}

Panics and asserts.

139
\begin{code}
140
141
142
panic, pgmError :: String -> a
panic    x = Exception.throwDyn (Panic x)
pgmError x = Exception.throwDyn (ProgramError x)
143
144
145
146
147

-- #-versions because panic can't return an unboxed int, and that's
-- what TAG_ is with GHC at the moment.  Ugh. (Simon)
-- No, man -- Too Beautiful! (Will)

148
149
panic# :: String -> FastInt
panic# s = case (panic s) of () -> _ILIT 0
150
151

assertPanic :: String -> Int -> a
152
assertPanic file line = 
153
  Exception.throw (Exception.AssertionFailed 
154
           ("ASSERT failed! file " ++ file ++ ", line " ++ show line))
155
\end{code}
156
157
158
159
160
161

\begin{code}
-- | tryMost is like try, but passes through Interrupted and Panic
-- exceptions.  Used when we want soft failures when reading interface
-- files, for example.

162
tryMost :: IO a -> IO (Either Exception.Exception a)
163
tryMost action = do r <- myTry action; filter r
164
  where
165
   filter (Left e@(Exception.DynException d))
166
167
	    | Just ghc_ex <- fromDynamic d
		= case ghc_ex of
168
169
		    Interrupted -> Exception.throw e
		    Panic _     -> Exception.throw e
170
171
172
173
174
		    _other      -> return (Left e)
   filter other 
     = return other

#if __GLASGOW_HASKELL__ <= 408
175
myTry = Exception.tryAllIO
176
#else
177
myTry = Exception.try
178
179
180
181
182
183
184
#endif
\end{code}	

Compatibility stuff:

\begin{code}
#if __GLASGOW_HASKELL__ <= 408
sof's avatar
sof committed
185
catchJust = Exception.catchIO
186
tryJust   = Exception.tryIO
sof's avatar
sof committed
187
188
ioErrors  = Exception.justIoErrors
throwTo   = Exception.raiseInThread
189
190
#endif
\end{code}
191
192
193
194
195
196
197
198

Standard signal handlers for catching ^C, which just throw an
exception in the main thread.  NOTE: must be called from the main
thread.

\begin{code}
installSignalHandlers :: IO ()
installSignalHandlers = do
199
#ifndef mingw32_HOST_OS
200
201
202
203
204
205
206
207
  main_thread <- myThreadId
  let sig_handler = Catch (throwTo main_thread 
				(Exception.DynException (toDyn Interrupted)))
  installHandler sigQUIT sig_handler Nothing 
  installHandler sigINT  sig_handler Nothing
#endif
  return ()
\end{code}