summaryrefslogtreecommitdiff
path: root/compiler
diff options
context:
space:
mode:
authorBen Gamari <ben@smart-cactus.org>2022-03-07 15:15:12 -0500
committerMarge Bot <ben+marge-bot@smart-cactus.org>2022-03-18 05:11:35 -0400
commitd147428a5041adcdda2ffe02e3012c7987be3c23 (patch)
tree32f711d9024393a1fd72a6e71d3102ffc827367c /compiler
parentac3b2e7deb0e2987ae3f27d62c03716de46ebc79 (diff)
downloadhaskell-d147428a5041adcdda2ffe02e3012c7987be3c23.tar.gz
codeGen: Fix signedness of jump table indexing
Previously while constructing the jump table index we would zero-extend the discriminant before subtracting the start of the jump-table. This goes subtly wrong in the case of a sub-word, signed discriminant, as described in the included Note. Fix this in both the PPC and X86 NCGs. Fixes #21186.
Diffstat (limited to 'compiler')
-rw-r--r--compiler/GHC/CmmToAsm/PPC/CodeGen.hs12
-rw-r--r--compiler/GHC/CmmToAsm/X86/CodeGen.hs46
2 files changed, 47 insertions, 11 deletions
diff --git a/compiler/GHC/CmmToAsm/PPC/CodeGen.hs b/compiler/GHC/CmmToAsm/PPC/CodeGen.hs
index 6d7f61d56d..844e4744da 100644
--- a/compiler/GHC/CmmToAsm/PPC/CodeGen.hs
+++ b/compiler/GHC/CmmToAsm/PPC/CodeGen.hs
@@ -2143,12 +2143,14 @@ genSwitch config expr targets
]
return code
where
- indexExpr = cmmOffset platform exprWidened offset
+ -- See Note [Sub-word subtlety during jump-table indexing] in
+ -- GHC.CmmToAsm.X86.CodeGen for why we must first offset, then widen.
+ indexExpr0 = cmmOffset platform expr offset
-- We widen to a native-width register to santize the high bits
- exprWidened = CmmMachOp
- (MO_UU_Conv (cmmExprWidth platform expr)
- (platformWordWidth platform))
- [expr]
+ indexExpr = CmmMachOp
+ (MO_UU_Conv expr_w (platformWordWidth platform))
+ [indexExpr0]
+ expr_w = cmmExprWidth platform expr
(offset, ids) = switchTargetsToTable targets
platform = ncgPlatform config
diff --git a/compiler/GHC/CmmToAsm/X86/CodeGen.hs b/compiler/GHC/CmmToAsm/X86/CodeGen.hs
index 0d67f306dc..1fefe3a346 100644
--- a/compiler/GHC/CmmToAsm/X86/CodeGen.hs
+++ b/compiler/GHC/CmmToAsm/X86/CodeGen.hs
@@ -2773,18 +2773,52 @@ maybePromoteCArg platform wto arg
-- -----------------------------------------------------------------------------
-- Generating a table-branch
+{-
+Note [Sub-word subtlety during jump-table indexing]
+~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
+Offset the index by the start index of the jump table.
+It's important that we do this *before* the widening below. To see
+why, consider a switch with a sub-word, signed discriminant such as:
+
+ switch [-5...+2] x::I16 {
+ case -5: ...
+ ...
+ case +2: ...
+ }
+
+Consider what happens if we offset *after* widening in the case that
+x=-4:
+
+ // x == -4 == 0xfffc::I16
+ indexWidened = UU_Conv(x); // == 0xfffc::I64
+ indexExpr = indexWidened - (-5); // == 0x10000::I64
+
+This index is clearly nonsense given that the jump table only has
+eight entries.
+
+By contrast, if we widen *after* we offset then we get the correct
+index (1),
+
+ // x == -4 == 0xfffc::I16
+ indexOffset = x - (-5); // == 1::I16
+ indexExpr = UU_Conv(indexOffset); // == 1::I64
+
+See #21186.
+-}
+
genSwitch :: CmmExpr -> SwitchTargets -> NatM InstrBlock
genSwitch expr targets = do
config <- getConfig
let platform = ncgPlatform config
- -- We widen to a native-width register because we cannot use arbitry sizes
+ expr_w = cmmExprWidth platform expr
+ indexExpr0 = cmmOffset platform expr offset
+ -- We widen to a native-width register because we cannot use arbitrary sizes
-- in x86 addressing modes.
- exprWidened = CmmMachOp
- (MO_UU_Conv (cmmExprWidth platform expr)
- (platformWordWidth platform))
- [expr]
- indexExpr = cmmOffset platform exprWidened offset
+ -- See Note [Sub-word subtlety during jump-table indexing].
+ indexExpr = CmmMachOp
+ (MO_UU_Conv expr_w (platformWordWidth platform))
+ [indexExpr0]
if ncgPIC config
then do
(reg,e_code) <- getNonClobberedReg indexExpr