From 87c2162187413fe2631cef8801288f4619990886 Mon Sep 17 00:00:00 2001 From: Milan Bouchet-Valat Date: Tue, 10 Oct 2023 12:20:52 +0200 Subject: [PATCH] Fix `@spawn_or_run_task` with interactive threads (#3385) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit When Julia ≥ 1.9 is started with `-tX,Y`, by default tasks use the pool of interactive threads instead of the default pool. Fix this by updating `spawn_or_run_task` and `spawn_or_run` to match the code used by `@spawn` in Julia master. --- .github/workflows/ci.yml | 2 +- NEWS.md | 7 +++++++ docs/src/lib/functions.md | 3 ++- src/other/utils.jl | 26 ++++++++++++++++++++------ 4 files changed, 30 insertions(+), 8 deletions(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 4222741a09..5bbb7f0772 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -36,7 +36,7 @@ jobs: - uses: julia-actions/julia-buildpkg@v1 - uses: julia-actions/julia-runtest@v1 env: - JULIA_NUM_THREADS: 4 + JULIA_NUM_THREADS: 4,1 - uses: julia-actions/julia-processcoverage@v1 - uses: codecov/codecov-action@v1 with: diff --git a/NEWS.md b/NEWS.md index 468c3656e7..8a46730dac 100644 --- a/NEWS.md +++ b/NEWS.md @@ -10,6 +10,13 @@ keyword argument ([#3380](https://github.com/JuliaData/DataFrames.jl/pull/3380)) +## Bug fixes + +* Always use the default thread pool for multithreaded operations, + instead of using the interactive thread pool when Julia was started + with `-tM,N` with N > 0 + ([#3385](https://github.com/JuliaData/DataFrames.jl/pull/3385)) + # DataFrames.jl v1.6.1 Release Notes ## Bug fixes diff --git a/docs/src/lib/functions.md b/docs/src/lib/functions.md index 625cf60665..1cb8dfd66c 100644 --- a/docs/src/lib/functions.md +++ b/docs/src/lib/functions.md @@ -7,7 +7,8 @@ CurrentModule = DataFrames ## Multithreading support By default, selected operations in DataFrames.jl automatically use multiple threads -when available. It is task-based and implemented using the `@spawn` macro from Julia Base. +when available. Multi-threading is task-based and implemented using the `@spawn` +macro from Julia Base. Tasks are therefore scheduled on the `:default` threadpool. Functions that take user-defined functions and may run it in parallel accept a `threads` keyword argument which allows disabling multithreading when the provided function requires serial execution or is not thread-safe. diff --git a/src/other/utils.jl b/src/other/utils.jl index 455c406f46..78f91785b1 100644 --- a/src/other/utils.jl +++ b/src/other/utils.jl @@ -221,16 +221,23 @@ end Equivalent to `Threads.@spawn` if `threads === true`, otherwise run `expr` and return a `Task` that returns its value. """ -macro spawn_or_run_task(threads, expr) - letargs = Base._lift_one_interp!(expr) +macro spawn_or_run_task(threads, ex) + letargs = Base._lift_one_interp!(ex) - thunk = esc(:(()->($expr))) + thunk = :(()->($(esc(ex)))) + @static if VERSION >= v"1.10.0-DEV" + Base.replace_linenums!(thunk, __source__) + end var = esc(Base.sync_varname) + spawn_set_thrpool = VERSION >= v"1.9.0" ? + :(Base.Threads._spawn_set_thrpool(task, :default)) : + :() quote let $(letargs...) if $(esc(threads)) local task = Task($thunk) task.sticky = false + $(spawn_set_thrpool) else # Run expr immediately res = $thunk() @@ -253,16 +260,23 @@ end Equivalent to `Threads.@spawn` if `threads === true`, otherwise run `expr`. """ -macro spawn_or_run(threads, expr) - letargs = Base._lift_one_interp!(expr) +macro spawn_or_run(threads, ex) + letargs = Base._lift_one_interp!(ex) - thunk = esc(:(()->($expr))) + thunk = :(()->($(esc(ex)))) + if VERSION >= v"1.10.0-DEV" + Base.replace_linenums!(thunk, __source__) + end var = esc(Base.sync_varname) + spawn_set_thrpool = VERSION >= v"1.9.0" ? + :(Base.Threads._spawn_set_thrpool(task, :default)) : + :() quote let $(letargs...) if $(esc(threads)) local task = Task($thunk) task.sticky = false + $(spawn_set_thrpool) if $(Expr(:islocal, var)) put!($var, task) end