Skip to content

Conversation

@tom-pytel
Copy link
Contributor

@tom-pytel tom-pytel commented Apr 16, 2025

Full pyperformance suite times (can anyone can suggest better benchmark for BytesIO?):

             time diff
         mean    stdev
GIL    -0.01% +- 2.31%
noGIL  +0.22% +- 2.93%

Changes, * functions had critical section added, - not. Atomics were added for exports field:

static PyGetSetDef bytesio_getsetlist[] = {
*   {"closed",  bytesio_get_closed, NULL,    // @critical_section to avoid make `buf` atomic many places

static struct PyMethodDef bytesio_methods[] = {
-   _IO_BYTESIO_READABLE_METHODDEF
-   _IO_BYTESIO_SEEKABLE_METHODDEF
-   _IO_BYTESIO_WRITABLE_METHODDEF
*   _IO_BYTESIO_CLOSE_METHODDEF
-   _IO_BYTESIO_FLUSH_METHODDEF
-   _IO_BYTESIO_ISATTY_METHODDEF
*   _IO_BYTESIO_TELL_METHODDEF      // @critical_section to avoid make `pos` atomic many places
*   _IO_BYTESIO_WRITE_METHODDEF
*   _IO_BYTESIO_WRITELINES_METHODDEF
*   _IO_BYTESIO_READ1_METHODDEF
*   _IO_BYTESIO_READINTO_METHODDEF
*   _IO_BYTESIO_READLINE_METHODDEF
*   _IO_BYTESIO_READLINES_METHODDEF
*   _IO_BYTESIO_READ_METHODDEF
*   _IO_BYTESIO_GETBUFFER_METHODDEF
*   _IO_BYTESIO_GETVALUE_METHODDEF
*   _IO_BYTESIO_SEEK_METHODDEF      // @critical_section to avoid make `pos` atomic many places
*   _IO_BYTESIO_TRUNCATE_METHODDEF

*   {"__getstate__", bytesio_getstate,  METH_NOARGS, NULL},
*   {"__setstate__", bytesio_setstate,  METH_O, NULL},
*   {"__sizeof__",   bytesio_sizeof,     METH_NOARGS, NULL},

static PyType_Slot bytesio_slots[] = {
-   {Py_tp_dealloc,   bytesio_dealloc},
-   {Py_tp_traverse,  bytesio_traverse},
-   {Py_tp_clear,     bytesio_clear},
-   {Py_tp_iter,      PyObject_SelfIter},
*   {Py_tp_iternext,  bytesio_iternext},
*   {Py_tp_init,      _io_BytesIO___init__},
-   {Py_tp_new,       bytesio_new},

static PyType_Slot bytesiobuf_slots[] = {
-   {Py_tp_dealloc, bytesiobuf_dealloc},
-   {Py_tp_traverse, bytesiobuf_traverse},
-   {Py_bf_getbuffer, bytesiobuf_getbuffer},
-   {Py_bf_releasebuffer, bytesiobuf_releasebuffer},

Notes:

  • Opted for critical sections in three functions instead of making buf and or pos atomic everywhere.
  • New test test_free_threading.test_io.TestBytesIO clean under --parallel-threads TSAN.
  • bytesiobuf_getbuffer doesn't need critical sections because as far as I see its only used for BytesIO in bytesio.c where its use is protected by a critical section in _io_BytesIO_getbuffer_impl (its also used for a random non-BytesIO-related test in _testcapimodule.c).

@tom-pytel
Copy link
Contributor Author

Ping @ZeroIntensity, @serhiy-storchaka

Copy link
Member

@ZeroIntensity ZeroIntensity left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The implementation looks fine, but I'm worried that concurrent use of BytesIO is an actual use-case that we need to consider.

For example, I could see someone using a logger that writes to an IO-like object from multiple threads. If they wanted to log to a BytesIO buffer (e.g., for temporarily suppressing the output), then that would scale pretty terribly with critical sections. I'm speculating, though.

@tom-pytel
Copy link
Contributor Author

For example, I could see someone using a logger that writes to an IO-like object from multiple threads. If they wanted to log to a BytesIO buffer (e.g., for temporarily suppressing the output), then that would scale pretty terribly with critical sections. I'm speculating, though.

Well, that's why I asked about a specific better benchmark to test with because pyperformance is not specific enough to this to show a perf hit. But as for the case you mention, if someone is writing the BytesIO object from multiple threads then performance is not really an issue as they will eventually segfault.

$ ./python ../t.py 
1
2
Segmentation fault (core dumped)

t.py:

from io import BytesIO
from random import randint
import threading


def write(barrier, b):
    barrier.wait()
    b.write(b'0' * randint(100, 1000))


def check(funcs, *args):
    barrier = threading.Barrier(len(funcs))
    thrds = []

    for func in funcs:
        thrd = threading.Thread(target=func, args=(barrier, *args))

        thrds.append(thrd)
        thrd.start()

    for thrd in thrds:
        thrd.join()


if __name__ == "__main__":
    count = 0

    while True:
        print(count := count + 1)

        check([write] * 10, BytesIO())

@ZeroIntensity
Copy link
Member

ZeroIntensity commented Apr 16, 2025

Well, yeah, we need to fix the races. I'm just wondering how we should fix it. Would you mind benchmarking (with this PR) the case I brought up against different numbers of threads? I want to see how bad the contention gets (and compare it to a GILful build as well).

@tom-pytel
Copy link
Contributor Author

Well, yeah, we need to fix the races. I'm just wondering how we should fix it. Would you mind benchmarking (with this PR) the case I brought up against different numbers of threads? I want to see how bad the contention gets (and compare it to a GILful build as well).

Ok, will do a bunch of tests tomorrow.

@tom-pytel
Copy link
Contributor Author

tom-pytel commented Apr 17, 2025

EDIT: I just realized with a good bit of rewriting lock-free reads might be possible. EDIT2: Nope, see below.

TL;DR: GIL build performance is same as expected while noGIL makes multithread writes possible and otherwise seems ok.

Detail: noGIL PR has a minor hit to single-threaded read and a smaller hit to single-threaded write (cmdline timeit). Multithreaded timings are unstable (especially write, run the script and u will get 2x variation for that), but read looks slower than main while write is much faster than main with explicit locking (which is the only thing to compare it to as otherwise writes crash).

./python -m timeit -s "from io import BytesIO; b = BytesIO((b'.'*99+b'\n')*50_000)" "while b.read(100): pass"
./python -m timeit -s "from io import BytesIO; b = BytesIO((b'.'*99+b'\n')*50_000)" "while b.readline(): pass"
./python -m timeit -s "from io import BytesIO; b = BytesIO(); l = b'.'*99+b'\n'" "for _ in range(50_000): b.write(l)"

Times in ns, lower is better.

SINGLE MAIN THREAD '-m timeit' cmdline (above):  (per loop, read/readline = 20000000 loops, write = 100 loops)
=====================================
             read  readline     write
GIL main     18.3      14.4      2580
GIL PR       18.2      14.2      2590
noGIL main   18.8      14.7      3710
noGIL PR     20.3      16.1      3860


MULTITHREADING benchmark script (below):
========================================

read:
-----
cfg / #thrds     1      2      3      4      5
GIL main      24.4   23.6   23.8   23.7   23.7
GIL PR        23.4   23.7   24.1   23.5   24.8
noGIL main    27.4   44.0   64.8    124    116
noGIL PR      28.6   57.6   90.5    129    163

readline:
---------
cfg / #thrds     1      2      3      4      5
GIL main      22.4   22.3   22.8   22.2   23.7
GIL PR        22.0   22.2   22.7   21.9   25.0
noGIL main    25.4   46.2   85.8    198    261
noGIL PR      26.1   54.7   82.8    114    152

write:  (* = with threading.Lock for 'noGIL main', otherwise crash)
------
cfg / #thrds     1      2      3      4      5
GIL main      22.6   22.2   24.6   22.4   23.0
GIL PR        22.1   23.2   22.4   23.0   23.4
noGIL main    42.2    286*   571*   805*  1077*
noGIL PR      39.7    123    287    535    677

write:  (different script (not below) to double check unstable timings, total time)
------
cfg / #thrds         1          2          3          4          5
GIL main       1171556    2474471    3664172    4817181    6343492
GIL PR         1195404    2442720    3698038    4949697    6276943
noGIL main     5997412   22573890*  27660777*  43626582*  61317669*
noGIL PR       2332928    6786042   19316932   38951468   31712621

Ping @ZeroIntensity, timings as you requested.

import sys
from io import BytesIO
from threading import Barrier, Lock, Thread
from time import time_ns

BESTOF = 5
NLINES = 50_000
LINE   = b'.'*99 + b'\n'
LLEN   = len(LINE)
LOCK   = Lock()


def read(barrier, out, b):
    barrier.wait()
    t0 = time_ns()
    while b.read(LLEN): pass
    out[0] = time_ns() - t0

def readline(barrier, out, b):
    barrier.wait()
    t0 = time_ns()
    while b.readline(): pass
    out[0] = time_ns() - t0

def write_locked(barrier, out, b):
    barrier.wait()
    t0 = time_ns()
    for _ in range(NLINES):
        with LOCK:
            b.write(LINE)
    out[0] = time_ns() - t0

def write(barrier, out, b):
    barrier.wait()
    t0 = time_ns()
    for _ in range(NLINES):
        b.write(LINE)
    out[0] = time_ns() - t0


def check(func, nthrds, b, is_wr):  # is_wr compensates for write total size *= nthrds while read total size is fixed
    barrier = Barrier(nthrds)
    outs = []
    thrds = []

    for _ in range(nthrds):
        out = [0]
        thrd = Thread(target=func, args=(barrier, out, b))

        outs.append(out)
        thrds.append(thrd)
        thrd.start()

    for thrd in thrds:
        thrd.join()

    return sum(o[0] for o in outs) / ((nthrds if is_wr else 1) * NLINES)

def checkbest(func, nthrds, setup, is_wr):
    best = float('inf')

    for _ in range(BESTOF):
        best = min(best, check(func, nthrds, setup(), is_wr))

    return best

if __name__ == "__main__":
    nthrds = int((sys.argv + ['1'])[1])

    print(f'read:         ', checkbest(read, nthrds, lambda: BytesIO(LINE * NLINES), False), 'ns')
    print(f'readline:     ', checkbest(readline, nthrds, lambda: BytesIO(LINE * NLINES), False), 'ns')
    print(f'write_locked: ', checkbest(write_locked, nthrds, lambda: BytesIO(), True), 'ns')
    print(f'write:        ', checkbest(write, nthrds, lambda: BytesIO(), True), 'ns')

Copy link
Member

@serhiy-storchaka serhiy-storchaka left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Many functions are just wrapped in critical section. I wonder, what if add special flag, METH_FT_SYNC, and when it set, wrap the method calling code in critical section?

@tom-pytel
Copy link
Contributor Author

Many functions are just wrapped in critical section. I wonder, what if add special flag, METH_FT_SYNC, and when it set, wrap the method calling code in critical section?

If you're talking about stuff like bytesio_sizeof, I could have just handed that over to argument clinic to wrap but I guess I was being lazy. Argument clinic works well enough for the critical section wrapping I think.

@serhiy-storchaka
Copy link
Member

I mean that this would work also for methods that do not use Argument Clinic. Was this idea discussed before?

@tom-pytel
Copy link
Contributor Author

I mean that this would work also for methods that do not use Argument Clinic. Was this idea discussed before?

Couldn't tell you that, I'm relatively new here (: But one drawback I can see would be that it would only work when calling via py mechanism, if u called directly from C you would still have to do the critical section on your own as opposed to a clinic-generated function which would have it, no?

@ZeroIntensity
Copy link
Member

I mean that this would work also for methods that do not use Argument Clinic. Was this idea discussed before?

Yeah, I've suggested it in the past. Apparently, it's been tried and failed before. I don't know much of the details.

@tom-pytel
Copy link
Contributor Author

Ok, had a look and lock-free reads can almost certainly be done but they would not have the data integrity guarantees you would want from a file-like object without sync. They would work in the same way lock-free multithreaded iteration works, could get dup data and might not sync position correctly wrt writes, which is not what you expect from a "file". So scratch that idea.

@kumaraditya303 kumaraditya303 enabled auto-merge (squash) May 8, 2025 18:21
@kumaraditya303 kumaraditya303 merged commit 5dd3a3a into python:main May 8, 2025
42 checks passed
Pranjal095 pushed a commit to Pranjal095/cpython that referenced this pull request Jul 12, 2025
@ngoldbaum
Copy link
Contributor

Just to clarify - this didn't make it into 3.14, right?

@tom-pytel
Copy link
Contributor Author

Just to clarify - this didn't make it into 3.14, right?

Looking at the code I don't see it, so no.

@ngoldbaum
Copy link
Contributor

ngoldbaum commented Jul 24, 2025

OK - I think we'll make it so pytest-run-parallel marks tests that use BytesIO on 3.13 and 3.14 as thread-unsafe: Quansight-Labs/pytest-run-parallel#100

I happened to hit the issues fixed by this PR running the tests for python-isal under pytest-run-parallel.

@picnixz
Copy link
Member

picnixz commented Sep 2, 2025

@kumaraditya303 should this be backported?

@kumaraditya303
Copy link
Contributor

should this be backported?

I didn't backport this as the change is large and it only crashes when multiple threads modify it concurrently which shouldn't be done.

@tom-pytel
Copy link
Contributor Author

should this be backported?

I didn't backport this as the change is large and it only crashes when multiple threads modify it concurrently which shouldn't be done.

It was at the end of the 3.14 cycle so a backport shouldn't be very painful, if you want I could have a look.

@ngoldbaum
Copy link
Contributor

It was at the end of the 3.14 cycle so a backport shouldn't be very painful, if you want I could have a look.

A backport would be appreciated, I think it would help resolve this discussion and get a downstream release supporting 3.14 out: pycompression/python-isal#233 (comment)

@kumaraditya303 kumaraditya303 added the needs backport to 3.14 bugs and security fixes label Sep 5, 2025
@miss-islington-app
Copy link

Thanks @tom-pytel for the PR, and @kumaraditya303 for merging it 🌮🎉.. I'm working now to backport this PR to: 3.14.
🐍🍒⛏🤖

@miss-islington-app
Copy link

Sorry, @tom-pytel and @kumaraditya303, I could not cleanly backport this to 3.14 due to a conflict.
Please backport using cherry_picker on command line.

cherry_picker 5dd3a3a58cca4798ebfc2edd673211067453e81e 3.14

@bedevere-app
Copy link

bedevere-app bot commented Sep 5, 2025

GH-138551 is a backport of this pull request to the 3.14 branch.

@bedevere-app bedevere-app bot removed the needs backport to 3.14 bugs and security fixes label Sep 5, 2025
kumaraditya303 added a commit to kumaraditya303/cpython that referenced this pull request Sep 5, 2025
(cherry picked from commit 5dd3a3a)

Co-authored-by: Tomasz Pytel <[email protected]>
Co-authored-by: Kumar Aditya <[email protected]>
@kumaraditya303
Copy link
Contributor

I created #138551 to backport it if it helps, although I think that the isal issue is about sharing io.BytesIO in a fixture across threads and that is still incorrect.

kumaraditya303 added a commit that referenced this pull request Oct 7, 2025
* [3.14] gh-132551: make `io.BytesIO` thread safe (GH-132616)
(cherry picked from commit 5dd3a3a)

Co-authored-by: Tomasz Pytel <[email protected]>
Co-authored-by: Kumar Aditya <[email protected]>
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.

6 participants