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
·
SiteGround - Reliable hosting with speed, security, and support you can count on.
→
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
51
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
機械学習を「社会実装」するということ 2026年夏版 / Social Implementation of Machine Learning June 2026 Version
moepy_stats
6
2.4k
現地で盛り上がった WWDC26 Keynote
zozotech
PRO
1
250
SONiCの統計情報を取得したい
sonic
0
180
スキルと MCP ツール、責務をどう分けるか? AI が迷わないインターフェース設計の戦略
cdataj
1
1.1k
ルールやカスタム機能、どう活かす?ハンズオンで体感するIBM Bobの出力コントロール
muehara
1
170
あなたの知らないPDFのアクセシビリティ
lycorptech_jp
PRO
0
200
いまさら聞けない「仕様駆動開発入門」 〜AI活用時代の開発プロセスを考える〜
findy_eventslides
2
140
2026 TECHFRESH 畢業分享會 - 開發日常大解密!從領域驅動到企業級上線
line_developers_tw
PRO
0
1.1k
LayerXにおけるセキュリティ管理の現在地と次の一手
tosho
0
220
エンジニアリング戦略の作り方 / Crafting Engineering Strategy
iwashi86
21
7k
連合学習と機密コンピューティング
lycorptech_jp
PRO
0
120
ACE-Step-1.5で見る 音楽生成AIのしくみと“破綻だけ直す”Retake機能の開発【zennfes spring 2026 登壇資料】
personabb
1
500
Featured
See All Featured
Neural Spatial Audio Processing for Sound Field Analysis and Control
skoyamalab
0
330
Dominate Local Search Results - an insider guide to GBP, reviews, and Local SEO
greggifford
PRO
0
190
The Art of Delivering Value - GDevCon NA Keynote
reverentgeek
16
2k
Templates, Plugins, & Blocks: Oh My! Creating the theme that thinks of everything
marktimemedia
31
2.8k
Connecting the Dots Between Site Speed, User Experience & Your Business [WebExpo 2025]
tammyeverts
11
940
Save Time (by Creating Custom Rails Generators)
garrettdimon
PRO
32
3.4k
End of SEO as We Know It (SMX Advanced Version)
ipullrank
3
4.2k
Evolution of real-time – Irina Nazarova, EuRuKo, 2024
irinanazarova
9
1.4k
Building Applications with DynamoDB
mza
96
7.1k
<Decoding/> the Language of Devs - We Love SEO 2024
nikkihalliwell
1
240
Tell your own story through comics
letsgokoyo
1
950
Reflections from 52 weeks, 52 projects
jeffersonlam
356
21k
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͕ͬͱΓ্͕͍ͬͯ ͖͍ͨʂ
͓͠·͍