Skip to content

Commit 1f6e5b1

Browse files
Make unique_lock and shared_lock use self-swap implementation of move assignment (#5337)
Co-authored-by: Stephan T. Lavavej <stl@microsoft.com>
1 parent 713dd95 commit 1f6e5b1

File tree

5 files changed

+119
-19
lines changed

5 files changed

+119
-19
lines changed

stl/inc/mutex

+2-11
Original file line numberDiff line numberDiff line change
@@ -170,17 +170,8 @@ public:
170170
_Other._Owns = false;
171171
}
172172

173-
unique_lock& operator=(unique_lock&& _Other) noexcept /* strengthened */ {
174-
if (this != _STD addressof(_Other)) {
175-
if (_Owns) {
176-
_Pmtx->unlock();
177-
}
178-
179-
_Pmtx = _Other._Pmtx;
180-
_Owns = _Other._Owns;
181-
_Other._Pmtx = nullptr;
182-
_Other._Owns = false;
183-
}
173+
unique_lock& operator=(unique_lock&& _Other) noexcept {
174+
unique_lock{_STD move(_Other)}.swap(*this);
184175
return *this;
185176
}
186177

stl/inc/shared_mutex

+1-8
Original file line numberDiff line numberDiff line change
@@ -251,14 +251,7 @@ public:
251251
}
252252

253253
shared_lock& operator=(shared_lock&& _Right) noexcept {
254-
if (_Owns) {
255-
_Pmtx->unlock_shared();
256-
}
257-
258-
_Pmtx = _Right._Pmtx;
259-
_Owns = _Right._Owns;
260-
_Right._Pmtx = nullptr;
261-
_Right._Owns = false;
254+
shared_lock{_STD move(_Right)}.swap(*this);
262255
return *this;
263256
}
264257

tests/std/test.lst

+1
Original file line numberDiff line numberDiff line change
@@ -269,6 +269,7 @@ tests\LWG3561_discard_block_engine_counter
269269
tests\LWG3610_iota_view_size_and_integer_class
270270
tests\LWG4084_iostream_uppercase_inf_nan
271271
tests\LWG4105_ranges_ends_with_and_integer_class
272+
tests\LWG4172_unique_lock_self_move_assignment
272273
tests\P0009R18_mdspan_default_accessor
273274
tests\P0009R18_mdspan_extents
274275
tests\P0009R18_mdspan_extents_death
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
# Copyright (c) Microsoft Corporation.
2+
# SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
3+
4+
RUNALL_INCLUDE ..\impure_matrix.lst
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,111 @@
1+
// Copyright (c) Microsoft Corporation.
2+
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
3+
4+
#include <cassert>
5+
#include <mutex>
6+
#include <shared_mutex>
7+
#include <utility>
8+
9+
using namespace std;
10+
11+
struct lockable_with_counters {
12+
void lock() {
13+
++lock_count;
14+
}
15+
16+
void unlock() {
17+
++unlock_count;
18+
}
19+
20+
void lock_shared() {
21+
++shared_lock_count;
22+
}
23+
24+
void unlock_shared() {
25+
++shared_unlock_count;
26+
}
27+
28+
int lock_count = 0;
29+
int unlock_count = 0;
30+
int shared_lock_count = 0;
31+
int shared_unlock_count = 0;
32+
};
33+
34+
#ifdef __clang__
35+
#pragma clang diagnostic push
36+
#pragma clang diagnostic ignored "-Wself-move"
37+
#endif // __clang__
38+
#pragma warning(push)
39+
#pragma warning(disable : 26800) // use a moved-from object
40+
int main() {
41+
lockable_with_counters lockable1;
42+
lockable_with_counters lockable2;
43+
44+
{
45+
unique_lock<lockable_with_counters> lock_a(lockable1);
46+
assert(lockable1.lock_count == 1);
47+
assert(lockable1.unlock_count == 0);
48+
49+
lock_a = move(lock_a);
50+
assert(lockable1.lock_count == 1);
51+
assert(lockable1.unlock_count == 0);
52+
{
53+
unique_lock<lockable_with_counters> lock_b(lockable2);
54+
assert(lockable2.lock_count == 1);
55+
assert(lockable2.unlock_count == 0);
56+
57+
lock_a = move(lock_b);
58+
59+
assert(lockable1.lock_count == 1);
60+
assert(lockable1.unlock_count == 1);
61+
assert(lockable2.lock_count == 1);
62+
assert(lockable2.unlock_count == 0);
63+
}
64+
65+
assert(lockable1.lock_count == 1);
66+
assert(lockable1.unlock_count == 1);
67+
assert(lockable2.lock_count == 1);
68+
assert(lockable2.unlock_count == 0);
69+
}
70+
71+
assert(lockable1.lock_count == 1);
72+
assert(lockable1.unlock_count == 1);
73+
assert(lockable2.lock_count == 1);
74+
assert(lockable2.unlock_count == 1);
75+
76+
{
77+
shared_lock<lockable_with_counters> lock_a(lockable1);
78+
assert(lockable1.shared_lock_count == 1);
79+
assert(lockable1.shared_unlock_count == 0);
80+
81+
lock_a = move(lock_a);
82+
assert(lockable1.shared_lock_count == 1);
83+
assert(lockable1.shared_unlock_count == 0);
84+
{
85+
shared_lock<lockable_with_counters> lock_b(lockable2);
86+
assert(lockable2.shared_lock_count == 1);
87+
assert(lockable2.shared_unlock_count == 0);
88+
89+
lock_a = move(lock_b);
90+
91+
assert(lockable1.shared_lock_count == 1);
92+
assert(lockable1.shared_unlock_count == 1);
93+
assert(lockable2.shared_lock_count == 1);
94+
assert(lockable2.shared_unlock_count == 0);
95+
}
96+
97+
assert(lockable1.shared_lock_count == 1);
98+
assert(lockable1.shared_unlock_count == 1);
99+
assert(lockable2.shared_lock_count == 1);
100+
assert(lockable2.shared_unlock_count == 0);
101+
}
102+
103+
assert(lockable1.shared_lock_count == 1);
104+
assert(lockable1.shared_unlock_count == 1);
105+
assert(lockable2.shared_lock_count == 1);
106+
assert(lockable2.shared_unlock_count == 1);
107+
}
108+
#pragma warning(pop)
109+
#ifdef __clang__
110+
#pragma clang diagnostic pop
111+
#endif // __clang__

0 commit comments

Comments
 (0)