Skip to content

Latest commit

 

History

History
637 lines (522 loc) · 35.2 KB

coroutines.md

File metadata and controls

637 lines (522 loc) · 35.2 KB

コルーチン [P0912R5]

  • cpp20[meta cpp]

このページはC++20に採用された言語機能の変更を解説しています。

のちのC++規格でさらに変更される場合があるため関連項目を参照してください。

概要

処理途中でのサスペンド(中断)/レジューム(再開)をサポートする一般化された関数として、コルーチンが導入される。

C++20時点では、コルーチン動作に関する言語仕様と新キーワードco_return, co_await, co_yieldによる新しい構文、コルーチンライブラリ実装者向けの低レベルライブラリ<coroutine>のみが規定される。

// コルーチンiotaを定義
my_generator iota(int end)
{
  for (int n = 0; n < end; ++n) {
    co_yield n;
  }
}

// コルーチンの呼び出し
auto g = iota(10);
for (int v: g) {
  std::cout << v;
}

// "my_generator"はライブラリが提供するべきクラス。
// 動作可能なサンプルコード全体は後述例を参照のこと。
  • co_yield[color ff0000]

一般的なアプリケーション実装者からの利用を想定した、ジェネレータや非同期タスク・非同期I/Oといったハイレベルなコルーチンライブラリは、C++23以降での導入にむけて検討されている。

C++23ではジェネレータコルーチンを実現する<generator>が追加された。

特徴

C++コルーチンの特徴は次の通り:

  • 関数からコルーチンへの拡張: 従来からある関数(function)の呼出し(call)と復帰(return)に加えて、コルーチン(coroutine)では中断(suspend)と再開(resume)動作をサポートする。また中断状態のまま再開不要となったコルーチンに対しては、リソースリークを防ぐため明示的に破棄(destroy)を行える。
  • 多数の カスタマイズポイント: コルーチンライブラリ実装者向けに、コルーチン動作の制御を可能とするカスタマイズポイントを規定する。後述するPromise、Awaitable、Awaiterなど。
  • 軽量な スタックレス(Stackless)コルーチン: コルーチンの中断は実行中コルーチンのレキシカル・スコープ内でのみで許可され、コルーチンが呼び出した関数内では中断操作を行えない。(C++コルーチンの定義上、co_awaitco_yieldを用いて中断処理を記述すると、関数ではなくコルーチンとみなされる。)
  • コルーチン毎の 動的メモリ確保: コルーチン実引数の保持や進行状況を管理するため、動的メモリ確保が行われる可能性がある。ただし一定の条件を満たす場合には、C++コンパイラ最適化により動的メモリ確保は省略されると期待できる。
  • 非対称(Asymmetric)・対称(Symmetric)コルーチン: 中断処理によりコルーチン再開元へ制御を戻す非対称コルーチンのほか、明示的に別コルーチンの再開に制御を移す対称コルーチンをサポートする。待機動作をカスタマイズするAwaiterオブジェクトawait_suspendにて制御する。
  • スレッド(thread)との直交: あるスレッド上で実行されるコルーチンを中断し、その後に別スレッドから同コルーチンを再開させることもできる。ただしスレッドローカルストレージと組合せには注意が必要。

動作概略

C++コルーチン動作理解の助けとなるよう、ここでは細部を省略した説明を行う。 コルーチン実行の基本動作は、それぞれ下記ように説明される:

  • 呼出し: 通常の関数と同様に、カッコ(())を用いた関数呼び出し構文を用いる。コルーチンに対応するユーザ定義Promiseオブジェクトが自動的に生成されるため、そこからコルーチンハンドルを取得する。
  • 中断: コルーチン本体にてco_yield式またはco_await式を記述する。中断されたコルーチンから呼出元へは、コルーチンハンドルを内包するコルーチン戻り値型オブジェクトを返す。
  • 再開: コルーチン中断により返された戻り値型オブジェクトを介して、コルーチンハンドルの再開関数resumeを呼び出す。
  • 復帰: コルーチン本体にてco_return文を記述、またはコルーチン本体終端まで到達する。全てのローカル変数とPromiseオブジェクトは破棄される。

この4種類の基本動作に対して、次のカスタマイズポイントが提供される。カッコ内はコルーチンライブラリが実装すべきカスタマイズポイント名:

  • コルーチン呼出し直後の動作: 初期サスペンドポイント(initial_suspend)にて、コルーチン本体の開始前に中断して戻り値型オブジェクトを返すか、そのままコルーチン本体を実行継続するかを制御する。
  • コルーチン復帰直前の動作: 最終サスペンドポイント(final_suspend)にて、コルーチンを最後に中断して戻り値型オブジェクトを返すか、そのままコルーチンに関するリソースを破棄するかを制御する。前者を選択した場合、リソースリークを防ぐためコルーチンハンドルの破棄関数destroy呼出しが必要となる。
  • 値を伴うコルーチン中断: co_yield式により、コルーチンを中断すると同時に呼出元へ値を返す(yield_value)。
  • 値を伴うコルーチン復帰: co_return文により、呼出元へ値を返す(return_value)。
  • コルーチン中断/再開制御: co_await式により、コルーチン中断と再開に関する振る舞いを詳細に制御する(await_transform, operator co_await)。
    • コルーチン中断の条件: co_await式に対して、コルーチンを中断するか否かを判断する(await_ready)。
    • コルーチン中断直前の動作: co_await式に対して、コルーチンを中断する直前の動作を制御する(await_suspend)。
    • コルーチン再開直後の動作: co_await式に対して、コルーチンが再開された直後の動作を制御する(await_resume)。

プログラマが記述するコルーチンは、コンパイル時にソースコード変換が行われると解釈できる。 (従来のC++仕様範囲ではコルーチン動作を正確に表現できないため、下記はあくまでも疑似的なコードとなる):

// プログラマが記述するコルーチン
my_generator iota(int end)
{
  for (int n = 0; n < end; ++n) {
    co_yield n;
  }
}

// C++コンパイラにより展開されたコード
my_generator iota(int end)
{
  // コルーチンに対応するPromiseオブジェクトを初期化
  my_generator::promise_type promise;

  // 戻り値型オブジェクトの初期化
  my_generator result = promise.get_return_object();
  // コルーチンハンドルをget_return_object内で取得し、resultメンバで保持する。
  // 生成したresultオブジェクトは、初回のコルーチン中断時に呼出元へ返される。

  // 本例では全て co_await std::suspend_always{} 相当のため、
  // 以降のco_await式(★箇所)においてコルーチンは中断/再開される。

  // 初期サスペンドポイント
  co_await promise.initial_suspend(); //

  // コルーチン本体部
  {
    for (int n = 0; n < end; ++n) {
      // co_yield式は下記co_await式に展開される
      co_await promise.yield_value(n); //
    }
  }
  promise.return_void();

  // 最終サスペンドポイント
  co_await promise.final_suspend(); //

  // 本例では最終サスペンドポイントでコルーチンを中断するため、ここには制御が到達しない。
  // 呼出側で戻り値オブジェクトを破棄すると、デストラクタ経由で本コルーチンは破棄される。
}
  • co_yield[color ff0000]
  • co_await[color ff0000]
  • std::suspend_always{}[link /reference/coroutine/suspend_always.md]

仕様

コルーチン定義

C++におけるコルーチンは、関数の一種として定義される。

関数本体に新キーワードco_await(Await式), co_yield(Yield式), co_returnのいずれかが含まれるとき、その関数はコルーチンとなる。 つまり、戻り値型や引数リストなどのシグニチャからコルーチン/関数を区別することはできない。 コルーチンの引数宣言リストはC言語由来の可変引数リスト(...)を含んではならないが、可変引数テンプレートのパラメータパック(...)は利用できる。

task<int> f();

task<void> g1() {
  int i = co_await f();
  std::cout << "f() => " << i << std::endl;
}

template <typename... Args>
task<void> g2(Args&&...) { // OK, "..."はパック展開
  int i = co_await f();
  std::cout << "f() => " << i << std::endl;
}

task<void> g3(int a, ...) { // エラー: 可変引数リストは許可されない
  int i = co_await f();
  std::cout << "f() => " << i << std::endl;
}
  • co_await[color ff0000]

プログラムエントリポイントのmain関数、constexpr関数、戻り値型をプレースホルダ(auto)で宣言された関数、クラス型のコンストラクタとデストラクタは、コルーチンとして定義できない。

Promise型とコルーチン動作仕様

コルーチンのPromise型は、コルーチンの戻り値型Rと引数リストP1, P2, ..., Pnから決定されるクラス型である。

  • デフォルト動作ではR::promise_typeがPromise型となる。
  • ユーザプログラム中でstd::coroutine_traitsトレイトを特殊化した場合は、coroutine_traits<R, P1, P2, ..., Pn>::promise_typeがPromise型となる。
  • コルーチンがクラスの非静的メンバの場合、P1は暗黙のオブジェクトパラメータ(*thisの型)となる。

コルーチンは、その本体 function-body が下記の通り置き換えられたかのように動作する:

{
  promise-type promise promise-constructor-arguments ;
  try {
    co_await promise.initial_suspend() ;
    function-body
  } catch ( ... ) {
    if (! initial-await-resume-called )
      throw ;
    promise.unhandled_exception() ;
  }
final-suspend :
  co_await promise.final_suspend() ;
}
  • promise-type[italic]
  • promise-constructor-arguments[italic]
  • promise[italic]
  • function-body[italic]
  • initial-await-resume-called[italic]
  • final-suspend[italic]
  • initial_suspend呼び出しを含むAwait式は、初期サスペンドポイントとなる。
  • final_suspend呼び出しを含むAwait式は、最終サスペンドポイントとなる。
  • initial-await-resume-calledfalseで初期化され、初期サスペンドポイントの式 await-resume が評価される直前にtrueが設定される。
  • promise-type はPromise型を表す。
  • 説明用の変数名 promise は、コルーチンのPromiseオブジェクトを表す。
  • ラベル final-suspend は説明のためにのみ定義される。
  • promise-constructor-arguments は次の通りに決定される:
    • 引数リストPiの左辺値をpiとする。コルーチンが非静的メンバの場合、p1*thisを表しp(i+1)はi番目の関数パラメータを表す。
    • 左辺値p1...pnの実引数リストを用いて、Promiseコンストラクタ呼び出しのオーバーロード解決を試みる。
    • 適合するコンストラクタが見つかった場合は、promise-constructor-arguments(p1, ..., pn) となる。見つからなかった場合、promise-constructor-arguments は空のリストとなる。

Promise型のスコープにおいて、非修飾なreturn_voidおよびreturn_valueの探索が行われる。両方が見つかった場合、プログラムは不適格となる。

コルーチン呼び出しのglvalue結果またはprvalue結果オブジェクトを初期化するために、式 promise.get_return_object()が使われる。 get_return_object呼び出しは高々1回であり、initial_suspend呼び出しよりも前に順序付けられる。

中断状態にあるコルーチンは、そのコルーチンを指すコルーチンハンドルの再開メンバ関数呼び出しによって、継続実行を再開できる。 再開メンバ関数を呼び出した関数は、再開元(resumer)と呼ばれる。 中断状態にないコルーチンに対する再開メンバ関数呼び出しは、未定義の動作をもたらす。

処理系はコルーチンのために追加のメモリ領域を確保する必要があるかもしれない。 このメモリ領域はコルーチン・ステートとして知られ、非配列版のメモリ確保関数(operator new)によって確保される。 メモリ確保関数はPromise型のスコープで名前探索が行われる。 名前探索に失敗した場合は、グローバルスコープで探索が行われる。 名前探索がPromise型のスコープで確保関数を見つけた場合は、実引数リストを用いて関数呼び出しのオーバーロード解決が行われる。 第1引数はstd::size_t型であり、要求メモリサイズの合計値となる。続く実引数は左辺値p1...pnとなる。 適合する関数が見つからなかった場合、std::size_t型の要求メモリサイズ合計値のみで再度オーバーロード解決が行われる。

Promise型のスコープにおいて、非修飾なget_return_object_on_allocation_failureの探索が行われる。 何らかの宣言が見つかった場合、グローバルな::operator new(size_t nothrow_t)形式の確保関数が選択され、メモリ領域確保に失敗すると、コルーチン・ステート用メモリ領域取得のための確保関数呼び出し結果はnullptrを返すと想定される。 このケースにおける確保関数は、例外を投げないnoexcept指定されるべきである。 メモリ確保関数がnullptrを返した場合、コルーチンはその呼び出し元に制御を戻し、戻り値はT::get_return_object_on_allocation_failure()呼び出しにより取得する。 ここでTはPromise型を表す。

#include <iostream>
#include <coroutine>

// メモリ確保が必要となったときは::operator new(size_t, nothrow_t)が使われる
struct generator {
  struct promise_type;
  using handle = std::coroutine_handle<promise_type>;
  struct promise_type {
    int current_value;
    static auto get_return_object_on_allocation_failure() { return generator{nullptr}; }
    auto get_return_object() { return generator{handle::from_promise(*this)}; }
    auto initial_suspend() { return std::suspend_always{}; }
    auto final_suspend() noexcept { return std::suspend_always{}; }
    void unhandled_exception() { std::terminate(); }
    void return_void() {}
    auto yield_value(int value) {
      current_value = value;
      return std::suspend_always{};
    }
  };
  bool move_next() { return coro ? (coro.resume(), !coro.done()) : false; }
  int current_value() { return coro.promise().current_value; }
  generator(generator const&) = delete;
  generator(generator && rhs) : coro(rhs.coro) { rhs.coro = nullptr; }
  ~generator() { if (coro) coro.destroy(); }
private:
  generator(handle h) : coro(h) {}
  handle coro;
};

generator f() { co_yield 1; co_yield 2; }

int main() {
  auto g = f();
  while (g.move_next()) std::cout << g.current_value() << std::endl;
}
  • co_yield[color ff0000]
  • std::coroutine_handle<promise_type>[link /reference/coroutine/coroutine_handle.md]
  • from_promise[link /reference/coroutine/coroutine_handle/from_promise.md]
  • resume()[link /reference/coroutine/coroutine_handle/resume.md]
  • done()[link /reference/coroutine/coroutine_handle/done.md]
  • promise()[link /reference/coroutine/coroutine_handle/promise.md]
  • destroy()[link /reference/coroutine/coroutine_handle/destroy.md]
  • suspend_always[link /reference/coroutine/suspend_always.md]
  • std::terminate[link /reference/exception/terminate.md]

コルーチンの終端まで制御が到達、またはコルーチンを指すコルーチンハンドルのdestroyメンバ関数が呼び出されると、コルーチン・ステートは破棄される。

メモリ解放関数はPromise型のスコープで名前探索が行われる。 名前探索に失敗した場合は、グローバルスコープで探索が行われる。 解放関数の探索が、ポインタパラメータのみの通常の解放関数と、ポインタとサイズをパラメータにとる通常の解放関数の両方を見つける場合、2個のパラメータをとる解放関数が選択される。 そうでなければ、1個のパラメータをとる解放関数が選択される。 通常の解放関数が見つからなければ、プログラムは不適格となる。 選択された解放関数の呼び出しでは、その第1実引数に解放すべきメモリブロックのアドレスが渡される。 解放関数のパラメータにstd::size_tが使われる場合、その実引数としてメモリブロックのサイズが渡される。

コルーチンが呼び出されるとき、パラメータ初期化が行われたのち、各コルーチンパラメータのコピーが作成される。 型cv Tをもつパラメータにおいて、そのコピーはパラメータを参照するT型のxvalueで直接初期化された自動記憶域期間をもつcv T型の変数となる。 各パラメータのコピーの初期化と破棄は、呼び出されたコルーチンのコンテキストで行われる。 パラメータのコピーの初期化は、コルーチンPromiseコンストラクタの呼び出しより前に順序付けられ、それぞれは互いに非決定順で順序付けられる。 パラメータのコピーの生存期間は、パラメータPromiseオブジェクトの終了直後で終了する。 (コルーチンが参照渡しのパラメータを持つ場合、そのパラメータにより参照されるエンティティ生存期間終了後のコルーチン再開は未定義動作を引き起こしやすい。)

promise.unhandled_exception()の評価が例外で終了した場合、コルーチンは最終サスペンドポイントで中断したとみなされる。

co_await promise.final_suspend() は例外送出してはならない。

Await式

co_await式は、そのオペランド式で表される計算の完了を待機しているあいだ、コルーチン評価をサスペンド(中断)するために用いる。

co_await cast-expression
  • cast-expression[italic]

Await式は、コルーチン本体複合文の内側(かつtrycatch構文のcatch節の外側)において潜在的に評価される式(potentially-evaluated expression)でのみ、出現してよい。 宣言文やfor構文の宣言を伴う初期化部では、その初期化子の中でのみAwait式が出現してよい。 デフォルト引数ではAwait式を用いることはできない。 Await式は、静的記憶域もしくはスレッドローカルなブロックスコープ変数の初期化に出現してはならない。 関数内でAwait式を置けるコンテキストを、関数の中断コンテキストと呼ぶ。

Await式の評価では、次のような補助的な型、式、オブジェクトを用いる:

  • p を同Await式を含むコルーチンのPromiseオブジェクトの左辺値名とし、Pを同オブジェクトの型とする。
  • a (Awaitable) を下記のように定義する:
    • Await式がYield式または初期サスペンドポイントまたは最終サスペンドポイントにより暗黙に生成された場合、a をその cast-expression とする。
    • Pのスコープで非修飾なawait_transformの探索により一つ以上の名前がみつかった場合は、 ap.await_transform( cast-expression )とする。
    • それ以外では acast-expression とする。
  • o (Awaiter) を下記のように定義する。o がprvalueの場合はTemporary materialization conversionが行われる:
    • 実引数 a に対して適用可能なoperator co_await関数を列挙し、o をオーバーロード解決により選択された関数呼び出しとする。
    • 適合する関数が見つからない場合、oa とする。
    • オーバーロード解決が曖昧な場合、プログラムは不適格となる。
  • e を、o の評価結果を参照する左辺値とする。
  • h を、同Await式を含むコルーチンを参照するstd::coroutine_handle<P>型のオブジェクトとする。
  • await-ready を、boolに変換されうる式 e.await_ready()とする。
  • await-suspend を、式 e.await_suspend( h )とする。この式(の結果)はvoidであるか、boolまたは任意の型Zに対するstd::coroutine_handle<Z>型のprvalueであるべき。
  • await-resume を、式 e.await_resume()とする。

Await式は式 await-resume と同じ型、同じ値カテゴリを持つ。

Await式は式 o と式 await-ready を評価し、続いて:

  • await-ready の結果がfalseの場合、コルーチンは中断状態とみなされる。その後に:
    • await-suspend の型がstd::coroutine_handle<Z>の場合、await-suspend.resume()が評価される。
    • そうではなく await-suspend の型がboolの場合、await-suspend が評価され、その結果がfalseであればコルーチンは再開する。
    • それ以外の場合、await-suspend が評価される。
  • await-suspend の評価が例外で終了した場合、例外が捕捉されてコルーチンが再開し、その例外は即座に再スローされる。そうでなければ、スコープ終了をともなわずに現在のコルーチンの呼出元もしくは再開元へ制御フローを戻す。
  • await-ready の結果がtrueまたはコルーチンが再開した場合、await-resume の評価結果がAwait式の結果となる。
template <typename T>
struct my_future {
  /* ... */
  bool await_ready();
  void await_suspend(std::coroutine_handle<>);
  T await_resume();
};

template <class Rep, class Period>
auto operator co_await(std::chrono::duration<Rep, Period> d) {
  struct awaiter {
    std::chrono::system_clock::duration duration;
    /* ... */
    awaiter(std::chrono::system_clock::duration d) : duration(d) {}
    bool await_ready() const { return duration.count() <= 0; }
    void await_resume() {}
    void await_suspend(std::coroutine_handle<> h) { /* ... */ }
  };
  return awaiter{d};
}

using namespace std::chrono;

my_future<int> h();

my_future<void> g() {
  std::cout << "just about go to sleep...\n";
  co_await 10ms;
  std::cout << "resumed\n";
  co_await h();
}

auto f(int x = co_await h()); // エラー: await式は関数中断コンテキストの外
int a[] = { co_await h() };   // エラー: await式は関数中断コンテキストの外
  • co_await[color ff0000]
  • std::coroutine_handle<>[link /reference/coroutine/coroutine_handle.md]

Yield式

co_yield式は、コルーチンから値を生成(yield)するときに用いる。 その動作はAwait式にて書き換え可能であり、コルーチン利用者向けのシンタックスシュガーとも解釈できる。

co_yield assignment-expression
co_yield braced-init-list
  • assignment-expression[italic]
  • braced-init-list[italic]

Yield式は関数の中断コンテキストにのみ出現してよい。 e をYield式のオペランド、p を同式を含むコルーチンのPromiseオブジェクトのlvalue名としたとき、Yield式は式co_await p.yield_value( e )と等価である。

template <typename T>
struct generator {
  struct promise_type {
    T current_value;
    /* ... */
    auto yield_value(T v) {
      current_value = std::move(v);
      return std::suspend_always{};
    }
  };
  struct iterator { /* ... */ };
  iterator begin();
  iterator end();
};

generator<pair<int,int>> g1() {
  for (int i = i; i < 10; ++i) co_yield {i,i};
}
generator<pair<int,int>> g2() {
  for (int i = i; i < 10; ++i) co_yield make_pair(i,i);
}

auto f(int x = co_yield 5); // エラー: yield式は関数中断コンテキストの外
int a[] = { co_yield 1 };   // エラー: yield式は関数中断コンテキストの外

int main() {
  auto r1 = g1();
  auto r2 = g2();
  assert(std::equal(r1.begin(), r1.end(), r2.begin(), r2.end()));
}
  • co_yield[color ff0000]
  • std::suspend_always[link /reference/coroutine/suspend_always.md]
  • std::move[link /reference/utility/move.md]
  • pair[link /reference/utility/pair.md]
  • make_pair[link /reference/utility/make_pair.md]
  • std::equal[link /reference/algorithm/equal.md]

co_return文

co_return文は、コルーチンを終了し呼出元へ制御を戻すために用いる。 co_yieldco_awaitいずれも含まないコルーチンを定義する場合にも利用できる。

co_return expr-or-braced-init-list opt ;
  • expr-or-braced-init-list[italic]
  • opt[italic]

co_return文または中断により、コルーチンは呼出元もしくは再開元に制御を戻す。 コルーチンは通常のreturn文を含んではならない。

co_return文の expr-or-braced-init-list はオペランドと呼ばれる。 p をコルーチンPromiseオブジェクトのlvalue名とすると、co_return文は次と等価である:

{ S; goto final-suspend ; }
  • S[italic]
  • final-suspend[italic]

ここで final-suspend はコルーチン動作説明用の最終サスペンドポイントラベル名であり、S は次の通り定義される:

  • オペランドが braced-init-list または非void型の式の場合、Sp.return_value( expr-or-braced-init-list )とする。式 Svoid型のprvalueであるべき。
  • そうでなければ、S を複合文 { expression opt ; p.return_void(); }とする。式 p.return_void()void型のprvalueであるべき。

p.return_void()が有効な式のとき、コルーチン本体の終端到達はオペランド無しco_returnと等価である。 そうでなければ、コルーチン本体の終端到達は未定義の動作を引き起こす。

#include <iostream>
#include <coroutine>
#include <utility>

// コルーチン利用ライブラリ: ジェネレータ型
struct my_generator {
  // ジェネレータに関連付けられるPromise型
  struct promise_type {
    // co_yield式で指定されるint値を保持する変数
    int value_;

    auto get_return_object()
    {
      // コルーチンに紐づくPromiseオブジェクト(*this)から
      // ジェネレータ型のコルーチン戻り値オブジェクトを生成
      return my_generator{*this};
    };
    auto initial_suspend()
    {
      // コルーチン本体処理の開始前に無条件サスペンド
      return std::suspend_always{};
    }
    auto final_suspend() noexcept
    {
      // コルーチン本体処理の終了後に無条件サスペンド
      return std::suspend_always{};
    }
    auto yield_value(int v)
    {
      // co_yield式で渡される値を保持し、コルーチンを無条件サスペンド
      value_ = v;
      return std::suspend_always{};
    }
    void return_void() {}
    void unhandled_exception() { std::terminate(); }
  };
  // ジェネレータに関連付けられるコルーチンハンドル型
  using coro_handle = std::coroutine_handle<promise_type>;

  // 範囲for構文サポート用イテレータ型
  struct iterator {
    // 対象のコルーチンハンドル
    coro_handle coro_;
    // 対象コルーチン本体処理が終了したかを表すフラグ
    bool done_;

    iterator& operator++()
    {
      // yield_value()で中断したコルーチンを再開する
      coro_.resume();
      // (co_yield式評価もしくはコルーチン本体処理の終了により制御が戻ってくる)
      done_ = coro_.done();
      return *this;
    }
    bool operator!=(const iterator& rhs) const
    {
      return done_ != rhs.done_;
    }
    int operator*() const
    {
      // Promiseオブジェクトが保持している値を返す
      return coro_.promise().value_;
    }
  };

  ~my_generator()
  {
    if (coro_)
      coro_.destroy();
  }

  my_generator(my_generator const&) = delete;
  my_generator(my_generator&& rhs) 
    : coro_(std::exchange(rhs.coro_, nullptr)) {}

  // 範囲for構文サポート用のメンバ関数
  iterator begin()
  {
    // initial_suspend()で中断したコルーチンを再開する
    coro_.resume();
    // (初回co_yield式評価により制御が戻ってくる)
    return {coro_, coro_.done()};
  }
  iterator end()
  {
    // 終端位置を表現する番兵イテレータ
    return {{}, true};
  }

private:
  // Promiseオブジェクト経由でコルーチンハンドルを取得する
  explicit my_generator(promise_type& p)
    : coro_(coro_handle::from_promise(p)) {}

  coro_handle coro_;
};


// ユーザ定義コルーチン
my_generator iota(int end)
{
  // コルーチンに対応したPromise型 generator::promise_typeの
  // Promiseオブジェクト(p)が生成される。

  for (int n = 0; n < end; ++n) {
    // 下式は co_await p.yield_value(n) と等価
    co_yield n;
  }
  // コルーチン本体の終端到達により p.return_void() 呼び出し
}

int main()
{
  // コルーチンを呼び出し、整数生成ジェネレータを取得する。
  auto g = iota(10);
  // このタイミングではまだコルーチン本体は実行されない。

  // 範囲for構文を用いてコルーチン本体を実行する。
  // ここではコルーチンiotaの値生成ループ処理ステップと、
  // main関数の表示ループ処理ステップが交互に実行される。
  for (int v: g) {
    std::cout << v;
  }
}
  • co_yield[color ff0000]
  • std::exchange[link /reference/utility/exchange.md]
  • std::terminate()[link /reference/exception/terminate.md]
  • std::coroutine_handle[link /reference/coroutine/coroutine_handle.md]
  • std::suspend_always[link /reference/coroutine/suspend_always.md]
  • destroy()[link /reference/coroutine/coroutine_handle/destroy.md]
  • resume()[link /reference/coroutine/coroutine_handle/resume.md]
  • done()[link /reference/coroutine/coroutine_handle/done.md]
  • from_promise[link /reference/coroutine/coroutine_handle/from_promise.md]
  • promise()[link /reference/coroutine/coroutine_handle/promise.md]

出力

0123456789

この機能が必要になった背景・経緯

多くのプログラミング言語で対応されており広い実績のあるコルーチン機能を、C++言語でも使えるよう2013年頃から検討が始まっている。

2017年には ISO/IEC TS 22277 C++ Extensions for Coroutines(通称"Coroutines TS") として正式発効され、いくつかの追加の仕様修正をへてC++20言語仕様本体への統合が決定された。

C++言語仕様へのコルーチン導入によって、ジェネレータの協調的マルチタスクのサポート、ファイルやネットワークなど非同期I/Oライブラリとの統合が期待されている。

検討されたほかの選択肢

C++20コルーチンはスタックレスコルーチンとして導入されたが、スタックフル(Stackful)コルーチン=ファイバー(Fiber)の導入検討も長らく行われてきた。 スタックフルコルーチンは将来のC++仕様導入に向けて引き続き検討されている。 (本ページ執筆時点では提案文書P0876R10が最新)

C++20コルーチンでは、コルーチン・ステートのために動的メモリ確保が行われる可能性がある。 一定条件を満たせばコンパイラ最適化によって動的メモリ確保が省略されるとしているが、言語仕様として動的メモリ確保を避ける仕様も検討された(通称"Core Coroutines")。 最終的には既に実績のあるCoroutinesTS(発案者の名前にちなみ"Gor-routines"と呼ばれた)ベースのコルーチン仕様が採用されることになった。

C++20コルーチンに関するキーワードは、いずれも接頭辞co_が付与されている。 何度かの改名提案(P0071R0P1485R1)も提出されたが、いずれも否決されてC++20仕様に落ち着いた。

参照