DriverPipeline.hs 52.5 KB
Newer Older
1
2
3
4
-----------------------------------------------------------------------------
--
-- GHC Driver
--
5
-- (c) The University of Glasgow 2005
6
7
8
9
--
-----------------------------------------------------------------------------

module DriverPipeline (
10
11
	-- Run a series of compilation steps in a pipeline, for a
	-- collection of source files.
12
   oneShot, compileFile,
13

14
	-- Interfaces for the batch-mode driver
15
   staticLink,
16

17
18
19
20
	-- Interfaces for the compilation manager (interpreted/batch-mode)
   preprocess, 
   compile, CompResult(..), 
   link, 
21

rrt's avatar
rrt committed
22
        -- DLL building
23
24
   doMkDLL,

25
26
   getOptionsFromStringBuffer,	-- used in module GHC
   optionsErrorMsgs,	  	-- ditto
27
28
29
30
  ) where

#include "HsVersions.h"

31
import Packages
32
import GetImports
33
import DriverPhases
sof's avatar
sof committed
34
import SysTools		( newTempName, addFilesToClean, getSysMan, copy )
35
import qualified SysTools	
36
import HscMain
37
import Finder
38
39
40
import HscTypes
import Outputable
import Module
41
import ErrUtils
42
43
import DynFlags
import StaticFlags	( v_Ld_inputs, opt_Static, WayName(..) )
44
import Config
45
import Panic
46
import Util
47
import StringBuffer	( hGetStringBuffer )
48
import BasicTypes	( SuccessFlag(..) )
49
import Maybes		( expectJust )
50
import Ctype		( is_ident )
51
52
import StringBuffer	( StringBuffer(..), lexemeToString )
import ParserCoreUtils	( getCoreModuleName )
53
54
55
import SrcLoc		( srcLocSpan, mkSrcLoc )
import FastString	( mkFastString )
import Bag		( listToBag, emptyBag )
56
import SrcLoc		( Located(..) )
57

58
59
import Distribution.Compiler ( extensionsToGHCFlag )

60
import EXCEPTION
61
import DATA_IOREF	( readIORef, writeIORef, IORef )
62
import GLAEXTS		( Int(..) )
63

64
65
import Directory
import System
66
67
import IO
import Monad
68
import Data.List	( isSuffixOf )
69
70
import Maybe

71

72
73
-- ---------------------------------------------------------------------------
-- Pre-process
74
75
76

-- Just preprocess a file, put the result in a temp. file (used by the
-- compilation manager during the summary phase).
77
78
79
--
-- We return the augmented DynFlags, because they contain the result
-- of slurping in the OPTIONS pragmas
80

81
82
83
84
preprocess :: DynFlags -> (FilePath, Maybe Phase) -> IO (DynFlags, FilePath)
preprocess dflags (filename, mb_phase) =
  ASSERT2(isJust mb_phase || isHaskellSrcFilename filename, text filename) 
  runPipeline anyHsc dflags (filename, mb_phase) Temporary Nothing{-no ModLocation-}
85

86
87
-- ---------------------------------------------------------------------------
-- Compile
88
89
90
91
92
93
94
95
96
97

-- Compile a single module, under the control of the compilation manager.
--
-- This is the interface between the compilation manager and the
-- compiler proper (hsc), where we deal with tedious details like
-- reading the OPTIONS pragma from the source file, and passing the
-- output of hsc through the C compiler.

-- NB.  No old interface can also mean that the source has changed.

98
compile :: HscEnv
99
	-> ModSummary
100
	-> Maybe Linkable	-- Just linkable <=> source unchanged
101
        -> Maybe ModIface       -- Old interface, if available
102
        -> Int -> Int
103
104
105
        -> IO CompResult

data CompResult
106
107
108
   = CompOK   ModDetails 	-- New details
              ModIface		-- New iface
              (Maybe Linkable)	-- a Maybe, for the same reasons as hm_linkable
109

110
   | CompErrs 
111
112


113
compile hsc_env mod_summary maybe_old_linkable old_iface mod_index nmods = do 
114

115
   let dflags0     = hsc_dflags hsc_env
116
117
       this_mod    = ms_mod mod_summary
       src_flavour = ms_hsc_src mod_summary
118

119
120
121
122
       have_object 
	       | Just l <- maybe_old_linkable, isObjectLinkable l = True
	       | otherwise = False

123
   showPass dflags0 ("Compiling " ++ showModMsg have_object mod_summary)
124

125
   let location	  = ms_location mod_summary
126
   let input_fn   = expectJust "compile:hs" (ml_hs_file location) 
127
   let input_fnpp = expectJust "compile:hspp" (ms_hspp_file mod_summary)
128

129
   debugTraceMsg dflags0 2 (text "compile: input file" <+> text input_fnpp)
130

131
132
133
   -- Add in the OPTIONS from the source file
   -- This is nasty: we've done this once already, in the compilation manager
   -- It might be better to cache the flags in the ml_hspp_file field,say
134
   let hspp_buf = expectJust "compile:hspp_buf" (ms_hspp_buf mod_summary)
135
       opts = getOptionsFromStringBuffer hspp_buf input_fn
136
137
   (dflags1,unhandled_flags) <- parseDynamicFlags dflags0 (map snd opts)
   if (not (null unhandled_flags))
138
	then do printErrorsAndWarnings dflags1 (optionsErrorMsgs unhandled_flags opts input_fn)
139
140
		return CompErrs
	else do
141

142
   let (basename, _) = splitFilename input_fn
143

144
145
146
147
  -- We add the directory in which the .hs files resides) to the import path.
  -- This is needed when we try to compile the .hc file later, if it
  -- imports a _stub.h file that we created here.
   let current_dir = directoryOf basename
148
149
       old_paths   = includePaths dflags1
       dflags      = dflags1 { includePaths = current_dir : old_paths }
150

151
   -- Figure out what lang we're generating
152
   let hsc_lang = hscMaybeAdjustTarget dflags StopLn src_flavour (hscTarget dflags)
153
   -- ... and what the next phase should be
154
   let next_phase = hscNextPhase dflags src_flavour hsc_lang
155
   -- ... and what file to generate the output into
156
157
   output_fn <- getOutputFilename dflags next_phase 
			Temporary basename next_phase (Just location)
158

159
   let dflags' = dflags { hscTarget = hsc_lang,
160
				hscOutName = output_fn,
161
162
163
				extCoreName = basename ++ ".hcr" }

   -- -no-recomp should also work with --make
164
   let do_recomp = dopt Opt_RecompChecking dflags
165
       source_unchanged = isJust maybe_old_linkable && do_recomp
166
       hsc_env' = hsc_env { hsc_dflags = dflags' }
167
       object_filename = ml_obj_file location
168
169

   -- run the compiler
170
   hsc_result <- hscMain hsc_env' mod_summary
171
			 source_unchanged have_object old_iface
172
                         (Just (mod_index, nmods))
173
174

   case hsc_result of
175
      HscFail -> return CompErrs
176

177
178
179
      HscNoRecomp details iface -> 
	  ASSERT(isJust maybe_old_linkable)
	  return (CompOK details iface maybe_old_linkable)
180

181
      HscRecomp details iface stub_h_exists stub_c_exists maybe_interpreted_code 
182
183

	| isHsBoot src_flavour	-- No further compilation to do
184
185
186
187
188
	-> do	case hsc_lang of
		   HscInterpreted -> return ()
		   _other -> SysTools.touch dflags' "Touching object file" 
					    object_filename
		return (CompOK details iface Nothing)
189

190
	| otherwise	-- Normal source file
191
	-> do
192
193
	   stub_unlinked <-
	     if stub_c_exists then do
194
		stub_o <- compileStub dflags' this_mod location
195
196
197
		return [ DotO stub_o ]
	     else
		return []
198
199
200
201
202
203

	   (hs_unlinked, unlinked_time) <-
	     case hsc_lang of

		-- in interpreted mode, just return the compiled code
		-- as our "unlinked" object.
204
205
		HscInterpreted
		  -> case maybe_interpreted_code of
206
#ifdef GHCI
207
		       Just comp_bc -> return ([BCOs comp_bc], ms_hs_date mod_summary)
208
209
210
211
212
213
			-- Why do we use the timestamp of the source file here,
			-- rather than the current time?  This works better in
			-- the case where the local clock is out of sync
			-- with the filesystem's clock.  It's just as accurate:
			-- if the source is modified, then the linkable will
			-- be out of date.
214
215
216
#endif
		       Nothing -> panic "compile: no interpreted code"

217
218
219
		HscNothing
		  -> return ([], ms_hs_date mod_summary)

220
221
		-- We're in --make mode: finish the compilation pipeline.
		_other
222
		  -> do	runPipeline StopLn dflags (output_fn,Nothing) Persistent
223
224
				    (Just location)
				-- The object filename comes from the ModLocation
225

226
227
			o_time <- getModificationTime object_filename
			return ([DotO object_filename], o_time)
228

229
	   let linkable = LM unlinked_time this_mod
230
231
			     (hs_unlinked ++ stub_unlinked)

232
	   return (CompOK details iface (Just linkable))
233

234
-----------------------------------------------------------------------------
235
236
-- stub .h and .c files (for foreign export support)

237
238
239
-- The _stub.c file is derived from the haskell source file, possibly taking
-- into account the -stubdir option.
--
240
241
242
243
244
245
246
247
248
249
250
251
252
-- Consequently, we derive the _stub.o filename from the haskell object
-- filename.  
--
-- This isn't necessarily the same as the object filename we
-- would get if we just compiled the _stub.c file using the pipeline.
-- For example:
--
--    ghc src/A.hs -odir obj
-- 
-- results in obj/A.o, and src/A_stub.c.  If we compile src/A_stub.c with
-- -odir obj, we would get obj/src/A_stub.o, which is wrong; we want
-- obj/A_stub.o.

253
254
255
compileStub :: DynFlags -> Module -> ModLocation -> IO FilePath
compileStub dflags mod location = do
	let (o_base, o_ext) = splitFilename (ml_obj_file location)
256
257
	    stub_o = o_base ++ "_stub" `joinFileExt` o_ext

258
	-- compile the _stub.c file w/ gcc
259
	let (stub_c,_) = mkStubPaths dflags mod location
260
261
262
263
	runPipeline StopLn dflags (stub_c,Nothing) 
		(SpecificFile stub_o) Nothing{-no ModLocation-}

	return stub_o
264
265


266
267
-- ---------------------------------------------------------------------------
-- Link
268

269
link :: GhcMode			-- interactive or batch
270
271
     -> DynFlags		-- dynamic flags
     -> Bool			-- attempt linking in batch mode?
272
     -> HomePackageTable	-- what to link
273
274
275
276
277
278
279
280
281
282
     -> IO SuccessFlag

-- For the moment, in the batch linker, we don't bother to tell doLink
-- which packages to link -- it just tries all that are available.
-- batch_attempt_linking should only be *looked at* in batch mode.  It
-- should only be True if the upsweep was successful and someone
-- exports main, i.e., we have good reason to believe that linking
-- will succeed.

#ifdef GHCI
283
284
link Interactive dflags batch_attempt_linking hpt
    = do -- Not Linking...(demand linker will do the job)
285
286
287
	 return Succeeded
#endif

288
289
290
link JustTypecheck dflags batch_attempt_linking hpt
   = return Succeeded

291
link BatchCompile dflags batch_attempt_linking hpt
292
   | batch_attempt_linking
293
   = do 
294
295
296
297
298
299
300
	let 
	    home_mod_infos = moduleEnvElts hpt

	    -- the packages we depend on
	    pkg_deps  = concatMap (dep_pkgs . mi_deps . hm_iface) home_mod_infos

	    -- the linkables to link
301
	    linkables = map (expectJust "link".hm_linkable) home_mod_infos
302

303
        debugTraceMsg dflags 3 (text "link: linkables are ..." $$ vcat (map ppr linkables))
304

305
	-- check for the -no-link flag
306
	if isNoLink (ghcLink dflags)
307
	  then do debugTraceMsg dflags 3 (text "link(batch): linking omitted (-c flag given).")
308
309
310
	          return Succeeded
	  else do

311
312
313
	let getOfiles (LM _ _ us) = map nameOfObject (filter isObject us)
	    obj_files = concatMap getOfiles linkables

314
315
316
317
318
319
320
321
322
323
324
325
	    exe_file = exeFileName dflags

	-- if the modification time on the executable is later than the
	-- modification times on all of the objects, then omit linking
	-- (unless the -no-recomp flag was given).
	e_exe_time <- IO.try $ getModificationTime exe_file
	let linking_needed 
		| Left _  <- e_exe_time = True
		| Right t <- e_exe_time = 
			any (t <) (map linkableTime linkables)

	if dopt Opt_RecompChecking dflags && not linking_needed
326
	   then do debugTraceMsg dflags 1 (text exe_file <+> ptext SLIT("is up to date, linking not required."))
327
328
329
		   return Succeeded
	   else do

330
331
	debugTraceMsg dflags 1 (ptext SLIT("Linking") <+> text exe_file
				 <+> text "...")
332

333
	-- Don't showPass in Batch mode; doLink will do that for us.
334
335
336
337
	let link = case ghcLink dflags of
	        MkDLL       -> doMkDLL
	        StaticLink  -> staticLink
	link dflags obj_files pkg_deps
338

339
        debugTraceMsg dflags 3 (text "link: done")
340
341
342
343
344

	-- staticLink only returns if it succeeds
        return Succeeded

   | otherwise
345
346
   = do debugTraceMsg dflags 3 (text "link(batch): upsweep (partially) failed OR" $$
                                text "   Main.main not exported; not linking.")
347
        return Succeeded
348
      
349
350
351
352

-- -----------------------------------------------------------------------------
-- Compile files in one-shot mode.

353
oneShot :: DynFlags -> Phase -> [(String, Maybe Phase)] -> IO ()
354
355
356
357
oneShot dflags stop_phase srcs = do
  o_files <- mapM (compileFile dflags stop_phase) srcs
  doLink dflags stop_phase o_files

358
359
compileFile :: DynFlags -> Phase -> (FilePath, Maybe Phase) -> IO FilePath
compileFile dflags stop_phase (src, mb_phase) = do
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
   exists <- doesFileExist src
   when (not exists) $ 
   	throwDyn (CmdLineError ("does not exist: " ++ src))
   
   let
	split     = dopt Opt_SplitObjs dflags
	mb_o_file = outputFile dflags
	ghc_link  = ghcLink dflags	-- Set by -c or -no-link

	-- When linking, the -o argument refers to the linker's output.	
	-- otherwise, we use it as the name for the pipeline's output.
        output
	 | StopLn <- stop_phase, not (isNoLink ghc_link) = Persistent
		-- -o foo applies to linker
	 | Just o_file <- mb_o_file = SpecificFile o_file
		-- -o foo applies to the file we are compiling now
	 | otherwise = Persistent

        stop_phase' = case stop_phase of 
			As | split -> SplitAs
			other      -> stop_phase

   (_, out_file) <- runPipeline stop_phase' dflags
383
			  (src, mb_phase) output Nothing{-no ModLocation-}
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
   return out_file


doLink :: DynFlags -> Phase -> [FilePath] -> IO ()
doLink dflags stop_phase o_files
  | not (isStopLn stop_phase)
  = return ()		-- We stopped before the linking phase

  | otherwise
  = case ghcLink dflags of
	NoLink     -> return ()
	StaticLink -> staticLink dflags o_files link_pkgs
	MkDLL      -> doMkDLL dflags o_files link_pkgs
  where
   -- Always link in the haskell98 package for static linking.  Other
   -- packages have to be specified via the -package flag.
    link_pkgs
	  | ExtPackage h98_id <- haskell98PackageId (pkgState dflags) = [h98_id]
	  | otherwise = []


405
406
-- ---------------------------------------------------------------------------
-- Run a compilation pipeline, consisting of multiple phases.
407

408
409
410
411
-- This is the interface to the compilation pipeline, which runs
-- a series of compilation steps on a single source file, specifying
-- at which stage to stop.

412
-- The DynFlags can be modified by phases in the pipeline (eg. by
413
414
415
416
417
418
419
420
421
422
423
424
425
-- GHC_OPTIONS pragmas), and the changes affect later phases in the
-- pipeline.

data PipelineOutput 
  = Temporary
	-- output should be to a temporary file: we're going to
	-- run more compilation steps on this output later
  | Persistent
	-- we want a persistent file, i.e. a file in the current directory
	-- derived from the input filename, but with the appropriate extension.
	-- eg. in "ghc -c Foo.hs" the output goes into ./Foo.o.
  | SpecificFile FilePath
	-- the output must go into the specified file.
426

427
runPipeline
428
429
430
431
432
  :: Phase		        -- When to stop
  -> DynFlags		        -- Dynamic flags
  -> (FilePath,Maybe Phase)     -- Input filename (and maybe -x suffix)
  -> PipelineOutput	        -- Output filename
  -> Maybe ModLocation          -- A ModLocation, if this is a Haskell module
433
  -> IO (DynFlags, FilePath)	-- (final flags, output filename)
434

435
runPipeline stop_phase dflags (input_fn, mb_phase) output maybe_loc
436
437
  = do
  let (basename, suffix) = splitFilename input_fn
438
439
440
441
442

	-- If we were given a -x flag, then use that phase to start from
      start_phase
	| Just x_phase <- mb_phase = x_phase
	| otherwise                = startPhase suffix
443

444
445
446
447
448
449
  -- We want to catch cases of "you can't get there from here" before
  -- we start the pipeline, because otherwise it will just run off the
  -- end.
  --
  -- There is a partial ordering on phases, where A < B iff A occurs
  -- before B in a normal compilation pipeline.
450

451
452
  when (not (start_phase `happensBefore` stop_phase)) $
	throwDyn (UsageError 
453
454
		    ("cannot compile this file to desired target: "
		       ++ input_fn))
455

456
457
458
  -- this is a function which will be used to calculate output file names
  -- as we go along (we partially apply it to some of its inputs here)
  let get_output_fn = getOutputFilename dflags stop_phase output basename
459

460
  -- Execute the pipeline...
461
462
463
  (dflags', output_fn, maybe_loc) <- 
	pipeLoop dflags start_phase stop_phase input_fn 
	  	 basename suffix get_output_fn maybe_loc
464

465
  -- Sometimes, a compilation phase doesn't actually generate any output
466
467
468
  -- (eg. the CPP phase when -fcpp is not turned on).  If we end on this
  -- stage, but we wanted to keep the output, then we have to explicitly
  -- copy the file.
469
470
471
472
473
474
475
  case output of
    Temporary -> 
	return (dflags', output_fn)
    _other ->
	do final_fn <- get_output_fn stop_phase maybe_loc
	   when (final_fn /= output_fn) $
		  copy dflags ("Copying `" ++ output_fn ++ "' to `" ++ final_fn
476
			++ "'") output_fn final_fn
477
478
	   return (dflags', final_fn)
	        
479
480


481
pipeLoop :: DynFlags -> Phase -> Phase 
482
483
484
485
	 -> FilePath  -> String -> Suffix
	 -> (Phase -> Maybe ModLocation -> IO FilePath)
	 -> Maybe ModLocation
	 -> IO (DynFlags, FilePath, Maybe ModLocation)
486

487
pipeLoop dflags phase stop_phase 
488
489
	 input_fn orig_basename orig_suff 
	 orig_get_output_fn maybe_loc
490

491
492
  | phase `eqPhase` stop_phase		  -- All done
  = return (dflags, input_fn, maybe_loc)
493

494
  | not (phase `happensBefore` stop_phase)
495
496
497
498
	-- Something has gone wrong.  We'll try to cover all the cases when
	-- this could happen, so if we reach here it is a panic.
	-- eg. it might happen if the -C flag is used on a source file that
	-- has {-# OPTIONS -fasm #-}.
499
500
501
502
503
  = panic ("pipeLoop: at phase " ++ show phase ++ 
	   " but I wanted to stop at phase " ++ show stop_phase)

  | otherwise 
  = do	{ (next_phase, dflags', maybe_loc, output_fn)
504
		<- runPhase phase stop_phase dflags orig_basename 
505
			    orig_suff input_fn orig_get_output_fn maybe_loc
506
	; pipeLoop dflags' next_phase stop_phase output_fn
507
508
		   orig_basename orig_suff orig_get_output_fn maybe_loc }

509
510
511
512
getOutputFilename
  :: DynFlags -> Phase -> PipelineOutput -> String
  -> Phase{-next phase-} -> Maybe ModLocation -> IO FilePath
getOutputFilename dflags stop_phase output basename
513
514
515
 = func
 where
	hcsuf      = hcSuf dflags
516
	odir       = objectDir dflags
517
518
519
520
521
	osuf       = objectSuf dflags
	keep_hc    = dopt Opt_KeepHcFiles dflags
	keep_raw_s = dopt Opt_KeepRawSFiles dflags
	keep_s     = dopt Opt_KeepSFiles dflags

522
523
524
        myPhaseInputExt HCc    = hcsuf
        myPhaseInputExt StopLn = osuf
        myPhaseInputExt other  = phaseInputExt other
525

526
	func next_phase maybe_location
527
528
529
530
	   | is_last_phase, Persistent <- output     = persistent_fn
	   | is_last_phase, SpecificFile f <- output = return f
	   | keep_this_output	   		     = persistent_fn
     	   | otherwise        	   		     = newTempName dflags suffix
531
	   where
532
		is_last_phase = next_phase `eqPhase` stop_phase
533
534
535
536

		-- sometimes, we keep output from intermediate stages
		keep_this_output = 
     		     case next_phase of
537
     			     StopLn              -> True
538
539
540
541
542
     			     Mangle | keep_raw_s -> True
     			     As     | keep_s     -> True
     			     HCc    | keep_hc    -> True
     			     _other              -> False

543
		suffix = myPhaseInputExt next_phase
544
545
546

		-- persistent object files get put in odir
	        persistent_fn 
547
548
		   | StopLn <- next_phase = return odir_persistent
		   | otherwise            = return persistent
549

550
		persistent = basename `joinFileExt` suffix
551

552
		odir_persistent
553
		   | Just loc <- maybe_location = ml_obj_file loc
554
		   | Just d <- odir = d `joinFileName` persistent
555
556
		   | otherwise      = persistent

557

558
559
560
561
562
563
564
565
566
567
-- -----------------------------------------------------------------------------
-- Each phase in the pipeline returns the next phase to execute, and the
-- name of the file in which the output was placed.
--
-- We must do things dynamically this way, because we often don't know
-- what the rest of the phases will be until part-way through the
-- compilation: for example, an {-# OPTIONS -fasm #-} at the beginning
-- of a source file can change the latter stages of the pipeline from
-- taking the via-C route to using the native code generator.

568
569
runPhase :: Phase	-- Do this phase first
	 -> Phase	-- Stop just before this phase
570
571
572
573
574
	 -> DynFlags
	 -> String	-- basename of original input source
	 -> String	-- its extension
	 -> FilePath	-- name of file which contains the input to this phase.
	 -> (Phase -> Maybe ModLocation -> IO FilePath)
575
			-- how to calculate the output filename
576
	 -> Maybe ModLocation		-- the ModLocation, if we have one
577
	 -> IO (Phase,	  		-- next phase
578
579
580
		DynFlags,		-- new dynamic flags
		Maybe ModLocation,	-- the ModLocation, if we have one
		FilePath)		-- output filename
581

582
583
584
585
	-- Invariant: the output filename always contains the output
	-- Interesting case: Hsc when there is no recompilation to do
	--		     Then the output filename is still a .o file 

586
587
588
-------------------------------------------------------------------------------
-- Unlit phase 

589
runPhase (Unlit sf) _stop dflags _basename _suff input_fn get_output_fn maybe_loc
590
  = do let unlit_flags = getOpts dflags opt_L
rrt's avatar
rrt committed
591
       -- The -h option passes the file name for unlit to put in a #line directive
592
       output_fn <- get_output_fn (Cpp sf) maybe_loc
593

594
595
       SysTools.runUnlit dflags 
		(map SysTools.Option unlit_flags ++
sof's avatar
sof committed
596
597
       			  [ SysTools.Option     "-h"
 			  , SysTools.Option     input_fn
sof's avatar
sof committed
598
599
			  , SysTools.FileOption "" input_fn
			  , SysTools.FileOption "" output_fn
sof's avatar
sof committed
600
			  ])
601

602
       return (Cpp sf, dflags, maybe_loc, output_fn)
603
604

-------------------------------------------------------------------------------
605
606
-- Cpp phase : (a) gets OPTIONS out of file
--	       (b) runs cpp if necessary
607

608
runPhase (Cpp sf) _stop dflags0 basename suff input_fn get_output_fn maybe_loc
609
  = do src_opts <- getOptionsFromSource input_fn
610
       (dflags,unhandled_flags) <- parseDynamicFlags dflags0 src_opts
611
       checkProcessArgsResult unhandled_flags (basename `joinFileExt` suff)
612

613
       if not (dopt Opt_Cpp dflags) then
sof's avatar
sof committed
614
615
           -- no need to preprocess CPP, just pass input file along
	   -- to the next phase of the pipeline.
616
          return (HsPp sf, dflags, maybe_loc, input_fn)
sof's avatar
sof committed
617
	else do
618
	    output_fn <- get_output_fn (HsPp sf) maybe_loc
619
	    doCpp dflags True{-raw-} False{-no CC opts-} input_fn output_fn
620
	    return (HsPp sf, dflags, maybe_loc, output_fn)
621

sof's avatar
sof committed
622
623
624
-------------------------------------------------------------------------------
-- HsPp phase 

625
runPhase (HsPp sf) _stop dflags basename suff input_fn get_output_fn maybe_loc
626
  = do if not (dopt Opt_Pp dflags) then
sof's avatar
sof committed
627
628
           -- no need to preprocess, just pass input file along
	   -- to the next phase of the pipeline.
629
          return (Hsc sf, dflags, maybe_loc, input_fn)
sof's avatar
sof committed
630
	else do
631
	    let hspp_opts = getOpts dflags opt_F
632
	    let orig_fn = basename `joinFileExt` suff
633
	    output_fn <- get_output_fn (Hsc sf) maybe_loc
634
635
	    SysTools.runPp dflags
			   ( [ SysTools.Option     orig_fn
sof's avatar
sof committed
636
637
638
639
640
			     , SysTools.Option     input_fn
			     , SysTools.FileOption "" output_fn
			     ] ++
			     map SysTools.Option hspp_opts
			   )
641
	    return (Hsc sf, dflags, maybe_loc, output_fn)
642

643
644
645
-----------------------------------------------------------------------------
-- Hsc phase

646
647
-- Compilation of a single module, in "legacy" mode (_not_ under
-- the direction of the compilation manager).
648
runPhase (Hsc src_flavour) stop dflags0 basename suff input_fn get_output_fn _maybe_loc 
649
 = do	-- normal Hsc mode, not mkdependHS
650

651
652
653
  -- we add the current directory (i.e. the directory in which
  -- the .hs files resides) to the import path, since this is
  -- what gcc does, and it's probably what you want.
654
	let current_dir = directoryOf basename
655
	
656
657
	    paths = includePaths dflags0
	    dflags = dflags0 { includePaths = current_dir : paths }
658
	
659
  -- gather the imports and module name
660
        (hspp_buf,mod_name) <- 
661
662
663
664
665
666
            case src_flavour of
		ExtCoreFile -> do {  -- no explicit imports in ExtCore input.
			          ; m <- getCoreModuleName input_fn
			          ; return (Nothing, mkModule m) }

		other -> do { buf <- hGetStringBuffer input_fn
667
			    ; (_,_,L _ mod_name) <- getImports dflags buf input_fn
668
669
670
671
672
673
674
675
			    ; return (Just buf, mod_name) }

  -- Build a ModLocation to pass to hscMain.
  -- The source filename is rather irrelevant by now, but it's used
  -- by hscMain for messages.  hscMain also needs 
  -- the .hi and .o filenames, and this is as good a way
  -- as any to generate them, and better than most. (e.g. takes 
  -- into accout the -osuf flags)
676
	location1 <- mkHomeModLocation2 dflags mod_name basename suff
677
678
679
680
681
682
683
684
685

  -- Boot-ify it if necessary
	let location2 | isHsBoot src_flavour = addBootSuffixLocn location1
		      | otherwise	     = location1 
					

  -- Take -ohi into account if present
  -- This can't be done in mkHomeModuleLocation because
  -- it only applies to the module being compiles
686
687
	let ohi = outputHi dflags
	    location3 | Just fn <- ohi = location2{ ml_hi_file = fn }
688
689
690
691
692
693
694
		      | otherwise      = location2

  -- Take -o into account if present
  -- Very like -ohi, but we must *only* do this if we aren't linking
  -- (If we're linking then the -o applies to the linked thing, not to
  -- the object file for one module.)
  -- Note the nasty duplication with the same computation in compileFile above
695
696
697
	let expl_o_file = outputFile dflags
	    location4 | Just ofile <- expl_o_file
		      , isNoLink (ghcLink dflags)
698
699
700
701
		      = location3 { ml_obj_file = ofile }
		      | otherwise = location3

  -- Make the ModSummary to hand to hscMain
702
	src_timestamp <- getModificationTime (basename `joinFileExt` suff)
703
704
705
706
707
708
709
710
711
	let
	    unused_field = panic "runPhase:ModSummary field"
		-- Some fields are not looked at by hscMain
	    mod_summary = ModSummary {	ms_mod 	     = mod_name, 
					ms_hsc_src   = src_flavour,
				 	ms_hspp_file = Just input_fn,
					ms_hspp_buf  = hspp_buf,
					ms_location  = location4,
					ms_hs_date   = src_timestamp,
sof's avatar
sof committed
712
					ms_obj_date  = Nothing,
713
714
715
716
717
718
719
					ms_imps	     = unused_field,
					ms_srcimps   = unused_field }

	    o_file = ml_obj_file location4 	-- The real object file


  -- Figure out if the source has changed, for recompilation avoidance.
720
  --
721
  -- Setting source_unchanged to True means that M.o seems
722
723
  -- to be up to date wrt M.hs; so no need to recompile unless imports have
  -- changed (which the compiler itself figures out).
724
  -- Setting source_unchanged to False tells the compiler that M.o is out of
725
  -- date wrt M.hs (or M.o doesn't exist) so we must recompile regardless.
726
	let do_recomp = dopt Opt_RecompChecking dflags
727
	source_unchanged <- 
728
          if not do_recomp || not (isStopLn stop)
729
730
		-- Set source_unchanged to False unconditionally if
		--	(a) recompilation checker is off, or
731
		-- 	(b) we aren't going all the way to .o file (e.g. ghc -S)
732
733
	     then return False	
		-- Otherwise look at file modification dates
734
	     else do o_file_exists <- doesFileExist o_file
735
		     if not o_file_exists
736
		        then return False	-- Need to recompile
737
			else do t2 <- getModificationTime o_file
738
			        if t2 > src_timestamp
739
740
				  then return True
				  else return False
741

742
  -- get the DynFlags
743
744
	let hsc_lang = hscMaybeAdjustTarget dflags stop src_flavour (hscTarget dflags)
	let next_phase = hscNextPhase dflags src_flavour hsc_lang
745
	output_fn  <- get_output_fn next_phase (Just location4)
746

747
        let dflags' = dflags { hscTarget = hsc_lang,
748
749
			       hscOutName = output_fn,
			       extCoreName = basename ++ ".hcr" }
750

751
	hsc_env <- newHscEnv dflags'
752

753
754
755
  -- Tell the finder cache about this module
	addHomeModuleToFinder hsc_env mod_name location4

756
  -- run the compiler!
757
	result <- hscMain hsc_env
758
759
760
			  mod_summary source_unchanged 
			  False		-- No object file
			  Nothing	-- No iface
761
                          Nothing       -- No "module i of n" progress info
762

763
	case result of
764

765
	    HscFail -> throwDyn (PhaseFailed "hsc" (ExitFailure 1))
766

767
            HscNoRecomp details iface -> do
768
		SysTools.touch dflags' "Touching object file" o_file
769
770
771
			-- The .o file must have a later modification date
			-- than the source file (else we wouldn't be in HscNoRecomp)
			-- but we touch it anyway, to keep 'make' happy (we think).
772
		return (StopLn, dflags', Just location4, o_file)
773

774
	    HscRecomp _details _iface 
775
		      stub_h_exists stub_c_exists
776
		      _maybe_interpreted_code -> do
777

778
		when stub_c_exists $ do
779
			stub_o <- compileStub dflags' mod_name location4
780
			consIORef v_Ld_inputs stub_o
781
782
783
784
785
786
787
788

		-- In the case of hs-boot files, generate a dummy .o-boot 
		-- stamp file for the benefit of Make
		case src_flavour of
		  HsBootFile -> SysTools.touch dflags' "Touching object file" o_file
		  other	     -> return ()

		return (next_phase, dflags', Just location4, output_fn)
789

790
791
792
-----------------------------------------------------------------------------
-- Cmm phase

793
runPhase CmmCpp stop dflags basename suff input_fn get_output_fn maybe_loc
794
795
  = do
       output_fn <- get_output_fn Cmm maybe_loc
796
       doCpp dflags False{-not raw-} True{-include CC opts-} input_fn output_fn	
797
       return (Cmm, dflags, maybe_loc, output_fn)
798

799
runPhase Cmm stop dflags basename suff input_fn get_output_fn maybe_loc
800
  = do
801
802
	let hsc_lang = hscMaybeAdjustTarget dflags stop HsSrcFile (hscTarget dflags)
	let next_phase = hscNextPhase dflags HsSrcFile hsc_lang
803
804
	output_fn <- get_output_fn next_phase maybe_loc

805
        let dflags' = dflags { hscTarget = hsc_lang,
806
807
			       hscOutName = output_fn,
			       extCoreName = basename ++ ".hcr" }
808

809
	ok <- hscCmmFile dflags' input_fn
810
811
812

	when (not ok) $ throwDyn (PhaseFailed "cmm" (ExitFailure 1))

813
	return (next_phase, dflags, maybe_loc, output_fn)
814

815
816
817
818
819
820
-----------------------------------------------------------------------------
-- Cc phase

-- we don't support preprocessing .c files (with -E) now.  Doing so introduces
-- way too many hacks, and I can't say I've ever used it anyway.

821
runPhase cc_phase stop dflags basename suff input_fn get_output_fn maybe_loc
822
   | cc_phase `eqPhase` Cc || cc_phase `eqPhase` HCc
823
   = do	let cc_opts = getOpts dflags opt_c
824
	    hcc = cc_phase `eqPhase` HCc
825

826
       	let cmdline_include_paths = includePaths dflags
827

828
829
830
831
832
833
	-- HC files have the dependent packages stamped into them
	pkgs <- if hcc then getHCFilePackages input_fn else return []

	-- add package include paths even if we're just compiling .c
	-- files; this is the Value Add(TM) that using ghc instead of
	-- gcc gives you :)
834
        pkg_include_dirs <- getPackageIncludePath dflags pkgs
rrt's avatar
rrt committed
835
836
        let include_paths = foldr (\ x xs -> "-I" : x : xs) []
			      (cmdline_include_paths ++ pkg_include_dirs)
837

838
839
	let (md_c_flags, md_regd_c_flags) = machdepCCOpts dflags
        let pic_c_flags = picCCOpts dflags
840

841
        let verb = getVerbFlag dflags
842

843
	pkg_extra_cc_opts <- getPackageExtraCcOpts dflags pkgs
844

845
846
	let split_objs = dopt Opt_SplitObjs dflags
	    split_opt | hcc && split_objs = [ "-DUSE_SPLIT_MARKERS" ]
sof's avatar
sof committed
847
		      | otherwise         = [ ]
848

849
	let excessPrecision = dopt Opt_ExcessPrecision dflags
850

Simon Marlow's avatar
Simon Marlow committed
851
852
853
	let cc_opt | optLevel dflags >= 2 = "-O2"
		   | otherwise            = "-O"

854
	-- Decide next phase
855
856
857
	
        let mangle = dopt Opt_DoAsmMangling dflags
            next_phase
858
859
860
861
		| hcc && mangle     = Mangle
		| otherwise         = As
	output_fn <- get_output_fn next_phase maybe_loc

862
863
	let
	  more_hcc_opts =
864
#if i386_TARGET_ARCH
865
866
867
868
869
870
871
872
873
874
875
876
877
878
	   	-- on x86 the floating point regs have greater precision
	     	-- than a double, which leads to unpredictable results.
		-- By default, we turn this off with -ffloat-store unless
		-- the user specified -fexcess-precision.
		(if excessPrecision then [] else [ "-ffloat-store" ]) ++
#endif
		-- gcc's -fstrict-aliasing allows two accesses to memory
		-- to be considered non-aliasing if they have different types.
		-- This interacts badly with the C code we generate, which is
		-- very weakly typed, being derived from C--.
		["-fno-strict-aliasing"]



879
880
881
882
883
884
	SysTools.runCc dflags (
		-- force the C compiler to interpret this file as C when
		-- compiling .hc files, by adding the -x c option.
		-- Also useful for plain .c files, just in case GHC saw a 
		-- -x c option.
			[ SysTools.Option "-x", SysTools.Option "c"] ++
885
			[ SysTools.FileOption "" input_fn
sof's avatar
sof committed
886
			, SysTools.Option "-o"
sof's avatar
sof committed
887
			, SysTools.FileOption "" output_fn
sof's avatar
sof committed
888
889
890
			]
		       ++ map SysTools.Option (
		          md_c_flags
891
                       ++ pic_c_flags
892
		       ++ (if hcc && mangle
893
894
		  	     then md_regd_c_flags
		  	     else [])
895
896
897
		       ++ (if hcc 
			     then more_hcc_opts
			     else [])
Simon Marlow's avatar
Simon Marlow committed
898
		       ++ [ verb, "-S", "-Wimplicit", cc_opt ]
899
900
		       ++ [ "-D__GLASGOW_HASKELL__="++cProjectVersionInt ]
		       ++ cc_opts
sof's avatar
sof committed
901
		       ++ split_opt
902
903
		       ++ include_paths
		       ++ pkg_extra_cc_opts
sof's avatar
sof committed
904
		       ))
905

906
	return (next_phase, dflags, maybe_loc, output_fn)
907
908
909
910
911
912

	-- ToDo: postprocess the output from gcc

-----------------------------------------------------------------------------
-- Mangle phase

913
runPhase Mangle stop dflags _basename _suff input_fn get_output_fn maybe_loc
914
   = do let mangler_opts = getOpts dflags opt_m
915
916
917
918
919
920

#if i386_TARGET_ARCH
        machdep_opts <- return [ show (stolen_x86_regs dflags) ]
#else
	machdep_opts <- return []
#endif
921

922
923
	let split = dopt Opt_SplitObjs dflags
            next_phase
924
925
		| split = SplitMangle
		| otherwise = As
926
	output_fn <- get_output_fn next_phase maybe_loc
927

928
	SysTools.runMangle dflags (map SysTools.Option mangler_opts
sof's avatar
sof committed
929
930
		          ++ [ SysTools.FileOption "" input_fn
			     , SysTools.FileOption "" output_fn
sof's avatar
sof committed
931
932
			     ]
			  ++ map SysTools.Option machdep_opts)
933

934
	return (next_phase, dflags, maybe_loc, output_fn)