I'm currently trying to get two boost::thread
s to get equal processing time on the CPU using a shared mutex called mutexCOM
. As it currently stands both threads are set to run infinitely. One thread (The main thread) simply prints out "testing!" while the other thread, running the Navigation::begin()
function, prints "-----shared function-----", waits one second, then prints "-----Shared function will now end...-----". The wait is meant to simulate the larger amount of processing necessary for the function that will replace this stub. Each iteration of the infinite loops is preceded within the loop by a scoped lock on mutexCOM
. The expected output should be something like the following ad infinitum:
testing!
-----shared function-----
----Shared function will now end...-----
testing!
-----shared function-----
----Shared function will now end...-----
Unfortunately, something I've done causes the output instead to be:
`
It seems that the iterations are being grouped together instead of being interlaced like I would expect. If anyone can explain what's going on here I would be very appreciative, as it seems that my understanding of thread management using mutexes could use some work.
My only theory is that my assumption that boost handles queuing for locking the mutex is wrong and that I need to somehow create my own queue to give the threads fair access to avoid starvation. However, I'm doubtful about this because it would basically render the point of mutexes moot as I would still need to handle all of the management on my own. At that point why not just use a semaphore?
My code is as follows. Only relevant code is included, so don't worry about the missing class structure. Assume all headers are properly included and that SharedMutex.h is included in both classes:
SharedMutex.h -
#ifndef SHARED_MUTEX_
#define SHARED_MUTEX_
#include <boost/thread/mutex.hpp>
extern boost::mutex mutexCOM;
#endif
Navigation.cpp -
Navigation::begin() {
boost::mutex::scoped_lock lock(mutexCOM);
cout<< "-----shared function-----" << endl;
sleep(1);
cout<< "----Shared function will now end...-----" << endl;
}
mainclass.cpp -
void mainclass::run() {
this->runNav();
while(1) {
boost::mutex::scoped_lock lock(mutexCOM);
cout << "testing!" << endl;
}
}
void mainclass::runNav() {
//this->nav is an instance of Navigation within mainclass
boost::thread navThread(boost::bind(&Navigation::begin, *(this->nav)));
//this->navPtr is a pointer to the boost::thread for later management
this->navPtr=&navThread;
}
This is wrong:
You're taking the address of a local variable there. That's Undefined Behaviour if you use
navPtr
later on. Instead, just store the thread in a member: see sample below.Indeed. If you want task queueing, write it. Mutexes are not for that. As the name implies, mutexes are just for mutual exclusion, the basic guarantee that two blocks of code (the critical sections) do not run at the same time.
It doesn't guarantee anything else.
Thread starvation is indeed the issue, as can be shown by introducing yield/sleeps at some times:
Live On Coliru
Which prints:
./a.out | uniq -c