Upgrade to Pro — share decks privately, control downloads, hide ads and more …

PHPからはじめるコンピュータアーキテクチャ / From Scripts to Silico...

PHPからはじめるコンピュータアーキテクチャ / From Scripts to Silicon: A Journey Through the Layers of Computing

PHPカンファレンス関西2025の資料です
---
現代のコンピュータはハードウェアから私たちプログラマが書くプログラムの動作までの間が多くのレイヤーに分けられて動作しています。
レイヤーは自分より下を抽象化し、下のレイヤーを詳しく理解しなくても多くの場合プログラマはプログラムを書けます。
一方、プログラムが期待した様に動作しない時には下のレイヤーの動作の理解が問題の解決の助けになることもあります。

このトークでは私たちが愛するPHPをスタート地点にして、「VMって何?」「 PHPやJavaとC言語の根本的な違い」など、コンピュータプログラムがどの様に動作するのかを解説します。

コンピュータのレイヤー構造を理解すると、いままでは見えていなかった角度からプログラミングを楽しめるようになります。
このトークを通じて、低レイヤーが好きになったり、いろいろなレイヤーで面白いことをしたりする方が増えることを期待しています!

Avatar for HASEGAWA Tomoki

HASEGAWA Tomoki

July 18, 2025
Tweet

More Decks by HASEGAWA Tomoki

Other Decks in Technology

Transcript

  1. 長谷川智希 𝕏 @tomzoh 2025/07/18 PHPカンファレンス関西2025 From Scripts to Silicon: A

    Journey Through the Layers of Computing PHPからはじめる コ ンピュータ ア ーキテクチャ
  2. ௕୩઒ஐر ͸͕ͤΘ ͱ΋͖ @tomzoh ٕज़ΧϯϑΝϨϯεӡӦࢀՃ ֤छϓϩάϥϜ։ൃ   $16 ϨτϩήʔϜػ

    ిࢠ޻࡞  Ϗʔϧ αοΧʔ؍ઓ ϨϯλϧΧʔτϨʔε  ΩϟϯϐϯάΧʔʜ ϥΠϑϫʔΫ 𝕏
  3. 長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ 実行方式1: CPUでの実行 • CPUが実行できるのはマシン語(機械語・アセンブラ・アセンブリ言語)のみ • 単純な命令のセット /

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

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

    • 解釈 = パース • 読み込む対象はソースコード(テキスト)でないこともある • Java: ソースコード(テキストファイル / HelloWorld.java) → バイトコード(バイナリ / HelloWorld.class) → java HelloWorld • 読み込むものがテキストでもバイナリでも 実行 = 命令を読み込んでその通りに動作する ということ 11
  6. 長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ PHPでの実行 php test.php • PHPが $a=100; という文字列を見たら…

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

    何から何に変換してもコンパイル • Cコンパイラ C言語のソースコード(テキストファイル)をアセンブリ(テキストファイル)に変換する • Javaコンパイラ Javaのソースコード(テキストファイル)をバイトコード(バイナリファイル)に変換 する • インタープリタとコンパイラは言語の特徴として対比して使われる • が昨今はインタープリタ言語でも内部でコンパイルしてたりして単純ではない 13
  8. 長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ VM: Virtual Machine • バイナリのプログラムを実行するプログラムをVMと言ったりする • PC全体を仮想化したVMとは違う文脈

    • PHP関連で言うと… • PHP VM, HHVM • PHP Opcodeを実行するプログラム • PHP Opcode = PHPのソースコードからコンパイルされたバイナリ • Java VM • Javaバイトコードを実行するプログラム • Javaバイトコード = Javaのソースコードからコンパイルされたバイナリ 14
  9. 長谷川智希 @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
  10. 長谷川智希 @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 機械語
  11. 長谷川智希 @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コマンド 機械語
  12. 長谷川智希 @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 機械語 機械語
  13. 長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ hello-world.php PHPプログラム phpコマンド macOS Apple Silicon CPU

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

    Mac Book Pro ここは何でもいい 下のレイヤが好きにやる Javaで書かれ、 Javaバイトコードに 変換されたプログラム (HelloWorld.class) javaコマンドより下は 気にしていない なら好きにしてみよう 27
  15. 長谷川智希 @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
  16. 長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ みんな大好き ファミコン • カートリッジ左半分には6502*1用の マシン語プログラムが格納されている • カートリッジ右半分には

    キャラクタのパターンが格納されている • このマシン語プログラムも当然 下のレイヤは気にしない 30 CPU 6502 PPU RAM V-RAM APU カートリッジ マシン語 プログラム キャラクタ パターン *1 実際は6502ベースのカスタムIC RP2A03 ここは何でもいい 話は聞かせて もらった!! 好きにさせて もらおう!
  17. 長谷川智希 @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
  18. 長谷川智希 @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のコードではありません めちゃ単純
  19. 長谷川智希 @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); }
  20. 長谷川智希 @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
  21. 長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ まとめ • そもそもプログラムを実行するとは何か • CPUでの実行 (機械語 /

    電気回路による実行) or プログラムでの実行 • PHP, Java, Go の実行モデルの違い • 近代のコンピュータはレイヤー化されている • 下のレイヤのことは気にしなくて良くなっている • 仮想サーバとかDockerとかエミュレータはその特性を活かしている • エミュレータは難しくない • 低レイヤを楽しもう 40 長谷川 智希 @tomzoh 𝕏
  22. 長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ 電気回路として実装されたCPU 42 加算機 演算データの セレクタ 1byteの上位4bit =

    機械語命令 何と何を 演算に使うかを決定 演算結果を どこに保存するかを 決定 PCレジスタ レジスタか I/Oのどれかと… Output 機械語の 引数を… 演算して どこかに保存する 1byteの下位4bit = 機械語の引数 汎用(演算用) レジスタ 2 × 4bit Input
  23. 長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ JavaScript インタープリタ ソースコードを nodeコマンドで実行 複数のエンジン実装 • Node.js

    • ブラウザ • Bun 複数の実装があるのは珍しくはない • RubyとかPythonも実装が複数ある • PHPもHackがあったり 46 console.log("Hello, world."); hello-world.js
  24. 長谷川智希 @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 |.....<init>...()| 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
  25. 長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ C コンパイラ ソースコード(テキストファイル) Cコンパイラ → ニーモニック(テキストファイル) アセンブラ

    → マシン語(バイナリ) CPUで実行 $ ./hello-world 48 #include <stdio.h> 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
  26. 長谷川智希 @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 $ݴޠͱಉ͡ʹݟ͑Δ͚Ͳɺ&-'ϔομ͸ όΠτͱ͔͋ΔΒ͍͠ͷͰʜ
  27. 長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ PHP インタープリタ ソースコードをphpコマンドで実行 $ php helllo-world.php 50

    <?php echo 'Hello, world.' hello-world.php $ php -dopcache.enable_cli=1 -dopcache.opt_debug_level=0x10000 hello-world.php $_main: ; (lines=2, args=0, vars=0, tmps=0) ; (before optimizer) ; /home/tom/tmp/php/hello-world.php:1-3 L0 (2): ECHO string("Hello, world.") L1 (3): RETURN int(1)
  28. 長谷川智希 @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
  29. 長谷川智希 @tomzoh PHPからはじめるコンピュータアーキテクチャ 余談: トランスパイル • トランスパイルという言葉もある • 何かから何かに変換するという意味ではコンパイルと同じ •

    プログラム言語からほかのプログラム言語への変換で使われる • TypeScript → JavaScript とか • JavaScriptのPoly fi llとか • 動作としてはコンパイルと同じ • C言語 → アセンブラの場合にはコンパイルと言う • 高水準言語 → 高水準言語の場合を明示的に示すために使われ始めた…? 53
  30. 長谷川智希 @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*