Причина для C++ IntAtomicGet, GotW

В GotW статье № 45 Херб утверждает следующее:

void String::AboutToModify(
  size_t n,
  bool   bMarkUnshareable /* = false */
) {
  if( data_->refs > 1 && data_->refs != Unshareable ) {
    /* ... etc. ... */

Это условие if не является потокобезопасным. С одной стороны, оценка даже "data_->refs > 1" может быть не атомарной; если это так, возможно, что если поток 1 попытается вычислить "data_->refs > 1", когда поток 2 обновляет значение refs, значение, считанное из data _-> refs, может быть любым - 1, 2 или даже чем-то другим. это ни оригинальное значение, ни новое значение.

Кроме того, он указывает, что данные _-> ссылки могут быть изменены между сравнением с 1 и сравнением с Unshareable.

Далее мы находим решение:

void String::AboutToModify(
  size_t n,
  bool   bMarkUnshareable /* = false */
) {
  int refs = IntAtomicGet( data_->refs );
  if( refs > 1 && refs != Unshareable ) {
    /* ... etc. ...*/

Теперь я понимаю, что одни и те же ссылки используются для обоих сравнений, для решения задачи 2. Но почему IntAtomicGet? Я ничего не нашел в поисках по этой теме - все атомарные операции сосредоточены на операциях чтения, изменения, записи, и здесь мы только что прочитали. Так что мы можем просто сделать...

int refs = data_->refs;

... которая, вероятно, должна быть всего лишь одной инструкцией в конце концов?

2 ответа

Разные платформы дают разные обещания об атомарности операций чтения / записи. x86 например гарантирует, что чтение двойного слова (4 bytes) будет атомная операция. Однако вы не можете предполагать, что это будет верно для любой архитектуры и, вероятно, не будет.

Если вы планируете переносить код на разные платформы, такие предположения могут создать проблемы и привести к странным условиям гонки в вашем коде. Поэтому лучше защитить себя и сделать операции чтения / записи явно атомарными.

Чтение из общей памяти (data_->refs) в то время как другой поток пишет в него, это определение гонки данных.

Что происходит, когда мы не атомно читаем из data_->refs в то время как другой поток пытается написать в это же время?

Представь, что нить А делает ++data_->refs (напишите), пока поток B делает int x = data_->refs (читать). Представьте, что поток B читает первые несколько байтов из data_->refs и этот поток А заканчивает записывать свое значение в data_->refs прежде чем поток B закончил чтение. Затем поток B читает остальные байты в data_->refs,

Вы не получите ни исходного значения, ни нового значения; Вы получите совершенно другое значение! Этот сценарий просто иллюстрирует, что подразумевается под:

[...] значение, считываемое из данных _->refs, может быть любым - 1, 2 или даже чем-то, что не является ни исходным, ни новым значением.

Цель атомарных операций состоит в том, чтобы гарантировать, что операция неделима: она либо выполняется как выполненная, либо не выполняется. Поэтому мы используем атомарную операцию чтения, чтобы гарантировать, что мы получим значение data_->refs либо до его обновления, либо после (это зависит от времени потоков).

Другие вопросы по тегам