I'm trying to wrap my head around how I can ensure that an objects reference count is thread safe.
class MyObject{
//Other implementation details
private:
mutable volatile LONGLONG * m_count;
IData * m_data;
};
Assume the necessary class declaration are there, just keeping it simple. Here is the implementation of the copy constructor and destructor.
MyObject::MyObject(const MyObject& rhs) : m_count(rhs.m_count), m_data(rhs.m_data){
InterlockedIncrement64(m_count);
}
MyObject::~MyObject(){
if(InterlockedDecrement64(m_count) == 0)
delete m_data;
}
Is this thread safe? How is the intilization list of the copy constructor seen, atomic or not? Does that even matter? Should I be setting the incremented value of the count in the initilization list (is this possible)?
As it stands is this good enough. I'm thinking it is, otherwise, how could I get into a scenario where thread1
is copying and thread2
is destroying concurrently when the count == 1
. There has to be a handshake between the threads, meaning thread1 has to copy the object entirely before thread2's object goes out of scope correct?
After reading some of these responses I went back and did a little research. Boost implements their shared_ptr very similiarly. Here is the destructor call.
void release() // nothrow
{
if( BOOST_INTERLOCKED_DECREMENT( &use_count_ ) == 0 )
{
dispose();
weak_release();
}
}
Some have suggested that in the boost documentation that it clearly states assignment is not thread safe. I agree and disagree. I think in my case I disagree. I only need the handshake between threadA and threadB. I don't think some of the problems described in some of the replies apply here (although they were eye opening replies that I did not completely think through).
Example ThreadA atach(SharedObject); //Shared object passed by value, count incremented etc etc.
ThreadB //Accepts the object, adds it to a list of shared objects. ThreadB is on a timer that notifies all SharedObjects of an event. Before Notifying, a copy of the list is made protected by a critical section. The CS is released, the copies are notified.
ThreadA detach(SharedObject); //Removes the shared object from the list of objects
Now, concurrently ThreadB is singaling the SharedOjbect and already made a copy of the list before ThreadA detached said shared object. Everything is ok no?
Technically, it should be safe.
Why? Because in order to copy an object, the source needs to have a "reference", so it's not going away during the copy. Also, nobody is accessing an object that's currently under construction.
Destructor is safe too, since there are no references left anyway.
You may want to reconsider copying the reference count, though. Those references don't actually exist; everyone with a reference to the original would somehow have to decrease the refcount for the copy too, provided it got the original reference before the copy was made. Copies should start like new objects, with a refcount of 1.
EDIT: likewise, if you're implementing an assignment operator (which is like a copy into an existing object), the refcount of the destination object should be left as it is.