線形化可能性

出典: フリー百科事典『ウィキペディア(Wikipedia)』

並行プログラミングにおいて操作(または操作の集合)は、呼び出しイベントと応答イベント(コールバック)の順序付きリストで構成されており、応答イベントを追加することで以下のように拡張できる場合、線形化可能である。

  1. 拡張されたリストは逐次履歴として再表現することができる(直列化可能である)。
  2. その逐次履歴は元の拡張されていないリストの部分集合である。

これは、非公式には、変更されていないイベントのリストは、その呼び出しが直列化可能であるが、直列スケジュールの応答の一部がまだ戻ってきていない場合に限り、線形化可能であることを意味する[1]

並行システムではプロセスが同時に共有オブジェクトにアクセスできる。複数のプロセスが1つのオブジェクトにアクセスしているため、あるプロセスがオブジェクトにアクセスしている間に、別のプロセスがオブジェクトの内容を変更するという事態が発生することがある。

この例では線形化可能性の必要性を示している。線形化可能なシステムでは、共有されたオブジェクトに対する操作が重なっても、それぞれの操作は瞬時に行われているように見える。線形化可能性は強力な正当性条件であり、複数のプロセスが同時にオブジェクトにアクセスした場合に、どのような出力が可能であるかを制約する。線形化可能性は、操作が予期せぬ方法で完了しないことを保証する安全特性でもある。システムが線形化可能であれば、プログラマーはシステムについて推論することができる[2]

線形化可能性の歴史[編集]

線形化可能性は、1987年にハーリー(Maurice Peter Herlihy)とWingによって初めて一貫性モデルとして紹介された。このモデルは「アトミックな操作とは、同時実行される操作によって割り込みされることがない操作である」といった、より限定的なアトミックの定義を含んでおり、操作の開始と終了がいつであるかについては通常曖昧である。

アトミックなオブジェクトは、逐次的・直列的な定義から即座に完全に理解することができる。並列に実行される一連の操作が常に次々と発生しているように見えることであり、矛盾が生じることはない。具体的には、線形性は、システムの不変性がすべての操作によって観察され、保存されることを保証する。すべての処理が個々に不変性を保持していれば、システム全体としても不変性を保持していることになる。

線形化可能性の定義[編集]

並行システムは、共有データ構造やオブジェクトを介して通信するプロセスの集合体である。並行システムでは、複数のプロセスが同時にオブジェクトにアクセスする可能性があり、プログラマーは期待される結果を推論できる必要があるため、線形化可能性が重要となる。並行システムの実行結果は、完了した操作の順序付けられたシーケンスである履歴となる。

履歴(history)は、一連のスレッドやプロセスがオブジェクトに対して行った呼び出しと応答のシーケンスと定義する。呼び出しは操作の開始であり、応答はその操作の終了を知らせるものと考えることができる。関数を呼び出すたびに、それに続く応答が発生する。これを利用して、オブジェクトのあらゆる用途をモデル化することができる。例えば、2つのスレッド、AとBが両方ともロックを取得しようとし、既に取得されている場合には手を引くとする。これは、両方のスレッドがロック操作を呼び出し、両方のスレッドが応答を受け取り、一方は成功し、一方は失敗したとモデル化される。

Aがロックを呼び出す Bがロックを呼び出す Aが失敗の応答を受け取る Bが成功の応答を受け取る

逐次履歴とは、すべての呼び出しが即座に応答するもので、つまり、呼び出しと応答が瞬時に行われると考えられる。逐次履歴は実際の並行性を持たないため、推論するのは簡単なはずだが、前の例は逐次履歴ではないため、推論するのは困難となる。ここで線形性が必要になる。

履歴が線形化可能なのは、次のような完了した操作の線形順序がある場合である。

  1. の中で完了したすべての操作について,その操作は,すべての操作がの順に1つずつ完了した場合に返すのと同じ結果を実行時に返す。
  2. op2が始まる(起動する)前にop1が完了する(応答を得る)場合、内ではop1がop2に先行することになる[1]

言い換えれば

  • その呼び出しと応答を並べ替えると、逐次的な履歴が得られる。
  • その逐次履歴は、オブジェクトの逐次定義に基づいて正しいものである。
  • 元の履歴で呼び出しの前に応答があった場合、順次並べ替えてもその前に応答がなければならない。

(ここで最初の2つの箇条書きは直列性と一致していることに注意すべきである。つまり操作は何らかの順序で起こるように見える。最後の点は線形化可能性に特有のものであり,HerlihyとWingの主要な貢献である)[1]

先ほどのロックの例を、2つの方法で並び替えて見る。

Aがロックを呼び出す Aが失敗の応答を受け取る Bがロックを呼び出す Bが成功の応答を受け取る

Aの応答の下にあるBの呼び出しを並べ替えると、連続した履歴になる。すべての操作が明らかな順序で行われるようになったので、これは簡単に推論できる。しかし残念なことに、これはオブジェクトの逐次的な定義とは一致しない(プログラムのセマンティクスとは一致しない)。Aはロックの取得に成功し、Bはその後中止したはずである。

Bがロックを呼び出す Bが成功の応答を受け取る Aがロックを呼び出す Aが失敗の応答を受け取る

これは正しい順列の履歴かつ線形化されている。線形化可能性の定義は、呼び出しの前にある応答を並べ替えることを妨げるだけであることに注意する。したがって、元の履歴は確かに線形化可能である.

オブジェクト(ヒストリーではなく)は、その使用に関するすべての有効な履歴が線形化できる場合、線形化可能である。なお、これを証明するのは困難とされている。

線形化可能性と直列化可能性[編集]

次のような履歴があるとする。これも2つのオブジェクトがロックを使って相互作用する。

Aがロックを呼び出す Aが成功の応答を受け取る Bがロック解除を呼び出す Bがロック解除に成功する Aがロック解除を呼び出す Aがロック解除に成功する

この履歴は、AとBの両方がロックを保持している時点があるため有効ではない。さらに順序規則に違反しない限り、有効な順序履歴に並び替えることはできない。したがって線形化可能ではない。しかし直列化可能性の下では、Bのロック解除操作をAの最初のロックの「前」に移動させることができ、これは有効な履歴となる(オブジェクトがロック状態で履歴を開始したと仮定)。

Bがロック解除を呼び出す Bがロック解除に成功する Aがロックを呼び出す Aがロックに成功する Aがロック解除を呼び出す Aがロック解除に成功する

この並び替えは、AとBの間で別の通信手段がない場合には意味がある。並び替えの制限により複数の線形化可能なオブジェクトが全体として線形化可能であることが保証されるため、個々のオブジェクトを個別に検討する場合には、線形化可能性が向上する。

線形化ポイント[編集]

線形化可能性のこの定義は、次のものと同等である。

  • すべての関数の呼び出しは、その呼び出しと応答の間のある瞬間に線形化点(線形化ポイント)を持つ。
  • すべての関数はその線形化点(線形化ポイント)で瞬時に発生しているように見え、逐次定義で指定された通りに動作する。

この方法は、通常、証明するのがはるかに簡単である。また、直感的に理解できることから、ユーザーとしても推論しやすいものとなっている。この瞬間的に発生する、あるいは不可分に発生するという特性から、長い「線形化可能」の代わりに「アトミック」という用語が使われている[1]

以下の例では、compare-and-swapで作られたカウンターの線形化ポイントは、最初の(そして唯一の)compare-and-swap更新の成功の線形化ポイントとなる。ロックを使用したカウンターは、ロックが保持されている間は、競合する可能性のある操作が実行されないため、いつでも線形化されると考えることができる。

線形化可能性の例[編集]

カウンター[編集]

線形化可能性の威力と必要性を示すために、異なるプロセスがインクリメントできる単純なカウンタを考える。

複数のプロセスがアクセスできるカウンタオブジェクトを実装する。多くの一般的なシステムでは、あるイベントの発生回数を記録するためにカウンターを使用している。

カウンターオブジェクトは、複数のプロセスからアクセスでき、2つの操作が可能である。

  • インクリメント:カウンターに格納されている値に1を加え、確認応答を返す。
  • 読み込み:カウンターに格納されている現在の値を変更せずに返す。

このカウンターオブジェクトを、共有レジスタを使って実装する。

最初の試みは、線形化できないことがわかっているので、プロセス間で1つの共有レジスタを使用して、次のような実装を行う。

ノンアトミック[編集]

素朴で非アトミックな実装。

インクリメント

  1. レジスタRの値を読む
  2. 値に1を加える
  3. 新しい値をレジスタRに書き戻す

読み込み

レジスタRの読み出し

この単純な実装では線形化はできない。

値が0になるように初期化された1つのカウンタ・オブジェクトに、2つのプロセスがアクセスしているとする。

  1. 最初のプロセスは,レジスタの値を0として読み取る
  2. 最初のプロセスは値に1を加え、カウンタの値は1になるはずだが、新しい値をレジスタに書き戻す前に中断(割り込まれ)してしまい、その間に2番目のプロセスが実行される
  3. 2つ目のプロセスは、レジスタの値を読み取るが、値はまだ0になっている
  4. 2つ目のプロセスは、その値に1を加える
  5. 2つ目のプロセスが新しい値をレジスタに書き込み、レジスタの値は1になる

2つ目のプロセスは実行を終了し、1つ目のプロセスは中断したところから実行を続ける。

  1. 第1のプロセスは、他のプロセスがすでにレジスタの値を1に更新していることを知らずに、レジスタに1を書き込む

上記の例では、2つのプロセスがインクリメントコマンドを呼び出しましたが、オブジェクトの値は本来の2ではなく、0から1にしか増えなかった。システムが線形化できないために、インクリメント操作の1つが失われた。

上記の例は、データ構造の実装を注意深く考える必要があり、線形性がシステムの正しさにどのような影響を与えるかを示している。

アトミック[編集]

線形化可能なアトミックカウンタオブジェクトを実装するために,これまでの実装を変更し,各プロセスPiが独自のレジスタRiを使用するようにする。

各プロセスは、以下のアルゴリズムに従ってインクリメントとリードを行う。

インクリメント

  1. レジスタRiの値を読み込む
  2. その値に1を加える
  3. 新しい値をRiに戻す

読み込み

  1. レジスタ R1,R2...Rn を読む
  2. すべてのレジスタの合計を返す

この実装では最初の実装の問題点が解決されている。このシステムでは、インクリメント演算は書き込みステップで線形化されている。インクリメント演算の線形化可能性ポイントは、その演算がレジスタRiに新しい値を書き込むときとなる。読み取り操作は、読み取りによって返された値が、各レジスタRiに格納されたすべての値の合計に等しいとき、システムのある時点で線形化される。

これは些細な例ではあるが、実際のシステムでは操作はより複雑になり、導入されるエラーも極めて微妙なものになる。例えば、64ビットの値をメモリーから読み出す場合、実際には2つの32ビットのメモリーロケーションを2回連続して読み出すように実装されることがある。プロセスが最初の32ビットだけを読み取った後、2番目の32ビットを読み取る前にメモリー内の値が変更された場合、元の値でも新しい値でもなく、混ざった値を持つことになる。

さらにプロセスの実行順序によっても結果が変わってしまうため、このようなエラーの検出、再現、デバッグは困難である。

比較・交換[編集]

多くのシステムでは、アトミックな比較スワップ命令が用意されている。これはメモリーロケーションから値を読み取り、その値とユーザーが指定した「期待値」を比較し、両者が一致した場合に「新しい」値を書き出し、更新が成功したかどうかを返すものである。これを利用してノンアトミックカウンターのアルゴリズムを以下のように修正することができる。

  1. メモリーロケーションの値を読み取る
  2. その値に1を加える
  3. 比較スワップ機能を使って、増加した値を書き戻す
  4. 比較スワップによって読み込まれた値が、最初に読み込んだ値と一致しない場合は再試行する

比較スワップは瞬時に行われる(ように見える)ので、進行中に他のプロセスがその場所を更新した場合、比較スワップは確実に失敗する。

関連項目[編集]

脚注[編集]

  1. ^ a b c d Herlihy, Maurice P.; Wing, Jeannette M. (1990). “Linearizability: A Correctness Condition for Concurrent Objects”. ACM Transactions on Programming Languages and Systems 12 (3): 463–492. doi:10.1145/78969.78972. 
  2. ^ Shavit, Nir and Taubenfel,Gadi (2016). “The Computability of Relaxed Data Structures: Queues and Stacks as Examples”. Distributed Computing 29 (5): 396–407. doi:10.1007/s00446-016-0272-0. http://www.faculty.idc.ac.il/gadi/MyPapers/2015ST-RelaxedDataStructures.pdf. 

参考文献[編集]

  • Herlihy, Maurice P.; Wing, Jeannette M. (1987). Axioms for Concurrent Objects. 13. doi:10.1145/41625.41627. ISBN 978-0-89791-215-0. http://repository.cmu.edu/cgi/viewcontent.cgi?article=2597&context=compsci 
  • Herlihy, Maurice P. (1990). A Methodology for Implementing Highly Concurrent Data Structures. 25. 197–206. doi:10.1145/99164.99185. ISBN 978-0-89791-350-8 
  • Herlihy, Maurice P.; Wing, Jeannette M. (1990). “Linearizability: A Correctness Condition for Concurrent Objects”. ACM Transactions on Programming Languages and Systems 12 (3): 463–492. doi:10.1145/78969.78972. 
  • Aphyr. “Strong Consistency Models”. aphyr.com. Aphyr. 2018年4月13日閲覧。