c++multithreadingatomicvolatilestdatomic

Does atomic read guarantee reading of the latest value?


In C++ we have a volatile keyword and an atomic class. Difference between them is that volatile does not guarantee thread-safe concurrent reading and writing, but ensures that compiler will not store variable's value in cache and instead will load it directly from the memory, while atomic guarantees thread-safe concurrent reading and writing.

As we know, an atomic read operation is indivisible, i.e. neither thread can write a new value to the variable while one or more threads are reading that variable's value. That makes me think that we always read the latest value, but I'm not sure :)

So, my question is: if we declare a variable as atomic, do we always get the latest value of the variable when calling load() operation?


Solution

  • When we talk about memory access on modern architectures, we usually ignore the "exact location" the value is read from.

    A read operation can fetch data from the cache (L0/L1/...), the RAM or even the hard-drive (e.g. when the memory is swapped).

    These keywords tell the compiler which assembly operations to use when accessing the data.

    volatile

    A keyword that tells the compiler to always read the variable's value from memory, and never from the register.

    This "memory" can still be the cache, but, in case that this "address" in the cache is considered "dirty", meaning that the value has changed by a different processor, the value will be reloaded.

    This ensures we never read a stale value.

    Clarification: According to the standard, if the volatile type is not a primitive, whose read/write operations are atomic (in regard to the assembly instructions that read/write it) by nature, one might possibly read an intermediate value (the writer managed to write only half of the bytes by the time the reader read it). However, modern implementations do not behave this way.

    atomic

    When the compiler sees a load (read) operation, it basically does the exact same thing it would have done for a volatile value.

    So, what is the difference???

    The difference is cross-CPU write operations. When working with a volatile variable, if CPU 1 sets the value, and CPU 2 reads it, the reader might read an old value.

    But, how can that be? The volatile keyword promises that we won't read a stale value!

    Well, that's because the writer didn't publish the value! And though the reader tries to read it, it reads the old one.

    When the compiler stumbles upon a store (write) operation for an atomic variable it:

    After the announcement, all the CPUs will know that they should re-read the value of the variable because their caches will be marked "dirty".

    This mechanism is very similar to operations performed on files. When your application writes to a file on the hard-drive, other applications may or may not see the new information, depending on whether or not your application flushed the data to the hard-drive.

    If the data wasn't flushed, then it merely resides somewhere in your application's caches and visible only itself. Once you flush it, anyone who opens the file will see the new state.

    Clarification: Common modern compiler & cache implementations ensure correct publishing of volatile writes as well. However, this is NOT a reason to prefer that over std::atomic. For example, just like some comments pointed out, Linux's atomic read and writes for x86_64 are implemented using volatiles.