1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94
|
// Copyright (C) 2013 Vicente Botet
//
// Distributed under the Boost Software License, Version 1.0. (See accompanying
// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
////////////////////////////////////////////
//#define BOOST_THREAD_PROVIDES_GENERIC_SHARED_MUTEX_ON_WIN
#include <iostream>
#include <boost/thread/thread_only.hpp>
#include <boost/thread/shared_mutex.hpp>
// shared_mutex_deadlock.cpp : Defines the entry point for the console application.
//
boost::shared_mutex mutex;
void thread2_func()
{
int i (0);
for (;;)
{
if (mutex.timed_lock(boost::posix_time::milliseconds(500)))
{
std::cout << "Unique lock acquired" << std::endl
<< "Test successful" << std::endl;
mutex.unlock();
break;
}
++i;
if (i == 100)
{
std::cout << "Test failed. App is deadlocked" << std::endl;
break;
}
boost::this_thread::sleep(boost::posix_time::seconds(1));
}
}
void thread3_func()
{
boost::shared_lock<boost::shared_mutex> lock (mutex);
std::cout << "Shared lock acquired" << std::endl
<< "Test successful" << std::endl;
}
void thread3_func_workaround()
{
for (;;)
{
if (mutex.timed_lock_shared(boost::posix_time::milliseconds(200)))
{
std::cout << "Shared lock acquired" << std::endl
<< "Test successful" << std::endl;
mutex.unlock_shared();
break;
}
boost::this_thread::sleep(boost::posix_time::milliseconds(100));
}
}
int main()
{
std::cout << "Starting" << std::endl;
// 1 - lock the mutex
boost::shared_lock<boost::shared_mutex> lock (mutex);
// 2 - start thread#2
boost::thread thread2(&thread2_func);
// 3 - sleep
boost::this_thread::sleep(boost::posix_time::milliseconds(10));
std::cout << "Thread#2 is waiting" << std::endl;
// - start thread3
boost::thread thread3(&thread3_func);
//boost::thread thread3(&thread3_func_workaround);
std::cout << "Thread#3 is started and blocked. It never will waked" << std::endl;
thread3.join(); // will never return
lock.unlock(); // release shared ownership. thread#2 will take unique ownership
thread2.join();
std::cout << "Finished" << std::endl;
return 0;
}
|