Slide 1

Slide 1 text

長谷川智希 𝕏 @tomzoh 2025/07/18 PHPカンファレンス関西2025 From Scripts to Silicon: A Journey Through the Layers of Computing PHPからはじめる コ ンピュータ ア ーキテクチャ

Slide 2

Slide 2 text

௕୩઒ஐر ͸͕ͤΘ ͱ΋͖ @tomzoh http://www.dgcircus.com デジタルサーカス株式会社 副団長CTO ॴଐ ٕज़ΧϯϑΝϨϯεओ࠻ دߘɾஶॻ 来たれ!PHPer!We are hiring! 𝕏

Slide 3

Slide 3 text

௕୩઒ஐر ͸͕ͤΘ ͱ΋͖ @tomzoh ٕज़ΧϯϑΝϨϯεӡӦࢀՃ ֤छϓϩάϥϜ։ൃ   $16 ϨτϩήʔϜػ ిࢠ޻࡞  Ϗʔϧ αοΧʔ؍ઓ ϨϯλϧΧʔτϨʔε  ΩϟϯϐϯάΧʔʜ ϥΠϑϫʔΫ 𝕏

Slide 4

Slide 4 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ 4 PHPからはじめる コ ンピュータ ア ーキテクチャ From Scripts to Silicon: A Journey Through the Layers of Computing

Slide 5

Slide 5 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ 今日のテーマ テーマ1 そもそもプログラムを実行するとは何か テーマ2 レイヤ化された近代のコンピュータアーキテクチャ 5

Slide 6

Slide 6 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ 👉 プログラムの実行 レイヤ化 6

Slide 7

Slide 7 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ プログラムの実行 • プログラムを実行するというのは具体的に何をしているか • みなさんはふだん何を実行していますか? • PHPのプログラム • Docker • JavaScript (ブラウザ, Node.js) • Safari, Chrome, … • PhpStorm • macOS 7

Slide 8

Slide 8 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ • プログラムの実行方法は2種類ある • CPUでの実行 (電気回路による実行) • プログラムでの実行 2つの実行 8 • マシン語(機械語) • アセンブラ • コンパイラ • インタープリタ • VM

Slide 9

Slide 9 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ 実行方式1: CPUでの実行 • CPUが実行できるのはマシン語(機械語・アセンブラ・アセンブリ言語)のみ • 単純な命令のセット / Instruction Set: IS • メインの機能は: データをメモリ/レジスタ(変数)の間でコピーする, 演算する, (条件付きで)ジャンプする ぐらい • CPUは「どこかにあるデータとデータを演算してどこかに格納する」装置 • レジスタは決まった名前/サイズで決まった個数あるだけ • Apple Siliconなら、X0〜X30 という名前の31本の64bit汎用レジスタと、V0〜V31という名前の 32本の128bit SIMD/浮動小数点レジスタ、いくつかの特定用途レジスタだけ • 命令は単体で完結し、文法はない • プログラム言語とは言えないぐらい原始的 • 電気回路として実装されている 9

Slide 10

Slide 10 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ マシン語 • 見た目は生のバイナリ • C3 11 11 の3バイトでアドレス 1111 にジャンプするとかそんな感じ • スパルタすぎる… • C3 = JP みたいに人間にとってわかりやすい名前が付いてたりする • JP 1111 って書くと少しマシ • ニーモニックという • ニーモニックで書かれたプログラム(テキストファイル)をマシン語のバイナリに変換する プログラムをアセンブラと呼ぶ • 転じてニーモニックで書かれたプログラムをアセンブラとかアセンブリ言語と言ったり 10

Slide 11

Slide 11 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ 実行方式2: プログラムでの実行 • プログラムがプログラムを読み込んで解釈して実行する • 例: PHPコマンド(プログラム)がPHPのソースコード(テキスト)を読み込んで解釈して実行する • 解釈 = パース • 読み込む対象はソースコード(テキスト)でないこともある • Java: ソースコード(テキストファイル / HelloWorld.java) → バイトコード(バイナリ / HelloWorld.class) → java HelloWorld • 読み込むものがテキストでもバイナリでも 実行 = 命令を読み込んでその通りに動作する ということ 11

Slide 12

Slide 12 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ PHPでの実行 php test.php • PHPが $a=100; という文字列を見たら… • 「$があるのでここから変数名で=は変数名に使えない文字なので 変数名は$aで確定。次の = は後ろに = が無いので代入確定で、 100 は数字。次に ; があるからここまで実行しよう」と解釈する • $a が既に存在しているかを調べる • 存在していなければメモリ上に $a の中身を保存する領域を確保する • メモリ上の $a の領域に 100 を書く • プログラマからは$aという変数に100が代入された様に見え る 12

Slide 13

Slide 13 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ インタープリタとコンパイラ • プログラムを順次実行していく実行形式をインタープリタと呼ぶ • プログラムを他の形式に変換するプログラムをコンパイラと呼ぶ • 変換動作はコンパイル 何から何に変換してもコンパイル • Cコンパイラ C言語のソースコード(テキストファイル)をアセンブリ(テキストファイル)に変換する • Javaコンパイラ Javaのソースコード(テキストファイル)をバイトコード(バイナリファイル)に変換 する • インタープリタとコンパイラは言語の特徴として対比して使われる • が昨今はインタープリタ言語でも内部でコンパイルしてたりして単純ではない 13

Slide 14

Slide 14 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ VM: Virtual Machine • バイナリのプログラムを実行するプログラムをVMと言ったりする • PC全体を仮想化したVMとは違う文脈 • PHP関連で言うと… • PHP VM, HHVM • PHP Opcodeを実行するプログラム • PHP Opcode = PHPのソースコードからコンパイルされたバイナリ • Java VM • Javaバイトコードを実行するプログラム • Javaバイトコード = Javaのソースコードからコンパイルされたバイナリ 14

Slide 15

Slide 15 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ ✅ プログラムの実行 👉 レイヤ化 15

Slide 16

Slide 16 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ レイヤ化されたコンピュータ • 現代のコンピュータはソフトウェア/ハードウェアともにレイヤ化・抽象化されている • 各レイヤは下のレイヤのことは気にしなくて良い • PHPプログラム(hello-world.php)から見たら下のレイヤは… • PHPコマンド • macOS, Linux, FreeBSD, Windows, Raspberry Pi OS, Solaris, … • Mac, PC, Raspberry Pi, … • Intel x86, Apple Silicon, ARM, SPARC, … • PHPプログラムはPHPコマンドで実行されることだけ意識すれば良い (だいたいは) 16

Slide 17

Slide 17 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ 我々がPHPプログラムを実行すると… $ php hello-world.php • CPUはphpコマンドを実行する • Cで書かれたプログラム • = 機械語の実行ファイル • CPUでそのまま実行できる • phpコマンドはPHPプログラムを読み込み、解釈し、 実行する 17 hello-world.php PHPプログラム phpコマンド macOS Apple Silicon CPU Mac Book Pro 機械語

Slide 18

Slide 18 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ Goだと…? $ go build hello-world.go; ./hello-world • CPUはhello-worldコマンドを実行する • Goで書かれたプログラムがコンパイルされ マシン語の実行ファイルになっている • CPUでそのまま実行できる 18 macOS Apple Silicon CPU Mac Book Pro hello-worldコマンド 機械語

Slide 19

Slide 19 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ PHPとGoの違い PHPもGoも CPUで動いているのは マシン語のプログラム PHP phpコマンドが PHPプログラムを都度解釈して 実行する → コンパイル不要 Go 事前にコンパイルしてCPUが 直接実行可能なファイルを作る → ハイパフォーマンス 19 hello-world.php PHPプログラム phpコマンド macOS Apple Silicon CPU Mac Book Pro hello-worldコマンド macOS Apple Silicon CPU Mac Book Pro 機械語 機械語

Slide 20

Slide 20 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ もっと下のレイヤ 下のレイヤを気にしないのは もっと下のレイヤでも同じ Linuxは自分を実行しているの が実CPUなのか仮想CPUなの かは気にしない VM = CPUとハードウェアのフ リをするソフトウェア 20 hello-world.php PHPプログラム phpコマンド Linux CPU PC

Slide 21

Slide 21 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ ✅ プログラムの実行 ✅ レイヤ化 21

Slide 22

Slide 22 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ さて… 22

Slide 23

Slide 23 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ これで普通(?)の話は 終わりです 23

Slide 24

Slide 24 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ プログラムが 下のレイヤを気にしないなら 24

Slide 25

Slide 25 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ 下は下で好きにやりましょう 25

Slide 26

Slide 26 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ hello-world.php PHPプログラム phpコマンド macOS Apple Silicon CPU Mac Book Pro ここは何でもいい 下のレイヤが好きにやる • PHPで書かれたプログラム(hello-world.php)は 自分より下を気にしていない • 自分を実行してくれれば何でも良い 26

Slide 27

Slide 27 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ HelloWorld.class Javaバイトコード javaコマンド macOS Apple Silicon CPU Mac Book Pro ここは何でもいい 下のレイヤが好きにやる Javaで書かれ、 Javaバイトコードに 変換されたプログラム (HelloWorld.class) javaコマンドより下は 気にしていない なら好きにしてみよう 27

Slide 28

Slide 28 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ PHPで書かれたJVM PHPJava • @m3m0r7 さん • バイトコードを解釈し実行する • PHPerKaigi 2019 にトークあります 28 https://fortee.jp/phperkaigi-2019/proposal/6f792375-335a-432e-b1b4-7b649a5152e4 https://github.com/php-java/php-java

Slide 29

Slide 29 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ ほかにも… 29

Slide 30

Slide 30 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ みんな大好き ファミコン • カートリッジ左半分には6502*1用の マシン語プログラムが格納されている • カートリッジ右半分には キャラクタのパターンが格納されている • このマシン語プログラムも当然 下のレイヤは気にしない 30 CPU 6502 PPU RAM V-RAM APU カートリッジ マシン語 プログラム キャラクタ パターン *1 実際は6502ベースのカスタムIC RP2A03 ここは何でもいい 話は聞かせて もらった!! 好きにさせて もらおう!

Slide 31

Slide 31 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ PHPで書かれたファミコンエミュレータ • php-terminal-nes-emulator *1 • 長谷川が書いた • bokuweb/ fl ownes という既存の JavaScript実装の写経 • PHPでファミコンのCPU/PPUの フリをする 31 *1 https://github.com/hasegawa-tomoki/php-terminal-nes-emulator

Slide 32

Slide 32 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ ファミコンエミュレータの実装 32

Slide 33

Slide 33 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ ファミコンエミュレータの実装 - CPU • CPUのエミュレーションは簡単 • マシン語を実行するだけだから • マシン語は構文解析なしに実行できる • 前提事項: PCレジスタ = イマココ • STA abs • 引数で指定されたメモリにAレジスタの データを保存する 33 class Cpu { : $instruction = $this->ram[$this->pc]; $this->pc++; : switch ($instruction){ : case 0x8d: $addr = $this->ram[$this->pc] * 16 + $this->ram[$this->pc + 1]; $this->ram[$addr] = $this->a; $this->pc += 2; break; : $8D $00 $20 $8004 STA abs ※ php-terminal-nes-emulatorのコードではありません めちゃ単純 0x8004 0x8005 0x8006 16bitの値 上位8bit 下位8bit

Slide 34

Slide 34 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ ファミコンエミュレータの実装 - 画面 • CPUから特定のメモリアドレスを 読み書きするとPPUにアクセスできる • ということは↓の世界 34 CPU 6502 PPU RAM V-RAM APU カートリッジ マシン語 プログラム キャラクタ パターン $this->ram[$addr] = $value • CPU命令実行したあとに PPUのターンを作る • PPUのターンではVRAM(とキャラクタパタ ーン)の内容を絵にするだけ while (true) { $this->cpu->run(); $renderingData = $this->ppu->run(); $this->renderer->render($renderingData); }

Slide 35

Slide 35 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ エミュレータ、 難しくはないでしょ? 35

Slide 36

Slide 36 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ ほかにも… 36

Slide 37

Slide 37 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ ほかの下のレイヤが好きにやるパターン • Rosetta 2 • Apple SiliconのMacでIntel CPU用のバイナリ(マシン語)を動作させる • Intel CPU用バイナリはIntel CPUの上で動いているつもりで動作してる • Switch 2 • Switch (1)のソフトは(だいたい)そのまま動く • GPUまわり、JIT的に下のレイヤで変換してるっぽいですね • Docker • Mac上でLinux用プログラムを動作させる • 仮想マシン • 物理PC上で仮想PCを動作させる • EC2とかさくらのVPSとか 37

Slide 38

Slide 38 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ という訳で… 38

Slide 39

Slide 39 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ ✅ プログラムの実行 ✅ レイヤ化 ✅ 好きにやる 39

Slide 40

Slide 40 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ まとめ • そもそもプログラムを実行するとは何か • CPUでの実行 (機械語 / 電気回路による実行) or プログラムでの実行 • PHP, Java, Go の実行モデルの違い • 近代のコンピュータはレイヤー化されている • 下のレイヤのことは気にしなくて良くなっている • 仮想サーバとかDockerとかエミュレータはその特性を活かしている • エミュレータは難しくない • 低レイヤを楽しもう 40 長谷川 智希 @tomzoh 𝕏

Slide 41

Slide 41 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ おまけ 41

Slide 42

Slide 42 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ 電気回路として実装されたCPU 42 加算機 演算データの セレクタ 1byteの上位4bit = 機械語命令 何と何を 演算に使うかを決定 演算結果を どこに保存するかを 決定 PCレジスタ レジスタか I/Oのどれかと… Output 機械語の 引数を… 演算して どこかに保存する 1byteの下位4bit = 機械語の引数 汎用(演算用) レジスタ 2 × 4bit Input

Slide 43

Slide 43 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ プログラムの実行 43 実例いくつか

Slide 44

Slide 44 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ PHP インタープリタ ソースコードをphpコマンドで実行 44

Slide 45

Slide 45 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ Python インタープリタ ソースコードをpython3コマンドで実行 45 print('Hello, world.') hello-world.py

Slide 46

Slide 46 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ JavaScript インタープリタ ソースコードを nodeコマンドで実行 複数のエンジン実装 • Node.js • ブラウザ • Bun 複数の実装があるのは珍しくはない • RubyとかPythonも実装が複数ある • PHPもHackがあったり 46 console.log("Hello, world."); hello-world.js

Slide 47

Slide 47 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ Java コンパイラ + インタープリタ ソースコード(テキストファイル) Javaコンパイラ → バイトコード(バイナリ) javaコマンドで実行 $ java HelloWorld 47 class HelloWorld { public static void main(String[] args){ System.out.println("Hello, world."); } } HelloWorld.java 00000000 ca fe ba be 00 00 00 37 00 1d 0a 00 06 00 0f 09 |.......7........| 00000010 00 10 00 11 08 00 12 0a 00 13 00 14 07 00 15 07 |................| 00000020 00 16 01 00 06 3c 69 6e 69 74 3e 01 00 03 28 29 |........()| 00000030 56 01 00 04 43 6f 64 65 01 00 0f 4c 69 6e 65 4e |V...Code...LineN| 00000040 75 6d 62 65 72 54 61 62 6c 65 01 00 04 6d 61 69 |umberTable...mai| : HelloWorld.class

Slide 48

Slide 48 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ C コンパイラ ソースコード(テキストファイル) Cコンパイラ → ニーモニック(テキストファイル) アセンブラ → マシン語(バイナリ) CPUで実行 $ ./hello-world 48 #include int main(void){ printf("Hello, world."); } hello-world.c .LC0: .string "Hello, world." : main: endbr64 pushq %rbp movq %rsp, %rbp leaq .LC0(%rip), %rdi movl $0, %eax call printf@PLT movl $0, %eax : hello-world.s 00000000 7f 45 4c 46 02 01 01 00 00 00 00 00 00 00 00 00 |.ELF............| 00000010 03 00 3e 00 01 00 00 00 60 10 00 00 00 00 00 00 |..>.....`.......| 00000020 40 00 00 00 00 00 00 00 80 39 00 00 00 00 00 00 |@........9......| : hello-world

Slide 49

Slide 49 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ Go コンパイラ ソースコード(テキスト) go build コマンド → マシン語(バイナリ) CPUで実行 ./hello-world go run hello-world.go でも実行できる (内部的に go build してるんじゃないかな…) 49 package main import "fmt" func main() { fmt.Printf("Hello, world.") } hello-world.go 00000000 7f 45 4c 46 02 01 01 00 00 00 00 00 00 00 00 00 |.ELF............| 00000010 03 00 3e 00 01 00 00 00 60 10 00 00 00 00 00 00 |..>.....`.......| 00000020 40 00 00 00 00 00 00 00 80 39 00 00 00 00 00 00 |@........9......| : hello-world $ݴޠͱಉ͡ʹݟ͑Δ͚Ͳɺ&-'ϔομ͸ όΠτͱ͔͋ΔΒ͍͠ͷͰʜ

Slide 50

Slide 50 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ PHP インタープリタ ソースコードをphpコマンドで実行 $ php helllo-world.php 50

Slide 51

Slide 51 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ 6502のマシン語 • 基本的に1バイト = 1命令 • 命令に引数が付くこともある • $78(16進数の78) = SEI 命令 • $A9 = LDA 命令 • 引数の値をAレジスタに入れる 51 $78 $D8 $A9 $10 $8D $00 $20 $A2 $FF $9A $8000 $8001 $8002 $8004 $8007 $8009 SEI CLD LDA #imm STA abs LDX #imm TXS 78 D8 A9 10 8D 00 20 A2 FF 9A AD 02 20 10 FB AD 02 20 10 FB A0 FE A2 05 BD D7 07 C9 0A B0 0C CA $8000 $8010 00 01 02 03 04 05 06 07 08 09 0A 0B 0C 0D 0E 0F

Slide 52

Slide 52 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ 余談: 本当に左右に分かれてるファミコンカートリッジ 52 プログラム CPUに接続 キャラクタ PPUに接続

Slide 53

Slide 53 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ 余談: トランスパイル • トランスパイルという言葉もある • 何かから何かに変換するという意味ではコンパイルと同じ • プログラム言語からほかのプログラム言語への変換で使われる • TypeScript → JavaScript とか • JavaScriptのPoly fi llとか • 動作としてはコンパイルと同じ • C言語 → アセンブラの場合にはコンパイルと言う • 高水準言語 → 高水準言語の場合を明示的に示すために使われ始めた…? 53

Slide 54

Slide 54 text

長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ 余談: CとGoの違い • どちらも最終的にマシン語の実行ファイルができる • Cで作った hello-world と Go で作った hello-world • 動作は同じなのにサイズがだいぶ違う Cは17KB, Goは 2MB • 動的リンクと静的リンク • libc • 静的リンクすると852KB 54 $ ls -alFh 合計 60K -rwxrwxr-x 1 tom tom 17K Sep 12 04:16 hello-world* $ ls -alFh 合計 2.0M -rwxrwxr-x 1 tom tom 2.0M Sep 12 04:57 hello-world*