c++11: don't throw from the reverselock destructor (#1421)
noexcept is default for destructors as of c++11. By throwing in reverselock's destructor if it's lock has been tampered with, the likely result is std::terminate being called. Indeed that happened before this change. Once reverselock has taken another lock (its ctor didn't throw), it makes no sense to try to grab or lock the parent lock. That is be broken/undefined behavior depending on the parent lock's implementation, but it shouldn't cause the reverselock to fail to re-lock when destroyed. To avoid those problems, simply swap the parent lock's contents with a dummy for the duration of the lock. That will ensure that any undefined behavior is caught at the call-site rather than the reverse lock's destruction. Barring a failed mutex unlock which would be indicative of a larger problem, the destructor should now never throw.
This commit is contained in:
parent
8bbcf62000
commit
f3b92a95d9
@ -15,10 +15,12 @@ public:
|
|||||||
|
|
||||||
explicit reverse_lock(Lock& lock) : lock(lock) {
|
explicit reverse_lock(Lock& lock) : lock(lock) {
|
||||||
lock.unlock();
|
lock.unlock();
|
||||||
|
lock.swap(templock);
|
||||||
}
|
}
|
||||||
|
|
||||||
~reverse_lock() {
|
~reverse_lock() {
|
||||||
lock.lock();
|
templock.lock();
|
||||||
|
templock.swap(lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
@ -26,6 +28,7 @@ private:
|
|||||||
reverse_lock& operator=(reverse_lock const&);
|
reverse_lock& operator=(reverse_lock const&);
|
||||||
|
|
||||||
Lock& lock;
|
Lock& lock;
|
||||||
|
Lock templock;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // BITCOIN_REVERSELOCK_H
|
#endif // BITCOIN_REVERSELOCK_H
|
||||||
|
@ -42,22 +42,18 @@ BOOST_AUTO_TEST_CASE(reverselock_errors)
|
|||||||
BOOST_CHECK(failed);
|
BOOST_CHECK(failed);
|
||||||
BOOST_CHECK(!lock.owns_lock());
|
BOOST_CHECK(!lock.owns_lock());
|
||||||
|
|
||||||
// Make sure trying to lock a lock after it has been reverse locked fails
|
// Locking the original lock after it has been taken by a reverse lock
|
||||||
failed = false;
|
// makes no sense. Ensure that the original lock no longer owns the lock
|
||||||
bool locked = false;
|
// after giving it to a reverse one.
|
||||||
|
|
||||||
lock.lock();
|
lock.lock();
|
||||||
BOOST_CHECK(lock.owns_lock());
|
BOOST_CHECK(lock.owns_lock());
|
||||||
|
{
|
||||||
try {
|
|
||||||
reverse_lock<boost::unique_lock<boost::mutex> > rlock(lock);
|
reverse_lock<boost::unique_lock<boost::mutex> > rlock(lock);
|
||||||
lock.lock();
|
BOOST_CHECK(!lock.owns_lock());
|
||||||
locked = true;
|
|
||||||
} catch(...) {
|
|
||||||
failed = true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
BOOST_CHECK(locked && failed);
|
BOOST_CHECK(failed);
|
||||||
BOOST_CHECK(lock.owns_lock());
|
BOOST_CHECK(lock.owns_lock());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user