blob: 061ed448c33f48f97db15ca9bc912f1619fe3ae2 [file] [log] [blame]
// Copyright (C) 2020-2021 Free Software Foundation, Inc.
//
// This file is part of the GNU ISO C++ Library. This library is free
// software; you can redistribute it and/or modify it under the
// terms of the GNU General Public License as published by the
// Free Software Foundation; either version 3, or (at your option)
// any later version.
// This library is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
// You should have received a copy of the GNU General Public License along
// with this library; see the file COPYING3. If not see
// <http://www.gnu.org/licenses/>.
// { dg-options "-std=gnu++2a -pthread" }
// { dg-add-options libatomic }
// { dg-require-effective-target c++2a }
// { dg-require-effective-target pthread }
// { dg-require-gthreads "" }
#include <stop_token>
#include <atomic>
#include <thread>
#include <testsuite_hooks.h>
struct F
{
static std::atomic<int> stage;
F(int) { }
~F()
{
// PR libstdc++/92895
// Callback function must not be destroyed while still executing.
VERIFY( stage == 4 );
}
void operator()() const noexcept
{
stage = 2; // trigger destructor of stop_callback that owns *this
while (stage == 2)
std::this_thread::sleep_for(std::chrono::milliseconds(10));
std::this_thread::sleep_for(std::chrono::milliseconds(500));
stage = 4; // destructor checks for this
}
};
std::atomic<int> F::stage{0};
void
test01()
{
std::stop_source ssrc;
std::stop_token stok = ssrc.get_token();
std::thread t1([&ssrc] {
while (F::stage == 0)
std::this_thread::sleep_for(std::chrono::milliseconds(10));
ssrc.request_stop();
while (F::stage != 5)
std::this_thread::sleep_for(std::chrono::milliseconds(10));
});
std::thread t2([&ssrc] {
std::stop_callback<F> cb(ssrc.get_token(), 0);
F::stage = 1; // trigger stop request in other thread, which runs callback
while (F::stage == 1)
std::this_thread::sleep_for(std::chrono::milliseconds(10));
F::stage = 3;
// stop_callback destructor should block until callback completes
});
t2.join();
F::stage = 5; // allow first thread to exit
t1.join();
}
int main()
{
test01();
}