-
-
Notifications
You must be signed in to change notification settings - Fork 32.1k
nogil list reverse
breaks atomicity
#129619
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Comments
Adding the tests mentioned in the main issue here, with sample outputs.
import threading
import sys
def t0(b1,s,res):
b1.wait()
res.append(s.count(3))
def t1(b1,s,res):
b1.wait()
s.reverse()
def Test():
l = [1, 2, 3]
threads=[]
barrier = threading.Barrier(2)
res = []
threads.append(threading.Thread(target= t0, args=(barrier, l,res)))
threads.append(threading.Thread(target= t1, args=(barrier, l,res)))
for i in range(0, len(threads)):
threads[i].start()
for i in range(0, len(threads)):
threads[i].join()
if res[0] != 1:
print("found bug: " + str(res[0]))
print("test begin...")
for i in range(0,50000):
threads = []
if i % 1000 == 0:
print(i)
for i in range(0,100):
threads.append(threading.Thread(target= Test))
for t in threads:
t.start()
for t in threads:
t.join()
print("test Done") Sample output:
import threading
import sys
def t0(b1,s,res):
b1.wait()
s.reverse()
def t1(b1,s,res):
b1.wait()
try:
s.index(3)
except Exception as e:
res.append(e)
def Test():
l = [1, 2, 3]
threads=[]
barrier = threading.Barrier(2)
e = []
threads.append(threading.Thread(target= t0, args=(barrier, l,e)))
threads.append(threading.Thread(target= t1, args=(barrier, l,e)))
for i in range(0, len(threads)):
threads[i].start()
for i in range(0, len(threads)):
threads[i].join()
if len(e) != 0:
print("found bug: " + str(e[0]))
print("test begin...")
for i in range(0,50000):
threads = []
if i % 1000 == 0:
print(i)
for i in range(0,100):
threads.append(threading.Thread(target= Test))
for t in threads:
t.start()
for t in threads:
t.join()
print("test Done") Sample output:
|
I tried all examples several times, but I can't reproduce any of the errors on my macos with m2 with |
I don't think you need to attempt to reproduce the issue. It's clear to me from the implementation that this can happen: I don't think we should change the implementation, but we will want to document the behavior. Making reverse behave atomically would require either locking in all read operations or changing reverse to perform the operation out of place before swapping the In general, we should not rush to make multi-element operations atomic. |
Uh oh!
There was an error while loading. Please reload this page.
Bug report
Bug description:
Hi,
We're a research group focused on testing concurrent runtimes. Our work-in-progress prototype found a violation of atomicity on the current nogil build when using list
reverse
with other concurrent operations on the same list. The program below shows the wrong behavior forreverse
and__contains__
:Operation
__contains__
should always see element3
in the list, either the original or reversed. However, it can see a list without element3
and returnFalse
, as shown in the sample output:We observed the same behavior with operations
count
andindex
executing concurrently withreverse
on the same list, I'll add a comment with those tests and sample outputs.@flypoodles and @overlorde are part of the team, adding them so they get notified about further discussion.
We note that we observed these outputs in several x86_64 machines and one ARM machine.
CPython versions tested on:
3.14, CPython main branch
Operating systems tested on:
Linux
The text was updated successfully, but these errors were encountered: