tag:crieit.net,2005:https://crieit.net/tags/MinRuby/feed 「MinRuby」の記事 - Crieit Crieitでタグ「MinRuby」に投稿された最近の記事 2023-01-22T17:44:40+09:00 https://crieit.net/tags/MinRuby/feed tag:crieit.net,2005:PublicArticle/18373 2023-01-22T17:38:20+09:00 2023-01-22T17:44:40+09:00 https://crieit.net/posts/ruby-minruby-recursive-descent-parser 『RubyでつくるRuby』のMinRubyのパーサを書いた(手書きの再帰下降パーサ) <p>これは <a target="_blank" rel="nofollow noopener" href="https://qiita.com/advent-calendar/2022/ruby">Ruby Advent Calendar 2022</a> の25日目の記事です。</p> <p>書籍 <strong>『RubyでつくるRuby ゼロから学びなおすプログラミング言語入門』</strong>(以下、 <strong>『RubyでつくるRuby』</strong> )で扱われているミニ言語 MinRuby のパーサを書いてみました。</p> <hr /> <p>『RubyでつくるRuby』は、Ruby を使った基本的なプログラミングの入門から始まり、後半ではミニ言語 MinRuby(Ruby のサブセット)のインタプリタを作って最終的に MinRuby言語で MinRuby インタプリタを書くところまでやってしまう、という内容の入門書です。</p> <hr /> <p>……と自分なりに内容紹介してみましたが、以下もあわせて参考にしていただければと思います。</p> <ul> <li><a target="_blank" rel="nofollow noopener" href="https://mametter.hatenablog.com/entry/20170315/p1">書籍『Ruby でつくる Ruby』が発売されます - まめめも</a> <ul> <li>著者の遠藤さん</li> </ul></li> <li><a target="_blank" rel="nofollow noopener" href="https://golden-lucky.hatenablog.com/entry/2023/01/10/131903">『RubyでつくるRuby』の読み方(私論) - golden-luckyの日記</a> <ul> <li>ラムダノートの鹿野さん</li> </ul></li> </ul> <hr /> <p><a target="_blank" rel="nofollow noopener" href="https://www.lambdanote.com/products/ruby-ruby">RubyでつくるRuby ゼロから学びなおすプログラミング言語入門 – 技術書出版と販売のラムダノート</a></p> <ul> <li><a target="_blank" rel="nofollow noopener" href="https://www.lambdanote.com/products/ruby-ruby-ebook">PDF版のみの販売ページ</a></li> <li><a target="_blank" rel="nofollow noopener" href="https://www.amazon.co.jp/dp/4908686017">Amazon</a></li> </ul> <blockquote> <p>プログラミングを始めるなら、プログラミング言語を自分でつくってみるのがいちばん! 最低限の機能なら、こんなに簡単にインタプリタを作れます。よくわからなかったプログラミングも、裏側の仕組みから分かってしまえば怖くない! </p> <p>2016年9月から2017年1月にかけて<a target="_blank" rel="nofollow noopener" href="https://ascii.jp/serialarticles/1230449/">アスキーjpの「プログラミング+」コーナーで連載された大好評のWebコンテンツ『Rubyで学ぶRuby』</a>を、さらにわかりやすく紙版の書籍として編纂しなおして発売するものです。豊富なイラストもカラーで完全採録。</p> </blockquote> <ul> <li>プログラミング初心者を想定して丁寧に解説してある</li> <li>分量が多すぎず(144ページ)読み通しやすい</li> <li>評価器にフォーカスしており、パーサはスコープ外 <ul> <li>パースは <a target="_blank" rel="nofollow noopener" href="https://rubygems.org/gems/minruby">minruby gem</a> にまかせる</li> </ul></li> </ul> <p>たしか発売された頃に読んで「せっかくだからパーサも自作したい」と思った記憶があります。当時はパーサの作り方についての知識が足りずどうすればよいか分からなかったのですが、その後あれこれあって簡単なものなら作れるようになりました。</p> <h1 id="できたもの"><a href="#%E3%81%A7%E3%81%8D%E3%81%9F%E3%82%82%E3%81%AE">できたもの</a></h1> <p><a target="_blank" rel="nofollow noopener" href="https://github.com/sonota88/cookpad-hackarade-minruby/tree/recursive-descent-parser">https://github.com/sonota88/cookpad-hackarade-minruby/tree/recursive-descent-parser</a></p> <pre><code class="terminal"> $ wc -l rcl_*.rb my_minruby_parser.rb 33 rcl_common.rb 65 rcl_lexer.rb 496 rcl_parser.rb 17 my_minruby_parser.rb 611 合計 </code></pre> <p>一応動いてはいます。advent calendar には間に合った……リファクタリングなどは冬休み以降の宿題にします。</p> <p>(2023-01-22 追記) interp.rb がパースできるようになりました。</p> <h1 id="動作例"><a href="#%E5%8B%95%E4%BD%9C%E4%BE%8B">動作例</a></h1> <pre><code class="terminal"> $ cat test3-4.rb n = 1 while n < 100 if n % 3 == 0 if n % 5 == 0 p("FizzBuzz") else p("Fizz") end else if n % 5 == 0 p("Buzz") else p(n) end end n = n + 1 end $ ruby my_minruby_parser.rb test3-4.rb [:stmts, [:var_assign, "n", [:lit, 1]], [:while, [:<, [:var_ref, "n"], [:lit, 100]], [:stmts, [:if, [:==, [:%, [:var_ref, "n"], [:lit, 3]], [:lit, 0]], [:if, [:==, [:%, [:var_ref, "n"], [:lit, 5]], [:lit, 0]], [:func_call, "p", [:lit, "FizzBuzz"]], [:func_call, "p", [:lit, "Fizz"]]], [:if, [:==, [:%, [:var_ref, "n"], [:lit, 5]], [:lit, 0]], [:func_call, "p", [:lit, "Buzz"]], [:func_call, "p", [:var_ref, "n"]]]], [:var_assign, "n", [:+, [:var_ref, "n"], [:lit, 1]]]]]] </code></pre> <h1 id="cookpad-hackarade-minruby"><a href="#cookpad-hackarade-minruby">cookpad-hackarade-minruby</a></h1> <p>まずは先行事例を軽く調べました。</p> <p><a target="_blank" rel="nofollow noopener" href="https://techlife.cookpad.com/entry/2018/10/16/131000">Hackarade #04: Create Your Own Interpreter - クックパッド開発者ブログ</a></p> <blockquote> <p>完全セルフホスト:MinRubyでパーサを書き、minruby gemに依存せずにinterp.rb単体でセルフホストするようにした</p> </blockquote> <p>さすがのクックパッドさん。今回私はここまではやっていません。パーサを MinRuby で書くのもおもしろそう……。</p> <p>同記事により、 mame/cookpad-hackarade-minruby というリポジトリでテストケースが用意されていることを知りました。ありがたく有効活用させていただきましょう。</p> <p><a target="_blank" rel="nofollow noopener" href="https://github.com/mame/cookpad-hackarade-minruby">mame/cookpad-hackarade-minruby: material for Cookpad's Hackarade #4</a></p> <p>今回はこのテストケースを使い、自作パーサと minruby gem に同じ入力を与えて同じ結果になればヨシ! ということにしました。</p> <h1 id="方針"><a href="#%E6%96%B9%E9%87%9D">方針</a></h1> <p>パーサ入門でよくある再帰下降パーサにしました。</p> <p>専用のパーサライブラリを使うのに比べるとデメリットもあり本格的な言語実装ではあまり採用されない手法だと思いますが、入門者目線だと「全部自分で書いたぞ感」「全部把握できてる感覚」が得られるという、とても良い良さがあります。</p> <p>(ちなみに、Rust製 Ruby 実装 <a target="_blank" rel="nofollow noopener" href="https://github.com/sisshiki1969/ruruby">ruruby</a> のパーサは手書きだそうです。すごい……。)</p> <p>ここで選択肢が2つあり、</p> <ul> <li>(1) 何もないところから全部書く</li> <li>(2) Ruccola のパーサを改造する</li> </ul> <p>ちょっとだけ (1) を試した後、時間(というか計画性)がないこともあって (2) に切り替えました。<br /> さて、Ruccola というのがスッと出てきましたね。これはなんでしょうか。</p> <h1 id="Ruccola とは"><a href="#Ruccola+%E3%81%A8%E3%81%AF">Ruccola とは</a></h1> <p>Ruccola は私が自分の勉強のために作っている素朴な自作プログラミング言語です。</p> <p><a target="_blank" rel="nofollow noopener" href="https://github.com/sonota88/ruccola">https://github.com/sonota88/ruccola</a></p> <p>最新の知見を盛り込んだキラリと光る言語……とかでは全然なく、現代の水準からするとかなり原始的なものです。自分の勉強用・入門用なので素朴でよいと割り切っています。</p> <p>2021年にセルフホストできた(詳しくは↓の記事を参照)後も趣味の盆栽プログラミング的にちびちびと機能追加やリファクタリングを続けています。</p> <p><a target="_blank" rel="nofollow noopener" href="https://qiita.com/sonota88/items/1e683276541cf1b87b76">素朴な自作言語Ruccolaのコンパイラをセルフホストした - Qiita</a></p> <ul> <li>コンパイラを Ruby で書いている</li> <li>Ruccola言語のコードの見た目は Ruby っぽい</li> <li>なんとなくC言語っぽいものをイメージして作ったが、現時点では型がない(整数しかない)ので <a target="_blank" rel="nofollow noopener" href="https://ja.wikipedia.org/wiki/B%E8%A8%80%E8%AA%9E">B言語</a> や <a target="_blank" rel="nofollow noopener" href="https://ja.wikipedia.org/wiki/BCPL">BCPL</a> の方が近いのかも(詳しくないのでボンヤリした書き方)</li> </ul> <p>ざっくり書くとこんな感じです。このうち「見た目は Ruby っぽい」というのがポイントで、Ruccola のパーサ( <a target="_blank" rel="nofollow noopener" href="https://github.com/sonota88/ruccola/blob/00fc70c2f76e56737e290b541a69e9c36f9bb968/rcl_parser.rb">202-12-25 時点のソース</a> )にいくつか手を加えれば MinRuby のパーサとして使えそうでした。</p> <p>コード例です。以下は Ruccola言語で書いた cat コマンド。</p> <pre><code class="ruby">def main() var EOF = -1; var c; while (true) c = getchar(); if (c == EOF) break; end write(c, 1); end end </code></pre> <p>だいたい Ruby ですね。中身は原始的なのに見た目は Ruby 風なので、書いているとちょっと不思議な気分になります。</p> <p>作者の目にはだいたい Ruby と同じに見えますが、Ruby に慣れている方には次のような点が目に付くのではないでしょうか。</p> <ul> <li>文末の <code>;</code> (必須)</li> <li>while の条件式を囲む <code>(</code> <code>)</code> (必須)</li> <li>変数宣言の var(必須)</li> </ul> <p>他にも関数呼び出しの <code>(</code> <code>)</code>、関数定義の仮引数を囲む <code>(</code> <code>)</code>、エントリポイントとなる main 関数も必須です。</p> <p>自分が言語実装ビギナーなので、難しいことをしなくていいように Ruby の文法にいろいろ制限を加えてこうなっています。Ruby と完全にコンパチではありませんが、構文ハイライトやエディタのインデント支援は Ruby 向けのものがだいたいそのまま使えています。</p> <p>ちなみに、Ruccola よりももっとコンパクトな <strong>mini-ruccola</strong>(元は vm2gol-v2 という名前だった)もあります。VM〜コンパイラを割と素朴に書いて1500行以下というもの。自分が欲しい(難しすぎず、簡単すぎず、読み書きに慣れている言語向けの)教材が見つけられず、じゃあ自分で作ってしまえとなって作ったもの。</p> <p><a target="_blank" rel="nofollow noopener" href="https://github.com/sonota88/vm2gol-v2">https://github.com/sonota88/vm2gol-v2</a></p> <p>以前週刊Railsウォッチで紹介していただきました。</p> <p><a target="_blank" rel="nofollow noopener" href="https://techracho.bpsinc.jp/hachi8833/2021_02_09/103838">週刊Railsウォッチ(20210209後編)Rubyでミニ言語処理系を作る、Kernel#getsの意外な機能、CSSのcontent-visibilityほか|TechRacho by BPS株式会社</a></p> <p>作った順番としては mini-ruccola の方が先で、最初にこっちで簡単なコンパイラ実装になんとか入門し、その後セルフホストに必要な機能を追加していく形で Ruccola を作っていきました。</p> <h1 id="主な変更点"><a href="#%E4%B8%BB%E3%81%AA%E5%A4%89%E6%9B%B4%E7%82%B9">主な変更点</a></h1> <h2 id="MinRuby の AST に合わせる"><a href="#MinRuby+%E3%81%AE+AST+%E3%81%AB%E5%90%88%E3%82%8F%E3%81%9B%E3%82%8B">MinRuby の AST に合わせる</a></h2> <p>Ruccola の AST は現時点では MinRuby と同じく入れ子の配列(のようなもの)で木構造を表す方式です。そこは同じ。ただ細かいところが違うので合わせていきます。</p> <p>簡単なところでいえば、たとえば以下は変数への代入です。</p> <pre><code class="diff">- [:set, var_name, expr] + [:var_assign, var_name, expr] </code></pre> <p>ものによってはこの程度の変更でOK。</p> <h2 id="演算子の優先順位"><a href="#%E6%BC%94%E7%AE%97%E5%AD%90%E3%81%AE%E5%84%AA%E5%85%88%E9%A0%86%E4%BD%8D">演算子の優先順位</a></h2> <p>Ruccola では自分でも手に負えるようにいろいろと単純化しています。演算子の優先順位もそのひとつで、たとえば <code>1 + 2 * 3</code> という式は <code>(1 + 2) * 3</code> として扱われます(単純に左結合とする)。</p> <p>えっそれでいいの? と思いましたか? 思いますよね。</p> <p>言語処理系の解説を見ると必ずといっていいほど演算子の優先順位の話題が出てきて、自作言語を作るには避けて通れないトピックであるかのように思われます。私もある時までそう思い込んでいたのですが、実はそんなことはありません……と思うんですよね(微妙な自信のなさ)。</p> <p>Ruccola では <code>1 + (2 * 3)</code> と解釈させたい場合は明示的に括弧で優先順位を指定して <code>1 + (2 * 3)</code> と書けばよい、ということにしています。たまに自分でも忘れていて素でびっくりします。まあ自分の勉強用の言語だからいいんですよこれで。最初から実装しなくてもよいので後回しにしています。Ruccola言語の記述力がまだ低いため、今入れてもコード量が膨れるデメリット(セルフホストしているのでばかにならず、下手に機能追加すると自分が困る)の方が大きそうで嫌かなという考えもあり。</p> <p>ちなみに、似たような方針を採っている言語として <a target="_blank" rel="nofollow noopener" href="http://middleriver.chagasi.com/electronics/vtl.html">VTL (Very Tiny Language)</a> という言語があると後から知りました。仲間がいた……。</p> <blockquote> <p>VTLでは,演算子の間に優先順位は存在せず,左から順番に演算が行われます.演算の順序を変えたい場合は括弧"()"をつけます.例えば,"2+3*4"の結果は"20"となり,"2+(3*4)"の結果は"14"となります.</p> </blockquote> <p>あ、ていうか優先順位を明示する言語といえば Lisp があるじゃないか、とこの記事を書いていて今思い至りました。Lisp、心強いですね。</p> <p>さて、Ruccola ではそれでいいとして、演算子の優先順位に対応しないと cookpad-hackarade-minruby のテストが通りません。なんとかしなければ。</p> <p>これについては自分にとって今回初めてという訳ではなく、以前単体で履修していたのでした。</p> <p><a target="_blank" rel="nofollow noopener" href="https://memo88.hatenablog.com/entry/2020/07/05/111103">四則演算と剰余のみのexprコマンドをRubyで作ってみた</a></p> <p>これを組み込めばいけるはず。</p> <p>test1-4.rb と test1-5.rb の間の飛躍が大きかったので小さなテストを追加しながら進めました。</p> <ul> <li>まずは一番優先順位の低い <code>1 == 2</code> から</li> <li>二番目に優先順位の低い <code><</code> を右側に加えた <code>1 == 2 < 3</code></li> <li>...</li> <li><code>1 == 2 < 3 + 4 * 5</code> が <code>(1 == (2 < (3 + (4 * 5))))</code> と解釈できればOK</li> <li>ここまでできたら test1-5.rb が通る</li> </ul> <p>という流れで進めることで意外とすんなり書き換え完了しました。</p> <p>参考: <a target="_blank" rel="nofollow noopener" href="https://docs.ruby-lang.org/ja/latest/doc/spec=2foperator.html">演算子式 (Ruby 3.1 リファレンスマニュアル)</a></p> <p>優先順位まわりについては書籍『<a target="_blank" rel="nofollow noopener" href="http://esolang-book.route477.net/">Rubyで作る奇妙なプログラミング言語</a>』もおすすめです。私はこの本の写経で入門しました。</p> <h2 id="if と case"><a href="#if+%E3%81%A8+case">if と case</a></h2> <p>Ruccola では if 文は case 文のシンタックスシュガーという扱いになっています。「case文が動けば if文はそのサブセット扱いにできるよね」という、大は小を兼ねる的な素朴な発想によるものです。あと Ruccola の前身の mini-ruccola は最初はパーサがなくて構文木を手書きするところからスタートしたため、その都合もあります。</p> <p>MinRuby では逆になっていて、 case 式は内部的に入れ子の if 式としてパースされます(『RubyでつくるRuby』 p85)。ここは今回『RubyでつくるRuby』を読み返してなるほどと思ったところでした。</p> <h2 id="改行の扱い"><a href="#%E6%94%B9%E8%A1%8C%E3%81%AE%E6%89%B1%E3%81%84">改行の扱い</a></h2> <p>Ruccola では現時点では if文の条件式全体を囲む括弧は省略不可です(case文、while文も同様)。また、代入や関数呼び出しなどの文の末尾にはセミコロンが必須です。改行はスペースと同じように字句解析の段階で捨てています。</p> <pre><code class="ruby">if (i == 10) p(1); end </code></pre> <p>一方 MinRuby では条件式を囲む括弧や関数呼び出しなどの末尾のセミコロンは省略可能です。改行の考慮が必要そう。</p> <pre><code class="ruby">if i == 10 p(1) end </code></pre> <p>改行を考慮したパースはこれまで経験がなく不確実な部分でしたが、結論からいえば特別な対応は不要で、適当に改行を読み飛ばすだけで cookpad-hackarade-minruby のテストは通りました(というか試しにやってみたら字句解析の段階で全部捨てても大丈夫だった)。</p> <p>今の実装では、式とみなせるまとまりの次に二項演算子が来たら「まだ式の続きがあるな」と判断し、そうでなければ「いったんそこで式が終わっているな」と判断する、という動作になっています。<br /> たとえば上のコード例だと <code>i == 10</code> の次に <code>p</code> というトークンが来ています。これは二項演算子ではありませんから、ここで式の区切りと判断され、<code>i == 10</code> と <code>p</code> 以降は別のまとまりだということになります。</p> <pre><code>i == 10 + ... ^ 二項演算子なのでまだ式が続いていると判断 i == 10 p ... ^ 二項演算子ではないので p からは別のまとまり </code></pre> <p>あくまで cookpad-hackarade-minruby のテストのカバー範囲と今回作ったパーサの実装の組み合わせでは問題なかったというだけで、Ruby に近づけていこうとするとどこかでちゃんとした対応が必要になってくるでしょうね。</p> <h2 id="配列・ハッシュ"><a href="#%E9%85%8D%E5%88%97%E3%83%BB%E3%83%8F%E3%83%83%E3%82%B7%E3%83%A5">配列・ハッシュ</a></h2> <p>Ruccola では配列リテラルや配列の要素にアクセスするための専用の構文は(まだ)ありません。ハッシュはサポート予定なし。</p> <p>なので配列・ハッシュまわりは新たに書きました。ここはすんなり書けてしまって特に書くことがありません。</p> <h2 id="TODO"><a href="#TODO">TODO</a></h2> <ul> <li>if, case まわりがいいかげんなのでもうちょいちゃんとやる</li> <li>interp.rb をパースできるようにする(ここまではやりたい)</li> </ul> <p>(2023-01-22 追記) 上記2つ完了しました。「これで完璧!」というものではありませんが、とりあえず interp.rb はパースできています。</p> <h1 id="おわりに"><a href="#%E3%81%8A%E3%82%8F%E3%82%8A%E3%81%AB">おわりに</a></h1> <p>以上のように自作言語 Ruccola のパーサにいくつかの変更を加えることで MinRuby のパーサを作ることができました(ただし cookpad-hackarade-minruby のテストを通すところまで)。</p> <p>以前から MinRuby のパーサ作れないかなーとボンヤリ考えていたのでこの機会に実現できてよかったです。</p> <hr /> <p>以下、おまけ的な雑多な話題・メモなどです。</p> <h1 id="おまけ"><a href="#%E3%81%8A%E3%81%BE%E3%81%91">おまけ</a></h1> <h2 id="再帰下降パーサについて"><a href="#%E5%86%8D%E5%B8%B0%E4%B8%8B%E9%99%8D%E3%83%91%E3%83%BC%E3%82%B5%E3%81%AB%E3%81%A4%E3%81%84%E3%81%A6">再帰下降パーサについて</a></h2> <p>再帰下降パーサについては書籍やウェブに解説がたくさんありますので「再帰下降 パーサ」「再帰下降 構文解析」などで調べてください。英語だと recursive descent parser です。</p> <p>今回作ったものはパーサだけで 500行くらいありますし、再帰下降パーサの一番最初の入門用としてはちょっと大きいかもしれません。そういう場合は分解してちょっとずつ手を付けるのがおすすめです。</p> <ul> <li>くりかえしの構造 <ul> <li>関数定義の仮引数、関数呼び出し時の引数、演算子を含む式、配列リテラル、ハッシュリテラルなど、いろんなところで登場する</li> <li><code>数</code> のあとに <code>, 数</code> をくりかえし</li> <li>例: <code>1, 2, 3</code></li> <li><code>数</code> のあとに <code>演算子 数</code> のくりかえし</li> <li>例: <code>1 + 2 + 3</code></li> </ul></li> <li><code>[[]]</code>, <code>[[[]]]</code> のような入れ子の構造</li> <li>演算子の優先順位</li> </ul> <p>上の方でも書いたように、演算子の優先順位は後回しにできますから、先にそれ以外の部分を作ってから後付けで追加するのもよいと思います。</p> <p>「MinRuby のパーサを書いてみたいけどいきなり全部書くのは無理そう……また今度にしよう」と思った方は、まず次のような簡単なものから手を付けてみるのはどうでしょう。どちらも50行くらいです。</p> <ul> <li><a target="_blank" rel="nofollow noopener" href="https://qiita.com/sonota88/items/816b6b8d7a362ef4e332">入れ子の配列だけをパースできるパーサを作る(手書きの再帰下降パーサ)</a> <ul> <li><code>((), (()))</code> のような入れ子の括弧だけをパースできるパーサ。コード例があった方がよさそうと思って簡単なものを昨日急いで書きました。</li> <li>n年前の自分に「これで入門しろ」「最初はこれでいいと思うよ」といって送りつけるとしたらこれ</li> </ul></li> <li><a target="_blank" rel="nofollow noopener" href="https://qiita.com/sonota88/items/5902f255e196b5b8b048">正規表現エンジン(ロブ・パイクのバックトラック実装)をRubyで写経した</a> <ul> <li>バックトラック方式の正規表現エンジンもおすすめです。こっちも手軽。手軽だけどおもしろい。再帰下降パーサに考え方が似ています。</li> </ul></li> </ul> <h2 id="MinRuby 関連"><a href="#MinRuby+%E9%96%A2%E9%80%A3">MinRuby 関連</a></h2> <p>今回調べていて見つけたもの。</p> <p><a target="_blank" rel="nofollow noopener" href="https://speakerdeck.com/m_seki/extend-your-own-programming-language-rubykaigi-2018">https://speakerdeck.com/m_seki/extend-your-own-programming-language-rubykaigi-2018</a></p> <p>MinRuby に末尾呼び出し最適化を実装する話や Rinda + MinRuby の話。</p> <p><a target="_blank" rel="nofollow noopener" href="https://matsubara0507.github.io/posts/2019-05-16-minruby-with-patternmatch.html">https://matsubara0507.github.io/posts/2019-05-16-minruby-with-patternmatch.html</a></p> <p>パーサは「木を組み立てる」処理がメインなのでパターンマッチの出番はそんなにありませんが、評価器は「木を使う」処理がメインなのでパターンマッチを使うといい感じに書けますね。</p> <h2 id="関連書籍"><a href="#%E9%96%A2%E9%80%A3%E6%9B%B8%E7%B1%8D">関連書籍</a></h2> <ul> <li><a target="_blank" rel="nofollow noopener" href="http://esolang-book.route477.net/">Rubyで作る奇妙なプログラミング言語</a></li> <li><a target="_blank" rel="nofollow noopener" href="https://tatsu-zine.com/books/scheme-in-ruby">つくって学ぶプログラミング言語 RubyによるScheme処理系の実装 - 達人出版会</a></li> </ul> <p>『RubyでつくるRuby』もそうですが、どの本も古びていない、これからも通用する、使い回しのきく内容だと思います。Ruby に慣れていて言語処理系に興味があるという人におすすめです。</p> <h2 id="mal (Make a Lisp)"><a href="#mal+%28Make+a+Lisp%29">mal (Make a Lisp)</a></h2> <p><a target="_blank" rel="nofollow noopener" href="https://github.com/kanaka/mal">https://github.com/kanaka/mal</a></p> <p>Lisp に興味のある方には mal(Make a Lisp)もおすすめです。ちょうど『RubyでつくるRuby』の Lisp 版といった感じで、簡単な Lisp インタプリタを作ってセルフホストするまでが11ステップに分かれており、テストを通しながら作っていきます。入門者向けのシンプルな実装ですが、マクロや例外機構、末尾呼び出しの最適化まで含まれています。すでに <a target="_blank" rel="nofollow noopener" href="https://github.com/kanaka/mal/tree/03b6cfd45c99db651ece70381a7c0f596fc14336/impls/ruby">Ruby 版の実装</a>(主な部分は500行程度)もリポジトリに含まれていますから、そのまま写経してもいいですし、自力で書いて詰まったときに参考にすることもできます。</p> <p>ちなみに私は一度 Ruby で写経した後 LibreOffice Basic で書くということをやりました。</p> <p><a target="_blank" rel="nofollow noopener" href="https://qiita.com/sonota88/items/501f0efa437bc4d53cc7">LibreOffice BasicでLispインタプリタ(mal)を書いた - Qiita</a></p> <h2 id="PicoRuby"><a href="#PicoRuby">PicoRuby</a></h2> <p>入門者向けに作られているものではありませんが、Ruby 関連の小さめの処理系というと PicoRuby がありますね。</p> <p><a target="_blank" rel="nofollow noopener" href="https://github.com/picoruby">https://github.com/picoruby</a></p> <p><a target="_blank" rel="nofollow noopener" href="https://github.com/picoruby/mruby-pico-compiler">https://github.com/picoruby/mruby-pico-compiler</a></p> <p>この記事の前日(2022-12-24)に開発者募集されていました。</p> <p><a target="_blank" rel="nofollow noopener" href="https://github.com/picoruby/picoruby/wiki/%E3%80%90%E3%81%8A%E8%AA%98%E3%81%84%E3%80%91PicoRuby%E3%81%AE%E9%96%8B%E7%99%BA%E8%80%85%E3%82%92%E5%8B%9F%E9%9B%86%E3%81%97%E3%81%A6%E3%81%84%E3%81%BE%E3%81%99">【お誘い】PicoRubyの開発者を募集しています · picoruby/picoruby Wiki</a></p> <p>Ruby よりは小さくて難しくなさそう?? 読めそうだったら参考のためにソース読んでみようかな……と思いつつまだ何もできていません。</p> <p>Lemon というパーサジェネレータを使ってるんですね。Lemon 知らなかった。SQLite が使っているものだそうです。</p> <ul> <li><a target="_blank" rel="nofollow noopener" href="https://sqlite.org/lemon.html">The Lemon LALR(1) Parser Generator</a></li> <li><a target="_blank" rel="nofollow noopener" href="https://github.com/picoruby/picoruby/wiki/Proposal-for-the-Grant-Program-2020-of-Ruby-Association">Proposal for the Grant Program 2020 of Ruby Association · picoruby/picoruby Wiki</a></li> </ul> <h2 id="雑多"><a href="#%E9%9B%91%E5%A4%9A">雑多</a></h2> <ul> <li><a target="_blank" rel="nofollow noopener" href="https://qiita.com/sonota88/items/97ba1f0c377fbe86d5b1">Raccでかんたんな自作言語のパーサを書いた</a></li> <li><a target="_blank" rel="nofollow noopener" href="https://qiita.com/sonota88/items/edce05e86d3248f49401">Parsletでかんたんな自作言語のパーサを書いた</a></li> </ul> <p>mini-ruccola のパーサを Racc と Parslet を使って書いたもの。</p> <hr /> <p><a target="_blank" rel="nofollow noopener" href="https://qiita.com/sonota88/items/6870ef5fe5b2a9275ab7">自作言語のコンパイラにオレオレアセンブリではなくx86_64アセンブリを生成させる(関数呼び出しと足し算だけ) - Qiita</a></p> <p>コンパイラ作ったらやはりこれもやっておきたい、ということで(ちょっとだけ)やったみたもの。RISC-V 版や WebAssembly 版もやってみたい。</p> <hr /> <p><a target="_blank" rel="nofollow noopener" href="https://qiita.com/sonota88/items/a5d6d3539e0fb8040f74">Ruby+DXOpalでリレー式論理回路シミュレータを自作して1bit CPUまで動かした - Qiita</a></p> <p>言語処理系をやるとさらに下のレイヤーも気になってきますよね? ということでついでに紹介。これも楽しかったです。Ruby(と <a target="_blank" rel="nofollow noopener" href="https://yhara.github.io/dxopal/doc/ja/index.html">DXOpal</a>)で作れます!</p> <h2 id="今年 Ruby 関連で書いたもの"><a href="#%E4%BB%8A%E5%B9%B4+Ruby+%E9%96%A2%E9%80%A3%E3%81%A7%E6%9B%B8%E3%81%84%E3%81%9F%E3%82%82%E3%81%AE">今年 Ruby 関連で書いたもの</a></h2> <p>せっかくのアドベントカレンダーなので今年書いたものも並べてみます。</p> <ul> <li><a target="_blank" rel="nofollow noopener" href="https://qiita.com/sonota88/items/b586e96f7805a5695a34">Galaaz を触ってみた(TruffleRuby + ggplot2 で散布図を描いてみた)</a></li> <li><a target="_blank" rel="nofollow noopener" href="https://qiita.com/sonota88/items/539c631b0f7dfb295fec">Ubuntu 22.04にJupyter NotebookとIRubyをインストール(pyenv, rbenv, Bundler を使用)</a></li> <li><a target="_blank" rel="nofollow noopener" href="https://qiita.com/sonota88/items/7e2ab396163c081657b1">Ruby + Victor でSVGお絵描き(簡単な散布図を描いてみた)</a></li> <li><a target="_blank" rel="nofollow noopener" href="https://qiita.com/sonota88/items/86afece3b2595648a656">SVG::Graph(svg-graph gem)で散布図を描く</a></li> <li><a target="_blank" rel="nofollow noopener" href="https://qiita.com/sonota88/items/50609275668fcd7511e4">Ruby + ruby_gnuplot(gnuplot gem)で散布図を描く</a></li> <li><a target="_blank" rel="nofollow noopener" href="https://qiita.com/sonota88/items/da02c6f9f85c5a33f0a1">Ruby + Numo::Gnuplot(numo-gnuplot gem)で散布図を描く</a></li> </ul> sonota486