diff options
author | Thomas Lively <7121787+tlively@users.noreply.github.com> | 2021-04-14 14:43:01 -0700 |
---|---|---|
committer | GitHub <noreply@github.com> | 2021-04-14 14:43:01 -0700 |
commit | b2c63a9665a9758c50eac60af605f0399f66580f (patch) | |
tree | a69f69ae7bf9ef6272c7fad6eb33d86e6625e55d /src/mixed_arena.h | |
parent | d4f8cd5ccabaaf2c55561548d88c075cfa4f765d (diff) | |
download | binaryen-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.h | 2 |
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; |