(root)/
gcc-13.2.0/
libstdc++-v3/
testsuite/
30_threads/
stop_token/
stop_callback/
destroy.cc
// Copyright (C) 2020-2023 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();
}