diff options
Diffstat (limited to 'compiler')
| -rw-r--r-- | compiler/cmm/CLabel.hs | 17 | ||||
| -rw-r--r-- | compiler/nativeGen/X86/Ppr.hs | 69 | 
2 files changed, 80 insertions, 6 deletions
| diff --git a/compiler/cmm/CLabel.hs b/compiler/cmm/CLabel.hs index 1ba0d89c37..12c3357e47 100644 --- a/compiler/cmm/CLabel.hs +++ b/compiler/cmm/CLabel.hs @@ -104,7 +104,9 @@ module CLabel (          -- * Conversions          toClosureLbl, toSlowEntryLbl, toEntryLbl, toInfoLbl, hasHaskellName, -        pprCLabel +        pprCLabel, +        isInfoTableLabel, +        isConInfoTableLabel      ) where  #include "HsVersions.h" @@ -621,6 +623,19 @@ isSomeRODataLabel (IdLabel _ _ BlockInfoTable) = True  isSomeRODataLabel (CmmLabel _ _ CmmInfo) = True  isSomeRODataLabel _lbl = False +-- | Whether label is points to some kind of info table +isInfoTableLabel :: CLabel -> Bool +isInfoTableLabel (IdLabel _ _ InfoTable)      = True +isInfoTableLabel (IdLabel _ _ LocalInfoTable) = True +isInfoTableLabel (IdLabel _ _ ConInfoTable)   = True +isInfoTableLabel (IdLabel _ _ BlockInfoTable) = True +isInfoTableLabel _                            = False + +-- | Whether label is points to constructor info table +isConInfoTableLabel :: CLabel -> Bool +isConInfoTableLabel (IdLabel _ _ ConInfoTable)   = True +isConInfoTableLabel _                            = False +  -- | Get the label size field from a ForeignLabel  foreignLabelStdcallInfo :: CLabel -> Maybe Int  foreignLabelStdcallInfo (ForeignLabel _ info _ _) = info diff --git a/compiler/nativeGen/X86/Ppr.hs b/compiler/nativeGen/X86/Ppr.hs index c5fbeb544e..03d4fce794 100644 --- a/compiler/nativeGen/X86/Ppr.hs +++ b/compiler/nativeGen/X86/Ppr.hs @@ -170,16 +170,76 @@ pprGloblDecl lbl    | not (externallyVisibleCLabel lbl) = empty    | otherwise = text ".globl " <> ppr lbl -pprTypeAndSizeDecl :: CLabel -> SDoc -pprTypeAndSizeDecl lbl +pprLabelType' :: DynFlags -> CLabel -> SDoc +pprLabelType' dflags lbl = +  if isCFunctionLabel lbl || functionOkInfoTable then +    text "@function" +  else +    text "@object" +  where +    {- +    NOTE: This is a bit hacky. + +    With the `tablesNextToCode` info tables look like this: +    ``` +      <info table data> +    label_info: +      <info table code> +    ``` +    So actually info table label points exactly to the code and we can mark +    the label as @function. (This is required to make perf and potentially other +    tools to work on Haskell binaries). +    This usually works well but it can cause issues with a linker. +    A linker uses different algorithms for the relocation depending on +    the symbol type.For some reason, a linker will generate JUMP_SLOT relocation +    when constructor info table is referenced from a data section. +    This only happens with static constructor call so +    we mark _con_info symbols as `@object` to avoid the issue with relocations. + +    @SimonMarlow hack explanation: +    "The reasoning goes like this: + +    * The danger when we mark a symbol as `@function` is that the linker will +      redirect it to point to the PLT and use a `JUMP_SLOT` relocation when +      the symbol refers to something outside the current shared object. +      A PLT / JUMP_SLOT reference only works for symbols that we jump to, not +      for symbols representing data,, nor for info table symbol references which +      we expect to point directly to the info table. +    * GHC generates code that might refer to any info table symbol from the text +      segment, but that's OK, because those will be explicit GOT references +      generated by the code generator. +    * When we refer to info tables from the data segment, it's either +      * a FUN_STATIC/THUNK_STATIC local to this module +      * a `con_info` that could be from anywhere + +    So, the only info table symbols that we might refer to from the data segment +    of another shared object are `con_info` symbols, so those are the ones we +    need to exclude from getting the @function treatment. +    " + +    A good place to check for more +    https://ghc.haskell.org/trac/ghc/wiki/Commentary/PositionIndependentCode + +    Another possible hack is to create an extra local function symbol for +    every code-like thing to give the needed information for to the tools +    but mess up with the relocation. https://phabricator.haskell.org/D4730 +    -} +    functionOkInfoTable = tablesNextToCode dflags && +      isInfoTableLabel lbl && not (isConInfoTableLabel lbl) + + +pprTypeDecl :: CLabel -> SDoc +pprTypeDecl lbl      = sdocWithPlatform $ \platform ->        if osElfTarget (platformOS platform) && externallyVisibleCLabel lbl -      then text ".type " <> ppr lbl <> ptext (sLit ", @object") +      then +        sdocWithDynFlags $ \df -> +          text ".type " <> ppr lbl <> ptext (sLit  ", ") <> pprLabelType' df lbl        else empty  pprLabel :: CLabel -> SDoc  pprLabel lbl = pprGloblDecl lbl -            $$ pprTypeAndSizeDecl lbl +            $$ pprTypeDecl lbl              $$ (ppr lbl <> char ':')  {- @@ -1346,4 +1406,3 @@ pprFormatOpOpCoerce name format1 format2 op1 op2  pprCondInstr :: LitString -> Cond -> SDoc -> SDoc  pprCondInstr name cond arg    = hcat [ char '\t', ptext name, pprCond cond, space, arg] - | 
