summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorSimon Marlow <marlowsd@gmail.com>2010-06-10 08:06:36 +0000
committerSimon Marlow <marlowsd@gmail.com>2010-06-10 08:06:36 +0000
commit565f3c753ccedeb19b5b8190090241a2185eb0cb (patch)
tree9396ad7223765464ccd7071209ef2e4e2faf39ab
parent5ac6359ebd27b7a84eb63a3ea779d9ce6659ce76 (diff)
downloadhaskell-565f3c753ccedeb19b5b8190090241a2185eb0cb.tar.gz
messageBlackHole: fix deadlock bug caused by a missing 'volatile'
-rw-r--r--includes/stg/SMP.h8
-rw-r--r--rts/Messages.c4
2 files changed, 11 insertions, 1 deletions
diff --git a/includes/stg/SMP.h b/includes/stg/SMP.h
index b302f4886b..ad8c0baef9 100644
--- a/includes/stg/SMP.h
+++ b/includes/stg/SMP.h
@@ -293,6 +293,12 @@ load_load_barrier(void) {
#endif
}
+// Load a pointer from a memory location that might be being modified
+// concurrently. This prevents the compiler from optimising away
+// multiple loads of the memory location, as it might otherwise do in
+// a busy wait loop for example.
+#define VOLATILE_LOAD(p) (*((StgVolatilePtr)(p)))
+
/* ---------------------------------------------------------------------- */
#else /* !THREADED_RTS */
@@ -331,6 +337,8 @@ atomic_dec(StgVolatilePtr p)
return --(*p);
}
+#define VOLATILE_LOAD(p) ((StgWord)*((StgWord*)(p)))
+
#endif /* !THREADED_RTS */
#endif /* SMP_H */
diff --git a/rts/Messages.c b/rts/Messages.c
index 91ee9a6574..7a37a86259 100644
--- a/rts/Messages.c
+++ b/rts/Messages.c
@@ -186,7 +186,9 @@ nat messageBlackHole(Capability *cap, MessageBlackHole *msg)
// The blackhole must indirect to a TSO, a BLOCKING_QUEUE, an IND,
// or a value.
loop:
- p = UNTAG_CLOSURE(((StgInd*)bh)->indirectee);
+ // NB. VOLATILE_LOAD(), because otherwise gcc hoists the load
+ // and turns this into an infinite loop.
+ p = UNTAG_CLOSURE((StgClosure*)VOLATILE_LOAD(&((StgInd*)bh)->indirectee));
info = p->header.info;
if (info == &stg_IND_info)