diff options
| author | Michael D. Adams <t-madams@microsoft.com> | 2007-06-27 15:21:30 +0000 | 
|---|---|---|
| committer | Michael D. Adams <t-madams@microsoft.com> | 2007-06-27 15:21:30 +0000 | 
| commit | d31dfb32ea936c22628b508c28a36c12e631430a (patch) | |
| tree | 76bc1a29b3c5646a8f552af820a81abff49aa492 /compiler | |
| parent | c9c4951cc1d76273be541fc4791e131e418956aa (diff) | |
| download | haskell-d31dfb32ea936c22628b508c28a36c12e631430a.tar.gz | |
Implemented and fixed bugs in CmmInfo handling
Diffstat (limited to 'compiler')
| -rw-r--r-- | compiler/cmm/CLabel.hs | 8 | ||||
| -rw-r--r-- | compiler/cmm/Cmm.hs | 12 | ||||
| -rw-r--r-- | compiler/cmm/CmmCPS.hs | 12 | ||||
| -rw-r--r-- | compiler/cmm/CmmInfo.hs | 236 | ||||
| -rw-r--r-- | compiler/cmm/CmmParse.y | 49 | ||||
| -rw-r--r-- | compiler/cmm/PprC.hs | 10 | ||||
| -rw-r--r-- | compiler/cmm/PprCmm.hs | 19 | ||||
| -rw-r--r-- | compiler/codeGen/CgBindery.lhs | 12 | ||||
| -rw-r--r-- | compiler/codeGen/CgCallConv.hs | 64 | ||||
| -rw-r--r-- | compiler/codeGen/CgClosure.lhs | 2 | ||||
| -rw-r--r-- | compiler/codeGen/CgForeignCall.hs | 15 | ||||
| -rw-r--r-- | compiler/codeGen/CgInfoTbls.hs | 383 | ||||
| -rw-r--r-- | compiler/codeGen/CgProf.hs | 3 | ||||
| -rw-r--r-- | compiler/codeGen/CgUtils.hs | 29 | ||||
| -rw-r--r-- | compiler/codeGen/ClosureInfo.lhs | 9 | ||||
| -rw-r--r-- | compiler/main/HscMain.lhs | 7 | ||||
| -rw-r--r-- | compiler/nativeGen/AsmCodeGen.lhs | 21 | ||||
| -rw-r--r-- | compiler/nativeGen/MachCodeGen.hs | 4 | 
18 files changed, 471 insertions, 424 deletions
| diff --git a/compiler/cmm/CLabel.hs b/compiler/cmm/CLabel.hs index 94ae64af55..ffca61d0a0 100644 --- a/compiler/cmm/CLabel.hs +++ b/compiler/cmm/CLabel.hs @@ -521,6 +521,8 @@ externallyVisibleCLabel (CCS_Label _)	   = True  externallyVisibleCLabel (DynamicLinkerLabel _ _)  = False  externallyVisibleCLabel (HpcTicksLabel _)   = True  externallyVisibleCLabel HpcModuleNameLabel      = False +externallyVisibleCLabel (LargeBitmapLabel _) = False +externallyVisibleCLabel (LargeSRTLabel _) = False  -- -----------------------------------------------------------------------------  -- Finding the "type" of a CLabel  @@ -702,7 +704,11 @@ pprCLbl (CaseLabel u CaseDefault)    = hcat [pprUnique u, ptext SLIT("_dflt")]  pprCLbl (LargeSRTLabel u)  = pprUnique u <> pp_cSEP <> ptext SLIT("srtd") -pprCLbl (LargeBitmapLabel u)  = pprUnique u <> pp_cSEP <> ptext SLIT("btm") +pprCLbl (LargeBitmapLabel u)  = text "b" <> pprUnique u <> pp_cSEP <> ptext SLIT("btm") +-- Some bitsmaps for tuple constructors have a numeric tag (e.g. '7') +-- until that gets resolved we'll just force them to start +-- with a letter so the label will be legal assmbly code. +          pprCLbl (RtsLabel (RtsCode str))   = ptext str  pprCLbl (RtsLabel (RtsData str))   = ptext str diff --git a/compiler/cmm/Cmm.hs b/compiler/cmm/Cmm.hs index 903853489f..530fab570d 100644 --- a/compiler/cmm/Cmm.hs +++ b/compiler/cmm/Cmm.hs @@ -9,9 +9,10 @@  module Cmm (   	GenCmm(..), Cmm, RawCmm,  	GenCmmTop(..), CmmTop, RawCmmTop, -	CmmInfo(..), ClosureTypeInfo(..), ProfilingInfo(..), +	CmmInfo(..), ClosureTypeInfo(..), ProfilingInfo(..), ClosureTypeTag,  	GenBasicBlock(..), CmmBasicBlock, blockId, blockStmts,  	CmmStmt(..), CmmActuals, CmmFormal, CmmFormals, CmmHintFormals, +        CmmSafety(..),  	CmmCallTarget(..),  	CmmStatic(..), Section(..),  	CmmExpr(..), cmmExprRep,  @@ -133,12 +134,14 @@ data ClosureTypeInfo  -- TODO: These types may need refinement  data ProfilingInfo = ProfilingInfo CmmLit CmmLit -- closure_type, closure_desc  type ClosureTypeTag = StgHalfWord -type ClosureLayout = (StgHalfWord, StgHalfWord) -- pts, nptrs +type ClosureLayout = (StgHalfWord, StgHalfWord) -- ptrs, nptrs  type ConstrTag = StgHalfWord  type ConstrDescription = CmmLit  type FunType = StgHalfWord  type FunArity = StgHalfWord -type SlowEntry = CLabel +type SlowEntry = CmmLit +  -- ^We would like this to be a CLabel but +  -- for now the parser sets this to zero on an INFO_TABLE_FUN.  type SelectorOffset = StgWord  ----------------------------------------------------------------------------- @@ -161,7 +164,7 @@ data CmmStmt       CmmCallTarget       CmmHintFormals		 -- zero or more results       CmmActuals			 -- zero or more arguments -     C_SRT			 -- SRT for the continuation of the call +     CmmSafety			 -- whether to build a continuation    | CmmBranch BlockId             -- branch to another BB in this fn @@ -184,6 +187,7 @@ type CmmActuals = [(CmmActual,MachHint)]  type CmmFormal = LocalReg  type CmmHintFormals = [(CmmFormal,MachHint)]  type CmmFormals = [CmmFormal] +data CmmSafety = CmmUnsafe | CmmSafe C_SRT  {-  Discussion diff --git a/compiler/cmm/CmmCPS.hs b/compiler/cmm/CmmCPS.hs index be9f474cbe..b6c57eea9d 100644 --- a/compiler/cmm/CmmCPS.hs +++ b/compiler/cmm/CmmCPS.hs @@ -70,9 +70,9 @@ cmmCPS dflags abstractC = do    return continuationC  stg_gc_gen = mkRtsApFastLabel SLIT("gen_cg_TODO") --panic "Need the label for gc" -make_gc_block block_id fun_label formals srt = BasicBlock block_id stmts +make_gc_block block_id fun_label formals safety = BasicBlock block_id stmts      where -      stmts = [CmmCall stg_gc_gen_target [] [] srt, +      stmts = [CmmCall stg_gc_gen_target [] [] safety,                 CmmJump fun_expr actuals]        stg_gc_gen_target =            CmmForeignCall (CmmLit (CmmLabel stg_gc_gen)) CmmCallConv @@ -85,10 +85,10 @@ force_gc_block old_info block_id fun_label formals blocks =        CmmInfo _ (Just _) _ _ -> (old_info, [])        CmmNonInfo Nothing            -> (CmmNonInfo (Just block_id), -              [make_gc_block block_id fun_label formals NoC_SRT]) +              [make_gc_block block_id fun_label formals (CmmSafe NoC_SRT)])        CmmInfo prof Nothing type_tag type_info          -> (CmmInfo prof (Just block_id) type_tag type_info, -            [make_gc_block block_id fun_label formals srt]) +            [make_gc_block block_id fun_label formals (CmmSafe srt)])             where               srt = case type_info of                       ConstrInfo _ _ _ -> NoC_SRT @@ -361,9 +361,7 @@ applyStackFormat formats (Continuation (Left srt) label formals blocks) =        -- TODO prof: this is the same as the current implementation        -- but I think it could be improved        prof = ProfilingInfo zeroCLit zeroCLit -      tag = if stack_frame_size format > mAX_SMALL_BITMAP_SIZE -            then rET_BIG -            else rET_SMALL +      tag = rET_SMALL -- cmmToRawCmm will convert this to rET_BIG if needed        format = maybe unknown_block id $ lookup label formats        unknown_block = panic "unknown BlockId in applyStackFormat" diff --git a/compiler/cmm/CmmInfo.hs b/compiler/cmm/CmmInfo.hs index ab46f1e58d..5937dd4fb9 100644 --- a/compiler/cmm/CmmInfo.hs +++ b/compiler/cmm/CmmInfo.hs @@ -1,4 +1,5 @@  module CmmInfo ( +  cmmToRawCmm,    mkInfoTable  ) where @@ -6,30 +7,81 @@ module CmmInfo (  import Cmm  import CmmUtils +import PprCmm  import CLabel +import MachOp  import Bitmap  import ClosureInfo  import CgInfoTbls  import CgCallConv  import CgUtils +import SMRep  import Constants  import StaticFlags +import DynFlags  import Unique +import UniqSupply  import Panic  import Data.Bits +cmmToRawCmm :: [Cmm] -> IO [RawCmm] +cmmToRawCmm cmm = do +  info_tbl_uniques <- mkSplitUniqSupply 'i' +  return $ zipWith raw_cmm (listSplitUniqSupply info_tbl_uniques) cmm +    where +      raw_cmm uniq_supply (Cmm procs) = +          Cmm $ concat $ zipWith mkInfoTable (uniqsFromSupply uniq_supply) procs + +-- Make a concrete info table, represented as a list of CmmStatic +-- (it can't be simply a list of Word, because the SRT field is +-- represented by a label+offset expression). +-- +-- With tablesNextToCode, the layout is +--	<reversed variable part> +--	<normal forward StgInfoTable, but without  +--		an entry point at the front> +--	<code> +-- +-- Without tablesNextToCode, the layout of an info table is +--	<entry label> +--	<normal forward rest of StgInfoTable> +--	<forward variable part> +-- +--	See includes/InfoTables.h +-- +-- For return-points these are as follows +-- +-- Tables next to code: +-- +--			<srt slot> +--			<standard info table> +--  	ret-addr -->	<entry code (if any)> +-- +-- Not tables-next-to-code: +-- +--	ret-addr -->	<ptr to entry code> +--			<standard info table> +--			<srt slot> +-- +--  * The SRT slot is only there if there is SRT info to record +  mkInfoTable :: Unique -> CmmTop -> [RawCmmTop]  mkInfoTable uniq (CmmData sec dat) = [CmmData sec dat]  mkInfoTable uniq (CmmProc info entry_label arguments blocks) =      case info of +      -- | Code without an info table.  Easy.        CmmNonInfo _ -> [CmmProc [] entry_label arguments blocks] + +      -- | A function entry point.        CmmInfo (ProfilingInfo ty_prof cl_prof) _ type_tag -              (FunInfo (ptrs, nptrs) srt fun_type fun_arity pap_bitmap slow_entry) -> -          mkInfoTableAndCode info_label std_info fun_extra_bits entry_label arguments blocks +              (FunInfo (ptrs, nptrs) srt fun_type fun_arity +                       pap_bitmap slow_entry) -> +          mkInfoTableAndCode info_label std_info fun_extra_bits entry_label +                             arguments blocks            where              fun_extra_bits =                 [packHalfWordsCLit fun_type fun_arity] ++ @@ -37,71 +89,74 @@ mkInfoTable uniq (CmmProc info entry_label arguments blocks) =                 case pap_bitmap of                   ArgGen liveness ->                       [makeRelativeRefTo info_label $ mkLivenessCLit liveness, -                      makeRelativeRefTo info_label (CmmLabel slow_entry)] +                      makeRelativeRefTo info_label slow_entry]                   _ -> []              std_info = mkStdInfoTable ty_prof cl_prof type_tag srt_bitmap layout              info_label = entryLblToInfoLbl entry_label -            (srt_label, srt_bitmap) = -                case srt of -                  NoC_SRT -> ([], 0) -                  (C_SRT lbl off bitmap) -> -                      ([makeRelativeRefTo info_label (cmmLabelOffW lbl off)], -                       bitmap) +            (srt_label, srt_bitmap) = mkSRTLit info_label srt              layout = packHalfWordsCLit ptrs nptrs +      -- | A constructor.        CmmInfo (ProfilingInfo ty_prof cl_prof) _ type_tag                (ConstrInfo (ptrs, nptrs) con_tag descr) -> -          mkInfoTableAndCode info_label std_info [con_name] entry_label arguments blocks +          mkInfoTableAndCode info_label std_info [con_name] entry_label +                             arguments blocks            where              std_info = mkStdInfoTable ty_prof cl_prof type_tag con_tag layout              info_label = entryLblToInfoLbl entry_label              con_name = makeRelativeRefTo info_label descr              layout = packHalfWordsCLit ptrs nptrs +      -- | A thunk.        CmmInfo (ProfilingInfo ty_prof cl_prof) _ type_tag                (ThunkInfo (ptrs, nptrs) srt) -> -          mkInfoTableAndCode info_label std_info srt_label entry_label arguments blocks +          mkInfoTableAndCode info_label std_info srt_label entry_label +                             arguments blocks            where              std_info = mkStdInfoTable ty_prof cl_prof type_tag srt_bitmap layout              info_label = entryLblToInfoLbl entry_label -            (srt_label, srt_bitmap) = -                case srt of -                  NoC_SRT -> ([], 0) -                  (C_SRT lbl off bitmap) -> -                      ([makeRelativeRefTo info_label (cmmLabelOffW lbl off)], -                       bitmap) +            (srt_label, srt_bitmap) = mkSRTLit info_label srt              layout = packHalfWordsCLit ptrs nptrs +      -- | A selector thunk.        CmmInfo (ProfilingInfo ty_prof cl_prof) _ type_tag                (ThunkSelectorInfo offset srt) -> -          mkInfoTableAndCode info_label std_info srt_label entry_label arguments blocks +          mkInfoTableAndCode info_label std_info srt_label entry_label +                             arguments blocks            where              std_info = mkStdInfoTable ty_prof cl_prof type_tag srt_bitmap (mkWordCLit offset)              info_label = entryLblToInfoLbl entry_label -            (srt_label, srt_bitmap) = -                case srt of -                  NoC_SRT -> ([], 0) -                  (C_SRT lbl off bitmap) -> -                      ([makeRelativeRefTo info_label (cmmLabelOffW lbl off)], -                       bitmap) +            (srt_label, srt_bitmap) = mkSRTLit info_label srt +      -- A continuation/return-point.        CmmInfo (ProfilingInfo ty_prof cl_prof) _ type_tag (ContInfo stack_layout srt) ->            liveness_data ++ -          mkInfoTableAndCode info_label std_info srt_label entry_label arguments blocks +          mkInfoTableAndCode info_label std_info srt_label entry_label +                             arguments blocks            where -            std_info = mkStdInfoTable ty_prof cl_prof type_tag srt_bitmap liveness_lit +            std_info = mkStdInfoTable ty_prof cl_prof maybe_big_type_tag srt_bitmap +                                      (makeRelativeRefTo info_label liveness_lit)              info_label = entryLblToInfoLbl entry_label -            (liveness_lit, liveness_data) = mkLiveness uniq stack_layout -            (srt_label, srt_bitmap) = -                case srt of -                  NoC_SRT -> ([], 0) -                  (C_SRT lbl off bitmap) -> -                      ([makeRelativeRefTo info_label (cmmLabelOffW lbl off)], -                       bitmap) +            (liveness_lit, liveness_data, liveness_tag) = +                mkLiveness uniq stack_layout +            maybe_big_type_tag = if type_tag == rET_SMALL +                                 then liveness_tag +                                 else type_tag +            (srt_label, srt_bitmap) = mkSRTLit info_label srt +-- Handle the differences between tables-next-to-code +-- and not tables-next-to-code +mkInfoTableAndCode :: CLabel +                   -> [CmmLit] +                   -> [CmmLit] +                   -> CLabel +                   -> CmmFormals +                   -> [CmmBasicBlock] +                   -> [RawCmmTop]  mkInfoTableAndCode info_lbl std_info extra_bits entry_lbl args blocks    | tablesNextToCode 	-- Reverse the extra_bits; and emit the top-level proc -  = [CmmProc (map CmmStaticLit (reverse extra_bits ++ std_info)) entry_lbl args blocks] +  = [CmmProc (map CmmStaticLit (reverse extra_bits ++ std_info)) +             entry_lbl args blocks]    | null blocks -- No actual code; only the info table is significant    =		-- Use a zero place-holder in place of the  @@ -113,27 +168,108 @@ mkInfoTableAndCode info_lbl std_info extra_bits entry_lbl args blocks      [mkDataLits info_lbl (CmmLabel entry_lbl : std_info ++ extra_bits),       CmmProc [] entry_lbl args blocks] +mkSRTLit :: CLabel +         -> C_SRT +         -> ([CmmLit],    -- srt_label +             StgHalfWord) -- srt_bitmap +mkSRTLit info_label NoC_SRT = ([], 0) +mkSRTLit info_label (C_SRT lbl off bitmap) = +    ([makeRelativeRefTo info_label (cmmLabelOffW lbl off)], bitmap) + +------------------------------------------------------------------------- +-- +--		Build a liveness mask for the stack layout +-- +------------------------------------------------------------------------- + +-- There are four kinds of things on the stack: +-- +--	- pointer variables (bound in the environment) +-- 	- non-pointer variables (bound in the environment) +-- 	- free slots (recorded in the stack free list) +-- 	- non-pointer data slots (recorded in the stack free list) +-- +-- The first two are represented with a 'Just' of a 'LocalReg'. +-- The last two with one or more 'Nothing' constructors. +-- Each 'Nothing' represents one used word. +-- +-- The head of the stack layout is the top of the stack and +-- the least-significant bit. +  -- TODO: refactor to use utility functions -mkLiveness :: Unique -> [Maybe LocalReg] -> (CmmLit, [GenCmmTop CmmStatic [CmmStatic] CmmStmt]) -mkLiveness uniq live -  = if length live > mAX_SMALL_BITMAP_SIZE -    then (CmmLabel big_liveness, [data_lits]) -- does not fit in one word -    else (mkWordCLit small_liveness, []) -- fits in one word +-- TODO: combine with CgCallConv.mkLiveness (see comment there) +mkLiveness :: Unique +           -> [Maybe LocalReg] +           -> (CmmLit,           -- ^ The bitmap (literal value or label) +               [RawCmmTop],      -- ^ Large bitmap CmmData if needed +               ClosureTypeTag)   -- ^ rET_SMALL or rET_BIG +mkLiveness uniq live = +  if length bits > mAX_SMALL_BITMAP_SIZE +    -- does not fit in one word +    then (CmmLabel big_liveness, [data_lits], rET_BIG) +    -- fits in one word +    else (mkWordCLit small_liveness, [], rET_SMALL)    where -    size = length live +    mkBits [] = [] +    mkBits (reg:regs) = take sizeW bits ++ mkBits regs where +        sizeW = case reg of +                  Nothing -> 1 +                  Just r -> machRepByteWidth (localRegRep r) `quot` wORD_SIZE +        bits = repeat $ is_non_ptr reg -- True <=> Non Ptr -    bits = mkBitmap (map is_non_ptr live)      is_non_ptr Nothing = True -    is_non_ptr (Just reg) | localRegGCFollow reg == KindNonPtr = True -    is_non_ptr (Just reg) | localRegGCFollow reg == KindPtr = False +    is_non_ptr (Just reg) = +        case localRegGCFollow reg of +          KindNonPtr -> True +          KindPtr -> False -    big_liveness = mkBitmapLabel uniq -    data_lits = mkRODataLits big_liveness lits -    lits = mkWordCLit (fromIntegral size) : map mkWordCLit bits -   -    small_liveness = -        fromIntegral size .|. (small_bits `shiftL` bITMAP_BITS_SHIFT) -    small_bits = case bits of  +    bits :: [Bool] +    bits = mkBits live + +    bitmap :: Bitmap +    bitmap = mkBitmap bits + +    small_bitmap = case bitmap of   		   []  -> 0  		   [b] -> fromIntegral b  		   _   -> panic "mkLiveness" +    small_liveness = +        fromIntegral (length bits) .|. (small_bitmap `shiftL` bITMAP_BITS_SHIFT) + +    big_liveness = mkBitmapLabel uniq +    lits = mkWordCLit (fromIntegral (length bits)) : map mkWordCLit bitmap +    data_lits = mkRODataLits big_liveness lits + +------------------------------------------------------------------------- +-- +--	Generating a standard info table +-- +------------------------------------------------------------------------- + +-- The standard bits of an info table.  This part of the info table +-- corresponds to the StgInfoTable type defined in InfoTables.h. +-- +-- Its shape varies with ticky/profiling/tables next to code etc +-- so we can't use constant offsets from Constants + +mkStdInfoTable +   :: CmmLit		-- closure type descr (profiling) +   -> CmmLit		-- closure descr (profiling) +   -> StgHalfWord	-- closure type +   -> StgHalfWord	-- SRT length +   -> CmmLit		-- layout field +   -> [CmmLit] + +mkStdInfoTable type_descr closure_descr cl_type srt_len layout_lit + = 	-- Parallel revertible-black hole field +    prof_info +	-- Ticky info (none at present) +	-- Debug info (none at present) + ++ [layout_lit, type_lit] + + where   +    prof_info  +	| opt_SccProfilingOn = [type_descr, closure_descr] +	| otherwise	     = [] + +    type_lit = packHalfWordsCLit cl_type srt_len diff --git a/compiler/cmm/CmmParse.y b/compiler/cmm/CmmParse.y index 7fc4c430f9..840b564a83 100644 --- a/compiler/cmm/CmmParse.y +++ b/compiler/cmm/CmmParse.y @@ -231,7 +231,9 @@ info	:: { ExtFCode (CLabel, CmmInfo) }  		{ do prof <- profilingInfo $11 $13  		     return (mkRtsInfoLabelFS $3,  			CmmInfo prof Nothing (fromIntegral $9) -				(FunInfo (fromIntegral $5, fromIntegral $7) NoC_SRT (fromIntegral $15) 0 (panic "INFO_TABLE_FUN:ArgDesr") (panic "INFO_TABLE_FUN:SlowEntry"))) } +				(FunInfo (fromIntegral $5, fromIntegral $7) NoC_SRT (fromIntegral $15) 0 +				 (ArgSpec 0) +				 zeroCLit)) }  		-- we leave most of the fields zero here.  This is only used  		-- to generate the BCO info table in the RTS at the moment. @@ -258,7 +260,7 @@ info	:: { ExtFCode (CLabel, CmmInfo) }  			CmmInfo (ProfilingInfo zeroCLit zeroCLit) Nothing (fromIntegral $5)  				(ContInfo [] NoC_SRT)) } -	| 'INFO_TABLE_RET' '(' NAME ',' INT ',' formals ')' +	| 'INFO_TABLE_RET' '(' NAME ',' INT ',' formals0 ')'  		-- closure type, live regs  		{ do live <- sequence (map (liftM Just) $7)  		     return (mkRtsInfoLabelFS $3, @@ -792,48 +794,6 @@ forkLabelledCodeEC ec = do    stmts <- getCgStmtsEC ec    code (forkCgStmts stmts) -retInfo name size live_bits cl_type = do -  let liveness = smallLiveness (fromIntegral size) (fromIntegral live_bits) -      info_lbl = mkRtsRetInfoLabelFS name -      (info1,info2) = mkRetInfoTable info_lbl liveness NoC_SRT  -				(fromIntegral cl_type) -  return (info_lbl, info1, info2) - -stdInfo name ptrs nptrs srt_bitmap cl_type desc_str ty_str = -  basicInfo name (packHalfWordsCLit ptrs nptrs)  -	srt_bitmap cl_type desc_str ty_str - -conInfo name ptrs nptrs srt_bitmap cl_type desc_str ty_str = do -  (lbl, info1, _) <- basicInfo name (packHalfWordsCLit ptrs nptrs)  -                	srt_bitmap cl_type desc_str ty_str -  desc_lit <- code $ mkStringCLit desc_str -  let desc_field = makeRelativeRefTo lbl desc_lit -  return (lbl, info1, [desc_field]) - -basicInfo name layout srt_bitmap cl_type desc_str ty_str = do -  let info_lbl = mkRtsInfoLabelFS name -  lit1 <- if opt_SccProfilingOn  -		   then code $ do lit <- mkStringCLit desc_str -                                  return (makeRelativeRefTo info_lbl lit) -		   else return (mkIntCLit 0) -  lit2 <- if opt_SccProfilingOn  -		   then code $ do lit <- mkStringCLit ty_str -                                  return (makeRelativeRefTo info_lbl lit) -		   else return (mkIntCLit 0) -  let info1 = mkStdInfoTable lit1 lit2 (fromIntegral cl_type)  -			(fromIntegral srt_bitmap) -			layout -  return (info_lbl, info1, []) - -funInfo name ptrs nptrs cl_type desc_str ty_str fun_type = do -  (label,info1,_) <- stdInfo name ptrs nptrs 0{-srt_bitmap-} -			 cl_type desc_str ty_str  -  let info2 = mkFunGenInfoExtraBits (fromIntegral fun_type) 0 zero zero zero -		-- we leave most of the fields zero here.  This is only used -		-- to generate the BCO info table in the RTS at the moment. -  return (label,info1,info2) - where -   zero = mkIntCLit 0  profilingInfo desc_str ty_str = do    lit1 <- if opt_SccProfilingOn  @@ -907,6 +867,7 @@ emitRetUT args = do    emitStmts stmts    when (sp /= 0) $ stmtC (CmmAssign spReg (cmmRegOffW spReg (-sp)))    stmtC (CmmJump (entryCode (CmmLoad (cmmRegOffW spReg sp) wordRep)) []) +  -- TODO (when using CPS): emitStmt (CmmReturn (map snd args))  -- -----------------------------------------------------------------------------  -- If-then-else and boolean expressions diff --git a/compiler/cmm/PprC.hs b/compiler/cmm/PprC.hs index 8726547ae9..1a909f26d3 100644 --- a/compiler/cmm/PprC.hs +++ b/compiler/cmm/PprC.hs @@ -199,11 +199,11 @@ pprStmt stmt = case stmt of  	where  	  rep = cmmExprRep src -    CmmCall (CmmForeignCall fn cconv) results args srt -> +    CmmCall (CmmForeignCall fn cconv) results args safety ->  	-- Controversial: leave this out for now.  	-- pprUndef fn $$ -	pprCall ppr_fn cconv results args srt +	pprCall ppr_fn cconv results args safety  	where      	ppr_fn = case fn of  		   CmmLit (CmmLabel lbl) -> pprCLabel lbl @@ -220,8 +220,8 @@ pprStmt stmt = case stmt of  	   ptext SLIT("#undef") <+> pprCLabel lbl  	pprUndef _ = empty -    CmmCall (CmmPrim op) results args srt -> -	pprCall ppr_fn CCallConv results args srt +    CmmCall (CmmPrim op) results args safety -> +	pprCall ppr_fn CCallConv results args safety  	where      	ppr_fn = pprCallishMachOp_for_C op @@ -719,7 +719,7 @@ pprLocalReg (LocalReg uniq _ _) = char '_' <> ppr uniq  -- -----------------------------------------------------------------------------  -- Foreign Calls -pprCall :: SDoc -> CCallConv -> CmmHintFormals -> CmmActuals -> C_SRT +pprCall :: SDoc -> CCallConv -> CmmHintFormals -> CmmActuals -> CmmSafety  	-> SDoc  pprCall ppr_fn cconv results args _ diff --git a/compiler/cmm/PprCmm.hs b/compiler/cmm/PprCmm.hs index 97170a1c33..163c86bcc7 100644 --- a/compiler/cmm/PprCmm.hs +++ b/compiler/cmm/PprCmm.hs @@ -117,7 +117,10 @@ pprTop (CmmData section ds) =      (hang (pprSection section <+> lbrace) 4 (vcat (map pprStatic ds)))      $$ rbrace - +-- -------------------------------------------------------------------------- +instance Outputable CmmSafety where +  ppr CmmUnsafe = ptext SLIT("_unsafe_call_") +  ppr (CmmSafe srt) = ppr srt  -- --------------------------------------------------------------------------  -- Info tables. The current pretty printer needs refinement @@ -128,13 +131,15 @@ pprTop (CmmData section ds) =  -- and were labelled with the procedure name ++ "_info".  pprInfo (CmmNonInfo gc_target) =      ptext SLIT("gc_target: ") <> -          maybe (ptext SLIT("<none>")) pprBlockId gc_target +          ptext SLIT("TODO") --maybe (ptext SLIT("<none>")) pprBlockId gc_target +          -- ^ gc_target is currently unused and wired to a panic  pprInfo (CmmInfo (ProfilingInfo closure_type closure_desc)                   gc_target tag info) =      vcat [ptext SLIT("type: ") <> pprLit closure_type,            ptext SLIT("desc: ") <> pprLit closure_desc,            ptext SLIT("gc_target: ") <> -                maybe (ptext SLIT("<none>")) pprBlockId gc_target, +                ptext SLIT("TODO"), --maybe (ptext SLIT("<none>")) pprBlockId gc_target, +                -- ^ gc_target is currently unused and wired to a panic            ptext SLIT("tag: ") <> integer (toInteger tag),            pprTypeInfo info] @@ -192,7 +197,7 @@ pprStmt stmt = case stmt of      -- call "ccall" foo(x, y)[r1, r2];      -- ToDo ppr volatile -    CmmCall (CmmForeignCall fn cconv) results args srt -> +    CmmCall (CmmForeignCall fn cconv) results args safety ->          hcat [ if null results                    then empty                    else parens (commafy $ map ppr results) <> @@ -200,14 +205,14 @@ pprStmt stmt = case stmt of                 ptext SLIT("call"), space,                  doubleQuotes(ppr cconv), space,                 target fn, parens  ( commafy $ map ppr args ), -               brackets (ppr srt), semi ] +               brackets (ppr safety), semi ]          where              target (CmmLit lit) = pprLit lit              target fn'          = parens (ppr fn') -    CmmCall (CmmPrim op) results args srt -> +    CmmCall (CmmPrim op) results args safety ->          pprStmt (CmmCall (CmmForeignCall (CmmLit lbl) CCallConv) -                        results args srt) +                        results args safety)          where            lbl = CmmLabel (mkForeignLabel (mkFastString (show op)) Nothing False) diff --git a/compiler/codeGen/CgBindery.lhs b/compiler/codeGen/CgBindery.lhs index 66ac9bf491..d5a2c69d60 100644 --- a/compiler/codeGen/CgBindery.lhs +++ b/compiler/codeGen/CgBindery.lhs @@ -19,6 +19,7 @@ module CgBindery (  	nukeVolatileBinds,  	nukeDeadBindings,  	getLiveStackSlots, +        getLiveStackBindings,  	bindArgsToStack,  rebindToStack,  	bindNewToNode, bindNewToReg, bindArgsToRegs, @@ -494,3 +495,14 @@ getLiveStackSlots  				   cg_rep = rep } <- varEnvElts binds,   		        isFollowableArg rep] }  \end{code} + +\begin{code} +getLiveStackBindings :: FCode [(VirtualSpOffset, CgIdInfo)] +getLiveStackBindings +  = do { binds <- getBinds +       ; return [(off, bind) | +                 bind <- varEnvElts binds, +                 CgIdInfo { cg_stb = VirStkLoc off, +                            cg_rep = rep} <- [bind], +                 isFollowableArg rep] } +\end{code} diff --git a/compiler/codeGen/CgCallConv.hs b/compiler/codeGen/CgCallConv.hs index b0fab89f82..34c9bee026 100644 --- a/compiler/codeGen/CgCallConv.hs +++ b/compiler/codeGen/CgCallConv.hs @@ -15,7 +15,7 @@ module CgCallConv (  	mkArgDescr, argDescrType,  	-- Liveness -	isBigLiveness, buildContLiveness, mkRegLiveness,  +	isBigLiveness, mkRegLiveness,   	smallLiveness, mkLivenessCLit,  	-- Register assignment @@ -71,7 +71,7 @@ import Data.Bits  #include "../includes/StgFun.h"  ------------------------- -argDescrType :: ArgDescr -> Int +argDescrType :: ArgDescr -> StgHalfWord  -- The "argument type" RTS field type  argDescrType (ArgSpec n) = n  argDescrType (ArgGen liveness) @@ -98,7 +98,7 @@ argBits [] 		= []  argBits (PtrArg : args) = False : argBits args  argBits (arg    : args) = take (cgRepSizeW arg) (repeat True) ++ argBits args -stdPattern :: [CgRep] -> Maybe Int +stdPattern :: [CgRep] -> Maybe StgHalfWord  stdPattern []          = Just ARG_NONE	-- just void args, probably  stdPattern [PtrArg]    = Just ARG_P @@ -133,6 +133,14 @@ stdPattern other = Nothing  --  ------------------------------------------------------------------------- +-- TODO: This along with 'mkArgDescr' should be unified +-- with 'CmmInfo.mkLiveness'.  However that would require +-- potentially invasive changes to the 'ClosureInfo' type. +-- For now, 'CmmInfo.mkLiveness' handles only continuations and +-- this one handles liveness everything else.  Another distinction +-- between these two is that 'CmmInfo.mkLiveness' information +-- about the stack layout, and this one is information about +-- the heap layout of PAPs.  mkLiveness :: Name -> Int -> Bitmap -> FCode Liveness  mkLiveness name size bits    | size > mAX_SMALL_BITMAP_SIZE		-- Bitmap does not fit in one word @@ -284,56 +292,6 @@ getSequelAmode  -------------------------------------------------------------------------  -- ---		Build a liveness mask for the current stack --- -------------------------------------------------------------------------- - --- There are four kinds of things on the stack: --- ---	- pointer variables (bound in the environment) --- 	- non-pointer variables (bound in the environment) --- 	- free slots (recorded in the stack free list) --- 	- non-pointer data slots (recorded in the stack free list) ---  --- We build up a bitmap of non-pointer slots by searching the environment --- for all the pointer variables, and subtracting these from a bitmap --- with initially all bits set (up to the size of the stack frame). - -buildContLiveness :: Name		-- Basis for label (only) -		  -> [VirtualSpOffset] 	-- Live stack slots -		  -> FCode Liveness -buildContLiveness name live_slots - = do	{ stk_usg    <- getStkUsage -	; let	StackUsage { realSp = real_sp,  -			     frameSp = frame_sp } = stk_usg - -		start_sp :: VirtualSpOffset -		start_sp = real_sp - retAddrSizeW -		-- In a continuation, we want a liveness mask that  -		-- starts from just after the return address, which is  -		-- on the stack at real_sp. - -		frame_size :: WordOff -		frame_size = start_sp - frame_sp -		-- real_sp points to the frame-header for the current -		-- stack frame, and the end of this frame is frame_sp. -		-- The size is therefore real_sp - frame_sp - retAddrSizeW -		-- (subtract one for the frame-header = return address). -	 -		rel_slots :: [WordOff] -	 	rel_slots = sortLe (<=)  -	    	    [ start_sp - ofs  -- Get slots relative to top of frame -	    	    | ofs <- live_slots ] - -		bitmap = intsToReverseBitmap frame_size rel_slots - -	; WARN( not (all (>=0) rel_slots),  -		ppr name $$ ppr live_slots $$ ppr frame_size $$ ppr start_sp $$ ppr rel_slots ) -	  mkLiveness name frame_size bitmap } - - -------------------------------------------------------------------------- ---  --		Register assignment  --  ------------------------------------------------------------------------- diff --git a/compiler/codeGen/CgClosure.lhs b/compiler/codeGen/CgClosure.lhs index 2c72860a29..98e5b0d0f2 100644 --- a/compiler/codeGen/CgClosure.lhs +++ b/compiler/codeGen/CgClosure.lhs @@ -533,7 +533,7 @@ link_caf cl_info is_upd = do  	-- so that the garbage collector can find them  	-- This must be done *before* the info table pointer is overwritten,   	-- because the old info table ptr is needed for reversion -  ; emitRtsCallWithVols SLIT("newCAF") [(CmmReg nodeReg,PtrHint)] [node] +  ; emitRtsCallWithVols SLIT("newCAF") [(CmmReg nodeReg,PtrHint)] [node] False  	-- node is live, so save it.  	-- Overwrite the closure with a (static) indirection  diff --git a/compiler/codeGen/CgForeignCall.hs b/compiler/codeGen/CgForeignCall.hs index b2ca5b166a..5d84da773c 100644 --- a/compiler/codeGen/CgForeignCall.hs +++ b/compiler/codeGen/CgForeignCall.hs @@ -116,7 +116,7 @@ emitForeignCall' safety results target args vols srt      temp_args <- load_args_into_temps args      let (caller_save, caller_load) = callerSaveVolatileRegs vols      stmtsC caller_save -    stmtC (CmmCall target results temp_args srt) +    stmtC (CmmCall target results temp_args CmmUnsafe)      stmtsC caller_load    | otherwise = do @@ -129,17 +129,20 @@ emitForeignCall' safety results target args vols srt      let (caller_save, caller_load) = callerSaveVolatileRegs vols      emitSaveThreadState      stmtsC caller_save -    -- Using the same SRT for each of these is a little bit conservative -    -- but it should work for now. +    -- The CmmUnsafe arguments are only correct because this part +    -- of the code hasn't been moved into the CPS pass yet. +    -- Once that happens, this function will just emit a (CmmSafe srt) call, +    -- and the CPS will will be the one to convert that +    -- to this sequence of three CmmUnsafe calls.      stmtC (CmmCall (CmmForeignCall suspendThread CCallConv)   			[ (id,PtrHint) ]  			[ (CmmReg (CmmGlobal BaseReg), PtrHint) ]  -			srt) -    stmtC (CmmCall temp_target results temp_args srt) +			CmmUnsafe) +    stmtC (CmmCall temp_target results temp_args CmmUnsafe)      stmtC (CmmCall (CmmForeignCall resumeThread CCallConv)   			[ (new_base, PtrHint) ]  			[ (CmmReg (CmmLocal id), PtrHint) ] -			srt) +			CmmUnsafe)      -- Assign the result to BaseReg: we      -- might now have a different Capability!      stmtC (CmmAssign (CmmGlobal BaseReg) (CmmReg (CmmLocal new_base))) diff --git a/compiler/codeGen/CgInfoTbls.hs b/compiler/codeGen/CgInfoTbls.hs index 6b7fcd563e..6d270aef16 100644 --- a/compiler/codeGen/CgInfoTbls.hs +++ b/compiler/codeGen/CgInfoTbls.hs @@ -12,10 +12,7 @@ module CgInfoTbls (  	dataConTagZ,  	emitReturnTarget, emitAlgReturnTarget,  	emitReturnInstr, -	mkRetInfoTable, -	mkStdInfoTable,  	stdInfoTableSizeB, -	mkFunGenInfoExtraBits,  	entryCode, closureInfoPtr,  	getConstrTag,  	infoTable, infoTableClosureType, @@ -46,6 +43,8 @@ import StaticFlags  import Maybes  import Constants  import Panic +import Util +import Outputable  -------------------------------------------------------------------------  -- @@ -53,114 +52,80 @@ import Panic  --  ------------------------------------------------------------------------- --- Here we make a concrete info table, represented as a list of CmmAddr --- (it can't be simply a list of Word, because the SRT field is --- represented by a label+offset expression). - --- With tablesNextToCode, the layout is ---	<reversed variable part> ---	<normal forward StgInfoTable, but without  ---		an entry point at the front> ---	<code> --- --- Without tablesNextToCode, the layout of an info table is ---	<entry label> ---	<normal forward rest of StgInfoTable> ---	<forward variable part> --- ---	See includes/InfoTables.h +-- Here we make an info table of type 'CmmInfo'.  The concrete +-- representation as a list of 'CmmAddr' is handled later +-- in the pipeline by 'cmmToRawCmm'.  emitClosureCodeAndInfoTable :: ClosureInfo -> CmmFormals -> CgStmts -> Code  emitClosureCodeAndInfoTable cl_info args body - = do	{ ty_descr_lit <-  -		if opt_SccProfilingOn  -		   then do lit <- mkStringCLit (closureTypeDescr cl_info) -                           return (makeRelativeRefTo info_lbl lit) -		   else return (mkIntCLit 0) -  	; cl_descr_lit <-  -		if opt_SccProfilingOn  -		   then do lit <- mkStringCLit cl_descr_string -                           return (makeRelativeRefTo info_lbl lit) -		   else return (mkIntCLit 0) -	; let std_info = mkStdInfoTable ty_descr_lit cl_descr_lit  -					cl_type srt_len layout_lit - -	; blks <- cgStmtsToBlocks body - -        ; conName <-   -             if is_con -                then do cstr <- mkByteStringCLit $ fromJust conIdentity -                        return (makeRelativeRefTo info_lbl cstr) -                else return (mkIntCLit 0) - -	; panic "emitClosureCodeAndInfoTable" } --emitInfoTableAndCode info_lbl std_info (extra_bits conName) args blks } + = do	{ blks <- cgStmtsToBlocks body +        ; info <- mkCmmInfo cl_info +        ; emitInfoTableAndCode info_lbl info args blks }    where      info_lbl  = infoTableLabelFromCI cl_info -    cl_descr_string = closureValDescr cl_info -    cl_type = smRepClosureTypeInt (closureSMRep cl_info) - -    srt = closureSRT cl_info	      -    needs_srt = needsSRT srt - -    mb_con = isConstrClosure_maybe  cl_info -    is_con = isJust mb_con - -    (srt_label,srt_len,conIdentity) -	= case mb_con of -	    Just con -> -- Constructors don't have an SRT -			-- We keep the *zero-indexed* tag in the srt_len -			-- field of the info table.  -			(mkIntCLit 0, fromIntegral (dataConTagZ con),  -                         Just $ dataConIdentity con)  - -	    Nothing  -> -- Not a constructor -                        let (label, len) = srtLabelAndLength srt info_lbl -                        in (label, len, Nothing) - -    ptrs       = closurePtrsSize cl_info -    nptrs      = size - ptrs -    size       = closureNonHdrSize cl_info -    layout_lit = packHalfWordsCLit ptrs nptrs - -    extra_bits conName  -	| is_fun    = fun_extra_bits -	| is_con    = [conName] -	| needs_srt = [srt_label] - 	| otherwise = [] - -    maybe_fun_stuff = closureFunInfo cl_info -    is_fun = isJust maybe_fun_stuff -    (Just (arity, arg_descr)) = maybe_fun_stuff - -    fun_extra_bits -	| ArgGen liveness <- arg_descr -	= [ fun_amode, -	    srt_label, -	    makeRelativeRefTo info_lbl $ mkLivenessCLit liveness,  -	    slow_entry ] -	| needs_srt = [fun_amode, srt_label] -	| otherwise = [fun_amode] - -    slow_entry = makeRelativeRefTo info_lbl (CmmLabel slow_entry_label) -    slow_entry_label = mkSlowEntryLabel (closureName cl_info) - -    fun_amode = packHalfWordsCLit fun_type arity -    fun_type  = argDescrType arg_descr -  -- We keep the *zero-indexed* tag in the srt_len field of the info  -- table of a data constructor.  dataConTagZ :: DataCon -> ConTagZ  dataConTagZ con = dataConTag con - fIRST_TAG --- A low-level way to generate the variable part of a fun-style info table. --- (must match fun_extra_bits above).  Used by the C-- parser. -mkFunGenInfoExtraBits :: Int -> Int -> CmmLit -> CmmLit -> CmmLit -> [CmmLit] -mkFunGenInfoExtraBits fun_type arity srt_label liveness slow_entry -  = [ packHalfWordsCLit fun_type arity, -      srt_label, -      liveness, -      slow_entry ] +-- Convert from 'ClosureInfo' to 'CmmInfo'. +-- Not used for return points.  (The 'smRepClosureTypeInt' call would panic.) +mkCmmInfo :: ClosureInfo -> FCode CmmInfo +mkCmmInfo cl_info = do +  prof <-  +      if opt_SccProfilingOn  +      then do ty_descr_lit <- mkStringCLit (closureTypeDescr cl_info) +              cl_descr_lit <- mkStringCLit (closureValDescr cl_info) +              return $ ProfilingInfo +                         (makeRelativeRefTo info_lbl ty_descr_lit) +                         (makeRelativeRefTo info_lbl cl_descr_lit) +      else return $ ProfilingInfo (mkIntCLit 0) (mkIntCLit 0) + +  case cl_info of +    ConInfo { closureCon = con } -> do +       cstr <- mkByteStringCLit $ dataConIdentity con +       let conName = makeRelativeRefTo info_lbl cstr +           info = ConstrInfo (ptrs, nptrs) +                             (fromIntegral (dataConTagZ con)) +                             conName +       return $ CmmInfo prof gc_target cl_type info + +    ClosureInfo { closureName   = name, +                  closureLFInfo = lf_info, +                  closureSRT    = srt } -> +       return $ CmmInfo prof gc_target cl_type info +       where +         info = +             case lf_info of +               LFReEntrant _ arity _ arg_descr -> +                   FunInfo (ptrs, nptrs) +                           srt  +                           (argDescrType arg_descr) +                           (fromIntegral arity) +                           arg_descr  +                           (CmmLabel (mkSlowEntryLabel name)) +               LFThunk _ _ _ (SelectorThunk offset) _ -> +                   ThunkSelectorInfo (fromIntegral offset) srt +               LFThunk _ _ _ _ _ -> +                   ThunkInfo (ptrs, nptrs) srt +               _ -> panic "unexpected lambda form in mkCmmInfo" +  where +    info_lbl = infoTableLabelFromCI cl_info + +    cl_type  = smRepClosureTypeInt (closureSMRep cl_info) + +    ptrs     = fromIntegral $ closurePtrsSize cl_info +    size     = fromIntegral $ closureNonHdrSize cl_info +    nptrs    = size - ptrs + +    -- The gc_target is to inform the CPS pass when it inserts a stack check. +    -- Since that pass isn't used yet we'll punt for now. +    -- When the CPS pass is fully integrated, this should +    -- be replaced by the label that any heap check jumped to, +    -- so that branch can be shared by both the heap (from codeGen) +    -- and stack checks (from the CPS pass). +    gc_target = panic "TODO: gc_target"  -------------------------------------------------------------------------  -- @@ -168,63 +133,134 @@ mkFunGenInfoExtraBits fun_type arity srt_label liveness slow_entry  --  ------------------------------------------------------------------------- ---	Here's the layout of a return-point info table --- --- Tables next to code: --- ---			<srt slot> ---			<standard info table> ---  	ret-addr -->	<entry code (if any)> --- --- Not tables-next-to-code: --- ---	ret-addr -->	<ptr to entry code> ---			<standard info table> ---			<srt slot> --- ---  * The SRT slot is only there is SRT info to record +-- The concrete representation as a list of 'CmmAddr' is handled later +-- in the pipeline by 'cmmToRawCmm'.  emitReturnTarget     :: Name     -> CgStmts			-- The direct-return code (if any)     -> FCode CLabel  emitReturnTarget name stmts -  = do	{ live_slots <- getLiveStackSlots -	; liveness   <- buildContLiveness name live_slots -	; srt_info   <- getSRTInfo - -	; let -	      cl_type | isBigLiveness liveness = rET_BIG -                      | otherwise              = rET_SMALL -  -	      (std_info, extra_bits) =  -		   mkRetInfoTable info_lbl liveness srt_info cl_type - +  = do	{ srt_info   <- getSRTInfo  	; blks <- cgStmtsToBlocks stmts -	; panic "emitReturnTarget" --emitInfoTableAndCode info_lbl std_info extra_bits args blks +        ; frame <- mkStackLayout +        ; let info = CmmInfo +                       (ProfilingInfo zeroCLit zeroCLit) +                       gc_target +                       rET_SMALL -- cmmToRawCmm may convert it to rET_BIG +                       (ContInfo frame srt_info) +        ; emitInfoTableAndCode info_lbl info args blks  	; return info_lbl }    where      args      = {- trace "emitReturnTarget: missing args" -} []      uniq      = getUnique name      info_lbl  = mkReturnInfoLabel uniq +    -- The gc_target is to inform the CPS pass when it inserts a stack check. +    -- Since that pass isn't used yet we'll punt for now. +    -- When the CPS pass is fully integrated, this should +    -- be replaced by the label that any heap check jumped to, +    -- so that branch can be shared by both the heap (from codeGen) +    -- and stack checks (from the CPS pass). +    gc_target = panic "TODO: gc_target" + -mkRetInfoTable -  :: CLabel             -- info label -  -> Liveness		-- liveness -  -> C_SRT		-- SRT Info -  -> StgHalfWord	-- type (eg. rET_SMALL) -  -> ([CmmLit],[CmmLit]) -mkRetInfoTable info_lbl liveness srt_info cl_type -  =  (std_info, srt_slot) +-- Build stack layout information from the state of the 'FCode' monad. +-- Should go away once 'codeGen' starts using the CPS conversion +-- pass to handle the stack.  Until then, this is really just +-- here to convert from the 'codeGen' representation of the stack +-- to the 'CmmInfo' representation of the stack. +-- +-- See 'CmmInfo.mkLiveness' for where this is converted to a bitmap. + +{- +This seems to be a very error prone part of the code. +It is surprisingly prone to off-by-one errors, because +it converts between offset form (codeGen) and list form (CmmInfo). +Thus a bit of explanation is in order. +Fortunately, this code should go away once the code generator +starts using the CPS conversion pass to handle the stack. + +The stack looks like this: + +             |             | +             |-------------| +frame_sp --> | return addr | +             |-------------| +             | dead slot   | +             |-------------| +             | live ptr b  | +             |-------------| +             | live ptr a  | +             |-------------| +real_sp  --> | return addr | +             +-------------+ + +Both 'frame_sp' and 'real_sp' are measured downwards +(i.e. larger frame_sp means smaller memory address). + +For that frame we want a result like: [Just a, Just b, Nothing] +Note that the 'head' of the list is the top +of the stack, and that the return address +is not present in the list (it is always assumed). +-} +mkStackLayout :: FCode [Maybe LocalReg] +mkStackLayout = do +  StackUsage { realSp = real_sp, +               frameSp = frame_sp } <- getStkUsage +  binds <- getLiveStackBindings +  let frame_size = real_sp - frame_sp - retAddrSizeW +      rel_binds = reverse $ sortWith fst +                    [(offset - frame_sp - retAddrSizeW, b) +                    | (offset, b) <- binds] + +  WARN( not (all (\bind -> fst bind >= 0) rel_binds), +	ppr binds $$ ppr rel_binds $$ +        ppr frame_size $$ ppr real_sp $$ ppr frame_sp ) +    return $ stack_layout rel_binds frame_size + +stack_layout :: [(VirtualSpOffset, CgIdInfo)] +             -> WordOff +             -> [Maybe LocalReg] +stack_layout [] sizeW = replicate sizeW Nothing +stack_layout ((off, bind):binds) sizeW | off == sizeW - 1 = +  (Just stack_bind) : (stack_layout binds (sizeW - rep_size)) +  where +    rep_size = cgRepSizeW (cgIdInfoArgRep bind) +    stack_bind = LocalReg unique machRep kind +    unique = getUnique (cgIdInfoId bind) +    machRep = argMachRep (cgIdInfoArgRep bind) +    kind = if isFollowableArg (cgIdInfoArgRep bind) +           then KindPtr +           else KindNonPtr +stack_layout binds@((off, _):_) sizeW | otherwise = +  Nothing : (stack_layout binds (sizeW - 1)) + +{- Another way to write the function that might be less error prone (untested) +stack_layout offsets sizeW = result    where -	(srt_label, srt_len) = srtLabelAndLength srt_info info_lbl -  -	srt_slot | needsSRT srt_info = [srt_label] -	         | otherwise         = [] -  -	liveness_lit = makeRelativeRefTo info_lbl $ mkLivenessCLit liveness -	std_info = mkStdInfoTable zeroCLit zeroCLit cl_type srt_len liveness_lit +    y = map (flip lookup offsets) [0..] +      -- offsets -> nothing and just (each slot is one word) +    x = take sizeW y -- set the frame size +    z = clip x -- account for multi-word slots +    result = map mk_reg z + +    clip [] = [] +    clip list@(x : _) = x : clip (drop count list) +      ASSERT(all isNothing (tail (take count list))) +     +    count Nothing = 1 +    count (Just x) = cgRepSizeW (cgIdInfoArgRep x) + +    mk_reg Nothing = Nothing +    mk_reg (Just x) = LocalReg unique machRep kind +      where +        unique = getUnique (cgIdInfoId x) +        machRep = argMachrep (cgIdInfoArgRep bind) +        kind = if isFollowableArg (cgIdInfoArgRep bind) +           then KindPtr +           else KindNonPtr +-}  emitAlgReturnTarget  	:: Name				-- Just for its unique @@ -250,39 +286,11 @@ emitReturnInstr    = do 	{ info_amode <- getSequelAmode  	; stmtC (CmmJump (entryCode info_amode) []) } -------------------------------------------------------------------------- --- ---	Generating a standard info table +-----------------------------------------------------------------------------  -- -------------------------------------------------------------------------- - --- The standard bits of an info table.  This part of the info table --- corresponds to the StgInfoTable type defined in InfoTables.h. +--	Info table offsets  -- --- Its shape varies with ticky/profiling/tables next to code etc --- so we can't use constant offsets from Constants - -mkStdInfoTable -   :: CmmLit		-- closure type descr (profiling) -   -> CmmLit		-- closure descr (profiling) -   -> StgHalfWord	-- closure type -   -> StgHalfWord	-- SRT length -   -> CmmLit		-- layout field -   -> [CmmLit] - -mkStdInfoTable type_descr closure_descr cl_type srt_len layout_lit - = 	-- Parallel revertible-black hole field -    prof_info -	-- Ticky info (none at present) -	-- Debug info (none at present) - ++ [layout_lit, type_lit] - - where   -    prof_info  -	| opt_SccProfilingOn = [type_descr, closure_descr] -	| otherwise	     = [] - -    type_lit = packHalfWordsCLit cl_type srt_len +-----------------------------------------------------------------------------  stdInfoTableSizeW :: WordOff  -- The size of a standard info table varies with profiling/ticky etc, @@ -402,35 +410,6 @@ emitInfoTableAndCode info_lbl info args blocks    where  	entry_lbl = infoLblToEntryLbl info_lbl -{- -emitInfoTableAndCode  -	:: CLabel 		-- Label of info table -	-> [CmmLit]		-- ...its invariant part -	-> [CmmLit] 		-- ...and its variant part -	-> CmmFormals		-- ...args -	-> [CmmBasicBlock]	-- ...and body -	-> Code - -emitInfoTableAndCode info_lbl std_info extra_bits args blocks -  | tablesNextToCode 	-- Reverse the extra_bits; and emit the top-level proc -  = emitProc (reverse extra_bits ++ std_info)  -	     entry_lbl args blocks -	-- NB: the info_lbl is discarded - -  | null blocks -- No actual code; only the info table is significant -  =		-- Use a zero place-holder in place of the  -		-- entry-label in the info table -    emitRODataLits info_lbl (zeroCLit : std_info ++ extra_bits) - -  | otherwise	-- Separately emit info table (with the function entry  -  =		-- point as first entry) and the entry code  -    do	{ emitDataLits info_lbl (CmmLabel entry_lbl : std_info ++ extra_bits) -	; emitProc [] entry_lbl args blocks } - -  where -	entry_lbl = infoLblToEntryLbl info_lbl --} -  -------------------------------------------------------------------------  --  --	Static reference tables diff --git a/compiler/codeGen/CgProf.hs b/compiler/codeGen/CgProf.hs index 3ba9d059fe..27ee54c50d 100644 --- a/compiler/codeGen/CgProf.hs +++ b/compiler/codeGen/CgProf.hs @@ -257,7 +257,7 @@ enterCostCentreThunk closure =    ifProfiling $ do       stmtC $ CmmStore curCCSAddr (costCentreFrom closure) -enter_ccs_fun stack = emitRtsCall SLIT("EnterFunCCS") [(stack,PtrHint)] +enter_ccs_fun stack = emitRtsCall SLIT("EnterFunCCS") [(stack,PtrHint)] False  			-- ToDo: vols  enter_ccs_fsub = enteringPAP 0 @@ -407,6 +407,7 @@ pushCostCentre result ccs cc    = emitRtsCallWithResult result PtrHint  	SLIT("PushCostCentre") [(ccs,PtrHint),   				(CmmLit (mkCCostCentre cc), PtrHint)] +        False  bumpSccCount :: CmmExpr -> CmmStmt  bumpSccCount ccs diff --git a/compiler/codeGen/CgUtils.hs b/compiler/codeGen/CgUtils.hs index 13de2136f5..c48b584fda 100644 --- a/compiler/codeGen/CgUtils.hs +++ b/compiler/codeGen/CgUtils.hs @@ -269,18 +269,18 @@ emitIfThenElse cond then_part else_part         ; labelC join_id         } -emitRtsCall :: LitString -> [(CmmExpr,MachHint)] -> Code -emitRtsCall fun args = emitRtsCall' [] fun args Nothing +emitRtsCall :: LitString -> [(CmmExpr,MachHint)] -> Bool -> Code +emitRtsCall fun args safe = emitRtsCall' [] fun args Nothing safe     -- The 'Nothing' says "save all global registers" -emitRtsCallWithVols :: LitString -> [(CmmExpr,MachHint)] -> [GlobalReg] -> Code -emitRtsCallWithVols fun args vols -   = emitRtsCall' [] fun args (Just vols) +emitRtsCallWithVols :: LitString -> [(CmmExpr,MachHint)] -> [GlobalReg] -> Bool -> Code +emitRtsCallWithVols fun args vols safe +   = emitRtsCall' [] fun args (Just vols) safe  emitRtsCallWithResult :: LocalReg -> MachHint -> LitString -	-> [(CmmExpr,MachHint)] -> Code -emitRtsCallWithResult res hint fun args -   = emitRtsCall' [(res,hint)] fun args Nothing +	-> [(CmmExpr,MachHint)] -> Bool -> Code +emitRtsCallWithResult res hint fun args safe +   = emitRtsCall' [(res,hint)] fun args Nothing safe  -- Make a call to an RTS C procedure  emitRtsCall' @@ -288,12 +288,15 @@ emitRtsCall'     -> LitString     -> [(CmmExpr,MachHint)]     -> Maybe [GlobalReg] +   -> Bool -- True <=> CmmSafe call     -> Code -emitRtsCall' res fun args vols = do -    srt <- getSRTInfo -    stmtsC caller_save -    stmtC (CmmCall target res args srt) -    stmtsC caller_load +emitRtsCall' res fun args vols safe = do +  safety <- if safe +            then getSRTInfo >>= (return . CmmSafe) +            else return CmmUnsafe +  stmtsC caller_save +  stmtC (CmmCall target res args safety) +  stmtsC caller_load    where      (caller_save, caller_load) = callerSaveVolatileRegs vols      target   = CmmForeignCall fun_expr CCallConv diff --git a/compiler/codeGen/ClosureInfo.lhs b/compiler/codeGen/ClosureInfo.lhs index ad26b2ec7c..db4636866d 100644 --- a/compiler/codeGen/ClosureInfo.lhs +++ b/compiler/codeGen/ClosureInfo.lhs @@ -13,8 +13,9 @@ the STG paper.  \begin{code}  module ClosureInfo ( -	ClosureInfo, LambdaFormInfo, SMRep, 	-- all abstract -	StandardFormInfo,  +	ClosureInfo(..), LambdaFormInfo(..),	-- would be abstract but +	StandardFormInfo(..),			-- mkCmmInfo looks inside +        SMRep,  	ArgDescr(..), Liveness(..),   	C_SRT(..), needsSRT, @@ -188,7 +189,7 @@ data LambdaFormInfo  data ArgDescr    = ArgSpec		-- Fits one of the standard patterns -	!Int		-- RTS type identifier ARG_P, ARG_N, ... +	!StgHalfWord	-- RTS type identifier ARG_P, ARG_N, ...    | ArgGen	 	-- General case  	Liveness	-- Details about the arguments @@ -957,5 +958,3 @@ getTyDescription ty  getPredTyDescription (ClassP cl tys) = getOccString cl  getPredTyDescription (IParam ip ty)  = getOccString (ipNameName ip)  \end{code} - - diff --git a/compiler/main/HscMain.lhs b/compiler/main/HscMain.lhs index 0ae942cafa..f0fd95da23 100644 --- a/compiler/main/HscMain.lhs +++ b/compiler/main/HscMain.lhs @@ -76,6 +76,7 @@ import SimplStg		( stg2stg )  import CodeGen		( codeGen )  import CmmParse		( parseCmmFile )  import CmmCPS +import CmmInfo  import CodeOutput	( codeOutput )  import NameEnv          ( emptyNameEnv ) @@ -605,7 +606,8 @@ hscCompile cgguts                                foreign_stubs dir_imps cost_centre_info                                stg_binds hpc_info           ------------------  Convert to CPS -------------------- -         continuationC <- {-return abstractC-} cmmCPS dflags abstractC +         --continuationC <- cmmCPS dflags abstractC +         continuationC <- cmmToRawCmm abstractC           ------------------  Code output -----------------------           (stub_h_exists,stub_c_exists)               <- codeOutput dflags this_mod location foreign_stubs  @@ -721,7 +723,8 @@ hscCmmFile dflags filename = do    case maybe_cmm of      Nothing -> return False      Just cmm -> do -        continuationC <- {-return [cmm]-} cmmCPS dflags [cmm] +        --continuationC <- cmmCPS dflags [cmm] +        continuationC <- cmmToRawCmm [cmm]  	codeOutput dflags no_mod no_loc NoStubs [] continuationC  	return True    where diff --git a/compiler/nativeGen/AsmCodeGen.lhs b/compiler/nativeGen/AsmCodeGen.lhs index f954d524c9..a04c5c7527 100644 --- a/compiler/nativeGen/AsmCodeGen.lhs +++ b/compiler/nativeGen/AsmCodeGen.lhs @@ -429,9 +429,6 @@ fixAssigns stmts =    returnUs (concat stmtss)  fixAssign :: CmmStmt -> UniqSM [CmmStmt] -fixAssign (CmmAssign (CmmGlobal BaseReg) src) -   = panic "cmmStmtConFold: assignment to BaseReg"; -  fixAssign (CmmAssign (CmmGlobal reg) src)    | Left  realreg <- reg_or_addr    = returnUs [CmmAssign (CmmGlobal reg) src] @@ -444,24 +441,6 @@ fixAssign (CmmAssign (CmmGlobal reg) src)    where  	reg_or_addr = get_GlobalReg_reg_or_addr reg -{- -fixAssign (CmmCall target results args) -  = mapAndUnzipUs fixResult results `thenUs` \ (results',stores) -> -    returnUs (CmmCall target results' args : -	      concat stores) -  where -	fixResult g@(CmmGlobal reg,hint) =  -	  case get_GlobalReg_reg_or_addr reg of -		Left realreg -> returnUs (g, []) -		Right baseRegAddr -> -		    getUniqueUs `thenUs` \ uq -> -		    let local = CmmLocal (LocalReg uq (globalRegRep reg)) in -		    returnUs ((local,hint),  -			      [CmmStore baseRegAddr (CmmReg local)]) -	fixResult other = -	  returnUs (other,[]) --} -  fixAssign other_stmt = returnUs [other_stmt]  -- ----------------------------------------------------------------------------- diff --git a/compiler/nativeGen/MachCodeGen.hs b/compiler/nativeGen/MachCodeGen.hs index 154eed866e..1d1cfa1596 100644 --- a/compiler/nativeGen/MachCodeGen.hs +++ b/compiler/nativeGen/MachCodeGen.hs @@ -3182,13 +3182,13 @@ outOfLineFloatOp mop res args        if localRegRep res == F64          then -          stmtToInstrs (CmmCall target [(res,FloatHint)] args NoC_SRT) +          stmtToInstrs (CmmCall target [(res,FloatHint)] args CmmUnsafe)          else do            uq <- getUniqueNat            let               tmp = LocalReg uq F64 KindNonPtr            -- in -          code1 <- stmtToInstrs (CmmCall target [(tmp,FloatHint)] args NoC_SRT) +          code1 <- stmtToInstrs (CmmCall target [(tmp,FloatHint)] args CmmUnsafe)            code2 <- stmtToInstrs (CmmAssign (CmmLocal res) (CmmReg (CmmLocal tmp)))            return (code1 `appOL` code2)    where | 
