summaryrefslogtreecommitdiff
path: root/compiler
diff options
context:
space:
mode:
authorBen Gamari <ben@smart-cactus.org>2022-01-27 15:06:59 -0500
committerMarge Bot <ben+marge-bot@smart-cactus.org>2022-02-04 16:14:47 -0500
commit606b59a55c543607742092897bb774f7cb88e65d (patch)
treec4ed573bd525910c9d822f757c9419e2f9f79dc0 /compiler
parent99ea5f2cfa09f50bf3ea105821dc095942552e59 (diff)
downloadhaskell-606b59a55c543607742092897bb774f7cb88e65d.tar.gz
Fix array primop alignment
Diffstat (limited to 'compiler')
-rw-r--r--compiler/GHC/StgToCmm/Monad.hs9
-rw-r--r--compiler/GHC/StgToCmm/Prim.hs47
2 files changed, 35 insertions, 21 deletions
diff --git a/compiler/GHC/StgToCmm/Monad.hs b/compiler/GHC/StgToCmm/Monad.hs
index 04654d40d3..6aa0a2dd08 100644
--- a/compiler/GHC/StgToCmm/Monad.hs
+++ b/compiler/GHC/StgToCmm/Monad.hs
@@ -21,7 +21,7 @@ module GHC.StgToCmm.Monad (
emit, emitDecl,
emitProcWithConvention, emitProcWithStackFrame,
- emitOutOfLine, emitAssign, emitStore,
+ emitOutOfLine, emitAssign, emitStore, emitStore',
emitComment, emitTick, emitUnwind,
newTemp,
@@ -725,8 +725,11 @@ emitAssign :: CmmReg -> CmmExpr -> FCode ()
emitAssign l r = emitCgStmt (CgStmt (CmmAssign l r))
-- | Assumes natural alignment.
-emitStore :: CmmExpr -> CmmExpr -> FCode ()
-emitStore l r = emitCgStmt (CgStmt (CmmStore l r NaturallyAligned))
+emitStore :: CmmExpr -> CmmExpr -> FCode ()
+emitStore = emitStore' NaturallyAligned
+
+emitStore' :: AlignmentSpec -> CmmExpr -> CmmExpr -> FCode ()
+emitStore' alignment l r = emitCgStmt (CgStmt (CmmStore l r alignment))
emit :: CmmAGraph -> FCode ()
emit ag
diff --git a/compiler/GHC/StgToCmm/Prim.hs b/compiler/GHC/StgToCmm/Prim.hs
index 17fe2d9373..4d7294a3ec 100644
--- a/compiler/GHC/StgToCmm/Prim.hs
+++ b/compiler/GHC/StgToCmm/Prim.hs
@@ -2057,13 +2057,20 @@ genericFabsOp _ _ _ = panic "genericFabsOp"
------------------------------------------------------------------------------
-- Helpers for translating various minor variants of array indexing.
+alignmentFromTypes :: CmmType -- ^ element type
+ -> CmmType -- ^ index type
+ -> AlignmentSpec
+alignmentFromTypes ty idx_ty
+ | typeWidth ty < typeWidth idx_ty = NaturallyAligned
+ | otherwise = Unaligned
+
doIndexOffAddrOp :: Maybe MachOp
-> CmmType
-> [LocalReg]
-> [CmmExpr]
-> FCode ()
doIndexOffAddrOp maybe_post_read_cast rep [res] [addr,idx]
- = mkBasicIndexedRead 0 maybe_post_read_cast rep res addr rep idx
+ = mkBasicIndexedRead NaturallyAligned 0 maybe_post_read_cast rep res addr rep idx
doIndexOffAddrOp _ _ _ _
= panic "GHC.StgToCmm.Prim: doIndexOffAddrOp"
@@ -2074,7 +2081,8 @@ doIndexOffAddrOpAs :: Maybe MachOp
-> [CmmExpr]
-> FCode ()
doIndexOffAddrOpAs maybe_post_read_cast rep idx_rep [res] [addr,idx]
- = mkBasicIndexedRead 0 maybe_post_read_cast rep res addr idx_rep idx
+ = let alignment = alignmentFromTypes rep idx_rep
+ in mkBasicIndexedRead alignment 0 maybe_post_read_cast rep res addr idx_rep idx
doIndexOffAddrOpAs _ _ _ _ _
= panic "GHC.StgToCmm.Prim: doIndexOffAddrOpAs"
@@ -2086,7 +2094,7 @@ doIndexByteArrayOp :: Maybe MachOp
doIndexByteArrayOp maybe_post_read_cast rep [res] [addr,idx]
= do profile <- getProfile
doByteArrayBoundsCheck idx addr rep rep
- mkBasicIndexedRead (arrWordsHdrSize profile) maybe_post_read_cast rep res addr rep idx
+ mkBasicIndexedRead NaturallyAligned (arrWordsHdrSize profile) maybe_post_read_cast rep res addr rep idx
doIndexByteArrayOp _ _ _ _
= panic "GHC.StgToCmm.Prim: doIndexByteArrayOp"
@@ -2099,7 +2107,8 @@ doIndexByteArrayOpAs :: Maybe MachOp
doIndexByteArrayOpAs maybe_post_read_cast rep idx_rep [res] [addr,idx]
= do profile <- getProfile
doByteArrayBoundsCheck idx addr idx_rep rep
- mkBasicIndexedRead (arrWordsHdrSize profile) maybe_post_read_cast rep res addr idx_rep idx
+ let alignment = alignmentFromTypes rep idx_rep
+ mkBasicIndexedRead alignment (arrWordsHdrSize profile) maybe_post_read_cast rep res addr idx_rep idx
doIndexByteArrayOpAs _ _ _ _ _
= panic "GHC.StgToCmm.Prim: doIndexByteArrayOpAs"
@@ -2111,7 +2120,7 @@ doReadPtrArrayOp res addr idx
= do profile <- getProfile
platform <- getPlatform
doPtrArrayBoundsCheck idx addr
- mkBasicIndexedRead (arrPtrsHdrSize profile) Nothing (gcWord platform) res addr (gcWord platform) idx
+ mkBasicIndexedRead NaturallyAligned (arrPtrsHdrSize profile) Nothing (gcWord platform) res addr (gcWord platform) idx
doWriteOffAddrOp :: Maybe MachOp
-> CmmType
@@ -2150,7 +2159,7 @@ doWritePtrArrayOp addr idx val
-- Update remembered set for non-moving collector
whenUpdRemSetEnabled
- $ emitUpdRemSetPush (cmmLoadIndexOffExpr platform hdr_size ty addr ty idx)
+ $ emitUpdRemSetPush (cmmLoadIndexOffExpr platform NaturallyAligned hdr_size ty addr ty idx)
-- This write barrier is to ensure that the heap writes to the object
-- referred to by val have happened before we write val into the array.
-- See #12469 for details.
@@ -2164,8 +2173,7 @@ doWritePtrArrayOp addr idx val
cmmOffsetExpr platform
(cmmOffsetExprW platform (cmmOffsetB platform addr hdr_size)
(loadArrPtrsSize profile addr))
- (CmmMachOp (mo_wordUShr platform) [idx,
- mkIntExpr platform (pc_MUT_ARR_PTRS_CARD_BITS (platformConstants platform))])
+ (CmmMachOp (mo_wordUShr platform) [idx, mkIntExpr platform (pc_MUT_ARR_PTRS_CARD_BITS (platformConstants platform))])
) (CmmLit (CmmInt 1 W8))
loadArrPtrsSize :: Profile -> CmmExpr -> CmmExpr
@@ -2173,7 +2181,8 @@ loadArrPtrsSize profile addr = cmmLoadBWord platform (cmmOffsetB platform addr o
where off = fixedHdrSize profile + pc_OFFSET_StgMutArrPtrs_ptrs (profileConstants profile)
platform = profilePlatform profile
-mkBasicIndexedRead :: ByteOff -- Initial offset in bytes
+mkBasicIndexedRead :: AlignmentSpec
+ -> ByteOff -- Initial offset in bytes
-> Maybe MachOp -- Optional result cast
-> CmmType -- Type of element we are accessing
-> LocalReg -- Destination
@@ -2181,13 +2190,13 @@ mkBasicIndexedRead :: ByteOff -- Initial offset in bytes
-> CmmType -- Type of element by which we are indexing
-> CmmExpr -- Index
-> FCode ()
-mkBasicIndexedRead off Nothing ty res base idx_ty idx
+mkBasicIndexedRead alignment off Nothing ty res base idx_ty idx
= do platform <- getPlatform
- emitAssign (CmmLocal res) (cmmLoadIndexOffExpr platform off ty base idx_ty idx)
-mkBasicIndexedRead off (Just cast) ty res base idx_ty idx
+ emitAssign (CmmLocal res) (cmmLoadIndexOffExpr platform alignment off ty base idx_ty idx)
+mkBasicIndexedRead alignment off (Just cast) ty res base idx_ty idx
= do platform <- getPlatform
emitAssign (CmmLocal res) (CmmMachOp cast [
- cmmLoadIndexOffExpr platform off ty base idx_ty idx])
+ cmmLoadIndexOffExpr platform alignment off ty base idx_ty idx])
mkBasicIndexedWrite :: ByteOff -- Initial offset in bytes
-> Maybe MachOp -- Optional value cast
@@ -2198,7 +2207,8 @@ mkBasicIndexedWrite :: ByteOff -- Initial offset in bytes
-> FCode ()
mkBasicIndexedWrite off Nothing base idx_ty idx val
= do platform <- getPlatform
- emitStore (cmmIndexOffExpr platform off (typeWidth idx_ty) base idx) val
+ let alignment = alignmentFromTypes (cmmExprType platform val) idx_ty
+ emitStore' alignment (cmmIndexOffExpr platform off (typeWidth idx_ty) base idx) val
mkBasicIndexedWrite off (Just cast) base idx_ty idx val
= mkBasicIndexedWrite off Nothing base idx_ty idx (CmmMachOp cast [val])
@@ -2215,14 +2225,15 @@ cmmIndexOffExpr platform off width base idx
= cmmIndexExpr platform width (cmmOffsetB platform base off) idx
cmmLoadIndexOffExpr :: Platform
+ -> AlignmentSpec
-> ByteOff -- Initial offset in bytes
-> CmmType -- Type of element we are accessing
-> CmmExpr -- Base address
-> CmmType -- Type of element by which we are indexing
-> CmmExpr -- Index
-> CmmExpr
-cmmLoadIndexOffExpr platform off ty base idx_ty idx
- = CmmLoad (cmmIndexOffExpr platform off (typeWidth idx_ty) base idx) ty NaturallyAligned
+cmmLoadIndexOffExpr platform alignment off ty base idx_ty idx
+ = CmmLoad (cmmIndexOffExpr platform off (typeWidth idx_ty) base idx) ty alignment
setInfo :: CmmExpr -> CmmExpr -> CmmAGraph
setInfo closure_ptr info_ptr = mkStore closure_ptr info_ptr
@@ -2988,7 +2999,7 @@ doReadSmallPtrArrayOp res addr idx = do
profile <- getProfile
platform <- getPlatform
doSmallPtrArrayBoundsCheck idx addr
- mkBasicIndexedRead (smallArrPtrsHdrSize profile) Nothing (gcWord platform) res addr
+ mkBasicIndexedRead NaturallyAligned (smallArrPtrsHdrSize profile) Nothing (gcWord platform) res addr
(gcWord platform) idx
doWriteSmallPtrArrayOp :: CmmExpr
@@ -3004,7 +3015,7 @@ doWriteSmallPtrArrayOp addr idx val = do
-- Update remembered set for non-moving collector
tmp <- newTemp ty
- mkBasicIndexedRead (smallArrPtrsHdrSize profile) Nothing ty tmp addr ty idx
+ mkBasicIndexedRead NaturallyAligned (smallArrPtrsHdrSize profile) Nothing ty tmp addr ty idx
whenUpdRemSetEnabled $ emitUpdRemSetPush (CmmReg (CmmLocal tmp))
emitPrimCall [] MO_WriteBarrier [] -- #12469