32.5K Views
March 24, 23
スライド概要
第四回関数型プログラミング(仮)の会
都内在住のエンジニア
脱・初級者のための 自作GBエミュレータ開発 押谷 倫 Lin Oshitani @linoscope 第四回関数型プログラミング(仮)の会
新しいプログラミング言語 を勉強する際、 このように感じたことはありませんか?
● 簡単なプログラム なら書けるが、 中規模以上のプログラム をどうやって書けばよいのか分からない ● 高度な言語機能 を勉強しなんとなく理解した気になったが、 実践のなかで どのように活用 すればいいのかが分からない
● 簡単なプログラム なら書けるが、 中規模以上のプログラム をどうやって書けばよいのか分からない ● 高度な言語機能 を勉強しなんとなく理解した気になったが、 実践のなかで どのように活用 すればいいのかが分からない 一年半前に を本格的に勉強しはじめたときの 自分
このときの心情
初級者 本 趣味 自分 簡単なプログラム
初級者 中級者 本 趣味 自分 簡単なプログラム 業務 オープンソース活動
初級者 中級者 本 趣味 自分 簡単なプログラム いまやってることの延長線上で たどりつける気がしない。。 業務 オープンソース活動
初級者 中級者 本 趣味 自分 簡単なプログラム 業務 いまやってることの延長線上で たどりつける気がしない。。 中級の谷 オープンソース活動
どうする?
とりあえず実践をつむために なにか実装してみよう
ゲームボーイエミュレータ!
OCaml で書かれた ゲームボーイエミュレータ js_of_ocaml という ライブラリを使って JS にコンパイル https://github.com/linoscope/CAMLBOY
僕は CAMLBOY の実装を通して、 前述の「谷」をある程度乗り越えることができたと 感じました。 本発表の目標は、その経験を少しでも共有することです。
なぜGBエミュレータ?
仕様がはっきりしていて「何を実装するか」 を考える必要がない 競技プログラミングに近い。 「テストケースが通るかどうか」=「ゲームのカートリッジが動くかどうか」
「中規模以上」のコード 「中規模以上」 =「テストなしでは開発しづらく、その結果、テストを書きやすいように設計を しなければいけないコード」
思い入れがある 家族が寝静まってから、布団に潜って、 ゲームボーイライトでポケモンを遊んだ記憶
実装
そもそもエミュレータって?
ゲーム機 ハードウェア マシンコード ... VRAM LD A, B CPU JMP 0x1200 ADD A, 0x12 LD A, C 入力 RAM JMP 0x1201 出力 ADD A, 0x12 LD A, 0x12 カートリッジ挿入口 ... https://www.suruga-ya.jp/produ ct/detail/165005345 https://item.fril.jp/76e17f08bf11abdf5d244a3dde1c3079 https://www.copetti.org/writings/consoles/game-boy/
エミュレータ ハードウェア マシンコード ... VRAM LD A, B CPU JMP 0x1200 ADD A, 0x12 入力 LD A, C RAM JMP 0x1201 出力 ADD A, 0x12 LD A, 0x12 カートリッジ挿入口 ... https://www.suruga-ya.jp/produ ct/detail/165005345 https://item.fril.jp/76e17f08bf11abdf5d244a3dde1c3079 https://www.copetti.org/writings/consoles/game-boy/ 入 ハードウェアを 「模倣」するから 「エミュレータ」 力 ソフトウェア 出力
実装の概略図
RAM Timer Joypad CPU Bus Registers Cartridge GPU Serial Port
RAM Timer Joypad CPU Bus Registers Cartridge GPU Serial Port
RAM Timer Joypad CPU Bus Registers Cartridge GPU Serial Port
CPU と IO デバイスを つなぐの 「仲介役」 RAM Timer Joypad CPU Bus Registers Cartridge GPU Serial Port
RAM Timer Joypad CPU Bus Registers read 0x0000~0x3FFF カートリッジから データを 読み出す Cartridge GPU Serial Port
メモリに データを 読み書きする RAM Timer Joypad read/write 0xC000~0xCFFF CPU Bus Registers Cartridge GPU Serial Port
タイマーの スピードを 変更する RAM CPU Timer Joypad write 0xFF07 Bus Registers Cartridge GPU Serial Port
CPU と Bus の実装
最初の実装 cpu.ml bus.ml type t = { type t = { bus : Bus.t; ... : Cartridge.t; ram : Ram.t; ... } ... cartridge } let x = Bus.read t.bus addr in ... let read t addr = (* read を中継 *) let write t addr data = (* write を中継 *) ...
最初の実装 cpu.ml bus.ml type t = { type t = { bus ... } : Bus.t; cartridge : Cartridge.t; ram : Ram.t; 問題がある!... ... } let x = Bus.read t.bus addr in ... let read t addr = (* read を中継 *) let write t addr data = (* write を中継 *) ...
問題: CPU→Bus→Cartridge/… が密結合していてCPUを単体でテストできない bus.ml cpu.ml type t = { type t = { bus : Bus.t; ... } cartridge : Cartridge.t; ram : Ram.t; ... 依存 } ... let x = Bus.read t.bus addr in ... let read t addr = ... let write t addr data = ... ... 依存 cartridge .ml ram.ml ...
そういえば Functorってのがあっ たな。。
解決法: Functor でCPUをBusの実装ではなく
インターフェースに依存させることができる。
cpu.ml
bus_intf.ml
module Make (Bus : Bus_intf.S) = struct
type t = {
bus
: Bus.t;
...
}
...
let x = Bus.read t.bus addr in
...
end
module type S = sig
依存
type t
val read
val write
: t -> addr -> uint16
: t -> addr -> uint16 -> unit
...
end
実装
bus.ml
実装
mock_bus.ml
CPUの単体テスト内では実際の Bus の実装ではなく Mock_bus をつかう test_cpu.ml module Cpu = Cpu.Make(Mock_bus) ...
Functorを使えば、 「実装」ではなく「インターフェース」に依存する 疎結合 で テスタブル なコードがかける
CPUの実装(より詳しく)
CPUには 8-bit レジスタ と 16-bit レジスタがある registers.mli type t (** 8-bit レジスタの識別子 *) type r = A | B | C | D | E | F | H | L (** 16-bit レジスタの識別子 *) type rr = AF | BC | DE | HL (** 8-bit レジスタの read/write *) val read_r : t -> r -> uint8 val write_r : t -> r -> uint8 -> unit (*16-bit レジスタの read/write*) val read_rr : t -> rr -> uint16 val write_rr : t -> rr -> uint16 -> unit
命令セットには 8-bit命令 と 16-bit命令 がある # 8-bit バージョン。 # A レジスタの中身 (8-bit) に 0x12 を足し、結果を Aレジスタに保存する ADD8 A, 0x12 # 16-bit バージョン。 # AF レジスタの中身 (16-bit) に 0x1234 を足し、結果を AFレジスタに保存する ADD16 AF, 0x1234 では、この命令セットはどのようにして定義すればよい?
最初に試した方法: ヴァリアント(代数的データ型) instruction.ml type arg = | R of Registers.r (* 8-bit レジスタ *) | RR of Registers.rr (* 16-bit レジスタ *) | Immediate8 of uint8 (* 8-bit の値 *) | Immediate16 of uint16 (* 16-bit の値 *) | ... type t = | ADD8 of arg * arg | ADD16 of arg * arg | ...
最初に試した方法: ヴァリアント(代数的データ型) instruction.ml type arg = | R of Registers.r (* 8-bit レジスタ *) | RR of Registers.rr (* 16-bit レジスタ *) | Immediate8 of uint8 (* 8-bit の値 *) | Immediate16 of uint16 (* 16-bit の値 *) | ... type t = | ADD8 of arg * arg | ADD16 of arg * arg | ... うまくいかない!
なんで? → 命令を実行する execute の実装でつまる! cpu.ml let execute t (inst : Instruction.t) = let read_arg arg = match arg with | R r -> Registers.read_r r | RR rr -> Registers.read_rr rr | ... in match inst with | Add8 (x, y) -> let sum = Uint8.add (read_arg x) (read_arg y) in ... | Add16 (x, y) -> let sum = Uint16.add (read_arg x) (read_arg y) in ...
なんで? → 命令を実行する execute の実装でつまる! cpu.ml let execute t (inst : Instruction.t) = let read_arg arg = match arg with | R r -> Registers.read_r r ここでつまる! | RR rr -> Registers.read_rr rr | ... in match inst with | Add8 (x, y) -> let sum = Uint8.add (read_arg x) (read_arg y) in ... | Add16 (x, y) -> let sum = Uint16.add (read_arg x) (read_arg y) in ...
問題:マッチしたコンストラクタによって返り値が変わって型がつかない cpu.ml (* 返り値の型は??? *) let read_arg (arg : Instruction.arg) -> ??? = match arg with | R r -> (* R にマッチしたときの返り値は uint8 *) Registers.read_r r | RR rr -> (* RR にマッチしたときはの返り値は uint16 *) Registers.read_rr rr | ... in
そういえば GADTってのがあった な。。
解決法: GADT をつかう instruction.ml type arg = | R : Registers.r | RR : Registers.rr -> uint16 arg | Immediate8 : uint8 | Immediate16 : uint16 -> uint8 -> uint8 arg arg -> uint16 arg | ... type t =... (* type t はヴァリアントのときと一緒 *)
解決法: GADT (Generalized Algebraic Data Type) をつかう instruction.ml type arg = | R : Registers.r | RR : Registers.rr -> uint16 arg | Immediate8 : uint8 | Immediate16 : uint16 -> uint8 -> uint8 arg arg -> uint16 arg | ... type t =... (* type t はヴァリアントのときと一緒 *) どういう意味?
-> の左側: ヴァリアントの of のあとの部分に対応 コンストラクタをマッチしたときに手に入る値の型 | R : Registers.r -> uint8 arg | R of Registers.r let read_arg = function | R r -> .. (* r の型は Registers.r *) | RR rr -> .. (* rr の型は Registers.rr *) ...
| R : Registers.r -> uint8 arg | R of Registers.r let read_arg = function | R r -> .. (* r の型は Registers.r *) | RR rr -> .. (* rr の型は Registers.rr *) 右側は? ... ヴァリアントに対応 するものはない。
-> の右側: コンストラクタをマッチしたときに返す値の型 | R : Registers.r -> uint8 arg | RR : Registers.rr -> uint16 arg let read_arg = function | R r -> Registers.read_r r (* uint8 を返す *) | RR rr -> Registers.read_rr r (* uint16 を返す *) ...
ヴァリアント は パターンマッチで手に入る値の型をパラメータ化できる。 GADT は パターンマッチで返す値の型もパラメータ化できる だから、"Generalized" Algebraic Data Type (多分)
cpu.ml let execute t (inst : Instruction.t) = let read_arg : type a. a Instruction.arg -> a = fun arg -> | R r -> Registers.read_r r | RR rr -> Registers.read_rr rr | ... in match inst with | Add8 (x, y) -> let sum = Uint8.add (read_arg x) (read_arg y) in ... | Add16 (x, y) -> let sum = Uint16.add (read_arg x) (read_arg y) in
実装のタイムライン
タイムライン (git log から) ● 2021/8/15: カートリッジからの命令の読み込み・命令の実行 を実装しはじめる ○ テストロムを使って最終テスト。結果はシリアルポートに出力されるので、 GPUが実装されていなくても動く。 ● 2021/9/19: 命令セットの実装を完了、GPU の実装を開始 ● 2021/10/5: Tetris の開始画面の描画に成功、Joypad や Cartridge の実装を開始 ● 2021/10/23: ポケモンが動く ● 2021/10/24: JS にコンパイル、ブラウザで動くが、めちゃくちゃ遅い。最適化に取 り組む。 ● 2021/11/14: スマホブラウザで動くくらいの速度がでる。
まとめ
なにか 実装してみる 問題 本やドキュメント で勉強 そういえば あれが 使えるかも... 解決 自分の理解に 落とし込む 勉強した内容を 思い出す
なにか 実装してみる 問題 本やドキュメント で勉強 言語機能だけではなく、 テストフレームワーク・ビルドシステム・ライブラリ の理解も同様にすすんだ! そういえば あれが 使えるかも... 解決 自分の理解に 落とし込む 勉強した内容を 思い出す
最後に
初級者 中級者 本 趣味 簡単なプログラム 業務 中級の谷 オープンソース活動
初級者 中級者 「谷」どう越える? 本 趣味 簡単なプログラム 業務 中級の谷 オープンソース活動
初級者 中級者 「谷」どう越える? 本 趣味 簡単なプログラム 中規模以上の プログラムを書いて 「問題」「解決」 を繰り返す 中級の谷 業務 オープンソース活動
僕にとってこの「中規模以上のプログラム」 は CAMLBOY でした
ほかには?
おすすめする三つの条件 1. 適度に複雑 ○ ○ ある程度以上の複雑性からのみ出現する「問題」がある (テスタビリティ、保守性、など) 書く自信がないくらいがちょうどいい。 2. 仕様が決まっている ○ 「何を書くか」と「どう書くか」を同時に考えるのは大変。 ○ ゴールがわかりやすい。例:自作エミュレータ上でポケモンを動かす 3. 楽しい ○ テストプレイと称して何時間も溶かした。 例: ゲームエミュレータ(ゲームボーイ、ファミコン、chip8、…)、 既存言語(のサブセット)のコンパイラ、ISUCON移植、など
リンク集 ● ● ブログ記事: ○ https://qiita.com/linoscope/items/244d931aaae07df2c27e ○ https://linoscope.github.io/writing-a-game-boy-emulator-in-ocaml/ CAMLBOY のレポジトリ: ○ ● https://github.com/linoscope/CAMLBOY CHIP8 エミュレータ: ○ https://yukinarit.github.io/cowgod-chip8-tech-reference-ja/1_about_chip8.html ● Build your own X: https://github.com/codecrafters-io/build-your-own-x ● 発表者の Twitter (質問・コメントなどがあればこちらまで!) ○ https://twitter.com/linoscope
ボツスライドおきば
なぜ OCaml を勉強しはじめた ?
2021年半ば
仕事 同僚 オフィス コロナ前
仕事 仕事 同僚 オフィス コロナ前 オフィス 同僚 コロナ後
重要度が増した! せっかくなら面白い ことをやりたい。 仕事 同僚 オフィス コロナ前 仕事 オフィス 同僚 コロナ後
重要度が増した! せっかくなら面白い ことをやりたい。 仕事 同僚 オフィス コロナ前 仕事 オフィス で仕事したら面白いことができそう。 同僚 コロナ後