Upgrade to Pro
— share decks privately, control downloads, hide ads and more …
Speaker Deck
Features
Speaker Deck
PRO
Sign in
Sign up for free
Search
Search
PHPとRustを組み合わせて音声ファイルをエンコードする話
Search
Sponsored
·
Your Podcast. Everywhere. Effortlessly.
Share. Educate. Inspire. Entertain. You do you. We'll handle the rest.
→
Shumpei Urabe
February 11, 2020
Technology
3.9k
0
Share
Embed
Copy iframe code
Copy JS code
Copy link
Start on current slide
PHPとRustを組み合わせて音声ファイルをエンコードする話
PHPerKaigi 2020
https://phperkaigi.jp/2020/
Shumpei Urabe
February 11, 2020
More Decks by Shumpei Urabe
See All by Shumpei Urabe
放送局の“SIer依存”は終わるのか? 〜Claude Code・Devin時代のAIネイティブ放送DX〜
yaminoma
0
52
AIを活用した放送技術スタートアップの制作現場改革について
yaminoma
0
16k
動画配信技術について
yaminoma
2
1.8k
Multimedia Security Topics
yaminoma
0
2.6k
コロナ禍でのライブストリーミングの変化とテレビ放送の規格について
yaminoma
0
1.2k
今日から分かる Android Audioの全て
yaminoma
0
500
Apple Low-Latency HLSを使った 超低遅延配信について
yaminoma
4
5.3k
今日から分かる AVAudioEngineの全て
yaminoma
4
6.4k
Cross Platform Video Player
yaminoma
0
570
Other Decks in Technology
See All in Technology
LayerXにおけるセキュリティ管理の現在地と次の一手
tosho
0
240
[チョークトーク資料]AWS DevOps Agent を使いこなす / AWS Dev Ops Agent Chalk Talk AWS Summit Japan 2026
kinunori
3
570
IaC コードを資産へ:AWS CDK 社内ライブラリと横断展開 / aws-summit-japan-2026
gotok365
5
1.1k
2026年6月23日 Syncable Tech + Start Python Club にて
hamukazu
0
140
サイバーエージェントにおけるAI推進戦略と変革への取り組み
shotatsuge
0
130
不要なレビューをAIにまかせて AIコーディングの環境改善を加速した
shoota
1
230
iAEONの段階的リアーキテクト戦略 / iAEON's_Gradual_Re-architecture_Strategy
aeonpeople
0
230
AWS Security Hub CSPMの成功・失敗体験
cmusudakeisuke
0
260
Claude Codeをどのように キャッチアップしているか
oikon48
13
8.6k
SONiCの統計情報を取得したい
sonic
0
230
10年間のブログ発信を振り返って見えたWebアプリケーションエンジニアとしての軌跡
stefafafan
0
160
Kiro Ambassador を目指す話
k_adachi_01
0
110
Featured
See All Featured
HTML-Aware ERB: The Path to Reactive Rendering @ RubyCon 2026, Rimini, Italy
marcoroth
1
200
Making Projects Easy
brettharned
120
6.7k
WENDY [Excerpt]
tessaabrams
11
38k
[RailsConf 2023] Rails as a piece of cake
palkan
59
6.7k
Fireside Chat
paigeccino
42
4k
AI: The stuff that nobody shows you
jnunemaker
PRO
8
720
Large-scale JavaScript Application Architecture
addyosmani
515
110k
Have SEOs Ruined the Internet? - User Awareness of SEO in 2025
akashhashmi
0
370
Discover your Explorer Soul
emna__ayadi
2
1.1k
Practical Tips for Bootstrapping Information Extraction Pipelines
honnibal
25
2k
"I'm Feeling Lucky" - Building Great Search Experiences for Today's Users (#IAC19)
danielanewman
230
23k
VelocityConf: Rendering Performance Case Studies
addyosmani
333
25k
Transcript
PHPͱRustΛΈ߹ΘͤͯԻ ϑΝΠϧΛΤϯίʔυ͢Δ PHPerKaigi 2020 LT @yaminoma_tw
Shumpei Urabe (@yaminoma_tw) ͖ͳϑϨʔϜϫʔΫ BEAR.Sunday PHPerݟश͍
PHP FFIͷ͓ͳ͠
FFIͱʁ • PHPҎ֎ͷݴޠͰ࣮͞ΕͨωΠςΟϒίʔυΛݺͼग़͢͜ͱ ͕Ͱ͖ΔΈ • FFIωΠςΟϒ࣮ߦͳͷͰɺύϑΥʔϚϯεͷ্͕ظͰ͖ Δ
PHP FFIͷྺ࢙ • Ҏલ͔ΒPECLͰFFIͷΤΫεςϯγϣϯ͋ͬͨ • ͔͠͠ɺ10Ҏ্ߋ৽͞Ε͍ͯͳ͍ঢ়گͰ์ஔ͞Ε͍ͯͨ
PHP FFIͷྺ࢙ • ͦ͜ͰZendࣾͷDmitry StogovࢯɺlibffiΛ֦ͬͨுΛ։ൃͨ͠ • 20191݄ʹࢍ24ɺର15ͱ͍͏ൺֱతᷮࠩͰPHP 7.4ͷ࠾༻ ͕ܾ·Γɺ͜ͷ֦ு͕ແࣄPHP 7.4ʹऔΓࠐ·Εͨ
• ͦͷ݁ՌɺPECLͷFFIΤΫεςϯγϣϯʹൺΔͱɺΠϯλʔϑΣ ΠεͷΈهड़͢Δ͚ͩͰग़དྷΔΑ͏ʹͳͬͨ
ຊ
PHPͰԻϑΝΠϧΛΤϯίʔυͯ͠ΈΔ • ԻϑΝΠϧͷΤϯίʔυͱ͍͑FFmpeg͕ఆ൪Ͱ͋Δ • FFmpegͷίϚϯυΛexecͰୟ͘ͷ؆୯Ͱ͋Δ͕ɺexecͰ ࣮ߦͨ͘͠ͳ͍ͱ͖͋Δ • PHP୯ମͰԻϑΝΠϧΛΤϯίʔυͰ͖Δͱศརʂ
ͱ͍͏͜ͱͰ࡞Γ·ͨ͠
php-mp3-encoder • https://github.com/yaminoma/php-mp3-encoder • RustͰMP3ͷΤϯίʔμʔͰ͋Δlibmp3lameΛͬͯɺWAVϑΝ Πϧ͔ΒMP3ʹΤϯίʔυ͢Δαϯϓϧ • ͜ͷlibmp3lamebindgenΛͬͯRust͔ΒFFIͰݺͼग़͍ͯ͠Δ
͘͠Έ
wavͷதΛ uint8_t* Ͱ͢ Τϯίʔυͨ͠ϑΝΠϧ໊Λ char* Ͱ͢
PHPͰFFIΛ༗ޮʹͯ͠ΈΔ git clone https://github.com/php/php-src.git cd php-src git checkout PHP-7.4 ./configure
--with-ffi make
PHPͰFFIΛ༗ޮʹͯ͠ΈΔ • σϑΥϧτ ffi.enable=preload ʹͳ͓ͬͯΓɺϓϦϩʔυΛ ߦͬͨؔ·ͨCLIΠϯλʔϑΣΠεͰͷΈFFIػೳΛ༻͢ Δ͜ͱ͕Մೳ • ffi.enable=trueʹ͢ΔͱɺશͯͷFFIͷػೳΛ͏͜ͱ͕ग़དྷΔ
RustͰΤϯίʔυ͢Δ෦Λ࣮͢Δ #[no_mangle] pub extern "C" fn encode(data: *mut u8, length:
u32) -> *const c_char { unsafe { let buf: &mut [u8] = core::slice::from_raw_parts_mut(data, length as usize); let mut ptr = io::Cursor::new(buf); let mut wav_reader = hound::WavReader::new(&mut ptr).unwrap(); let mut data = Vec::new(); for sample in wav_reader.samples() { match sample { Ok(smpl) => data.push(smpl), _ => panic!("failed"), } } let timestamp = Rational64::from_integer(0); let raw_frame = RawFrame { timestamp, data }; let mut encoder = Encoder::new(); let output_file_name = encoder.encode(raw_frame); return CString::new(output_file_name) .expect("failed") .into_raw(); } }
RustͰdylibΛϏϧυͯ͠PHPͰαʔϒ͢Δ $ cargo build --release $ php -d ffi.enable=1 -S
localhost:8000 -t src
μΠφϛοΫϥΠϒϥϦΛಡΈࠐΉ $signature = "const char* encode(uint8_t *data, uint32_t length);"; return
\FFI::cdef( $signature, __DIR__ . "/../target/ release/libmp3encoder.dylib");
freadͰϑΝΠϧΛಡΈऔΔ $file = $_FILES['audio_file']['tmp_name']; $handle = fopen($file, "rb"); $fsize =
filesize($file); $value = fread($handle, $fsize);
FFIͷݺͼग़͠ͱ࣮ߦ $ffi = loadLibrary(); $length = \strlen($value); $buffer = FFI::new(FFI::arrayType($ffi->type('uint8_t'),
[$length])); FFI::memcpy($buffer, $value, $length); $bufferPtr = FFI::cast($ffi->type('uint8_t*'), $buffer); $outputFileName = $ffi->encode($bufferPtr, $length);
݁Ռ • 5ͷۂ͕େମ8ඵ͘Β͍Ͱม͕ऴΘͬͨ • PHPͷFFIͷυΩϡϝϯτ͕ใྔগͳͯ݁͘ߏۤ͠Μͩ • FFI͕༗ޮʹͳ͍ͬͯΕͲͷαʔόͰMP3͕ΤϯίʔυͰ͖ Δʂ
FFIͷকདྷ • ܭࢉྔͷଟ͍ϓϩάϥϜΛFFIͰݺͼग़͢ͱྑ͍͔͠Εͳ͍ • ·ͩ·࣮ͩݧతͳཁૉ͕ଟ͍ͷͰɺϓϩμΫγϣϯϨσΟʔͰ ͳ͍ • ͜Ε͔ΒଞͷݴޠͱΈ߹ΘͤͯPHP͕ͬͱΓ্͕͍ͬͯ ͖͍ͨʂ
͓͠·͍