I’ve been reading Joe Duffy’s book on Concurrent programming. I have kind of an academic question about lockless threading.
First: I know that lockless threading is fraught with peril (if you don’t believe me, read the sections in the book about memory model)
Nevertheless, I have a question:
suppose I have an class with an int property on it.
The value referenced by this property will be read very frequently by multiple threads
It is extremely rare that the value will change, and when it does it will be a single thread that changes it.
If it does change while another operation that uses it is in flight, no one is going to lose a finger (the first thing anyone using it does is copy it to a local variable)
I could use locks (or a readerwriterlockslim to keep the reads concurrent).
I could mark the variable volatile (lots of examples where this is done)
However, even volatile can impose a performance hit.
What if I use VolatileWrite when it changes, and leave the access normal for reads. Something like this:
public class MyClass
{
private int _TheProperty;
internal int TheProperty
{
get { return _TheProperty; }
set { System.Threading.Thread.VolatileWrite(ref _TheProperty, value); }
}
}
I don’t think that I would ever try this in real life, but I’m curious about the answer (more than anything, as a checkpoint of whether I understand the memory model stuff I’ve been reading).
Marking a variable as “volatile” has two effects.
1) Reads and writes have acquire and release semantics, so that reads and writes of other memory locations will not “move forwards and backwards in time” with respect to reads and writes of this memory location. (This is a simplification, but you take my point.)
2) The code generated by the jitter will not “cache” a value that seems to logically be unchanging.
Whether the former point is relevant in your scenario, I don’t know; you’ve only described one memory location. Whether or not it is important that you have only volatile writes but not volatile reads is something that is up to you to decide.
But it seems to me that the latter point is quite relevant. If you have a spin lock on a non-volatile variable:
the jitter is within its rights to generate this code as though you’d written
Whether it actually does so or not, I don’t know, but it has the right to. If what you want is for the code to actually re-check the property on each go round the loop, marking it as volatile is the right way to go.