view presen/slide.html @ 24:86184ba4927b

fix
author ikkun
date Tue, 16 May 2017 09:46:11 +0900
parents 1e7bc71c1abe
children 6f5ce3e82fbf
line wrap: on
line source

<!DOCTYPE html>
<html>
<head>
   <meta http-equiv="content-type" content="text/html;charset=utf-8">
   <title>Gears OS における並列処理</title>

<meta name="generator" content="Slide Show (S9) v2.5.0 on Ruby 2.4.1 (2017-03-22) [x86_64-darwin16]">
<meta name="author"    content="東恩納 琢偉" >

<!-- style sheet links -->
<link rel="stylesheet" href="s6/themes/projection.css"   media="screen,projection">
<link rel="stylesheet" href="s6/themes/screen.css"       media="screen">
<link rel="stylesheet" href="s6/themes/print.css"        media="print">
<link rel="stylesheet" href="s6/themes/blank.css"        media="screen,projection">

<!-- JS -->
<script src="s6/js/jquery-1.11.3.min.js"></script>
<script src="s6/js/jquery.slideshow.js"></script>
<script src="s6/js/jquery.slideshow.counter.js"></script>
<script src="s6/js/jquery.slideshow.controls.js"></script>
<script src="s6/js/jquery.slideshow.footer.js"></script>
<script src="s6/js/jquery.slideshow.autoplay.js"></script>

<!-- prettify -->
<link rel="stylesheet" href="scripts/prettify.css">
<script src="scripts/prettify.js"></script>

<script>
  $(document).ready( function() {
    Slideshow.init();

    $('code').each(function(_, el) {
      if (!el.classList.contains('noprettyprint')) {
        el.classList.add('prettyprint');
      }
    });
    prettyPrint();
  } );

  
</script>

<!-- Better Browser Banner for Microsoft Internet Explorer (IE) -->
<!--[if IE]>
<script src="s6/js/jquery.microsoft.js"></script>
<![endif]-->



</head>
<body>

<div class="layout">
  <div id="header"></div>
  <div id="footer">
    <div align="right">
      <img src="s6/images/logo.svg" width="200px">
    </div>
  </div>
</div>

<div class="presentation">

  <div class='slide cover'>
    <table width="90%" height="90%" border="0" align="center">
      <tr>
        <td>
          <div align="center">
            <h1><font color="#808db5">Gears OS における並列処理</font></h1>
          </div>
        </td>
      </tr>
      <tr>
        <td>
          <div align="left">
            東恩納 琢偉
            琉球大学工学部
            <hr style="color:#ffcc00;background-color:#ffcc00;text-align:left;border:none;width:100%;height:0.2em;">
          </div>
        </td>
      </tr>
    </table>
  </div>

<div class='slide '>
<!-- === begin markdown block ===

      generated by markdown/1.2.0 on Ruby 2.4.1 (2017-03-22) [x86_64-darwin16]
                on 2017-05-16 09:45:10 +0900 with Markdown engine kramdown (1.13.2)
                  using options {}
  -->

<!-- _S9SLIDE_ -->
<h1 id="section">研究目的</h1>
<ul>
  <li>当研究室では  処理の単位を Code Gear、 データの単位を Data Gear を用いて 信頼性が高い並列処理を行う Gears OS を開発している</li>
  <li>Gears OS では Task を Code Gear が実行するときに必要な Input Data Gear と出力の Output Data Gear の組で表現される。 Input Data Gear/Output Data Gear によって依存関係が決定し、それにそって並列実行を行う.</li>
  <li>信頼性の確保はモデルチェック、検証等を使用して行う。この信頼性のための計算は通常の計算とは別の階層のメタ計算として記述する。</li>
  <li>また、 メタ計算は信頼性の他に CPU, GPU などの実行環境の切り替え, データ拡張等の柔軟性を提供する。</li>
  <li>本研究では、 Gears OS の並列処理機構の実装を行う。また、並列処理の実行の検証をメタ計算として記述することで、 並列処理の精度を保証する。</li>
</ul>


</div>
<div class='slide '>
<!-- _S9SLIDE_ -->
<h1 id="section-1">メタ計算を使った並列処理</h1>

<ul>
  <li>並列処理のプログラミングはチューニングや信頼性の確保が困難である。</li>
  <li>これらのチューニングや信頼性の確保を、アルゴリズムから分離しメタ計算で行う</li>
  <li>当研究室ではメタ計算をサポートする言語として Continuation based C (CbC) を開発している</li>
  <li>CbC は Code Gear と Data Gear という処理とデータの単位を持っており、メタ計算を行うための Meta Code Gear、Meta Data Gear がある。</li>
  <li>Gears OS は CbC を用いて記述されている。</li>
</ul>


</div>
<div class='slide '>
<!-- _S9SLIDE_ -->
<h1 id="code-gear--data-gear">Code Gear と Data Gear</h1>

<ul>
  <li>Code Gear、Data Gear とはプログラムを分解し、処理の部分を Code Gear、int や文字列などの部分を Data Gear という単位で分割したものである。</li>
  <li>Code Gear の処理はその処理に必要な Data Gear が揃ってから行われる。</li>
  <li>Code Gear の処理の移行は goto を使って行い、この移行の仕方を継続という。</li>
  <li>goto による移行は戻り値を持たないためスタックを積む必要がない、スタックに積まない継続を軽量継続と呼ぶ。</li>
</ul>


</div>
<div class='slide '>
<!-- _S9SLIDE_ -->
<h1 id="code-gear--data-gear-">Code Gear と Data Gear を用いた計算</h1>

<ul>
  <li>Code Gear の処理の実行は、接続された Data Gear が揃ってから実行される。</li>
  <li>Data Gear には Output Data Gear と Input Data Gear があり、Code Gear は接続された Data Gear 以外を変更することはない。</li>
  <li>そのため通常の計算ではポインタを気にすることはなく、ポインタ演算はメタ計算部分で行われる。</li>
  <li>Code Gear の処理は tail call のみで遷移を行う関数型プログラミングである。</li>
  <li>agda に変換することができ、プログラムの正しさを証明できる。</li>
</ul>


</div>
<div class='slide '>
<!-- _S9SLIDE_ -->
<h1 id="code-gear-">Code Gear の例</h1>
<ul>
  <li>下記のコードは Code Gear で記述された stack のコードの一部である</li>
  <li>Code Gear は __code Code Gear で始まり、次の Code Gear へ goto で遷移する。</li>
</ul>

<pre lang="c"><code>_code clearSingleLinkedStack(struct SingleLinkedStack* stack,__code next(...)) {
    stack-&gt;top = NULL;
    goto next(...);
}

__code pushSingleLinkedStack(struct SingleLinkedStack* stack,union Data* data, __code next(...)) {
    Element* element = new Element();
    element-&gt;next = stack-&gt;top;
    element-&gt;data = data;
    stack-&gt;top = element;
    goto next(...);
}
</code></pre>


</div>
<div class='slide '>
<!-- _S9SLIDE_ -->
<h1 id="section-2">メタ計算</h1>

<ul>
  <li>メタ計算は通常の計算のための計算である</li>
  <li>並列処理の依存関係の解決、GPUなどのアーキテクチャ実行のための処理を行う。</li>
  <li>Gears OSではメタ計算は Meta Code Gear、Meta Data Gear で表現される。</li>
</ul>

<div style="text-align: center;">
    <img src="./fig/meta_gear.svg" alt="message" width="800" />
</div>


</div>
<div class='slide '>
<!-- _S9SLIDE_ -->
<h1 id="gears-os">Gears OS</h1>

<ul>
  <li>Gears OS は Code Gear、Data Gearの単位を用いて開発されており、CbCで記述される。</li>
  <li>Meta Code Gear は 通常の Code Gear の直後に遷移され、メタ計算を実行する。</li>
  <li>Meta Code Gear で OS の機能であるメモリ管理やスレッド管理を行う。</li>
  <li>
    <p>stub
# Gears OS の並列性</p>
  </li>
  <li>Code Gear が処理するのに必要な Input Data Gear と処理実行後に出力される Input Data Gear の組を Task と呼ぶ。</li>
  <li>Code Gear は Task 以外とは依存関係がない</li>
  <li>依存関係が明確化されている Code Gear Data Gear で記述することで、並列化し易い。</li>
</ul>

<div style="text-align: center;">
    <img src="./fig/codeGear_dataGear.svg" alt="message" width="600" />
</div>


</div>
<div class='slide '>
<!-- _S9SLIDE_ -->
<h1 id="gears-os-">Gears OS の構成</h1>

<ul>
  <li>Gears OS は以下の要素で構成される。
    <ul>
      <li>Context - TaskQueue - TaskManager - Worker</li>
    </ul>
  </li>
</ul>

<div style="text-align: center;">
    <img src="./fig/gears_structure.svg" alt="message" width="800" />
</div>


</div>
<div class='slide '>
<!-- _S9SLIDE_ -->
<h1 id="context">Context</h1>

<ul>
  <li>Gears OS は Context と呼ばれる接続可能な Code Gear、Data Gear のリスト、Temporal Data Gear のためのメモリ空間等を持っている Meta Data Gear を持つ。</li>
  <li>Gears OS は必要な Code Gear、Data Gear に参照したい場合、この Context を通す必要がある。</li>
  <li>Context は 接続に必要な Code/Data Gear のリスト、 Data Gear を確保するためのメモリ空間、実行する Code Gear、Code Gear の実行に必要な Input Data Gear のカウンタ等をもっている。
<!-- context のこーど?  --></li>
</ul>

<div style="text-align: center;">
    <img src="./fig/gears_structure.svg" alt="message" width="800" />
</div>


</div>
<div class='slide '>
<!-- _S9SLIDE_ -->
<h1 id="taskmanager">TaskManager</h1>
<ul>
  <li>TaskManager は Task、Worker の生成</li>
  <li>Worker に生成した Task の送信</li>
  <li>生成した Worker の終了処理等を行う</li>
</ul>

<div style="text-align: center;">
    <img src="./fig/gears_structure.svg" alt="message" width="800" />
</div>


</div>
<div class='slide '>
<!-- _S9SLIDE_ -->
<h1 id="worker">Worker</h1>
<ul>
  <li>Worker は thread と実行する Task が入っている Queue を持っている。</li>
  <li>Worker は TaskManager から送信された Task を Queue から取り出し、Code Gear を実行する。</li>
  <li>Task は Context なので、Code Gear の実行に必要な Input Data Gear はその Context から参照される。</li>
  <li>Code Gear を実行した後は出力される Output Data Gear から依存関係を解決する。</li>
</ul>
<div style="text-align: center;">
    <img src="./fig/gears_structure.svg" alt="message" width="800" />
</div>


</div>
<div class='slide '>
<!-- _S9SLIDE_ -->
<h1 id="gpgpu">GPGPU</h1>
<ul>
  <li>もともとは画像出力や画像編集などの画像処理に用いられるGPUを画像処理以外に利用する技術のこと。</li>
  <li>GPUにはCPUに比べ多数のコアがあり、並列に処理することによってCPUよりも高速に処理を行うことができる。</li>
  <li>CPUに比べ複雑な計算ができない、GPU単体に直接命令を書き込むこともできないなど、問題点も存在する。</li>
</ul>


</div>
<div class='slide '>
<!-- _S9SLIDE_ -->
<h1 id="cpuwoker">CPUWoker</h1>
<ul>
  <li>Worker thread で動くTaskスケジューラーである。</li>
  <li>synchronized queue からTask の List を読み込み実行する。</li>
  <li>Data Gear の待ち合わせ管理を行う。</li>
  <li>CPU Worker はreceive Task というAPI を持ち、Taskがなくなるまで繰り返す。</li>
</ul>


</div>
<div class='slide '>
<!-- _S9SLIDE_ -->
<h1 id="cudawoker">CUDAWoker</h1>
<ul>
  <li>CPUWoker を再利用して作成するTask スケジューラー</li>
  <li>CUDA ライブラリの初期化を行う以外の動作はCPUWoker と全く同じとなる。</li>
  <li>GPU へのデータ転送及びGPU側でのTaskの実行はTaskのMeta Code Gear で行われる。</li>
</ul>

<!--  
# CbC による Gears OS 記述の問題

- Gears OS を CbC で実装する上でメタ計算の記述が煩雑であることがわかった。
- これらのメタ計算を自動生成することにより Gears OS を記述する上においてより良い構文をユーザーに提供することにした。  
-->

<!-- 順番まだ考えてない 

# interface の記述

- interface は呼び出しの引数になる Data Gear の集合であり、そこで呼び出される Code Gear のエントリである。  
- 呼び出される Code Gear の引数となる Data Gear はここで全て定義される。
- Code Gear、Data Gear に参照するために Context を通す必要があるが、interface を記述することでデータ構造の API と Data Gear を結びつけることが出来、 呼び出しが容易になった。

{:lang="c"}
~~~
typedef struct Stack<Impl>{
    union Data* stack;
    union Data* data;
    union Data* data1;
    __code whenEmpty(...);
    __code clear(Impl* stack,__code next(...));
    __code push(Impl* stack,union Data* data, __code next(...));
    __code pop(Impl* stack, __code next(union Data*, ...));
    __code pop2(Impl* stack, union Data** data, union Data** data1, __code next(union Data**, union Data**, ...));
    __code isEmpty(Impl* stack, __code next(...), __code whenEmpty(...));
    __code get(Impl* stack, union Data** data, __code next(...));
    __code get2(Impl* stack,..., __code next(...));
    __code next(...);
} Stack;
~~~

# stub Code Gear

- Code Gear が必要とする Data Gear を取り出す際に Context を通す必要があるが、Context を直接扱うのはセキュリティ上好ましくない。
- そこで Context から必要なデータを取り出して Code Gear に接続する stub Code Gear を定義し、これを介して間接的に必要な Data Gear にアクセスする。
- stub Code Gear は Code Gear 毎に生成され、次の Code Gear へと継続する間に挟まれる。

# stub Code Gear の生成

- stub Code Gear は Code Gear 毎に記述する必要があり、そのCode Gear の引数を見て取り出す Data Gear を選択する。
- stub Code Gear を 自動生成する generate stub を Perl スクリプトで作成することによって Code Gear の記述量を約半分にすることができる。
- stub を生成するために generate_stub は指定された cbc ファイルの __code型である Code Gear を取得し、引数から必要な Data Gear を選択する。
- generate_stub は引数と interface を照らし合わせ、Gearef または GearImpl を決定する。
- また、この時既に stub Code Gear が記述されている Code Gear は無視される。
- cbc ファイルから、生成した stub Code Gear を加えて stub を加えたコードに変換を行う。 

# Context の生成

- generate_context は Context.h、Interface.cbc、generate_stub で生成されたImpl.cbc を見て Context を生成する Perl スクリプトである。

-->


</div>
<div class='slide '>
<!-- _S9SLIDE_ -->
<h1 id="section-3">依存関係の解決</h1>
<ul>
  <li>Gears での依存関係の解決は Data Gear にQueueをもたせることで行う</li>
  <li>Queue には その Data Gear を待っている Task が入っている</li>
  <li>Task は実行に必要な Input Data Gear のカウンタを持っており, Data Gear は書き出されると、依存関係にある Taskのカウンタをデクリメントする</li>
  <li>全ての Input Data Gear が揃ったら, Taskを Worker に送信する</li>
</ul>

<div style="text-align: center;">
    <img src="./fig/dependency.svg" alt="message" width="800" />
</div>


</div>
<div class='slide '>
<!-- _S9SLIDE_ -->
<h1 id="task-">Task の設定におけるメタ計算の問題</h1>
<ul>
  <li>現在のGears OS では並列実行するTask の設定を以下の用に行っている。</li>
  <li>実行するCode Gear 揃っていない Input Data Gear の数、Input Data Gear/Output Data Gear への参照等をノーマルレベルで記述している。</li>
</ul>

<pre lang="c"><code>__code createTask(TaskManager* taskManager, Context* task, Integer *integer1, Integer *integer2, Integer *output) {
    task-&gt;next = C_add; // set Code Gear                                                                                                                          
    task-&gt;idgCount = 2; // set Input Data Gear Counter                                                                                                            
    task-&gt;data[task-&gt;idg] = (union Data*)integer1; // set Input Data Gear reference                                                                               
    task-&gt;data[task-&gt;idg+1] = (union Data*)integer2;
    task-&gt;maxIdg = task-&gt;idg + 2;
    task-&gt;odg = task-&gt;maxIdg; // Output Data Gear index                                                                                                           
    task-&gt;data[task-&gt;odg] = (union Data*)output; // set Output Data Gear reference                                                                                
    task-&gt;maxOdg = task-&gt;odg + 1;
    taskManager-&gt;next = C_createTask1;
    goto meta(context, taskManager-&gt;taskManager-&gt;TaskManager.spawn); // spawn task                                                                                
}

// code gear                                                                                                                                                      
__code add(Integer *integer1, Integer *integer2, Integer *output) {
    ....
}


</code></pre>


</div>
<div class='slide '>
<!-- _S9SLIDE_ -->
<h1 id="task--1">Task の設定におけるメタ計算の問題</h1>
<ul lang="c">
  <li>この記述方法では Meta Data Gearである Task を直接参照しているためノーマルレベルでの記述は好ましくなく、メタレベルでの記述を行いたい。</li>
  <li>そこで以下のような記述を新たに考案した。</li>
  <li>par goto は先に上げたCode1 に変換される記述であり、これによりノーマルレベルでは直接 Taskを参照せずに par goto の引数で Task の設定を行うことができる。</li>
  <li>この記述を拡張することでCPU GPUでの切り替えを行うことも可能であると考える。
- ここに論文で使っていたcode2をいれる</li>
</ul>
<pre><code>__code createTask(Integer *integer1, Integer *integer2, Integer *output, __code next(...)) {
    par goto add(integer1, integer2, output, __exit);
    goto next(...);
}
</code></pre>


</div>
<div class='slide '>
<!-- _S9SLIDE_ -->
<h1 id="section-4">従来のシステムとの比較</h1>
<ul>
  <li>Microware OS/9 モジュール単位で実装されたりreal-time OS</li>
  <li>compiler directve を使用する HPC</li>
  <li>CUDA</li>
  <li>golang</li>
  <li>Unix</li>
  <li>Aspect J</li>
</ul>


</div>
<div class='slide '>
<!-- _S9SLIDE_ -->
<h1 id="microware-os9">Microware OS/9</h1>
<ul>
  <li>モジュール単位で記述された real-time OS</li>
  <li>モジュール内部では普通のシステムコールを行う</li>
  <li>ユーザーレベルとシステムレベルの区別がある</li>
  <li>Gears OS はメタ計算のサポートがある</li>
  <li>データモジュールの待ち合わせなどの機能はOS/9にはない</li>
</ul>


</div>
<div class='slide '>
<!-- _S9SLIDE_ -->
<h1 id="compiler-directve">compiler directve</h1>
<ul>
  <li>C や Fortran のコードに並列用の指示を埋め込む手法</li>
  <li>専用のコンパイラが用意される</li>
  <li>チューニングはコンパイラが用意された機能の範囲内で行う</li>
  <li>Gears OS ではメタ計算を使った個別のチューニングが可能</li>
</ul>


</div>
<div class='slide '>
<!-- _S9SLIDE_ -->
<h1 id="cuda">CUDA</h1>
<ul>
  <li>NVIDIA GPGPU 用の並列計算ライブラリ</li>
  <li>並列計算を行うカーネルとデータ転送を記述する部分からなる</li>
  <li>細かい最適化を行う場合はDriver APIを用いる</li>
  <li>Gears OS ではCUDAのカーネルをそのまま Code Gear として使うことができる</li>
  <li>データ転送部分はメタレベルで記述する</li>
</ul>


</div>
<div class='slide '>
<!-- _S9SLIDE_ -->
<h1 id="golang">golang</h1>
<ul>
  <li>シンプルな構文を持つ並列言語</li>
  <li>go routine 単位で並列処理を行う</li>
  <li>データの単位としてunion が入っている</li>
  <li>go routine 間の通信はchannel を用いて行う</li>
  <li>Gears OSでは並列処理の実装はmeta Code Gear で行う</li>
  <li>通信は Data Gear の待ち合わせで行う</li>
</ul>


</div>
<div class='slide '>
<!-- _S9SLIDE_ -->
<h1 id="aspect-j">Aspect J</h1>
<ul>
  <li>Java用のメタ計算記述</li>
  <li>ポイントカットと呼ばれる方法でメタ計算を挟む位置を指定する</li>
  <li>計算のログを取るのに使われている</li>
  <li>Code Gear ではCode segment 間の任意の位置にメタ計算を挿入することができる</li>
  <li>Aspect J よりも下位のメタ計算を記述することが可能</li>
</ul>


</div>
<div class='slide '>
<!-- _S9SLIDE_ -->
<h1 id="section-5">まとめ</h1>
<ul>
  <li>Gears OS のメタ計算を用いた並列処理の実装について考察した</li>
  <li>CPUWoker とGPUWoker が稼働している</li>
  <li>ノーマルレベルでの記述からstub とContext そしてメタ計算部分を生成するスクリプトを作成した</li>
  <li>他のシステムとの比較を行った</li>
  <li>今後は実装の詳細化、raspberry pi への実装</li>
  <li>メタ計算を用いたプログラム検証などを行っていく</li>
</ul>
<!-- === end markdown block === -->
</div>


</div><!-- presentation -->
</body>
</html>