view slide.pdf.html @ 9:642787982a80

update
author anatofuz <anatofuz@cr.ie.u-ryukyu.ac.jp>
date Fri, 19 Apr 2019 18:24:04 +0900
parents d8feb607c44e
children 4b1eb4d69695
line wrap: on
line source






<!DOCTYPE html>
<html>
<head>
   <meta http-equiv="content-type" content="text/html;charset=utf-8">
   <title>Perl6の内部表現</title>

   <meta name="generator" content="Slide Show (S9) v4.1.0 on Ruby 2.6.3 (2019-04-16) [x86_64-darwin17]">
   <meta name="author"    content="Takahiro Shimizu" >

<!-- style sheet links -->
<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>

<style>
  .slide {page-break-after: always;}
</style>




</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">Perl6の内部表現</font></h1>
          </div>
        </td>
      </tr>
      <tr>
        <td>
          <div align="left">
               Takahiro Shimizu
               
            <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'>
  
<!-- _S9SLIDE_ -->
<h2 id="このセッションの内容">このセッションの内容</h2>

<ul>
  <li>Perl6の主要な実装であるRakudoの内部構造を探ります</li>
  <li>Rakudoの内部で利用されているVMや, Perl6のサブセットなどについて探索します</li>
  <li>スクリプト言語で主に使われているバイトコードインタプリタの気持ちになります</li>
</ul>



</div>

<div class='slide'>
  <!-- _S9SLIDE_ -->
<h2 id="内容">内容</h2>
<ul>
  <li>Perl6とは?</li>
  <li>スクリプト言語処理系の動き</li>
  <li>Perl6の内部構造
    <ul>
      <li>NQP</li>
      <li>MoarVM</li>
    </ul>
  </li>
  <li>MoarVMのバイトコード実行</li>
  <li>まとめ</li>
</ul>



</div>

<div class='slide'>
  <!-- _S9SLIDE_ -->
<h2 id="perl6とは">Perl6とは</h2>
<ul>
  <li>当初Perl5の時期バージョンとして開発されていたプログラミング言語
    <ul>
      <li>現在は別の言語として開発がそれぞれ進んでいる</li>
    </ul>
  </li>
  <li>仕様と実装が分離しており, 現在はテストが仕様となっている</li>
  <li>実装は歴史上複数存在しているが,主流な実装はRakudo</li>
  <li>言語的にはスクリプト言語であり, 漸進的型付き言語</li>
  <li>動作環境は、独自のVMのMoarVM, JVM、一部JavaScript上で動作する</li>
</ul>

<p><img src="2000px-Camelia.svg.png" alt="" style="width: 31%; height: auto;" /></p>



</div>

<div class='slide'>
  <!-- _S9SLIDE_ -->
<h2 id="現在のperl6">現在のPerl6</h2>

<ul>
  <li>現在のバージョンは <code>6.d</code></li>
  <li><a href="https://perl6.github.io/6pad/">ブラウザ上で実行可能な環境</a>が存在する</li>
  <li><a href="https://commaide.com/">IDE</a>が開発されている</li>
  <li>WebApplicationFrameworkなども開発されており、 Perl5のモジュールを移行したものがいくつか存在する</li>
  <li>日本では趣味のプロダクト以外社会では使用されていない
    <ul>
      <li>海外では実際に使われているケースも存在する</li>
    </ul>
  </li>
  <li>処理速度では一部Perl5に勝っているが、それでも大分遅い</li>
</ul>



</div>

<div class='slide'>
  <!-- _S9SLIDE_ -->
<h2 id="参考perl5のソースコード">[参考]Perl5のソースコード</h2>

<ul>
  <li>Perl5時代
    <ul>
      <li>スカラ、配列、ハッシュの3種類</li>
      <li>それぞれの変数への参照であるリファレンスが使用可能</li>
    </ul>
  </li>
</ul>

<pre><code class="language-perl">use ustrict;
use warnings;

my $scalar_value = "hello!";
print "$scalar_value\n";

my @array = (1..10);
print "$array[0]\n";

my %hash = ( this_is_key =&gt; "this_is_value");
print "$hash{this_is_key}\n";

my $hash_ref = \%hash;
print "$hash_ref-&gt;{this_is_key}\n";
</code></pre>



</div>

<div class='slide'>
  <!-- _S9SLIDE_ -->
<h2 id="perl6のソースコード概要">Perl6のソースコード概要</h2>

<ul>
  <li>Perl5の文法とは比較的変更が多い
    <ul>
      <li>雰囲気は似ている</li>
    </ul>
  </li>
  <li>変数がオブジェクトと化した事により, 変数からsayメソッドを呼ぶことが可能</li>
</ul>

<pre><code>my $str_value = 'hello world!';
$str_value.say; # hello world!
</code></pre>

<ul>
  <li>Perl5と同様に,変数にはデフォルトでは型がないような振る舞いをする</li>
</ul>

<pre><code>my $sample_value = 'hello world!';
$sample_value.say; # hello world!

$sample_value = '31';
$sample_value.say; # 31

say($sample_value * 3);
</code></pre>



</div>

<div class='slide'>
  <!-- _S9SLIDE_ -->
<h2 id="perl6の言語的な特徴">Perl6の言語的な特徴</h2>

<ul>
  <li>漸進的型付き言語である為, 型を強制することも可能となる</li>
</ul>

<pre><code>my Int $int_value  = 31;
$int_value = "hello"; # Compile error!
</code></pre>

<pre><code>$ perl6 type_invalid.p6
Type check failed in assignment to $int_value; expected Int but got Str ("hello")
  in block &lt;unit&gt; at type_invalid.p6 line 4
</code></pre>



</div>

<div class='slide'>
  <!-- _S9SLIDE_ -->
<h2 id="perl6の言語的な特徴-1">Perl6の言語的な特徴</h2>

<ul>
  <li>型を独自に定義することも可能</li>
  <li>入力の型によって実行する関数を変える事などができる</li>
</ul>

<pre><code class="language-perl6">my subset Fizz of Int where * %% 3;
my subset Buzz of Int where * %% 5;
my subset FizzBuzz of Int where Fizz&amp;Buzz;
my subset Number of Int where none Fizz|Buzz;

proto sub fizzbuzz ($) { * }
multi sub fizzbuzz (FizzBuzz) { "FuzzBuzz" }
multi sub fizzbuzz (Fizz) { "Fizz" }
multi sub fizzbuzz (Buzz) { "Buzz" }
multi sub fizzbuzz (Number $number) { $number }

fizzbuzz($_).say for 1..15;
</code></pre>

<ul>
  <li>型を利用したFizzBuzz</li>
</ul>



</div>

<div class='slide'>
  <!-- _S9SLIDE_ -->
<h2 id="スクリプト言語">スクリプト言語</h2>
<ul>
  <li>Perl6は現状コンパイルすることはできない
    <ul>
      <li>スクリプト言語の分類</li>
    </ul>
  </li>
  <li>現在広く使われているスクリプト言語(Perl,Python,Ruby…)などとPerl6の構成は類似している</li>
  <li>今回はPerl6の実装を追いながら、最近のスクリプト言語処理系の大まかな実装を理解する</li>
</ul>



</div>

<div class='slide'>
  <!-- _S9SLIDE_ -->
<h2 id="スクリプト言語処理系">スクリプト言語処理系</h2>
<ul>
  <li>スクリプト言語は入力として与えられたソースコードを、 直接評価せずにバイトコードにコンパイルする形式が主流となっている</li>
  <li>その為スクリプト言語の実装は大きく2つで構成されている
    <ul>
      <li>バイトコードに変換するフロントエンド部分</li>
      <li>バイトコードを解釈する仮想機械</li>
    </ul>
  </li>
</ul>

<p><img src="fig/bytecode_sample_generally_lang.svg" width="80%" /></p>



</div>

<div class='slide'>
  <!-- _S9SLIDE_ -->
<h2 id="perl6以外のスクリプト言語">Perl6以外のスクリプト言語</h2>

<ul>
  <li>現在使われているプロセスVMは言語に組み込まれているものが多い</li>
  <li>JVMやElixirなどのVMは複数の言語で使用されている</li>
  <li>Java
    <ul>
      <li>JVM</li>
    </ul>
  </li>
  <li>Ruby
    <ul>
      <li>YARV</li>
    </ul>
  </li>
  <li>Python
    <ul>
      <li>PythonVM</li>
    </ul>
  </li>
  <li>Elixir
    <ul>
      <li>BEAM</li>
    </ul>
  </li>
</ul>



</div>

<div class='slide'>
  <!-- _S9SLIDE_ -->
<h2 id="perl6の処理系の構成">Perl6の処理系の構成</h2>

<ul>
  <li>Perl6の処理系で現在主流なものはRakudoと呼ばれる実装である(歴史上複数存在する)</li>
  <li>Rakudoは3つのレイヤーから構成されている
    <ul>
      <li>Perl6インタプリタ</li>
      <li>Perl6インタプリタを記述するPerl6のサブセットNQP</li>
      <li>Perl6のバイトコードを解釈するMoarVM</li>
    </ul>
  </li>
  <li>Perl6/NQPがフロントエンドに相当し、MoarVMがバックエンドに相当する</li>
</ul>



</div>

<div class='slide'>
  <!-- _S9SLIDE_ -->
<h2 id="rakudoの構成図">Rakudoの構成図</h2>

<p><img src="fig/Rakudo_System_overview.png" alt="" /></p>

<p>(http://brrt-to-the-future.blogspot.com/2015/03/advancing-jit-compiler.html)</p>



</div>

<div class='slide'>
  <!-- _S9SLIDE_ -->
<h2 id="perl6とnqp">Perl6とNQP</h2>

<ul>
  <li>NQP(NotQuitPerl Perl)
    <ul>
      <li>Perl6のサブセット。Perl6っぽい言語</li>
    </ul>
  </li>
  <li>Perl6、 NQP自体がNQPで記述されている</li>
  <li>NQPもNQPで記述されている為、 セルフビルド(自分自身で自分自身をコンパイルする)を行う</li>
  <li>NQPはPerl6の文法をベースにしているが、 制約がいくつか存在する</li>
  <li>元々はPerl6の主力実装がParrotだった時代に登場
    <ul>
      <li>文法がアップデートされており、当時の資料は古くなっている</li>
    </ul>
  </li>
</ul>

<pre><code>my $value := "hello!";
say($value);
</code></pre>



</div>

<div class='slide'>
  <!-- _S9SLIDE_ -->
<h2 id="nqpスクリプト">NQPスクリプト</h2>

<ul>
  <li>変数は束縛 <code>:=</code> を使う</li>
  <li>関数の間に空白を入れてはいけない</li>
  <li>再帰呼び出しを使うフィボナッチ数列</li>
</ul>

<pre><code>#! nqp
sub fib($n) {
    $n &lt; 2 ?? $n !! fib($n-1) + fib($n - 2);
}

my $N := 29;

my $z  := fib($N);

nqp::say("fib($N) = " ~ fib($N));
</code></pre>



</div>

<div class='slide'>
  <!-- _S9SLIDE_ -->
<h2 id="nqpスクリプトnまでの整数の和">NQPスクリプト(nまでの整数の和)</h2>

<pre><code class="language-perl6">sub add_test($n){
    mu $sum := 0;
    while ( $n &gt; 1) {
        $sum := $sum + $n;
        --$n;
    }
    return $sum;
}

say(add_test(10000));
</code></pre>



</div>

<div class='slide'>
  <!-- _S9SLIDE_ -->
<h2 id="nqpとオペコード">NQPとオペコード</h2>

<ul>
  <li>NQPはPerl6の中で一番レイヤーが低い言語</li>
  <li>その為、 実行するVMのオペコード(処理単位)を使用することができる</li>
</ul>



</div>

<div class='slide'>
  <!-- _S9SLIDE_ -->
<h2 id="nqpとmoarvm">NQPとMoarVM</h2>
<ul>
  <li>NQPそのものは実行することはできない</li>
  <li>NQPの実行にはMoarVM/JVMが必要となる
    <ul>
      <li>NQPコンパイラが各VMに対応したバイトコードに変換する</li>
    </ul>
  </li>
</ul>



</div>

<div class='slide'>
  <!-- _S9SLIDE_ -->
<h2 id="perl6のvm">Perl6のVM</h2>
<ul>
  <li>MoarVM, JVM , JavaScriptが選択可能
    <ul>
      <li>メインで開発されているのはMoarVMであり、 他のVMは機能が実装されていないものが存在する</li>
    </ul>
  </li>
  <li><code>rakudo-star</code> というPerl6のパッケージ環境では、 MoarVMがデフォルトでインストールされる</li>
</ul>



</div>

<div class='slide'>
  <!-- _S9SLIDE_ -->
<h2 id="moarvm">MoarVM</h2>
<ul>
  <li>C言語で記述されているPerl6専用の仮想機械</li>
  <li>レジスタマシン
    <ul>
      <li>型情報を持つレジスタに対しての演算として処理される</li>
      <li>Rubyなどはスタックマシンとして実装されている</li>
    </ul>
  </li>
  <li>LuaJITなどを利用したJITコンパイルなども可能</li>
  <li>Perl6やNQPは、MoarVMに対してライブラリなどを設定して起動する</li>
</ul>



</div>

<div class='slide'>
  <!-- _S9SLIDE_ -->
<h2 id="バイトコード">バイトコード</h2>
<ul>
  <li>Perl6も、Rakudo/NQPはバイトコードに変換され、 バイトコードをVMが実行する</li>
  <li>バイトコード実行部分は、 命令に対応するバイト列を読み込み、 解釈し、 次の命令を読み取ることを繰り返す</li>
</ul>


</div>


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