Понимание memory_order_relaxed

Я пытаюсь понять специфику memory_order_relaxed. Я имею в виду эту ссылку: ссылка CPP.

#include <future>
#include <atomic>

std::atomic<int*> ptr {nullptr};

void fun1(){
        ptr.store(new int{0}, std::memory_order_relaxed);
}

void fun2(){
        while(!ptr.load(std::memory_order_relaxed));
}

int main(){
        std::async(std::launch::async, fun1);
        std::async(std::launch::async, fun2);
}

Вопрос 1: В приведенном выше коде технически возможно, чтобы fun2 находился в бесконечном цикле, где он видит значение ptr как nullptr, даже если поток, который устанавливает ptr, завершил работу?

Если предположить, я изменяю код выше на что-то вроде этого:

#include <future>
#include <atomic>

std::atomic<int> i {0};
std::atomic<int*> ptr {nullptr};

void fun1(){
        i.store(1, std::memory_order_relaxed);
        i.store(2, std::memory_order_relaxed);
        ptr.store(new int{0}, std::memory_order_release);

}

void fun2(){
        while(!ptr.load(std::memory_order_acquire));
        int x = i.load(std::memory_order_relaxed);
}

int main(){
        std::async(std::launch::async, fun1);
        std::async(std::launch::async, fun2);
}

Смежный вопрос: возможно ли в коде выше для fun2 увидеть значение атомарного i как 1 или оно гарантированно увидит значение 2?

1 ответ

Интересное наблюдение состоит в том, что в вашем коде нет фактического параллелизма; т.е. fun1 а также fun2 работать последовательно, причина в том, что при определенных условиях (включая вызов std::async с std::launch::async политика запуска), std::future объект, возвращенный std::async имеет свой блок деструктора, пока не будет запущен вызов функции. Поскольку вы игнорируете возвращаемый объект, его деструктор вызывается до конца инструкции. Если бы вы поменяли два утверждения в main() (т.е. запустить fun2 до fun1), ваша программа была бы поймана в бесконечном цикле, так как fun1 никогда бы не побежал.

это std::future поведение с ожиданием уничтожения несколько противоречиво (даже в рамках комитета по стандартам), и, поскольку я предполагаю, что вы не имели в виду это, я позволю себе переписать 2 утверждения в main для (оба примера):

auto tmp1 = std::async(std::launch::async, fun1);
auto tmp2 = std::async(std::launch::async, fun2);

Это откладывает фактическое std::future вернуть уничтожение объекта до конца main чтобы fun1 а также fun2 получить работать асинхронно.

технически возможно для fun2 находиться в бесконечном цикле, где он видит значение ptr как nullptr, даже если поток, который устанавливает ptr, завершил работу?

Нет, это невозможно с std::atomic (на реальной платформе, как уже упоминалось в разделе комментариев). С неstd::atomic переменной, компилятор мог (теоретически) выбрать сохранение значения только в регистре, но std::atomic сохраняется и когерентность кеша распространит значение в другие потоки. С помощью std::memory_order_relaxed здесь хорошо, если вы не разыменовываете указатель.

Возможно ли в коде выше для fun2 увидеть значение атомарного i как 1, или он уверен, что увидит значение 2?

Гарантируется увидеть значение 2 в переменной x,
fun1 хранит два разных значения в одной и той же переменной, но поскольку существует четкая зависимость, они не переупорядочиваются.

В fun1, ptr.store с std::memory_order_release предотвращает i.store(2) с std::memory_order_relaxed от движения ниже его барьера выпуска. В fun2, ptr.load с std::memory_order_acquire предотвращает i.load с std::memory_order_relaxed от продвижения через свой барьер приобретения. Это гарантирует, что x в fun2 будет иметь значение 2.

Обратите внимание, что с помощью std::memory_order_relaxed по всем атомикам можно было бы увидеть x со значением 0, 1 или 2, в зависимости от относительного порядка доступа к атомарной переменной i в отношении ptr.store а также ptr.load,

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