2010-05-12 03:42:16 +08:00
|
|
|
// -*- C++ -*-
|
2021-11-18 05:25:01 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2010-05-12 03:42:16 +08:00
|
|
|
//
|
2019-01-19 18:56:40 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2010-05-12 03:42:16 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#ifndef _LIBCPP_FUTURE
|
|
|
|
#define _LIBCPP_FUTURE
|
|
|
|
|
|
|
|
/*
|
|
|
|
future synopsis
|
|
|
|
|
|
|
|
namespace std
|
|
|
|
{
|
|
|
|
|
|
|
|
enum class future_errc
|
|
|
|
{
|
2013-09-15 02:20:10 +08:00
|
|
|
future_already_retrieved = 1,
|
2010-05-12 03:42:16 +08:00
|
|
|
promise_already_satisfied,
|
2013-09-15 02:20:10 +08:00
|
|
|
no_state,
|
|
|
|
broken_promise
|
2010-05-12 03:42:16 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
enum class launch
|
|
|
|
{
|
2010-11-24 02:33:54 +08:00
|
|
|
async = 1,
|
|
|
|
deferred = 2,
|
|
|
|
any = async | deferred
|
2010-05-12 03:42:16 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
enum class future_status
|
|
|
|
{
|
|
|
|
ready,
|
|
|
|
timeout,
|
|
|
|
deferred
|
|
|
|
};
|
|
|
|
|
|
|
|
template <> struct is_error_code_enum<future_errc> : public true_type { };
|
2012-07-22 01:46:55 +08:00
|
|
|
error_code make_error_code(future_errc e) noexcept;
|
|
|
|
error_condition make_error_condition(future_errc e) noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
|
2012-07-22 01:46:55 +08:00
|
|
|
const error_category& future_category() noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
|
|
|
|
class future_error
|
|
|
|
: public logic_error
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
future_error(error_code ec); // exposition only
|
2016-11-15 02:56:24 +08:00
|
|
|
explicit future_error(future_errc); // C++17
|
2012-07-22 01:46:55 +08:00
|
|
|
const error_code& code() const noexcept;
|
|
|
|
const char* what() const noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
template <class R>
|
|
|
|
class promise
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
promise();
|
|
|
|
template <class Allocator>
|
|
|
|
promise(allocator_arg_t, const Allocator& a);
|
2012-07-22 01:46:55 +08:00
|
|
|
promise(promise&& rhs) noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
promise(const promise& rhs) = delete;
|
|
|
|
~promise();
|
|
|
|
|
|
|
|
// assignment
|
2012-07-22 01:46:55 +08:00
|
|
|
promise& operator=(promise&& rhs) noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
promise& operator=(const promise& rhs) = delete;
|
2012-07-22 01:46:55 +08:00
|
|
|
void swap(promise& other) noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
|
|
|
|
// retrieving the result
|
|
|
|
future<R> get_future();
|
|
|
|
|
|
|
|
// setting the result
|
|
|
|
void set_value(const R& r);
|
|
|
|
void set_value(R&& r);
|
|
|
|
void set_exception(exception_ptr p);
|
|
|
|
|
|
|
|
// setting the result with deferred notification
|
|
|
|
void set_value_at_thread_exit(const R& r);
|
|
|
|
void set_value_at_thread_exit(R&& r);
|
|
|
|
void set_exception_at_thread_exit(exception_ptr p);
|
|
|
|
};
|
|
|
|
|
|
|
|
template <class R>
|
|
|
|
class promise<R&>
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
promise();
|
|
|
|
template <class Allocator>
|
|
|
|
promise(allocator_arg_t, const Allocator& a);
|
2012-07-22 01:46:55 +08:00
|
|
|
promise(promise&& rhs) noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
promise(const promise& rhs) = delete;
|
|
|
|
~promise();
|
|
|
|
|
|
|
|
// assignment
|
2012-07-22 01:46:55 +08:00
|
|
|
promise& operator=(promise&& rhs) noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
promise& operator=(const promise& rhs) = delete;
|
2012-07-22 01:46:55 +08:00
|
|
|
void swap(promise& other) noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
|
|
|
|
// retrieving the result
|
2010-08-28 04:10:19 +08:00
|
|
|
future<R&> get_future();
|
2010-05-12 03:42:16 +08:00
|
|
|
|
|
|
|
// setting the result
|
|
|
|
void set_value(R& r);
|
|
|
|
void set_exception(exception_ptr p);
|
|
|
|
|
|
|
|
// setting the result with deferred notification
|
|
|
|
void set_value_at_thread_exit(R&);
|
|
|
|
void set_exception_at_thread_exit(exception_ptr p);
|
|
|
|
};
|
|
|
|
|
|
|
|
template <>
|
|
|
|
class promise<void>
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
promise();
|
|
|
|
template <class Allocator>
|
|
|
|
promise(allocator_arg_t, const Allocator& a);
|
2012-07-22 01:46:55 +08:00
|
|
|
promise(promise&& rhs) noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
promise(const promise& rhs) = delete;
|
|
|
|
~promise();
|
|
|
|
|
|
|
|
// assignment
|
2012-07-22 01:46:55 +08:00
|
|
|
promise& operator=(promise&& rhs) noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
promise& operator=(const promise& rhs) = delete;
|
2012-07-22 01:46:55 +08:00
|
|
|
void swap(promise& other) noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
|
|
|
|
// retrieving the result
|
2010-08-28 04:10:19 +08:00
|
|
|
future<void> get_future();
|
2010-05-12 03:42:16 +08:00
|
|
|
|
|
|
|
// setting the result
|
|
|
|
void set_value();
|
|
|
|
void set_exception(exception_ptr p);
|
|
|
|
|
|
|
|
// setting the result with deferred notification
|
|
|
|
void set_value_at_thread_exit();
|
|
|
|
void set_exception_at_thread_exit(exception_ptr p);
|
|
|
|
};
|
|
|
|
|
2012-07-22 01:46:55 +08:00
|
|
|
template <class R> void swap(promise<R>& x, promise<R>& y) noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
|
|
|
|
template <class R, class Alloc>
|
|
|
|
struct uses_allocator<promise<R>, Alloc> : public true_type {};
|
|
|
|
|
|
|
|
template <class R>
|
|
|
|
class future
|
|
|
|
{
|
|
|
|
public:
|
2012-07-22 01:46:55 +08:00
|
|
|
future() noexcept;
|
|
|
|
future(future&&) noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
future(const future& rhs) = delete;
|
|
|
|
~future();
|
|
|
|
future& operator=(const future& rhs) = delete;
|
2012-07-22 01:46:55 +08:00
|
|
|
future& operator=(future&&) noexcept;
|
2017-01-26 04:14:03 +08:00
|
|
|
shared_future<R> share() noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
|
|
|
|
// retrieving the value
|
|
|
|
R get();
|
|
|
|
|
|
|
|
// functions to check state
|
2012-07-22 01:46:55 +08:00
|
|
|
bool valid() const noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
|
|
|
|
void wait() const;
|
|
|
|
template <class Rep, class Period>
|
|
|
|
future_status
|
|
|
|
wait_for(const chrono::duration<Rep, Period>& rel_time) const;
|
|
|
|
template <class Clock, class Duration>
|
|
|
|
future_status
|
|
|
|
wait_until(const chrono::time_point<Clock, Duration>& abs_time) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <class R>
|
|
|
|
class future<R&>
|
|
|
|
{
|
|
|
|
public:
|
2012-07-22 01:46:55 +08:00
|
|
|
future() noexcept;
|
|
|
|
future(future&&) noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
future(const future& rhs) = delete;
|
|
|
|
~future();
|
|
|
|
future& operator=(const future& rhs) = delete;
|
2012-07-22 01:46:55 +08:00
|
|
|
future& operator=(future&&) noexcept;
|
2017-01-25 07:28:25 +08:00
|
|
|
shared_future<R&> share() noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
|
|
|
|
// retrieving the value
|
|
|
|
R& get();
|
|
|
|
|
|
|
|
// functions to check state
|
2012-07-22 01:46:55 +08:00
|
|
|
bool valid() const noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
|
|
|
|
void wait() const;
|
|
|
|
template <class Rep, class Period>
|
|
|
|
future_status
|
|
|
|
wait_for(const chrono::duration<Rep, Period>& rel_time) const;
|
|
|
|
template <class Clock, class Duration>
|
|
|
|
future_status
|
|
|
|
wait_until(const chrono::time_point<Clock, Duration>& abs_time) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <>
|
|
|
|
class future<void>
|
|
|
|
{
|
|
|
|
public:
|
2012-07-22 01:46:55 +08:00
|
|
|
future() noexcept;
|
|
|
|
future(future&&) noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
future(const future& rhs) = delete;
|
|
|
|
~future();
|
|
|
|
future& operator=(const future& rhs) = delete;
|
2012-07-22 01:46:55 +08:00
|
|
|
future& operator=(future&&) noexcept;
|
2017-01-25 07:28:25 +08:00
|
|
|
shared_future<void> share() noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
|
|
|
|
// retrieving the value
|
|
|
|
void get();
|
|
|
|
|
|
|
|
// functions to check state
|
2012-07-22 01:46:55 +08:00
|
|
|
bool valid() const noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
|
|
|
|
void wait() const;
|
|
|
|
template <class Rep, class Period>
|
|
|
|
future_status
|
|
|
|
wait_for(const chrono::duration<Rep, Period>& rel_time) const;
|
|
|
|
template <class Clock, class Duration>
|
|
|
|
future_status
|
|
|
|
wait_until(const chrono::time_point<Clock, Duration>& abs_time) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <class R>
|
|
|
|
class shared_future
|
|
|
|
{
|
|
|
|
public:
|
2012-07-22 01:46:55 +08:00
|
|
|
shared_future() noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
shared_future(const shared_future& rhs);
|
2012-07-22 01:46:55 +08:00
|
|
|
shared_future(future<R>&&) noexcept;
|
|
|
|
shared_future(shared_future&& rhs) noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
~shared_future();
|
|
|
|
shared_future& operator=(const shared_future& rhs);
|
2012-07-22 01:46:55 +08:00
|
|
|
shared_future& operator=(shared_future&& rhs) noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
|
|
|
|
// retrieving the value
|
|
|
|
const R& get() const;
|
|
|
|
|
|
|
|
// functions to check state
|
2012-07-22 01:46:55 +08:00
|
|
|
bool valid() const noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
|
|
|
|
void wait() const;
|
|
|
|
template <class Rep, class Period>
|
|
|
|
future_status
|
|
|
|
wait_for(const chrono::duration<Rep, Period>& rel_time) const;
|
|
|
|
template <class Clock, class Duration>
|
|
|
|
future_status
|
|
|
|
wait_until(const chrono::time_point<Clock, Duration>& abs_time) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <class R>
|
|
|
|
class shared_future<R&>
|
|
|
|
{
|
|
|
|
public:
|
2012-07-22 01:46:55 +08:00
|
|
|
shared_future() noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
shared_future(const shared_future& rhs);
|
2012-07-22 01:46:55 +08:00
|
|
|
shared_future(future<R&>&&) noexcept;
|
|
|
|
shared_future(shared_future&& rhs) noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
~shared_future();
|
|
|
|
shared_future& operator=(const shared_future& rhs);
|
2012-07-22 01:46:55 +08:00
|
|
|
shared_future& operator=(shared_future&& rhs) noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
|
|
|
|
// retrieving the value
|
|
|
|
R& get() const;
|
|
|
|
|
|
|
|
// functions to check state
|
2012-07-22 01:46:55 +08:00
|
|
|
bool valid() const noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
|
|
|
|
void wait() const;
|
|
|
|
template <class Rep, class Period>
|
|
|
|
future_status
|
|
|
|
wait_for(const chrono::duration<Rep, Period>& rel_time) const;
|
|
|
|
template <class Clock, class Duration>
|
|
|
|
future_status
|
|
|
|
wait_until(const chrono::time_point<Clock, Duration>& abs_time) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <>
|
|
|
|
class shared_future<void>
|
|
|
|
{
|
|
|
|
public:
|
2012-07-22 01:46:55 +08:00
|
|
|
shared_future() noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
shared_future(const shared_future& rhs);
|
2012-07-22 01:46:55 +08:00
|
|
|
shared_future(future<void>&&) noexcept;
|
|
|
|
shared_future(shared_future&& rhs) noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
~shared_future();
|
|
|
|
shared_future& operator=(const shared_future& rhs);
|
2012-07-22 01:46:55 +08:00
|
|
|
shared_future& operator=(shared_future&& rhs) noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
|
|
|
|
// retrieving the value
|
|
|
|
void get() const;
|
|
|
|
|
|
|
|
// functions to check state
|
2012-07-22 01:46:55 +08:00
|
|
|
bool valid() const noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
|
|
|
|
void wait() const;
|
|
|
|
template <class Rep, class Period>
|
|
|
|
future_status
|
|
|
|
wait_for(const chrono::duration<Rep, Period>& rel_time) const;
|
|
|
|
template <class Clock, class Duration>
|
|
|
|
future_status
|
|
|
|
wait_until(const chrono::time_point<Clock, Duration>& abs_time) const;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <class F, class... Args>
|
2013-09-22 02:17:23 +08:00
|
|
|
future<typename result_of<typename decay<F>::type(typename decay<Args>::type...)>::type>
|
2010-05-12 03:42:16 +08:00
|
|
|
async(F&& f, Args&&... args);
|
|
|
|
|
|
|
|
template <class F, class... Args>
|
2013-09-22 02:17:23 +08:00
|
|
|
future<typename result_of<typename decay<F>::type(typename decay<Args>::type...)>::type>
|
2010-05-12 03:42:16 +08:00
|
|
|
async(launch policy, F&& f, Args&&... args);
|
|
|
|
|
|
|
|
template <class> class packaged_task; // undefined
|
|
|
|
|
|
|
|
template <class R, class... ArgTypes>
|
|
|
|
class packaged_task<R(ArgTypes...)>
|
|
|
|
{
|
|
|
|
public:
|
2016-06-02 05:05:53 +08:00
|
|
|
typedef R result_type; // extension
|
2010-05-12 03:42:16 +08:00
|
|
|
|
|
|
|
// construction and destruction
|
2012-07-22 01:46:55 +08:00
|
|
|
packaged_task() noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
template <class F>
|
|
|
|
explicit packaged_task(F&& f);
|
2017-11-28 04:47:54 +08:00
|
|
|
template <class F, class Allocator>
|
|
|
|
packaged_task(allocator_arg_t, const Allocator& a, F&& f);
|
2010-05-12 03:42:16 +08:00
|
|
|
~packaged_task();
|
|
|
|
|
|
|
|
// no copy
|
2012-07-22 03:34:12 +08:00
|
|
|
packaged_task(const packaged_task&) = delete;
|
|
|
|
packaged_task& operator=(const packaged_task&) = delete;
|
2010-05-12 03:42:16 +08:00
|
|
|
|
|
|
|
// move support
|
2012-07-22 01:46:55 +08:00
|
|
|
packaged_task(packaged_task&& other) noexcept;
|
|
|
|
packaged_task& operator=(packaged_task&& other) noexcept;
|
|
|
|
void swap(packaged_task& other) noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
|
2012-07-22 01:46:55 +08:00
|
|
|
bool valid() const noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
|
|
|
|
// result retrieval
|
|
|
|
future<R> get_future();
|
|
|
|
|
|
|
|
// execution
|
|
|
|
void operator()(ArgTypes... );
|
|
|
|
void make_ready_at_thread_exit(ArgTypes...);
|
|
|
|
|
|
|
|
void reset();
|
|
|
|
};
|
|
|
|
|
|
|
|
template <class R>
|
2012-07-22 01:46:55 +08:00
|
|
|
void swap(packaged_task<R(ArgTypes...)&, packaged_task<R(ArgTypes...)>&) noexcept;
|
2010-05-12 03:42:16 +08:00
|
|
|
|
2017-11-28 04:47:54 +08:00
|
|
|
template <class R, class Alloc> struct uses_allocator<packaged_task<R>, Alloc>;
|
|
|
|
|
2010-05-12 03:42:16 +08:00
|
|
|
} // std
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
2022-03-26 00:55:36 +08:00
|
|
|
#include <__assert> // all public C++ headers provide the assertion handler
|
2020-11-05 04:01:25 +08:00
|
|
|
#include <__availability>
|
2022-02-24 06:05:22 +08:00
|
|
|
#include <__chrono/duration.h>
|
|
|
|
#include <__chrono/time_point.h>
|
2021-05-19 23:57:04 +08:00
|
|
|
#include <__config>
|
2021-07-01 21:25:35 +08:00
|
|
|
#include <__memory/allocator_arg_t.h>
|
|
|
|
#include <__memory/uses_allocator.h>
|
2021-12-14 08:21:38 +08:00
|
|
|
#include <__utility/auto_cast.h>
|
2021-06-05 10:47:47 +08:00
|
|
|
#include <__utility/forward.h>
|
2022-03-06 02:17:07 +08:00
|
|
|
#include <__utility/move.h>
|
2010-08-28 04:10:19 +08:00
|
|
|
#include <exception>
|
2021-05-13 11:04:03 +08:00
|
|
|
#include <memory>
|
2010-09-04 05:46:37 +08:00
|
|
|
#include <mutex>
|
2021-05-13 11:04:03 +08:00
|
|
|
#include <system_error>
|
2010-08-28 04:10:19 +08:00
|
|
|
#include <thread>
|
2021-12-23 01:14:14 +08:00
|
|
|
#include <version>
|
2010-05-12 03:42:16 +08:00
|
|
|
|
[libc++] Re-add transitive includes that had been removed since LLVM 14
This commit re-adds transitive includes that had been removed by
4cd04d1687f1, c36870c8e79c, a83f4b9cda57, 1458458b558d, 2e2f3158c604,
and 489637e66dd3. This should cover almost all the includes that had
been removed since LLVM 14 and that would contribute to breaking user
code when releasing LLVM 15.
It is possible to disable the inclusion of these headers by defining
_LIBCPP_REMOVE_TRANSITIVE_INCLUDES. The intent is that vendors will
enable that macro and start fixing downstream issues immediately. We
can then remove the macro (and the transitive includes) by default in
a future release. That way, we will break users only once by removing
transitive includes in bulk instead of doing it bit by bit a every
release, which is more disruptive for users.
Note 1: The set of headers to re-add was found by re-generating the
transitive include test on a checkout of release/14.x, which
provided the list of all transitive includes we used to provide.
Note 2: Several includes of <vector>, <optional>, <array> and <unordered_map>
have been added in this commit. These transitive inclusions were
added when we implemented boyer_moore_searcher in <functional>.
Note 3: This is a best effort patch to try and resolve downstream breakage
caused since branching LLVM 14. I wasn't able to perfectly mirror
transitive includes in LLVM 14 for a few headers, so I added a
release note explaining it. To summarize, adding boyer_moore_searcher
created a bunch of circular dependencies, so we have to break
backwards compatibility in a few cases.
Differential Revision: https://reviews.llvm.org/D128661
2022-06-28 03:53:41 +08:00
|
|
|
#ifndef _LIBCPP_REMOVE_TRANSITIVE_INCLUDES
|
|
|
|
# include <chrono>
|
|
|
|
#endif
|
|
|
|
|
2011-10-18 04:05:10 +08:00
|
|
|
#if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER)
|
2022-02-02 09:16:40 +08:00
|
|
|
# pragma GCC system_header
|
2011-10-18 04:05:10 +08:00
|
|
|
#endif
|
2010-05-12 03:42:16 +08:00
|
|
|
|
2014-09-06 04:28:44 +08:00
|
|
|
#ifdef _LIBCPP_HAS_NO_THREADS
|
2022-05-19 01:17:14 +08:00
|
|
|
# error "<future> is not supported since libc++ has been configured without support for threads."
|
|
|
|
#endif
|
2014-09-06 03:45:05 +08:00
|
|
|
|
2010-05-12 03:42:16 +08:00
|
|
|
_LIBCPP_BEGIN_NAMESPACE_STD
|
|
|
|
|
|
|
|
//enum class future_errc
|
2011-12-03 03:36:40 +08:00
|
|
|
_LIBCPP_DECLARE_STRONG_ENUM(future_errc)
|
2010-05-12 03:42:16 +08:00
|
|
|
{
|
2013-09-15 02:20:10 +08:00
|
|
|
future_already_retrieved = 1,
|
2010-05-12 03:42:16 +08:00
|
|
|
promise_already_satisfied,
|
2013-09-15 02:20:10 +08:00
|
|
|
no_state,
|
|
|
|
broken_promise
|
2010-05-12 03:42:16 +08:00
|
|
|
};
|
2011-12-03 03:36:40 +08:00
|
|
|
_LIBCPP_DECLARE_STRONG_ENUM_EPILOG(future_errc)
|
2010-05-12 03:42:16 +08:00
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
template <>
|
2017-01-05 07:56:00 +08:00
|
|
|
struct _LIBCPP_TEMPLATE_VIS is_error_code_enum<future_errc> : public true_type {};
|
2010-08-26 01:32:05 +08:00
|
|
|
|
2022-02-12 02:34:33 +08:00
|
|
|
#ifdef _LIBCPP_CXX03_LANG
|
2011-12-03 03:36:40 +08:00
|
|
|
template <>
|
2017-01-05 07:56:00 +08:00
|
|
|
struct _LIBCPP_TEMPLATE_VIS is_error_code_enum<future_errc::__lx> : public true_type { };
|
2011-12-03 03:36:40 +08:00
|
|
|
#endif
|
|
|
|
|
2010-05-12 03:42:16 +08:00
|
|
|
//enum class launch
|
2011-12-03 03:36:40 +08:00
|
|
|
_LIBCPP_DECLARE_STRONG_ENUM(launch)
|
2010-05-12 03:42:16 +08:00
|
|
|
{
|
2010-11-24 02:33:54 +08:00
|
|
|
async = 1,
|
|
|
|
deferred = 2,
|
|
|
|
any = async | deferred
|
2010-05-12 03:42:16 +08:00
|
|
|
};
|
2011-12-03 03:36:40 +08:00
|
|
|
_LIBCPP_DECLARE_STRONG_ENUM_EPILOG(launch)
|
2010-05-12 03:42:16 +08:00
|
|
|
|
2022-02-12 02:34:33 +08:00
|
|
|
#ifndef _LIBCPP_CXX03_LANG
|
2013-06-30 02:38:17 +08:00
|
|
|
|
|
|
|
typedef underlying_type<launch>::type __launch_underlying_type;
|
|
|
|
|
|
|
|
inline _LIBCPP_INLINE_VISIBILITY
|
|
|
|
_LIBCPP_CONSTEXPR
|
|
|
|
launch
|
|
|
|
operator&(launch __x, launch __y)
|
|
|
|
{
|
|
|
|
return static_cast<launch>(static_cast<__launch_underlying_type>(__x) &
|
|
|
|
static_cast<__launch_underlying_type>(__y));
|
|
|
|
}
|
|
|
|
|
|
|
|
inline _LIBCPP_INLINE_VISIBILITY
|
|
|
|
_LIBCPP_CONSTEXPR
|
|
|
|
launch
|
|
|
|
operator|(launch __x, launch __y)
|
|
|
|
{
|
|
|
|
return static_cast<launch>(static_cast<__launch_underlying_type>(__x) |
|
|
|
|
static_cast<__launch_underlying_type>(__y));
|
|
|
|
}
|
|
|
|
|
|
|
|
inline _LIBCPP_INLINE_VISIBILITY
|
|
|
|
_LIBCPP_CONSTEXPR
|
|
|
|
launch
|
|
|
|
operator^(launch __x, launch __y)
|
|
|
|
{
|
|
|
|
return static_cast<launch>(static_cast<__launch_underlying_type>(__x) ^
|
|
|
|
static_cast<__launch_underlying_type>(__y));
|
|
|
|
}
|
|
|
|
|
|
|
|
inline _LIBCPP_INLINE_VISIBILITY
|
|
|
|
_LIBCPP_CONSTEXPR
|
|
|
|
launch
|
|
|
|
operator~(launch __x)
|
|
|
|
{
|
2013-07-03 02:01:41 +08:00
|
|
|
return static_cast<launch>(~static_cast<__launch_underlying_type>(__x) & 3);
|
2013-06-30 02:38:17 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
inline _LIBCPP_INLINE_VISIBILITY
|
|
|
|
launch&
|
|
|
|
operator&=(launch& __x, launch __y)
|
|
|
|
{
|
|
|
|
__x = __x & __y; return __x;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline _LIBCPP_INLINE_VISIBILITY
|
|
|
|
launch&
|
|
|
|
operator|=(launch& __x, launch __y)
|
|
|
|
{
|
|
|
|
__x = __x | __y; return __x;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline _LIBCPP_INLINE_VISIBILITY
|
|
|
|
launch&
|
|
|
|
operator^=(launch& __x, launch __y)
|
|
|
|
{
|
|
|
|
__x = __x ^ __y; return __x;
|
|
|
|
}
|
|
|
|
|
2022-02-12 02:34:33 +08:00
|
|
|
#endif // !_LIBCPP_CXX03_LANG
|
2013-06-30 02:38:17 +08:00
|
|
|
|
2010-05-12 03:42:16 +08:00
|
|
|
//enum class future_status
|
2011-12-03 03:36:40 +08:00
|
|
|
_LIBCPP_DECLARE_STRONG_ENUM(future_status)
|
2010-05-12 03:42:16 +08:00
|
|
|
{
|
|
|
|
ready,
|
|
|
|
timeout,
|
|
|
|
deferred
|
|
|
|
};
|
2011-12-03 03:36:40 +08:00
|
|
|
_LIBCPP_DECLARE_STRONG_ENUM_EPILOG(future_status)
|
2010-05-12 03:42:16 +08:00
|
|
|
|
2013-03-07 07:30:19 +08:00
|
|
|
_LIBCPP_FUNC_VIS
|
2012-07-22 01:46:55 +08:00
|
|
|
const error_category& future_category() _NOEXCEPT;
|
2010-08-26 01:32:05 +08:00
|
|
|
|
|
|
|
inline _LIBCPP_INLINE_VISIBILITY
|
|
|
|
error_code
|
2012-07-22 01:46:55 +08:00
|
|
|
make_error_code(future_errc __e) _NOEXCEPT
|
2010-08-26 01:32:05 +08:00
|
|
|
{
|
|
|
|
return error_code(static_cast<int>(__e), future_category());
|
|
|
|
}
|
|
|
|
|
|
|
|
inline _LIBCPP_INLINE_VISIBILITY
|
|
|
|
error_condition
|
2012-07-22 01:46:55 +08:00
|
|
|
make_error_condition(future_errc __e) _NOEXCEPT
|
2010-08-26 01:32:05 +08:00
|
|
|
{
|
|
|
|
return error_condition(static_cast<int>(__e), future_category());
|
|
|
|
}
|
|
|
|
|
2017-05-05 01:08:54 +08:00
|
|
|
class _LIBCPP_EXCEPTION_ABI _LIBCPP_AVAILABILITY_FUTURE_ERROR future_error
|
2010-08-26 01:32:05 +08:00
|
|
|
: public logic_error
|
|
|
|
{
|
|
|
|
error_code __ec_;
|
|
|
|
public:
|
|
|
|
future_error(error_code __ec);
|
2021-04-01 14:29:55 +08:00
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
const error_code& code() const _NOEXCEPT {return __ec_;}
|
2011-07-08 05:03:52 +08:00
|
|
|
|
2020-03-14 02:36:26 +08:00
|
|
|
future_error(const future_error&) _NOEXCEPT = default;
|
2011-07-08 05:03:52 +08:00
|
|
|
virtual ~future_error() _NOEXCEPT;
|
2010-08-26 01:32:05 +08:00
|
|
|
};
|
|
|
|
|
2018-07-12 07:14:33 +08:00
|
|
|
_LIBCPP_NORETURN inline _LIBCPP_INLINE_VISIBILITY
|
2017-05-05 01:08:54 +08:00
|
|
|
#ifndef _LIBCPP_NO_EXCEPTIONS
|
|
|
|
_LIBCPP_AVAILABILITY_FUTURE_ERROR
|
|
|
|
#endif
|
2022-07-09 00:17:26 +08:00
|
|
|
void __throw_future_error(future_errc __ev)
|
2015-09-03 23:11:32 +08:00
|
|
|
{
|
|
|
|
#ifndef _LIBCPP_NO_EXCEPTIONS
|
2022-07-09 00:17:26 +08:00
|
|
|
throw future_error(make_error_code(__ev));
|
2015-09-03 23:11:32 +08:00
|
|
|
#else
|
2022-07-09 00:17:26 +08:00
|
|
|
((void)__ev);
|
2016-08-25 23:09:01 +08:00
|
|
|
_VSTD::abort();
|
2015-09-03 23:11:32 +08:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2017-05-05 01:08:54 +08:00
|
|
|
class _LIBCPP_TYPE_VIS _LIBCPP_AVAILABILITY_FUTURE __assoc_sub_state
|
2010-08-28 04:10:19 +08:00
|
|
|
: public __shared_count
|
|
|
|
{
|
|
|
|
protected:
|
|
|
|
exception_ptr __exception_;
|
|
|
|
mutable mutex __mut_;
|
|
|
|
mutable condition_variable __cv_;
|
|
|
|
unsigned __state_;
|
|
|
|
|
2011-05-28 22:41:13 +08:00
|
|
|
virtual void __on_zero_shared() _NOEXCEPT;
|
2010-08-31 02:46:21 +08:00
|
|
|
void __sub_wait(unique_lock<mutex>& __lk);
|
2010-08-28 04:10:19 +08:00
|
|
|
public:
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
__constructed = 1,
|
|
|
|
__future_attached = 2,
|
|
|
|
ready = 4,
|
|
|
|
deferred = 8
|
|
|
|
};
|
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-08-28 04:10:19 +08:00
|
|
|
__assoc_sub_state() : __state_(0) {}
|
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-08-28 04:10:19 +08:00
|
|
|
bool __has_value() const
|
|
|
|
{return (__state_ & __constructed) || (__exception_ != nullptr);}
|
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2018-08-24 22:00:59 +08:00
|
|
|
void __attach_future() {
|
2013-01-15 04:01:24 +08:00
|
|
|
lock_guard<mutex> __lk(__mut_);
|
2018-08-24 22:00:59 +08:00
|
|
|
bool __has_future_attached = (__state_ & __future_attached) != 0;
|
|
|
|
if (__has_future_attached)
|
|
|
|
__throw_future_error(future_errc::future_already_retrieved);
|
|
|
|
this->__add_shared();
|
2013-01-15 04:01:24 +08:00
|
|
|
__state_ |= __future_attached;
|
|
|
|
}
|
2010-08-28 04:10:19 +08:00
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-08-31 02:46:21 +08:00
|
|
|
void __set_deferred() {__state_ |= deferred;}
|
|
|
|
|
2010-08-28 04:10:19 +08:00
|
|
|
void __make_ready();
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2013-10-13 09:02:45 +08:00
|
|
|
bool __is_ready() const {return (__state_ & ready) != 0;}
|
2010-08-28 04:10:19 +08:00
|
|
|
|
|
|
|
void set_value();
|
|
|
|
void set_value_at_thread_exit();
|
|
|
|
|
|
|
|
void set_exception(exception_ptr __p);
|
|
|
|
void set_exception_at_thread_exit(exception_ptr __p);
|
|
|
|
|
|
|
|
void copy();
|
|
|
|
|
2010-08-31 02:46:21 +08:00
|
|
|
void wait();
|
2010-08-28 04:10:19 +08:00
|
|
|
template <class _Rep, class _Period>
|
|
|
|
future_status
|
2015-11-07 09:22:13 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-08-28 04:10:19 +08:00
|
|
|
wait_for(const chrono::duration<_Rep, _Period>& __rel_time) const;
|
|
|
|
template <class _Clock, class _Duration>
|
2017-03-02 11:22:18 +08:00
|
|
|
_LIBCPP_METHOD_TEMPLATE_IMPLICIT_INSTANTIATION_VIS
|
2010-08-28 04:10:19 +08:00
|
|
|
future_status
|
|
|
|
wait_until(const chrono::time_point<_Clock, _Duration>& __abs_time) const;
|
2010-08-31 02:46:21 +08:00
|
|
|
|
|
|
|
virtual void __execute();
|
2010-08-28 04:10:19 +08:00
|
|
|
};
|
|
|
|
|
2010-08-29 05:01:06 +08:00
|
|
|
template <class _Clock, class _Duration>
|
|
|
|
future_status
|
|
|
|
__assoc_sub_state::wait_until(const chrono::time_point<_Clock, _Duration>& __abs_time) const
|
|
|
|
{
|
|
|
|
unique_lock<mutex> __lk(__mut_);
|
2010-08-31 02:46:21 +08:00
|
|
|
if (__state_ & deferred)
|
|
|
|
return future_status::deferred;
|
|
|
|
while (!(__state_ & ready) && _Clock::now() < __abs_time)
|
2010-08-29 05:01:06 +08:00
|
|
|
__cv_.wait_until(__lk, __abs_time);
|
|
|
|
if (__state_ & ready)
|
|
|
|
return future_status::ready;
|
|
|
|
return future_status::timeout;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class _Rep, class _Period>
|
2015-11-07 09:22:13 +08:00
|
|
|
inline
|
2010-08-29 05:01:06 +08:00
|
|
|
future_status
|
|
|
|
__assoc_sub_state::wait_for(const chrono::duration<_Rep, _Period>& __rel_time) const
|
|
|
|
{
|
2010-11-21 03:16:30 +08:00
|
|
|
return wait_until(chrono::steady_clock::now() + __rel_time);
|
2010-08-29 05:01:06 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2019-12-11 07:00:42 +08:00
|
|
|
class _LIBCPP_AVAILABILITY_FUTURE _LIBCPP_HIDDEN __assoc_state
|
2010-08-28 04:10:19 +08:00
|
|
|
: public __assoc_sub_state
|
|
|
|
{
|
|
|
|
typedef __assoc_sub_state base;
|
2011-11-30 02:15:50 +08:00
|
|
|
typedef typename aligned_storage<sizeof(_Rp), alignment_of<_Rp>::value>::type _Up;
|
2010-08-28 04:10:19 +08:00
|
|
|
protected:
|
2011-11-30 02:15:50 +08:00
|
|
|
_Up __value_;
|
2010-08-28 04:10:19 +08:00
|
|
|
|
2011-05-28 22:41:13 +08:00
|
|
|
virtual void __on_zero_shared() _NOEXCEPT;
|
2010-08-28 04:10:19 +08:00
|
|
|
public:
|
|
|
|
|
|
|
|
template <class _Arg>
|
2020-07-30 21:42:23 +08:00
|
|
|
void set_value(_Arg&& __arg);
|
2010-08-28 04:10:19 +08:00
|
|
|
|
|
|
|
template <class _Arg>
|
2020-07-30 21:42:23 +08:00
|
|
|
void set_value_at_thread_exit(_Arg&& __arg);
|
2010-08-28 04:10:19 +08:00
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
_Rp move();
|
|
|
|
typename add_lvalue_reference<_Rp>::type copy();
|
2010-08-28 04:10:19 +08:00
|
|
|
};
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2010-08-28 04:10:19 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
__assoc_state<_Rp>::__on_zero_shared() _NOEXCEPT
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
|
|
|
if (this->__state_ & base::__constructed)
|
2011-11-30 02:15:50 +08:00
|
|
|
reinterpret_cast<_Rp*>(&__value_)->~_Rp();
|
2010-08-28 04:10:19 +08:00
|
|
|
delete this;
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2010-08-28 04:10:19 +08:00
|
|
|
template <class _Arg>
|
2017-05-05 01:08:54 +08:00
|
|
|
_LIBCPP_AVAILABILITY_FUTURE
|
2010-08-28 04:10:19 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
__assoc_state<_Rp>::set_value(_Arg&& __arg)
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
|
|
|
unique_lock<mutex> __lk(this->__mut_);
|
|
|
|
if (this->__has_value())
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::promise_already_satisfied);
|
2020-12-12 09:30:28 +08:00
|
|
|
::new ((void*)&__value_) _Rp(_VSTD::forward<_Arg>(__arg));
|
2010-08-28 04:10:19 +08:00
|
|
|
this->__state_ |= base::__constructed | base::ready;
|
|
|
|
__cv_.notify_all();
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2010-08-28 04:10:19 +08:00
|
|
|
template <class _Arg>
|
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
__assoc_state<_Rp>::set_value_at_thread_exit(_Arg&& __arg)
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
|
|
|
unique_lock<mutex> __lk(this->__mut_);
|
|
|
|
if (this->__has_value())
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::promise_already_satisfied);
|
2020-12-12 09:30:28 +08:00
|
|
|
::new ((void*)&__value_) _Rp(_VSTD::forward<_Arg>(__arg));
|
2010-08-28 04:10:19 +08:00
|
|
|
this->__state_ |= base::__constructed;
|
2010-10-15 03:18:04 +08:00
|
|
|
__thread_local_data()->__make_ready_at_thread_exit(this);
|
2010-08-28 04:10:19 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
|
|
|
_Rp
|
|
|
|
__assoc_state<_Rp>::move()
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
|
|
|
unique_lock<mutex> __lk(this->__mut_);
|
2010-08-31 02:46:21 +08:00
|
|
|
this->__sub_wait(__lk);
|
2010-08-28 04:10:19 +08:00
|
|
|
if (this->__exception_ != nullptr)
|
|
|
|
rethrow_exception(this->__exception_);
|
2011-11-30 02:15:50 +08:00
|
|
|
return _VSTD::move(*reinterpret_cast<_Rp*>(&__value_));
|
2010-08-28 04:10:19 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
|
|
|
typename add_lvalue_reference<_Rp>::type
|
|
|
|
__assoc_state<_Rp>::copy()
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
|
|
|
unique_lock<mutex> __lk(this->__mut_);
|
2010-08-31 02:46:21 +08:00
|
|
|
this->__sub_wait(__lk);
|
2010-08-28 04:10:19 +08:00
|
|
|
if (this->__exception_ != nullptr)
|
|
|
|
rethrow_exception(this->__exception_);
|
2011-11-30 02:15:50 +08:00
|
|
|
return *reinterpret_cast<_Rp*>(&__value_);
|
2010-08-28 04:10:19 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2017-05-05 01:08:54 +08:00
|
|
|
class _LIBCPP_AVAILABILITY_FUTURE __assoc_state<_Rp&>
|
2010-08-29 05:01:06 +08:00
|
|
|
: public __assoc_sub_state
|
|
|
|
{
|
|
|
|
typedef __assoc_sub_state base;
|
2011-11-30 02:15:50 +08:00
|
|
|
typedef _Rp* _Up;
|
2010-08-29 05:01:06 +08:00
|
|
|
protected:
|
2011-11-30 02:15:50 +08:00
|
|
|
_Up __value_;
|
2010-08-29 05:01:06 +08:00
|
|
|
|
2011-05-28 22:41:13 +08:00
|
|
|
virtual void __on_zero_shared() _NOEXCEPT;
|
2010-08-29 05:01:06 +08:00
|
|
|
public:
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
void set_value(_Rp& __arg);
|
|
|
|
void set_value_at_thread_exit(_Rp& __arg);
|
2010-08-29 05:01:06 +08:00
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
_Rp& copy();
|
2010-08-29 05:01:06 +08:00
|
|
|
};
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2010-08-29 05:01:06 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
__assoc_state<_Rp&>::__on_zero_shared() _NOEXCEPT
|
2010-08-29 05:01:06 +08:00
|
|
|
{
|
|
|
|
delete this;
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2010-08-29 05:01:06 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
__assoc_state<_Rp&>::set_value(_Rp& __arg)
|
2010-08-29 05:01:06 +08:00
|
|
|
{
|
|
|
|
unique_lock<mutex> __lk(this->__mut_);
|
|
|
|
if (this->__has_value())
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::promise_already_satisfied);
|
2013-08-09 02:38:55 +08:00
|
|
|
__value_ = _VSTD::addressof(__arg);
|
2010-08-29 05:01:06 +08:00
|
|
|
this->__state_ |= base::__constructed | base::ready;
|
|
|
|
__cv_.notify_all();
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2010-08-29 05:01:06 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
__assoc_state<_Rp&>::set_value_at_thread_exit(_Rp& __arg)
|
2010-08-29 05:01:06 +08:00
|
|
|
{
|
|
|
|
unique_lock<mutex> __lk(this->__mut_);
|
|
|
|
if (this->__has_value())
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::promise_already_satisfied);
|
2013-08-09 02:38:55 +08:00
|
|
|
__value_ = _VSTD::addressof(__arg);
|
2010-08-29 05:01:06 +08:00
|
|
|
this->__state_ |= base::__constructed;
|
2010-10-15 03:18:04 +08:00
|
|
|
__thread_local_data()->__make_ready_at_thread_exit(this);
|
2010-08-29 05:01:06 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
|
|
|
_Rp&
|
|
|
|
__assoc_state<_Rp&>::copy()
|
2010-08-29 05:01:06 +08:00
|
|
|
{
|
|
|
|
unique_lock<mutex> __lk(this->__mut_);
|
2010-08-31 02:46:21 +08:00
|
|
|
this->__sub_wait(__lk);
|
2010-08-29 05:01:06 +08:00
|
|
|
if (this->__exception_ != nullptr)
|
|
|
|
rethrow_exception(this->__exception_);
|
|
|
|
return *__value_;
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp, class _Alloc>
|
2017-05-05 01:08:54 +08:00
|
|
|
class _LIBCPP_AVAILABILITY_FUTURE __assoc_state_alloc
|
2011-11-30 02:15:50 +08:00
|
|
|
: public __assoc_state<_Rp>
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
2011-11-30 02:15:50 +08:00
|
|
|
typedef __assoc_state<_Rp> base;
|
2010-08-28 04:10:19 +08:00
|
|
|
_Alloc __alloc_;
|
|
|
|
|
2011-05-28 22:41:13 +08:00
|
|
|
virtual void __on_zero_shared() _NOEXCEPT;
|
2010-08-28 04:10:19 +08:00
|
|
|
public:
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-08-28 04:10:19 +08:00
|
|
|
explicit __assoc_state_alloc(const _Alloc& __a)
|
|
|
|
: __alloc_(__a) {}
|
|
|
|
};
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp, class _Alloc>
|
2010-08-28 04:10:19 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
__assoc_state_alloc<_Rp, _Alloc>::__on_zero_shared() _NOEXCEPT
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
|
|
|
if (this->__state_ & base::__constructed)
|
2013-08-09 02:38:55 +08:00
|
|
|
reinterpret_cast<_Rp*>(_VSTD::addressof(this->__value_))->~_Rp();
|
2015-02-06 07:01:40 +08:00
|
|
|
typedef typename __allocator_traits_rebind<_Alloc, __assoc_state_alloc>::type _Al;
|
|
|
|
typedef allocator_traits<_Al> _ATraits;
|
2014-10-23 14:24:45 +08:00
|
|
|
typedef pointer_traits<typename _ATraits::pointer> _PTraits;
|
2015-02-06 07:01:40 +08:00
|
|
|
_Al __a(__alloc_);
|
2010-08-28 04:10:19 +08:00
|
|
|
this->~__assoc_state_alloc();
|
2014-10-23 14:24:45 +08:00
|
|
|
__a.deallocate(_PTraits::pointer_to(*this), 1);
|
2010-08-28 04:10:19 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp, class _Alloc>
|
2017-05-05 01:08:54 +08:00
|
|
|
class _LIBCPP_AVAILABILITY_FUTURE __assoc_state_alloc<_Rp&, _Alloc>
|
2011-11-30 02:15:50 +08:00
|
|
|
: public __assoc_state<_Rp&>
|
2010-08-29 05:01:06 +08:00
|
|
|
{
|
2011-11-30 02:15:50 +08:00
|
|
|
typedef __assoc_state<_Rp&> base;
|
2010-08-29 05:01:06 +08:00
|
|
|
_Alloc __alloc_;
|
|
|
|
|
2011-05-28 22:41:13 +08:00
|
|
|
virtual void __on_zero_shared() _NOEXCEPT;
|
2010-08-29 05:01:06 +08:00
|
|
|
public:
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-08-29 05:01:06 +08:00
|
|
|
explicit __assoc_state_alloc(const _Alloc& __a)
|
|
|
|
: __alloc_(__a) {}
|
|
|
|
};
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp, class _Alloc>
|
2010-08-29 05:01:06 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
__assoc_state_alloc<_Rp&, _Alloc>::__on_zero_shared() _NOEXCEPT
|
2010-08-29 05:01:06 +08:00
|
|
|
{
|
2015-02-06 07:01:40 +08:00
|
|
|
typedef typename __allocator_traits_rebind<_Alloc, __assoc_state_alloc>::type _Al;
|
|
|
|
typedef allocator_traits<_Al> _ATraits;
|
2014-10-23 14:24:45 +08:00
|
|
|
typedef pointer_traits<typename _ATraits::pointer> _PTraits;
|
2015-02-06 07:01:40 +08:00
|
|
|
_Al __a(__alloc_);
|
2010-08-29 05:01:06 +08:00
|
|
|
this->~__assoc_state_alloc();
|
2014-10-23 14:24:45 +08:00
|
|
|
__a.deallocate(_PTraits::pointer_to(*this), 1);
|
2010-08-29 05:01:06 +08:00
|
|
|
}
|
|
|
|
|
2010-08-28 04:10:19 +08:00
|
|
|
template <class _Alloc>
|
2017-05-05 01:08:54 +08:00
|
|
|
class _LIBCPP_AVAILABILITY_FUTURE __assoc_sub_state_alloc
|
2010-08-28 04:10:19 +08:00
|
|
|
: public __assoc_sub_state
|
|
|
|
{
|
|
|
|
typedef __assoc_sub_state base;
|
|
|
|
_Alloc __alloc_;
|
|
|
|
|
2011-05-28 22:41:13 +08:00
|
|
|
virtual void __on_zero_shared() _NOEXCEPT;
|
2010-08-28 04:10:19 +08:00
|
|
|
public:
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-08-28 04:10:19 +08:00
|
|
|
explicit __assoc_sub_state_alloc(const _Alloc& __a)
|
|
|
|
: __alloc_(__a) {}
|
|
|
|
};
|
|
|
|
|
|
|
|
template <class _Alloc>
|
|
|
|
void
|
2011-05-28 22:41:13 +08:00
|
|
|
__assoc_sub_state_alloc<_Alloc>::__on_zero_shared() _NOEXCEPT
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
2015-02-06 07:01:40 +08:00
|
|
|
typedef typename __allocator_traits_rebind<_Alloc, __assoc_sub_state_alloc>::type _Al;
|
|
|
|
typedef allocator_traits<_Al> _ATraits;
|
2014-10-23 14:24:45 +08:00
|
|
|
typedef pointer_traits<typename _ATraits::pointer> _PTraits;
|
2015-02-06 07:01:40 +08:00
|
|
|
_Al __a(__alloc_);
|
2010-08-28 04:10:19 +08:00
|
|
|
this->~__assoc_sub_state_alloc();
|
2014-10-23 14:24:45 +08:00
|
|
|
__a.deallocate(_PTraits::pointer_to(*this), 1);
|
2010-08-28 04:10:19 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp, class _Fp>
|
2017-05-05 01:08:54 +08:00
|
|
|
class _LIBCPP_AVAILABILITY_FUTURE __deferred_assoc_state
|
2011-11-30 02:15:50 +08:00
|
|
|
: public __assoc_state<_Rp>
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
2011-11-30 02:15:50 +08:00
|
|
|
typedef __assoc_state<_Rp> base;
|
2010-08-31 02:46:21 +08:00
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
_Fp __func_;
|
2010-08-31 02:46:21 +08:00
|
|
|
|
|
|
|
public:
|
2015-11-07 09:22:13 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2011-11-30 02:15:50 +08:00
|
|
|
explicit __deferred_assoc_state(_Fp&& __f);
|
2010-08-31 02:46:21 +08:00
|
|
|
|
|
|
|
virtual void __execute();
|
|
|
|
};
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp, class _Fp>
|
2015-11-07 09:22:13 +08:00
|
|
|
inline
|
2011-11-30 02:15:50 +08:00
|
|
|
__deferred_assoc_state<_Rp, _Fp>::__deferred_assoc_state(_Fp&& __f)
|
|
|
|
: __func_(_VSTD::forward<_Fp>(__f))
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
|
|
|
this->__set_deferred();
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp, class _Fp>
|
2010-08-31 02:46:21 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
__deferred_assoc_state<_Rp, _Fp>::__execute()
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
|
|
|
#ifndef _LIBCPP_NO_EXCEPTIONS
|
|
|
|
try
|
|
|
|
{
|
2021-04-21 00:03:32 +08:00
|
|
|
#endif // _LIBCPP_NO_EXCEPTIONS
|
2010-08-31 02:46:21 +08:00
|
|
|
this->set_value(__func_());
|
|
|
|
#ifndef _LIBCPP_NO_EXCEPTIONS
|
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
this->set_exception(current_exception());
|
|
|
|
}
|
2021-04-21 00:03:32 +08:00
|
|
|
#endif // _LIBCPP_NO_EXCEPTIONS
|
2010-08-31 02:46:21 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Fp>
|
2017-05-05 01:08:54 +08:00
|
|
|
class _LIBCPP_AVAILABILITY_FUTURE __deferred_assoc_state<void, _Fp>
|
2010-08-31 02:46:21 +08:00
|
|
|
: public __assoc_sub_state
|
|
|
|
{
|
|
|
|
typedef __assoc_sub_state base;
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
_Fp __func_;
|
2010-08-31 02:46:21 +08:00
|
|
|
|
|
|
|
public:
|
2015-11-07 09:22:13 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2011-11-30 02:15:50 +08:00
|
|
|
explicit __deferred_assoc_state(_Fp&& __f);
|
2010-08-31 02:46:21 +08:00
|
|
|
|
|
|
|
virtual void __execute();
|
|
|
|
};
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Fp>
|
2015-11-07 09:22:13 +08:00
|
|
|
inline
|
2011-11-30 02:15:50 +08:00
|
|
|
__deferred_assoc_state<void, _Fp>::__deferred_assoc_state(_Fp&& __f)
|
|
|
|
: __func_(_VSTD::forward<_Fp>(__f))
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
|
|
|
this->__set_deferred();
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Fp>
|
2010-08-31 02:46:21 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
__deferred_assoc_state<void, _Fp>::__execute()
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
|
|
|
#ifndef _LIBCPP_NO_EXCEPTIONS
|
|
|
|
try
|
|
|
|
{
|
2021-04-21 00:03:32 +08:00
|
|
|
#endif // _LIBCPP_NO_EXCEPTIONS
|
2010-08-31 02:46:21 +08:00
|
|
|
__func_();
|
|
|
|
this->set_value();
|
|
|
|
#ifndef _LIBCPP_NO_EXCEPTIONS
|
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
this->set_exception(current_exception());
|
|
|
|
}
|
2021-04-21 00:03:32 +08:00
|
|
|
#endif // _LIBCPP_NO_EXCEPTIONS
|
2010-08-31 02:46:21 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp, class _Fp>
|
2017-05-05 01:08:54 +08:00
|
|
|
class _LIBCPP_AVAILABILITY_FUTURE __async_assoc_state
|
2011-11-30 02:15:50 +08:00
|
|
|
: public __assoc_state<_Rp>
|
2011-05-19 23:05:04 +08:00
|
|
|
{
|
2011-11-30 02:15:50 +08:00
|
|
|
typedef __assoc_state<_Rp> base;
|
2011-05-19 23:05:04 +08:00
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
_Fp __func_;
|
2011-05-19 23:05:04 +08:00
|
|
|
|
2011-05-28 22:41:13 +08:00
|
|
|
virtual void __on_zero_shared() _NOEXCEPT;
|
2011-05-19 23:05:04 +08:00
|
|
|
public:
|
2015-11-07 09:22:13 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2011-11-30 02:15:50 +08:00
|
|
|
explicit __async_assoc_state(_Fp&& __f);
|
2011-05-19 23:05:04 +08:00
|
|
|
|
|
|
|
virtual void __execute();
|
|
|
|
};
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp, class _Fp>
|
2015-11-07 09:22:13 +08:00
|
|
|
inline
|
2011-11-30 02:15:50 +08:00
|
|
|
__async_assoc_state<_Rp, _Fp>::__async_assoc_state(_Fp&& __f)
|
|
|
|
: __func_(_VSTD::forward<_Fp>(__f))
|
2011-05-19 23:05:04 +08:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp, class _Fp>
|
2011-05-19 23:05:04 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
__async_assoc_state<_Rp, _Fp>::__execute()
|
2011-05-19 23:05:04 +08:00
|
|
|
{
|
|
|
|
#ifndef _LIBCPP_NO_EXCEPTIONS
|
|
|
|
try
|
|
|
|
{
|
2021-04-21 00:03:32 +08:00
|
|
|
#endif // _LIBCPP_NO_EXCEPTIONS
|
2011-05-19 23:05:04 +08:00
|
|
|
this->set_value(__func_());
|
|
|
|
#ifndef _LIBCPP_NO_EXCEPTIONS
|
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
this->set_exception(current_exception());
|
|
|
|
}
|
2021-04-21 00:03:32 +08:00
|
|
|
#endif // _LIBCPP_NO_EXCEPTIONS
|
2011-05-19 23:05:04 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp, class _Fp>
|
2011-05-19 23:05:04 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
__async_assoc_state<_Rp, _Fp>::__on_zero_shared() _NOEXCEPT
|
2011-05-19 23:05:04 +08:00
|
|
|
{
|
|
|
|
this->wait();
|
|
|
|
base::__on_zero_shared();
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Fp>
|
2017-05-05 01:08:54 +08:00
|
|
|
class _LIBCPP_AVAILABILITY_FUTURE __async_assoc_state<void, _Fp>
|
2011-05-19 23:05:04 +08:00
|
|
|
: public __assoc_sub_state
|
|
|
|
{
|
|
|
|
typedef __assoc_sub_state base;
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
_Fp __func_;
|
2011-05-19 23:05:04 +08:00
|
|
|
|
2011-05-28 22:41:13 +08:00
|
|
|
virtual void __on_zero_shared() _NOEXCEPT;
|
2011-05-19 23:05:04 +08:00
|
|
|
public:
|
2015-11-07 09:22:13 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2011-11-30 02:15:50 +08:00
|
|
|
explicit __async_assoc_state(_Fp&& __f);
|
2011-05-19 23:05:04 +08:00
|
|
|
|
|
|
|
virtual void __execute();
|
|
|
|
};
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Fp>
|
2015-11-07 09:22:13 +08:00
|
|
|
inline
|
2011-11-30 02:15:50 +08:00
|
|
|
__async_assoc_state<void, _Fp>::__async_assoc_state(_Fp&& __f)
|
|
|
|
: __func_(_VSTD::forward<_Fp>(__f))
|
2011-05-19 23:05:04 +08:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Fp>
|
2011-05-19 23:05:04 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
__async_assoc_state<void, _Fp>::__execute()
|
2011-05-19 23:05:04 +08:00
|
|
|
{
|
|
|
|
#ifndef _LIBCPP_NO_EXCEPTIONS
|
|
|
|
try
|
|
|
|
{
|
2021-04-21 00:03:32 +08:00
|
|
|
#endif // _LIBCPP_NO_EXCEPTIONS
|
2011-05-19 23:05:04 +08:00
|
|
|
__func_();
|
|
|
|
this->set_value();
|
|
|
|
#ifndef _LIBCPP_NO_EXCEPTIONS
|
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
this->set_exception(current_exception());
|
|
|
|
}
|
2021-04-21 00:03:32 +08:00
|
|
|
#endif // _LIBCPP_NO_EXCEPTIONS
|
2011-05-19 23:05:04 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Fp>
|
2011-05-19 23:05:04 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
__async_assoc_state<void, _Fp>::__on_zero_shared() _NOEXCEPT
|
2011-05-19 23:05:04 +08:00
|
|
|
{
|
|
|
|
this->wait();
|
|
|
|
base::__on_zero_shared();
|
|
|
|
}
|
|
|
|
|
2017-01-05 07:56:00 +08:00
|
|
|
template <class _Rp> class _LIBCPP_TEMPLATE_VIS promise;
|
|
|
|
template <class _Rp> class _LIBCPP_TEMPLATE_VIS shared_future;
|
2010-08-28 04:10:19 +08:00
|
|
|
|
|
|
|
// future
|
|
|
|
|
2017-01-05 07:56:00 +08:00
|
|
|
template <class _Rp> class _LIBCPP_TEMPLATE_VIS future;
|
2010-08-31 02:46:21 +08:00
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp, class _Fp>
|
2019-12-11 07:00:42 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY future<_Rp>
|
2011-11-30 02:15:50 +08:00
|
|
|
__make_deferred_assoc_state(_Fp&& __f);
|
2010-08-31 02:46:21 +08:00
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp, class _Fp>
|
2019-12-11 07:00:42 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY future<_Rp>
|
2011-11-30 02:15:50 +08:00
|
|
|
__make_async_assoc_state(_Fp&& __f);
|
2011-05-19 23:05:04 +08:00
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2017-05-05 01:08:54 +08:00
|
|
|
class _LIBCPP_TEMPLATE_VIS _LIBCPP_AVAILABILITY_FUTURE future
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
2011-11-30 02:15:50 +08:00
|
|
|
__assoc_state<_Rp>* __state_;
|
2010-08-28 04:10:19 +08:00
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
explicit future(__assoc_state<_Rp>* __state);
|
2010-08-28 04:10:19 +08:00
|
|
|
|
|
|
|
template <class> friend class promise;
|
2010-09-04 02:39:25 +08:00
|
|
|
template <class> friend class shared_future;
|
2010-08-31 02:46:21 +08:00
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _R1, class _Fp>
|
|
|
|
friend future<_R1> __make_deferred_assoc_state(_Fp&& __f);
|
|
|
|
template <class _R1, class _Fp>
|
|
|
|
friend future<_R1> __make_async_assoc_state(_Fp&& __f);
|
2010-08-31 02:46:21 +08:00
|
|
|
|
2010-08-28 04:10:19 +08:00
|
|
|
public:
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
future() _NOEXCEPT : __state_(nullptr) {}
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
future(future&& __rhs) _NOEXCEPT
|
2010-08-28 04:10:19 +08:00
|
|
|
: __state_(__rhs.__state_) {__rhs.__state_ = nullptr;}
|
|
|
|
future(const future&) = delete;
|
|
|
|
future& operator=(const future&) = delete;
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
future& operator=(future&& __rhs) _NOEXCEPT
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
[libc++] Consistently replace `std::` qualification with `_VSTD::` or nothing. NFCI.
I used a lot of `git grep` to find places where `std::` was being used
outside of comments and assert-messages. There were three outcomes:
- Qualified function calls, e.g. `std::move` becomes `_VSTD::move`.
This is the most common case.
- Typenames that don't need qualification, e.g. `std::allocator` becomes `allocator`.
Leaving these as `_VSTD::allocator` would also be fine, but I decided
that removing the qualification is more consistent with existing practice.
- Names that specifically need un-versioned `std::` qualification,
or that I wasn't sure about. For example, I didn't touch any code in
<atomic>, <math.h>, <new>, or any ext/ or experimental/ headers;
and I didn't touch any instances of `std::type_info`.
In some deduction guides, we were accidentally using `class Alloc = typename std::allocator<T>`,
despite `std::allocator<T>`'s type-ness not being template-dependent.
Because `std::allocator` is a qualified name, this did parse as we intended;
but what we meant was simply `class Alloc = allocator<T>`.
Differential Revision: https://reviews.llvm.org/D92250
2020-11-28 00:02:06 +08:00
|
|
|
future(_VSTD::move(__rhs)).swap(*this);
|
2010-08-28 04:10:19 +08:00
|
|
|
return *this;
|
|
|
|
}
|
2020-07-30 21:42:23 +08:00
|
|
|
|
2010-08-28 04:10:19 +08:00
|
|
|
~future();
|
2015-11-07 09:22:13 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2017-01-25 07:28:25 +08:00
|
|
|
shared_future<_Rp> share() _NOEXCEPT;
|
2010-08-28 04:10:19 +08:00
|
|
|
|
|
|
|
// retrieving the value
|
2011-11-30 02:15:50 +08:00
|
|
|
_Rp get();
|
2010-08-28 04:10:19 +08:00
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
void swap(future& __rhs) _NOEXCEPT {_VSTD::swap(__state_, __rhs.__state_);}
|
2010-08-28 04:10:19 +08:00
|
|
|
|
|
|
|
// functions to check state
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
bool valid() const _NOEXCEPT {return __state_ != nullptr;}
|
2010-08-28 04:10:19 +08:00
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-08-28 04:10:19 +08:00
|
|
|
void wait() const {__state_->wait();}
|
|
|
|
template <class _Rep, class _Period>
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-08-28 04:10:19 +08:00
|
|
|
future_status
|
|
|
|
wait_for(const chrono::duration<_Rep, _Period>& __rel_time) const
|
|
|
|
{return __state_->wait_for(__rel_time);}
|
|
|
|
template <class _Clock, class _Duration>
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-08-28 04:10:19 +08:00
|
|
|
future_status
|
|
|
|
wait_until(const chrono::time_point<_Clock, _Duration>& __abs_time) const
|
|
|
|
{return __state_->wait_until(__abs_time);}
|
|
|
|
};
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
|
|
|
future<_Rp>::future(__assoc_state<_Rp>* __state)
|
2010-08-28 04:10:19 +08:00
|
|
|
: __state_(__state)
|
|
|
|
{
|
2018-08-24 22:00:59 +08:00
|
|
|
__state_->__attach_future();
|
2010-08-28 04:10:19 +08:00
|
|
|
}
|
|
|
|
|
2010-08-31 02:46:21 +08:00
|
|
|
struct __release_shared_count
|
|
|
|
{
|
2022-07-09 00:17:26 +08:00
|
|
|
void operator()(__shared_count* __p) {__p->__release_shared();}
|
2010-08-31 02:46:21 +08:00
|
|
|
};
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
|
|
|
future<_Rp>::~future()
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
|
|
|
if (__state_)
|
|
|
|
__state_->__release_shared();
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
|
|
|
_Rp
|
|
|
|
future<_Rp>::get()
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
2010-08-31 02:46:21 +08:00
|
|
|
unique_ptr<__shared_count, __release_shared_count> __(__state_);
|
2011-11-30 02:15:50 +08:00
|
|
|
__assoc_state<_Rp>* __s = __state_;
|
2010-08-28 04:10:19 +08:00
|
|
|
__state_ = nullptr;
|
|
|
|
return __s->move();
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2017-05-05 01:08:54 +08:00
|
|
|
class _LIBCPP_TEMPLATE_VIS _LIBCPP_AVAILABILITY_FUTURE future<_Rp&>
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
2011-11-30 02:15:50 +08:00
|
|
|
__assoc_state<_Rp&>* __state_;
|
2010-08-28 04:10:19 +08:00
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
explicit future(__assoc_state<_Rp&>* __state);
|
2010-08-28 04:10:19 +08:00
|
|
|
|
|
|
|
template <class> friend class promise;
|
2010-09-04 02:39:25 +08:00
|
|
|
template <class> friend class shared_future;
|
2010-08-31 02:46:21 +08:00
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _R1, class _Fp>
|
|
|
|
friend future<_R1> __make_deferred_assoc_state(_Fp&& __f);
|
|
|
|
template <class _R1, class _Fp>
|
|
|
|
friend future<_R1> __make_async_assoc_state(_Fp&& __f);
|
2010-08-31 02:46:21 +08:00
|
|
|
|
2010-08-28 04:10:19 +08:00
|
|
|
public:
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
future() _NOEXCEPT : __state_(nullptr) {}
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
future(future&& __rhs) _NOEXCEPT
|
2010-08-28 04:10:19 +08:00
|
|
|
: __state_(__rhs.__state_) {__rhs.__state_ = nullptr;}
|
|
|
|
future(const future&) = delete;
|
|
|
|
future& operator=(const future&) = delete;
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
future& operator=(future&& __rhs) _NOEXCEPT
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
[libc++] Consistently replace `std::` qualification with `_VSTD::` or nothing. NFCI.
I used a lot of `git grep` to find places where `std::` was being used
outside of comments and assert-messages. There were three outcomes:
- Qualified function calls, e.g. `std::move` becomes `_VSTD::move`.
This is the most common case.
- Typenames that don't need qualification, e.g. `std::allocator` becomes `allocator`.
Leaving these as `_VSTD::allocator` would also be fine, but I decided
that removing the qualification is more consistent with existing practice.
- Names that specifically need un-versioned `std::` qualification,
or that I wasn't sure about. For example, I didn't touch any code in
<atomic>, <math.h>, <new>, or any ext/ or experimental/ headers;
and I didn't touch any instances of `std::type_info`.
In some deduction guides, we were accidentally using `class Alloc = typename std::allocator<T>`,
despite `std::allocator<T>`'s type-ness not being template-dependent.
Because `std::allocator` is a qualified name, this did parse as we intended;
but what we meant was simply `class Alloc = allocator<T>`.
Differential Revision: https://reviews.llvm.org/D92250
2020-11-28 00:02:06 +08:00
|
|
|
future(_VSTD::move(__rhs)).swap(*this);
|
2010-08-28 04:10:19 +08:00
|
|
|
return *this;
|
|
|
|
}
|
2020-07-30 21:42:23 +08:00
|
|
|
|
2010-08-28 04:10:19 +08:00
|
|
|
~future();
|
2015-11-07 09:22:13 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2017-01-25 07:28:25 +08:00
|
|
|
shared_future<_Rp&> share() _NOEXCEPT;
|
2010-08-28 04:10:19 +08:00
|
|
|
|
|
|
|
// retrieving the value
|
2011-11-30 02:15:50 +08:00
|
|
|
_Rp& get();
|
2010-08-28 04:10:19 +08:00
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
void swap(future& __rhs) _NOEXCEPT {_VSTD::swap(__state_, __rhs.__state_);}
|
2010-08-28 04:10:19 +08:00
|
|
|
|
|
|
|
// functions to check state
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
bool valid() const _NOEXCEPT {return __state_ != nullptr;}
|
2010-08-28 04:10:19 +08:00
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-08-28 04:10:19 +08:00
|
|
|
void wait() const {__state_->wait();}
|
|
|
|
template <class _Rep, class _Period>
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-08-28 04:10:19 +08:00
|
|
|
future_status
|
|
|
|
wait_for(const chrono::duration<_Rep, _Period>& __rel_time) const
|
|
|
|
{return __state_->wait_for(__rel_time);}
|
|
|
|
template <class _Clock, class _Duration>
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-08-28 04:10:19 +08:00
|
|
|
future_status
|
|
|
|
wait_until(const chrono::time_point<_Clock, _Duration>& __abs_time) const
|
|
|
|
{return __state_->wait_until(__abs_time);}
|
|
|
|
};
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
|
|
|
future<_Rp&>::future(__assoc_state<_Rp&>* __state)
|
2010-08-28 04:10:19 +08:00
|
|
|
: __state_(__state)
|
|
|
|
{
|
2018-08-24 22:00:59 +08:00
|
|
|
__state_->__attach_future();
|
2010-08-28 04:10:19 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
|
|
|
future<_Rp&>::~future()
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
|
|
|
if (__state_)
|
|
|
|
__state_->__release_shared();
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
|
|
|
_Rp&
|
|
|
|
future<_Rp&>::get()
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
2010-08-31 02:46:21 +08:00
|
|
|
unique_ptr<__shared_count, __release_shared_count> __(__state_);
|
2011-11-30 02:15:50 +08:00
|
|
|
__assoc_state<_Rp&>* __s = __state_;
|
2010-08-28 04:10:19 +08:00
|
|
|
__state_ = nullptr;
|
|
|
|
return __s->copy();
|
|
|
|
}
|
|
|
|
|
|
|
|
template <>
|
2017-05-05 01:08:54 +08:00
|
|
|
class _LIBCPP_TYPE_VIS _LIBCPP_AVAILABILITY_FUTURE future<void>
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
|
|
|
__assoc_sub_state* __state_;
|
|
|
|
|
|
|
|
explicit future(__assoc_sub_state* __state);
|
|
|
|
|
|
|
|
template <class> friend class promise;
|
2010-09-04 02:39:25 +08:00
|
|
|
template <class> friend class shared_future;
|
2010-08-31 02:46:21 +08:00
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _R1, class _Fp>
|
|
|
|
friend future<_R1> __make_deferred_assoc_state(_Fp&& __f);
|
|
|
|
template <class _R1, class _Fp>
|
|
|
|
friend future<_R1> __make_async_assoc_state(_Fp&& __f);
|
2010-08-31 02:46:21 +08:00
|
|
|
|
2010-08-28 04:10:19 +08:00
|
|
|
public:
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
future() _NOEXCEPT : __state_(nullptr) {}
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
future(future&& __rhs) _NOEXCEPT
|
2010-08-28 04:10:19 +08:00
|
|
|
: __state_(__rhs.__state_) {__rhs.__state_ = nullptr;}
|
|
|
|
future(const future&) = delete;
|
|
|
|
future& operator=(const future&) = delete;
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
future& operator=(future&& __rhs) _NOEXCEPT
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
[libc++] Consistently replace `std::` qualification with `_VSTD::` or nothing. NFCI.
I used a lot of `git grep` to find places where `std::` was being used
outside of comments and assert-messages. There were three outcomes:
- Qualified function calls, e.g. `std::move` becomes `_VSTD::move`.
This is the most common case.
- Typenames that don't need qualification, e.g. `std::allocator` becomes `allocator`.
Leaving these as `_VSTD::allocator` would also be fine, but I decided
that removing the qualification is more consistent with existing practice.
- Names that specifically need un-versioned `std::` qualification,
or that I wasn't sure about. For example, I didn't touch any code in
<atomic>, <math.h>, <new>, or any ext/ or experimental/ headers;
and I didn't touch any instances of `std::type_info`.
In some deduction guides, we were accidentally using `class Alloc = typename std::allocator<T>`,
despite `std::allocator<T>`'s type-ness not being template-dependent.
Because `std::allocator` is a qualified name, this did parse as we intended;
but what we meant was simply `class Alloc = allocator<T>`.
Differential Revision: https://reviews.llvm.org/D92250
2020-11-28 00:02:06 +08:00
|
|
|
future(_VSTD::move(__rhs)).swap(*this);
|
2010-08-28 04:10:19 +08:00
|
|
|
return *this;
|
|
|
|
}
|
2020-07-30 21:42:23 +08:00
|
|
|
|
2010-08-28 04:10:19 +08:00
|
|
|
~future();
|
2015-11-07 09:22:13 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2017-01-25 07:28:25 +08:00
|
|
|
shared_future<void> share() _NOEXCEPT;
|
2010-08-28 04:10:19 +08:00
|
|
|
|
|
|
|
// retrieving the value
|
|
|
|
void get();
|
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
void swap(future& __rhs) _NOEXCEPT {_VSTD::swap(__state_, __rhs.__state_);}
|
2010-08-28 04:10:19 +08:00
|
|
|
|
|
|
|
// functions to check state
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
bool valid() const _NOEXCEPT {return __state_ != nullptr;}
|
2010-08-28 04:10:19 +08:00
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-08-28 04:10:19 +08:00
|
|
|
void wait() const {__state_->wait();}
|
|
|
|
template <class _Rep, class _Period>
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-08-28 04:10:19 +08:00
|
|
|
future_status
|
|
|
|
wait_for(const chrono::duration<_Rep, _Period>& __rel_time) const
|
|
|
|
{return __state_->wait_for(__rel_time);}
|
|
|
|
template <class _Clock, class _Duration>
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-08-28 04:10:19 +08:00
|
|
|
future_status
|
|
|
|
wait_until(const chrono::time_point<_Clock, _Duration>& __abs_time) const
|
|
|
|
{return __state_->wait_until(__abs_time);}
|
|
|
|
};
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2010-09-04 02:39:25 +08:00
|
|
|
inline _LIBCPP_INLINE_VISIBILITY
|
|
|
|
void
|
2012-07-22 01:46:55 +08:00
|
|
|
swap(future<_Rp>& __x, future<_Rp>& __y) _NOEXCEPT
|
2010-09-04 02:39:25 +08:00
|
|
|
{
|
|
|
|
__x.swap(__y);
|
|
|
|
}
|
|
|
|
|
2010-08-28 04:10:19 +08:00
|
|
|
// promise<R>
|
|
|
|
|
2011-06-15 03:58:17 +08:00
|
|
|
template <class _Callable> class packaged_task;
|
2010-08-31 02:46:21 +08:00
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2017-05-05 01:08:54 +08:00
|
|
|
class _LIBCPP_TEMPLATE_VIS _LIBCPP_AVAILABILITY_FUTURE promise
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
2011-11-30 02:15:50 +08:00
|
|
|
__assoc_state<_Rp>* __state_;
|
2010-08-31 02:46:21 +08:00
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
explicit promise(nullptr_t) _NOEXCEPT : __state_(nullptr) {}
|
2010-08-31 02:46:21 +08:00
|
|
|
|
|
|
|
template <class> friend class packaged_task;
|
2010-08-28 04:10:19 +08:00
|
|
|
public:
|
|
|
|
promise();
|
|
|
|
template <class _Alloc>
|
|
|
|
promise(allocator_arg_t, const _Alloc& __a);
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
promise(promise&& __rhs) _NOEXCEPT
|
2010-08-28 04:10:19 +08:00
|
|
|
: __state_(__rhs.__state_) {__rhs.__state_ = nullptr;}
|
|
|
|
promise(const promise& __rhs) = delete;
|
|
|
|
~promise();
|
|
|
|
|
|
|
|
// assignment
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
promise& operator=(promise&& __rhs) _NOEXCEPT
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
[libc++] Consistently replace `std::` qualification with `_VSTD::` or nothing. NFCI.
I used a lot of `git grep` to find places where `std::` was being used
outside of comments and assert-messages. There were three outcomes:
- Qualified function calls, e.g. `std::move` becomes `_VSTD::move`.
This is the most common case.
- Typenames that don't need qualification, e.g. `std::allocator` becomes `allocator`.
Leaving these as `_VSTD::allocator` would also be fine, but I decided
that removing the qualification is more consistent with existing practice.
- Names that specifically need un-versioned `std::` qualification,
or that I wasn't sure about. For example, I didn't touch any code in
<atomic>, <math.h>, <new>, or any ext/ or experimental/ headers;
and I didn't touch any instances of `std::type_info`.
In some deduction guides, we were accidentally using `class Alloc = typename std::allocator<T>`,
despite `std::allocator<T>`'s type-ness not being template-dependent.
Because `std::allocator` is a qualified name, this did parse as we intended;
but what we meant was simply `class Alloc = allocator<T>`.
Differential Revision: https://reviews.llvm.org/D92250
2020-11-28 00:02:06 +08:00
|
|
|
promise(_VSTD::move(__rhs)).swap(*this);
|
2010-08-28 04:10:19 +08:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
promise& operator=(const promise& __rhs) = delete;
|
2020-07-30 21:42:23 +08:00
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
void swap(promise& __rhs) _NOEXCEPT {_VSTD::swap(__state_, __rhs.__state_);}
|
2010-08-28 04:10:19 +08:00
|
|
|
|
|
|
|
// retrieving the result
|
2011-11-30 02:15:50 +08:00
|
|
|
future<_Rp> get_future();
|
2010-08-28 04:10:19 +08:00
|
|
|
|
|
|
|
// setting the result
|
2011-11-30 02:15:50 +08:00
|
|
|
void set_value(const _Rp& __r);
|
|
|
|
void set_value(_Rp&& __r);
|
2010-08-28 04:10:19 +08:00
|
|
|
void set_exception(exception_ptr __p);
|
|
|
|
|
|
|
|
// setting the result with deferred notification
|
2011-11-30 02:15:50 +08:00
|
|
|
void set_value_at_thread_exit(const _Rp& __r);
|
|
|
|
void set_value_at_thread_exit(_Rp&& __r);
|
2010-08-28 04:10:19 +08:00
|
|
|
void set_exception_at_thread_exit(exception_ptr __p);
|
|
|
|
};
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
|
|
|
promise<_Rp>::promise()
|
|
|
|
: __state_(new __assoc_state<_Rp>)
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2010-08-28 04:10:19 +08:00
|
|
|
template <class _Alloc>
|
2011-11-30 02:15:50 +08:00
|
|
|
promise<_Rp>::promise(allocator_arg_t, const _Alloc& __a0)
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
2014-10-23 14:24:45 +08:00
|
|
|
typedef __assoc_state_alloc<_Rp, _Alloc> _State;
|
|
|
|
typedef typename __allocator_traits_rebind<_Alloc, _State>::type _A2;
|
2010-08-28 04:10:19 +08:00
|
|
|
typedef __allocator_destructor<_A2> _D2;
|
|
|
|
_A2 __a(__a0);
|
2014-10-23 14:24:45 +08:00
|
|
|
unique_ptr<_State, _D2> __hold(__a.allocate(1), _D2(__a, 1));
|
2020-12-12 09:30:28 +08:00
|
|
|
::new ((void*)_VSTD::addressof(*__hold.get())) _State(__a0);
|
2014-10-23 14:24:45 +08:00
|
|
|
__state_ = _VSTD::addressof(*__hold.release());
|
2010-08-28 04:10:19 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
|
|
|
promise<_Rp>::~promise()
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
|
|
|
if (__state_)
|
|
|
|
{
|
|
|
|
if (!__state_->__has_value() && __state_->use_count() > 1)
|
|
|
|
__state_->set_exception(make_exception_ptr(
|
|
|
|
future_error(make_error_code(future_errc::broken_promise))
|
|
|
|
));
|
|
|
|
__state_->__release_shared();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
|
|
|
future<_Rp>
|
|
|
|
promise<_Rp>::get_future()
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
|
|
|
if (__state_ == nullptr)
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::no_state);
|
2011-11-30 02:15:50 +08:00
|
|
|
return future<_Rp>(__state_);
|
2010-08-28 04:10:19 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2010-08-28 04:10:19 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
promise<_Rp>::set_value(const _Rp& __r)
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
|
|
|
if (__state_ == nullptr)
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::no_state);
|
2010-08-28 04:10:19 +08:00
|
|
|
__state_->set_value(__r);
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2010-08-28 04:10:19 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
promise<_Rp>::set_value(_Rp&& __r)
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
|
|
|
if (__state_ == nullptr)
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::no_state);
|
2011-07-01 05:18:19 +08:00
|
|
|
__state_->set_value(_VSTD::move(__r));
|
2010-08-28 04:10:19 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2010-08-28 04:10:19 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
promise<_Rp>::set_exception(exception_ptr __p)
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
2016-05-17 00:55:32 +08:00
|
|
|
_LIBCPP_ASSERT( __p != nullptr, "promise::set_exception: received nullptr" );
|
2010-08-28 04:10:19 +08:00
|
|
|
if (__state_ == nullptr)
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::no_state);
|
2010-08-28 04:10:19 +08:00
|
|
|
__state_->set_exception(__p);
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2010-08-28 04:10:19 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
promise<_Rp>::set_value_at_thread_exit(const _Rp& __r)
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
|
|
|
if (__state_ == nullptr)
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::no_state);
|
2010-08-28 04:10:19 +08:00
|
|
|
__state_->set_value_at_thread_exit(__r);
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2010-08-28 04:10:19 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
promise<_Rp>::set_value_at_thread_exit(_Rp&& __r)
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
|
|
|
if (__state_ == nullptr)
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::no_state);
|
2011-07-01 05:18:19 +08:00
|
|
|
__state_->set_value_at_thread_exit(_VSTD::move(__r));
|
2010-08-28 04:10:19 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2010-08-28 04:10:19 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
promise<_Rp>::set_exception_at_thread_exit(exception_ptr __p)
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
2016-05-31 09:50:55 +08:00
|
|
|
_LIBCPP_ASSERT( __p != nullptr, "promise::set_exception_at_thread_exit: received nullptr" );
|
2010-08-28 04:10:19 +08:00
|
|
|
if (__state_ == nullptr)
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::no_state);
|
2010-08-28 04:10:19 +08:00
|
|
|
__state_->set_exception_at_thread_exit(__p);
|
|
|
|
}
|
|
|
|
|
|
|
|
// promise<R&>
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2017-05-05 01:08:54 +08:00
|
|
|
class _LIBCPP_TEMPLATE_VIS _LIBCPP_AVAILABILITY_FUTURE promise<_Rp&>
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
2011-11-30 02:15:50 +08:00
|
|
|
__assoc_state<_Rp&>* __state_;
|
2010-08-31 02:46:21 +08:00
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
explicit promise(nullptr_t) _NOEXCEPT : __state_(nullptr) {}
|
2010-08-31 02:46:21 +08:00
|
|
|
|
|
|
|
template <class> friend class packaged_task;
|
|
|
|
|
2010-08-28 04:10:19 +08:00
|
|
|
public:
|
|
|
|
promise();
|
|
|
|
template <class _Allocator>
|
|
|
|
promise(allocator_arg_t, const _Allocator& __a);
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
promise(promise&& __rhs) _NOEXCEPT
|
2010-08-28 04:10:19 +08:00
|
|
|
: __state_(__rhs.__state_) {__rhs.__state_ = nullptr;}
|
|
|
|
promise(const promise& __rhs) = delete;
|
|
|
|
~promise();
|
|
|
|
|
|
|
|
// assignment
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
promise& operator=(promise&& __rhs) _NOEXCEPT
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
[libc++] Consistently replace `std::` qualification with `_VSTD::` or nothing. NFCI.
I used a lot of `git grep` to find places where `std::` was being used
outside of comments and assert-messages. There were three outcomes:
- Qualified function calls, e.g. `std::move` becomes `_VSTD::move`.
This is the most common case.
- Typenames that don't need qualification, e.g. `std::allocator` becomes `allocator`.
Leaving these as `_VSTD::allocator` would also be fine, but I decided
that removing the qualification is more consistent with existing practice.
- Names that specifically need un-versioned `std::` qualification,
or that I wasn't sure about. For example, I didn't touch any code in
<atomic>, <math.h>, <new>, or any ext/ or experimental/ headers;
and I didn't touch any instances of `std::type_info`.
In some deduction guides, we were accidentally using `class Alloc = typename std::allocator<T>`,
despite `std::allocator<T>`'s type-ness not being template-dependent.
Because `std::allocator` is a qualified name, this did parse as we intended;
but what we meant was simply `class Alloc = allocator<T>`.
Differential Revision: https://reviews.llvm.org/D92250
2020-11-28 00:02:06 +08:00
|
|
|
promise(_VSTD::move(__rhs)).swap(*this);
|
2010-08-28 04:10:19 +08:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
promise& operator=(const promise& __rhs) = delete;
|
2020-07-30 21:42:23 +08:00
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
void swap(promise& __rhs) _NOEXCEPT {_VSTD::swap(__state_, __rhs.__state_);}
|
2010-08-28 04:10:19 +08:00
|
|
|
|
|
|
|
// retrieving the result
|
2011-11-30 02:15:50 +08:00
|
|
|
future<_Rp&> get_future();
|
2010-08-28 04:10:19 +08:00
|
|
|
|
|
|
|
// setting the result
|
2011-11-30 02:15:50 +08:00
|
|
|
void set_value(_Rp& __r);
|
2010-08-28 04:10:19 +08:00
|
|
|
void set_exception(exception_ptr __p);
|
|
|
|
|
|
|
|
// setting the result with deferred notification
|
2011-11-30 02:15:50 +08:00
|
|
|
void set_value_at_thread_exit(_Rp&);
|
2010-08-28 04:10:19 +08:00
|
|
|
void set_exception_at_thread_exit(exception_ptr __p);
|
|
|
|
};
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
|
|
|
promise<_Rp&>::promise()
|
|
|
|
: __state_(new __assoc_state<_Rp&>)
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2010-08-28 04:10:19 +08:00
|
|
|
template <class _Alloc>
|
2011-11-30 02:15:50 +08:00
|
|
|
promise<_Rp&>::promise(allocator_arg_t, const _Alloc& __a0)
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
2014-10-23 14:24:45 +08:00
|
|
|
typedef __assoc_state_alloc<_Rp&, _Alloc> _State;
|
|
|
|
typedef typename __allocator_traits_rebind<_Alloc, _State>::type _A2;
|
2010-08-28 04:10:19 +08:00
|
|
|
typedef __allocator_destructor<_A2> _D2;
|
|
|
|
_A2 __a(__a0);
|
2014-10-23 14:24:45 +08:00
|
|
|
unique_ptr<_State, _D2> __hold(__a.allocate(1), _D2(__a, 1));
|
2020-12-12 09:30:28 +08:00
|
|
|
::new ((void*)_VSTD::addressof(*__hold.get())) _State(__a0);
|
2014-10-23 14:24:45 +08:00
|
|
|
__state_ = _VSTD::addressof(*__hold.release());
|
2010-08-28 04:10:19 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
|
|
|
promise<_Rp&>::~promise()
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
|
|
|
if (__state_)
|
|
|
|
{
|
|
|
|
if (!__state_->__has_value() && __state_->use_count() > 1)
|
|
|
|
__state_->set_exception(make_exception_ptr(
|
|
|
|
future_error(make_error_code(future_errc::broken_promise))
|
|
|
|
));
|
|
|
|
__state_->__release_shared();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
|
|
|
future<_Rp&>
|
|
|
|
promise<_Rp&>::get_future()
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
|
|
|
if (__state_ == nullptr)
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::no_state);
|
2011-11-30 02:15:50 +08:00
|
|
|
return future<_Rp&>(__state_);
|
2010-08-28 04:10:19 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2010-08-28 04:10:19 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
promise<_Rp&>::set_value(_Rp& __r)
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
|
|
|
if (__state_ == nullptr)
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::no_state);
|
2010-08-28 04:10:19 +08:00
|
|
|
__state_->set_value(__r);
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2010-08-28 04:10:19 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
promise<_Rp&>::set_exception(exception_ptr __p)
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
2016-05-17 00:55:32 +08:00
|
|
|
_LIBCPP_ASSERT( __p != nullptr, "promise::set_exception: received nullptr" );
|
2010-08-28 04:10:19 +08:00
|
|
|
if (__state_ == nullptr)
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::no_state);
|
2010-08-28 04:10:19 +08:00
|
|
|
__state_->set_exception(__p);
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2010-08-28 04:10:19 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
promise<_Rp&>::set_value_at_thread_exit(_Rp& __r)
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
|
|
|
if (__state_ == nullptr)
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::no_state);
|
2010-08-28 04:10:19 +08:00
|
|
|
__state_->set_value_at_thread_exit(__r);
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2010-08-28 04:10:19 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
promise<_Rp&>::set_exception_at_thread_exit(exception_ptr __p)
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
2016-05-31 09:50:55 +08:00
|
|
|
_LIBCPP_ASSERT( __p != nullptr, "promise::set_exception_at_thread_exit: received nullptr" );
|
2010-08-28 04:10:19 +08:00
|
|
|
if (__state_ == nullptr)
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::no_state);
|
2010-08-28 04:10:19 +08:00
|
|
|
__state_->set_exception_at_thread_exit(__p);
|
|
|
|
}
|
|
|
|
|
|
|
|
// promise<void>
|
|
|
|
|
|
|
|
template <>
|
2017-05-05 01:08:54 +08:00
|
|
|
class _LIBCPP_TYPE_VIS _LIBCPP_AVAILABILITY_FUTURE promise<void>
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
|
|
|
__assoc_sub_state* __state_;
|
2010-08-31 02:46:21 +08:00
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
explicit promise(nullptr_t) _NOEXCEPT : __state_(nullptr) {}
|
2010-08-31 02:46:21 +08:00
|
|
|
|
|
|
|
template <class> friend class packaged_task;
|
|
|
|
|
2010-08-28 04:10:19 +08:00
|
|
|
public:
|
|
|
|
promise();
|
|
|
|
template <class _Allocator>
|
2017-03-02 11:22:18 +08:00
|
|
|
_LIBCPP_METHOD_TEMPLATE_IMPLICIT_INSTANTIATION_VIS
|
2010-08-28 04:10:19 +08:00
|
|
|
promise(allocator_arg_t, const _Allocator& __a);
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
promise(promise&& __rhs) _NOEXCEPT
|
2010-08-28 04:10:19 +08:00
|
|
|
: __state_(__rhs.__state_) {__rhs.__state_ = nullptr;}
|
|
|
|
promise(const promise& __rhs) = delete;
|
|
|
|
~promise();
|
|
|
|
|
|
|
|
// assignment
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
promise& operator=(promise&& __rhs) _NOEXCEPT
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
[libc++] Consistently replace `std::` qualification with `_VSTD::` or nothing. NFCI.
I used a lot of `git grep` to find places where `std::` was being used
outside of comments and assert-messages. There were three outcomes:
- Qualified function calls, e.g. `std::move` becomes `_VSTD::move`.
This is the most common case.
- Typenames that don't need qualification, e.g. `std::allocator` becomes `allocator`.
Leaving these as `_VSTD::allocator` would also be fine, but I decided
that removing the qualification is more consistent with existing practice.
- Names that specifically need un-versioned `std::` qualification,
or that I wasn't sure about. For example, I didn't touch any code in
<atomic>, <math.h>, <new>, or any ext/ or experimental/ headers;
and I didn't touch any instances of `std::type_info`.
In some deduction guides, we were accidentally using `class Alloc = typename std::allocator<T>`,
despite `std::allocator<T>`'s type-ness not being template-dependent.
Because `std::allocator` is a qualified name, this did parse as we intended;
but what we meant was simply `class Alloc = allocator<T>`.
Differential Revision: https://reviews.llvm.org/D92250
2020-11-28 00:02:06 +08:00
|
|
|
promise(_VSTD::move(__rhs)).swap(*this);
|
2010-08-28 04:10:19 +08:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
promise& operator=(const promise& __rhs) = delete;
|
2020-07-30 21:42:23 +08:00
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
void swap(promise& __rhs) _NOEXCEPT {_VSTD::swap(__state_, __rhs.__state_);}
|
2010-08-28 04:10:19 +08:00
|
|
|
|
|
|
|
// retrieving the result
|
|
|
|
future<void> get_future();
|
|
|
|
|
|
|
|
// setting the result
|
|
|
|
void set_value();
|
|
|
|
void set_exception(exception_ptr __p);
|
|
|
|
|
|
|
|
// setting the result with deferred notification
|
|
|
|
void set_value_at_thread_exit();
|
|
|
|
void set_exception_at_thread_exit(exception_ptr __p);
|
|
|
|
};
|
|
|
|
|
|
|
|
template <class _Alloc>
|
|
|
|
promise<void>::promise(allocator_arg_t, const _Alloc& __a0)
|
|
|
|
{
|
2014-10-23 14:24:45 +08:00
|
|
|
typedef __assoc_sub_state_alloc<_Alloc> _State;
|
|
|
|
typedef typename __allocator_traits_rebind<_Alloc, _State>::type _A2;
|
2010-08-28 04:10:19 +08:00
|
|
|
typedef __allocator_destructor<_A2> _D2;
|
|
|
|
_A2 __a(__a0);
|
2014-10-23 14:24:45 +08:00
|
|
|
unique_ptr<_State, _D2> __hold(__a.allocate(1), _D2(__a, 1));
|
2020-12-12 09:30:28 +08:00
|
|
|
::new ((void*)_VSTD::addressof(*__hold.get())) _State(__a0);
|
2014-10-23 14:24:45 +08:00
|
|
|
__state_ = _VSTD::addressof(*__hold.release());
|
2010-08-28 04:10:19 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2010-08-28 04:10:19 +08:00
|
|
|
inline _LIBCPP_INLINE_VISIBILITY
|
|
|
|
void
|
2012-07-22 01:46:55 +08:00
|
|
|
swap(promise<_Rp>& __x, promise<_Rp>& __y) _NOEXCEPT
|
2010-08-28 04:10:19 +08:00
|
|
|
{
|
|
|
|
__x.swap(__y);
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp, class _Alloc>
|
2017-01-05 07:56:00 +08:00
|
|
|
struct _LIBCPP_TEMPLATE_VIS uses_allocator<promise<_Rp>, _Alloc>
|
2010-09-22 22:16:26 +08:00
|
|
|
: public true_type {};
|
2010-08-28 04:10:19 +08:00
|
|
|
|
2010-08-31 02:46:21 +08:00
|
|
|
// packaged_task
|
|
|
|
|
|
|
|
template<class _Fp> class __packaged_task_base;
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template<class _Rp, class ..._ArgTypes>
|
2017-05-05 01:08:54 +08:00
|
|
|
class _LIBCPP_AVAILABILITY_FUTURE __packaged_task_base<_Rp(_ArgTypes...)>
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
|
|
|
__packaged_task_base(const __packaged_task_base&);
|
|
|
|
__packaged_task_base& operator=(const __packaged_task_base&);
|
|
|
|
public:
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-08-31 02:46:21 +08:00
|
|
|
__packaged_task_base() {}
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-08-31 02:46:21 +08:00
|
|
|
virtual ~__packaged_task_base() {}
|
2012-07-22 01:46:55 +08:00
|
|
|
virtual void __move_to(__packaged_task_base*) _NOEXCEPT = 0;
|
2010-08-31 02:46:21 +08:00
|
|
|
virtual void destroy() = 0;
|
|
|
|
virtual void destroy_deallocate() = 0;
|
2011-11-30 02:15:50 +08:00
|
|
|
virtual _Rp operator()(_ArgTypes&& ...) = 0;
|
2010-08-31 02:46:21 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
template<class _FD, class _Alloc, class _FB> class __packaged_task_func;
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template<class _Fp, class _Alloc, class _Rp, class ..._ArgTypes>
|
2017-05-05 01:08:54 +08:00
|
|
|
class _LIBCPP_AVAILABILITY_FUTURE __packaged_task_func<_Fp, _Alloc, _Rp(_ArgTypes...)>
|
2011-11-30 02:15:50 +08:00
|
|
|
: public __packaged_task_base<_Rp(_ArgTypes...)>
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
2011-11-30 02:15:50 +08:00
|
|
|
__compressed_pair<_Fp, _Alloc> __f_;
|
2010-08-31 02:46:21 +08:00
|
|
|
public:
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2019-12-17 07:23:39 +08:00
|
|
|
explicit __packaged_task_func(const _Fp& __f) : __f_(__f, __default_init_tag()) {}
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2019-12-17 07:23:39 +08:00
|
|
|
explicit __packaged_task_func(_Fp&& __f) : __f_(_VSTD::move(__f), __default_init_tag()) {}
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2011-11-30 02:15:50 +08:00
|
|
|
__packaged_task_func(const _Fp& __f, const _Alloc& __a)
|
2010-08-31 02:46:21 +08:00
|
|
|
: __f_(__f, __a) {}
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2011-11-30 02:15:50 +08:00
|
|
|
__packaged_task_func(_Fp&& __f, const _Alloc& __a)
|
2011-07-01 05:18:19 +08:00
|
|
|
: __f_(_VSTD::move(__f), __a) {}
|
2012-07-22 01:46:55 +08:00
|
|
|
virtual void __move_to(__packaged_task_base<_Rp(_ArgTypes...)>*) _NOEXCEPT;
|
2010-08-31 02:46:21 +08:00
|
|
|
virtual void destroy();
|
|
|
|
virtual void destroy_deallocate();
|
2011-11-30 02:15:50 +08:00
|
|
|
virtual _Rp operator()(_ArgTypes&& ... __args);
|
2010-08-31 02:46:21 +08:00
|
|
|
};
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template<class _Fp, class _Alloc, class _Rp, class ..._ArgTypes>
|
2010-08-31 02:46:21 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
__packaged_task_func<_Fp, _Alloc, _Rp(_ArgTypes...)>::__move_to(
|
2012-07-22 01:46:55 +08:00
|
|
|
__packaged_task_base<_Rp(_ArgTypes...)>* __p) _NOEXCEPT
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
2020-12-12 09:30:28 +08:00
|
|
|
::new ((void*)__p) __packaged_task_func(_VSTD::move(__f_.first()), _VSTD::move(__f_.second()));
|
2010-08-31 02:46:21 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template<class _Fp, class _Alloc, class _Rp, class ..._ArgTypes>
|
2010-08-31 02:46:21 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
__packaged_task_func<_Fp, _Alloc, _Rp(_ArgTypes...)>::destroy()
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
2011-11-30 02:15:50 +08:00
|
|
|
__f_.~__compressed_pair<_Fp, _Alloc>();
|
2010-08-31 02:46:21 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template<class _Fp, class _Alloc, class _Rp, class ..._ArgTypes>
|
2010-08-31 02:46:21 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
__packaged_task_func<_Fp, _Alloc, _Rp(_ArgTypes...)>::destroy_deallocate()
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
2014-10-23 14:24:45 +08:00
|
|
|
typedef typename __allocator_traits_rebind<_Alloc, __packaged_task_func>::type _Ap;
|
|
|
|
typedef allocator_traits<_Ap> _ATraits;
|
|
|
|
typedef pointer_traits<typename _ATraits::pointer> _PTraits;
|
2011-11-30 02:15:50 +08:00
|
|
|
_Ap __a(__f_.second());
|
|
|
|
__f_.~__compressed_pair<_Fp, _Alloc>();
|
2014-10-23 14:24:45 +08:00
|
|
|
__a.deallocate(_PTraits::pointer_to(*this), 1);
|
2010-08-31 02:46:21 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template<class _Fp, class _Alloc, class _Rp, class ..._ArgTypes>
|
|
|
|
_Rp
|
|
|
|
__packaged_task_func<_Fp, _Alloc, _Rp(_ArgTypes...)>::operator()(_ArgTypes&& ... __arg)
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
2020-12-16 08:32:29 +08:00
|
|
|
return _VSTD::__invoke(__f_.first(), _VSTD::forward<_ArgTypes>(__arg)...);
|
2010-08-31 02:46:21 +08:00
|
|
|
}
|
|
|
|
|
2011-06-15 03:58:17 +08:00
|
|
|
template <class _Callable> class __packaged_task_function;
|
2010-08-31 02:46:21 +08:00
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template<class _Rp, class ..._ArgTypes>
|
2017-05-05 01:08:54 +08:00
|
|
|
class _LIBCPP_AVAILABILITY_FUTURE __packaged_task_function<_Rp(_ArgTypes...)>
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
2011-11-30 02:15:50 +08:00
|
|
|
typedef __packaged_task_base<_Rp(_ArgTypes...)> __base;
|
2020-08-07 02:32:33 +08:00
|
|
|
|
|
|
|
_LIBCPP_INLINE_VISIBILITY _LIBCPP_NO_CFI
|
|
|
|
__base* __get_buf() { return (__base*)&__buf_; }
|
|
|
|
|
2013-01-22 01:26:55 +08:00
|
|
|
typename aligned_storage<3*sizeof(void*)>::type __buf_;
|
2010-08-31 02:46:21 +08:00
|
|
|
__base* __f_;
|
|
|
|
|
|
|
|
public:
|
2011-11-30 02:15:50 +08:00
|
|
|
typedef _Rp result_type;
|
2010-08-31 02:46:21 +08:00
|
|
|
|
|
|
|
// construct/copy/destroy:
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
__packaged_task_function() _NOEXCEPT : __f_(nullptr) {}
|
2011-11-30 02:15:50 +08:00
|
|
|
template<class _Fp>
|
|
|
|
__packaged_task_function(_Fp&& __f);
|
|
|
|
template<class _Fp, class _Alloc>
|
|
|
|
__packaged_task_function(allocator_arg_t, const _Alloc& __a, _Fp&& __f);
|
2010-08-31 02:46:21 +08:00
|
|
|
|
2012-07-22 01:46:55 +08:00
|
|
|
__packaged_task_function(__packaged_task_function&&) _NOEXCEPT;
|
|
|
|
__packaged_task_function& operator=(__packaged_task_function&&) _NOEXCEPT;
|
2010-08-31 02:46:21 +08:00
|
|
|
|
|
|
|
__packaged_task_function(const __packaged_task_function&) = delete;
|
|
|
|
__packaged_task_function& operator=(const __packaged_task_function&) = delete;
|
|
|
|
|
|
|
|
~__packaged_task_function();
|
|
|
|
|
2012-07-22 01:46:55 +08:00
|
|
|
void swap(__packaged_task_function&) _NOEXCEPT;
|
2010-08-31 02:46:21 +08:00
|
|
|
|
2015-11-07 09:22:13 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2011-11-30 02:15:50 +08:00
|
|
|
_Rp operator()(_ArgTypes...) const;
|
2010-08-31 02:46:21 +08:00
|
|
|
};
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template<class _Rp, class ..._ArgTypes>
|
2012-07-22 01:46:55 +08:00
|
|
|
__packaged_task_function<_Rp(_ArgTypes...)>::__packaged_task_function(__packaged_task_function&& __f) _NOEXCEPT
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
|
|
|
if (__f.__f_ == nullptr)
|
|
|
|
__f_ = nullptr;
|
2020-08-07 02:32:33 +08:00
|
|
|
else if (__f.__f_ == __f.__get_buf())
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
2020-08-07 02:32:33 +08:00
|
|
|
__f.__f_->__move_to(__get_buf());
|
2010-08-31 02:46:21 +08:00
|
|
|
__f_ = (__base*)&__buf_;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
__f_ = __f.__f_;
|
|
|
|
__f.__f_ = nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template<class _Rp, class ..._ArgTypes>
|
|
|
|
template <class _Fp>
|
|
|
|
__packaged_task_function<_Rp(_ArgTypes...)>::__packaged_task_function(_Fp&& __f)
|
2010-08-31 02:46:21 +08:00
|
|
|
: __f_(nullptr)
|
|
|
|
{
|
2014-04-07 21:32:26 +08:00
|
|
|
typedef typename remove_reference<typename decay<_Fp>::type>::type _FR;
|
2011-11-30 02:15:50 +08:00
|
|
|
typedef __packaged_task_func<_FR, allocator<_FR>, _Rp(_ArgTypes...)> _FF;
|
2010-08-31 02:46:21 +08:00
|
|
|
if (sizeof(_FF) <= sizeof(__buf_))
|
|
|
|
{
|
2020-12-12 09:30:28 +08:00
|
|
|
::new ((void*)&__buf_) _FF(_VSTD::forward<_Fp>(__f));
|
2010-08-31 02:46:21 +08:00
|
|
|
__f_ = (__base*)&__buf_;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2011-11-30 02:15:50 +08:00
|
|
|
typedef allocator<_FF> _Ap;
|
|
|
|
_Ap __a;
|
|
|
|
typedef __allocator_destructor<_Ap> _Dp;
|
|
|
|
unique_ptr<__base, _Dp> __hold(__a.allocate(1), _Dp(__a, 1));
|
2020-12-12 09:30:28 +08:00
|
|
|
::new ((void*)__hold.get()) _FF(_VSTD::forward<_Fp>(__f), allocator<_FR>(__a));
|
2010-08-31 02:46:21 +08:00
|
|
|
__f_ = __hold.release();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template<class _Rp, class ..._ArgTypes>
|
|
|
|
template <class _Fp, class _Alloc>
|
|
|
|
__packaged_task_function<_Rp(_ArgTypes...)>::__packaged_task_function(
|
|
|
|
allocator_arg_t, const _Alloc& __a0, _Fp&& __f)
|
2010-08-31 02:46:21 +08:00
|
|
|
: __f_(nullptr)
|
|
|
|
{
|
2014-04-07 21:32:26 +08:00
|
|
|
typedef typename remove_reference<typename decay<_Fp>::type>::type _FR;
|
2011-11-30 02:15:50 +08:00
|
|
|
typedef __packaged_task_func<_FR, _Alloc, _Rp(_ArgTypes...)> _FF;
|
2010-08-31 02:46:21 +08:00
|
|
|
if (sizeof(_FF) <= sizeof(__buf_))
|
|
|
|
{
|
|
|
|
__f_ = (__base*)&__buf_;
|
2020-12-12 09:30:28 +08:00
|
|
|
::new ((void*)__f_) _FF(_VSTD::forward<_Fp>(__f));
|
2010-08-31 02:46:21 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2014-10-23 14:24:45 +08:00
|
|
|
typedef typename __allocator_traits_rebind<_Alloc, _FF>::type _Ap;
|
2011-11-30 02:15:50 +08:00
|
|
|
_Ap __a(__a0);
|
|
|
|
typedef __allocator_destructor<_Ap> _Dp;
|
|
|
|
unique_ptr<__base, _Dp> __hold(__a.allocate(1), _Dp(__a, 1));
|
2020-12-12 09:30:28 +08:00
|
|
|
::new ((void*)_VSTD::addressof(*__hold.get()))
|
2014-10-23 14:24:45 +08:00
|
|
|
_FF(_VSTD::forward<_Fp>(__f), _Alloc(__a));
|
|
|
|
__f_ = _VSTD::addressof(*__hold.release());
|
2010-08-31 02:46:21 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template<class _Rp, class ..._ArgTypes>
|
|
|
|
__packaged_task_function<_Rp(_ArgTypes...)>&
|
2012-07-22 01:46:55 +08:00
|
|
|
__packaged_task_function<_Rp(_ArgTypes...)>::operator=(__packaged_task_function&& __f) _NOEXCEPT
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
2020-08-07 02:32:33 +08:00
|
|
|
if (__f_ == __get_buf())
|
2010-08-31 02:46:21 +08:00
|
|
|
__f_->destroy();
|
|
|
|
else if (__f_)
|
|
|
|
__f_->destroy_deallocate();
|
|
|
|
__f_ = nullptr;
|
|
|
|
if (__f.__f_ == nullptr)
|
|
|
|
__f_ = nullptr;
|
2020-08-07 02:32:33 +08:00
|
|
|
else if (__f.__f_ == __f.__get_buf())
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
2020-08-07 02:32:33 +08:00
|
|
|
__f.__f_->__move_to(__get_buf());
|
|
|
|
__f_ = __get_buf();
|
2010-08-31 02:46:21 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
__f_ = __f.__f_;
|
|
|
|
__f.__f_ = nullptr;
|
|
|
|
}
|
2012-10-13 10:03:45 +08:00
|
|
|
return *this;
|
2010-08-31 02:46:21 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template<class _Rp, class ..._ArgTypes>
|
|
|
|
__packaged_task_function<_Rp(_ArgTypes...)>::~__packaged_task_function()
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
2020-08-07 02:32:33 +08:00
|
|
|
if (__f_ == __get_buf())
|
2010-08-31 02:46:21 +08:00
|
|
|
__f_->destroy();
|
|
|
|
else if (__f_)
|
|
|
|
__f_->destroy_deallocate();
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template<class _Rp, class ..._ArgTypes>
|
2020-08-07 02:32:33 +08:00
|
|
|
_LIBCPP_NO_CFI
|
2010-08-31 02:46:21 +08:00
|
|
|
void
|
2012-07-22 01:46:55 +08:00
|
|
|
__packaged_task_function<_Rp(_ArgTypes...)>::swap(__packaged_task_function& __f) _NOEXCEPT
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
|
|
|
if (__f_ == (__base*)&__buf_ && __f.__f_ == (__base*)&__f.__buf_)
|
|
|
|
{
|
|
|
|
typename aligned_storage<sizeof(__buf_)>::type __tempbuf;
|
|
|
|
__base* __t = (__base*)&__tempbuf;
|
|
|
|
__f_->__move_to(__t);
|
|
|
|
__f_->destroy();
|
|
|
|
__f_ = nullptr;
|
|
|
|
__f.__f_->__move_to((__base*)&__buf_);
|
|
|
|
__f.__f_->destroy();
|
|
|
|
__f.__f_ = nullptr;
|
|
|
|
__f_ = (__base*)&__buf_;
|
|
|
|
__t->__move_to((__base*)&__f.__buf_);
|
|
|
|
__t->destroy();
|
|
|
|
__f.__f_ = (__base*)&__f.__buf_;
|
|
|
|
}
|
|
|
|
else if (__f_ == (__base*)&__buf_)
|
|
|
|
{
|
|
|
|
__f_->__move_to((__base*)&__f.__buf_);
|
|
|
|
__f_->destroy();
|
|
|
|
__f_ = __f.__f_;
|
|
|
|
__f.__f_ = (__base*)&__f.__buf_;
|
|
|
|
}
|
|
|
|
else if (__f.__f_ == (__base*)&__f.__buf_)
|
|
|
|
{
|
|
|
|
__f.__f_->__move_to((__base*)&__buf_);
|
|
|
|
__f.__f_->destroy();
|
|
|
|
__f.__f_ = __f_;
|
|
|
|
__f_ = (__base*)&__buf_;
|
|
|
|
}
|
|
|
|
else
|
2011-07-01 05:18:19 +08:00
|
|
|
_VSTD::swap(__f_, __f.__f_);
|
2010-08-31 02:46:21 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template<class _Rp, class ..._ArgTypes>
|
2015-11-07 09:22:13 +08:00
|
|
|
inline
|
2011-11-30 02:15:50 +08:00
|
|
|
_Rp
|
|
|
|
__packaged_task_function<_Rp(_ArgTypes...)>::operator()(_ArgTypes... __arg) const
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
2011-07-01 05:18:19 +08:00
|
|
|
return (*__f_)(_VSTD::forward<_ArgTypes>(__arg)...);
|
2010-08-31 02:46:21 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template<class _Rp, class ..._ArgTypes>
|
2017-05-05 01:08:54 +08:00
|
|
|
class _LIBCPP_TEMPLATE_VIS _LIBCPP_AVAILABILITY_FUTURE packaged_task<_Rp(_ArgTypes...)>
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
|
|
|
public:
|
2016-06-02 05:05:53 +08:00
|
|
|
typedef _Rp result_type; // extension
|
2010-08-31 02:46:21 +08:00
|
|
|
|
|
|
|
private:
|
|
|
|
__packaged_task_function<result_type(_ArgTypes...)> __f_;
|
|
|
|
promise<result_type> __p_;
|
|
|
|
|
|
|
|
public:
|
|
|
|
// construction and destruction
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
packaged_task() _NOEXCEPT : __p_(nullptr) {}
|
2013-10-13 06:49:17 +08:00
|
|
|
template <class _Fp,
|
2022-02-18 05:53:20 +08:00
|
|
|
class = __enable_if_t<!is_same<__uncvref_t<_Fp>, packaged_task>::value> >
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2011-11-30 02:15:50 +08:00
|
|
|
explicit packaged_task(_Fp&& __f) : __f_(_VSTD::forward<_Fp>(__f)) {}
|
2017-11-28 04:47:54 +08:00
|
|
|
template <class _Fp, class _Allocator,
|
2022-02-18 05:53:20 +08:00
|
|
|
class = __enable_if_t<!is_same<__uncvref_t<_Fp>, packaged_task>::value> >
|
2017-11-28 04:47:54 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
|
|
|
packaged_task(allocator_arg_t, const _Allocator& __a, _Fp&& __f)
|
|
|
|
: __f_(allocator_arg, __a, _VSTD::forward<_Fp>(__f)),
|
|
|
|
__p_(allocator_arg, __a) {}
|
2010-08-31 02:46:21 +08:00
|
|
|
// ~packaged_task() = default;
|
|
|
|
|
|
|
|
// no copy
|
2012-07-22 03:34:12 +08:00
|
|
|
packaged_task(const packaged_task&) = delete;
|
|
|
|
packaged_task& operator=(const packaged_task&) = delete;
|
2010-08-31 02:46:21 +08:00
|
|
|
|
|
|
|
// move support
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
packaged_task(packaged_task&& __other) _NOEXCEPT
|
2011-07-01 05:18:19 +08:00
|
|
|
: __f_(_VSTD::move(__other.__f_)), __p_(_VSTD::move(__other.__p_)) {}
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
packaged_task& operator=(packaged_task&& __other) _NOEXCEPT
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
2011-07-01 05:18:19 +08:00
|
|
|
__f_ = _VSTD::move(__other.__f_);
|
|
|
|
__p_ = _VSTD::move(__other.__p_);
|
2010-08-31 02:46:21 +08:00
|
|
|
return *this;
|
|
|
|
}
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
void swap(packaged_task& __other) _NOEXCEPT
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
|
|
|
__f_.swap(__other.__f_);
|
|
|
|
__p_.swap(__other.__p_);
|
|
|
|
}
|
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
bool valid() const _NOEXCEPT {return __p_.__state_ != nullptr;}
|
2010-08-31 02:46:21 +08:00
|
|
|
|
|
|
|
// result retrieval
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-08-31 02:46:21 +08:00
|
|
|
future<result_type> get_future() {return __p_.get_future();}
|
|
|
|
|
|
|
|
// execution
|
|
|
|
void operator()(_ArgTypes... __args);
|
|
|
|
void make_ready_at_thread_exit(_ArgTypes... __args);
|
|
|
|
|
|
|
|
void reset();
|
|
|
|
};
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template<class _Rp, class ..._ArgTypes>
|
2010-08-31 02:46:21 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
packaged_task<_Rp(_ArgTypes...)>::operator()(_ArgTypes... __args)
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
|
|
|
if (__p_.__state_ == nullptr)
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::no_state);
|
2010-08-31 02:46:21 +08:00
|
|
|
if (__p_.__state_->__has_value())
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::promise_already_satisfied);
|
2015-09-03 23:11:32 +08:00
|
|
|
#ifndef _LIBCPP_NO_EXCEPTIONS
|
2010-08-31 02:46:21 +08:00
|
|
|
try
|
|
|
|
{
|
2021-04-21 00:03:32 +08:00
|
|
|
#endif // _LIBCPP_NO_EXCEPTIONS
|
2011-07-01 05:18:19 +08:00
|
|
|
__p_.set_value(__f_(_VSTD::forward<_ArgTypes>(__args)...));
|
2010-08-31 02:46:21 +08:00
|
|
|
#ifndef _LIBCPP_NO_EXCEPTIONS
|
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
__p_.set_exception(current_exception());
|
|
|
|
}
|
2021-04-21 00:03:32 +08:00
|
|
|
#endif // _LIBCPP_NO_EXCEPTIONS
|
2010-08-31 02:46:21 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template<class _Rp, class ..._ArgTypes>
|
2010-08-31 02:46:21 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
packaged_task<_Rp(_ArgTypes...)>::make_ready_at_thread_exit(_ArgTypes... __args)
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
|
|
|
if (__p_.__state_ == nullptr)
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::no_state);
|
2010-08-31 02:46:21 +08:00
|
|
|
if (__p_.__state_->__has_value())
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::promise_already_satisfied);
|
2015-09-03 23:11:32 +08:00
|
|
|
#ifndef _LIBCPP_NO_EXCEPTIONS
|
2010-08-31 02:46:21 +08:00
|
|
|
try
|
|
|
|
{
|
2021-04-21 00:03:32 +08:00
|
|
|
#endif // _LIBCPP_NO_EXCEPTIONS
|
2011-07-01 05:18:19 +08:00
|
|
|
__p_.set_value_at_thread_exit(__f_(_VSTD::forward<_ArgTypes>(__args)...));
|
2010-08-31 02:46:21 +08:00
|
|
|
#ifndef _LIBCPP_NO_EXCEPTIONS
|
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
__p_.set_exception_at_thread_exit(current_exception());
|
|
|
|
}
|
2021-04-21 00:03:32 +08:00
|
|
|
#endif // _LIBCPP_NO_EXCEPTIONS
|
2010-08-31 02:46:21 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template<class _Rp, class ..._ArgTypes>
|
2010-08-31 02:46:21 +08:00
|
|
|
void
|
2011-11-30 02:15:50 +08:00
|
|
|
packaged_task<_Rp(_ArgTypes...)>::reset()
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
2010-12-01 04:23:32 +08:00
|
|
|
if (!valid())
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::no_state);
|
2010-08-31 02:46:21 +08:00
|
|
|
__p_ = promise<result_type>();
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class ..._ArgTypes>
|
2017-05-05 01:08:54 +08:00
|
|
|
class _LIBCPP_TEMPLATE_VIS _LIBCPP_AVAILABILITY_FUTURE packaged_task<void(_ArgTypes...)>
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
|
|
|
public:
|
2016-06-02 05:05:53 +08:00
|
|
|
typedef void result_type; // extension
|
2010-08-31 02:46:21 +08:00
|
|
|
|
|
|
|
private:
|
|
|
|
__packaged_task_function<result_type(_ArgTypes...)> __f_;
|
|
|
|
promise<result_type> __p_;
|
|
|
|
|
|
|
|
public:
|
|
|
|
// construction and destruction
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
packaged_task() _NOEXCEPT : __p_(nullptr) {}
|
2013-10-13 06:49:17 +08:00
|
|
|
template <class _Fp,
|
2022-02-18 05:53:20 +08:00
|
|
|
class = __enable_if_t<!is_same<__uncvref_t<_Fp>, packaged_task>::value> >
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2011-11-30 02:15:50 +08:00
|
|
|
explicit packaged_task(_Fp&& __f) : __f_(_VSTD::forward<_Fp>(__f)) {}
|
2017-11-28 04:47:54 +08:00
|
|
|
template <class _Fp, class _Allocator,
|
2022-02-18 05:53:20 +08:00
|
|
|
class = __enable_if_t<!is_same<__uncvref_t<_Fp>, packaged_task>::value> >
|
2017-11-28 04:47:54 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
|
|
|
packaged_task(allocator_arg_t, const _Allocator& __a, _Fp&& __f)
|
|
|
|
: __f_(allocator_arg, __a, _VSTD::forward<_Fp>(__f)),
|
|
|
|
__p_(allocator_arg, __a) {}
|
2010-08-31 02:46:21 +08:00
|
|
|
// ~packaged_task() = default;
|
|
|
|
|
|
|
|
// no copy
|
2012-07-22 03:34:12 +08:00
|
|
|
packaged_task(const packaged_task&) = delete;
|
|
|
|
packaged_task& operator=(const packaged_task&) = delete;
|
2010-08-31 02:46:21 +08:00
|
|
|
|
|
|
|
// move support
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
packaged_task(packaged_task&& __other) _NOEXCEPT
|
2011-07-01 05:18:19 +08:00
|
|
|
: __f_(_VSTD::move(__other.__f_)), __p_(_VSTD::move(__other.__p_)) {}
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
packaged_task& operator=(packaged_task&& __other) _NOEXCEPT
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
2011-07-01 05:18:19 +08:00
|
|
|
__f_ = _VSTD::move(__other.__f_);
|
|
|
|
__p_ = _VSTD::move(__other.__p_);
|
2010-08-31 02:46:21 +08:00
|
|
|
return *this;
|
|
|
|
}
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
void swap(packaged_task& __other) _NOEXCEPT
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
|
|
|
__f_.swap(__other.__f_);
|
|
|
|
__p_.swap(__other.__p_);
|
|
|
|
}
|
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
bool valid() const _NOEXCEPT {return __p_.__state_ != nullptr;}
|
2010-08-31 02:46:21 +08:00
|
|
|
|
|
|
|
// result retrieval
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-08-31 02:46:21 +08:00
|
|
|
future<result_type> get_future() {return __p_.get_future();}
|
|
|
|
|
|
|
|
// execution
|
|
|
|
void operator()(_ArgTypes... __args);
|
|
|
|
void make_ready_at_thread_exit(_ArgTypes... __args);
|
|
|
|
|
|
|
|
void reset();
|
|
|
|
};
|
|
|
|
|
|
|
|
template<class ..._ArgTypes>
|
|
|
|
void
|
|
|
|
packaged_task<void(_ArgTypes...)>::operator()(_ArgTypes... __args)
|
|
|
|
{
|
|
|
|
if (__p_.__state_ == nullptr)
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::no_state);
|
2010-08-31 02:46:21 +08:00
|
|
|
if (__p_.__state_->__has_value())
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::promise_already_satisfied);
|
2015-09-03 23:11:32 +08:00
|
|
|
#ifndef _LIBCPP_NO_EXCEPTIONS
|
2010-08-31 02:46:21 +08:00
|
|
|
try
|
|
|
|
{
|
2021-04-21 00:03:32 +08:00
|
|
|
#endif // _LIBCPP_NO_EXCEPTIONS
|
2011-07-01 05:18:19 +08:00
|
|
|
__f_(_VSTD::forward<_ArgTypes>(__args)...);
|
2010-08-31 02:46:21 +08:00
|
|
|
__p_.set_value();
|
|
|
|
#ifndef _LIBCPP_NO_EXCEPTIONS
|
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
__p_.set_exception(current_exception());
|
|
|
|
}
|
2021-04-21 00:03:32 +08:00
|
|
|
#endif // _LIBCPP_NO_EXCEPTIONS
|
2010-08-31 02:46:21 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
template<class ..._ArgTypes>
|
|
|
|
void
|
|
|
|
packaged_task<void(_ArgTypes...)>::make_ready_at_thread_exit(_ArgTypes... __args)
|
|
|
|
{
|
|
|
|
if (__p_.__state_ == nullptr)
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::no_state);
|
2010-08-31 02:46:21 +08:00
|
|
|
if (__p_.__state_->__has_value())
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::promise_already_satisfied);
|
2015-09-03 23:11:32 +08:00
|
|
|
#ifndef _LIBCPP_NO_EXCEPTIONS
|
2010-08-31 02:46:21 +08:00
|
|
|
try
|
|
|
|
{
|
2021-04-21 00:03:32 +08:00
|
|
|
#endif // _LIBCPP_NO_EXCEPTIONS
|
2011-07-01 05:18:19 +08:00
|
|
|
__f_(_VSTD::forward<_ArgTypes>(__args)...);
|
2010-08-31 02:46:21 +08:00
|
|
|
__p_.set_value_at_thread_exit();
|
|
|
|
#ifndef _LIBCPP_NO_EXCEPTIONS
|
|
|
|
}
|
|
|
|
catch (...)
|
|
|
|
{
|
|
|
|
__p_.set_exception_at_thread_exit(current_exception());
|
|
|
|
}
|
2021-04-21 00:03:32 +08:00
|
|
|
#endif // _LIBCPP_NO_EXCEPTIONS
|
2010-08-31 02:46:21 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
template<class ..._ArgTypes>
|
|
|
|
void
|
|
|
|
packaged_task<void(_ArgTypes...)>::reset()
|
|
|
|
{
|
2010-12-01 04:23:32 +08:00
|
|
|
if (!valid())
|
2015-10-03 05:25:15 +08:00
|
|
|
__throw_future_error(future_errc::no_state);
|
2010-08-31 02:46:21 +08:00
|
|
|
__p_ = promise<result_type>();
|
|
|
|
}
|
|
|
|
|
2021-03-25 06:31:58 +08:00
|
|
|
template <class _Rp, class... _ArgTypes>
|
2010-08-31 02:46:21 +08:00
|
|
|
inline _LIBCPP_INLINE_VISIBILITY
|
|
|
|
void
|
2021-03-25 06:31:58 +08:00
|
|
|
swap(packaged_task<_Rp(_ArgTypes...)>& __x, packaged_task<_Rp(_ArgTypes...)>& __y) _NOEXCEPT
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
|
|
|
__x.swap(__y);
|
|
|
|
}
|
|
|
|
|
2017-11-28 04:47:54 +08:00
|
|
|
template <class _Callable, class _Alloc>
|
|
|
|
struct _LIBCPP_TEMPLATE_VIS uses_allocator<packaged_task<_Callable>, _Alloc>
|
|
|
|
: public true_type {};
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp, class _Fp>
|
2019-12-11 07:00:42 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY future<_Rp>
|
2011-11-30 02:15:50 +08:00
|
|
|
__make_deferred_assoc_state(_Fp&& __f)
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
2011-11-30 02:15:50 +08:00
|
|
|
unique_ptr<__deferred_assoc_state<_Rp, _Fp>, __release_shared_count>
|
|
|
|
__h(new __deferred_assoc_state<_Rp, _Fp>(_VSTD::forward<_Fp>(__f)));
|
|
|
|
return future<_Rp>(__h.get());
|
2010-08-31 02:46:21 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp, class _Fp>
|
2019-12-11 07:00:42 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY future<_Rp>
|
2011-11-30 02:15:50 +08:00
|
|
|
__make_async_assoc_state(_Fp&& __f)
|
2011-05-19 23:05:04 +08:00
|
|
|
{
|
2011-11-30 02:15:50 +08:00
|
|
|
unique_ptr<__async_assoc_state<_Rp, _Fp>, __release_shared_count>
|
|
|
|
__h(new __async_assoc_state<_Rp, _Fp>(_VSTD::forward<_Fp>(__f)));
|
|
|
|
_VSTD::thread(&__async_assoc_state<_Rp, _Fp>::__execute, __h.get()).detach();
|
|
|
|
return future<_Rp>(__h.get());
|
2011-05-19 23:05:04 +08:00
|
|
|
}
|
|
|
|
|
2020-07-30 22:00:53 +08:00
|
|
|
#ifndef _LIBCPP_CXX03_LANG
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Fp, class... _Args>
|
2019-12-11 07:00:42 +08:00
|
|
|
class _LIBCPP_HIDDEN __async_func
|
2011-05-19 23:05:04 +08:00
|
|
|
{
|
2011-11-30 02:15:50 +08:00
|
|
|
tuple<_Fp, _Args...> __f_;
|
2011-05-19 23:05:04 +08:00
|
|
|
|
|
|
|
public:
|
2011-11-30 02:15:50 +08:00
|
|
|
typedef typename __invoke_of<_Fp, _Args...>::type _Rp;
|
2011-05-19 23:05:04 +08:00
|
|
|
|
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2011-11-30 02:15:50 +08:00
|
|
|
explicit __async_func(_Fp&& __f, _Args&&... __args)
|
2011-07-01 05:18:19 +08:00
|
|
|
: __f_(_VSTD::move(__f), _VSTD::move(__args)...) {}
|
2011-05-19 23:05:04 +08:00
|
|
|
|
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2011-07-01 05:18:19 +08:00
|
|
|
__async_func(__async_func&& __f) : __f_(_VSTD::move(__f.__f_)) {}
|
2011-05-19 23:05:04 +08:00
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
_Rp operator()()
|
2011-05-19 23:05:04 +08:00
|
|
|
{
|
|
|
|
typedef typename __make_tuple_indices<1+sizeof...(_Args), 1>::type _Index;
|
|
|
|
return __execute(_Index());
|
|
|
|
}
|
|
|
|
private:
|
|
|
|
template <size_t ..._Indices>
|
2011-11-30 02:15:50 +08:00
|
|
|
_Rp
|
2011-05-19 23:05:04 +08:00
|
|
|
__execute(__tuple_indices<_Indices...>)
|
|
|
|
{
|
2020-12-16 08:32:29 +08:00
|
|
|
return _VSTD::__invoke(_VSTD::move(_VSTD::get<0>(__f_)), _VSTD::move(_VSTD::get<_Indices>(__f_))...);
|
2011-05-19 23:05:04 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2013-11-04 06:06:53 +08:00
|
|
|
inline _LIBCPP_INLINE_VISIBILITY bool __does_policy_contain(launch __policy, launch __value )
|
2013-11-03 23:43:35 +08:00
|
|
|
{ return (int(__policy) & int(__value)) != 0; }
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Fp, class... _Args>
|
2017-11-23 09:25:03 +08:00
|
|
|
_LIBCPP_NODISCARD_AFTER_CXX17
|
2011-11-30 02:15:50 +08:00
|
|
|
future<typename __invoke_of<typename decay<_Fp>::type, typename decay<_Args>::type...>::type>
|
|
|
|
async(launch __policy, _Fp&& __f, _Args&&... __args)
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
2011-11-30 02:15:50 +08:00
|
|
|
typedef __async_func<typename decay<_Fp>::type, typename decay<_Args>::type...> _BF;
|
|
|
|
typedef typename _BF::_Rp _Rp;
|
2013-11-03 23:43:35 +08:00
|
|
|
|
|
|
|
#ifndef _LIBCPP_NO_EXCEPTIONS
|
|
|
|
try
|
|
|
|
{
|
|
|
|
#endif
|
|
|
|
if (__does_policy_contain(__policy, launch::async))
|
2021-12-14 08:21:38 +08:00
|
|
|
return _VSTD::__make_async_assoc_state<_Rp>(_BF(_LIBCPP_AUTO_CAST(_VSTD::forward<_Fp>(__f)),
|
|
|
|
_LIBCPP_AUTO_CAST(_VSTD::forward<_Args>(__args))...));
|
2013-11-03 23:43:35 +08:00
|
|
|
#ifndef _LIBCPP_NO_EXCEPTIONS
|
|
|
|
}
|
|
|
|
catch ( ... ) { if (__policy == launch::async) throw ; }
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (__does_policy_contain(__policy, launch::deferred))
|
2021-12-14 08:21:38 +08:00
|
|
|
return _VSTD::__make_deferred_assoc_state<_Rp>(_BF(_LIBCPP_AUTO_CAST(_VSTD::forward<_Fp>(__f)),
|
|
|
|
_LIBCPP_AUTO_CAST(_VSTD::forward<_Args>(__args))...));
|
2013-11-03 23:43:35 +08:00
|
|
|
return future<_Rp>{};
|
2010-08-31 02:46:21 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Fp, class... _Args>
|
2017-11-23 09:25:03 +08:00
|
|
|
_LIBCPP_NODISCARD_AFTER_CXX17 inline _LIBCPP_INLINE_VISIBILITY
|
2011-11-30 02:15:50 +08:00
|
|
|
future<typename __invoke_of<typename decay<_Fp>::type, typename decay<_Args>::type...>::type>
|
|
|
|
async(_Fp&& __f, _Args&&... __args)
|
2010-08-31 02:46:21 +08:00
|
|
|
{
|
2011-11-30 02:15:50 +08:00
|
|
|
return _VSTD::async(launch::any, _VSTD::forward<_Fp>(__f),
|
2011-07-01 05:18:19 +08:00
|
|
|
_VSTD::forward<_Args>(__args)...);
|
2010-08-31 02:46:21 +08:00
|
|
|
}
|
|
|
|
|
2020-07-30 22:00:53 +08:00
|
|
|
#endif // C++03
|
2010-08-31 02:46:21 +08:00
|
|
|
|
2010-09-04 05:46:37 +08:00
|
|
|
// shared_future
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2017-01-05 07:56:00 +08:00
|
|
|
class _LIBCPP_TEMPLATE_VIS shared_future
|
2010-09-04 02:39:25 +08:00
|
|
|
{
|
2011-11-30 02:15:50 +08:00
|
|
|
__assoc_state<_Rp>* __state_;
|
2010-09-04 02:39:25 +08:00
|
|
|
|
|
|
|
public:
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
shared_future() _NOEXCEPT : __state_(nullptr) {}
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2016-11-15 03:58:05 +08:00
|
|
|
shared_future(const shared_future& __rhs) _NOEXCEPT : __state_(__rhs.__state_)
|
2010-09-04 02:39:25 +08:00
|
|
|
{if (__state_) __state_->__add_shared();}
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
shared_future(future<_Rp>&& __f) _NOEXCEPT : __state_(__f.__state_)
|
2010-09-04 02:39:25 +08:00
|
|
|
{__f.__state_ = nullptr;}
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
shared_future(shared_future&& __rhs) _NOEXCEPT : __state_(__rhs.__state_)
|
2010-09-04 02:39:25 +08:00
|
|
|
{__rhs.__state_ = nullptr;}
|
|
|
|
~shared_future();
|
2016-11-15 03:58:05 +08:00
|
|
|
shared_future& operator=(const shared_future& __rhs) _NOEXCEPT;
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
shared_future& operator=(shared_future&& __rhs) _NOEXCEPT
|
2010-09-04 02:39:25 +08:00
|
|
|
{
|
[libc++] Consistently replace `std::` qualification with `_VSTD::` or nothing. NFCI.
I used a lot of `git grep` to find places where `std::` was being used
outside of comments and assert-messages. There were three outcomes:
- Qualified function calls, e.g. `std::move` becomes `_VSTD::move`.
This is the most common case.
- Typenames that don't need qualification, e.g. `std::allocator` becomes `allocator`.
Leaving these as `_VSTD::allocator` would also be fine, but I decided
that removing the qualification is more consistent with existing practice.
- Names that specifically need un-versioned `std::` qualification,
or that I wasn't sure about. For example, I didn't touch any code in
<atomic>, <math.h>, <new>, or any ext/ or experimental/ headers;
and I didn't touch any instances of `std::type_info`.
In some deduction guides, we were accidentally using `class Alloc = typename std::allocator<T>`,
despite `std::allocator<T>`'s type-ness not being template-dependent.
Because `std::allocator` is a qualified name, this did parse as we intended;
but what we meant was simply `class Alloc = allocator<T>`.
Differential Revision: https://reviews.llvm.org/D92250
2020-11-28 00:02:06 +08:00
|
|
|
shared_future(_VSTD::move(__rhs)).swap(*this);
|
2010-09-04 02:39:25 +08:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
// retrieving the value
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2011-11-30 02:15:50 +08:00
|
|
|
const _Rp& get() const {return __state_->copy();}
|
2010-09-04 02:39:25 +08:00
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
void swap(shared_future& __rhs) _NOEXCEPT {_VSTD::swap(__state_, __rhs.__state_);}
|
2010-09-04 02:39:25 +08:00
|
|
|
|
|
|
|
// functions to check state
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
bool valid() const _NOEXCEPT {return __state_ != nullptr;}
|
2010-09-04 02:39:25 +08:00
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-09-04 02:39:25 +08:00
|
|
|
void wait() const {__state_->wait();}
|
|
|
|
template <class _Rep, class _Period>
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-09-04 02:39:25 +08:00
|
|
|
future_status
|
|
|
|
wait_for(const chrono::duration<_Rep, _Period>& __rel_time) const
|
|
|
|
{return __state_->wait_for(__rel_time);}
|
|
|
|
template <class _Clock, class _Duration>
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-09-04 02:39:25 +08:00
|
|
|
future_status
|
|
|
|
wait_until(const chrono::time_point<_Clock, _Duration>& __abs_time) const
|
|
|
|
{return __state_->wait_until(__abs_time);}
|
|
|
|
};
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
|
|
|
shared_future<_Rp>::~shared_future()
|
2010-09-04 02:39:25 +08:00
|
|
|
{
|
|
|
|
if (__state_)
|
|
|
|
__state_->__release_shared();
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
|
|
|
shared_future<_Rp>&
|
2016-11-15 03:58:05 +08:00
|
|
|
shared_future<_Rp>::operator=(const shared_future& __rhs) _NOEXCEPT
|
2010-09-04 02:39:25 +08:00
|
|
|
{
|
|
|
|
if (__rhs.__state_)
|
|
|
|
__rhs.__state_->__add_shared();
|
|
|
|
if (__state_)
|
|
|
|
__state_->__release_shared();
|
|
|
|
__state_ = __rhs.__state_;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2017-01-05 07:56:00 +08:00
|
|
|
class _LIBCPP_TEMPLATE_VIS shared_future<_Rp&>
|
2010-09-04 02:39:25 +08:00
|
|
|
{
|
2011-11-30 02:15:50 +08:00
|
|
|
__assoc_state<_Rp&>* __state_;
|
2010-09-04 02:39:25 +08:00
|
|
|
|
|
|
|
public:
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
shared_future() _NOEXCEPT : __state_(nullptr) {}
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-09-04 02:39:25 +08:00
|
|
|
shared_future(const shared_future& __rhs) : __state_(__rhs.__state_)
|
|
|
|
{if (__state_) __state_->__add_shared();}
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
shared_future(future<_Rp&>&& __f) _NOEXCEPT : __state_(__f.__state_)
|
2010-09-04 02:39:25 +08:00
|
|
|
{__f.__state_ = nullptr;}
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
shared_future(shared_future&& __rhs) _NOEXCEPT : __state_(__rhs.__state_)
|
2010-09-04 02:39:25 +08:00
|
|
|
{__rhs.__state_ = nullptr;}
|
|
|
|
~shared_future();
|
|
|
|
shared_future& operator=(const shared_future& __rhs);
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
shared_future& operator=(shared_future&& __rhs) _NOEXCEPT
|
2010-09-04 02:39:25 +08:00
|
|
|
{
|
[libc++] Consistently replace `std::` qualification with `_VSTD::` or nothing. NFCI.
I used a lot of `git grep` to find places where `std::` was being used
outside of comments and assert-messages. There were three outcomes:
- Qualified function calls, e.g. `std::move` becomes `_VSTD::move`.
This is the most common case.
- Typenames that don't need qualification, e.g. `std::allocator` becomes `allocator`.
Leaving these as `_VSTD::allocator` would also be fine, but I decided
that removing the qualification is more consistent with existing practice.
- Names that specifically need un-versioned `std::` qualification,
or that I wasn't sure about. For example, I didn't touch any code in
<atomic>, <math.h>, <new>, or any ext/ or experimental/ headers;
and I didn't touch any instances of `std::type_info`.
In some deduction guides, we were accidentally using `class Alloc = typename std::allocator<T>`,
despite `std::allocator<T>`'s type-ness not being template-dependent.
Because `std::allocator` is a qualified name, this did parse as we intended;
but what we meant was simply `class Alloc = allocator<T>`.
Differential Revision: https://reviews.llvm.org/D92250
2020-11-28 00:02:06 +08:00
|
|
|
shared_future(_VSTD::move(__rhs)).swap(*this);
|
2010-09-04 02:39:25 +08:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
// retrieving the value
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2011-11-30 02:15:50 +08:00
|
|
|
_Rp& get() const {return __state_->copy();}
|
2010-09-04 02:39:25 +08:00
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
void swap(shared_future& __rhs) _NOEXCEPT {_VSTD::swap(__state_, __rhs.__state_);}
|
2010-09-04 02:39:25 +08:00
|
|
|
|
|
|
|
// functions to check state
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
bool valid() const _NOEXCEPT {return __state_ != nullptr;}
|
2010-09-04 02:39:25 +08:00
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-09-04 02:39:25 +08:00
|
|
|
void wait() const {__state_->wait();}
|
|
|
|
template <class _Rep, class _Period>
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-09-04 02:39:25 +08:00
|
|
|
future_status
|
|
|
|
wait_for(const chrono::duration<_Rep, _Period>& __rel_time) const
|
|
|
|
{return __state_->wait_for(__rel_time);}
|
|
|
|
template <class _Clock, class _Duration>
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-09-04 02:39:25 +08:00
|
|
|
future_status
|
|
|
|
wait_until(const chrono::time_point<_Clock, _Duration>& __abs_time) const
|
|
|
|
{return __state_->wait_until(__abs_time);}
|
|
|
|
};
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
|
|
|
shared_future<_Rp&>::~shared_future()
|
2010-09-04 02:39:25 +08:00
|
|
|
{
|
|
|
|
if (__state_)
|
|
|
|
__state_->__release_shared();
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
|
|
|
shared_future<_Rp&>&
|
|
|
|
shared_future<_Rp&>::operator=(const shared_future& __rhs)
|
2010-09-04 02:39:25 +08:00
|
|
|
{
|
|
|
|
if (__rhs.__state_)
|
|
|
|
__rhs.__state_->__add_shared();
|
|
|
|
if (__state_)
|
|
|
|
__state_->__release_shared();
|
|
|
|
__state_ = __rhs.__state_;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <>
|
2017-05-05 01:08:54 +08:00
|
|
|
class _LIBCPP_TYPE_VIS _LIBCPP_AVAILABILITY_FUTURE shared_future<void>
|
2010-09-04 02:39:25 +08:00
|
|
|
{
|
|
|
|
__assoc_sub_state* __state_;
|
|
|
|
|
|
|
|
public:
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
shared_future() _NOEXCEPT : __state_(nullptr) {}
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-09-04 02:39:25 +08:00
|
|
|
shared_future(const shared_future& __rhs) : __state_(__rhs.__state_)
|
|
|
|
{if (__state_) __state_->__add_shared();}
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
shared_future(future<void>&& __f) _NOEXCEPT : __state_(__f.__state_)
|
2010-09-04 02:39:25 +08:00
|
|
|
{__f.__state_ = nullptr;}
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
shared_future(shared_future&& __rhs) _NOEXCEPT : __state_(__rhs.__state_)
|
2010-09-04 02:39:25 +08:00
|
|
|
{__rhs.__state_ = nullptr;}
|
|
|
|
~shared_future();
|
|
|
|
shared_future& operator=(const shared_future& __rhs);
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
shared_future& operator=(shared_future&& __rhs) _NOEXCEPT
|
2010-09-04 02:39:25 +08:00
|
|
|
{
|
[libc++] Consistently replace `std::` qualification with `_VSTD::` or nothing. NFCI.
I used a lot of `git grep` to find places where `std::` was being used
outside of comments and assert-messages. There were three outcomes:
- Qualified function calls, e.g. `std::move` becomes `_VSTD::move`.
This is the most common case.
- Typenames that don't need qualification, e.g. `std::allocator` becomes `allocator`.
Leaving these as `_VSTD::allocator` would also be fine, but I decided
that removing the qualification is more consistent with existing practice.
- Names that specifically need un-versioned `std::` qualification,
or that I wasn't sure about. For example, I didn't touch any code in
<atomic>, <math.h>, <new>, or any ext/ or experimental/ headers;
and I didn't touch any instances of `std::type_info`.
In some deduction guides, we were accidentally using `class Alloc = typename std::allocator<T>`,
despite `std::allocator<T>`'s type-ness not being template-dependent.
Because `std::allocator` is a qualified name, this did parse as we intended;
but what we meant was simply `class Alloc = allocator<T>`.
Differential Revision: https://reviews.llvm.org/D92250
2020-11-28 00:02:06 +08:00
|
|
|
shared_future(_VSTD::move(__rhs)).swap(*this);
|
2010-09-04 02:39:25 +08:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
// retrieving the value
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-09-04 02:39:25 +08:00
|
|
|
void get() const {__state_->copy();}
|
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
void swap(shared_future& __rhs) _NOEXCEPT {_VSTD::swap(__state_, __rhs.__state_);}
|
2010-09-04 02:39:25 +08:00
|
|
|
|
|
|
|
// functions to check state
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2012-07-22 01:46:55 +08:00
|
|
|
bool valid() const _NOEXCEPT {return __state_ != nullptr;}
|
2010-09-04 02:39:25 +08:00
|
|
|
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-09-04 02:39:25 +08:00
|
|
|
void wait() const {__state_->wait();}
|
|
|
|
template <class _Rep, class _Period>
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-09-04 02:39:25 +08:00
|
|
|
future_status
|
|
|
|
wait_for(const chrono::duration<_Rep, _Period>& __rel_time) const
|
|
|
|
{return __state_->wait_for(__rel_time);}
|
|
|
|
template <class _Clock, class _Duration>
|
2010-09-22 22:16:26 +08:00
|
|
|
_LIBCPP_INLINE_VISIBILITY
|
2010-09-04 02:39:25 +08:00
|
|
|
future_status
|
|
|
|
wait_until(const chrono::time_point<_Clock, _Duration>& __abs_time) const
|
|
|
|
{return __state_->wait_until(__abs_time);}
|
|
|
|
};
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2010-09-04 02:39:25 +08:00
|
|
|
inline _LIBCPP_INLINE_VISIBILITY
|
|
|
|
void
|
2012-07-22 01:46:55 +08:00
|
|
|
swap(shared_future<_Rp>& __x, shared_future<_Rp>& __y) _NOEXCEPT
|
2010-09-04 02:39:25 +08:00
|
|
|
{
|
|
|
|
__x.swap(__y);
|
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2015-11-07 09:22:13 +08:00
|
|
|
inline
|
2011-11-30 02:15:50 +08:00
|
|
|
shared_future<_Rp>
|
2017-01-25 07:28:25 +08:00
|
|
|
future<_Rp>::share() _NOEXCEPT
|
2010-09-04 05:46:37 +08:00
|
|
|
{
|
2011-11-30 02:15:50 +08:00
|
|
|
return shared_future<_Rp>(_VSTD::move(*this));
|
2010-09-04 05:46:37 +08:00
|
|
|
}
|
|
|
|
|
2011-11-30 02:15:50 +08:00
|
|
|
template <class _Rp>
|
2015-11-07 09:22:13 +08:00
|
|
|
inline
|
2011-11-30 02:15:50 +08:00
|
|
|
shared_future<_Rp&>
|
2017-01-25 07:28:25 +08:00
|
|
|
future<_Rp&>::share() _NOEXCEPT
|
2010-09-04 05:46:37 +08:00
|
|
|
{
|
2011-11-30 02:15:50 +08:00
|
|
|
return shared_future<_Rp&>(_VSTD::move(*this));
|
2010-09-04 05:46:37 +08:00
|
|
|
}
|
|
|
|
|
2015-11-07 09:22:13 +08:00
|
|
|
inline
|
2010-12-01 04:23:32 +08:00
|
|
|
shared_future<void>
|
2017-01-25 07:28:25 +08:00
|
|
|
future<void>::share() _NOEXCEPT
|
2010-09-04 05:46:37 +08:00
|
|
|
{
|
2011-07-01 05:18:19 +08:00
|
|
|
return shared_future<void>(_VSTD::move(*this));
|
2010-09-04 05:46:37 +08:00
|
|
|
}
|
|
|
|
|
2010-05-12 03:42:16 +08:00
|
|
|
_LIBCPP_END_NAMESPACE_STD
|
|
|
|
|
2021-04-21 00:03:32 +08:00
|
|
|
#endif // _LIBCPP_FUTURE
|