KEMBAR78
gh-123471: Make itertools.product and itertools.combinations thread-safe by eendebakpt · Pull Request #132814 · python/cpython · GitHub
Skip to content

Conversation

@eendebakpt
Copy link
Contributor

@eendebakpt eendebakpt commented Apr 22, 2025

We make concurrent iteration over itertools.combinations and itertools.product thread safe in the free-threading build.
The original combinations_next is renamed to combinations_next_with_lock_held and combinations_next is now calling combinations_next_with_lock_held with a lock (similar for itertools.product)

We use a lock because it is easy to implement and avoids quite a bit of complexity (we have two pieces of mutable state to deal with: op->indices and op->result).

Issues that can occur without the locks:

  • On initialization of the results structure the op->result can be overwritten, resulting in memory leaks

PyTuple_SET_ITEM(result, i, elem);

The tests in this PR trigger some of these issues, although some are not visible (e.g. the memory leak), and it typically requires more iterations to result in a segfault. On my system > 2000 iterations gives a very high probability of triggering a segfault. The number of iterations is set much lower to keep the duration of the test < 0.1 second.

Performance with the locks is about 5% less for a single-thread (see the corresponding issue).

I refactored the tests to avoid duplicated code. Currently combinations and product are in the test, but cwr and permutations have the same style and could be added as well (in a followup PR).

Could we do this without a full lock? It depends a bit on the iterator. For product we could make the rollover check safe by changing indices[i] == PyTuple_GET_SIZE(pool) into indices[i] >= PyTuple_GET_SIZE(pool) and use atomic operations in all operations dealing with op->indices or op->result. That would still leave memory leaks, but these are not crashes. And determining whether this actually safe (not crashing) requires some very careful reviews.

@rhettinger rhettinger removed their request for review April 22, 2025 22:22
@kumaraditya303 kumaraditya303 enabled auto-merge (squash) June 30, 2025 11:05
@kumaraditya303 kumaraditya303 merged commit 847d1c2 into python:main Jun 30, 2025
40 checks passed
AndPuQing pushed a commit to AndPuQing/cpython that referenced this pull request Jul 11, 2025
…read-safe (python#132814)

Co-authored-by: Kumar Aditya <kumaraditya@python.org>
Pranjal095 pushed a commit to Pranjal095/cpython that referenced this pull request Jul 12, 2025
…read-safe (python#132814)

Co-authored-by: Kumar Aditya <kumaraditya@python.org>
picnixz pushed a commit to picnixz/cpython that referenced this pull request Jul 13, 2025
…read-safe (python#132814)

Co-authored-by: Kumar Aditya <kumaraditya@python.org>
taegyunkim pushed a commit to taegyunkim/cpython that referenced this pull request Aug 4, 2025
…read-safe (python#132814)

Co-authored-by: Kumar Aditya <kumaraditya@python.org>
Agent-Hellboy pushed a commit to Agent-Hellboy/cpython that referenced this pull request Aug 19, 2025
…read-safe (python#132814)

Co-authored-by: Kumar Aditya <kumaraditya@python.org>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Projects

None yet

Development

Successfully merging this pull request may close these issues.

2 participants