diff options
| author | Michael D. Adams <t-madams@microsoft.com> | 2007-05-23 10:51:12 +0000 | 
|---|---|---|
| committer | Michael D. Adams <t-madams@microsoft.com> | 2007-05-23 10:51:12 +0000 | 
| commit | 53a82428d5e18a016dbc6b604d88577e7dc916e5 (patch) | |
| tree | 8cda5150cef4a14e8ef576885da6dad2cb054608 | |
| parent | 46b28f7bfdd535e9fe5217a1151bedfb2cc15472 (diff) | |
| download | haskell-53a82428d5e18a016dbc6b604d88577e7dc916e5.tar.gz | |
Refined the handling of stack frame headers
| -rw-r--r-- | compiler/cmm/CmmCPS.hs | 47 | ||||
| -rw-r--r-- | compiler/cmm/CmmLive.hs | 13 | 
2 files changed, 37 insertions, 23 deletions
| diff --git a/compiler/cmm/CmmCPS.hs b/compiler/cmm/CmmCPS.hs index ad494aadbb..4c1d025c8a 100644 --- a/compiler/cmm/CmmCPS.hs +++ b/compiler/cmm/CmmCPS.hs @@ -118,11 +118,12 @@ data FinalStmt    -- TODO: | ProcPointExit (needed?) +-- Describes the layout of a stack frame for a continuation  data StackFormat      = StackFormat -         BlockId {- block that is the start of the continuation. may or may not be the current block -} -         WordOff {- total frame size -} -         [(CmmReg, WordOff)] {- local reg offsets from stack top -} +         (Maybe CLabel)		-- The label occupying the top slot +         WordOff		-- Total frame size in words +         [(CmmReg, WordOff)]	-- local reg offsets from stack top  -- A block can be a continuation of a call  -- A block can be a continuation of another block (w/ or w/o joins) @@ -298,21 +299,23 @@ selectStackFormat2 live continuations =      map (\c -> (continuationLabel c, selectStackFormat' c)) continuations      where        selectStackFormat' (Continuation True info_table label formals blocks) = -          let ident = brokenBlockId $ head blocks -- TODO: CLabel isn't a uniquable, but we need a better way than this -          in StackFormat ident 0 [] +          --let ident = brokenBlockId $ head blocks -- TODO: CLabel isn't a uniquable, but we need a better way than this +          --in +          StackFormat (Just label) 0 []        selectStackFormat' (Continuation False info_table label formals blocks) = +          -- TODO: assumes the first block is the entry block            let ident = brokenBlockId $ head blocks -- TODO: CLabel isn't a uniquable, but we need a better way than this -          in live_to_format ident $ lookupWithDefaultUFM live unknown_block ident           +          in live_to_format label formals $ lookupWithDefaultUFM live unknown_block ident -      live_to_format :: BlockId -> CmmLive -> StackFormat -      live_to_format label live = +      live_to_format :: CLabel -> CmmFormals -> CmmLive -> StackFormat +      live_to_format label formals live =            foldl extend_format -                    (StackFormat label retAddrSizeW []) -                    (uniqSetToList live) +                    (StackFormat (Just label) retAddrSizeW []) +                    (uniqSetToList (live `minusUniqSet` mkUniqSet (cmmFormalsToLiveLocals formals)))        extend_format :: StackFormat -> LocalReg -> StackFormat -      extend_format (StackFormat block size offsets) reg = -          StackFormat block (slot_size reg + size) ((CmmLocal reg, size) : offsets) +      extend_format (StackFormat label size offsets) reg = +          StackFormat label (slot_size reg + size) ((CmmLocal reg, size) : offsets)        unknown_block = panic "unknown BlockId in selectStackFormat" @@ -361,9 +364,11 @@ constructContinuation2' curr_ident formats (BrokenBlock ident entry stmts _ exit  exit_function :: StackFormat -> CmmExpr -> CmmActuals -> [CmmStmt]  exit_function (StackFormat curr_id curr_frame_size curr_offsets) target arguments    = adjust_spReg ++ jump where -    adjust_spReg = [ -     CmmAssign spReg -     (CmmRegOff spReg (curr_frame_size*wORD_SIZE))] +    adjust_spReg = +        if curr_frame_size == 0 +        then [] +        else [CmmAssign spReg +                 (CmmRegOff spReg (curr_frame_size*wORD_SIZE))]      jump = [CmmJump target arguments]  enter_function :: WordOff -> [CmmStmt] @@ -388,9 +393,15 @@ pack_continuation (StackFormat curr_id curr_frame_size curr_offsets)            spReg (wORD_SIZE*(curr_frame_size - cont_frame_size + offset)))           (CmmReg reg)           | (reg, offset) <- cont_offsets] -    set_stack_header = -- TODO: only set when needed -        [CmmStore (CmmRegOff spReg (wORD_SIZE*(curr_frame_size - cont_frame_size))) continuation_function] -    continuation_function = CmmLit $ CmmLabel $ mkReturnPtLabel {-TODO: use the correct function -} $ getUnique cont_id +    needs_header = +      case (curr_id, cont_id) of +        (Just x, Just y) -> x /= y +        _ -> isJust cont_id +    set_stack_header = +      if not needs_header +         then [] +         else [CmmStore (CmmRegOff spReg (wORD_SIZE*(curr_frame_size - cont_frame_size))) continuation_function] +    continuation_function = CmmLit $ CmmLabel $ fromJust cont_id      adjust_spReg =          if curr_frame_size == cont_frame_size          then [] diff --git a/compiler/cmm/CmmLive.hs b/compiler/cmm/CmmLive.hs index 0a4eb67ae3..771d4760d4 100644 --- a/compiler/cmm/CmmLive.hs +++ b/compiler/cmm/CmmLive.hs @@ -1,6 +1,7 @@  module CmmLive (          CmmLive, BlockEntryLiveness, -        cmmLiveness +        cmmLiveness, +        cmmFormalsToLiveLocals    ) where  import Cmm @@ -156,6 +157,11 @@ addKilled new_killed live = live `minusUniqSet` new_killed  --------------------------------  -- Liveness of a CmmStmt  -------------------------------- +cmmFormalsToLiveLocals :: CmmFormals -> [LocalReg] +cmmFormalsToLiveLocals [] = [] +cmmFormalsToLiveLocals ((CmmGlobal _,_):args) = cmmFormalsToLiveLocals args +cmmFormalsToLiveLocals ((CmmLocal r,_):args) = r:cmmFormalsToLiveLocals args +  cmmStmtLive :: BlockEntryLiveness -> CmmStmt -> CmmLivenessTransformer  cmmStmtLive _ (CmmNop) = id  cmmStmtLive _ (CmmComment _) = id @@ -170,10 +176,7 @@ cmmStmtLive _ (CmmStore expr1 expr2) =  cmmStmtLive _ (CmmCall target results arguments _) =      target_liveness .      foldr ((.) . cmmExprLive) id (map fst arguments) . -    addKilled (mkUniqSet $ only_local_regs results) where -        only_local_regs [] = [] -        only_local_regs ((CmmGlobal _,_):args) = only_local_regs args -        only_local_regs ((CmmLocal r,_):args) = r:only_local_regs args +    addKilled (mkUniqSet $ cmmFormalsToLiveLocals results) where          target_liveness =              case target of                (CmmForeignCall target _) -> cmmExprLive target | 
