summaryrefslogtreecommitdiff
path: root/src/mixed_arena.h
diff options
context:
space:
mode:
authorThomas Lively <7121787+tlively@users.noreply.github.com>2021-04-14 14:43:01 -0700
committerGitHub <noreply@github.com>2021-04-14 14:43:01 -0700
commitb2c63a9665a9758c50eac60af605f0399f66580f (patch)
treea69f69ae7bf9ef6272c7fad6eb33d86e6625e55d /src/mixed_arena.h
parentd4f8cd5ccabaaf2c55561548d88c075cfa4f765d (diff)
downloadbinaryen-b2c63a9665a9758c50eac60af605f0399f66580f.tar.gz
binaryen-b2c63a9665a9758c50eac60af605f0399f66580f.tar.bz2
binaryen-b2c63a9665a9758c50eac60af605f0399f66580f.zip
Fix lock-free MixedArena allocation (#3807)
To reduce allocator contention, MixedArena transparently forwards allocations to a thread-local arena by traversing a linked list of arenas, looking for the one corresponding to the current thread. If the search reaches the end of the linked list, it allocates a new arena for the current thread and atomically appends it to the end of the list using a compare-and-swap operation. The problem was that the previous code used `compare_exchange_weak`, which is allowed to spuriously fail, i.e. it is allowed to behave as though the original value is not the same as the expected value, even when it is. In this case, that means that it might fail to append the new allocation to the list even if the `next` pointer is still null, which results in a subsequent null pointer dereference. The fix is to use `compare_exchange_strong`, which is not allowed to spuriously fail in this way. Reported in #3806.
Diffstat (limited to 'src/mixed_arena.h')
-rw-r--r--src/mixed_arena.h2
1 files changed, 1 insertions, 1 deletions
diff --git a/src/mixed_arena.h b/src/mixed_arena.h
index 4d2e24542..a36f72a8d 100644
--- a/src/mixed_arena.h
+++ b/src/mixed_arena.h
@@ -104,7 +104,7 @@ struct MixedArena {
if (!allocated) {
allocated = new MixedArena(); // has our thread id
}
- if (curr->next.compare_exchange_weak(seen, allocated)) {
+ if (curr->next.compare_exchange_strong(seen, allocated)) {
// we replaced it, so we are the next in the chain
// we can forget about allocated, it is owned by the chain now
allocated = nullptr;