Skip to content

Commit 13bb8cc

Browse files
NHDalyKristoffer Carlsson
authored and
Kristoffer Carlsson
committed
Fix assert_havelock(::ReentrantLock) to assert that the _current-task_ has the lock. (#33159)
* Fix `assert_havelock(::ReentrantLock)` to assert that the _current-task_ has the lock. Before this commit, new threads would incorrectly believe that they held a lock on a Condition when they actually didn't, and would allow illegal operations, e.g. notify: ```julia julia> c = Threads.Condition() Base.GenericCondition{ReentrantLock}(Base.InvasiveLinkedList{Task}(nothing, nothing), ReentrantLock(nothing, Base.GenericCondition{Base.Threads.SpinLock}(Base.InvasiveLinkedList{Task}(nothing, nothing), Base.Threads.SpinLock(Base.Threads.Atomic{Int64}(0))), 0)) julia> lock(c) julia> fetch(Threads.@Spawn Base.assert_havelock(c)) # This should be an ERROR (the new thread doesn't have the lock) julia> fetch(Threads.@Spawn notify(c)) # This should be an ERROR (the new thread doesn't have the lock) 0 julia> fetch(Threads.@Spawn wait(c)) # This error should be caught earlier (in assert_havelock). ERROR: TaskFailedException: unlock from wrong thread Stacktrace: [1] error(::String) at ./error.jl:33 [2] unlockall(::ReentrantLock) at ./lock.jl:121 [3] wait(::Base.GenericCondition{ReentrantLock}) at ./condition.jl:105 [4] (::var"##19#20")() at ./threadingconstructs.jl:113 ``` (The same holds for `@async` as `@spawn`.) After this change, the assertion works correctly: ``` julia> c = Threads.Condition(); julia> lock(c) julia> fetch(Threads.@Spawn Base.assert_havelock(c)) # This correctly ERRORs ERROR: TaskFailedException: concurrency violation detected Stacktrace: [1] error(::String) at ./error.jl:33 [2] concurrency_violation() at ./condition.jl:8 [3] assert_havelock at ./condition.jl:28 [inlined] [4] assert_havelock at ./REPL[22]:1 [inlined] [5] assert_havelock(::Base.GenericCondition{ReentrantLock}) at ./condition.jl:73 [6] (::var"##21#22")() at ./threadingconstructs.jl:113 ``` Also adds unit test that failed before this commit but now succeeds * Remove default impl of `assert_havelock`; add `::SpinLock` impl (cherry picked from commit 784eb57)
1 parent 3e543a2 commit 13bb8cc

File tree

4 files changed

+20
-1
lines changed

4 files changed

+20
-1
lines changed

base/condition.jl

-1
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,6 @@ function trylock end
2222
function islocked end
2323
unlockall(l::AbstractLock) = unlock(l) # internal function for implementing `wait`
2424
relockall(l::AbstractLock, token::Nothing) = lock(l) # internal function for implementing `wait`
25-
assert_havelock(l::AbstractLock) = assert_havelock(l, Threads.threadid())
2625
assert_havelock(l::AbstractLock, tid::Integer) =
2726
(islocked(l) && tid == Threads.threadid()) ? nothing : concurrency_violation()
2827
assert_havelock(l::AbstractLock, tid::Task) =

base/lock.jl

+1
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@ mutable struct ReentrantLock <: AbstractLock
1616
ReentrantLock() = new(nothing, GenericCondition{Threads.SpinLock}(), 0)
1717
end
1818

19+
assert_havelock(l::ReentrantLock) = assert_havelock(l, l.locked_by)
1920

2021
"""
2122
islocked(lock) -> Status (Boolean)

base/locks-mt.jl

+4
Original file line numberDiff line numberDiff line change
@@ -30,6 +30,10 @@ struct SpinLock <: AbstractLock
3030
SpinLock() = new(Atomic{Int}(0))
3131
end
3232

33+
# Note: this cannot assert that the lock is held by the correct thread, because we do not
34+
# track which thread locked it. Users beware.
35+
Base.assert_havelock(l::SpinLock) = islocked(l) ? nothing : concurrency_violation()
36+
3337
function lock(l::SpinLock)
3438
while true
3539
if l.handle[] == 0

test/threads_exec.jl

+15
Original file line numberDiff line numberDiff line change
@@ -151,6 +151,21 @@ end
151151
threaded_gc_locked(SpinLock)
152152
threaded_gc_locked(Threads.ReentrantLock)
153153

154+
# Issue 33159
155+
# Make sure that a Threads.Condition can't be used without being locked, on any thread.
156+
@testset "Threads.Conditions must be locked" begin
157+
c = Threads.Condition()
158+
@test_throws Exception notify(c)
159+
@test_throws Exception wait(c)
160+
161+
# If it's locked, but on the wrong thread, it should still throw an exception
162+
lock(c)
163+
@test_throws Exception fetch(@async notify(c))
164+
@test_throws Exception fetch(@async notify(c, all=false))
165+
@test_throws Exception fetch(@async wait(c))
166+
unlock(c)
167+
end
168+
154169
# Issue 14726
155170
# Make sure that eval'ing in a different module doesn't mess up other threads
156171
orig_curmodule14726 = @__MODULE__

0 commit comments

Comments
 (0)