summaryrefslogtreecommitdiffstats
path: root/dom/media/test
diff options
context:
space:
mode:
Diffstat (limited to 'dom/media/test')
-rw-r--r--dom/media/test/16bit_wave_extrametadata.wavbin0 -> 97814 bytes
-rw-r--r--dom/media/test/16bit_wave_extrametadata.wav^headers^1
-rw-r--r--dom/media/test/320x240.ogvbin0 -> 28942 bytes
-rw-r--r--dom/media/test/320x240.ogv^headers^1
-rw-r--r--dom/media/test/448636.ogvbin0 -> 7799 bytes
-rw-r--r--dom/media/test/448636.ogv^headers^1
-rw-r--r--dom/media/test/A4.ogvbin0 -> 94372 bytes
-rw-r--r--dom/media/test/A4.ogv^headers^1
-rw-r--r--dom/media/test/TestPatternHDR.mp4bin0 -> 179294 bytes
-rw-r--r--dom/media/test/VID_0001.oggbin0 -> 633435 bytes
-rw-r--r--dom/media/test/VID_0001.ogg^headers^1
-rw-r--r--dom/media/test/adts.aacbin0 -> 8537 bytes
-rw-r--r--dom/media/test/adts.aac^headers^1
-rw-r--r--dom/media/test/allowed.sjs61
-rw-r--r--dom/media/test/ambisonics.mp4bin0 -> 1053904 bytes
-rw-r--r--dom/media/test/ambisonics.mp4^headers^1
-rw-r--r--dom/media/test/audio-gaps-short.oggbin0 -> 5233 bytes
-rw-r--r--dom/media/test/audio-gaps-short.ogg^headers^1
-rw-r--r--dom/media/test/audio-gaps.oggbin0 -> 12306 bytes
-rw-r--r--dom/media/test/audio-gaps.ogg^headers^1
-rw-r--r--dom/media/test/audio-overhang.oggbin0 -> 45463 bytes
-rw-r--r--dom/media/test/audio-overhang.ogg^headers^1
-rw-r--r--dom/media/test/audio.wavbin0 -> 1422 bytes
-rw-r--r--dom/media/test/audio.wav^headers^1
-rw-r--r--dom/media/test/av1.mp4bin0 -> 13089 bytes
-rw-r--r--dom/media/test/av1.mp4^headers^1
-rw-r--r--dom/media/test/background_video.js224
-rw-r--r--dom/media/test/badtags.oggbin0 -> 5033 bytes
-rw-r--r--dom/media/test/badtags.ogg^headers^1
-rw-r--r--dom/media/test/bear-640x360-a_frag-cenc-key_rotation.mp4bin0 -> 80372 bytes
-rw-r--r--dom/media/test/bear-640x360-v_frag-cenc-key_rotation.mp4bin0 -> 280361 bytes
-rw-r--r--dom/media/test/beta-phrasebook.oggbin0 -> 47411 bytes
-rw-r--r--dom/media/test/beta-phrasebook.ogg^headers^1
-rw-r--r--dom/media/test/big-buck-bunny-cenc-avc3-1.m4sbin0 -> 60041 bytes
-rw-r--r--dom/media/test/big-buck-bunny-cenc-avc3-1.m4s^headers^1
-rw-r--r--dom/media/test/big-buck-bunny-cenc-avc3-init.mp4bin0 -> 819 bytes
-rw-r--r--dom/media/test/big-buck-bunny-cenc-avc3-init.mp4^headers^1
-rw-r--r--dom/media/test/big-short.wavbin0 -> 12366 bytes
-rw-r--r--dom/media/test/big-short.wav^headers^1
-rw-r--r--dom/media/test/big.wavbin0 -> 102444 bytes
-rw-r--r--dom/media/test/big.wav^headers^1
-rw-r--r--dom/media/test/bipbop-cenc-audio-key1.xml28
-rw-r--r--dom/media/test/bipbop-cenc-audio-key2.xml28
-rw-r--r--dom/media/test/bipbop-cenc-audio1.m4sbin0 -> 921 bytes
-rw-r--r--dom/media/test/bipbop-cenc-audio1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop-cenc-audio2.m4sbin0 -> 565 bytes
-rw-r--r--dom/media/test/bipbop-cenc-audio2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop-cenc-audio3.m4sbin0 -> 977 bytes
-rw-r--r--dom/media/test/bipbop-cenc-audio3.m4s^headers^1
-rw-r--r--dom/media/test/bipbop-cenc-audioinit.mp4bin0 -> 1000 bytes
-rw-r--r--dom/media/test/bipbop-cenc-audioinit.mp4^headers^1
-rw-r--r--dom/media/test/bipbop-cenc-video-10s.mp4bin0 -> 299914 bytes
-rw-r--r--dom/media/test/bipbop-cenc-video-10s.mp4^headers^1
-rw-r--r--dom/media/test/bipbop-cenc-video-key1.xml28
-rw-r--r--dom/media/test/bipbop-cenc-video-key2.xml28
-rw-r--r--dom/media/test/bipbop-cenc-video1.m4sbin0 -> 25211 bytes
-rw-r--r--dom/media/test/bipbop-cenc-video1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop-cenc-video2.m4sbin0 -> 22934 bytes
-rw-r--r--dom/media/test/bipbop-cenc-video2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop-cenc-videoinit.mp4bin0 -> 1058 bytes
-rw-r--r--dom/media/test/bipbop-cenc-videoinit.mp4^headers^1
-rw-r--r--dom/media/test/bipbop-cenc.sh29
-rw-r--r--dom/media/test/bipbop-clearkey-keyrotation-clear-lead-audio.mp4bin0 -> 8675 bytes
-rw-r--r--dom/media/test/bipbop-clearkey-keyrotation-clear-lead-audio.mp4^headers^1
-rw-r--r--dom/media/test/bipbop-clearkey-keyrotation-clear-lead-video.mp4bin0 -> 278040 bytes
-rw-r--r--dom/media/test/bipbop-clearkey-keyrotation-clear-lead-video.mp4^headers^1
-rw-r--r--dom/media/test/bipbop-frag-cenc.xml57
-rw-r--r--dom/media/test/bipbop-lateaudio.mp4bin0 -> 70404 bytes
-rw-r--r--dom/media/test/bipbop-lateaudio.mp4^headers^1
-rw-r--r--dom/media/test/bipbop-no-edts.mp4bin0 -> 285681 bytes
-rw-r--r--dom/media/test/bipbop.mp4bin0 -> 285765 bytes
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-1.m4sbin0 -> 921 bytes
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-2.m4sbin0 -> 565 bytes
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-3.m4sbin0 -> 977 bytes
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-3.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-4.m4sbin0 -> 389 bytes
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-4.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-init.mp4bin0 -> 1020 bytes
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-1.m4sbin0 -> 921 bytes
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-2.m4sbin0 -> 565 bytes
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-3.m4sbin0 -> 977 bytes
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-3.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-4.m4sbin0 -> 389 bytes
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-4.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-init.mp4bin0 -> 1020 bytes
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-video-key1-1.m4sbin0 -> 37646 bytes
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-video-key1-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-video-key1-init.mp4bin0 -> 1086 bytes
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-video-key1-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-video-key2-1.m4sbin0 -> 37646 bytes
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-video-key2-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-video-key2-init.mp4bin0 -> 1086 bytes
-rw-r--r--dom/media/test/bipbop_225w_175kbps-cenc-video-key2-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_225w_175kbps.mp4bin0 -> 38713 bytes
-rw-r--r--dom/media/test/bipbop_225w_175kbps.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-audio-key1-1.m4sbin0 -> 921 bytes
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-audio-key1-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-audio-key1-2.m4sbin0 -> 565 bytes
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-audio-key1-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-audio-key1-3.m4sbin0 -> 977 bytes
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-audio-key1-3.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-audio-key1-4.m4sbin0 -> 389 bytes
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-audio-key1-4.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-audio-key1-init.mp4bin0 -> 874 bytes
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-audio-key1-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-audio-key2-1.m4sbin0 -> 921 bytes
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-audio-key2-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-audio-key2-2.m4sbin0 -> 565 bytes
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-audio-key2-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-audio-key2-3.m4sbin0 -> 977 bytes
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-audio-key2-3.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-audio-key2-4.m4sbin0 -> 389 bytes
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-audio-key2-4.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-audio-key2-init.mp4bin0 -> 874 bytes
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-audio-key2-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-video-key1-1.m4sbin0 -> 25211 bytes
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-video-key1-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-video-key1-2.m4sbin0 -> 22938 bytes
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-video-key1-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-video-key1-init.mp4bin0 -> 932 bytes
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-video-key1-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-video-key2-1.m4sbin0 -> 25211 bytes
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-video-key2-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-video-key2-2.m4sbin0 -> 22938 bytes
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-video-key2-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-video-key2-init.mp4bin0 -> 932 bytes
-rw-r--r--dom/media/test/bipbop_300_215kbps-cenc-video-key2-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_300_215kbps.mp4bin0 -> 48393 bytes
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-1.m4sbin0 -> 921 bytes
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-2.m4sbin0 -> 565 bytes
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-3.m4sbin0 -> 977 bytes
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-3.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-4.m4sbin0 -> 389 bytes
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-4.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-init.mp4bin0 -> 1020 bytes
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-1.m4sbin0 -> 921 bytes
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-2.m4sbin0 -> 565 bytes
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-3.m4sbin0 -> 977 bytes
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-3.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-4.m4sbin0 -> 389 bytes
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-4.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-init.mp4bin0 -> 1020 bytes
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-1.m4sbin0 -> 25211 bytes
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-2.m4sbin0 -> 22938 bytes
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-init.mp4bin0 -> 1094 bytes
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-1.m4sbin0 -> 25211 bytes
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-2.m4sbin0 -> 22938 bytes
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-init.mp4bin0 -> 1094 bytes
-rw-r--r--dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_300wp_227kbps.mp4bin0 -> 48355 bytes
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-1.m4sbin0 -> 921 bytes
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-2.m4sbin0 -> 565 bytes
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-3.m4sbin0 -> 977 bytes
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-3.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-4.m4sbin0 -> 389 bytes
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-4.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-init.mp4bin0 -> 1020 bytes
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-1.m4sbin0 -> 921 bytes
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-2.m4sbin0 -> 565 bytes
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-3.m4sbin0 -> 977 bytes
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-3.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-4.m4sbin0 -> 389 bytes
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-4.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-init.mp4bin0 -> 1020 bytes
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-video-key1-1.m4sbin0 -> 53149 bytes
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-video-key1-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-video-key1-init.mp4bin0 -> 1088 bytes
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-video-key1-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-video-key2-1.m4sbin0 -> 53149 bytes
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-video-key2-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-video-key2-init.mp4bin0 -> 1088 bytes
-rw-r--r--dom/media/test/bipbop_360w_253kbps-cenc-video-key2-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_360w_253kbps-clearkey-audio.webmbin0 -> 7553 bytes
-rw-r--r--dom/media/test/bipbop_360w_253kbps-clearkey-audio.webm^headers^1
-rw-r--r--dom/media/test/bipbop_360w_253kbps-clearkey-video-vp8.webmbin0 -> 44671 bytes
-rw-r--r--dom/media/test/bipbop_360w_253kbps-clearkey-video-vp8.webm^headers^1
-rw-r--r--dom/media/test/bipbop_360w_253kbps-clearkey-video-vp9.webmbin0 -> 46030 bytes
-rw-r--r--dom/media/test/bipbop_360w_253kbps-clearkey-video-vp9.webm^headers^1
-rw-r--r--dom/media/test/bipbop_360w_253kbps.mp4bin0 -> 54218 bytes
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-audio-key1-1.m4sbin0 -> 921 bytes
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-audio-key1-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-audio-key1-2.m4sbin0 -> 565 bytes
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-audio-key1-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-audio-key1-3.m4sbin0 -> 977 bytes
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-audio-key1-3.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-audio-key1-4.m4sbin0 -> 389 bytes
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-audio-key1-4.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-audio-key1-init.mp4bin0 -> 874 bytes
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-audio-key1-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-audio-key2-1.m4sbin0 -> 921 bytes
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-audio-key2-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-audio-key2-2.m4sbin0 -> 565 bytes
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-audio-key2-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-audio-key2-3.m4sbin0 -> 977 bytes
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-audio-key2-3.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-audio-key2-4.m4sbin0 -> 389 bytes
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-audio-key2-4.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-audio-key2-init.mp4bin0 -> 874 bytes
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-audio-key2-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-video-key1-1.m4sbin0 -> 68025 bytes
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-video-key1-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-video-key1-2.m4sbin0 -> 66457 bytes
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-video-key1-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-video-key1-init.mp4bin0 -> 932 bytes
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-video-key1-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-video-key2-1.m4sbin0 -> 68025 bytes
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-video-key2-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-video-key2-2.m4sbin0 -> 66457 bytes
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-video-key2-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-video-key2-init.mp4bin0 -> 932 bytes
-rw-r--r--dom/media/test/bipbop_480_624kbps-cenc-video-key2-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_480_624kbps.mp4bin0 -> 133264 bytes
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-audio-key1-1.m4sbin0 -> 921 bytes
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-audio-key1-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-audio-key1-2.m4sbin0 -> 565 bytes
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-audio-key1-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-audio-key1-3.m4sbin0 -> 977 bytes
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-audio-key1-3.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-audio-key1-4.m4sbin0 -> 389 bytes
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-audio-key1-4.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-audio-key1-init.mp4bin0 -> 874 bytes
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-audio-key1-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-audio-key2-1.m4sbin0 -> 921 bytes
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-audio-key2-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-audio-key2-2.m4sbin0 -> 565 bytes
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-audio-key2-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-audio-key2-3.m4sbin0 -> 977 bytes
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-audio-key2-3.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-audio-key2-4.m4sbin0 -> 389 bytes
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-audio-key2-4.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-audio-key2-init.mp4bin0 -> 874 bytes
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-audio-key2-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-video-key1-1.m4sbin0 -> 101203 bytes
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-video-key1-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-video-key1-2.m4sbin0 -> 99366 bytes
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-video-key1-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-video-key1-init.mp4bin0 -> 932 bytes
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-video-key1-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-video-key2-1.m4sbin0 -> 101203 bytes
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-video-key2-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-video-key2-2.m4sbin0 -> 99366 bytes
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-video-key2-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-video-key2-init.mp4bin0 -> 932 bytes
-rw-r--r--dom/media/test/bipbop_480_959kbps-cenc-video-key2-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_480_959kbps.mp4bin0 -> 199351 bytes
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-1.m4sbin0 -> 921 bytes
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-2.m4sbin0 -> 565 bytes
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-3.m4sbin0 -> 977 bytes
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-4.m4sbin0 -> 389 bytes
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4bin0 -> 1020 bytes
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-1.m4sbin0 -> 921 bytes
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-2.m4sbin0 -> 565 bytes
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-3.m4sbin0 -> 977 bytes
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-4.m4sbin0 -> 389 bytes
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4bin0 -> 1020 bytes
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-1.m4sbin0 -> 101203 bytes
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-2.m4sbin0 -> 99366 bytes
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-init.mp4bin0 -> 1094 bytes
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-1.m4sbin0 -> 101203 bytes
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-2.m4sbin0 -> 99366 bytes
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-init.mp4bin0 -> 1094 bytes
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_1001kbps.mp4bin0 -> 199911 bytes
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-1.m4sbin0 -> 921 bytes
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-2.m4sbin0 -> 565 bytes
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-3.m4sbin0 -> 977 bytes
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-3.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-4.m4sbin0 -> 389 bytes
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-4.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-init.mp4bin0 -> 1020 bytes
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-1.m4sbin0 -> 921 bytes
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-2.m4sbin0 -> 565 bytes
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-3.m4sbin0 -> 977 bytes
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-3.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-4.m4sbin0 -> 389 bytes
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-4.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-init.mp4bin0 -> 1020 bytes
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-1.m4sbin0 -> 68025 bytes
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-2.m4sbin0 -> 66457 bytes
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-init.mp4bin0 -> 1094 bytes
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-1.m4sbin0 -> 68025 bytes
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-2.m4sbin0 -> 66457 bytes
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-2.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-init.mp4bin0 -> 1094 bytes
-rw-r--r--dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_480wp_663kbps.mp4bin0 -> 133824 bytes
-rw-r--r--dom/media/test/bipbop_audio_aac_22.05k.mp4bin0 -> 2424 bytes
-rw-r--r--dom/media/test/bipbop_audio_aac_22.05k.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_audio_aac_44.1k.mp4bin0 -> 3239 bytes
-rw-r--r--dom/media/test/bipbop_audio_aac_44.1k.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_audio_aac_48k.mp4bin0 -> 3286 bytes
-rw-r--r--dom/media/test/bipbop_audio_aac_48k.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_audio_aac_88.2k.mp4bin0 -> 3769 bytes
-rw-r--r--dom/media/test/bipbop_audio_aac_88.2k.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_audio_aac_8k.mp4bin0 -> 1707 bytes
-rw-r--r--dom/media/test/bipbop_audio_aac_8k.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_audio_aac_96k.mp4bin0 -> 4010 bytes
-rw-r--r--dom/media/test/bipbop_audio_aac_96k.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_cbcs_10_0_audio_1.m4sbin0 -> 1364 bytes
-rw-r--r--dom/media/test/bipbop_cbcs_10_0_audio_1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_cbcs_10_0_audio_init.mp4bin0 -> 936 bytes
-rw-r--r--dom/media/test/bipbop_cbcs_10_0_audio_init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_cbcs_10_0_video_1.m4sbin0 -> 57044 bytes
-rw-r--r--dom/media/test/bipbop_cbcs_10_0_video_1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_cbcs_10_0_video_init.mp4bin0 -> 972 bytes
-rw-r--r--dom/media/test/bipbop_cbcs_10_0_video_init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_cbcs_1_9_audio_1.m4sbin0 -> 1364 bytes
-rw-r--r--dom/media/test/bipbop_cbcs_1_9_audio_1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_cbcs_1_9_audio_init.mp4bin0 -> 936 bytes
-rw-r--r--dom/media/test/bipbop_cbcs_1_9_audio_init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_cbcs_1_9_video_1.m4sbin0 -> 57044 bytes
-rw-r--r--dom/media/test/bipbop_cbcs_1_9_video_1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_cbcs_1_9_video_init.mp4bin0 -> 972 bytes
-rw-r--r--dom/media/test/bipbop_cbcs_1_9_video_init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_cbcs_5_5_audio_1.m4sbin0 -> 1364 bytes
-rw-r--r--dom/media/test/bipbop_cbcs_5_5_audio_1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_cbcs_5_5_audio_init.mp4bin0 -> 936 bytes
-rw-r--r--dom/media/test/bipbop_cbcs_5_5_audio_init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_cbcs_5_5_video_1.m4sbin0 -> 57044 bytes
-rw-r--r--dom/media/test/bipbop_cbcs_5_5_video_1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_cbcs_5_5_video_init.mp4bin0 -> 972 bytes
-rw-r--r--dom/media/test/bipbop_cbcs_5_5_video_init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_cbcs_7_7_audio_1.m4sbin0 -> 1364 bytes
-rw-r--r--dom/media/test/bipbop_cbcs_7_7_audio_1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_cbcs_7_7_audio_init.mp4bin0 -> 936 bytes
-rw-r--r--dom/media/test/bipbop_cbcs_7_7_audio_init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_cbcs_7_7_video_1.m4sbin0 -> 57044 bytes
-rw-r--r--dom/media/test/bipbop_cbcs_7_7_video_1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_cbcs_7_7_video_init.mp4bin0 -> 972 bytes
-rw-r--r--dom/media/test/bipbop_cbcs_7_7_video_init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_cbcs_9_8_audio_1.m4sbin0 -> 1364 bytes
-rw-r--r--dom/media/test/bipbop_cbcs_9_8_audio_1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_cbcs_9_8_audio_init.mp4bin0 -> 936 bytes
-rw-r--r--dom/media/test/bipbop_cbcs_9_8_audio_init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_cbcs_9_8_video_1.m4sbin0 -> 57044 bytes
-rw-r--r--dom/media/test/bipbop_cbcs_9_8_video_1.m4s^headers^1
-rw-r--r--dom/media/test/bipbop_cbcs_9_8_video_init.mp4bin0 -> 972 bytes
-rw-r--r--dom/media/test/bipbop_cbcs_9_8_video_init.mp4^headers^1
-rw-r--r--dom/media/test/bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webmbin0 -> 48942 bytes
-rw-r--r--dom/media/test/bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm^headers^1
-rw-r--r--dom/media/test/bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webmbin0 -> 48942 bytes
-rw-r--r--dom/media/test/bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm^headers^1
-rw-r--r--dom/media/test/bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webmbin0 -> 48942 bytes
-rw-r--r--dom/media/test/bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm^headers^1
-rw-r--r--dom/media/test/bipbop_short_vp8.webmbin0 -> 48942 bytes
-rw-r--r--dom/media/test/bipbop_short_vp8.webm^headers^1
-rw-r--r--dom/media/test/black100x100-aspect3to2.ogvbin0 -> 3428 bytes
-rw-r--r--dom/media/test/black100x100-aspect3to2.ogv^headers^1
-rw-r--r--dom/media/test/bogus.duh45
-rw-r--r--dom/media/test/bogus.ogv45
-rw-r--r--dom/media/test/bogus.ogv^headers^1
-rw-r--r--dom/media/test/bogus.wav45
-rw-r--r--dom/media/test/bogus.wav^headers^1
-rw-r--r--dom/media/test/browser/browser.toml26
-rw-r--r--dom/media/test/browser/browser_encrypted_play_time_telemetry.js269
-rw-r--r--dom/media/test/browser/browser_partial.js56
-rw-r--r--dom/media/test/browser/browser_tab_visibility_and_play_time.js218
-rw-r--r--dom/media/test/browser/browser_telemetry_video_hardware_decoding_support.js106
-rw-r--r--dom/media/test/browser/file_empty_page.html8
-rw-r--r--dom/media/test/browser/file_media.html10
-rw-r--r--dom/media/test/browser/wmfme/browser.toml13
-rw-r--r--dom/media/test/browser/wmfme/browser_wmfme_crash.js52
-rw-r--r--dom/media/test/browser/wmfme/browser_wmfme_max_crashes.js69
-rw-r--r--dom/media/test/browser/wmfme/file_video.html9
-rw-r--r--dom/media/test/browser/wmfme/head.js201
-rw-r--r--dom/media/test/bug1066943.webmbin0 -> 18442 bytes
-rw-r--r--dom/media/test/bug1066943.webm^headers^1
-rw-r--r--dom/media/test/bug1301226-odd.wavbin0 -> 244 bytes
-rw-r--r--dom/media/test/bug1301226-odd.wav^headers^1
-rw-r--r--dom/media/test/bug1301226.wavbin0 -> 240 bytes
-rw-r--r--dom/media/test/bug1301226.wav^headers^1
-rw-r--r--dom/media/test/bug1377278.webmbin0 -> 215594 bytes
-rw-r--r--dom/media/test/bug1377278.webm^headers^1
-rw-r--r--dom/media/test/bug1535980.webmbin0 -> 81467 bytes
-rw-r--r--dom/media/test/bug1535980.webm^headers^1
-rw-r--r--dom/media/test/bug1799787.webmbin0 -> 1053 bytes
-rw-r--r--dom/media/test/bug1799787.webm^headers^1
-rw-r--r--dom/media/test/bug461281.oggbin0 -> 16521 bytes
-rw-r--r--dom/media/test/bug461281.ogg^headers^1
-rw-r--r--dom/media/test/bug482461-theora.ogvbin0 -> 280904 bytes
-rw-r--r--dom/media/test/bug482461-theora.ogv^headers^1
-rw-r--r--dom/media/test/bug482461.ogvbin0 -> 305785 bytes
-rw-r--r--dom/media/test/bug482461.ogv^headers^1
-rw-r--r--dom/media/test/bug495129.ogvbin0 -> 122207 bytes
-rw-r--r--dom/media/test/bug495129.ogv^headers^1
-rw-r--r--dom/media/test/bug495794.oggbin0 -> 4837 bytes
-rw-r--r--dom/media/test/bug495794.ogg^headers^1
-rw-r--r--dom/media/test/bug498380.ogvbin0 -> 65535 bytes
-rw-r--r--dom/media/test/bug498380.ogv^headers^1
-rw-r--r--dom/media/test/bug498855-1.ogvbin0 -> 20480 bytes
-rw-r--r--dom/media/test/bug498855-1.ogv^headers^1
-rw-r--r--dom/media/test/bug498855-2.ogvbin0 -> 20480 bytes
-rw-r--r--dom/media/test/bug498855-2.ogv^headers^1
-rw-r--r--dom/media/test/bug498855-3.ogvbin0 -> 20480 bytes
-rw-r--r--dom/media/test/bug498855-3.ogv^headers^1
-rw-r--r--dom/media/test/bug499519.ogvbin0 -> 20480 bytes
-rw-r--r--dom/media/test/bug499519.ogv^headers^1
-rw-r--r--dom/media/test/bug500311.ogvbin0 -> 55834 bytes
-rw-r--r--dom/media/test/bug500311.ogv^headers^1
-rw-r--r--dom/media/test/bug501279.oggbin0 -> 2361 bytes
-rw-r--r--dom/media/test/bug501279.ogg^headers^1
-rw-r--r--dom/media/test/bug504613.ogvbin0 -> 35000 bytes
-rw-r--r--dom/media/test/bug504613.ogv^headers^1
-rw-r--r--dom/media/test/bug504644.ogvbin0 -> 131114 bytes
-rw-r--r--dom/media/test/bug504644.ogv^headers^1
-rw-r--r--dom/media/test/bug504843.ogvbin0 -> 65536 bytes
-rw-r--r--dom/media/test/bug504843.ogv^headers^1
-rw-r--r--dom/media/test/bug506094.ogvbin0 -> 8195 bytes
-rw-r--r--dom/media/test/bug506094.ogv^headers^1
-rw-r--r--dom/media/test/bug516323.indexed.ogvbin0 -> 162193 bytes
-rw-r--r--dom/media/test/bug516323.indexed.ogv^headers^1
-rw-r--r--dom/media/test/bug516323.ogvbin0 -> 161789 bytes
-rw-r--r--dom/media/test/bug516323.ogv^headers^1
-rw-r--r--dom/media/test/bug520493.oggbin0 -> 3901 bytes
-rw-r--r--dom/media/test/bug520493.ogg^headers^1
-rw-r--r--dom/media/test/bug520500.oggbin0 -> 21978 bytes
-rw-r--r--dom/media/test/bug520500.ogg^headers^1
-rw-r--r--dom/media/test/bug520908.ogvbin0 -> 28942 bytes
-rw-r--r--dom/media/test/bug520908.ogv^headers^1
-rw-r--r--dom/media/test/bug523816.ogvbin0 -> 40585 bytes
-rw-r--r--dom/media/test/bug523816.ogv^headers^1
-rw-r--r--dom/media/test/bug533822.oggbin0 -> 35010 bytes
-rw-r--r--dom/media/test/bug533822.ogg^headers^1
-rw-r--r--dom/media/test/bug556821.ogvbin0 -> 196608 bytes
-rw-r--r--dom/media/test/bug556821.ogv^headers^1
-rw-r--r--dom/media/test/bug557094.ogvbin0 -> 76966 bytes
-rw-r--r--dom/media/test/bug557094.ogv^headers^1
-rw-r--r--dom/media/test/bug604067.webmbin0 -> 103227 bytes
-rw-r--r--dom/media/test/bug604067.webm^headers^1
-rw-r--r--dom/media/test/bunny.webmbin0 -> 195455 bytes
-rw-r--r--dom/media/test/bunny_hd_5s.mp4bin0 -> 845651 bytes
-rw-r--r--dom/media/test/can_play_type_dash.js27
-rw-r--r--dom/media/test/can_play_type_ogg.js72
-rw-r--r--dom/media/test/can_play_type_wave.js30
-rw-r--r--dom/media/test/can_play_type_webm.js39
-rw-r--r--dom/media/test/cancellable_request.sjs162
-rw-r--r--dom/media/test/chain.oggbin0 -> 63610 bytes
-rw-r--r--dom/media/test/chain.ogg^headers^1
-rw-r--r--dom/media/test/chain.ogvbin0 -> 45463 bytes
-rw-r--r--dom/media/test/chain.ogv^headers^1
-rw-r--r--dom/media/test/chain.opusbin0 -> 50101 bytes
-rw-r--r--dom/media/test/chain.opus^headers^1
-rw-r--r--dom/media/test/chained-audio-video.oggbin0 -> 92552 bytes
-rw-r--r--dom/media/test/chained-audio-video.ogg^headers^1
-rw-r--r--dom/media/test/chained-video.ogvbin0 -> 57906 bytes
-rw-r--r--dom/media/test/chained-video.ogv^headers^1
-rw-r--r--dom/media/test/chrome/chrome.toml12
-rw-r--r--dom/media/test/chrome/test_accumulated_play_time.html694
-rw-r--r--dom/media/test/chrome/test_telemetry_source_buffer_type.html105
-rw-r--r--dom/media/test/chromeHelper.js23
-rw-r--r--dom/media/test/cloneElementVisually_helpers.js232
-rw-r--r--dom/media/test/contentType.sjs77
-rw-r--r--dom/media/test/crashtests/0-timescale.html14
-rw-r--r--dom/media/test/crashtests/0-timescale.mp4bin0 -> 14718 bytes
-rw-r--r--dom/media/test/crashtests/1012609.html9
-rw-r--r--dom/media/test/crashtests/1015662.html4
-rw-r--r--dom/media/test/crashtests/1028458.html23
-rw-r--r--dom/media/test/crashtests/1041466.html21
-rw-r--r--dom/media/test/crashtests/1045650.html18
-rw-r--r--dom/media/test/crashtests/1080986.html3
-rw-r--r--dom/media/test/crashtests/1080986.wavbin0 -> 592 bytes
-rw-r--r--dom/media/test/crashtests/1122218.html24
-rw-r--r--dom/media/test/crashtests/1127188.html3
-rw-r--r--dom/media/test/crashtests/1157994.html21
-rw-r--r--dom/media/test/crashtests/1158427.html21
-rw-r--r--dom/media/test/crashtests/1180881.html8
-rw-r--r--dom/media/test/crashtests/1180881.webmbin0 -> 524 bytes
-rw-r--r--dom/media/test/crashtests/1185176.html24
-rw-r--r--dom/media/test/crashtests/1185191.html21
-rw-r--r--dom/media/test/crashtests/1185192.html18
-rw-r--r--dom/media/test/crashtests/1197935.html8
-rw-r--r--dom/media/test/crashtests/1197935.mp4bin0 -> 1806042 bytes
-rw-r--r--dom/media/test/crashtests/1223670.html23
-rw-r--r--dom/media/test/crashtests/1236639.html9
-rw-r--r--dom/media/test/crashtests/1236639.mp3bin0 -> 1080 bytes
-rw-r--r--dom/media/test/crashtests/1257700.html8
-rw-r--r--dom/media/test/crashtests/1257700.webmbin0 -> 59264 bytes
-rw-r--r--dom/media/test/crashtests/1267263.html19
-rw-r--r--dom/media/test/crashtests/1270303.html8
-rw-r--r--dom/media/test/crashtests/1270303.webmbin0 -> 5822 bytes
-rw-r--r--dom/media/test/crashtests/1291702.html72
-rw-r--r--dom/media/test/crashtests/1368490.html30
-rw-r--r--dom/media/test/crashtests/1378826.html46
-rw-r--r--dom/media/test/crashtests/1384248.html10
-rw-r--r--dom/media/test/crashtests/1388372.html13
-rw-r--r--dom/media/test/crashtests/1389304.html32
-rw-r--r--dom/media/test/crashtests/1389304.mp4bin0 -> 198320 bytes
-rw-r--r--dom/media/test/crashtests/1393272.webmbin0 -> 6781 bytes
-rw-r--r--dom/media/test/crashtests/1411322.html18
-rw-r--r--dom/media/test/crashtests/1414444.mp4bin0 -> 34009 bytes
-rw-r--r--dom/media/test/crashtests/1450845.html34
-rw-r--r--dom/media/test/crashtests/1489160.html10
-rw-r--r--dom/media/test/crashtests/1494073.html19
-rw-r--r--dom/media/test/crashtests/1517199.html17
-rw-r--r--dom/media/test/crashtests/1526044.html19
-rw-r--r--dom/media/test/crashtests/1530897.webmbin0 -> 509 bytes
-rw-r--r--dom/media/test/crashtests/1538727.html14
-rw-r--r--dom/media/test/crashtests/1545133.html34
-rw-r--r--dom/media/test/crashtests/1547784.html33
-rw-r--r--dom/media/test/crashtests/1547899.html20
-rw-r--r--dom/media/test/crashtests/1560215.html20
-rw-r--r--dom/media/test/crashtests/1569645.html23
-rw-r--r--dom/media/test/crashtests/1575271.html25
-rw-r--r--dom/media/test/crashtests/1577184.html15
-rw-r--r--dom/media/test/crashtests/1587248.html23
-rw-r--r--dom/media/test/crashtests/1594466.html22
-rw-r--r--dom/media/test/crashtests/1601385.html12
-rw-r--r--dom/media/test/crashtests/1601422.html20
-rw-r--r--dom/media/test/crashtests/1604941.html22
-rw-r--r--dom/media/test/crashtests/1608286.html50
-rw-r--r--dom/media/test/crashtests/1673525.html15
-rw-r--r--dom/media/test/crashtests/1673526-1.html20
-rw-r--r--dom/media/test/crashtests/1673526-2.html20
-rw-r--r--dom/media/test/crashtests/1693043.html21
-rw-r--r--dom/media/test/crashtests/1696511.html22
-rw-r--r--dom/media/test/crashtests/1697521.html19
-rw-r--r--dom/media/test/crashtests/1708790.html22
-rw-r--r--dom/media/test/crashtests/1709130.html19
-rw-r--r--dom/media/test/crashtests/1734008.html22
-rw-r--r--dom/media/test/crashtests/1734008.webmbin0 -> 8253 bytes
-rw-r--r--dom/media/test/crashtests/1741677.html15
-rw-r--r--dom/media/test/crashtests/1748272.html12
-rw-r--r--dom/media/test/crashtests/1752917.html18
-rw-r--r--dom/media/test/crashtests/1762620.html8
-rw-r--r--dom/media/test/crashtests/1765842.html8
-rw-r--r--dom/media/test/crashtests/1765842.webmbin0 -> 17210 bytes
-rw-r--r--dom/media/test/crashtests/1787281.html13
-rw-r--r--dom/media/test/crashtests/1787281.mp4bin0 -> 2736 bytes
-rw-r--r--dom/media/test/crashtests/1798778.html11
-rw-r--r--dom/media/test/crashtests/1830206.html12
-rw-r--r--dom/media/test/crashtests/1830206.mp4bin0 -> 41224 bytes
-rw-r--r--dom/media/test/crashtests/1833894.mp4bin0 -> 1000465 bytes
-rw-r--r--dom/media/test/crashtests/1833896.mp4bin0 -> 38215 bytes
-rw-r--r--dom/media/test/crashtests/1835118.adtsbin0 -> 132 bytes
-rw-r--r--dom/media/test/crashtests/1835164.html13
-rw-r--r--dom/media/test/crashtests/1835164.opusbin0 -> 2250 bytes
-rw-r--r--dom/media/test/crashtests/1839193.adtsbin0 -> 484 bytes
-rw-r--r--dom/media/test/crashtests/1839193.html9
-rw-r--r--dom/media/test/crashtests/1840002.webmbin0 -> 512 bytes
-rw-r--r--dom/media/test/crashtests/1845350.mp4bin0 -> 1045 bytes
-rw-r--r--dom/media/test/crashtests/1848660.html10
-rw-r--r--dom/media/test/crashtests/1848660.wav0
-rw-r--r--dom/media/test/crashtests/1850453.flacbin0 -> 104472 bytes
-rw-r--r--dom/media/test/crashtests/1850453.html12
-rw-r--r--dom/media/test/crashtests/1859384.mp4bin0 -> 4128 bytes
-rw-r--r--dom/media/test/crashtests/1859600.mp4bin0 -> 5172 bytes
-rw-r--r--dom/media/test/crashtests/1860840.mp4bin0 -> 5172 bytes
-rw-r--r--dom/media/test/crashtests/1864450.html12
-rw-r--r--dom/media/test/crashtests/1872787.html14
-rw-r--r--dom/media/test/crashtests/255ch.wavbin0 -> 68318 bytes
-rw-r--r--dom/media/test/crashtests/459439-1.html36
-rw-r--r--dom/media/test/crashtests/466607-1.html14
-rw-r--r--dom/media/test/crashtests/466945-1.html25
-rw-r--r--dom/media/test/crashtests/468763-1.html1
-rw-r--r--dom/media/test/crashtests/474744-1.html15
-rw-r--r--dom/media/test/crashtests/481136-1.html3
-rw-r--r--dom/media/test/crashtests/492286-1.xhtml1
-rw-r--r--dom/media/test/crashtests/493915-1.html18
-rw-r--r--dom/media/test/crashtests/495794-1.html8
-rw-r--r--dom/media/test/crashtests/495794-1.oggbin0 -> 4837 bytes
-rw-r--r--dom/media/test/crashtests/497734-1.xhtml21
-rw-r--r--dom/media/test/crashtests/497734-2.html17
-rw-r--r--dom/media/test/crashtests/576612-1.html15
-rw-r--r--dom/media/test/crashtests/691096-1.html31
-rw-r--r--dom/media/test/crashtests/752784-1.html15
-rw-r--r--dom/media/test/crashtests/789075-1.html20
-rw-r--r--dom/media/test/crashtests/789075.webmbin0 -> 12294 bytes
-rw-r--r--dom/media/test/crashtests/795892-1.html23
-rw-r--r--dom/media/test/crashtests/844563.html5
-rw-r--r--dom/media/test/crashtests/846612.html8
-rw-r--r--dom/media/test/crashtests/852838.html11
-rw-r--r--dom/media/test/crashtests/865004.html19
-rw-r--r--dom/media/test/crashtests/865537-1.html13
-rw-r--r--dom/media/test/crashtests/865550.html22
-rw-r--r--dom/media/test/crashtests/868504.html14
-rw-r--r--dom/media/test/crashtests/874869.html15
-rw-r--r--dom/media/test/crashtests/874915.html24
-rw-r--r--dom/media/test/crashtests/874934.html23
-rw-r--r--dom/media/test/crashtests/874952.html11
-rw-r--r--dom/media/test/crashtests/875144.html81
-rw-r--r--dom/media/test/crashtests/875596.html12
-rw-r--r--dom/media/test/crashtests/875911.html3
-rw-r--r--dom/media/test/crashtests/876024-1.html5
-rw-r--r--dom/media/test/crashtests/876024-2.html17
-rw-r--r--dom/media/test/crashtests/876118.html16
-rw-r--r--dom/media/test/crashtests/876207.html30
-rw-r--r--dom/media/test/crashtests/876215.html14
-rw-r--r--dom/media/test/crashtests/876249.html27
-rw-r--r--dom/media/test/crashtests/876252.html23
-rw-r--r--dom/media/test/crashtests/876834.html4
-rw-r--r--dom/media/test/crashtests/877527.html37
-rw-r--r--dom/media/test/crashtests/877820.html4
-rw-r--r--dom/media/test/crashtests/878014.html31
-rw-r--r--dom/media/test/crashtests/878328.html5
-rw-r--r--dom/media/test/crashtests/878407.html11
-rw-r--r--dom/media/test/crashtests/878478.html30
-rw-r--r--dom/media/test/crashtests/880129.html9
-rw-r--r--dom/media/test/crashtests/880202.html33
-rw-r--r--dom/media/test/crashtests/880342-1.html208
-rw-r--r--dom/media/test/crashtests/880342-2.html8
-rw-r--r--dom/media/test/crashtests/880384.html8
-rw-r--r--dom/media/test/crashtests/880404.html6
-rw-r--r--dom/media/test/crashtests/880724.html13
-rw-r--r--dom/media/test/crashtests/881775.html25
-rw-r--r--dom/media/test/crashtests/882956.html15
-rw-r--r--dom/media/test/crashtests/884459.html12
-rw-r--r--dom/media/test/crashtests/889042.html4
-rw-r--r--dom/media/test/crashtests/907986-1.html17
-rw-r--r--dom/media/test/crashtests/907986-2.html17
-rw-r--r--dom/media/test/crashtests/907986-3.html17
-rw-r--r--dom/media/test/crashtests/907986-4.html15
-rw-r--r--dom/media/test/crashtests/910171-1.html17
-rw-r--r--dom/media/test/crashtests/920987.html6
-rw-r--r--dom/media/test/crashtests/925619-1.html14
-rw-r--r--dom/media/test/crashtests/925619-2.html15
-rw-r--r--dom/media/test/crashtests/926619.html24
-rw-r--r--dom/media/test/crashtests/933151.html16
-rw-r--r--dom/media/test/crashtests/933156.html23
-rw-r--r--dom/media/test/crashtests/944851.html17
-rw-r--r--dom/media/test/crashtests/952756.html19
-rw-r--r--dom/media/test/crashtests/986901.html16
-rw-r--r--dom/media/test/crashtests/990794.html22
-rw-r--r--dom/media/test/crashtests/995289.html9
-rw-r--r--dom/media/test/crashtests/adts-truncated.aacbin0 -> 512 bytes
-rw-r--r--dom/media/test/crashtests/adts.aacbin0 -> 8537 bytes
-rw-r--r--dom/media/test/crashtests/analyser-channels-1.html16
-rw-r--r--dom/media/test/crashtests/audiocontext-after-unload-1.html27
-rw-r--r--dom/media/test/crashtests/audiocontext-after-xhr.html13
-rw-r--r--dom/media/test/crashtests/audiocontext-double-suspend.html5
-rw-r--r--dom/media/test/crashtests/audioworkletnode-after-unload-1.html27
-rw-r--r--dom/media/test/crashtests/buffer-source-duration-1.html14
-rw-r--r--dom/media/test/crashtests/buffer-source-ended-1.html16
-rw-r--r--dom/media/test/crashtests/buffer-source-resampling-start-1.html16
-rw-r--r--dom/media/test/crashtests/buffer-source-slow-resampling-1.html34
-rw-r--r--dom/media/test/crashtests/channel-count-in-metadata-different-than-in-content.mp4bin0 -> 13651 bytes
-rw-r--r--dom/media/test/crashtests/convolver-memory-report-1.html25
-rw-r--r--dom/media/test/crashtests/copyFromChannel-2.html16
-rw-r--r--dom/media/test/crashtests/cors.webmbin0 -> 215529 bytes
-rw-r--r--dom/media/test/crashtests/cors.webm^headers^1
-rw-r--r--dom/media/test/crashtests/crashtests.list182
-rw-r--r--dom/media/test/crashtests/disconnect-wrong-destination.html13
-rw-r--r--dom/media/test/crashtests/doppler-1.html23
-rw-r--r--dom/media/test/crashtests/empty-buffer-source.html14
-rw-r--r--dom/media/test/crashtests/empty-samples.webm0
-rw-r--r--dom/media/test/crashtests/encrypted-track-with-bad-sample-description-index.mp4bin0 -> 198320 bytes
-rw-r--r--dom/media/test/crashtests/encrypted-track-with-sample-missing-cenc-aux.mp4bin0 -> 152132 bytes
-rw-r--r--dom/media/test/crashtests/encrypted-track-without-tenc.mp4bin0 -> 152132 bytes
-rw-r--r--dom/media/test/crashtests/invalidfmt.html8
-rw-r--r--dom/media/test/crashtests/invalidfmt.wavbin0 -> 115 bytes
-rw-r--r--dom/media/test/crashtests/media-element-source-seek-1.html27
-rw-r--r--dom/media/test/crashtests/mp4_box_emptyrange.mp4bin0 -> 918 bytes
-rw-r--r--dom/media/test/crashtests/noextradata-8ch.wavbin0 -> 684 bytes
-rw-r--r--dom/media/test/crashtests/offline-buffer-source-ended-1.html15
-rw-r--r--dom/media/test/crashtests/oscillator-ended-1.html15
-rw-r--r--dom/media/test/crashtests/oscillator-ended-2.html15
-rw-r--r--dom/media/test/crashtests/small-timebase.html10
-rw-r--r--dom/media/test/crashtests/small-timebase.mp4bin0 -> 1355 bytes
-rw-r--r--dom/media/test/crashtests/sound.oggbin0 -> 2603 bytes
-rw-r--r--dom/media/test/crashtests/test.mp4bin0 -> 11817 bytes
-rw-r--r--dom/media/test/crashtests/track-with-zero-dimensions.mp4bin0 -> 11817 bytes
-rw-r--r--dom/media/test/crashtests/trimming_needed_and_last_sample_invalid_duration.oggbin0 -> 18307 bytes
-rw-r--r--dom/media/test/crashtests/video-crash.webmbin0 -> 58482 bytes
-rw-r--r--dom/media/test/crashtests/video-replay-after-audio-end.html43
-rw-r--r--dom/media/test/dash/dash-manifest-garbled-webm.mpd35
-rw-r--r--dom/media/test/dash/dash-manifest-garbled.mpd1
-rw-r--r--dom/media/test/dash/dash-manifest-sjs.mpd35
-rw-r--r--dom/media/test/dash/dash-manifest.mpd35
-rw-r--r--dom/media/test/dash/dash-webm-audio-128k.webmbin0 -> 41946 bytes
-rw-r--r--dom/media/test/dash/dash-webm-video-320x180.webmbin0 -> 35123 bytes
-rw-r--r--dom/media/test/dash/dash-webm-video-428x240.webmbin0 -> 50206 bytes
-rw-r--r--dom/media/test/dash/garbled.webm1
-rw-r--r--dom/media/test/dash_detect_stream_switch.sjs143
-rw-r--r--dom/media/test/detodos-recorder-test.opusbin0 -> 1507 bytes
-rw-r--r--dom/media/test/detodos-recorder-test.opus^headers^1
-rw-r--r--dom/media/test/detodos-short.opusbin0 -> 648 bytes
-rw-r--r--dom/media/test/detodos-short.opus^headers^1
-rw-r--r--dom/media/test/detodos-short.webmbin0 -> 1085 bytes
-rw-r--r--dom/media/test/detodos-short.webm^headers^1
-rw-r--r--dom/media/test/detodos.opusbin0 -> 6270 bytes
-rw-r--r--dom/media/test/detodos.opus^headers^1
-rw-r--r--dom/media/test/detodos.webmbin0 -> 11701 bytes
-rw-r--r--dom/media/test/detodos.webm^headers^1
-rw-r--r--dom/media/test/dirac.oggbin0 -> 106338 bytes
-rw-r--r--dom/media/test/dirac.ogg^headers^1
-rw-r--r--dom/media/test/dynamic_resource.sjs53
-rw-r--r--dom/media/test/eme.js479
-rw-r--r--dom/media/test/eme_standalone.js286
-rw-r--r--dom/media/test/empty_size.mp3bin0 -> 90368 bytes
-rw-r--r--dom/media/test/file_access_controls.html160
-rw-r--r--dom/media/test/file_eme_createMediaKeys.html47
-rw-r--r--dom/media/test/file_playback_and_bfcache.html57
-rw-r--r--dom/media/test/flac-noheader-s16.flacbin0 -> 242826 bytes
-rw-r--r--dom/media/test/flac-noheader-s16.flac^headers^1
-rw-r--r--dom/media/test/flac-s24.flacbin0 -> 980951 bytes
-rw-r--r--dom/media/test/flac-s24.flac^headers^1
-rw-r--r--dom/media/test/flac-sample-cenc.mp4bin0 -> 336823 bytes
-rw-r--r--dom/media/test/flac-sample-cenc.mp4^headers^1
-rw-r--r--dom/media/test/flac-sample.mp4bin0 -> 876556 bytes
-rw-r--r--dom/media/test/flac-sample.mp4^headers^1
-rw-r--r--dom/media/test/force_octet_stream.mp4bin0 -> 13708 bytes
-rw-r--r--dom/media/test/force_octet_stream.mp4^headers^2
-rw-r--r--dom/media/test/fragment_noplay.js19
-rw-r--r--dom/media/test/fragment_play.js92
-rw-r--r--dom/media/test/gUM_support.js103
-rw-r--r--dom/media/test/gizmo-frag.mp4bin0 -> 152132 bytes
-rw-r--r--dom/media/test/gizmo-noaudio.mp4bin0 -> 342980 bytes
-rw-r--r--dom/media/test/gizmo-noaudio.mp4^headers^1
-rw-r--r--dom/media/test/gizmo-noaudio.webmbin0 -> 112663 bytes
-rw-r--r--dom/media/test/gizmo-noaudio.webm^headers^1
-rw-r--r--dom/media/test/gizmo-short.mp4bin0 -> 29905 bytes
-rw-r--r--dom/media/test/gizmo-short.mp4^headers^1
-rw-r--r--dom/media/test/gizmo.mp4bin0 -> 455255 bytes
-rw-r--r--dom/media/test/gizmo.mp4^headers^1
-rw-r--r--dom/media/test/gizmo.webmbin0 -> 159035 bytes
-rw-r--r--dom/media/test/gizmo.webm^headers^1
-rw-r--r--dom/media/test/gzipped_mp4.sjs25
-rw-r--r--dom/media/test/hevc_white_frame.mp4bin0 -> 3358 bytes
-rw-r--r--dom/media/test/hevc_white_frame.mp4^headers^1
-rw-r--r--dom/media/test/hevc_white_red_frames.mp4bin0 -> 29235 bytes
-rw-r--r--dom/media/test/hevc_white_red_frames.mp4^headers^1
-rw-r--r--dom/media/test/hls/400x300_prog_index.m3u810
-rw-r--r--dom/media/test/hls/400x300_prog_index_5s.m3u88
-rw-r--r--dom/media/test/hls/400x300_seg0.tsbin0 -> 291588 bytes
-rw-r--r--dom/media/test/hls/400x300_seg0_5s.tsbin0 -> 168636 bytes
-rw-r--r--dom/media/test/hls/400x300_seg1.tsbin0 -> 288204 bytes
-rw-r--r--dom/media/test/hls/416x243_prog_index_5s.m3u88
-rw-r--r--dom/media/test/hls/416x243_seg0_5s.tsbin0 -> 197400 bytes
-rw-r--r--dom/media/test/hls/640x480_prog_index.m3u810
-rw-r--r--dom/media/test/hls/640x480_seg0.tsbin0 -> 814228 bytes
-rw-r--r--dom/media/test/hls/640x480_seg1.tsbin0 -> 796368 bytes
-rw-r--r--dom/media/test/hls/960x720_prog_index.m3u810
-rw-r--r--dom/media/test/hls/960x720_seg0.tsbin0 -> 1878120 bytes
-rw-r--r--dom/media/test/hls/960x720_seg1.tsbin0 -> 1839392 bytes
-rw-r--r--dom/media/test/hls/bipbop_16x9_single.m3u85
-rw-r--r--dom/media/test/hls/bipbop_4x3_single.m3u84
-rw-r--r--dom/media/test/hls/bipbop_4x3_variant.m3u810
-rw-r--r--dom/media/test/huge-id3.mp3bin0 -> 141774 bytes
-rw-r--r--dom/media/test/huge-id3.mp3^headers^1
-rw-r--r--dom/media/test/id3tags.mp3bin0 -> 3530 bytes
-rw-r--r--dom/media/test/id3tags.mp3^headers^1
-rw-r--r--dom/media/test/id3v1afterlongid3v2.mp3bin0 -> 10229 bytes
-rw-r--r--dom/media/test/invalid-cmap-s0c0.opusbin0 -> 6835 bytes
-rw-r--r--dom/media/test/invalid-cmap-s0c0.opus^headers^1
-rw-r--r--dom/media/test/invalid-cmap-s0c2.opusbin0 -> 6834 bytes
-rw-r--r--dom/media/test/invalid-cmap-s0c2.opus^headers^1
-rw-r--r--dom/media/test/invalid-cmap-s1c2.opusbin0 -> 6848 bytes
-rw-r--r--dom/media/test/invalid-cmap-s1c2.opus^headers^1
-rw-r--r--dom/media/test/invalid-cmap-short.opusbin0 -> 6854 bytes
-rw-r--r--dom/media/test/invalid-cmap-short.opus^headers^1
-rw-r--r--dom/media/test/invalid-discard_on_multi_blocks.webmbin0 -> 19636 bytes
-rw-r--r--dom/media/test/invalid-discard_on_multi_blocks.webm^headers^1
-rw-r--r--dom/media/test/invalid-excess_discard.webmbin0 -> 18442 bytes
-rw-r--r--dom/media/test/invalid-excess_discard.webm^headers^1
-rw-r--r--dom/media/test/invalid-excess_neg_discard.webmbin0 -> 18442 bytes
-rw-r--r--dom/media/test/invalid-excess_neg_discard.webm^headers^1
-rw-r--r--dom/media/test/invalid-m0c0.opusbin0 -> 2471 bytes
-rw-r--r--dom/media/test/invalid-m0c0.opus^headers^1
-rw-r--r--dom/media/test/invalid-m0c3.opusbin0 -> 2471 bytes
-rw-r--r--dom/media/test/invalid-m0c3.opus^headers^1
-rw-r--r--dom/media/test/invalid-m1c0.opusbin0 -> 6836 bytes
-rw-r--r--dom/media/test/invalid-m1c0.opus^headers^1
-rw-r--r--dom/media/test/invalid-m1c9.opusbin0 -> 6836 bytes
-rw-r--r--dom/media/test/invalid-m1c9.opus^headers^1
-rw-r--r--dom/media/test/invalid-m2c0.opusbin0 -> 2471 bytes
-rw-r--r--dom/media/test/invalid-m2c0.opus^headers^1
-rw-r--r--dom/media/test/invalid-m2c1.opusbin0 -> 2455 bytes
-rw-r--r--dom/media/test/invalid-m2c1.opus^headers^1
-rw-r--r--dom/media/test/invalid-neg_discard.webmbin0 -> 18442 bytes
-rw-r--r--dom/media/test/invalid-neg_discard.webm^headers^1
-rw-r--r--dom/media/test/invalid-preskip.webmbin0 -> 7251 bytes
-rw-r--r--dom/media/test/invalid-preskip.webm^headers^1
-rw-r--r--dom/media/test/make-headers.sh18
-rw-r--r--dom/media/test/manifest.js2556
-rw-r--r--dom/media/test/midflight-redirect.sjs87
-rw-r--r--dom/media/test/mochitest.toml951
-rw-r--r--dom/media/test/mochitest_background_video.toml800
-rw-r--r--dom/media/test/mochitest_bugs.toml812
-rw-r--r--dom/media/test/mochitest_compat.toml1033
-rw-r--r--dom/media/test/mochitest_eme.toml850
-rw-r--r--dom/media/test/mochitest_eme_compat.toml793
-rw-r--r--dom/media/test/mochitest_media_engine.toml10
-rw-r--r--dom/media/test/mochitest_media_recorder.toml856
-rw-r--r--dom/media/test/mochitest_seek.toml824
-rw-r--r--dom/media/test/mochitest_stream.toml792
-rw-r--r--dom/media/test/multi_id3v2.mp3bin0 -> 5039737 bytes
-rw-r--r--dom/media/test/multiple-bos-more-header-fileds.oggbin0 -> 27527 bytes
-rw-r--r--dom/media/test/multiple-bos-more-header-fileds.ogg^headers^1
-rw-r--r--dom/media/test/multiple-bos.oggbin0 -> 33045 bytes
-rw-r--r--dom/media/test/multiple-bos.ogg^headers^1
-rw-r--r--dom/media/test/no-container-codec-delay.webmbin0 -> 66250 bytes
-rw-r--r--dom/media/test/no-cues.webmbin0 -> 220609 bytes
-rw-r--r--dom/media/test/no-cues.webm^headers^1
-rw-r--r--dom/media/test/notags.mp3bin0 -> 2506 bytes
-rw-r--r--dom/media/test/notags.mp3^headers^1
-rw-r--r--dom/media/test/opus-mapping2.mp4bin0 -> 308048 bytes
-rw-r--r--dom/media/test/opus-mapping2.mp4^headers^1
-rw-r--r--dom/media/test/opus-mapping2.webmbin0 -> 309387 bytes
-rw-r--r--dom/media/test/opus-mapping2.webm^headers^1
-rw-r--r--dom/media/test/opus-sample-cenc.mp4bin0 -> 21958 bytes
-rw-r--r--dom/media/test/opus-sample-cenc.mp4^headers^1
-rw-r--r--dom/media/test/opus-sample.mp4bin0 -> 105690 bytes
-rw-r--r--dom/media/test/opus-sample.mp4^headers^1
-rw-r--r--dom/media/test/owl-funnier-id3.mp3bin0 -> 69603 bytes
-rw-r--r--dom/media/test/owl-funnier-id3.mp3^headers^1
-rw-r--r--dom/media/test/owl-funny-id3.mp3bin0 -> 71696 bytes
-rw-r--r--dom/media/test/owl-funny-id3.mp3^headers^1
-rw-r--r--dom/media/test/owl-short.mp3bin0 -> 11016 bytes
-rw-r--r--dom/media/test/owl-short.mp3^headers^1
-rw-r--r--dom/media/test/owl.mp3bin0 -> 67430 bytes
-rw-r--r--dom/media/test/owl.mp3^headers^1
-rw-r--r--dom/media/test/padding-spanning-multiple-packets.mp3bin0 -> 117600 bytes
-rw-r--r--dom/media/test/pixel_aspect_ratio.mp4bin0 -> 1806042 bytes
-rw-r--r--dom/media/test/play_promise.js3
-rw-r--r--dom/media/test/poster-test.jpgbin0 -> 58493 bytes
-rw-r--r--dom/media/test/r11025_msadpcm_c1.wavbin0 -> 5978 bytes
-rw-r--r--dom/media/test/r11025_msadpcm_c1.wav^headers^1
-rw-r--r--dom/media/test/r11025_s16_c1-short.wavbin0 -> 8270 bytes
-rw-r--r--dom/media/test/r11025_s16_c1-short.wav^headers^1
-rw-r--r--dom/media/test/r11025_s16_c1.wavbin0 -> 22094 bytes
-rw-r--r--dom/media/test/r11025_s16_c1.wav^headers^1
-rw-r--r--dom/media/test/r11025_s16_c1_trailing.wavbin0 -> 22095 bytes
-rw-r--r--dom/media/test/r11025_s16_c1_trailing.wav^headers^1
-rw-r--r--dom/media/test/r11025_u8_c1.wavbin0 -> 11069 bytes
-rw-r--r--dom/media/test/r11025_u8_c1.wav^headers^1
-rw-r--r--dom/media/test/r11025_u8_c1_trunc.wavbin0 -> 20000 bytes
-rw-r--r--dom/media/test/r11025_u8_c1_trunc.wav^headers^1
-rw-r--r--dom/media/test/r16000_u8_c1_list.wavbin0 -> 68318 bytes
-rw-r--r--dom/media/test/r16000_u8_c1_list.wav^headers^1
-rw-r--r--dom/media/test/rdd_process_xpcom/RddProcessTest.cpp69
-rw-r--r--dom/media/test/rdd_process_xpcom/RddProcessTest.h28
-rw-r--r--dom/media/test/rdd_process_xpcom/components.conf15
-rw-r--r--dom/media/test/rdd_process_xpcom/moz.build21
-rw-r--r--dom/media/test/rdd_process_xpcom/nsIRddProcessTest.idl25
-rw-r--r--dom/media/test/reactivate_helper.html57
-rw-r--r--dom/media/test/red-46x48.mp4bin0 -> 1548 bytes
-rw-r--r--dom/media/test/red-46x48.mp4^headers^1
-rw-r--r--dom/media/test/red-48x46.mp4bin0 -> 1548 bytes
-rw-r--r--dom/media/test/red-48x46.mp4^headers^1
-rw-r--r--dom/media/test/redirect.sjs35
-rw-r--r--dom/media/test/referer.sjs49
-rw-r--r--dom/media/test/reftest/av1hdr2020.mp4bin0 -> 109327 bytes
-rw-r--r--dom/media/test/reftest/av1hdr2020.pngbin0 -> 4162799 bytes
-rw-r--r--dom/media/test/reftest/bipbop_300_215kbps.mp4.lastframe-ref.html4
-rw-r--r--dom/media/test/reftest/bipbop_300_215kbps.mp4.lastframe.html19
-rw-r--r--dom/media/test/reftest/color_quads/720p.pngbin0 -> 8722 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.gbrp.av1.mp4bin0 -> 968 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.gbrp.av1.webmbin0 -> 669 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.gbrp.h264.mp4bin0 -> 1874 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.gbrp.vp9.mp4bin0 -> 1102 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.gbrp.vp9.webmbin0 -> 808 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p.av1.mp4bin0 -> 1016 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p.av1.webmbin0 -> 717 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p.h264.mp4bin0 -> 1951 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p.vp9.mp4bin0 -> 1116 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p.vp9.webmbin0 -> 822 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p10.av1.mp4bin0 -> 1031 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p10.av1.webmbin0 -> 732 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p10.h264.mp4bin0 -> 1990 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p10.vp9.mp4bin0 -> 1153 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p10.vp9.webmbin0 -> 859 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.gbrp.av1.mp4bin0 -> 968 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.gbrp.av1.webmbin0 -> 669 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.gbrp.h264.mp4bin0 -> 1873 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.gbrp.vp9.mp4bin0 -> 1102 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.gbrp.vp9.webmbin0 -> 808 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p.av1.mp4bin0 -> 1012 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p.av1.webmbin0 -> 713 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p.h264.mp4bin0 -> 1946 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p.vp9.mp4bin0 -> 1111 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p.vp9.webmbin0 -> 817 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p10.av1.mp4bin0 -> 1036 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p10.av1.webmbin0 -> 737 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p10.h264.mp4bin0 -> 1989 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p10.vp9.mp4bin0 -> 1148 bytes
-rw-r--r--dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p10.vp9.webmbin0 -> 854 bytes
-rw-r--r--dom/media/test/reftest/color_quads/reftest.list69
-rw-r--r--dom/media/test/reftest/frame_order.mp4bin0 -> 7971 bytes
-rw-r--r--dom/media/test/reftest/frame_order_mp4-ref.html13
-rw-r--r--dom/media/test/reftest/frame_order_mp4.html37
-rw-r--r--dom/media/test/reftest/gen_combos.py257
-rw-r--r--dom/media/test/reftest/generateREF.html104
-rw-r--r--dom/media/test/reftest/gizmo.mp4.55thframe-ref.html7
-rw-r--r--dom/media/test/reftest/gizmo.mp4.seek.html36
-rw-r--r--dom/media/test/reftest/image-10bits-rendering-720-90-ref.html4
-rw-r--r--dom/media/test/reftest/image-10bits-rendering-720-90-video.html22
-rw-r--r--dom/media/test/reftest/image-10bits-rendering-720-ref.html4
-rw-r--r--dom/media/test/reftest/image-10bits-rendering-720-video.html19
-rw-r--r--dom/media/test/reftest/image-10bits-rendering-720.video.html19
-rw-r--r--dom/media/test/reftest/image-10bits-rendering-90-ref.html4
-rw-r--r--dom/media/test/reftest/image-10bits-rendering-90-video.html22
-rw-r--r--dom/media/test/reftest/image-10bits-rendering-ref.html4
-rw-r--r--dom/media/test/reftest/image-10bits-rendering-video.html22
-rw-r--r--dom/media/test/reftest/incorrect_display_in_bytestream_vp8-ref.html13
-rw-r--r--dom/media/test/reftest/incorrect_display_in_bytestream_vp8.html33
-rw-r--r--dom/media/test/reftest/incorrect_display_in_bytestream_vp8.webmbin0 -> 84160 bytes
-rw-r--r--dom/media/test/reftest/incorrect_display_in_bytestream_vp9-ref.html12
-rw-r--r--dom/media/test/reftest/incorrect_display_in_bytestream_vp9.html33
-rw-r--r--dom/media/test/reftest/incorrect_display_in_bytestream_vp9.webmbin0 -> 740554 bytes
-rw-r--r--dom/media/test/reftest/reftest.list15
-rw-r--r--dom/media/test/reftest/reftest_img.html20
-rw-r--r--dom/media/test/reftest/reftest_video.html64
-rw-r--r--dom/media/test/reftest/short.mp4.firstframe-ref.html4
-rw-r--r--dom/media/test/reftest/short.mp4.firstframe.html19
-rw-r--r--dom/media/test/reftest/short.mp4.lastframe-ref.html4
-rw-r--r--dom/media/test/reftest/short.mp4.lastframe.html42
-rw-r--r--dom/media/test/reftest/uneven_frame_duration_video-ref.html7
-rw-r--r--dom/media/test/reftest/uneven_frame_duration_video.html39
-rw-r--r--dom/media/test/reftest/uneven_frame_durations.mp4bin0 -> 2424023 bytes
-rw-r--r--dom/media/test/reftest/uneven_frame_durations_3.8s_frame.pngbin0 -> 224136 bytes
-rw-r--r--dom/media/test/reftest/vp9hdr2020.pngbin0 -> 5083456 bytes
-rw-r--r--dom/media/test/reftest/vp9hdr2020.webmbin0 -> 108855 bytes
-rw-r--r--dom/media/test/resolution-change.webmbin0 -> 7166 bytes
-rw-r--r--dom/media/test/resolution-change.webm^headers^1
-rw-r--r--dom/media/test/sample-encrypted-sgpdstbl-sbgptraf.mp4bin0 -> 122703 bytes
-rw-r--r--dom/media/test/sample-encrypted-sgpdstbl-sbgptraf.mp4^headers^1
-rw-r--r--dom/media/test/sample-fisbone-skeleton4.ogvbin0 -> 8747 bytes
-rw-r--r--dom/media/test/sample-fisbone-skeleton4.ogv^headers^1
-rw-r--r--dom/media/test/sample-fisbone-wrong-header.ogvbin0 -> 8703 bytes
-rw-r--r--dom/media/test/sample-fisbone-wrong-header.ogv^headers^1
-rw-r--r--dom/media/test/sample.3g2bin0 -> 28561 bytes
-rw-r--r--dom/media/test/sample.3gpbin0 -> 28561 bytes
-rw-r--r--dom/media/test/seek-short.ogvbin0 -> 79921 bytes
-rw-r--r--dom/media/test/seek-short.ogv^headers^1
-rw-r--r--dom/media/test/seek-short.webmbin0 -> 19267 bytes
-rw-r--r--dom/media/test/seek-short.webm^headers^1
-rw-r--r--dom/media/test/seek.ogvbin0 -> 285310 bytes
-rw-r--r--dom/media/test/seek.ogv^headers^1
-rw-r--r--dom/media/test/seek.webmbin0 -> 215529 bytes
-rw-r--r--dom/media/test/seek.webm^headers^1
-rw-r--r--dom/media/test/seekLies.sjs22
-rw-r--r--dom/media/test/seek_support.js61
-rw-r--r--dom/media/test/seek_with_sound.oggbin0 -> 299507 bytes
-rw-r--r--dom/media/test/seek_with_sound.ogg^headers^1
-rw-r--r--dom/media/test/short-aac-encrypted-audio.mp4bin0 -> 5267 bytes
-rw-r--r--dom/media/test/short-aac-encrypted-audio.mp4^headers^1
-rw-r--r--dom/media/test/short-audio-fragmented-cenc-without-pssh.mp4bin0 -> 9261 bytes
-rw-r--r--dom/media/test/short-audio-fragmented-cenc-without-pssh.mp4^headers^1
-rw-r--r--dom/media/test/short-cenc-pssh-in-moof.mp4bin0 -> 14860 bytes
-rw-r--r--dom/media/test/short-cenc.mp4bin0 -> 14860 bytes
-rw-r--r--dom/media/test/short-cenc.xml37
-rw-r--r--dom/media/test/short-video.ogvbin0 -> 16049 bytes
-rw-r--r--dom/media/test/short-video.ogv^headers^1
-rw-r--r--dom/media/test/short-vp9-encrypted-video.mp4bin0 -> 6727 bytes
-rw-r--r--dom/media/test/short-vp9-encrypted-video.mp4^headers^1
-rw-r--r--dom/media/test/short.mp4bin0 -> 13708 bytes
-rw-r--r--dom/media/test/short.mp4.gzbin0 -> 6708 bytes
-rw-r--r--dom/media/test/short.mp4^headers^1
-rw-r--r--dom/media/test/shorter_audio_than_video_3s.webmbin0 -> 31229 bytes
-rw-r--r--dom/media/test/shorter_audio_than_video_3s.webm^headers^1
-rw-r--r--dom/media/test/sin-441-1s-44100-afconvert.mp4bin0 -> 9738 bytes
-rw-r--r--dom/media/test/sin-441-1s-44100-fdk_aac.mp4bin0 -> 9986 bytes
-rw-r--r--dom/media/test/sin-441-1s-44100-lame.mp3bin0 -> 8586 bytes
-rw-r--r--dom/media/test/sin-441-1s-44100.flacbin0 -> 24203 bytes
-rw-r--r--dom/media/test/sin-441-1s-44100.oggbin0 -> 5180 bytes
-rw-r--r--dom/media/test/sin-441-1s-44100.opusbin0 -> 10634 bytes
-rw-r--r--dom/media/test/sine.webmbin0 -> 17510 bytes
-rw-r--r--dom/media/test/sine.webm^headers^1
-rw-r--r--dom/media/test/single-xing-header-no-content-length.mp3bin0 -> 88834 bytes
-rw-r--r--dom/media/test/single-xing-header-no-content-length.mp3^headers^3
-rw-r--r--dom/media/test/sintel-short-clearkey-subsample-encrypted-audio.webmbin0 -> 30362 bytes
-rw-r--r--dom/media/test/sintel-short-clearkey-subsample-encrypted-audio.webm^headers^1
-rw-r--r--dom/media/test/sintel-short-clearkey-subsample-encrypted-video.webmbin0 -> 46703 bytes
-rw-r--r--dom/media/test/sintel-short-clearkey-subsample-encrypted-video.webm^headers^1
-rw-r--r--dom/media/test/small-shot-mp3.mp4bin0 -> 7491 bytes
-rw-r--r--dom/media/test/small-shot-mp3.mp4^headers^1
-rw-r--r--dom/media/test/small-shot.flacbin0 -> 16430 bytes
-rw-r--r--dom/media/test/small-shot.m4abin0 -> 2710 bytes
-rw-r--r--dom/media/test/small-shot.mp3bin0 -> 6825 bytes
-rw-r--r--dom/media/test/small-shot.mp3^headers^1
-rw-r--r--dom/media/test/small-shot.oggbin0 -> 6416 bytes
-rw-r--r--dom/media/test/small-shot.ogg^headers^1
-rw-r--r--dom/media/test/sound.oggbin0 -> 2603 bytes
-rw-r--r--dom/media/test/sound.ogg^headers^1
-rw-r--r--dom/media/test/spacestorm-1000Hz-100ms.oggbin0 -> 3270 bytes
-rw-r--r--dom/media/test/spacestorm-1000Hz-100ms.ogg^headers^1
-rw-r--r--dom/media/test/split.webmbin0 -> 105755 bytes
-rw-r--r--dom/media/test/split.webm^headers^1
-rw-r--r--dom/media/test/street.mp4bin0 -> 1505751 bytes
-rw-r--r--dom/media/test/street.mp4^headers^1
-rw-r--r--dom/media/test/sync.webmbin0 -> 397383 bytes
-rw-r--r--dom/media/test/test-1-mono.opusbin0 -> 4086 bytes
-rw-r--r--dom/media/test/test-1-mono.opus^headers^1
-rw-r--r--dom/media/test/test-2-stereo.opusbin0 -> 24973 bytes
-rw-r--r--dom/media/test/test-2-stereo.opus^headers^1
-rw-r--r--dom/media/test/test-3-LCR.opusbin0 -> 39471 bytes
-rw-r--r--dom/media/test/test-3-LCR.opus^headers^1
-rw-r--r--dom/media/test/test-4-quad.opusbin0 -> 129906 bytes
-rw-r--r--dom/media/test/test-4-quad.opus^headers^1
-rw-r--r--dom/media/test/test-5-5.0.opusbin0 -> 164935 bytes
-rw-r--r--dom/media/test/test-5-5.0.opus^headers^1
-rw-r--r--dom/media/test/test-6-5.1.opusbin0 -> 288195 bytes
-rw-r--r--dom/media/test/test-6-5.1.opus^headers^1
-rw-r--r--dom/media/test/test-7-6.1.opusbin0 -> 401668 bytes
-rw-r--r--dom/media/test/test-7-6.1.opus^headers^1
-rw-r--r--dom/media/test/test-8-7.1.opusbin0 -> 543119 bytes
-rw-r--r--dom/media/test/test-8-7.1.opus^headers^1
-rw-r--r--dom/media/test/test-stereo-phase-inversion-180.opusbin0 -> 14011 bytes
-rw-r--r--dom/media/test/test-stereo-phase-inversion-180.opus^headers^1
-rw-r--r--dom/media/test/test_VideoPlaybackQuality.html61
-rw-r--r--dom/media/test/test_VideoPlaybackQuality_disabled.html37
-rw-r--r--dom/media/test/test_access_control.html62
-rw-r--r--dom/media/test/test_arraybuffer.html83
-rw-r--r--dom/media/test/test_aspectratio_mp4.html46
-rw-r--r--dom/media/test/test_audio1.html33
-rw-r--r--dom/media/test/test_audio2.html33
-rw-r--r--dom/media/test/test_audioDocumentTitle.html56
-rw-r--r--dom/media/test/test_background_video_cancel_suspend_taint.html70
-rw-r--r--dom/media/test/test_background_video_cancel_suspend_visible.html69
-rw-r--r--dom/media/test/test_background_video_drawimage_with_suspended_video.html76
-rw-r--r--dom/media/test/test_background_video_ended_event.html48
-rw-r--r--dom/media/test/test_background_video_no_suspend_disabled.html36
-rw-r--r--dom/media/test/test_background_video_no_suspend_not_in_tree.html56
-rw-r--r--dom/media/test/test_background_video_no_suspend_short_vid.html38
-rw-r--r--dom/media/test/test_background_video_resume_after_end_show_last_frame.html141
-rw-r--r--dom/media/test/test_background_video_resume_looping_video_without_audio.html81
-rw-r--r--dom/media/test/test_background_video_suspend.html81
-rw-r--r--dom/media/test/test_background_video_suspend_ends.html55
-rw-r--r--dom/media/test/test_background_video_suspend_ready_state.html74
-rw-r--r--dom/media/test/test_background_video_tainted_by_capturestream.html46
-rw-r--r--dom/media/test/test_background_video_tainted_by_createimagebitmap.html42
-rw-r--r--dom/media/test/test_background_video_tainted_by_drawimage.html58
-rw-r--r--dom/media/test/test_buffered.html117
-rw-r--r--dom/media/test/test_bug1113600.html50
-rw-r--r--dom/media/test/test_bug1120222.html42
-rw-r--r--dom/media/test/test_bug1242338.html66
-rw-r--r--dom/media/test/test_bug1248229.html35
-rw-r--r--dom/media/test/test_bug1431810_opus_downmix_to_mono.html139
-rw-r--r--dom/media/test/test_bug1512958.html74
-rw-r--r--dom/media/test/test_bug1553262.html31
-rw-r--r--dom/media/test/test_bug448534.html71
-rw-r--r--dom/media/test/test_bug463162.xhtml78
-rw-r--r--dom/media/test/test_bug465498.html83
-rw-r--r--dom/media/test/test_bug495145.html95
-rw-r--r--dom/media/test/test_bug495300.html63
-rw-r--r--dom/media/test/test_bug654550.html84
-rw-r--r--dom/media/test/test_bug686942.html68
-rw-r--r--dom/media/test/test_bug726904.html56
-rw-r--r--dom/media/test/test_bug874897.html68
-rw-r--r--dom/media/test/test_bug879717.html130
-rw-r--r--dom/media/test/test_bug895305.html42
-rw-r--r--dom/media/test/test_bug919265.html30
-rw-r--r--dom/media/test/test_can_play_type.html40
-rw-r--r--dom/media/test/test_can_play_type_mpeg.html166
-rw-r--r--dom/media/test/test_can_play_type_no_ogg.html42
-rw-r--r--dom/media/test/test_can_play_type_ogg.html37
-rw-r--r--dom/media/test/test_can_play_type_wave.html30
-rw-r--r--dom/media/test/test_can_play_type_webm.html39
-rw-r--r--dom/media/test/test_capture_stream_av_sync.html276
-rw-r--r--dom/media/test/test_chaining.html92
-rw-r--r--dom/media/test/test_cloneElementVisually_ended_video.html48
-rw-r--r--dom/media/test/test_cloneElementVisually_mediastream.html70
-rw-r--r--dom/media/test/test_cloneElementVisually_mediastream_multitrack.html88
-rw-r--r--dom/media/test/test_cloneElementVisually_no_suspend.html90
-rw-r--r--dom/media/test/test_cloneElementVisually_paused.html45
-rw-r--r--dom/media/test/test_cloneElementVisually_poster.html53
-rw-r--r--dom/media/test/test_cloneElementVisually_resource_change.html67
-rw-r--r--dom/media/test/test_clone_media_element.html54
-rw-r--r--dom/media/test/test_closing_connections.html58
-rw-r--r--dom/media/test/test_constants.html228
-rw-r--r--dom/media/test/test_controls.html33
-rw-r--r--dom/media/test/test_cueless_webm_seek-1.html136
-rw-r--r--dom/media/test/test_cueless_webm_seek-2.html126
-rw-r--r--dom/media/test/test_cueless_webm_seek-3.html120
-rw-r--r--dom/media/test/test_currentTime.html19
-rw-r--r--dom/media/test/test_debug_data_helpers.html74
-rw-r--r--dom/media/test/test_decode_error.html66
-rw-r--r--dom/media/test/test_decode_error_crossorigin.html54
-rw-r--r--dom/media/test/test_decoder_disable.html78
-rw-r--r--dom/media/test/test_defaultMuted.html54
-rw-r--r--dom/media/test/test_delay_load.html108
-rw-r--r--dom/media/test/test_duration_after_error.html54
-rw-r--r--dom/media/test/test_eme_autoplay.html115
-rw-r--r--dom/media/test/test_eme_canvas_blocked.html58
-rw-r--r--dom/media/test/test_eme_createMediaKeys_iframes.html192
-rw-r--r--dom/media/test/test_eme_detach_media_keys.html63
-rw-r--r--dom/media/test/test_eme_detach_reattach_same_mediakeys_during_playback.html141
-rw-r--r--dom/media/test/test_eme_getstatusforpolicy.html93
-rw-r--r--dom/media/test/test_eme_initDataTypes.html130
-rw-r--r--dom/media/test/test_eme_mfcdm_generate_request.html93
-rw-r--r--dom/media/test/test_eme_mfcdm_getstatusforpolicy.html115
-rw-r--r--dom/media/test/test_eme_missing_pssh.html92
-rw-r--r--dom/media/test/test_eme_non_mse_fails.html95
-rw-r--r--dom/media/test/test_eme_playback.html188
-rw-r--r--dom/media/test/test_eme_protection_query.html250
-rw-r--r--dom/media/test/test_eme_pssh_in_moof.html141
-rw-r--r--dom/media/test/test_eme_requestKeySystemAccess.html477
-rw-r--r--dom/media/test/test_eme_requestMediaKeySystemAccess_with_app_approval.html202
-rw-r--r--dom/media/test/test_eme_request_notifications.html82
-rw-r--r--dom/media/test/test_eme_sample_groups_playback.html130
-rw-r--r--dom/media/test/test_eme_session_callable_value.html34
-rw-r--r--dom/media/test/test_eme_setMediaKeys_before_attach_MediaSource.html37
-rw-r--r--dom/media/test/test_eme_special_key_system.html63
-rw-r--r--dom/media/test/test_eme_stream_capture_blocked_case1.html59
-rw-r--r--dom/media/test/test_eme_stream_capture_blocked_case2.html52
-rw-r--r--dom/media/test/test_eme_stream_capture_blocked_case3.html50
-rw-r--r--dom/media/test/test_eme_unsetMediaKeys_then_capture.html108
-rw-r--r--dom/media/test/test_eme_waitingforkey.html78
-rw-r--r--dom/media/test/test_eme_wideinve_l1_installation.html86
-rw-r--r--dom/media/test/test_eme_wv_privacy.html53
-rw-r--r--dom/media/test/test_empty_resource.html58
-rw-r--r--dom/media/test/test_error_in_video_document.html59
-rw-r--r--dom/media/test/test_error_on_404.html84
-rw-r--r--dom/media/test/test_fastSeek-forwards.html77
-rw-r--r--dom/media/test/test_fastSeek.html88
-rw-r--r--dom/media/test/test_fragment_noplay.html127
-rw-r--r--dom/media/test/test_fragment_play.html91
-rw-r--r--dom/media/test/test_hevc_playback.html45
-rw-r--r--dom/media/test/test_hevc_support.html47
-rw-r--r--dom/media/test/test_hls_player_independency.html53
-rw-r--r--dom/media/test/test_hw_video_decoding.html119
-rw-r--r--dom/media/test/test_imagecapture.html128
-rw-r--r--dom/media/test/test_info_leak.html174
-rw-r--r--dom/media/test/test_invalid_reject.html58
-rw-r--r--dom/media/test/test_invalid_reject_play.html44
-rw-r--r--dom/media/test/test_invalid_seek.html32
-rw-r--r--dom/media/test/test_load.html217
-rw-r--r--dom/media/test/test_load_candidates.html84
-rw-r--r--dom/media/test/test_load_same_resource.html106
-rw-r--r--dom/media/test/test_load_source.html76
-rw-r--r--dom/media/test/test_load_source_empty_type.html36
-rw-r--r--dom/media/test/test_loop.html57
-rw-r--r--dom/media/test/test_looping_eventsOrder.html52
-rw-r--r--dom/media/test/test_media_selection.html142
-rw-r--r--dom/media/test/test_media_sniffer.html67
-rw-r--r--dom/media/test/test_mediacapabilities_resistfingerprinting.html69
-rw-r--r--dom/media/test/test_mediarecorder_avoid_recursion.html61
-rw-r--r--dom/media/test/test_mediarecorder_bitrate.html127
-rw-r--r--dom/media/test/test_mediarecorder_creation.html45
-rw-r--r--dom/media/test/test_mediarecorder_creation_fail.html61
-rw-r--r--dom/media/test/test_mediarecorder_fires_start_event_once_when_erroring.html45
-rw-r--r--dom/media/test/test_mediarecorder_multipletracks.html68
-rw-r--r--dom/media/test/test_mediarecorder_onerror_pause.html107
-rw-r--r--dom/media/test/test_mediarecorder_pause_resume_video.html130
-rw-r--r--dom/media/test/test_mediarecorder_playback_can_repeat.html87
-rw-r--r--dom/media/test/test_mediarecorder_principals.html132
-rw-r--r--dom/media/test/test_mediarecorder_record_4ch_audiocontext.html76
-rw-r--r--dom/media/test/test_mediarecorder_record_addtracked_stream.html182
-rw-r--r--dom/media/test/test_mediarecorder_record_audiocontext.html65
-rw-r--r--dom/media/test/test_mediarecorder_record_audiocontext_mlk.html24
-rw-r--r--dom/media/test/test_mediarecorder_record_audionode.html135
-rw-r--r--dom/media/test/test_mediarecorder_record_canvas_captureStream.html75
-rw-r--r--dom/media/test/test_mediarecorder_record_changing_video_resolution.html174
-rw-r--r--dom/media/test/test_mediarecorder_record_downsize_resolution.html148
-rw-r--r--dom/media/test/test_mediarecorder_record_getdata_afterstart.html81
-rw-r--r--dom/media/test/test_mediarecorder_record_gum_video_timeslice.html94
-rw-r--r--dom/media/test/test_mediarecorder_record_gum_video_timeslice_mixed.html100
-rw-r--r--dom/media/test/test_mediarecorder_record_immediate_stop.html115
-rw-r--r--dom/media/test/test_mediarecorder_record_no_timeslice.html106
-rw-r--r--dom/media/test/test_mediarecorder_record_session.html75
-rw-r--r--dom/media/test/test_mediarecorder_record_startstopstart.html75
-rw-r--r--dom/media/test/test_mediarecorder_record_timeslice.html105
-rw-r--r--dom/media/test/test_mediarecorder_record_upsize_resolution.html148
-rw-r--r--dom/media/test/test_mediarecorder_reload_crash.html29
-rw-r--r--dom/media/test/test_mediarecorder_state_event_order.html83
-rw-r--r--dom/media/test/test_mediarecorder_state_transition.html280
-rw-r--r--dom/media/test/test_mediarecorder_webm_support.html56
-rw-r--r--dom/media/test/test_mediastream_as_eventarget.html33
-rw-r--r--dom/media/test/test_mediatrack_consuming_mediaresource.html198
-rw-r--r--dom/media/test/test_mediatrack_consuming_mediastream.html146
-rw-r--r--dom/media/test/test_mediatrack_events.html135
-rw-r--r--dom/media/test/test_mediatrack_parsing_ogg.html72
-rw-r--r--dom/media/test/test_mediatrack_replay_from_end.html160
-rw-r--r--dom/media/test/test_metadata.html81
-rw-r--r--dom/media/test/test_midflight_redirect_blocked.html87
-rw-r--r--dom/media/test/test_mixed_principals.html94
-rw-r--r--dom/media/test/test_mozHasAudio.html42
-rw-r--r--dom/media/test/test_mp3_broadcast.html52
-rw-r--r--dom/media/test/test_mp3_with_multiple_ID3v2.html30
-rw-r--r--dom/media/test/test_multiple_mediastreamtracks.html47
-rw-r--r--dom/media/test/test_networkState.html47
-rw-r--r--dom/media/test/test_new_audio.html48
-rw-r--r--dom/media/test/test_no_load_event.html53
-rw-r--r--dom/media/test/test_not_reset_playbackRate_when_removing_nonloaded_media_from_document.html46
-rw-r--r--dom/media/test/test_paused.html21
-rw-r--r--dom/media/test/test_paused_after_ended.html53
-rw-r--r--dom/media/test/test_periodic_timeupdate.html100
-rw-r--r--dom/media/test/test_play_events.html61
-rw-r--r--dom/media/test/test_play_events_2.html60
-rw-r--r--dom/media/test/test_play_promise_1.html42
-rw-r--r--dom/media/test/test_play_promise_10.html40
-rw-r--r--dom/media/test/test_play_promise_11.html40
-rw-r--r--dom/media/test/test_play_promise_12.html45
-rw-r--r--dom/media/test/test_play_promise_13.html49
-rw-r--r--dom/media/test/test_play_promise_14.html56
-rw-r--r--dom/media/test/test_play_promise_15.html51
-rw-r--r--dom/media/test/test_play_promise_16.html47
-rw-r--r--dom/media/test/test_play_promise_17.html43
-rw-r--r--dom/media/test/test_play_promise_18.html46
-rw-r--r--dom/media/test/test_play_promise_2.html43
-rw-r--r--dom/media/test/test_play_promise_3.html47
-rw-r--r--dom/media/test/test_play_promise_4.html41
-rw-r--r--dom/media/test/test_play_promise_5.html44
-rw-r--r--dom/media/test/test_play_promise_6.html45
-rw-r--r--dom/media/test/test_play_promise_7.html47
-rw-r--r--dom/media/test/test_play_promise_8.html47
-rw-r--r--dom/media/test/test_play_promise_9.html44
-rw-r--r--dom/media/test/test_play_twice.html95
-rw-r--r--dom/media/test/test_playback.html108
-rw-r--r--dom/media/test/test_playback_and_bfcache.html72
-rw-r--r--dom/media/test/test_playback_errors.html48
-rw-r--r--dom/media/test/test_playback_hls.html91
-rw-r--r--dom/media/test/test_playback_rate.html175
-rw-r--r--dom/media/test/test_playback_rate_playpause.html66
-rw-r--r--dom/media/test/test_playback_reactivate.html77
-rw-r--r--dom/media/test/test_played.html288
-rw-r--r--dom/media/test/test_preload_actions.html581
-rw-r--r--dom/media/test/test_preload_attribute.html44
-rw-r--r--dom/media/test/test_preload_suspend.html112
-rw-r--r--dom/media/test/test_preserve_playbackrate_after_ui_play.html60
-rw-r--r--dom/media/test/test_progress.html52
-rw-r--r--dom/media/test/test_reactivate.html64
-rw-r--r--dom/media/test/test_readyState.html51
-rw-r--r--dom/media/test/test_referer.html88
-rw-r--r--dom/media/test/test_replay_metadata.html120
-rw-r--r--dom/media/test/test_reset_events_async.html58
-rw-r--r--dom/media/test/test_reset_src.html98
-rw-r--r--dom/media/test/test_resolution_change.html52
-rw-r--r--dom/media/test/test_resume.html47
-rw-r--r--dom/media/test/test_seamless_looping.html199
-rw-r--r--dom/media/test/test_seamless_looping_cancel_looping_future_frames.html61
-rw-r--r--dom/media/test/test_seamless_looping_duration.html63
-rw-r--r--dom/media/test/test_seamless_looping_media_element_state.html51
-rw-r--r--dom/media/test/test_seamless_looping_not_keep_painting_old_video_frames.html57
-rw-r--r--dom/media/test/test_seamless_looping_resume_video_decoding.html64
-rw-r--r--dom/media/test/test_seamless_looping_seek_current_time.html62
-rw-r--r--dom/media/test/test_seamless_looping_shorter_audio_than_video.html37
-rw-r--r--dom/media/test/test_seamless_looping_video.html73
-rw-r--r--dom/media/test/test_seek-1.html84
-rw-r--r--dom/media/test/test_seek-10.html56
-rw-r--r--dom/media/test/test_seek-11.html76
-rw-r--r--dom/media/test/test_seek-12.html59
-rw-r--r--dom/media/test/test_seek-13.html72
-rw-r--r--dom/media/test/test_seek-14.html43
-rw-r--r--dom/media/test/test_seek-2.html76
-rw-r--r--dom/media/test/test_seek-3.html68
-rw-r--r--dom/media/test/test_seek-4.html70
-rw-r--r--dom/media/test/test_seek-5.html69
-rw-r--r--dom/media/test/test_seek-6.html64
-rw-r--r--dom/media/test/test_seek-7.html59
-rw-r--r--dom/media/test/test_seek-8.html42
-rw-r--r--dom/media/test/test_seek-9.html41
-rw-r--r--dom/media/test/test_seekLies.html28
-rw-r--r--dom/media/test/test_seekToNextFrame.html95
-rw-r--r--dom/media/test/test_seek_duration.html62
-rw-r--r--dom/media/test/test_seek_negative.html77
-rw-r--r--dom/media/test/test_seek_nosrc.html58
-rw-r--r--dom/media/test/test_seek_out_of_range.html48
-rw-r--r--dom/media/test/test_seek_promise_bug1344357.html35
-rw-r--r--dom/media/test/test_seekable1.html66
-rw-r--r--dom/media/test/test_setSinkId_after_loop.html126
-rw-r--r--dom/media/test/test_source.html91
-rw-r--r--dom/media/test/test_source_null.html33
-rw-r--r--dom/media/test/test_source_write.html40
-rw-r--r--dom/media/test/test_standalone.html61
-rw-r--r--dom/media/test/test_streams_capture_origin.html90
-rw-r--r--dom/media/test/test_streams_element_capture.html125
-rw-r--r--dom/media/test/test_streams_element_capture_mediatrack.html100
-rw-r--r--dom/media/test/test_streams_element_capture_playback.html47
-rw-r--r--dom/media/test/test_streams_element_capture_reset.html174
-rw-r--r--dom/media/test/test_streams_element_capture_twice.html79
-rw-r--r--dom/media/test/test_streams_firstframe.html67
-rw-r--r--dom/media/test/test_streams_gc.html44
-rw-r--r--dom/media/test/test_streams_individual_pause.html77
-rw-r--r--dom/media/test/test_streams_srcObject.html60
-rw-r--r--dom/media/test/test_streams_tracks.html66
-rw-r--r--dom/media/test/test_suspend_media_by_inactive_docshell.html72
-rw-r--r--dom/media/test/test_temporary_file_blob_video_plays.html75
-rw-r--r--dom/media/test/test_timeupdate_small_files.html88
-rw-r--r--dom/media/test/test_unseekable.html101
-rw-r--r--dom/media/test/test_videoDocumentTitle.html57
-rw-r--r--dom/media/test/test_videoPlaybackQuality_totalFrames.html46
-rw-r--r--dom/media/test/test_video_dimensions.html96
-rw-r--r--dom/media/test/test_video_gzip_encoding.html25
-rw-r--r--dom/media/test/test_video_in_audio_element.html68
-rw-r--r--dom/media/test/test_video_low_power_telemetry.html205
-rw-r--r--dom/media/test/test_video_stats_resistfingerprinting.html90
-rw-r--r--dom/media/test/test_video_to_canvas.html68
-rw-r--r--dom/media/test/test_volume.html41
-rw-r--r--dom/media/test/test_vp9_superframes.html31
-rw-r--r--dom/media/test/test_wav_ended1.html43
-rw-r--r--dom/media/test/test_wav_ended2.html62
-rw-r--r--dom/media/test/tone2s-silence4s-tone2s.opusbin0 -> 67207 bytes
-rw-r--r--dom/media/test/two-xing-header-no-content-length.mp3bin0 -> 45594 bytes
-rw-r--r--dom/media/test/two-xing-header-no-content-length.mp3^headers^3
-rw-r--r--dom/media/test/variable-channel.oggbin0 -> 27749 bytes
-rw-r--r--dom/media/test/variable-channel.ogg^headers^1
-rw-r--r--dom/media/test/variable-channel.opusbin0 -> 46597 bytes
-rw-r--r--dom/media/test/variable-channel.opus^headers^1
-rw-r--r--dom/media/test/variable-preskip.opusbin0 -> 17660 bytes
-rw-r--r--dom/media/test/variable-preskip.opus^headers^1
-rw-r--r--dom/media/test/variable-samplerate.oggbin0 -> 22325 bytes
-rw-r--r--dom/media/test/variable-samplerate.ogg^headers^1
-rw-r--r--dom/media/test/variable-samplerate.opusbin0 -> 28111 bytes
-rw-r--r--dom/media/test/variable-samplerate.opus^headers^1
-rw-r--r--dom/media/test/vbr-head.mp3bin0 -> 4474 bytes
-rw-r--r--dom/media/test/vbr-head.mp3^headers^1
-rw-r--r--dom/media/test/vbr.mp3bin0 -> 300553 bytes
-rw-r--r--dom/media/test/vbr.mp3^headers^1
-rw-r--r--dom/media/test/very-short.mp3bin0 -> 1612 bytes
-rw-r--r--dom/media/test/video-overhang.oggbin0 -> 301831 bytes
-rw-r--r--dom/media/test/video-overhang.ogg^headers^1
-rw-r--r--dom/media/test/vp9-short.webmbin0 -> 3107 bytes
-rw-r--r--dom/media/test/vp9-short.webm^headers^1
-rw-r--r--dom/media/test/vp9-superframes.webmbin0 -> 173187 bytes
-rw-r--r--dom/media/test/vp9-superframes.webm^headers^1
-rw-r--r--dom/media/test/vp9.webmbin0 -> 97465 bytes
-rw-r--r--dom/media/test/vp9.webm^headers^1
-rw-r--r--dom/media/test/vp9cake-short.webmbin0 -> 25155 bytes
-rw-r--r--dom/media/test/vp9cake-short.webm^headers^1
-rw-r--r--dom/media/test/vp9cake.webmbin0 -> 141743 bytes
-rw-r--r--dom/media/test/vp9cake.webm^headers^1
-rw-r--r--dom/media/test/wave_metadata.wavbin0 -> 42706 bytes
-rw-r--r--dom/media/test/wave_metadata.wav^headers^1
-rw-r--r--dom/media/test/wave_metadata_bad_len.wavbin0 -> 42706 bytes
-rw-r--r--dom/media/test/wave_metadata_bad_len.wav^headers^1
-rw-r--r--dom/media/test/wave_metadata_bad_no_null.wavbin0 -> 42706 bytes
-rw-r--r--dom/media/test/wave_metadata_bad_no_null.wav^headers^1
-rw-r--r--dom/media/test/wave_metadata_bad_utf8.wavbin0 -> 42704 bytes
-rw-r--r--dom/media/test/wave_metadata_bad_utf8.wav^headers^1
-rw-r--r--dom/media/test/wave_metadata_unknown_tag.wavbin0 -> 42706 bytes
-rw-r--r--dom/media/test/wave_metadata_unknown_tag.wav^headers^1
-rw-r--r--dom/media/test/wave_metadata_utf8.wavbin0 -> 42704 bytes
-rw-r--r--dom/media/test/wave_metadata_utf8.wav^headers^1
-rw-r--r--dom/media/test/wavedata_alaw.wavbin0 -> 11067 bytes
-rw-r--r--dom/media/test/wavedata_alaw.wav^headers^1
-rw-r--r--dom/media/test/wavedata_float.wavbin0 -> 176458 bytes
-rw-r--r--dom/media/test/wavedata_float.wav^headers^1
-rw-r--r--dom/media/test/wavedata_s16.wavbin0 -> 22062 bytes
-rw-r--r--dom/media/test/wavedata_s16.wav^headers^1
-rw-r--r--dom/media/test/wavedata_s24.wavbin0 -> 33071 bytes
-rw-r--r--dom/media/test/wavedata_s24.wav^headers^1
-rw-r--r--dom/media/test/wavedata_u8.wavbin0 -> 11037 bytes
-rw-r--r--dom/media/test/wavedata_u8.wav^headers^1
-rw-r--r--dom/media/test/wavedata_ulaw.wavbin0 -> 11067 bytes
-rw-r--r--dom/media/test/wavedata_ulaw.wav^headers^1
-rw-r--r--dom/media/test/white-3s-black-1s.webmbin0 -> 4662 bytes
-rw-r--r--dom/media/test/white-3s-black-1s.webm^headers^1
-rw-r--r--dom/media/test/white-short.webmbin0 -> 1573 bytes
1386 files changed, 43352 insertions, 0 deletions
diff --git a/dom/media/test/16bit_wave_extrametadata.wav b/dom/media/test/16bit_wave_extrametadata.wav
new file mode 100644
index 0000000000..443ec73a3d
--- /dev/null
+++ b/dom/media/test/16bit_wave_extrametadata.wav
Binary files differ
diff --git a/dom/media/test/16bit_wave_extrametadata.wav^headers^ b/dom/media/test/16bit_wave_extrametadata.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/16bit_wave_extrametadata.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/320x240.ogv b/dom/media/test/320x240.ogv
new file mode 100644
index 0000000000..093158432a
--- /dev/null
+++ b/dom/media/test/320x240.ogv
Binary files differ
diff --git a/dom/media/test/320x240.ogv^headers^ b/dom/media/test/320x240.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/320x240.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/448636.ogv b/dom/media/test/448636.ogv
new file mode 100644
index 0000000000..628df924f8
--- /dev/null
+++ b/dom/media/test/448636.ogv
Binary files differ
diff --git a/dom/media/test/448636.ogv^headers^ b/dom/media/test/448636.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/448636.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/A4.ogv b/dom/media/test/A4.ogv
new file mode 100644
index 0000000000..de99616ece
--- /dev/null
+++ b/dom/media/test/A4.ogv
Binary files differ
diff --git a/dom/media/test/A4.ogv^headers^ b/dom/media/test/A4.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/A4.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/TestPatternHDR.mp4 b/dom/media/test/TestPatternHDR.mp4
new file mode 100644
index 0000000000..2aeb5d7f05
--- /dev/null
+++ b/dom/media/test/TestPatternHDR.mp4
Binary files differ
diff --git a/dom/media/test/VID_0001.ogg b/dom/media/test/VID_0001.ogg
new file mode 100644
index 0000000000..0068b9af85
--- /dev/null
+++ b/dom/media/test/VID_0001.ogg
Binary files differ
diff --git a/dom/media/test/VID_0001.ogg^headers^ b/dom/media/test/VID_0001.ogg^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/VID_0001.ogg^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/adts.aac b/dom/media/test/adts.aac
new file mode 100644
index 0000000000..208515464a
--- /dev/null
+++ b/dom/media/test/adts.aac
Binary files differ
diff --git a/dom/media/test/adts.aac^headers^ b/dom/media/test/adts.aac^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/adts.aac^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/allowed.sjs b/dom/media/test/allowed.sjs
new file mode 100644
index 0000000000..4460cd05af
--- /dev/null
+++ b/dom/media/test/allowed.sjs
@@ -0,0 +1,61 @@
+function parseQuery(request, key) {
+ var params = request.queryString.split("&");
+ for (var j = 0; j < params.length; ++j) {
+ var p = params[j];
+ if (p == key) {
+ return true;
+ }
+ if (p.indexOf(key + "=") == 0) {
+ return p.substring(key.length + 1);
+ }
+ if (!p.includes("=") && key == "") {
+ return p;
+ }
+ }
+ return false;
+}
+
+var types = {
+ js: "text/javascript",
+ m4s: "video/mp4",
+ mp4: "video/mp4",
+ ogg: "video/ogg",
+ ogv: "video/ogg",
+ oga: "audio/ogg",
+ webm: "video/webm",
+ wav: "audio/x-wav",
+};
+
+// Return file with name as per the query string with access control
+// allow headers.
+function handleRequest(request, response) {
+ var resource = parseQuery(request, "");
+
+ var file = Services.dirsvc.get("CurWorkD", Ci.nsIFile);
+ var fis = Cc["@mozilla.org/network/file-input-stream;1"].createInstance(
+ Ci.nsIFileInputStream
+ );
+ var bis = Cc["@mozilla.org/binaryinputstream;1"].createInstance(
+ Ci.nsIBinaryInputStream
+ );
+ var paths = "tests/dom/media/test/" + resource;
+ var split = paths.split("/");
+ for (var i = 0; i < split.length; ++i) {
+ file.append(split[i]);
+ }
+ fis.init(file, -1, -1, false);
+ dump("file=" + file + "\n");
+ bis.setInputStream(fis);
+ var bytes = bis.readBytes(bis.available());
+ response.setStatusLine(request.httpVersion, 206, "Partial Content");
+ response.setHeader(
+ "Content-Range",
+ "bytes 0-" + (bytes.length - 1) + "/" + bytes.length
+ );
+ response.setHeader("Content-Length", "" + bytes.length, false);
+ var ext = resource.substring(resource.lastIndexOf(".") + 1);
+ response.setHeader("Content-Type", types[ext], false);
+ response.setHeader("Access-Control-Allow-Origin", "*");
+ response.write(bytes, bytes.length);
+ bis.close();
+}
diff --git a/dom/media/test/ambisonics.mp4 b/dom/media/test/ambisonics.mp4
new file mode 100644
index 0000000000..4f5bcdfd26
--- /dev/null
+++ b/dom/media/test/ambisonics.mp4
Binary files differ
diff --git a/dom/media/test/ambisonics.mp4^headers^ b/dom/media/test/ambisonics.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/ambisonics.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/audio-gaps-short.ogg b/dom/media/test/audio-gaps-short.ogg
new file mode 100644
index 0000000000..e01a24bfda
--- /dev/null
+++ b/dom/media/test/audio-gaps-short.ogg
Binary files differ
diff --git a/dom/media/test/audio-gaps-short.ogg^headers^ b/dom/media/test/audio-gaps-short.ogg^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/audio-gaps-short.ogg^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/audio-gaps.ogg b/dom/media/test/audio-gaps.ogg
new file mode 100644
index 0000000000..ce96748ccd
--- /dev/null
+++ b/dom/media/test/audio-gaps.ogg
Binary files differ
diff --git a/dom/media/test/audio-gaps.ogg^headers^ b/dom/media/test/audio-gaps.ogg^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/audio-gaps.ogg^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/audio-overhang.ogg b/dom/media/test/audio-overhang.ogg
new file mode 100644
index 0000000000..c07986e7a1
--- /dev/null
+++ b/dom/media/test/audio-overhang.ogg
Binary files differ
diff --git a/dom/media/test/audio-overhang.ogg^headers^ b/dom/media/test/audio-overhang.ogg^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/audio-overhang.ogg^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/audio.wav b/dom/media/test/audio.wav
new file mode 100644
index 0000000000..c6fd5cb869
--- /dev/null
+++ b/dom/media/test/audio.wav
Binary files differ
diff --git a/dom/media/test/audio.wav^headers^ b/dom/media/test/audio.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/audio.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/av1.mp4 b/dom/media/test/av1.mp4
new file mode 100644
index 0000000000..28de929a29
--- /dev/null
+++ b/dom/media/test/av1.mp4
Binary files differ
diff --git a/dom/media/test/av1.mp4^headers^ b/dom/media/test/av1.mp4^headers^
new file mode 100644
index 0000000000..2567dc2fe5
--- /dev/null
+++ b/dom/media/test/av1.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store \ No newline at end of file
diff --git a/dom/media/test/background_video.js b/dom/media/test/background_video.js
new file mode 100644
index 0000000000..508f8fd89a
--- /dev/null
+++ b/dom/media/test/background_video.js
@@ -0,0 +1,224 @@
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+// This file expects manager to be defined in the global scope.
+/* global manager */
+/* import-globals-from manifest.js */
+
+"use strict";
+
+function startTest(test) {
+ info(test.desc);
+ SimpleTest.waitForExplicitFinish();
+ SpecialPowers.pushPrefEnv({ set: test.prefs }, () => {
+ manager.runTests(test.tests, test.runTest);
+ });
+}
+
+function nextVideoEnded(video) {
+ return nextEvent(video, "ended");
+}
+
+function nextVideoPlaying(video) {
+ return nextEvent(video, "playing");
+}
+
+function nextVideoResumes(video) {
+ return nextEvent(video, "mozexitvideosuspend");
+}
+
+function nextVideoSuspends(video) {
+ return nextEvent(video, "mozentervideosuspend");
+}
+
+/**
+ * @param {string} url video src.
+ * @returns {HTMLMediaElement} The created video element.
+ */
+function appendVideoToDoc(url, token, width, height) {
+ // Default size of (160, 120) is used by other media tests.
+ if (width === undefined) {
+ width = 160;
+ }
+ if (height === undefined) {
+ height = (3 * width) / 4;
+ }
+
+ let v = document.createElement("video");
+ v.token = token;
+ v.width = width;
+ v.height = height;
+ v.src = url;
+ document.body.appendChild(v);
+ return v;
+}
+
+function appendVideoToDocWithoutLoad(token, width, height) {
+ // Default size of (160, 120) is used by other media tests.
+ if (width === undefined) {
+ width = 160;
+ }
+ if (height === undefined) {
+ height = (3 * width) / 4;
+ }
+
+ let v = document.createElement("video");
+ v.token = token;
+ document.body.appendChild(v);
+ v.width = width;
+ v.height = height;
+ return v;
+}
+
+function loadAndWaitUntilLoadedmetadata(video, url, preloadType = "metadata") {
+ return new Promise((resolve, reject) => {
+ video.preload = preloadType;
+ video.addEventListener(
+ "loadedmetadata",
+ () => {
+ resolve();
+ },
+ true
+ );
+ video.src = url;
+ });
+}
+
+/**
+ * @param {HTMLMediaElement} video Video element with under test.
+ * @returns {Promise} Promise that is resolved when video 'visibilitychanged' event fires.
+ */
+function waitUntilVisible(video) {
+ let videoChrome = SpecialPowers.wrap(video);
+ if (videoChrome.isInViewPort) {
+ return Promise.resolve();
+ }
+
+ return new Promise(resolve => {
+ videoChrome.addEventListener("visibilitychanged", () => {
+ if (videoChrome.isInViewPort) {
+ ok(true, `${video.token} is visible.`);
+ videoChrome.removeEventListener("visibilitychanged", this);
+ resolve();
+ }
+ });
+ });
+}
+
+/**
+ * @param {HTMLMediaElement} video Video element under test.
+ * @returns {Promise} Promise that is resolved when video 'playing' event fires.
+ */
+function waitUntilPlaying(video) {
+ var p = once(video, "playing", () => {
+ ok(true, `${video.token} played.`);
+ });
+ Log(video.token, "Start playing");
+ video.play();
+ return p;
+}
+
+/**
+ * @param {HTMLMediaElement} video Video element under test.
+ * @returns {Promise} Promise which is resolved when video 'ended' event fires.
+ */
+function waitUntilEnded(video) {
+ Log(video.token, "Waiting for ended");
+ if (video.ended) {
+ ok(true, video.token + " already ended");
+ return Promise.resolve();
+ }
+
+ return once(video, "ended", () => {
+ ok(true, `${video.token} ended`);
+ });
+}
+
+/**
+ * @param {HTMLMediaElement} video Video element under test.
+ * @returns {Promise} Promise that is resolved when video decode starts
+ * suspend timer.
+ */
+function testSuspendTimerStartedWhenHidden(video) {
+ var p = once(video, "mozstartvideosuspendtimer").then(() => {
+ ok(true, `${video.token} suspend begins`);
+ });
+ Log(video.token, "Set Hidden");
+ video.setVisible(false);
+ return p;
+}
+
+/**
+ * @param {HTMLMediaElement} video Video element under test.
+ * @returns {Promise} Promise that is resolved when video decode suspends.
+ */
+function testVideoSuspendsWhenHidden(video) {
+ let p = once(video, "mozentervideosuspend").then(() => {
+ ok(true, `${video.token} suspends`);
+ });
+ Log(video.token, "Set hidden");
+ video.setVisible(false);
+ return p;
+}
+
+/**
+ * @param {HTMLMediaElement} video Video element under test.
+ * @returns {Promise} Promise that is resolved when video decode resumes.
+ */
+function testVideoResumesWhenShown(video) {
+ var p = once(video, "mozexitvideosuspend").then(() => {
+ ok(true, `${video.token} resumes`);
+ });
+ Log(video.token, "Set visible");
+ video.setVisible(true);
+ return p;
+}
+
+/**
+ * @param {HTMLMediaElement} video Video element under test.
+ * @returns {Promise} Promise that is resolved when video decode resumes.
+ */
+function testVideoOnlySeekCompletedWhenShown(video) {
+ var p = once(video, "mozvideoonlyseekcompleted").then(() => {
+ ok(true, `${video.token} resumes`);
+ });
+ Log(video.token, "Set visible");
+ video.setVisible(true);
+ return p;
+}
+
+/**
+ * @param {HTMLVideoElement} video Video element under test.
+ * @returns {Promise} Promise that is resolved if video ends and rejects if video suspends.
+ */
+function checkVideoDoesntSuspend(video) {
+ let p = Promise.race([
+ waitUntilEnded(video).then(() => {
+ ok(true, `${video.token} ended before decode was suspended`);
+ }),
+ once(video, "mozentervideosuspend", () => {
+ Promise.reject(new Error(`${video.token} suspended`));
+ }),
+ ]);
+ Log(video.token, "Set hidden.");
+ video.setVisible(false);
+ return p;
+}
+
+/**
+ * @param {HTMLMediaElement} video Video element under test.
+ * @param {number} time video current time to wait til.
+ * @returns {Promise} Promise that is resolved once currentTime passes time.
+ */
+function waitTil(video, time) {
+ Log(video.token, `Waiting for time to reach ${time}s`);
+ return new Promise(resolve => {
+ video.addEventListener("timeupdate", function timeUpdateEvent() {
+ if (video.currentTime > time) {
+ video.removeEventListener(name, timeUpdateEvent);
+ resolve();
+ }
+ });
+ });
+}
diff --git a/dom/media/test/badtags.ogg b/dom/media/test/badtags.ogg
new file mode 100644
index 0000000000..12d8358730
--- /dev/null
+++ b/dom/media/test/badtags.ogg
Binary files differ
diff --git a/dom/media/test/badtags.ogg^headers^ b/dom/media/test/badtags.ogg^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/badtags.ogg^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bear-640x360-a_frag-cenc-key_rotation.mp4 b/dom/media/test/bear-640x360-a_frag-cenc-key_rotation.mp4
new file mode 100644
index 0000000000..dc4f197ffa
--- /dev/null
+++ b/dom/media/test/bear-640x360-a_frag-cenc-key_rotation.mp4
Binary files differ
diff --git a/dom/media/test/bear-640x360-v_frag-cenc-key_rotation.mp4 b/dom/media/test/bear-640x360-v_frag-cenc-key_rotation.mp4
new file mode 100644
index 0000000000..916c64e9ee
--- /dev/null
+++ b/dom/media/test/bear-640x360-v_frag-cenc-key_rotation.mp4
Binary files differ
diff --git a/dom/media/test/beta-phrasebook.ogg b/dom/media/test/beta-phrasebook.ogg
new file mode 100644
index 0000000000..7e6ef77ec4
--- /dev/null
+++ b/dom/media/test/beta-phrasebook.ogg
Binary files differ
diff --git a/dom/media/test/beta-phrasebook.ogg^headers^ b/dom/media/test/beta-phrasebook.ogg^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/beta-phrasebook.ogg^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/big-buck-bunny-cenc-avc3-1.m4s b/dom/media/test/big-buck-bunny-cenc-avc3-1.m4s
new file mode 100644
index 0000000000..266ec4c100
--- /dev/null
+++ b/dom/media/test/big-buck-bunny-cenc-avc3-1.m4s
Binary files differ
diff --git a/dom/media/test/big-buck-bunny-cenc-avc3-1.m4s^headers^ b/dom/media/test/big-buck-bunny-cenc-avc3-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/big-buck-bunny-cenc-avc3-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/big-buck-bunny-cenc-avc3-init.mp4 b/dom/media/test/big-buck-bunny-cenc-avc3-init.mp4
new file mode 100644
index 0000000000..7aeb3eca8a
--- /dev/null
+++ b/dom/media/test/big-buck-bunny-cenc-avc3-init.mp4
Binary files differ
diff --git a/dom/media/test/big-buck-bunny-cenc-avc3-init.mp4^headers^ b/dom/media/test/big-buck-bunny-cenc-avc3-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/big-buck-bunny-cenc-avc3-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/big-short.wav b/dom/media/test/big-short.wav
new file mode 100644
index 0000000000..c850e5fd14
--- /dev/null
+++ b/dom/media/test/big-short.wav
Binary files differ
diff --git a/dom/media/test/big-short.wav^headers^ b/dom/media/test/big-short.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/big-short.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/big.wav b/dom/media/test/big.wav
new file mode 100644
index 0000000000..5f66bc1f02
--- /dev/null
+++ b/dom/media/test/big.wav
Binary files differ
diff --git a/dom/media/test/big.wav^headers^ b/dom/media/test/big.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/big.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop-cenc-audio-key1.xml b/dom/media/test/bipbop-cenc-audio-key1.xml
new file mode 100644
index 0000000000..a1672eecef
--- /dev/null
+++ b/dom/media/test/bipbop-cenc-audio-key1.xml
@@ -0,0 +1,28 @@
+<?xml version="1.0" encoding="UTF-8"?>
+
+<!--
+ This XML file describes the encryption applied to |bipbop_<res>-cenc*|. To
+ generate the encrypted files, run bipbop-cenc.sh
+-->
+
+<GPACDRM type="CENC AES-CTR">
+
+ <DRMInfo type="pssh" version="1">
+ <!--
+ SystemID specified in
+ https://dvcs.w3.org/hg/html-media/raw-file/tip/encrypted-media/cenc-format.html
+ -->
+ <BS ID128="1077efecc0b24d02ace33c1e52e2fb4b" />
+ <!-- Number of KeyIDs = 1 -->
+ <BS bits="32" value="1" />
+ <!-- KeyID -->
+ <BS ID128="0x7e571d047e571d047e571d047e571d21" />
+ </DRMInfo>
+
+ <CrypTrack trackID="2" isEncrypted="1" IV_size="16" saiSavedBox="senc"
+ first_IV="0x00000000000000000000000000000000">
+ <key KID="0x7e571d047e571d047e571d047e571d21"
+ value="0x7e5744447e5744447e5744447e574421" />
+ </CrypTrack>
+
+</GPACDRM>
diff --git a/dom/media/test/bipbop-cenc-audio-key2.xml b/dom/media/test/bipbop-cenc-audio-key2.xml
new file mode 100644
index 0000000000..b706609052
--- /dev/null
+++ b/dom/media/test/bipbop-cenc-audio-key2.xml
@@ -0,0 +1,28 @@
+<?xml version="1.0" encoding="UTF-8"?>
+
+<!--
+ This XML file describes the encryption applied to |bipbop_<res>-cenc*|. To
+ generate the encrypted files, run bipbop-cenc.sh
+-->
+
+<GPACDRM type="CENC AES-CTR">
+
+ <DRMInfo type="pssh" version="1">
+ <!--
+ SystemID specified in
+ https://dvcs.w3.org/hg/html-media/raw-file/tip/encrypted-media/cenc-format.html
+ -->
+ <BS ID128="1077efecc0b24d02ace33c1e52e2fb4b" />
+ <!-- Number of KeyIDs = 1 -->
+ <BS bits="32" value="1" />
+ <!-- KeyID -->
+ <BS ID128="0x7e571d047e571d047e571d047e571d22" />
+ </DRMInfo>
+
+ <CrypTrack trackID="2" isEncrypted="1" IV_size="16" saiSavedBox="senc"
+ first_IV="0x00000000000000000000000000000000">
+ <key KID="0x7e571d047e571d047e571d047e571d22"
+ value="0x7e5744447e5744447e5744447e574422" />
+ </CrypTrack>
+
+</GPACDRM>
diff --git a/dom/media/test/bipbop-cenc-audio1.m4s b/dom/media/test/bipbop-cenc-audio1.m4s
new file mode 100644
index 0000000000..63cfd66f7e
--- /dev/null
+++ b/dom/media/test/bipbop-cenc-audio1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop-cenc-audio1.m4s^headers^ b/dom/media/test/bipbop-cenc-audio1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop-cenc-audio1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop-cenc-audio2.m4s b/dom/media/test/bipbop-cenc-audio2.m4s
new file mode 100644
index 0000000000..04a6cb6ff9
--- /dev/null
+++ b/dom/media/test/bipbop-cenc-audio2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop-cenc-audio2.m4s^headers^ b/dom/media/test/bipbop-cenc-audio2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop-cenc-audio2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop-cenc-audio3.m4s b/dom/media/test/bipbop-cenc-audio3.m4s
new file mode 100644
index 0000000000..ad0cd72f90
--- /dev/null
+++ b/dom/media/test/bipbop-cenc-audio3.m4s
Binary files differ
diff --git a/dom/media/test/bipbop-cenc-audio3.m4s^headers^ b/dom/media/test/bipbop-cenc-audio3.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop-cenc-audio3.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop-cenc-audioinit.mp4 b/dom/media/test/bipbop-cenc-audioinit.mp4
new file mode 100644
index 0000000000..b827aa49aa
--- /dev/null
+++ b/dom/media/test/bipbop-cenc-audioinit.mp4
Binary files differ
diff --git a/dom/media/test/bipbop-cenc-audioinit.mp4^headers^ b/dom/media/test/bipbop-cenc-audioinit.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop-cenc-audioinit.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop-cenc-video-10s.mp4 b/dom/media/test/bipbop-cenc-video-10s.mp4
new file mode 100644
index 0000000000..abbe4561fd
--- /dev/null
+++ b/dom/media/test/bipbop-cenc-video-10s.mp4
Binary files differ
diff --git a/dom/media/test/bipbop-cenc-video-10s.mp4^headers^ b/dom/media/test/bipbop-cenc-video-10s.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop-cenc-video-10s.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop-cenc-video-key1.xml b/dom/media/test/bipbop-cenc-video-key1.xml
new file mode 100644
index 0000000000..f0d9878fa2
--- /dev/null
+++ b/dom/media/test/bipbop-cenc-video-key1.xml
@@ -0,0 +1,28 @@
+<?xml version="1.0" encoding="UTF-8"?>
+
+<!--
+ This XML file describes the encryption applied to |bipbop_<res>-cenc*|. To
+ generate the encrypted files, run bipbop-cenc.sh
+-->
+
+<GPACDRM type="CENC AES-CTR">
+
+ <DRMInfo type="pssh" version="1">
+ <!--
+ SystemID specified in
+ https://dvcs.w3.org/hg/html-media/raw-file/tip/encrypted-media/cenc-format.html
+ -->
+ <BS ID128="1077efecc0b24d02ace33c1e52e2fb4b" />
+ <!-- Number of KeyIDs = 1 -->
+ <BS bits="32" value="1" />
+ <!-- KeyID -->
+ <BS ID128="0x7e571d037e571d037e571d037e571d11" />
+ </DRMInfo>
+
+ <CrypTrack trackID="1" isEncrypted="1" IV_size="16" saiSavedBox="senc"
+ first_IV="0x00000000000000000000000000000000">
+ <key KID="0x7e571d037e571d037e571d037e571d11"
+ value="0x7e5733337e5733337e5733337e573311" />
+ </CrypTrack>
+
+</GPACDRM>
diff --git a/dom/media/test/bipbop-cenc-video-key2.xml b/dom/media/test/bipbop-cenc-video-key2.xml
new file mode 100644
index 0000000000..1f320e6336
--- /dev/null
+++ b/dom/media/test/bipbop-cenc-video-key2.xml
@@ -0,0 +1,28 @@
+<?xml version="1.0" encoding="UTF-8"?>
+
+<!--
+ This XML file describes the encryption applied to |bipbop_<res>-cenc*|. To
+ generate the encrypted files, run bipbop-cenc.sh
+-->
+
+<GPACDRM type="CENC AES-CTR">
+
+ <DRMInfo type="pssh" version="1">
+ <!--
+ SystemID specified in
+ https://dvcs.w3.org/hg/html-media/raw-file/tip/encrypted-media/cenc-format.html
+ -->
+ <BS ID128="1077efecc0b24d02ace33c1e52e2fb4b" />
+ <!-- Number of KeyIDs = 1 -->
+ <BS bits="32" value="1" />
+ <!-- KeyID -->
+ <BS ID128="0x7e571d037e571d037e571d037e571d12" />
+ </DRMInfo>
+
+ <CrypTrack trackID="1" isEncrypted="1" IV_size="16" saiSavedBox="senc"
+ first_IV="0x00000000000000000000000000000000">
+ <key KID="0x7e571d037e571d037e571d037e571d12"
+ value="0x7e5733337e5733337e5733337e573312" />
+ </CrypTrack>
+
+</GPACDRM>
diff --git a/dom/media/test/bipbop-cenc-video1.m4s b/dom/media/test/bipbop-cenc-video1.m4s
new file mode 100644
index 0000000000..755013c11c
--- /dev/null
+++ b/dom/media/test/bipbop-cenc-video1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop-cenc-video1.m4s^headers^ b/dom/media/test/bipbop-cenc-video1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop-cenc-video1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop-cenc-video2.m4s b/dom/media/test/bipbop-cenc-video2.m4s
new file mode 100644
index 0000000000..c884bd95fc
--- /dev/null
+++ b/dom/media/test/bipbop-cenc-video2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop-cenc-video2.m4s^headers^ b/dom/media/test/bipbop-cenc-video2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop-cenc-video2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop-cenc-videoinit.mp4 b/dom/media/test/bipbop-cenc-videoinit.mp4
new file mode 100644
index 0000000000..aa87d0bbe6
--- /dev/null
+++ b/dom/media/test/bipbop-cenc-videoinit.mp4
Binary files differ
diff --git a/dom/media/test/bipbop-cenc-videoinit.mp4^headers^ b/dom/media/test/bipbop-cenc-videoinit.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop-cenc-videoinit.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop-cenc.sh b/dom/media/test/bipbop-cenc.sh
new file mode 100644
index 0000000000..a00c38ae80
--- /dev/null
+++ b/dom/media/test/bipbop-cenc.sh
@@ -0,0 +1,29 @@
+mkdir work.tmp
+
+for r in 225w_175kbps 300_215kbps 300wp_227kbps 360w_253kbps 480_624kbps 480wp_663kbps 480_959kbps 480wp_1001kbps
+do
+ for k in 1 2
+ do
+ # Encrypt bipbop_<res>.mp4 with the keys specified in this file,
+ # and output to |bipbop_<res>-cenc-{video,audio}.mp4|
+ MP4Box -crypt bipbop-cenc-audio-key${k}.xml -rem 1 -out work.tmp/bipbop_${r}-cenc-audio-key${k}.mp4 bipbop_${r}.mp4
+ MP4Box -crypt bipbop-cenc-video-key${k}.xml -rem 2 -out work.tmp/bipbop_${r}-cenc-video-key${k}.mp4 bipbop_${r}.mp4
+
+ # Fragment |bipbop_<res>-cenc-*.mp4| into 500ms segments:
+ MP4Box -dash 500 -rap -segment-name work.tmp/bipbop_${r}-cenc-audio-key${k}- -subsegs-per-sidx 5 work.tmp/bipbop_${r}-cenc-audio-key${k}.mp4
+ MP4Box -dash 500 -rap -segment-name work.tmp/bipbop_${r}-cenc-video-key${k}- -subsegs-per-sidx 5 work.tmp/bipbop_${r}-cenc-video-key${k}.mp4
+
+ # The above command will generate a set of fragments |bipbop_<res>-cenc-{video,audio}-*.m4s
+ # and |bipbop_<res>-cenc-{video,audio}-init.mp4| containing just the init segment.
+
+ # Remove unneeded mpd files.
+ rm bipbop_${r}-cenc-{audio,video}-key${k}_dash.mpd
+ done
+done
+
+# Only keep the first 4 audio & 2 video segments:
+cp work.tmp/*-init[.]mp4 ./
+cp work.tmp/*audio*-[1234][.]m4s ./
+cp work.tmp/*video*-[12][.]m4s ./
+
+rm -Rf work.tmp
diff --git a/dom/media/test/bipbop-clearkey-keyrotation-clear-lead-audio.mp4 b/dom/media/test/bipbop-clearkey-keyrotation-clear-lead-audio.mp4
new file mode 100644
index 0000000000..5e5e30c255
--- /dev/null
+++ b/dom/media/test/bipbop-clearkey-keyrotation-clear-lead-audio.mp4
Binary files differ
diff --git a/dom/media/test/bipbop-clearkey-keyrotation-clear-lead-audio.mp4^headers^ b/dom/media/test/bipbop-clearkey-keyrotation-clear-lead-audio.mp4^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop-clearkey-keyrotation-clear-lead-audio.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop-clearkey-keyrotation-clear-lead-video.mp4 b/dom/media/test/bipbop-clearkey-keyrotation-clear-lead-video.mp4
new file mode 100644
index 0000000000..447c657475
--- /dev/null
+++ b/dom/media/test/bipbop-clearkey-keyrotation-clear-lead-video.mp4
Binary files differ
diff --git a/dom/media/test/bipbop-clearkey-keyrotation-clear-lead-video.mp4^headers^ b/dom/media/test/bipbop-clearkey-keyrotation-clear-lead-video.mp4^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop-clearkey-keyrotation-clear-lead-video.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop-frag-cenc.xml b/dom/media/test/bipbop-frag-cenc.xml
new file mode 100644
index 0000000000..6f6a4d90a9
--- /dev/null
+++ b/dom/media/test/bipbop-frag-cenc.xml
@@ -0,0 +1,57 @@
+<?xml version="1.0" encoding="UTF-8"?>
+
+<!--
+ This XML file describes the encryption applied to |bipbop-cenc*|. To
+ generate the bipbop-cenc files, run the following commands:
+
+ Encrypt bipbop-no-edts.mp4 with the keys specified in this file,
+ and output to |bipbop-cenc-{video,audio}.mp4|
+ MP4Box -crypt bipbop-frag-cenc.xml -rem 2 -out bipbop-cenc-video.mp4 bipbop-no-edts.mp4
+ MP4Box -crypt bipbop-frag-cenc.xml -rem 1 -out bipbop-cenc-audio.mp4 bipbop-no-edts.mp4
+
+ Fragment |bipbop-cenc-*.mp4| into 500ms segments:
+ MP4Box -dash 500 -rap -segment-name bipbop-cenc-video -subsegs-per-sidx 5 bipbop-cenc-video.mp4
+ MP4Box -dash 500 -rap -segment-name bipbop-cenc-audio -subsegs-per-sidx 5 bipbop-cenc-audio.mp4
+
+ The above command will generate a set of fragments in |bipbop-cenc-{video,audio}*.m4s
+ and |bipbop-cenc-{video,audio}init.mp4| containing just the init segment.
+
+ To cut down the duration, we throw out all but the first 3 audio & 2 video segments:
+ rm bipbop-cenc-audio{[^123],[123][^.]}.m4s
+ rm bipbop-cenc-video{[^12],[12][^.]}.m4s
+
+ MP4Box will also have generated some *.mpd files we don't need:
+ rm bipbop-cenc-*.mpd
+
+ Delete intermediate encrypted files:
+ rm bipbop-cenc-{audio,video}.mp4
+-->
+
+<GPACDRM type="CENC AES-CTR">
+
+ <DRMInfo type="pssh" version="1">
+ <!--
+ SystemID specified in
+ https://dvcs.w3.org/hg/html-media/raw-file/tip/encrypted-media/cenc-format.html
+ -->
+ <BS ID128="1077efecc0b24d02ace33c1e52e2fb4b" />
+ <!-- Number of KeyIDs = 2 -->
+ <BS bits="32" value="2" />
+ <!-- KeyID -->
+ <BS ID128="0x7e571d037e571d037e571d037e571d03" />
+ <BS ID128="0x7e571d047e571d047e571d047e571d04" />
+ </DRMInfo>
+
+ <CrypTrack trackID="1" isEncrypted="1" IV_size="16" saiSavedBox="senc"
+ first_IV="0x00000000000000000000000000000000">
+ <key KID="0x7e571d037e571d037e571d037e571d03"
+ value="0x7e5733337e5733337e5733337e573333" />
+ </CrypTrack>
+
+ <CrypTrack trackID="2" isEncrypted="1" IV_size="16" saiSavedBox="senc"
+ first_IV="0x00000000000000000000000000000000">
+ <key KID="0x7e571d047e571d047e571d047e571d04"
+ value="0x7e5744447e5744447e5744447e574444" />
+ </CrypTrack>
+
+</GPACDRM>
diff --git a/dom/media/test/bipbop-lateaudio.mp4 b/dom/media/test/bipbop-lateaudio.mp4
new file mode 100644
index 0000000000..5b4cc57095
--- /dev/null
+++ b/dom/media/test/bipbop-lateaudio.mp4
Binary files differ
diff --git a/dom/media/test/bipbop-lateaudio.mp4^headers^ b/dom/media/test/bipbop-lateaudio.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop-lateaudio.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop-no-edts.mp4 b/dom/media/test/bipbop-no-edts.mp4
new file mode 100644
index 0000000000..63435887df
--- /dev/null
+++ b/dom/media/test/bipbop-no-edts.mp4
Binary files differ
diff --git a/dom/media/test/bipbop.mp4 b/dom/media/test/bipbop.mp4
new file mode 100644
index 0000000000..017d658f31
--- /dev/null
+++ b/dom/media/test/bipbop.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-1.m4s b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-1.m4s
new file mode 100644
index 0000000000..e2bd754c7e
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-1.m4s^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-2.m4s b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-2.m4s
new file mode 100644
index 0000000000..347835feee
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-2.m4s^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-3.m4s b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-3.m4s
new file mode 100644
index 0000000000..64b0da69a0
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-3.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-3.m4s^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-3.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-3.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-4.m4s b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-4.m4s
new file mode 100644
index 0000000000..864f4248af
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-4.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-4.m4s^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-4.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-4.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-init.mp4 b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-init.mp4
new file mode 100644
index 0000000000..40c3a7bb98
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-init.mp4^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key1-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-1.m4s b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-1.m4s
new file mode 100644
index 0000000000..a8896e069a
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-1.m4s^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-2.m4s b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-2.m4s
new file mode 100644
index 0000000000..0f0a35ce79
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-2.m4s^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-3.m4s b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-3.m4s
new file mode 100644
index 0000000000..fece52ff42
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-3.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-3.m4s^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-3.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-3.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-4.m4s b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-4.m4s
new file mode 100644
index 0000000000..70e61e3d5f
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-4.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-4.m4s^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-4.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-4.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-init.mp4 b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-init.mp4
new file mode 100644
index 0000000000..986e5fb186
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-init.mp4^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-audio-key2-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-video-key1-1.m4s b/dom/media/test/bipbop_225w_175kbps-cenc-video-key1-1.m4s
new file mode 100644
index 0000000000..547950e516
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-video-key1-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-video-key1-1.m4s^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-video-key1-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-video-key1-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-video-key1-init.mp4 b/dom/media/test/bipbop_225w_175kbps-cenc-video-key1-init.mp4
new file mode 100644
index 0000000000..3214f131d4
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-video-key1-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-video-key1-init.mp4^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-video-key1-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-video-key1-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-video-key2-1.m4s b/dom/media/test/bipbop_225w_175kbps-cenc-video-key2-1.m4s
new file mode 100644
index 0000000000..08713078d9
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-video-key2-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-video-key2-1.m4s^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-video-key2-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-video-key2-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-video-key2-init.mp4 b/dom/media/test/bipbop_225w_175kbps-cenc-video-key2-init.mp4
new file mode 100644
index 0000000000..0b13fed5f0
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-video-key2-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_225w_175kbps-cenc-video-key2-init.mp4^headers^ b/dom/media/test/bipbop_225w_175kbps-cenc-video-key2-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps-cenc-video-key2-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_225w_175kbps.mp4 b/dom/media/test/bipbop_225w_175kbps.mp4
new file mode 100644
index 0000000000..abe37b9f9d
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_225w_175kbps.mp4^headers^ b/dom/media/test/bipbop_225w_175kbps.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_225w_175kbps.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-1.m4s b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-1.m4s
new file mode 100644
index 0000000000..e2bd754c7e
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-1.m4s^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-2.m4s b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-2.m4s
new file mode 100644
index 0000000000..347835feee
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-2.m4s^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-3.m4s b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-3.m4s
new file mode 100644
index 0000000000..64b0da69a0
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-3.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-3.m4s^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-3.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-3.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-4.m4s b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-4.m4s
new file mode 100644
index 0000000000..864f4248af
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-4.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-4.m4s^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-4.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-4.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-init.mp4 b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-init.mp4
new file mode 100644
index 0000000000..21f3863274
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-init.mp4^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key1-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-1.m4s b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-1.m4s
new file mode 100644
index 0000000000..a8896e069a
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-1.m4s^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-2.m4s b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-2.m4s
new file mode 100644
index 0000000000..0f0a35ce79
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-2.m4s^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-3.m4s b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-3.m4s
new file mode 100644
index 0000000000..fece52ff42
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-3.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-3.m4s^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-3.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-3.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-4.m4s b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-4.m4s
new file mode 100644
index 0000000000..70e61e3d5f
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-4.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-4.m4s^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-4.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-4.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-init.mp4 b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-init.mp4
new file mode 100644
index 0000000000..bc741cdf86
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-init.mp4^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-audio-key2-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-video-key1-1.m4s b/dom/media/test/bipbop_300_215kbps-cenc-video-key1-1.m4s
new file mode 100644
index 0000000000..9c6818d06f
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-video-key1-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-video-key1-1.m4s^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-video-key1-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-video-key1-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-video-key1-2.m4s b/dom/media/test/bipbop_300_215kbps-cenc-video-key1-2.m4s
new file mode 100644
index 0000000000..f327aaa573
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-video-key1-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-video-key1-2.m4s^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-video-key1-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-video-key1-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-video-key1-init.mp4 b/dom/media/test/bipbop_300_215kbps-cenc-video-key1-init.mp4
new file mode 100644
index 0000000000..543f18c24b
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-video-key1-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-video-key1-init.mp4^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-video-key1-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-video-key1-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-video-key2-1.m4s b/dom/media/test/bipbop_300_215kbps-cenc-video-key2-1.m4s
new file mode 100644
index 0000000000..f850ceaf0a
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-video-key2-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-video-key2-1.m4s^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-video-key2-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-video-key2-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-video-key2-2.m4s b/dom/media/test/bipbop_300_215kbps-cenc-video-key2-2.m4s
new file mode 100644
index 0000000000..a28a106daf
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-video-key2-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-video-key2-2.m4s^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-video-key2-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-video-key2-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-video-key2-init.mp4 b/dom/media/test/bipbop_300_215kbps-cenc-video-key2-init.mp4
new file mode 100644
index 0000000000..a05a879970
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-video-key2-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_300_215kbps-cenc-video-key2-init.mp4^headers^ b/dom/media/test/bipbop_300_215kbps-cenc-video-key2-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps-cenc-video-key2-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300_215kbps.mp4 b/dom/media/test/bipbop_300_215kbps.mp4
new file mode 100644
index 0000000000..084d477430
--- /dev/null
+++ b/dom/media/test/bipbop_300_215kbps.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-1.m4s b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-1.m4s
new file mode 100644
index 0000000000..e2bd754c7e
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-1.m4s^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-2.m4s b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-2.m4s
new file mode 100644
index 0000000000..347835feee
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-2.m4s^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-3.m4s b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-3.m4s
new file mode 100644
index 0000000000..64b0da69a0
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-3.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-3.m4s^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-3.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-3.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-4.m4s b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-4.m4s
new file mode 100644
index 0000000000..864f4248af
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-4.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-4.m4s^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-4.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-4.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-init.mp4 b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-init.mp4
new file mode 100644
index 0000000000..40c3a7bb98
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-init.mp4^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key1-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-1.m4s b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-1.m4s
new file mode 100644
index 0000000000..a8896e069a
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-1.m4s^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-2.m4s b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-2.m4s
new file mode 100644
index 0000000000..0f0a35ce79
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-2.m4s^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-3.m4s b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-3.m4s
new file mode 100644
index 0000000000..fece52ff42
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-3.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-3.m4s^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-3.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-3.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-4.m4s b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-4.m4s
new file mode 100644
index 0000000000..70e61e3d5f
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-4.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-4.m4s^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-4.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-4.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-init.mp4 b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-init.mp4
new file mode 100644
index 0000000000..986e5fb186
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-init.mp4^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-audio-key2-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-1.m4s b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-1.m4s
new file mode 100644
index 0000000000..9c6818d06f
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-1.m4s^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-2.m4s b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-2.m4s
new file mode 100644
index 0000000000..ce2e64eb33
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-2.m4s^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-init.mp4 b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-init.mp4
new file mode 100644
index 0000000000..8592a5b0a3
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-init.mp4^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key1-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-1.m4s b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-1.m4s
new file mode 100644
index 0000000000..f850ceaf0a
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-1.m4s^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-2.m4s b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-2.m4s
new file mode 100644
index 0000000000..d07ce9753e
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-2.m4s^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-init.mp4 b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-init.mp4
new file mode 100644
index 0000000000..9d2fa23dd4
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-init.mp4^headers^ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps-cenc-video-key2-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_300wp_227kbps.mp4 b/dom/media/test/bipbop_300wp_227kbps.mp4
new file mode 100644
index 0000000000..1499355313
--- /dev/null
+++ b/dom/media/test/bipbop_300wp_227kbps.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-1.m4s b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-1.m4s
new file mode 100644
index 0000000000..e2bd754c7e
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-1.m4s^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-2.m4s b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-2.m4s
new file mode 100644
index 0000000000..347835feee
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-2.m4s^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-3.m4s b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-3.m4s
new file mode 100644
index 0000000000..64b0da69a0
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-3.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-3.m4s^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-3.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-3.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-4.m4s b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-4.m4s
new file mode 100644
index 0000000000..864f4248af
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-4.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-4.m4s^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-4.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-4.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-init.mp4 b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-init.mp4
new file mode 100644
index 0000000000..40c3a7bb98
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-init.mp4^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key1-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-1.m4s b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-1.m4s
new file mode 100644
index 0000000000..a8896e069a
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-1.m4s^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-2.m4s b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-2.m4s
new file mode 100644
index 0000000000..0f0a35ce79
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-2.m4s^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-3.m4s b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-3.m4s
new file mode 100644
index 0000000000..fece52ff42
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-3.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-3.m4s^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-3.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-3.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-4.m4s b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-4.m4s
new file mode 100644
index 0000000000..70e61e3d5f
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-4.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-4.m4s^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-4.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-4.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-init.mp4 b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-init.mp4
new file mode 100644
index 0000000000..986e5fb186
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-init.mp4^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-audio-key2-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-video-key1-1.m4s b/dom/media/test/bipbop_360w_253kbps-cenc-video-key1-1.m4s
new file mode 100644
index 0000000000..a571d47cfb
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-video-key1-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-video-key1-1.m4s^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-video-key1-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-video-key1-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-video-key1-init.mp4 b/dom/media/test/bipbop_360w_253kbps-cenc-video-key1-init.mp4
new file mode 100644
index 0000000000..42dbfec1ed
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-video-key1-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-video-key1-init.mp4^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-video-key1-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-video-key1-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-video-key2-1.m4s b/dom/media/test/bipbop_360w_253kbps-cenc-video-key2-1.m4s
new file mode 100644
index 0000000000..9e4224cac8
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-video-key2-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-video-key2-1.m4s^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-video-key2-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-video-key2-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-video-key2-init.mp4 b/dom/media/test/bipbop_360w_253kbps-cenc-video-key2-init.mp4
new file mode 100644
index 0000000000..21763ecbdd
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-video-key2-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_360w_253kbps-cenc-video-key2-init.mp4^headers^ b/dom/media/test/bipbop_360w_253kbps-cenc-video-key2-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-cenc-video-key2-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_360w_253kbps-clearkey-audio.webm b/dom/media/test/bipbop_360w_253kbps-clearkey-audio.webm
new file mode 100644
index 0000000000..4be8f340c3
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-clearkey-audio.webm
Binary files differ
diff --git a/dom/media/test/bipbop_360w_253kbps-clearkey-audio.webm^headers^ b/dom/media/test/bipbop_360w_253kbps-clearkey-audio.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-clearkey-audio.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_360w_253kbps-clearkey-video-vp8.webm b/dom/media/test/bipbop_360w_253kbps-clearkey-video-vp8.webm
new file mode 100644
index 0000000000..56cf4c483c
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-clearkey-video-vp8.webm
Binary files differ
diff --git a/dom/media/test/bipbop_360w_253kbps-clearkey-video-vp8.webm^headers^ b/dom/media/test/bipbop_360w_253kbps-clearkey-video-vp8.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-clearkey-video-vp8.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_360w_253kbps-clearkey-video-vp9.webm b/dom/media/test/bipbop_360w_253kbps-clearkey-video-vp9.webm
new file mode 100644
index 0000000000..9f411d0e34
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-clearkey-video-vp9.webm
Binary files differ
diff --git a/dom/media/test/bipbop_360w_253kbps-clearkey-video-vp9.webm^headers^ b/dom/media/test/bipbop_360w_253kbps-clearkey-video-vp9.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps-clearkey-video-vp9.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_360w_253kbps.mp4 b/dom/media/test/bipbop_360w_253kbps.mp4
new file mode 100644
index 0000000000..6c796f4e1f
--- /dev/null
+++ b/dom/media/test/bipbop_360w_253kbps.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-1.m4s b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-1.m4s
new file mode 100644
index 0000000000..e2bd754c7e
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-1.m4s^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-2.m4s b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-2.m4s
new file mode 100644
index 0000000000..347835feee
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-2.m4s^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-3.m4s b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-3.m4s
new file mode 100644
index 0000000000..64b0da69a0
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-3.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-3.m4s^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-3.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-3.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-4.m4s b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-4.m4s
new file mode 100644
index 0000000000..864f4248af
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-4.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-4.m4s^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-4.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-4.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-init.mp4 b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-init.mp4
new file mode 100644
index 0000000000..e626fa4564
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-init.mp4^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key1-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-1.m4s b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-1.m4s
new file mode 100644
index 0000000000..a8896e069a
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-1.m4s^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-2.m4s b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-2.m4s
new file mode 100644
index 0000000000..0f0a35ce79
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-2.m4s^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-3.m4s b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-3.m4s
new file mode 100644
index 0000000000..fece52ff42
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-3.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-3.m4s^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-3.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-3.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-4.m4s b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-4.m4s
new file mode 100644
index 0000000000..70e61e3d5f
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-4.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-4.m4s^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-4.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-4.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-init.mp4 b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-init.mp4
new file mode 100644
index 0000000000..d7cbb2b6b0
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-init.mp4^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-audio-key2-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-video-key1-1.m4s b/dom/media/test/bipbop_480_624kbps-cenc-video-key1-1.m4s
new file mode 100644
index 0000000000..805f4bbf3f
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-video-key1-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-video-key1-1.m4s^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-video-key1-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-video-key1-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-video-key1-2.m4s b/dom/media/test/bipbop_480_624kbps-cenc-video-key1-2.m4s
new file mode 100644
index 0000000000..5bf9994733
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-video-key1-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-video-key1-2.m4s^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-video-key1-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-video-key1-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-video-key1-init.mp4 b/dom/media/test/bipbop_480_624kbps-cenc-video-key1-init.mp4
new file mode 100644
index 0000000000..77c7daba5a
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-video-key1-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-video-key1-init.mp4^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-video-key1-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-video-key1-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-video-key2-1.m4s b/dom/media/test/bipbop_480_624kbps-cenc-video-key2-1.m4s
new file mode 100644
index 0000000000..c5127beec9
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-video-key2-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-video-key2-1.m4s^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-video-key2-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-video-key2-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-video-key2-2.m4s b/dom/media/test/bipbop_480_624kbps-cenc-video-key2-2.m4s
new file mode 100644
index 0000000000..b0ff51f74a
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-video-key2-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-video-key2-2.m4s^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-video-key2-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-video-key2-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-video-key2-init.mp4 b/dom/media/test/bipbop_480_624kbps-cenc-video-key2-init.mp4
new file mode 100644
index 0000000000..cfa099c043
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-video-key2-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_480_624kbps-cenc-video-key2-init.mp4^headers^ b/dom/media/test/bipbop_480_624kbps-cenc-video-key2-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps-cenc-video-key2-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_624kbps.mp4 b/dom/media/test/bipbop_480_624kbps.mp4
new file mode 100644
index 0000000000..27928b85f4
--- /dev/null
+++ b/dom/media/test/bipbop_480_624kbps.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-1.m4s b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-1.m4s
new file mode 100644
index 0000000000..e2bd754c7e
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-1.m4s^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-2.m4s b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-2.m4s
new file mode 100644
index 0000000000..347835feee
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-2.m4s^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-3.m4s b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-3.m4s
new file mode 100644
index 0000000000..64b0da69a0
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-3.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-3.m4s^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-3.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-3.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-4.m4s b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-4.m4s
new file mode 100644
index 0000000000..864f4248af
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-4.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-4.m4s^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-4.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-4.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-init.mp4 b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-init.mp4
new file mode 100644
index 0000000000..c9106aad99
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-init.mp4^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key1-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-1.m4s b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-1.m4s
new file mode 100644
index 0000000000..a8896e069a
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-1.m4s^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-2.m4s b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-2.m4s
new file mode 100644
index 0000000000..0f0a35ce79
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-2.m4s^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-3.m4s b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-3.m4s
new file mode 100644
index 0000000000..fece52ff42
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-3.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-3.m4s^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-3.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-3.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-4.m4s b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-4.m4s
new file mode 100644
index 0000000000..70e61e3d5f
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-4.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-4.m4s^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-4.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-4.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-init.mp4 b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-init.mp4
new file mode 100644
index 0000000000..888b20ab63
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-init.mp4^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-audio-key2-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-video-key1-1.m4s b/dom/media/test/bipbop_480_959kbps-cenc-video-key1-1.m4s
new file mode 100644
index 0000000000..796ad13670
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-video-key1-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-video-key1-1.m4s^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-video-key1-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-video-key1-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-video-key1-2.m4s b/dom/media/test/bipbop_480_959kbps-cenc-video-key1-2.m4s
new file mode 100644
index 0000000000..d02be53198
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-video-key1-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-video-key1-2.m4s^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-video-key1-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-video-key1-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-video-key1-init.mp4 b/dom/media/test/bipbop_480_959kbps-cenc-video-key1-init.mp4
new file mode 100644
index 0000000000..6e0c60f986
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-video-key1-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-video-key1-init.mp4^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-video-key1-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-video-key1-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-video-key2-1.m4s b/dom/media/test/bipbop_480_959kbps-cenc-video-key2-1.m4s
new file mode 100644
index 0000000000..06778e6f2b
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-video-key2-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-video-key2-1.m4s^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-video-key2-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-video-key2-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-video-key2-2.m4s b/dom/media/test/bipbop_480_959kbps-cenc-video-key2-2.m4s
new file mode 100644
index 0000000000..4c1c603e8d
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-video-key2-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-video-key2-2.m4s^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-video-key2-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-video-key2-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-video-key2-init.mp4 b/dom/media/test/bipbop_480_959kbps-cenc-video-key2-init.mp4
new file mode 100644
index 0000000000..f4a98eca97
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-video-key2-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_480_959kbps-cenc-video-key2-init.mp4^headers^ b/dom/media/test/bipbop_480_959kbps-cenc-video-key2-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps-cenc-video-key2-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480_959kbps.mp4 b/dom/media/test/bipbop_480_959kbps.mp4
new file mode 100644
index 0000000000..4a9f2ee823
--- /dev/null
+++ b/dom/media/test/bipbop_480_959kbps.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s
new file mode 100644
index 0000000000..e2bd754c7e
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s
new file mode 100644
index 0000000000..347835feee
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s
new file mode 100644
index 0000000000..64b0da69a0
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s
new file mode 100644
index 0000000000..864f4248af
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4 b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4
new file mode 100644
index 0000000000..416bc7a7ca
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s
new file mode 100644
index 0000000000..a8896e069a
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s
new file mode 100644
index 0000000000..0f0a35ce79
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s
new file mode 100644
index 0000000000..fece52ff42
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s
new file mode 100644
index 0000000000..70e61e3d5f
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4 b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4
new file mode 100644
index 0000000000..73d542cfe0
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-1.m4s b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-1.m4s
new file mode 100644
index 0000000000..796ad13670
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-1.m4s^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-2.m4s b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-2.m4s
new file mode 100644
index 0000000000..80824e9ffc
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-2.m4s^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-init.mp4 b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-init.mp4
new file mode 100644
index 0000000000..5db21d091b
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-init.mp4^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key1-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-1.m4s b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-1.m4s
new file mode 100644
index 0000000000..06778e6f2b
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-1.m4s^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-2.m4s b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-2.m4s
new file mode 100644
index 0000000000..38a081187a
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-2.m4s^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-init.mp4 b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-init.mp4
new file mode 100644
index 0000000000..bc8bddf505
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-init.mp4^headers^ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps-cenc-video-key2-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_1001kbps.mp4 b/dom/media/test/bipbop_480wp_1001kbps.mp4
new file mode 100644
index 0000000000..600376cf83
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_1001kbps.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-1.m4s b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-1.m4s
new file mode 100644
index 0000000000..e2bd754c7e
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-1.m4s^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-2.m4s b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-2.m4s
new file mode 100644
index 0000000000..347835feee
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-2.m4s^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-3.m4s b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-3.m4s
new file mode 100644
index 0000000000..64b0da69a0
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-3.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-3.m4s^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-3.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-3.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-4.m4s b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-4.m4s
new file mode 100644
index 0000000000..864f4248af
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-4.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-4.m4s^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-4.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-4.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-init.mp4 b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-init.mp4
new file mode 100644
index 0000000000..416bc7a7ca
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-init.mp4^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key1-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-1.m4s b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-1.m4s
new file mode 100644
index 0000000000..a8896e069a
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-1.m4s^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-2.m4s b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-2.m4s
new file mode 100644
index 0000000000..0f0a35ce79
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-2.m4s^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-3.m4s b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-3.m4s
new file mode 100644
index 0000000000..fece52ff42
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-3.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-3.m4s^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-3.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-3.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-4.m4s b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-4.m4s
new file mode 100644
index 0000000000..70e61e3d5f
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-4.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-4.m4s^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-4.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-4.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-init.mp4 b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-init.mp4
new file mode 100644
index 0000000000..73d542cfe0
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-init.mp4^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-audio-key2-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-1.m4s b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-1.m4s
new file mode 100644
index 0000000000..805f4bbf3f
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-1.m4s^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-2.m4s b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-2.m4s
new file mode 100644
index 0000000000..0a40d1cb73
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-2.m4s^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-init.mp4 b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-init.mp4
new file mode 100644
index 0000000000..5db21d091b
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-init.mp4^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key1-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-1.m4s b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-1.m4s
new file mode 100644
index 0000000000..c5127beec9
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-1.m4s^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-1.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-2.m4s b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-2.m4s
new file mode 100644
index 0000000000..3f344022a4
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-2.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-2.m4s^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-2.m4s^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-2.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-init.mp4 b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-init.mp4
new file mode 100644
index 0000000000..bc8bddf505
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-init.mp4^headers^ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-init.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps-cenc-video-key2-init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_480wp_663kbps.mp4 b/dom/media/test/bipbop_480wp_663kbps.mp4
new file mode 100644
index 0000000000..3cc1da69d2
--- /dev/null
+++ b/dom/media/test/bipbop_480wp_663kbps.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_audio_aac_22.05k.mp4 b/dom/media/test/bipbop_audio_aac_22.05k.mp4
new file mode 100644
index 0000000000..e89ba40a10
--- /dev/null
+++ b/dom/media/test/bipbop_audio_aac_22.05k.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_audio_aac_22.05k.mp4^headers^ b/dom/media/test/bipbop_audio_aac_22.05k.mp4^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_audio_aac_22.05k.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_audio_aac_44.1k.mp4 b/dom/media/test/bipbop_audio_aac_44.1k.mp4
new file mode 100644
index 0000000000..9a13333c12
--- /dev/null
+++ b/dom/media/test/bipbop_audio_aac_44.1k.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_audio_aac_44.1k.mp4^headers^ b/dom/media/test/bipbop_audio_aac_44.1k.mp4^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_audio_aac_44.1k.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_audio_aac_48k.mp4 b/dom/media/test/bipbop_audio_aac_48k.mp4
new file mode 100644
index 0000000000..0224350449
--- /dev/null
+++ b/dom/media/test/bipbop_audio_aac_48k.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_audio_aac_48k.mp4^headers^ b/dom/media/test/bipbop_audio_aac_48k.mp4^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_audio_aac_48k.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_audio_aac_88.2k.mp4 b/dom/media/test/bipbop_audio_aac_88.2k.mp4
new file mode 100644
index 0000000000..a653be973f
--- /dev/null
+++ b/dom/media/test/bipbop_audio_aac_88.2k.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_audio_aac_88.2k.mp4^headers^ b/dom/media/test/bipbop_audio_aac_88.2k.mp4^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_audio_aac_88.2k.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_audio_aac_8k.mp4 b/dom/media/test/bipbop_audio_aac_8k.mp4
new file mode 100644
index 0000000000..fb704090d6
--- /dev/null
+++ b/dom/media/test/bipbop_audio_aac_8k.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_audio_aac_8k.mp4^headers^ b/dom/media/test/bipbop_audio_aac_8k.mp4^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_audio_aac_8k.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_audio_aac_96k.mp4 b/dom/media/test/bipbop_audio_aac_96k.mp4
new file mode 100644
index 0000000000..30579f59a4
--- /dev/null
+++ b/dom/media/test/bipbop_audio_aac_96k.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_audio_aac_96k.mp4^headers^ b/dom/media/test/bipbop_audio_aac_96k.mp4^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_audio_aac_96k.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_cbcs_10_0_audio_1.m4s b/dom/media/test/bipbop_cbcs_10_0_audio_1.m4s
new file mode 100644
index 0000000000..98ff850161
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_10_0_audio_1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_cbcs_10_0_audio_1.m4s^headers^ b/dom/media/test/bipbop_cbcs_10_0_audio_1.m4s^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_10_0_audio_1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_cbcs_10_0_audio_init.mp4 b/dom/media/test/bipbop_cbcs_10_0_audio_init.mp4
new file mode 100644
index 0000000000..ef462e0d66
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_10_0_audio_init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_cbcs_10_0_audio_init.mp4^headers^ b/dom/media/test/bipbop_cbcs_10_0_audio_init.mp4^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_10_0_audio_init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_cbcs_10_0_video_1.m4s b/dom/media/test/bipbop_cbcs_10_0_video_1.m4s
new file mode 100644
index 0000000000..73a90b9a5b
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_10_0_video_1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_cbcs_10_0_video_1.m4s^headers^ b/dom/media/test/bipbop_cbcs_10_0_video_1.m4s^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_10_0_video_1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_cbcs_10_0_video_init.mp4 b/dom/media/test/bipbop_cbcs_10_0_video_init.mp4
new file mode 100644
index 0000000000..266e846e9b
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_10_0_video_init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_cbcs_10_0_video_init.mp4^headers^ b/dom/media/test/bipbop_cbcs_10_0_video_init.mp4^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_10_0_video_init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_cbcs_1_9_audio_1.m4s b/dom/media/test/bipbop_cbcs_1_9_audio_1.m4s
new file mode 100644
index 0000000000..98ff850161
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_1_9_audio_1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_cbcs_1_9_audio_1.m4s^headers^ b/dom/media/test/bipbop_cbcs_1_9_audio_1.m4s^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_1_9_audio_1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_cbcs_1_9_audio_init.mp4 b/dom/media/test/bipbop_cbcs_1_9_audio_init.mp4
new file mode 100644
index 0000000000..8f5521eaa3
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_1_9_audio_init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_cbcs_1_9_audio_init.mp4^headers^ b/dom/media/test/bipbop_cbcs_1_9_audio_init.mp4^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_1_9_audio_init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_cbcs_1_9_video_1.m4s b/dom/media/test/bipbop_cbcs_1_9_video_1.m4s
new file mode 100644
index 0000000000..7606c23199
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_1_9_video_1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_cbcs_1_9_video_1.m4s^headers^ b/dom/media/test/bipbop_cbcs_1_9_video_1.m4s^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_1_9_video_1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_cbcs_1_9_video_init.mp4 b/dom/media/test/bipbop_cbcs_1_9_video_init.mp4
new file mode 100644
index 0000000000..f9f20f9aec
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_1_9_video_init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_cbcs_1_9_video_init.mp4^headers^ b/dom/media/test/bipbop_cbcs_1_9_video_init.mp4^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_1_9_video_init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_cbcs_5_5_audio_1.m4s b/dom/media/test/bipbop_cbcs_5_5_audio_1.m4s
new file mode 100644
index 0000000000..98ff850161
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_5_5_audio_1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_cbcs_5_5_audio_1.m4s^headers^ b/dom/media/test/bipbop_cbcs_5_5_audio_1.m4s^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_5_5_audio_1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_cbcs_5_5_audio_init.mp4 b/dom/media/test/bipbop_cbcs_5_5_audio_init.mp4
new file mode 100644
index 0000000000..3118a175f1
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_5_5_audio_init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_cbcs_5_5_audio_init.mp4^headers^ b/dom/media/test/bipbop_cbcs_5_5_audio_init.mp4^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_5_5_audio_init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_cbcs_5_5_video_1.m4s b/dom/media/test/bipbop_cbcs_5_5_video_1.m4s
new file mode 100644
index 0000000000..8ed5b11aa0
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_5_5_video_1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_cbcs_5_5_video_1.m4s^headers^ b/dom/media/test/bipbop_cbcs_5_5_video_1.m4s^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_5_5_video_1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_cbcs_5_5_video_init.mp4 b/dom/media/test/bipbop_cbcs_5_5_video_init.mp4
new file mode 100644
index 0000000000..be9731c4c6
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_5_5_video_init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_cbcs_5_5_video_init.mp4^headers^ b/dom/media/test/bipbop_cbcs_5_5_video_init.mp4^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_5_5_video_init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_cbcs_7_7_audio_1.m4s b/dom/media/test/bipbop_cbcs_7_7_audio_1.m4s
new file mode 100644
index 0000000000..98ff850161
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_7_7_audio_1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_cbcs_7_7_audio_1.m4s^headers^ b/dom/media/test/bipbop_cbcs_7_7_audio_1.m4s^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_7_7_audio_1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_cbcs_7_7_audio_init.mp4 b/dom/media/test/bipbop_cbcs_7_7_audio_init.mp4
new file mode 100644
index 0000000000..3017ff231e
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_7_7_audio_init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_cbcs_7_7_audio_init.mp4^headers^ b/dom/media/test/bipbop_cbcs_7_7_audio_init.mp4^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_7_7_audio_init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_cbcs_7_7_video_1.m4s b/dom/media/test/bipbop_cbcs_7_7_video_1.m4s
new file mode 100644
index 0000000000..c3d41c16c6
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_7_7_video_1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_cbcs_7_7_video_1.m4s^headers^ b/dom/media/test/bipbop_cbcs_7_7_video_1.m4s^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_7_7_video_1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_cbcs_7_7_video_init.mp4 b/dom/media/test/bipbop_cbcs_7_7_video_init.mp4
new file mode 100644
index 0000000000..994e319e57
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_7_7_video_init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_cbcs_7_7_video_init.mp4^headers^ b/dom/media/test/bipbop_cbcs_7_7_video_init.mp4^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_7_7_video_init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_cbcs_9_8_audio_1.m4s b/dom/media/test/bipbop_cbcs_9_8_audio_1.m4s
new file mode 100644
index 0000000000..98ff850161
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_9_8_audio_1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_cbcs_9_8_audio_1.m4s^headers^ b/dom/media/test/bipbop_cbcs_9_8_audio_1.m4s^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_9_8_audio_1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_cbcs_9_8_audio_init.mp4 b/dom/media/test/bipbop_cbcs_9_8_audio_init.mp4
new file mode 100644
index 0000000000..0f436af510
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_9_8_audio_init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_cbcs_9_8_audio_init.mp4^headers^ b/dom/media/test/bipbop_cbcs_9_8_audio_init.mp4^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_9_8_audio_init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_cbcs_9_8_video_1.m4s b/dom/media/test/bipbop_cbcs_9_8_video_1.m4s
new file mode 100644
index 0000000000..1bb3dd493c
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_9_8_video_1.m4s
Binary files differ
diff --git a/dom/media/test/bipbop_cbcs_9_8_video_1.m4s^headers^ b/dom/media/test/bipbop_cbcs_9_8_video_1.m4s^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_9_8_video_1.m4s^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_cbcs_9_8_video_init.mp4 b/dom/media/test/bipbop_cbcs_9_8_video_init.mp4
new file mode 100644
index 0000000000..df5f9a13b8
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_9_8_video_init.mp4
Binary files differ
diff --git a/dom/media/test/bipbop_cbcs_9_8_video_init.mp4^headers^ b/dom/media/test/bipbop_cbcs_9_8_video_init.mp4^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_cbcs_9_8_video_init.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm b/dom/media/test/bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm
new file mode 100644
index 0000000000..7d0935aead
--- /dev/null
+++ b/dom/media/test/bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm
Binary files differ
diff --git a/dom/media/test/bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm^headers^ b/dom/media/test/bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm b/dom/media/test/bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm
new file mode 100644
index 0000000000..39d20e1e87
--- /dev/null
+++ b/dom/media/test/bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm
Binary files differ
diff --git a/dom/media/test/bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm^headers^ b/dom/media/test/bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm b/dom/media/test/bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm
new file mode 100644
index 0000000000..4d6a68fbc6
--- /dev/null
+++ b/dom/media/test/bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm
Binary files differ
diff --git a/dom/media/test/bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm^headers^ b/dom/media/test/bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bipbop_short_vp8.webm b/dom/media/test/bipbop_short_vp8.webm
new file mode 100644
index 0000000000..7c47b3d7e8
--- /dev/null
+++ b/dom/media/test/bipbop_short_vp8.webm
Binary files differ
diff --git a/dom/media/test/bipbop_short_vp8.webm^headers^ b/dom/media/test/bipbop_short_vp8.webm^headers^
new file mode 100644
index 0000000000..12a01c4a22
--- /dev/null
+++ b/dom/media/test/bipbop_short_vp8.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/black100x100-aspect3to2.ogv b/dom/media/test/black100x100-aspect3to2.ogv
new file mode 100644
index 0000000000..81fe51ffb3
--- /dev/null
+++ b/dom/media/test/black100x100-aspect3to2.ogv
Binary files differ
diff --git a/dom/media/test/black100x100-aspect3to2.ogv^headers^ b/dom/media/test/black100x100-aspect3to2.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/black100x100-aspect3to2.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bogus.duh b/dom/media/test/bogus.duh
new file mode 100644
index 0000000000..528ae275d0
--- /dev/null
+++ b/dom/media/test/bogus.duh
@@ -0,0 +1,45 @@
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
diff --git a/dom/media/test/bogus.ogv b/dom/media/test/bogus.ogv
new file mode 100644
index 0000000000..528ae275d0
--- /dev/null
+++ b/dom/media/test/bogus.ogv
@@ -0,0 +1,45 @@
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
diff --git a/dom/media/test/bogus.ogv^headers^ b/dom/media/test/bogus.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bogus.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bogus.wav b/dom/media/test/bogus.wav
new file mode 100644
index 0000000000..528ae275d0
--- /dev/null
+++ b/dom/media/test/bogus.wav
@@ -0,0 +1,45 @@
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
+bogus bogus bogus
diff --git a/dom/media/test/bogus.wav^headers^ b/dom/media/test/bogus.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bogus.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/browser/browser.toml b/dom/media/test/browser/browser.toml
new file mode 100644
index 0000000000..cf25369576
--- /dev/null
+++ b/dom/media/test/browser/browser.toml
@@ -0,0 +1,26 @@
+[DEFAULT]
+subsuite = "media-bc"
+prefs = ["gfx.font_loader.delay=0"]
+support-files = [
+ "file_empty_page.html",
+ "file_media.html",
+ "../av1.mp4",
+ "../bipbop_short_vp8.webm",
+ "../bunny_hd_5s.mp4",
+ "../eme_standalone.js",
+ "../gizmo.mp4",
+ "../gizmo.webm",
+ "../sintel-short-clearkey-subsample-encrypted-video.webm",
+ "../small-shot.flac",
+ "../small-shot.m4a",
+ "../small-shot.mp3",
+ "../small-shot.ogg",
+ "../TestPatternHDR.mp4",
+]
+
+["browser_encrypted_play_time_telemetry.js"]
+skip-if = ["apple_silicon"] # Disabled due to bleedover with other tests when run in regular suites; passes in "failures" jobs
+
+["browser_tab_visibility_and_play_time.js"]
+
+["browser_telemetry_video_hardware_decoding_support.js"]
diff --git a/dom/media/test/browser/browser_encrypted_play_time_telemetry.js b/dom/media/test/browser/browser_encrypted_play_time_telemetry.js
new file mode 100644
index 0000000000..ff4f2753ec
--- /dev/null
+++ b/dom/media/test/browser/browser_encrypted_play_time_telemetry.js
@@ -0,0 +1,269 @@
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this file,
+ * You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+// This test verifies that telemetry gathered around encrypted media playtime
+// is gathered as expected.
+
+"use strict";
+
+/* import-globals-from ../eme_standalone.js */
+
+// Clears any existing telemetry data that has been accumulated. Returns a
+// promise the will be resolved once the telemetry store is clear.
+async function clearTelemetry() {
+ // There's an arbitrary interval of 2 seconds in which the content
+ // processes sync their event data with the parent process, we wait
+ // this out to ensure that we clear everything that is left over from
+ // previous tests and don't receive random events in the middle of our tests.
+ // eslint-disable-next-line mozilla/no-arbitrary-setTimeout
+ await new Promise(resolve => setTimeout(resolve, 2000));
+
+ Services.telemetry.clearEvents();
+ return TestUtils.waitForCondition(() => {
+ let events = Services.telemetry.snapshotEvents(
+ Ci.nsITelemetry.DATASET_ALL_CHANNELS,
+ true
+ ).content;
+ return !events || !events.length;
+ });
+}
+
+// Opens a tab containing a blank page, returns a promise that will resolve
+// to that tab.
+async function openTab() {
+ const emptyPageUri =
+ "https://example.com/browser/dom/media/test/browser/file_empty_page.html";
+ return BrowserTestUtils.openNewForegroundTab(window.gBrowser, emptyPageUri);
+}
+
+// Creates and configures a video element for EME playback in `tab`. Does not
+// start playback for the element. Returns a promise that will resolve once
+// the element is setup and ready for playback.
+async function loadEmeVideo(tab) {
+ const emeHelperUri =
+ gTestPath.substr(0, gTestPath.lastIndexOf("/")) + "/eme_standalone.js";
+ return SpecialPowers.spawn(
+ tab.linkedBrowser,
+ [emeHelperUri],
+ async _emeHelperUri => {
+ // Begin helper functions.
+ async function once(target, name) {
+ return new Promise(r =>
+ target.addEventListener(name, r, { once: true })
+ );
+ }
+
+ // Helper to clone data into content so the EME helper can use the data.
+ function cloneIntoContent(data) {
+ return Cu.cloneInto(data, content.wrappedJSObject);
+ }
+ // End helper functions.
+
+ // Load the EME helper into content.
+ Services.scriptloader.loadSubScript(_emeHelperUri, content);
+ // Setup EME with the helper.
+ let video = content.document.createElement("video");
+ video.id = "media";
+ content.document.body.appendChild(video);
+ let emeHelper = new content.wrappedJSObject.EmeHelper();
+ emeHelper.SetKeySystem(
+ content.wrappedJSObject.EmeHelper.GetClearkeyKeySystemString()
+ );
+ emeHelper.SetInitDataTypes(cloneIntoContent(["webm"]));
+ emeHelper.SetVideoCapabilities(
+ cloneIntoContent([{ contentType: 'video/webm; codecs="vp9"' }])
+ );
+ emeHelper.AddKeyIdAndKey(
+ "2cdb0ed6119853e7850671c3e9906c3c",
+ "808b9adac384de1e4f56140f4ad76194"
+ );
+ emeHelper.onerror = error => {
+ is(false, `Got unexpected error from EME helper: ${error}`);
+ };
+ await emeHelper.ConfigureEme(video);
+ // Done setting up EME.
+
+ // Setup MSE.
+ const ms = new content.wrappedJSObject.MediaSource();
+ video.src = content.wrappedJSObject.URL.createObjectURL(ms);
+ await once(ms, "sourceopen");
+ const sb = ms.addSourceBuffer("video/webm");
+ const videoFile = "sintel-short-clearkey-subsample-encrypted-video.webm";
+ let fetchResponse = await content.fetch(videoFile);
+ sb.appendBuffer(await fetchResponse.arrayBuffer());
+ await once(sb, "updateend");
+ ms.endOfStream();
+ await once(ms, "sourceended");
+ }
+ );
+}
+
+// Plays the media in `tab` until the 'ended' event is fire. Returns a promise
+// that resolves once that state has been reached.
+async function playMediaThrough(tab) {
+ return SpecialPowers.spawn(tab.linkedBrowser, [], async () => {
+ let video = content.document.getElementById("media");
+ await Promise.all([new Promise(r => (video.onended = r)), video.play()]);
+ });
+}
+
+// Plays the media in `tab` until the 'timeupdate' event is fire. Returns a
+// promise that resolves once that state has been reached.
+async function playMediaToTimeUpdate(tab) {
+ return SpecialPowers.spawn(tab.linkedBrowser, [], async () => {
+ let video = content.document.getElementById("media");
+ await Promise.all([
+ new Promise(r => (video.ontimeupdate = r)),
+ video.play(),
+ ]);
+ });
+}
+
+// Aborts existing loads and replaces the media on the media element with an
+// unencrypted file.
+async function replaceMediaWithUnencrypted(tab) {
+ return SpecialPowers.spawn(tab.linkedBrowser, [], async () => {
+ let video = content.document.getElementById("media");
+ video.src = "gizmo.mp4";
+ video.load();
+ });
+}
+
+// Clears/nulls the media keys on the media in `tab`.
+async function clearMediaKeys(tab) {
+ return SpecialPowers.spawn(tab.linkedBrowser, [], async () => {
+ let video = content.document.getElementById("media");
+ await video.setMediaKeys(null);
+ });
+}
+
+// Wait for telemetry information to be received from the content process
+// then get the relevant histograms for the tests and return the sums of
+// those histograms. If a histogram does not exist this will return a 0
+// sum. Returns a promise the resolves to an object with sums for
+// - VIDEO_PLAY_TIME_MS
+// - VIDEO_ENCRYPTED_PLAY_TIME_MS
+// - VIDEO_CLEARKEY_PLAY_TIME_MS
+// This function clears the histograms as it gets them.
+async function getTelemetrySums() {
+ // The telemetry was gathered in the content process, so we have to wait
+ // until is arrived in the parent to check it. At time of writing there's
+ // not a more elegant way of doing this than polling.
+ return TestUtils.waitForCondition(() => {
+ let histograms = Services.telemetry.getSnapshotForHistograms(
+ "main",
+ true
+ ).content;
+ // All the histogram data should come at the same time, so we just check
+ // for playtime here as we always expect it in these tests, but we'll
+ // grab other values if present.
+ if (histograms.VIDEO_PLAY_TIME_MS) {
+ // We only expect to have one value for each histogram, so returning the
+ // sums is a short hand for returning that one value.
+ return {
+ VIDEO_PLAY_TIME_MS: histograms.VIDEO_PLAY_TIME_MS.sum,
+ VIDEO_ENCRYPTED_PLAY_TIME_MS: histograms.VIDEO_ENCRYPTED_PLAY_TIME_MS
+ ? histograms.VIDEO_ENCRYPTED_PLAY_TIME_MS.sum
+ : 0,
+ VIDEO_CLEARKEY_PLAY_TIME_MS: histograms.VIDEO_CLEARKEY_PLAY_TIME_MS
+ ? histograms.VIDEO_CLEARKEY_PLAY_TIME_MS.sum
+ : 0,
+ };
+ }
+ return null;
+ }, "recorded telemetry from playing media");
+}
+
+// Clear telemetry before other tests. Internally the tests clear the telemetry
+// when they check it, so we shouldn't need to do this between tests.
+add_task(clearTelemetry);
+
+add_task(async function testEncryptedMediaPlayback() {
+ let testTab = await openTab();
+
+ await loadEmeVideo(testTab);
+ await playMediaThrough(testTab);
+
+ BrowserTestUtils.removeTab(testTab);
+
+ let telemetrySums = await getTelemetrySums();
+
+ ok(telemetrySums, "Should get play time telemetry");
+ is(
+ telemetrySums.VIDEO_PLAY_TIME_MS,
+ telemetrySums.VIDEO_ENCRYPTED_PLAY_TIME_MS,
+ "Play time should be the same as encrypted play time"
+ );
+ is(
+ telemetrySums.VIDEO_PLAY_TIME_MS,
+ telemetrySums.VIDEO_CLEARKEY_PLAY_TIME_MS,
+ "Play time should be the same as clearkey play time"
+ );
+ Assert.greater(
+ telemetrySums.VIDEO_PLAY_TIME_MS,
+ 0,
+ "Should have a play time greater than zero"
+ );
+});
+
+add_task(async function testChangingFromEncryptedToUnencrypted() {
+ let testTab = await openTab();
+
+ await loadEmeVideo(testTab);
+ await replaceMediaWithUnencrypted(testTab);
+ await playMediaToTimeUpdate(testTab);
+
+ BrowserTestUtils.removeTab(testTab);
+
+ let telemetrySums = await getTelemetrySums();
+
+ ok(telemetrySums, "Should get play time telemetry");
+ is(
+ telemetrySums.VIDEO_ENCRYPTED_PLAY_TIME_MS,
+ 0,
+ "Encrypted play time should be 0"
+ );
+ is(
+ telemetrySums.VIDEO_PLAY_TIME_MS,
+ telemetrySums.VIDEO_CLEARKEY_PLAY_TIME_MS,
+ "Play time should be the same as clearkey play time because the media element still has a media keys attached"
+ );
+ Assert.greater(
+ telemetrySums.VIDEO_PLAY_TIME_MS,
+ 0,
+ "Should have a play time greater than zero"
+ );
+});
+
+add_task(
+ async function testChangingFromEncryptedToUnencryptedAndClearingMediaKeys() {
+ let testTab = await openTab();
+
+ await loadEmeVideo(testTab);
+ await replaceMediaWithUnencrypted(testTab);
+ await clearMediaKeys(testTab);
+ await playMediaToTimeUpdate(testTab);
+
+ BrowserTestUtils.removeTab(testTab);
+
+ let telemetrySums = await getTelemetrySums();
+
+ ok(telemetrySums, "Should get play time telemetry");
+ is(
+ telemetrySums.VIDEO_ENCRYPTED_PLAY_TIME_MS,
+ 0,
+ "Encrypted play time should be 0"
+ );
+ is(
+ telemetrySums.VIDEO_CLEARKEY_PLAY_TIME_MS,
+ 0,
+ "Clearkey play time should be 0"
+ );
+ Assert.greater(
+ telemetrySums.VIDEO_PLAY_TIME_MS,
+ 0,
+ "Should have a play time greater than zero"
+ );
+ }
+);
diff --git a/dom/media/test/browser/browser_partial.js b/dom/media/test/browser/browser_partial.js
new file mode 100644
index 0000000000..572287b392
--- /dev/null
+++ b/dom/media/test/browser/browser_partial.js
@@ -0,0 +1,56 @@
+async function test() {
+ waitForExplicitFinish();
+ const target =
+ "https://example.com/browser/dom/media/test/browser/file_empty_page.html";
+
+ info("Loading download page...");
+
+ let tab = BrowserTestUtils.addTab(gBrowser, target);
+
+ registerCleanupFunction(function () {
+ gBrowser.removeTab(tab);
+ window.restore();
+ });
+
+ gBrowser.selectedTab = tab;
+ BrowserTestUtils.browserLoaded(tab.linkedBrowser, false, target).then(
+ async () => {
+ info("Page loaded.");
+ let allDownloads = await Downloads.getList(Downloads.ALL);
+ let started = new Promise(resolve => {
+ // With no download modal, the download will begin on its own, so we need
+ // to wait to be notified by the downloads list when that happens.
+ let downloadView = {
+ onDownloadAdded(download) {
+ ok(true, "Download was started.");
+ download.cancel();
+ allDownloads.removeView(this);
+ allDownloads.removeFinished();
+ resolve();
+ },
+ };
+ allDownloads.addView(downloadView);
+ });
+
+ let revoked = SpecialPowers.spawn(
+ tab.linkedBrowser,
+ [],
+ () =>
+ new Promise(resolve => {
+ let link = content.document.createElement("a");
+ link.href = "force_octet_stream.mp4";
+ content.document.body.appendChild(link);
+ info("Clicking HTMLAnchorElement...");
+ link.click();
+ info("Clicked HTMLAnchorElement.");
+ resolve();
+ })
+ );
+
+ info("Waiting for async activities...");
+ await Promise.all([revoked, started]);
+ ok(true, "Exiting test.");
+ finish();
+ }
+ );
+}
diff --git a/dom/media/test/browser/browser_tab_visibility_and_play_time.js b/dom/media/test/browser/browser_tab_visibility_and_play_time.js
new file mode 100644
index 0000000000..4d33826091
--- /dev/null
+++ b/dom/media/test/browser/browser_tab_visibility_and_play_time.js
@@ -0,0 +1,218 @@
+/**
+ * This test is used to ensure that invisible play time would be accumulated
+ * when tab is in background. It also checks the HDR video accumulation time.
+ * However, this test won't directly check the reported telemetry result,
+ * because we can't check the snapshot histogram in the content process.
+ * The actual probe checking happens in `test_accumulated_play_time.html`.
+ */
+"use strict";
+
+const PAGE_URL =
+ "https://example.com/browser/dom/media/test/browser/file_media.html";
+
+// This HDR tests will only pass on platforms that accurately report color
+// depth in their VideoInfo structures. Presently, that is only true for
+// macOS.
+
+const reportsColorDepthFromVideoData = AppConstants.platform == "macosx";
+
+add_task(async function testChangingTabVisibilityAffectsInvisiblePlayTime() {
+ const originalTab = gBrowser.selectedTab;
+ const mediaTab = await openMediaTab(PAGE_URL);
+
+ info(`measuring play time when tab is in foreground`);
+ await startMedia({
+ mediaTab,
+ shouldAccumulateTime: true,
+ shouldAccumulateInvisibleTime: false,
+ shouldAccumulateHDRTime: reportsColorDepthFromVideoData,
+ });
+ await pauseMedia(mediaTab);
+
+ info(`measuring play time when tab is in background`);
+ await BrowserTestUtils.switchTab(window.gBrowser, originalTab);
+ await startMedia({
+ mediaTab,
+ shouldAccumulateTime: true,
+ shouldAccumulateInvisibleTime: true,
+ shouldAccumulateHDRTime: reportsColorDepthFromVideoData,
+ });
+ await pauseMedia(mediaTab);
+
+ BrowserTestUtils.removeTab(mediaTab);
+});
+
+/**
+ * Following are helper functions.
+ */
+async function openMediaTab(url) {
+ info(`open tab for media playback`);
+ const tab = await BrowserTestUtils.openNewForegroundTab(window.gBrowser, url);
+ info(`add content helper functions and variables`);
+ await SpecialPowers.spawn(tab.linkedBrowser, [], _ => {
+ content.waitForOnTimeUpdate = element => {
+ return new Promise(resolve => {
+ element.addEventListener(
+ "timeupdate",
+ e => {
+ resolve();
+ },
+ { once: true }
+ );
+ });
+ };
+
+ content.sleep = ms => {
+ return new Promise(resolve => content.setTimeout(resolve, ms));
+ };
+
+ content.assertAttributeDefined = (videoChrome, checkType) => {
+ Assert.notEqual(videoChrome[checkType], undefined, `${checkType} exists`);
+ };
+ content.assertValueEqualTo = (videoChrome, checkType, expectedValue) => {
+ content.assertAttributeDefined(videoChrome, checkType);
+ is(
+ videoChrome[checkType],
+ expectedValue,
+ `${checkType} equals to ${expectedValue}`
+ );
+ };
+ content.assertValueConstantlyIncreases = async (videoChrome, checkType) => {
+ content.assertAttributeDefined(videoChrome, checkType);
+ const valueSnapshot = videoChrome[checkType];
+ await content.waitForOnTimeUpdate(videoChrome);
+ Assert.greater(
+ videoChrome[checkType],
+ valueSnapshot,
+ `${checkType} keeps increasing`
+ );
+ };
+ content.assertValueKeptUnchanged = async (videoChrome, checkType) => {
+ content.assertAttributeDefined(videoChrome, checkType);
+ const valueSnapshot = videoChrome[checkType];
+ await content.sleep(1000);
+ Assert.equal(
+ videoChrome[checkType],
+ valueSnapshot,
+ `${checkType} keeps unchanged`
+ );
+ };
+ });
+ return tab;
+}
+
+function startMedia({
+ mediaTab,
+ shouldAccumulateTime,
+ shouldAccumulateInvisibleTime,
+ shouldAccumulateHDRTime,
+}) {
+ return SpecialPowers.spawn(
+ mediaTab.linkedBrowser,
+ [
+ shouldAccumulateTime,
+ shouldAccumulateInvisibleTime,
+ shouldAccumulateHDRTime,
+ ],
+ async (accumulateTime, accumulateInvisibleTime, accumulateHDRTime) => {
+ const video = content.document.getElementById("video");
+ ok(
+ await video.play().then(
+ () => true,
+ () => false
+ ),
+ "video started playing"
+ );
+ const videoChrome = SpecialPowers.wrap(video);
+ if (accumulateTime) {
+ await content.assertValueConstantlyIncreases(
+ videoChrome,
+ "totalVideoPlayTime"
+ );
+ } else {
+ await content.assertValueKeptUnchanged(
+ videoChrome,
+ "totalVideoPlayTime"
+ );
+ }
+ if (accumulateInvisibleTime) {
+ await content.assertValueConstantlyIncreases(
+ videoChrome,
+ "invisiblePlayTime"
+ );
+ } else {
+ await content.assertValueKeptUnchanged(
+ videoChrome,
+ "invisiblePlayTime"
+ );
+ }
+
+ const videoHDR = content.document.getElementById("videoHDR");
+
+ // HDR test video might not decode on all platforms, so catch
+ // the play() command and exit early in such a case. Failure to
+ // decode might manifest as a timeout, so add a rejection race
+ // to catch that.
+ let didDecode = true;
+ const playPromise = videoHDR.play().then(
+ () => true,
+ () => false
+ );
+ /* eslint-disable mozilla/no-arbitrary-setTimeout */
+ const tooSlowPromise = new Promise(resolve =>
+ setTimeout(() => {
+ info("videoHDR timed out.");
+ didDecode = false;
+ resolve(false);
+ }, 1000)
+ );
+ /* eslint-enable mozilla/no-arbitrary-setTimeout */
+
+ let didPlay = await Promise.race(playPromise, tooSlowPromise).catch(
+ err => {
+ info("videoHDR failed to decode with error: " + err.message);
+ didDecode = false;
+ return false;
+ }
+ );
+
+ if (!didDecode) {
+ return;
+ }
+
+ ok(didPlay, "videoHDR started playing");
+ const videoHDRChrome = SpecialPowers.wrap(videoHDR);
+ if (accumulateHDRTime) {
+ await content.assertValueConstantlyIncreases(
+ videoHDRChrome,
+ "totalVideoHDRPlayTime"
+ );
+ } else {
+ await content.assertValueKeptUnchanged(
+ videoHDRChrome,
+ "totalVideoHDRPlayTime"
+ );
+ }
+ }
+ );
+}
+
+function pauseMedia(tab) {
+ return SpecialPowers.spawn(tab.linkedBrowser, [], async _ => {
+ const video = content.document.getElementById("video");
+ video.pause();
+ ok(true, "video paused");
+ const videoChrome = SpecialPowers.wrap(video);
+ await content.assertValueKeptUnchanged(videoChrome, "totalVideoPlayTime");
+ await content.assertValueKeptUnchanged(videoChrome, "invisiblePlayTime");
+
+ const videoHDR = content.document.getElementById("videoHDR");
+ videoHDR.pause();
+ ok(true, "videoHDR paused");
+ const videoHDRChrome = SpecialPowers.wrap(videoHDR);
+ await content.assertValueKeptUnchanged(
+ videoHDRChrome,
+ "totalVideoHDRPlayTime"
+ );
+ });
+}
diff --git a/dom/media/test/browser/browser_telemetry_video_hardware_decoding_support.js b/dom/media/test/browser/browser_telemetry_video_hardware_decoding_support.js
new file mode 100644
index 0000000000..3b1b41c03f
--- /dev/null
+++ b/dom/media/test/browser/browser_telemetry_video_hardware_decoding_support.js
@@ -0,0 +1,106 @@
+/**
+ * This test is used to ensure that the scalar which indicates whether hardware
+ * decoding is supported for a specific video codec type can be recorded
+ * correctly.
+ */
+"use strict";
+
+add_task(async function setupTestingPref() {
+ await SpecialPowers.pushPrefEnv({
+ set: [
+ // In order to test av1 in the chrome process, see https://bit.ly/3oF0oan
+ ["media.rdd-process.enabled", false],
+ ],
+ });
+});
+
+const ALL_SCALAR = "media.video_hardware_decoding_support";
+const HD_SCALAR = "media.video_hd_hardware_decoding_support";
+
+add_task(async function testVideoCodecs() {
+ // There are still other video codecs, but we only care about these popular
+ // codec types.
+ const testFiles = [
+ { fileName: "gizmo.mp4", type: "video/avc" },
+ { fileName: "gizmo.webm", type: "video/vp9" },
+ { fileName: "bipbop_short_vp8.webm", type: "video/vp8" },
+ { fileName: "av1.mp4", type: "video/av1" },
+ { fileName: "bunny_hd_5s.mp4", type: "video/avc", hd: true },
+ ];
+
+ for (const file of testFiles) {
+ const { fileName, type, hd } = file;
+ let video = document.createElement("video");
+ video.src = GetTestWebBasedURL(fileName);
+ await video.play();
+ let snapshot = Services.telemetry.getSnapshotForKeyedScalars(
+ "main",
+ false
+ ).parent;
+ ok(
+ snapshot.hasOwnProperty(ALL_SCALAR),
+ `Found stored scalar '${ALL_SCALAR}'`
+ );
+ ok(
+ snapshot[ALL_SCALAR].hasOwnProperty(type),
+ `Found key '${type}' in '${ALL_SCALAR}'`
+ );
+ if (hd) {
+ ok(
+ snapshot.hasOwnProperty(HD_SCALAR),
+ `HD video '${fileName}' should record a scalar '${HD_SCALAR}'`
+ );
+ ok(
+ snapshot[HD_SCALAR].hasOwnProperty(type),
+ `Found key '${type}' in '${HD_SCALAR}'`
+ );
+ } else {
+ ok(
+ !snapshot.hasOwnProperty(HD_SCALAR),
+ `SD video won't store a scalar '${HD_SCALAR}'`
+ );
+ }
+ video.src = "";
+ Services.telemetry.clearScalars();
+ }
+});
+
+add_task(async function testAudioCodecs() {
+ const testFiles = [
+ "small-shot.ogg",
+ "small-shot.m4a",
+ "small-shot.mp3",
+ "small-shot.flac",
+ ];
+ for (const file of testFiles) {
+ let audio = document.createElement("audio");
+ info(GetTestWebBasedURL(file));
+ audio.src = GetTestWebBasedURL(file);
+ await audio.play();
+ let snapshot = Services.telemetry.getSnapshotForKeyedScalars(
+ "main",
+ false
+ ).parent;
+ ok(
+ !snapshot ||
+ (!snapshot.hasOwnProperty(ALL_SCALAR) &&
+ !snapshot.hasOwnProperty(HD_SCALAR)),
+ `Did not record scalar for ${file}`
+ );
+ audio.src = "";
+ }
+});
+
+/**
+ * Return a web-based URL for a given file based on the testing directory.
+ * @param {String} fileName
+ * file that caller wants its web-based url
+ */
+function GetTestWebBasedURL(fileName) {
+ return (
+ getRootDirectory(gTestPath).replace(
+ "chrome://mochitests/content",
+ "http://example.org"
+ ) + fileName
+ );
+}
diff --git a/dom/media/test/browser/file_empty_page.html b/dom/media/test/browser/file_empty_page.html
new file mode 100644
index 0000000000..cd1b7830be
--- /dev/null
+++ b/dom/media/test/browser/file_empty_page.html
@@ -0,0 +1,8 @@
+<!DOCTYPE html>
+<html>
+<head>
+<title>An empty page</title>
+</head>
+<body>
+</body>
+</html>
diff --git a/dom/media/test/browser/file_media.html b/dom/media/test/browser/file_media.html
new file mode 100644
index 0000000000..36dca8d01c
--- /dev/null
+++ b/dom/media/test/browser/file_media.html
@@ -0,0 +1,10 @@
+<!DOCTYPE html>
+<html>
+<head>
+<title>Non-Autoplay page</title>
+</head>
+<body>
+<video id="video" src="gizmo.mp4" loop></video>
+<video id="videoHDR" src="TestPatternHDR.mp4" loop></video>
+</body>
+</html>
diff --git a/dom/media/test/browser/wmfme/browser.toml b/dom/media/test/browser/wmfme/browser.toml
new file mode 100644
index 0000000000..422ea29228
--- /dev/null
+++ b/dom/media/test/browser/wmfme/browser.toml
@@ -0,0 +1,13 @@
+[DEFAULT]
+subsuite = "media-bc"
+tags = "media-engine-compatible"
+run-if = ["wmfme"]
+support-files = [
+ "head.js",
+ "file_video.html",
+ "../../gizmo.mp4",
+]
+
+["browser_wmfme_crash.js"]
+
+["browser_wmfme_max_crashes.js"]
diff --git a/dom/media/test/browser/wmfme/browser_wmfme_crash.js b/dom/media/test/browser/wmfme/browser_wmfme_crash.js
new file mode 100644
index 0000000000..57064cfe8d
--- /dev/null
+++ b/dom/media/test/browser/wmfme/browser_wmfme_crash.js
@@ -0,0 +1,52 @@
+"use strict";
+
+/**
+ * This test aims to ensure that the media engine playback will recover from a
+ * crash and keep playing without any problem.
+ */
+add_task(async function setupTestingPref() {
+ await SpecialPowers.pushPrefEnv({
+ set: [
+ ["media.wmf.media-engine.enabled", 1],
+ ["media.wmf.media-engine.channel-decoder.enabled", true],
+ ],
+ });
+});
+
+const VIDEO_PAGE = GetTestWebBasedURL("file_video.html");
+
+add_task(async function testPlaybackRecoveryFromCrash() {
+ info(`Create a tab and load test page`);
+ let tab = await BrowserTestUtils.openNewForegroundTab(
+ window.gBrowser,
+ "about:blank"
+ );
+ BrowserTestUtils.startLoadingURIString(tab.linkedBrowser, VIDEO_PAGE);
+ await BrowserTestUtils.browserLoaded(tab.linkedBrowser);
+
+ await playVideo(tab);
+
+ info("Ensure video is running via the media engine framework");
+ await assertRunningProcessAndDecoderName(tab, {
+ expectedProcess: "Utility MF Media Engine CDM",
+ expectedDecoder: "media engine video stream",
+ });
+
+ const pidBeforeCrash = await getMFCDMProcessId();
+ await crashUtilityProcess(pidBeforeCrash);
+
+ info("The CDM process should be recreated which makes media keep playing");
+ await assertRunningProcessAndDecoderName(tab, {
+ expectedProcess: "Utility MF Media Engine CDM",
+ expectedDecoder: "media engine video stream",
+ });
+
+ const pidAfterCrash = await getMFCDMProcessId();
+ isnot(
+ pidBeforeCrash,
+ pidAfterCrash,
+ `new process ${pidAfterCrash} is not previous crashed one ${pidBeforeCrash}`
+ );
+
+ BrowserTestUtils.removeTab(tab);
+});
diff --git a/dom/media/test/browser/wmfme/browser_wmfme_max_crashes.js b/dom/media/test/browser/wmfme/browser_wmfme_max_crashes.js
new file mode 100644
index 0000000000..8cc8dc08b8
--- /dev/null
+++ b/dom/media/test/browser/wmfme/browser_wmfme_max_crashes.js
@@ -0,0 +1,69 @@
+"use strict";
+
+/**
+ * This test aims to ensure that the MFCDM process won't be recovered once the
+ * amount of crashes has exceeded the amount of value which we tolerate.
+ */
+add_task(async function setupTestingPref() {
+ await SpecialPowers.pushPrefEnv({
+ set: [
+ ["media.wmf.media-engine.enabled", 1],
+ ["media.wmf.media-engine.channel-decoder.enabled", true],
+ ],
+ });
+});
+
+const VIDEO_PAGE = GetTestWebBasedURL("file_video.html");
+
+add_task(async function testPlaybackRecoveryFromCrash() {
+ const maxCrashes = Services.prefs.getIntPref(
+ "media.wmf.media-engine.max-crashes"
+ );
+ info(`The amount of tolerable crashes=${maxCrashes}`);
+
+ info(`Create a tab and load test page`);
+ let tab = await BrowserTestUtils.openNewForegroundTab(
+ window.gBrowser,
+ "about:blank"
+ );
+ BrowserTestUtils.startLoadingURIString(tab.linkedBrowser, VIDEO_PAGE);
+ await BrowserTestUtils.browserLoaded(tab.linkedBrowser);
+
+ await playVideo(tab);
+
+ info("Ensure video is running via the media engine framework");
+ await assertRunningProcessAndDecoderName(tab, {
+ expectedProcess: "Utility MF Media Engine CDM",
+ expectedDecoder: "media engine video stream",
+ });
+
+ let pidBeforeCrash, pidAfterCrash;
+ for (let idx = 0; idx < maxCrashes; idx++) {
+ pidBeforeCrash = await getMFCDMProcessId();
+ await crashUtilityProcess(pidBeforeCrash);
+
+ info("The CDM process should be recreated which makes media keep playing");
+ await assertRunningProcessAndDecoderName(tab, {
+ expectedProcess: "Utility MF Media Engine CDM",
+ expectedDecoder: "media engine video stream",
+ });
+
+ pidAfterCrash = await getMFCDMProcessId();
+ isnot(
+ pidBeforeCrash,
+ pidAfterCrash,
+ `new process ${pidAfterCrash} is not previous crashed one ${pidBeforeCrash}`
+ );
+ }
+
+ info("This crash should result in not spawning MFCDM process again");
+ pidBeforeCrash = await getMFCDMProcessId();
+ await crashUtilityProcess(pidBeforeCrash);
+
+ await assertNotEqualRunningProcessAndDecoderName(tab, {
+ givenProcess: "Utility MF Media Engine CDM",
+ givenDecoder: "media engine video stream",
+ });
+
+ BrowserTestUtils.removeTab(tab);
+});
diff --git a/dom/media/test/browser/wmfme/file_video.html b/dom/media/test/browser/wmfme/file_video.html
new file mode 100644
index 0000000000..3c70268fbb
--- /dev/null
+++ b/dom/media/test/browser/wmfme/file_video.html
@@ -0,0 +1,9 @@
+<!DOCTYPE html>
+<html>
+<head>
+<title>video</title>
+</head>
+<body>
+<video id="v" src="gizmo.mp4" controls loop></video>
+</body>
+</html>
diff --git a/dom/media/test/browser/wmfme/head.js b/dom/media/test/browser/wmfme/head.js
new file mode 100644
index 0000000000..1e6f3b18bb
--- /dev/null
+++ b/dom/media/test/browser/wmfme/head.js
@@ -0,0 +1,201 @@
+"use strict";
+
+/**
+ * Return a web-based URL for a given file based on the testing directory.
+ * @param {String} fileName
+ * file that caller wants its web-based url
+ * @param {Boolean} cors [optional]
+ * if set, then return a url with different origin
+ */
+function GetTestWebBasedURL(fileName) {
+ const origin = "https://example.com";
+ return (
+ getRootDirectory(gTestPath).replace("chrome://mochitests/content", origin) +
+ fileName
+ );
+}
+
+/**
+ * Return current process Id for the Media Foundation CDM process.
+ */
+async function getMFCDMProcessId() {
+ const process = (await ChromeUtils.requestProcInfo()).children.find(
+ p =>
+ p.type === "utility" &&
+ p.utilityActors.find(a => a.actorName === "mfMediaEngineCDM")
+ );
+ return process.pid;
+}
+
+/**
+ * Make the utility process with given process id crash.
+ * @param {int} pid
+ * the process id for the process which is going to crash
+ */
+async function crashUtilityProcess(utilityPid) {
+ info(`Crashing process ${utilityPid}`);
+ SimpleTest.expectChildProcessCrash();
+
+ const crashMan = Services.crashmanager;
+ const utilityProcessGone = TestUtils.topicObserved(
+ "ipc:utility-shutdown",
+ (subject, data) => {
+ info(`ipc:utility-shutdown: data=${data} subject=${subject}`);
+ return parseInt(data, 10) === utilityPid;
+ }
+ );
+
+ info("Prune any previous crashes");
+ const future = new Date(Date.now() + 1000 * 60 * 60 * 24);
+ await crashMan.pruneOldCrashes(future);
+
+ info("Crash Utility Process");
+ const ProcessTools = Cc["@mozilla.org/processtools-service;1"].getService(
+ Ci.nsIProcessToolsService
+ );
+
+ info(`Crash Utility Process ${utilityPid}`);
+ ProcessTools.crash(utilityPid);
+
+ info(`Waiting for utility process ${utilityPid} to go away.`);
+ let [subject, data] = await utilityProcessGone;
+ Assert.strictEqual(
+ parseInt(data, 10),
+ utilityPid,
+ `Should match the crashed PID ${utilityPid} with ${data}`
+ );
+ ok(
+ subject instanceof Ci.nsIPropertyBag2,
+ "Subject needs to be a nsIPropertyBag2 to clean up properly"
+ );
+
+ const dumpID = subject.getPropertyAsAString("dumpID");
+ ok(dumpID, "There should be a dumpID");
+
+ await crashMan.ensureCrashIsPresent(dumpID);
+ await crashMan.getCrashes().then(crashes => {
+ is(crashes.length, 1, "There should be only one record");
+ const crash = crashes[0];
+ ok(
+ crash.isOfType(
+ crashMan.processTypes[Ci.nsIXULRuntime.PROCESS_TYPE_UTILITY],
+ crashMan.CRASH_TYPE_CRASH
+ ),
+ "Record should be a utility process crash"
+ );
+ Assert.strictEqual(crash.id, dumpID, "Record should have an ID");
+ });
+
+ let minidumpDirectory = Services.dirsvc.get("ProfD", Ci.nsIFile);
+ minidumpDirectory.append("minidumps");
+
+ let dumpfile = minidumpDirectory.clone();
+ dumpfile.append(dumpID + ".dmp");
+ if (dumpfile.exists()) {
+ info(`Removal of ${dumpfile.path}`);
+ dumpfile.remove(false);
+ }
+
+ let extrafile = minidumpDirectory.clone();
+ extrafile.append(dumpID + ".extra");
+ info(`Removal of ${extrafile.path}`);
+ if (extrafile.exists()) {
+ extrafile.remove(false);
+ }
+}
+
+/**
+ * Make video in the tab play.
+ * @param {object} tab
+ * the tab contains at least one video element
+ */
+async function playVideo(tab) {
+ return SpecialPowers.spawn(tab.linkedBrowser, [], async _ => {
+ const video = content.document.querySelector("video");
+ ok(
+ await video.play().then(
+ () => true,
+ () => false
+ ),
+ "video started playing"
+ );
+ });
+}
+
+/**
+ * Check whether the video playback is performed in the right process and right decoder.
+ * @param {object} tab
+ * the tab which has a playing video
+ * @param {string} expectedProcess
+ * the expected process name
+ * @param {string} expectedDecoder
+ * the expected decoder name
+ */
+async function assertRunningProcessAndDecoderName(
+ tab,
+ { expectedProcess, expectedDecoder } = {}
+) {
+ return SpecialPowers.spawn(
+ tab.linkedBrowser,
+ [expectedProcess, expectedDecoder],
+ // eslint-disable-next-line no-shadow
+ async (expectedProcess, expectedDecoder) => {
+ const video = content.document.querySelector("video");
+ ok(!video.paused, "checking a playing video");
+
+ const debugInfo = await SpecialPowers.wrap(video).mozRequestDebugInfo();
+ const videoDecoderName = debugInfo.decoder.reader.videoDecoderName;
+
+ const isExpectedDecoder =
+ videoDecoderName.indexOf(`${expectedDecoder}`) == 0;
+ ok(
+ isExpectedDecoder,
+ `Playback running by decoder '${videoDecoderName}', expected '${expectedDecoder}'`
+ );
+
+ const isExpectedProcess =
+ videoDecoderName.indexOf(`(${expectedProcess} remote)`) > 0;
+ ok(
+ isExpectedProcess,
+ `Playback running in process '${videoDecoderName}', expected '${expectedProcess}'`
+ );
+ }
+ );
+}
+
+/**
+ * Check whether the video playback is not performed in the given process and given decoder.
+ * @param {object} tab
+ * the tab which has a playing video
+ * @param {string} givenProcess
+ * the process name on which the video playback should not be running
+ * @param {string} givenDecoder
+ * the decoder name with which the video playback should not be running
+ */
+async function assertNotEqualRunningProcessAndDecoderName(
+ tab,
+ { givenProcess, givenDecoder } = {}
+) {
+ return SpecialPowers.spawn(
+ tab.linkedBrowser,
+ [givenProcess, givenDecoder],
+ // eslint-disable-next-line no-shadow
+ async (givenProcess, givenDecoder) => {
+ const video = content.document.querySelector("video");
+ ok(!video.paused, "checking a playing video");
+
+ const debugInfo = await SpecialPowers.wrap(video).mozRequestDebugInfo();
+ const videoDecoderName = debugInfo.decoder.reader.videoDecoderName;
+ const pattern = /(.+?)\s+\((\S+)\s+remote\)/;
+ const match = videoDecoderName.match(pattern);
+ if (match) {
+ const decoder = match[1];
+ const process = match[2];
+ isnot(decoder, givenDecoder, `Decoder name is not equal`);
+ isnot(process, givenProcess, `Process name is not equal`);
+ } else {
+ ok(false, "failed to match decoder/process name?");
+ }
+ }
+ );
+}
diff --git a/dom/media/test/bug1066943.webm b/dom/media/test/bug1066943.webm
new file mode 100644
index 0000000000..64a24ec898
--- /dev/null
+++ b/dom/media/test/bug1066943.webm
Binary files differ
diff --git a/dom/media/test/bug1066943.webm^headers^ b/dom/media/test/bug1066943.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug1066943.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug1301226-odd.wav b/dom/media/test/bug1301226-odd.wav
new file mode 100644
index 0000000000..dd2df4e4dd
--- /dev/null
+++ b/dom/media/test/bug1301226-odd.wav
Binary files differ
diff --git a/dom/media/test/bug1301226-odd.wav^headers^ b/dom/media/test/bug1301226-odd.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug1301226-odd.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug1301226.wav b/dom/media/test/bug1301226.wav
new file mode 100644
index 0000000000..0128486f07
--- /dev/null
+++ b/dom/media/test/bug1301226.wav
Binary files differ
diff --git a/dom/media/test/bug1301226.wav^headers^ b/dom/media/test/bug1301226.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug1301226.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug1377278.webm b/dom/media/test/bug1377278.webm
new file mode 100644
index 0000000000..802019f39f
--- /dev/null
+++ b/dom/media/test/bug1377278.webm
Binary files differ
diff --git a/dom/media/test/bug1377278.webm^headers^ b/dom/media/test/bug1377278.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug1377278.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug1535980.webm b/dom/media/test/bug1535980.webm
new file mode 100644
index 0000000000..efc43959bf
--- /dev/null
+++ b/dom/media/test/bug1535980.webm
Binary files differ
diff --git a/dom/media/test/bug1535980.webm^headers^ b/dom/media/test/bug1535980.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug1535980.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug1799787.webm b/dom/media/test/bug1799787.webm
new file mode 100644
index 0000000000..e6e613a59a
--- /dev/null
+++ b/dom/media/test/bug1799787.webm
Binary files differ
diff --git a/dom/media/test/bug1799787.webm^headers^ b/dom/media/test/bug1799787.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug1799787.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug461281.ogg b/dom/media/test/bug461281.ogg
new file mode 100644
index 0000000000..d7f6a0ccf4
--- /dev/null
+++ b/dom/media/test/bug461281.ogg
Binary files differ
diff --git a/dom/media/test/bug461281.ogg^headers^ b/dom/media/test/bug461281.ogg^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug461281.ogg^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug482461-theora.ogv b/dom/media/test/bug482461-theora.ogv
new file mode 100644
index 0000000000..941b8d8efd
--- /dev/null
+++ b/dom/media/test/bug482461-theora.ogv
Binary files differ
diff --git a/dom/media/test/bug482461-theora.ogv^headers^ b/dom/media/test/bug482461-theora.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug482461-theora.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug482461.ogv b/dom/media/test/bug482461.ogv
new file mode 100644
index 0000000000..6cf6aed330
--- /dev/null
+++ b/dom/media/test/bug482461.ogv
Binary files differ
diff --git a/dom/media/test/bug482461.ogv^headers^ b/dom/media/test/bug482461.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug482461.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug495129.ogv b/dom/media/test/bug495129.ogv
new file mode 100644
index 0000000000..44eb9296f5
--- /dev/null
+++ b/dom/media/test/bug495129.ogv
Binary files differ
diff --git a/dom/media/test/bug495129.ogv^headers^ b/dom/media/test/bug495129.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug495129.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug495794.ogg b/dom/media/test/bug495794.ogg
new file mode 100644
index 0000000000..1c19a64061
--- /dev/null
+++ b/dom/media/test/bug495794.ogg
Binary files differ
diff --git a/dom/media/test/bug495794.ogg^headers^ b/dom/media/test/bug495794.ogg^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug495794.ogg^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug498380.ogv b/dom/media/test/bug498380.ogv
new file mode 100644
index 0000000000..1179ecb70a
--- /dev/null
+++ b/dom/media/test/bug498380.ogv
Binary files differ
diff --git a/dom/media/test/bug498380.ogv^headers^ b/dom/media/test/bug498380.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug498380.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug498855-1.ogv b/dom/media/test/bug498855-1.ogv
new file mode 100644
index 0000000000..95a524da4c
--- /dev/null
+++ b/dom/media/test/bug498855-1.ogv
Binary files differ
diff --git a/dom/media/test/bug498855-1.ogv^headers^ b/dom/media/test/bug498855-1.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug498855-1.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug498855-2.ogv b/dom/media/test/bug498855-2.ogv
new file mode 100644
index 0000000000..795a308ae1
--- /dev/null
+++ b/dom/media/test/bug498855-2.ogv
Binary files differ
diff --git a/dom/media/test/bug498855-2.ogv^headers^ b/dom/media/test/bug498855-2.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug498855-2.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug498855-3.ogv b/dom/media/test/bug498855-3.ogv
new file mode 100644
index 0000000000..714858dfed
--- /dev/null
+++ b/dom/media/test/bug498855-3.ogv
Binary files differ
diff --git a/dom/media/test/bug498855-3.ogv^headers^ b/dom/media/test/bug498855-3.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug498855-3.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug499519.ogv b/dom/media/test/bug499519.ogv
new file mode 100644
index 0000000000..62c0922d36
--- /dev/null
+++ b/dom/media/test/bug499519.ogv
Binary files differ
diff --git a/dom/media/test/bug499519.ogv^headers^ b/dom/media/test/bug499519.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug499519.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug500311.ogv b/dom/media/test/bug500311.ogv
new file mode 100644
index 0000000000..2cf27ef1ee
--- /dev/null
+++ b/dom/media/test/bug500311.ogv
Binary files differ
diff --git a/dom/media/test/bug500311.ogv^headers^ b/dom/media/test/bug500311.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug500311.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug501279.ogg b/dom/media/test/bug501279.ogg
new file mode 100644
index 0000000000..e266f07ee8
--- /dev/null
+++ b/dom/media/test/bug501279.ogg
Binary files differ
diff --git a/dom/media/test/bug501279.ogg^headers^ b/dom/media/test/bug501279.ogg^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug501279.ogg^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug504613.ogv b/dom/media/test/bug504613.ogv
new file mode 100644
index 0000000000..5c7fd015e9
--- /dev/null
+++ b/dom/media/test/bug504613.ogv
Binary files differ
diff --git a/dom/media/test/bug504613.ogv^headers^ b/dom/media/test/bug504613.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug504613.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug504644.ogv b/dom/media/test/bug504644.ogv
new file mode 100644
index 0000000000..46fb4a876b
--- /dev/null
+++ b/dom/media/test/bug504644.ogv
Binary files differ
diff --git a/dom/media/test/bug504644.ogv^headers^ b/dom/media/test/bug504644.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug504644.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug504843.ogv b/dom/media/test/bug504843.ogv
new file mode 100644
index 0000000000..94b4750865
--- /dev/null
+++ b/dom/media/test/bug504843.ogv
Binary files differ
diff --git a/dom/media/test/bug504843.ogv^headers^ b/dom/media/test/bug504843.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug504843.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug506094.ogv b/dom/media/test/bug506094.ogv
new file mode 100644
index 0000000000..142b7b9ad1
--- /dev/null
+++ b/dom/media/test/bug506094.ogv
Binary files differ
diff --git a/dom/media/test/bug506094.ogv^headers^ b/dom/media/test/bug506094.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug506094.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug516323.indexed.ogv b/dom/media/test/bug516323.indexed.ogv
new file mode 100644
index 0000000000..7bd76eeccc
--- /dev/null
+++ b/dom/media/test/bug516323.indexed.ogv
Binary files differ
diff --git a/dom/media/test/bug516323.indexed.ogv^headers^ b/dom/media/test/bug516323.indexed.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug516323.indexed.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug516323.ogv b/dom/media/test/bug516323.ogv
new file mode 100644
index 0000000000..8f2f38b983
--- /dev/null
+++ b/dom/media/test/bug516323.ogv
Binary files differ
diff --git a/dom/media/test/bug516323.ogv^headers^ b/dom/media/test/bug516323.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug516323.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug520493.ogg b/dom/media/test/bug520493.ogg
new file mode 100644
index 0000000000..6eb23198f4
--- /dev/null
+++ b/dom/media/test/bug520493.ogg
Binary files differ
diff --git a/dom/media/test/bug520493.ogg^headers^ b/dom/media/test/bug520493.ogg^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug520493.ogg^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug520500.ogg b/dom/media/test/bug520500.ogg
new file mode 100644
index 0000000000..b91d3dd97d
--- /dev/null
+++ b/dom/media/test/bug520500.ogg
Binary files differ
diff --git a/dom/media/test/bug520500.ogg^headers^ b/dom/media/test/bug520500.ogg^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug520500.ogg^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug520908.ogv b/dom/media/test/bug520908.ogv
new file mode 100644
index 0000000000..093158432a
--- /dev/null
+++ b/dom/media/test/bug520908.ogv
Binary files differ
diff --git a/dom/media/test/bug520908.ogv^headers^ b/dom/media/test/bug520908.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug520908.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug523816.ogv b/dom/media/test/bug523816.ogv
new file mode 100644
index 0000000000..ca9a31b6da
--- /dev/null
+++ b/dom/media/test/bug523816.ogv
Binary files differ
diff --git a/dom/media/test/bug523816.ogv^headers^ b/dom/media/test/bug523816.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug523816.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug533822.ogg b/dom/media/test/bug533822.ogg
new file mode 100644
index 0000000000..a8e506910e
--- /dev/null
+++ b/dom/media/test/bug533822.ogg
Binary files differ
diff --git a/dom/media/test/bug533822.ogg^headers^ b/dom/media/test/bug533822.ogg^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug533822.ogg^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug556821.ogv b/dom/media/test/bug556821.ogv
new file mode 100644
index 0000000000..8d76fee45e
--- /dev/null
+++ b/dom/media/test/bug556821.ogv
Binary files differ
diff --git a/dom/media/test/bug556821.ogv^headers^ b/dom/media/test/bug556821.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug556821.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug557094.ogv b/dom/media/test/bug557094.ogv
new file mode 100644
index 0000000000..b4fc0799a6
--- /dev/null
+++ b/dom/media/test/bug557094.ogv
Binary files differ
diff --git a/dom/media/test/bug557094.ogv^headers^ b/dom/media/test/bug557094.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug557094.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bug604067.webm b/dom/media/test/bug604067.webm
new file mode 100644
index 0000000000..86bdfdc91f
--- /dev/null
+++ b/dom/media/test/bug604067.webm
Binary files differ
diff --git a/dom/media/test/bug604067.webm^headers^ b/dom/media/test/bug604067.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/bug604067.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/bunny.webm b/dom/media/test/bunny.webm
new file mode 100644
index 0000000000..823439d1c5
--- /dev/null
+++ b/dom/media/test/bunny.webm
Binary files differ
diff --git a/dom/media/test/bunny_hd_5s.mp4 b/dom/media/test/bunny_hd_5s.mp4
new file mode 100644
index 0000000000..975a159574
--- /dev/null
+++ b/dom/media/test/bunny_hd_5s.mp4
Binary files differ
diff --git a/dom/media/test/can_play_type_dash.js b/dom/media/test/can_play_type_dash.js
new file mode 100644
index 0000000000..b4760545db
--- /dev/null
+++ b/dom/media/test/can_play_type_dash.js
@@ -0,0 +1,27 @@
+function check_dash(v, enabled) {
+ function check(type, expected) {
+ is(v.canPlayType(type), enabled ? expected : "", type);
+ }
+
+ // DASH types
+ check("application/dash+xml", "probably");
+
+ // Supported Webm codecs
+ check("application/dash+xml; codecs=vorbis", "probably");
+ check("application/dash+xml; codecs=vorbis", "probably");
+ check("application/dash+xml; codecs=vorbis,vp8", "probably");
+ check("application/dash+xml; codecs=vorbis,vp8.0", "probably");
+ check('application/dash+xml; codecs="vorbis,vp8"', "probably");
+ check('application/dash+xml; codecs="vorbis,vp8.0"', "probably");
+ check('application/dash+xml; codecs="vp8, vorbis"', "probably");
+ check('application/dash+xml; codecs="vp8.0, vorbis"', "probably");
+ check("application/dash+xml; codecs=vp8", "probably");
+ check("application/dash+xml; codecs=vp8.0", "probably");
+
+ // Unsupported codecs
+ check("application/dash+xml; codecs=xyz", "");
+ check("application/dash+xml; codecs=xyz,vorbis", "");
+ check("application/dash+xml; codecs=vorbis,xyz", "");
+ check("application/dash+xml; codecs=xyz,vp8.0", "");
+ check("application/dash+xml; codecs=vp8.0,xyz", "");
+}
diff --git a/dom/media/test/can_play_type_ogg.js b/dom/media/test/can_play_type_ogg.js
new file mode 100644
index 0000000000..79bf8a554c
--- /dev/null
+++ b/dom/media/test/can_play_type_ogg.js
@@ -0,0 +1,72 @@
+function check_ogg(v, enabled, finish) {
+ function check(type, expected) {
+ is(v.canPlayType(type), enabled ? expected : "", type);
+ }
+
+ function basic_test() {
+ return new Promise(function (resolve, reject) {
+ // Ogg types
+ check("video/ogg", "maybe");
+ check("audio/ogg", "maybe");
+ check("application/ogg", "maybe");
+
+ // Supported Ogg codecs
+ check("audio/ogg; codecs=vorbis", "probably");
+ check("video/ogg; codecs=vorbis", "probably");
+ check("video/ogg; codecs=vorbis,theora", "probably");
+ check('video/ogg; codecs="vorbis, theora"', "probably");
+ check("video/ogg; codecs=theora", "probably");
+
+ resolve();
+ });
+ }
+
+ // Verify Opus support
+ function verify_opus_support() {
+ return new Promise(function (resolve, reject) {
+ var OpusEnabled = SpecialPowers.getBoolPref(
+ "media.opus.enabled",
+ undefined
+ );
+ if (OpusEnabled != undefined) {
+ resolve();
+ } else {
+ console.log(
+ "media.opus.enabled pref not found; skipping Opus validation"
+ );
+ reject();
+ }
+ });
+ }
+
+ function opus_enable() {
+ return SpecialPowers.pushPrefEnv({
+ set: [["media.opus.enabled", true]],
+ }).then(function () {
+ check("audio/ogg; codecs=opus", "probably");
+ });
+ }
+
+ function opus_disable() {
+ return SpecialPowers.pushPrefEnv({
+ set: [["media.opus.enabled", false]],
+ }).then(function () {
+ check("audio/ogg; codecs=opus", "");
+ });
+ }
+
+ function unspported_ogg() {
+ // Unsupported Ogg codecs
+ check("video/ogg; codecs=xyz", "");
+ check("video/ogg; codecs=xyz,vorbis", "");
+ check("video/ogg; codecs=vorbis,xyz", "");
+
+ finish.call();
+ }
+
+ basic_test()
+ .then(verify_opus_support)
+ .then(opus_enable)
+ .then(opus_disable)
+ .then(unspported_ogg, unspported_ogg);
+}
diff --git a/dom/media/test/can_play_type_wave.js b/dom/media/test/can_play_type_wave.js
new file mode 100644
index 0000000000..a5e087aa40
--- /dev/null
+++ b/dom/media/test/can_play_type_wave.js
@@ -0,0 +1,30 @@
+function check_wave(v, enabled) {
+ function check(type, expected) {
+ is(v.canPlayType(type), enabled ? expected : "", type);
+ }
+
+ // Wave types
+ check("audio/wave", "maybe");
+ check("audio/wav", "maybe");
+ check("audio/x-wav", "maybe");
+ check("audio/x-pn-wav", "maybe");
+
+ // Supported Wave codecs
+ check("audio/wave; codecs=1", "probably");
+ check("audio/wave; codecs=3", "probably");
+ check("audio/wave; codecs=6", "probably");
+ check("audio/wave; codecs=7", "probably");
+ // "no codecs" should be supported, I guess
+ check("audio/wave; codecs=", "maybe");
+ check('audio/wave; codecs=""', "maybe");
+
+ // Unsupported Wave codecs
+ check("audio/wave; codecs=0", "");
+ check("audio/wave; codecs=2", "");
+ check("audio/wave; codecs=xyz,1", "");
+ check("audio/wave; codecs=1,xyz", "");
+ check('audio/wave; codecs="xyz, 1"', "");
+ // empty codec names
+ check("audio/wave; codecs=,", "");
+ check('audio/wave; codecs="0, 1,"', "");
+}
diff --git a/dom/media/test/can_play_type_webm.js b/dom/media/test/can_play_type_webm.js
new file mode 100644
index 0000000000..315a8ef3d9
--- /dev/null
+++ b/dom/media/test/can_play_type_webm.js
@@ -0,0 +1,39 @@
+async function check_webm(v, enabled) {
+ function check(type, expected) {
+ is(
+ v.canPlayType(type),
+ enabled ? expected : "",
+ type + "='" + expected + "'"
+ );
+ }
+
+ // WebM types
+ check("video/webm", "maybe");
+ check("audio/webm", "maybe");
+
+ var video = ["vp8", "vp8.0", "vp9", "vp9.0"];
+ var audio = ["vorbis", "opus"];
+
+ audio.forEach(function (acodec) {
+ check("audio/webm; codecs=" + acodec, "probably");
+ check("video/webm; codecs=" + acodec, "probably");
+ });
+ video.forEach(function (vcodec) {
+ check("video/webm; codecs=" + vcodec, "probably");
+ audio.forEach(function (acodec) {
+ check('video/webm; codecs="' + vcodec + ", " + acodec + '"', "probably");
+ check('video/webm; codecs="' + acodec + ", " + vcodec + '"', "probably");
+ });
+ });
+
+ // Unsupported WebM codecs
+ check("video/webm; codecs=xyz", "");
+ check("video/webm; codecs=xyz,vorbis", "");
+ check("video/webm; codecs=vorbis,xyz", "");
+
+ await SpecialPowers.pushPrefEnv({ set: [["media.av1.enabled", true]] });
+ check('video/webm; codecs="av1"', "probably");
+
+ await SpecialPowers.pushPrefEnv({ set: [["media.av1.enabled", false]] });
+ check('video/webm; codecs="av1"', "");
+}
diff --git a/dom/media/test/cancellable_request.sjs b/dom/media/test/cancellable_request.sjs
new file mode 100644
index 0000000000..ffeb0c7818
--- /dev/null
+++ b/dom/media/test/cancellable_request.sjs
@@ -0,0 +1,162 @@
+function parseQuery(request, key) {
+ var params = request.queryString.split("&");
+ for (var j = 0; j < params.length; ++j) {
+ var p = params[j];
+ if (p == key) {
+ return true;
+ }
+ if (p.indexOf(key + "=") == 0) {
+ return p.substring(key.length + 1);
+ }
+ if (!p.includes("=") && key == "") {
+ return p;
+ }
+ }
+ return false;
+}
+
+function push32BE(array, input) {
+ array.push(String.fromCharCode((input >> 24) & 0xff));
+ array.push(String.fromCharCode((input >> 16) & 0xff));
+ array.push(String.fromCharCode((input >> 8) & 0xff));
+ array.push(String.fromCharCode(input & 0xff));
+}
+
+function push32LE(array, input) {
+ array.push(String.fromCharCode(input & 0xff));
+ array.push(String.fromCharCode((input >> 8) & 0xff));
+ array.push(String.fromCharCode((input >> 16) & 0xff));
+ array.push(String.fromCharCode((input >> 24) & 0xff));
+}
+
+function push16LE(array, input) {
+ array.push(String.fromCharCode(input & 0xff));
+ array.push(String.fromCharCode((input >> 8) & 0xff));
+}
+
+function buildWave(samples, sample_rate) {
+ const RIFF_MAGIC = 0x52494646;
+ const WAVE_MAGIC = 0x57415645;
+ const FRMT_MAGIC = 0x666d7420;
+ const DATA_MAGIC = 0x64617461;
+ const RIFF_SIZE = 44;
+
+ var header = [];
+ push32BE(header, RIFF_MAGIC);
+ push32LE(header, RIFF_SIZE + samples.length * 2);
+ push32BE(header, WAVE_MAGIC);
+ push32BE(header, FRMT_MAGIC);
+ push32LE(header, 16);
+ push16LE(header, 1);
+ push16LE(header, 1);
+ push32LE(header, sample_rate);
+ push32LE(header, sample_rate);
+ push16LE(header, 2);
+ push16LE(header, 16);
+ push32BE(header, DATA_MAGIC);
+ push32LE(header, samples.length * 2);
+ for (var i = 0; i < samples.length; ++i) {
+ push16LE(header, samples[i], 2);
+ }
+ return header;
+}
+
+const CC = Components.Constructor;
+const Timer = CC("@mozilla.org/timer;1", "nsITimer", "initWithCallback");
+const BinaryOutputStream = CC(
+ "@mozilla.org/binaryoutputstream;1",
+ "nsIBinaryOutputStream",
+ "setOutputStream"
+);
+
+function poll(f) {
+ if (f()) {
+ return;
+ }
+ new Timer(
+ function () {
+ poll(f);
+ },
+ 100,
+ Ci.nsITimer.TYPE_ONE_SHOT
+ );
+}
+
+function handleRequest(request, response) {
+ var cancel = parseQuery(request, "cancelkey");
+ if (cancel) {
+ setState(cancel[1], "cancelled");
+ response.setStatusLine(request.httpVersion, 200, "OK");
+ response.write("Cancel approved!");
+ return;
+ }
+
+ var samples = [];
+ for (var i = 0; i < 1000000; ++i) {
+ samples.push(0);
+ }
+ var bytes = buildWave(samples, 44100).join("");
+
+ var key = parseQuery(request, "key");
+ response.setHeader("Content-Type", "audio/x-wav");
+ response.setHeader("Content-Length", "" + bytes.length, false);
+
+ var out = new BinaryOutputStream(response.bodyOutputStream);
+
+ var start = 0,
+ end = bytes.length - 1;
+ if (request.hasHeader("Range")) {
+ var rangeMatch = request.getHeader("Range").match(/^bytes=(\d+)?-(\d+)?$/);
+
+ if (rangeMatch[1] !== undefined) {
+ start = parseInt(rangeMatch[1], 10);
+ }
+
+ if (rangeMatch[2] !== undefined) {
+ end = parseInt(rangeMatch[2], 10);
+ }
+
+ // No start given, so the end is really the count of bytes from the
+ // end of the file.
+ if (start === undefined) {
+ start = Math.max(0, bytes.length - end);
+ end = bytes.length - 1;
+ }
+
+ // start and end are inclusive
+ if (end === undefined || end >= bytes.length) {
+ end = bytes.length - 1;
+ }
+
+ if (end < start) {
+ response.setStatusLine(request.httpVersion, 200, "OK");
+ start = 0;
+ end = bytes.length - 1;
+ } else {
+ response.setStatusLine(request.httpVersion, 206, "Partial Content");
+ var contentRange = "bytes " + start + "-" + end + "/" + bytes.length;
+ response.setHeader("Content-Range", contentRange);
+ }
+ }
+
+ if (start > 0) {
+ // Send all requested data
+ out.write(bytes.slice(start, end + 1), end + 1 - start);
+ return;
+ }
+
+ // Write the first 1.2M of the Wave file. We know the cache size is set to
+ // 100K so this will fill the cache and and cause a "suspend" event on
+ // the loading element.
+ out.write(bytes, 1200000);
+
+ response.processAsync();
+ // Now wait for the message to cancel this response
+ poll(function () {
+ if (getState(key[1]) != "cancelled") {
+ return false;
+ }
+ response.finish();
+ return true;
+ });
+}
diff --git a/dom/media/test/chain.ogg b/dom/media/test/chain.ogg
new file mode 100644
index 0000000000..3535b280f4
--- /dev/null
+++ b/dom/media/test/chain.ogg
Binary files differ
diff --git a/dom/media/test/chain.ogg^headers^ b/dom/media/test/chain.ogg^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/chain.ogg^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/chain.ogv b/dom/media/test/chain.ogv
new file mode 100644
index 0000000000..3e684b64a5
--- /dev/null
+++ b/dom/media/test/chain.ogv
Binary files differ
diff --git a/dom/media/test/chain.ogv^headers^ b/dom/media/test/chain.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/chain.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/chain.opus b/dom/media/test/chain.opus
new file mode 100644
index 0000000000..9fa67f94c3
--- /dev/null
+++ b/dom/media/test/chain.opus
Binary files differ
diff --git a/dom/media/test/chain.opus^headers^ b/dom/media/test/chain.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/chain.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/chained-audio-video.ogg b/dom/media/test/chained-audio-video.ogg
new file mode 100644
index 0000000000..adda68bb47
--- /dev/null
+++ b/dom/media/test/chained-audio-video.ogg
Binary files differ
diff --git a/dom/media/test/chained-audio-video.ogg^headers^ b/dom/media/test/chained-audio-video.ogg^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/chained-audio-video.ogg^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/chained-video.ogv b/dom/media/test/chained-video.ogv
new file mode 100644
index 0000000000..a6288ef6c9
--- /dev/null
+++ b/dom/media/test/chained-video.ogv
Binary files differ
diff --git a/dom/media/test/chained-video.ogv^headers^ b/dom/media/test/chained-video.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/chained-video.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/chrome/chrome.toml b/dom/media/test/chrome/chrome.toml
new file mode 100644
index 0000000000..57d1843841
--- /dev/null
+++ b/dom/media/test/chrome/chrome.toml
@@ -0,0 +1,12 @@
+[DEFAULT]
+subsuite = "media"
+support-files = [
+ "../gizmo.mp4",
+ "../gizmo-noaudio.mp4",
+ "../TestPatternHDR.mp4",
+ "../tone2s-silence4s-tone2s.opus",
+]
+
+["test_accumulated_play_time.html"]
+
+["test_telemetry_source_buffer_type.html"]
diff --git a/dom/media/test/chrome/test_accumulated_play_time.html b/dom/media/test/chrome/test_accumulated_play_time.html
new file mode 100644
index 0000000000..e59c0377b2
--- /dev/null
+++ b/dom/media/test/chrome/test_accumulated_play_time.html
@@ -0,0 +1,694 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+<title>Test Video Play Time Related Permanent Telemetry Probes</title>
+<script src="chrome://mochikit/content/tests/SimpleTest/SimpleTest.js"></script>
+<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+<script type="application/javascript">
+
+/**
+ * This test is used to ensure that we accumulate time for video playback
+ * correctly, and the results would be used in Telemetry probes.
+ * Currently this test covers following probes
+ * - VIDEO_PLAY_TIME_MS
+ * - VIDEO_HDR_PLAY_TIME_MS
+ * - VIDEO_HIDDEN_PLAY_TIME_MS
+ * - VIDEO_HIDDEN_PLAY_TIME_PERCENTAGE
+ * - VIDEO_INFERRED_DECODE_SUSPEND_PERCENTAGE
+ * - VIDEO_VISIBLE_PLAY_TIME_MS
+ * - MEDIA_PLAY_TIME_MS
+ * - MUTED_PLAY_TIME_PERCENT
+ * - AUDIBLE_PLAY_TIME_PERCENT
+ */
+const videoHistNames = [
+ "VIDEO_PLAY_TIME_MS",
+ "VIDEO_HIDDEN_PLAY_TIME_MS"
+];
+const videoHDRHistNames = [
+ "VIDEO_HDR_PLAY_TIME_MS"
+];
+const videoKeyedHistNames = [
+ "VIDEO_HIDDEN_PLAY_TIME_PERCENTAGE",
+ "VIDEO_INFERRED_DECODE_SUSPEND_PERCENTAGE",
+ "VIDEO_VISIBLE_PLAY_TIME_MS"
+];
+const audioKeyedHistNames = [
+ "MUTED_PLAY_TIME_PERCENT",
+ "AUDIBLE_PLAY_TIME_PERCENT"
+];
+
+add_task(async function setTestPref() {
+ await SpecialPowers.pushPrefEnv({
+ set: [["media.testing-only-events", true],
+ ["media.test.video-suspend", true],
+ ["media.suspend-background-video.enabled", true],
+ ["media.suspend-background-video.delay-ms", 0],
+ ["dom.media.silence_duration_for_audibility", 0.1]
+ ]});
+});
+
+add_task(async function testTotalPlayTime() {
+ const video = document.createElement('video');
+ video.src = "gizmo.mp4";
+ document.body.appendChild(video);
+
+ info(`all accumulated time should be zero`);
+ const videoChrome = SpecialPowers.wrap(video);
+ await new Promise(r => video.onloadeddata = r);
+ assertValueEqualTo(videoChrome, "totalVideoPlayTime", 0);
+ assertValueEqualTo(videoChrome, "invisiblePlayTime", 0);
+
+ info(`start accumulating play time after media starts`);
+ video.autoplay = true;
+ await Promise.all([
+ once(video, "playing"),
+ once(video, "moztotalplaytimestarted"),
+ ]);
+ await assertValueConstantlyIncreases(videoChrome, "totalVideoPlayTime");
+ assertValueKeptUnchanged(videoChrome, "invisiblePlayTime");
+ assertValueKeptUnchanged(videoChrome, "videoDecodeSuspendedTime");
+
+ info(`should not accumulate time for paused video`);
+ video.pause();
+ await once(video, "moztotalplaytimepaused");
+ assertValueKeptUnchanged(videoChrome, "totalVideoPlayTime");
+ assertValueEqualTo(videoChrome, "totalVideoPlayTime", 0);
+
+ info(`should start accumulating time again`);
+ let rv = await Promise.all([
+ onceWithTrueReturn(video, "moztotalplaytimestarted"),
+ video.play().then(_ => true, _ => false),
+ ]);
+ ok(returnTrueWhenAllValuesAreTrue(rv), "video started again");
+ await assertValueConstantlyIncreases(videoChrome, "totalVideoPlayTime");
+ await cleanUpMediaAndCheckTelemetry(video);
+});
+
+// The testHDRPlayTime task will only pass on platforms that accurately report
+// color depth in their VideoInfo structures. Presently, that is only true for
+// macOS.
+const {AppConstants} = ChromeUtils.importESModule(
+ "resource://gre/modules/AppConstants.sys.mjs"
+);
+const reportsColorDepthFromVideoData = (AppConstants.platform == "macosx");
+if (reportsColorDepthFromVideoData) {
+ add_task(async function testHDRPlayTime() {
+ // This task is different from the others because the HTMLMediaElement does
+ // not expose a chrome property for video hdr play time. But we do capture
+ // telemety for VIDEO_HDR_PLAY_TIME_MS. To ensure that this telemetry is
+ // generated, this task follows the same structure as the other tasks, but
+ // doesn't actually check the properties of the video player, other than to
+ // confirm that video has played for at least some time.
+ const video = document.createElement('video');
+ video.src = "TestPatternHDR.mp4"; // This is an HDR video with no audio.
+ document.body.appendChild(video);
+
+ info(`load the HDR video`);
+ const videoChrome = SpecialPowers.wrap(video);
+ await new Promise(r => video.onloadeddata = r);
+
+ info(`start accumulating play time after media starts`);
+ video.autoplay = true;
+ await Promise.all([
+ once(video, "playing"),
+ once(video, "moztotalplaytimestarted"),
+ ]);
+ // Check that we have at least some video play time, because the
+ // HDR play time telemetry is emitted by the same process.
+ await assertValueConstantlyIncreases(videoChrome, "totalVideoPlayTime");
+ await cleanUpMediaAndCheckTelemetry(video, {hasVideo: true, hasAudio: false, hasVideoHDR: true});
+ });
+}
+
+add_task(async function testVisiblePlayTime() {
+ const video = document.createElement('video');
+ video.src = "gizmo.mp4";
+ document.body.appendChild(video);
+
+ info(`all accumulated time should be zero`);
+ const videoChrome = SpecialPowers.wrap(video);
+ await new Promise(r => video.onloadeddata = r);
+ assertValueEqualTo(videoChrome, "totalVideoPlayTime", 0);
+ assertValueEqualTo(videoChrome, "visiblePlayTime", 0);
+ assertValueEqualTo(videoChrome, "invisiblePlayTime", 0);
+
+ info(`start accumulating play time after media starts`);
+ video.autoplay = true;
+ await Promise.all([
+ once(video, "playing"),
+ once(video, "moztotalplaytimestarted"),
+ ]);
+ await assertValueConstantlyIncreases(videoChrome, "totalVideoPlayTime");
+ await assertValueConstantlyIncreases(videoChrome, "visiblePlayTime");
+ assertValueKeptUnchanged(videoChrome, "invisiblePlayTime");
+
+ info(`make video invisible`);
+ video.style.display = "none";
+ await once(video, "mozinvisibleplaytimestarted");
+ await assertValueConstantlyIncreases(videoChrome, "totalVideoPlayTime");
+ await assertValueConstantlyIncreases(videoChrome, "invisiblePlayTime");
+ assertValueKeptUnchanged(videoChrome, "visiblePlayTime");
+ await cleanUpMediaAndCheckTelemetry(video);
+});
+
+add_task(async function testAudibleAudioPlayTime() {
+ const audio = document.createElement('audio');
+ audio.src = "tone2s-silence4s-tone2s.opus";
+ audio.controls = true;
+ audio.loop = true;
+ document.body.appendChild(audio);
+
+ info(`all accumulated time should be zero`);
+ const audioChrome = SpecialPowers.wrap(audio);
+ await new Promise(r => audio.onloadeddata = r);
+ assertValueEqualTo(audioChrome, "totalVideoPlayTime", 0);
+ assertValueEqualTo(audioChrome, "totalAudioPlayTime", 0);
+ assertValueEqualTo(audioChrome, "mutedPlayTime", 0);
+ assertValueEqualTo(audioChrome, "audiblePlayTime", 0);
+
+ info(`start accumulating play time after media starts`);
+ await Promise.all([
+ audio.play(),
+ once(audio, "moztotalplaytimestarted"),
+ ]);
+ await assertValueConstantlyIncreases(audioChrome, "totalAudioPlayTime");
+ await assertValueConstantlyIncreases(audioChrome, "audiblePlayTime");
+ assertValueKeptUnchanged(audioChrome, "mutedPlayTime");
+ assertValueKeptUnchanged(audioChrome, "totalVideoPlayTime");
+
+ info(`audio becomes inaudible for 4s`);
+ await once(audio, "mozinaudibleaudioplaytimestarted");
+
+ await assertValueConstantlyIncreases(audioChrome, "totalAudioPlayTime");
+ assertValueKeptUnchanged(audioChrome, "audiblePlayTime");
+ assertValueKeptUnchanged(audioChrome, "mutedPlayTime");
+ assertValueKeptUnchanged(audioChrome, "totalVideoPlayTime");
+
+ info(`audio becomes audible after 4s`);
+ await once(audio, "mozinaudibleaudioplaytimepaused");
+
+ await assertValueConstantlyIncreases(audioChrome, "totalAudioPlayTime");
+ await assertValueConstantlyIncreases(audioChrome, "audiblePlayTime");
+ assertValueKeptUnchanged(audioChrome, "mutedPlayTime");
+ assertValueKeptUnchanged(audioChrome, "totalVideoPlayTime");
+
+ await cleanUpMediaAndCheckTelemetry(audio, {hasVideo: false});
+});
+
+add_task(async function testHiddenPlayTime() {
+ const invisibleReasons = ["notInTree", "notInConnectedTree", "invisibleInDisplay"];
+ for (let reason of invisibleReasons) {
+ const video = document.createElement('video');
+ video.src = "gizmo.mp4";
+ video.loop = true;
+ info(`invisible video due to '${reason}'`);
+
+ if (reason == "notInConnectedTree") {
+ let disconnected = document.createElement("div")
+ disconnected.appendChild(video);
+ } else if (reason == "invisibleInDisplay") {
+ document.body.appendChild(video);
+ video.style.display = "none";
+ } else if (reason == "notInTree") {
+ // video is already created in the `notInTree` situation.
+ } else {
+ ok(false, "undefined reason");
+ }
+
+ info(`start invisible video should start accumulating timers`);
+ const videoChrome = SpecialPowers.wrap(video);
+ let rv = await Promise.all([
+ onceWithTrueReturn(video, "mozinvisibleplaytimestarted"),
+ video.play().then(_ => true, _ => false),
+ ]);
+ ok(returnTrueWhenAllValuesAreTrue(rv), "video started playing");
+ await assertValueConstantlyIncreases(videoChrome, "invisiblePlayTime");
+
+ info(`should not accumulate time for paused video`);
+ video.pause();
+ await once(video, "mozinvisibleplaytimepaused");
+ assertValueKeptUnchanged(videoChrome, "invisiblePlayTime");
+
+ info(`should start accumulating time again`);
+ rv = await Promise.all([
+ onceWithTrueReturn(video, "mozinvisibleplaytimestarted"),
+ video.play().then(_ => true, _ => false),
+ ]);
+ ok(returnTrueWhenAllValuesAreTrue(rv), "video started again");
+ await assertValueConstantlyIncreases(videoChrome, "invisiblePlayTime");
+
+ info(`make video visible should stop accumulating invisible related time`);
+ if (reason == "notInTree" || reason == "notInConnectedTree") {
+ document.body.appendChild(video);
+ } else if (reason == "invisibleInDisplay") {
+ video.style.display = "block";
+ } else {
+ ok(false, "undefined reason");
+ }
+ await once(video, "mozinvisibleplaytimepaused");
+ assertValueKeptUnchanged(videoChrome, "invisiblePlayTime");
+ await cleanUpMediaAndCheckTelemetry(video);
+ }
+});
+
+add_task(async function testAudioProbesWithoutAudio() {
+ const video = document.createElement('video');
+ video.src = "gizmo-noaudio.mp4";
+ video.loop = true;
+ document.body.appendChild(video);
+
+ info(`all accumulated time should be zero`);
+ const videoChrome = SpecialPowers.wrap(video);
+ await new Promise(r => video.onloadeddata = r);
+ assertValueEqualTo(videoChrome, "totalVideoPlayTime", 0);
+ assertValueEqualTo(videoChrome, "totalAudioPlayTime", 0);
+ assertValueEqualTo(videoChrome, "mutedPlayTime", 0);
+ assertValueEqualTo(videoChrome, "audiblePlayTime", 0);
+
+ info(`start accumulating play time after media starts`);
+ await Promise.all([
+ video.play(),
+ once(video, "moztotalplaytimestarted"),
+ ]);
+
+ async function checkInvariants() {
+ await assertValueConstantlyIncreases(videoChrome, "totalVideoPlayTime");
+ assertValueKeptUnchanged(videoChrome, "audiblePlayTime");
+ assertValueKeptUnchanged(videoChrome, "mutedPlayTime");
+ assertValueKeptUnchanged(videoChrome, "totalAudioPlayTime");
+ }
+
+ checkInvariants();
+
+ video.muted = true;
+
+ checkInvariants();
+
+ video.currentTime = 0.0;
+ await once(video, "seeked");
+
+ checkInvariants();
+
+ video.muted = false;
+
+ checkInvariants();
+
+ video.volume = 0.0;
+
+ checkInvariants();
+
+ video.volume = 1.0;
+
+ checkInvariants();
+
+ video.muted = true;
+
+ checkInvariants();
+
+ video.currentTime = 0.0;
+
+ checkInvariants();
+
+ await cleanUpMediaAndCheckTelemetry(video, {hasAudio: false});
+});
+
+add_task(async function testMutedAudioPlayTime() {
+ const audio = document.createElement('audio');
+ audio.src = "gizmo.mp4";
+ audio.controls = true;
+ audio.loop = true;
+ document.body.appendChild(audio);
+
+ info(`all accumulated time should be zero`);
+ const audioChrome = SpecialPowers.wrap(audio);
+ await new Promise(r => audio.onloadeddata = r);
+ assertValueEqualTo(audioChrome, "totalVideoPlayTime", 0);
+ assertValueEqualTo(audioChrome, "totalAudioPlayTime", 0);
+ assertValueEqualTo(audioChrome, "mutedPlayTime", 0);
+ assertValueEqualTo(audioChrome, "audiblePlayTime", 0);
+
+ info(`start accumulating play time after media starts`);
+ await Promise.all([
+ audio.play(),
+ once(audio, "moztotalplaytimestarted"),
+ ]);
+ await assertValueConstantlyIncreases(audioChrome, "totalAudioPlayTime");
+ await assertValueConstantlyIncreases(audioChrome, "audiblePlayTime");
+ assertValueKeptUnchanged(audioChrome, "mutedPlayTime");
+ assertValueKeptUnchanged(audioChrome, "totalVideoPlayTime");
+
+ audio.muted = true;
+ await once(audio, "mozmutedaudioplaytimestarted");
+
+ await assertValueConstantlyIncreases(audioChrome, "totalAudioPlayTime");
+ await assertValueConstantlyIncreases(audioChrome, "mutedPlayTime");
+ await assertValueConstantlyIncreases(audioChrome, "audiblePlayTime");
+ assertValueKeptUnchanged(audioChrome, "totalVideoPlayTime");
+
+ audio.currentTime = 0.0;
+ await once(audio, "seeked");
+
+ await assertValueConstantlyIncreases(audioChrome, "totalAudioPlayTime");
+ await assertValueConstantlyIncreases(audioChrome, "mutedPlayTime");
+ await assertValueConstantlyIncreases(audioChrome, "audiblePlayTime");
+ assertValueKeptUnchanged(audioChrome, "totalVideoPlayTime");
+
+ audio.muted = false;
+ await once(audio, "mozmutedeaudioplaytimepaused");
+
+ await assertValueConstantlyIncreases(audioChrome, "totalAudioPlayTime");
+ await assertValueConstantlyIncreases(audioChrome, "audiblePlayTime");
+ assertValueKeptUnchanged(audioChrome, "mutedPlayTime");
+ assertValueKeptUnchanged(audioChrome, "totalVideoPlayTime");
+
+ audio.volume = 0.0;
+ await once(audio, "mozmutedaudioplaytimestarted");
+
+ await assertValueConstantlyIncreases(audioChrome, "totalAudioPlayTime");
+ await assertValueConstantlyIncreases(audioChrome, "mutedPlayTime");
+ await assertValueConstantlyIncreases(audioChrome, "audiblePlayTime");
+ assertValueKeptUnchanged(audioChrome, "totalVideoPlayTime");
+
+ audio.volume = 1.0;
+ await once(audio, "mozmutedeaudioplaytimepaused");
+
+ await assertValueConstantlyIncreases(audioChrome, "totalAudioPlayTime");
+ await assertValueConstantlyIncreases(audioChrome, "audiblePlayTime");
+ assertValueKeptUnchanged(audioChrome, "mutedPlayTime");
+ assertValueKeptUnchanged(audioChrome, "totalVideoPlayTime");
+
+ audio.muted = true;
+ await once(audio, "mozmutedaudioplaytimestarted");
+
+ await assertValueConstantlyIncreases(audioChrome, "totalAudioPlayTime");
+ await assertValueConstantlyIncreases(audioChrome, "mutedPlayTime");
+ await assertValueConstantlyIncreases(audioChrome, "audiblePlayTime");
+ assertValueKeptUnchanged(audioChrome, "totalVideoPlayTime");
+
+ audio.currentTime = 0.0;
+
+ await assertValueConstantlyIncreases(audioChrome, "totalAudioPlayTime");
+ await assertValueConstantlyIncreases(audioChrome, "mutedPlayTime");
+ await assertValueConstantlyIncreases(audioChrome, "audiblePlayTime");
+ assertValueKeptUnchanged(audioChrome, "totalVideoPlayTime");
+
+ // The media has a video track, but it's being played back in an
+ // HTMLAudioElement, without video frame location.
+ await cleanUpMediaAndCheckTelemetry(audio, {hasVideo: false});
+});
+
+// Note that video suspended time is not always align with the invisible play
+// time even if `media.suspend-background-video.delay-ms` is `0`, because not all
+// invisible videos would be suspended under current strategy.
+add_task(async function testDecodeSuspendedTime() {
+ const video = document.createElement('video');
+ video.src = "gizmo.mp4";
+ video.loop = true;
+ document.body.appendChild(video);
+
+ info(`start video should start accumulating timers`);
+ const videoChrome = SpecialPowers.wrap(video);
+ let rv = await Promise.all([
+ onceWithTrueReturn(video, "moztotalplaytimestarted"),
+ video.play().then(_ => true, _ => false),
+ ]);
+ ok(returnTrueWhenAllValuesAreTrue(rv), "video started playing");
+ await assertValueConstantlyIncreases(videoChrome, "totalVideoPlayTime");
+ assertValueKeptUnchanged(videoChrome, "invisiblePlayTime");
+ assertValueKeptUnchanged(videoChrome, "videoDecodeSuspendedTime");
+
+ info(`make it invisible and force to suspend decoding`);
+ video.setVisible(false);
+ await once(video, "mozvideodecodesuspendedstarted");
+ await assertValueConstantlyIncreases(videoChrome, "totalVideoPlayTime");
+ await assertValueConstantlyIncreases(videoChrome, "invisiblePlayTime");
+ await assertValueConstantlyIncreases(videoChrome, "videoDecodeSuspendedTime");
+
+ info(`make it visible and resume decoding`);
+ video.setVisible(true);
+ await Promise.all([
+ once(video, "mozinvisibleplaytimepaused"),
+ once(video, "mozvideodecodesuspendedpaused"),
+ ]);
+ await assertValueConstantlyIncreases(videoChrome, "totalVideoPlayTime");
+ assertValueKeptUnchanged(videoChrome, "invisiblePlayTime");
+ assertValueKeptUnchanged(videoChrome, "videoDecodeSuspendedTime");
+ await cleanUpMediaAndCheckTelemetry(video);
+});
+
+add_task(async function reuseSameElementForPlayback() {
+ const video = document.createElement('video');
+ video.src = "gizmo.mp4";
+ document.body.appendChild(video);
+
+ info(`start accumulating play time after media starts`);
+ const videoChrome = SpecialPowers.wrap(video);
+ let rv = await Promise.all([
+ onceWithTrueReturn(video, "moztotalplaytimestarted"),
+ video.play().then(_ => true, _ => false),
+ ]);
+ ok(returnTrueWhenAllValuesAreTrue(rv), "video started again");
+ await assertValueConstantlyIncreases(videoChrome, "totalVideoPlayTime");
+
+ info(`reset its src and all accumulated value should be reset after then`);
+ // After setting its src to nothing, that would trigger a failed load and set
+ // the error. If the following step tries to set the new resource and `play()`
+ // , then they should be done after receving the `error` from that failed load
+ // first.
+ await Promise.all([
+ once(video, "error"),
+ cleanUpMediaAndCheckTelemetry(video),
+ ]);
+ // video doesn't have a decoder, so the return value would be -1 (error).
+ assertValueEqualTo(videoChrome, "totalVideoPlayTime", -1);
+ assertValueEqualTo(videoChrome, "invisiblePlayTime", -1);
+
+ info(`resue same element, make it visible and start playback again`);
+ video.src = "gizmo.mp4";
+ rv = await Promise.all([
+ onceWithTrueReturn(video, "moztotalplaytimestarted"),
+ video.play().then(_ => true, _ => false),
+ ]);
+ ok(returnTrueWhenAllValuesAreTrue(rv), "video started");
+ await assertValueConstantlyIncreases(videoChrome, "totalVideoPlayTime");
+ await cleanUpMediaAndCheckTelemetry(video);
+});
+
+add_task(async function testNoReportedTelemetryResult() {
+ info(`No result for empty video`);
+ const video = document.createElement('video');
+ assertAllProbeRelatedAttributesKeptUnchanged(video);
+ await assertNoReportedTelemetryResult(video);
+
+ info(`No result for video which hasn't started playing`);
+ video.src = "gizmo.mp4";
+ document.body.appendChild(video);
+ ok(await once(video, "loadeddata").then(_ => true), "video loaded data");
+ assertAllProbeRelatedAttributesKeptUnchanged(video);
+ await assertNoReportedTelemetryResult(video);
+
+ info(`No result for video with error`);
+ video.src = "filedoesnotexist.mp4";
+ ok(await video.play().then(_ => false, _ => true), "video failed to play");
+ ok(video.error != undefined, "video got error");
+ assertAllProbeRelatedAttributesKeptUnchanged(video);
+ await assertNoReportedTelemetryResult(video);
+});
+
+/**
+ * Following are helper functions
+ */
+async function cleanUpMediaAndCheckTelemetry(media, { reportExpected = true, hasVideo = true, hasAudio = true, hasVideoHDR = false } = {}) {
+ media.src = "";
+ await checkReportedTelemetry(media, reportExpected, hasVideo, hasAudio, hasVideoHDR);
+}
+
+async function assertNoReportedTelemetryResult(media) {
+ await checkReportedTelemetry(media, false, true, true);
+}
+
+async function checkReportedTelemetry(media, reportExpected, hasVideo, hasAudio, hasVideoHDR) {
+ const reportResultPromise = once(media, "mozreportedtelemetry");
+ info(`check telemetry result, reportExpected=${reportExpected}`);
+ if (reportExpected) {
+ await reportResultPromise;
+ }
+ for (const name of videoHistNames) {
+ try {
+ const hist = SpecialPowers.Services.telemetry.getHistogramById(name);
+ /**
+ * Histogram's snapshot looks like that
+ * {
+ * "bucket_count": X,
+ * "histogram_type": Y,
+ * "sum": Z,
+ * "range": [min, max],
+ * "values": { "value1" : "num1", "value2" : "num2", ...}
+ * }
+ */
+ const entriesNums = Object.entries(hist.snapshot().values).length;
+ if (reportExpected && hasVideo) {
+ ok(entriesNums > 0, `Reported result for ${name}`);
+ } else {
+ ok(entriesNums == 0, `Reported nothing for ${name}`);
+ }
+ hist.clear();
+ } catch (e) {
+ ok(false , `histogram '${name}' doesn't exist`);
+ }
+ }
+ // videoHDRHistNames are checked for total time, not for number of samples.
+ for (const name of videoHDRHistNames) {
+ try {
+ const hist = SpecialPowers.Services.telemetry.getHistogramById(name);
+ const totalTimeMS = hist.snapshot().sum;
+ if (reportExpected && hasVideoHDR) {
+ ok(totalTimeMS > 0, `Reported some time for ${name}`);
+ } else {
+ ok(totalTimeMS == 0, `Reported no time for ${name}`);
+ }
+ hist.clear();
+ } catch (e) {
+ ok(false , `histogram '${name}' doesn't exist`);
+ }
+ }
+ for (const name of videoKeyedHistNames) {
+ try {
+ const hist = SpecialPowers.Services.telemetry.getKeyedHistogramById(name);
+ /**
+ * Keyed Histogram's snapshot looks like that
+ * {
+ * "Key1" : {
+ * "bucket_count": X,
+ * "histogram_type": Y,
+ * "sum": Z,
+ * "range": [min, max],
+ * "values": { "value1" : "num1", "value2" : "num2", ...}
+ * },
+ * "Key2" : {...},
+ * }
+ */
+ const items = Object.entries(hist.snapshot());
+ if (items.length) {
+ for (const [key, value] of items) {
+ const entriesNums = Object.entries(value.values).length;
+ ok(reportExpected && entriesNums > 0, `Reported ${key} for ${name}`);
+ }
+ } else if (reportExpected) {
+ ok(!hasVideo, `No video telemetry reported but no video track in the media`);
+ } else {
+ ok(true, `No video telemetry expected, none reported`);
+ }
+ // Avoid to pollute next test task.
+ hist.clear();
+ } catch (e) {
+ ok(false , `keyed histogram '${name}' doesn't exist`);
+ }
+ }
+
+ // In any case, the combined probe MEDIA_PLAY_TIME_MS should be reported, if
+ // expected
+ {
+ const hist =
+ SpecialPowers.Services.telemetry.getKeyedHistogramById("MEDIA_PLAY_TIME_MS");
+ const items = Object.entries(hist.snapshot());
+ if (items.length) {
+ for (const item of items) {
+ ok(item[0].includes("V") != -1 || !hasVideo, "Video time is reported if video was present");
+ }
+ hist.clear();
+ } else {
+ ok(!reportExpected, "MEDIA_PLAY_TIME_MS should always be reported if a report is expected");
+ }
+ }
+
+ for (const name of audioKeyedHistNames) {
+ try {
+ const hist = SpecialPowers.Services.telemetry.getKeyedHistogramById(name);
+ const items = Object.entries(hist.snapshot());
+ if (items.length) {
+ for (const [key, value] of items) {
+ const entriesNums = Object.entries(value.values).length;
+ ok(reportExpected && entriesNums > 0, `Reported ${key} for ${name}`);
+ }
+ } else {
+ ok(!reportExpected || !hasAudio, `No audio telemetry expected, none reported`);
+ }
+ // Avoid to pollute next test task.
+ hist.clear();
+ } catch (e) {
+ ok(false , `keyed histogram '${name}' doesn't exist`);
+ }
+ }
+}
+
+function once(target, name) {
+ return new Promise(r => target.addEventListener(name, r, { once: true }));
+}
+
+function onceWithTrueReturn(target, name) {
+ return once(target, name).then(_ => true);
+}
+
+function returnTrueWhenAllValuesAreTrue(arr) {
+ for (let val of arr) {
+ if (!val) {
+ return false;
+ }
+ }
+ return true;
+}
+
+// Block the main thread for a number of milliseconds
+function blockMainThread(durationMS) {
+ const start = Date.now();
+ while (Date.now() - start < durationMS) { /* spin */ }
+}
+
+// Allows comparing two values from the system clocks that are not gathered
+// atomically. Allow up to 1ms of fuzzing when lhs and rhs are seconds.
+function timeFuzzyEquals(lhs, rhs, str) {
+ ok(Math.abs(lhs - rhs) < 1e-3, str);
+}
+
+function assertAttributeDefined(mediaChrome, checkType) {
+ ok(mediaChrome[checkType] != undefined, `${checkType} exists`);
+}
+
+function assertValueEqualTo(mediaChrome, checkType, expectedValue) {
+ assertAttributeDefined(mediaChrome, checkType);
+ is(mediaChrome[checkType], expectedValue, `${checkType} equals to ${expectedValue}`);
+}
+
+async function assertValueConstantlyIncreases(mediaChrome, checkType) {
+ assertAttributeDefined(mediaChrome, checkType);
+ const valueSnapshot = mediaChrome[checkType];
+ // 30ms is long enough to have a low-resolution system clock tick, but short
+ // enough to not slow the test down.
+ blockMainThread(30);
+ const current = mediaChrome[checkType];
+ ok(current > valueSnapshot, `${checkType} keeps increasing (${current} > ${valueSnapshot})`);
+}
+
+function assertValueKeptUnchanged(mediaChrome, checkType) {
+ assertAttributeDefined(mediaChrome, checkType);
+ const valueSnapshot = mediaChrome[checkType];
+ // 30ms is long enough to have a low-resolution system clock tick, but short
+ // enough to not slow the test down.
+ blockMainThread(30);
+ const newValue = mediaChrome[checkType];
+ timeFuzzyEquals(newValue, valueSnapshot, `${checkType} keeps unchanged (${newValue} vs. ${valueSnapshot})`);
+}
+
+function assertAllProbeRelatedAttributesKeptUnchanged(video) {
+ const videoChrome = SpecialPowers.wrap(video);
+ assertValueKeptUnchanged(videoChrome, "totalVideoPlayTime");
+ assertValueKeptUnchanged(videoChrome, "invisiblePlayTime");
+ assertValueKeptUnchanged(videoChrome, "videoDecodeSuspendedTime");
+}
+
+</script>
+</head>
+<body>
+</body>
+</html>
diff --git a/dom/media/test/chrome/test_telemetry_source_buffer_type.html b/dom/media/test/chrome/test_telemetry_source_buffer_type.html
new file mode 100644
index 0000000000..85ccfd1000
--- /dev/null
+++ b/dom/media/test/chrome/test_telemetry_source_buffer_type.html
@@ -0,0 +1,105 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+<title>Telemetry : test permanent probe MSE_SOURCE_BUFFER_TYPE</title>
+<script src="chrome://mochikit/content/tests/SimpleTest/SimpleTest.js"></script>
+<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+<script type="application/javascript">
+
+/**
+ * This test is used for the permanent Telemetry probe MSE_SOURCE_BUFFER_TYPE.
+ * which is used to measure the usage of mime type used in MSE.
+ */
+const VIDEO_HEVC = "video/hevc";
+const VIDEO_WEBM = "video/webm";
+const AUDIO_WEBM = "audio/webm";
+const VIDEO_MP4 = "video/mp4";
+const AUDIO_MP4 = "audio/mp4";
+const VIDEO_MP2T = "video/mp2t";
+const AUDIO_MP2T = "audio/mp2t";
+const AUDIO_MP3 = "audio/mpeg";
+const AUDIO_AAC = "audio/aac";
+
+// The order follows `MSE_SOURCE_BUFFER_TYPE` in `Histogram.json`.
+const gLabelNames = [VIDEO_HEVC, VIDEO_WEBM, AUDIO_WEBM, VIDEO_MP4, AUDIO_MP4,
+ VIDEO_MP2T, AUDIO_MP2T, AUDIO_MP3, AUDIO_AAC];
+
+add_task(async function setTestPref() {
+ await SpecialPowers.pushPrefEnv({
+ set: [["media.testing-only-events", true]]});
+});
+
+add_task(async function testSourceBufferTypeProbe() {
+ for (let label of gLabelNames) {
+ // HEVC will be tested in another task.
+ if (label == VIDEO_HEVC) {
+ continue;
+ }
+ info(`calling 'isTypeSupported()' should increase the probe value`);
+ MediaSource.isTypeSupported(label);
+ assertLabelValueEqualTo(label, 1);
+
+ info(`calling 'addSourceBuffer()' should increase the probe value`);
+ try {
+ const source = new MediaSource();
+ source.addSourceBuffer(label);
+ } catch (e) {
+ info(`ignore error for unsupported type ${label}`);
+ }
+ assertLabelValueEqualTo(label, 2);
+ }
+});
+
+add_task(async function testSourceBufferHEVCTypeProbe() {
+ // These are all possible mime types for HEVC
+ const mimeTypes = [
+ "video/mp4; codecs=\"hvc1\"",
+ "video/mp4; codecs=\"hev1\"",
+ "video/mp4; codecs=\"hvc1.1.6.L93.B0\"",
+ "video/mp4; codecs=\"hev1.1.6.L93.B0\"",
+ ];
+ for (let idx = 0; idx < mimeTypes.length; idx++) {
+ const mimeType = mimeTypes[idx];
+ info(`calling 'isTypeSupported()' should increase the probe value`);
+ MediaSource.isTypeSupported(mimeType);
+ assertLabelValueEqualTo(VIDEO_HEVC, idx * 2 + 1);
+
+ info(`calling 'addSourceBuffer()' should increase the probe value`);
+ try {
+ const source = new MediaSource();
+ source.addSourceBuffer(mimeType);
+ } catch (e) {
+ info(`ignore error for unsupported type ${mimeType}`);
+ }
+ assertLabelValueEqualTo(VIDEO_HEVC, idx * 2 + 2);
+ }
+});
+
+/**
+ * Following are helper functions
+ */
+async function assertLabelValueEqualTo(labelName, value) {
+ const hist = SpecialPowers.Services.telemetry.getHistogramById("MSE_SOURCE_BUFFER_TYPE");
+ /**
+ * Histogram's snapshot looks like that
+ * {
+ * "bucket_count": X,
+ * "histogram_type": Y,
+ * "sum": Z,
+ * "range": [min, max],
+ * "values": { "value1" : "num1", "value2" : "num2", ...}
+ * }
+ */
+ if (!gLabelNames.includes(labelName)) {
+ ok(false, `undefined label name=${labelName}`);
+ return;
+ }
+ const labelIdx = gLabelNames.indexOf(labelName);
+ is(hist.snapshot().values[labelIdx], value, `${labelName} equal to ${value}`);
+}
+
+</script>
+</head>
+<body>
+</body>
+</html>
diff --git a/dom/media/test/chromeHelper.js b/dom/media/test/chromeHelper.js
new file mode 100644
index 0000000000..a89d12d6dd
--- /dev/null
+++ b/dom/media/test/chromeHelper.js
@@ -0,0 +1,23 @@
+/* -*- Mode: javascript; indent-tabs-mode: nil; js-indent-level: 2 -*- */
+/* vim: set ft=javascript ts=2 et sw=2 tw=80: */
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+/* eslint-env mozilla/chrome-script */
+
+"use strict";
+
+// eslint-disable-next-line mozilla/use-services
+const dirSvc = Cc["@mozilla.org/file/directory_service;1"].getService(
+ Ci.nsIProperties
+);
+
+addMessageListener("media-test:getcwd", () => {
+ let cwd;
+ try {
+ cwd = dirSvc.get("CurWorkD", Ci.nsIFile).path;
+ } finally {
+ sendAsyncMessage("media-test:cwd", cwd);
+ }
+});
diff --git a/dom/media/test/cloneElementVisually_helpers.js b/dom/media/test/cloneElementVisually_helpers.js
new file mode 100644
index 0000000000..52e6cca36e
--- /dev/null
+++ b/dom/media/test/cloneElementVisually_helpers.js
@@ -0,0 +1,232 @@
+const TEST_VIDEO_1 =
+ "http://mochi.test:8888/tests/dom/media/test/bipbop_225w_175kbps.mp4";
+const TEST_VIDEO_2 =
+ "http://mochi.test:8888/tests/dom/media/test/pixel_aspect_ratio.mp4";
+const LONG_VIDEO = "http://mochi.test:8888/tests/dom/media/test/gizmo.mp4";
+
+/**
+ * Ensure that the original <video> is prepped and ready to play
+ * before starting any other tests.
+ */
+async function setup() {
+ await SpecialPowers.pushPrefEnv({
+ set: [
+ ["media.test.video-suspend", true],
+ ["media.suspend-background-video.enabled", true],
+ ["media.suspend-background-video.delay-ms", 500],
+ ["media.dormant-on-pause-timeout-ms", 0],
+ ["media.cloneElementVisually.testing", true],
+ ],
+ });
+
+ let originalVideo = document.getElementById("original");
+ await setVideoSrc(originalVideo, TEST_VIDEO_1);
+}
+
+/**
+ * Given a canvas, as well as a width and height of something to be
+ * blitted onto that canvas, makes any necessary adjustments to the
+ * canvas to work with the current display resolution.
+ *
+ * @param canvas (<canvas> element)
+ * The canvas to be adjusted.
+ * @param width (int)
+ * The width of the image to be blitted.
+ * @param height (int)
+ * The height of the image to be blitted.
+ * @return CanvasRenderingContext2D (SpecialPowers wrapper)
+ */
+function getWrappedScaledCanvasContext(canvas, width, height) {
+ let devRatio = window.devicePixelRatio || 1;
+ let ctx = SpecialPowers.wrap(canvas.getContext("2d"));
+ let backingRatio = ctx.webkitBackingStorePixelRatio || 1;
+
+ let ratio = 1;
+ ratio = devRatio / backingRatio;
+ canvas.width = ratio * width;
+ canvas.height = ratio * height;
+ canvas.style.width = width + "px";
+ canvas.style.height = height + "px";
+ ctx.scale(ratio, ratio);
+
+ return ctx;
+}
+
+/**
+ * Given a <video> element in the DOM, figures out its location, and captures
+ * a snapshot of what it's currently presenting.
+ *
+ * @param video (<video> element)
+ * @return ImageData (SpecialPowers wrapper)
+ */
+function captureFrameImageData(video) {
+ // Flush layout first, just in case the decoder has recently
+ // updated the dimensions of the video.
+ let rect = video.getBoundingClientRect();
+
+ let width = video.videoWidth;
+ let height = video.videoHeight;
+
+ let canvas = document.createElement("canvas");
+ let ctx = getWrappedScaledCanvasContext(canvas, width, height);
+ ctx.drawWindow(window, rect.left, rect.top, width, height, "rgb(0,0,0)");
+
+ return ctx.getImageData(0, 0, width, height);
+}
+
+/**
+ * Given two <video> elements, captures snapshots of what they're currently
+ * displaying, and asserts that they're identical.
+ *
+ * @param video1 (<video> element)
+ * A video element to compare against.
+ * @param video2 (<video> element)
+ * A video to compare with video1.
+ * @return Promise
+ * @resolves
+ * Resolves as true if the two videos match.
+ */
+async function assertVideosMatch(video1, video2) {
+ let video1Frame = captureFrameImageData(video1);
+ let video2Frame = captureFrameImageData(video2);
+
+ let left = document.getElementById("left");
+ let leftCtx = getWrappedScaledCanvasContext(
+ left,
+ video1Frame.width,
+ video1Frame.height
+ );
+ leftCtx.putImageData(video1Frame, 0, 0);
+
+ let right = document.getElementById("right");
+ let rightCtx = getWrappedScaledCanvasContext(
+ right,
+ video2Frame.width,
+ video2Frame.height
+ );
+ rightCtx.putImageData(video2Frame, 0, 0);
+
+ if (video1Frame.data.length != video2Frame.data.length) {
+ return false;
+ }
+
+ let leftDataURL = left.toDataURL();
+ let rightDataURL = right.toDataURL();
+
+ if (leftDataURL != rightDataURL) {
+ dump("Left frame: " + leftDataURL + "\n\n");
+ dump("Right frame: " + rightDataURL + "\n\n");
+
+ return false;
+ }
+
+ return true;
+}
+
+/**
+ * Testing helper function that constructs a node clone of a video,
+ * injects it into the DOM, and then runs an async testing function.
+ * This also does the work of removing the clone before resolving.
+ *
+ * @param video (<video> element)
+ * The video to clone the node from.
+ * @param asyncFn (async function)
+ * A test function that will be passed the new clone as its
+ * only argument.
+ * @return Promise
+ * @resolves
+ * When the asyncFn resolves and the clone has been removed
+ * from the DOM.
+ */
+async function withNewClone(video, asyncFn) {
+ let clone = video.cloneNode();
+ clone.id = "clone";
+ clone.src = "";
+ let content = document.getElementById("content");
+ content.appendChild(clone);
+
+ try {
+ await asyncFn(clone);
+ } finally {
+ clone.remove();
+ }
+}
+
+/**
+ * Sets the src on a video and waits until its ready.
+ *
+ * @param video (<video> element)
+ * The video to set the src on.
+ * @param src (string)
+ * The URL to set as the source on a video.
+ * @return Promise
+ * @resolves
+ * When the video fires the "canplay" event.
+ */
+async function setVideoSrc(video, src) {
+ let promiseReady = waitForEventOnce(video, "canplay");
+ video.src = src;
+ await promiseReady;
+}
+
+/**
+ * Returns a Promise once target emits a particular event
+ * once.
+ *
+ * @param target (DOM node)
+ * The target to monitor for the event.
+ * @param event (string)
+ * The name of the event to wait for.
+ * @return Promise
+ * @resolves
+ * When the event fires, and resolves to the event.
+ */
+function waitForEventOnce(target, event) {
+ return new Promise(resolve => {
+ target.addEventListener(event, resolve, { once: true });
+ });
+}
+
+/**
+ * Polls the video debug data as a hacky way of knowing when
+ * when the decoders have shut down.
+ *
+ * @param video (<video> element)
+ * @return Promise
+ * @resolves
+ * When the decoder has shut down.
+ */
+function waitForShutdownDecoder(video) {
+ return SimpleTest.promiseWaitForCondition(async () => {
+ let readerData = await SpecialPowers.wrap(video).mozRequestDebugInfo();
+ return readerData.decoder.reader.audioDecoderName == "shutdown";
+ }, "Video decoder should eventually shut down.");
+}
+
+/**
+ * Ensures that both hiding and pausing the video causes the
+ * video to suspend and make dormant its decoders, respectively.
+ *
+ * @param video (<video element)
+ */
+async function ensureVideoSuspendable(video) {
+ video = SpecialPowers.wrap(video);
+
+ ok(!video.hasSuspendTaint(), "Should be suspendable");
+
+ // First, we'll simulate putting the video in the background by
+ // making it invisible.
+ let suspendPromise = waitForEventOnce(video, "mozentervideosuspend");
+ video.setVisible(false);
+ await suspendPromise;
+ ok(true, "Suspended after the video was made invisible.");
+ video.setVisible(true);
+
+ ok(!video.hasSuspendTaint(), "Should still be suspendable.");
+
+ // Next, we'll pause the video.
+ await video.pause();
+ await waitForShutdownDecoder(video);
+ ok(true, "Shutdown decoder after the video was paused.");
+ await video.play();
+}
diff --git a/dom/media/test/contentType.sjs b/dom/media/test/contentType.sjs
new file mode 100644
index 0000000000..5a371564cf
--- /dev/null
+++ b/dom/media/test/contentType.sjs
@@ -0,0 +1,77 @@
+// Parse the query string, and give us the value for a certain key, or false if
+// it does not exist.
+function parseQuery(request, key) {
+ var params = request.queryString.split("?")[0].split("&");
+ for (var j = 0; j < params.length; ++j) {
+ var p = params[j];
+ if (p == key) {
+ return true;
+ }
+ if (p.indexOf(key + "=") == 0) {
+ return p.substring(key.length + 1);
+ }
+ if (!p.includes("=") && key == "") {
+ return p;
+ }
+ }
+ return false;
+}
+
+function handleRequest(request, response) {
+ try {
+ // Get the filename to send back.
+ var filename = parseQuery(request, "file");
+
+ var file = Services.dirsvc.get("CurWorkD", Ci.nsIFile);
+ var fis = Cc["@mozilla.org/network/file-input-stream;1"].createInstance(
+ Ci.nsIFileInputStream
+ );
+ var bis = Cc["@mozilla.org/binaryinputstream;1"].createInstance(
+ Ci.nsIBinaryInputStream
+ );
+ var paths = "tests/dom/media/test/" + filename;
+ dump(paths + "\n");
+ var split = paths.split("/");
+ for (var i = 0; i < split.length; ++i) {
+ file.append(split[i]);
+ }
+ fis.init(file, -1, -1, false);
+
+ // handle range requests
+ var partialstart = 0,
+ partialend = file.fileSize - 1;
+ if (request.hasHeader("Range")) {
+ var range = request.getHeader("Range");
+ var parts = range.replace(/bytes=/, "").split("-");
+ partialstart = parts[0];
+ partialend = parts[1];
+ if (!partialend.length) {
+ partialend = file.fileSize - 1;
+ }
+ response.setStatusLine(request.httpVersion, 206, "Partial Content");
+ var contentRange =
+ "bytes " + partialstart + "-" + partialend + "/" + file.fileSize;
+ response.setHeader("Content-Range", contentRange);
+ }
+
+ fis.seek(Ci.nsISeekableStream.NS_SEEK_SET, partialstart);
+ bis.setInputStream(fis);
+
+ var sendContentType = parseQuery(request, "nomime");
+ if (!sendContentType) {
+ var contentType = parseQuery(request, "type");
+ if (!contentType) {
+ // This should not happen.
+ dump("No type specified without having 'nomime' in parameters.");
+ return;
+ }
+ response.setHeader("Content-Type", contentType, false);
+ }
+ response.setHeader("Content-Length", "" + bis.available(), false);
+
+ var bytes = bis.readBytes(bis.available());
+ response.write(bytes, bytes.length);
+ } catch (e) {
+ dump("ERROR : " + e + "\n");
+ }
+}
diff --git a/dom/media/test/crashtests/0-timescale.html b/dom/media/test/crashtests/0-timescale.html
new file mode 100644
index 0000000000..db845096dd
--- /dev/null
+++ b/dom/media/test/crashtests/0-timescale.html
@@ -0,0 +1,14 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<head>
+</head>
+<body>
+<!-- This video file has a timescale of 0 in the mdhd atom -->
+<video src="0-timescale.mp4"
+ autoplay
+ onerror="document.documentElement.className=undefined"
+ onloadedmetadata="this.src=''; document.documentElement.className=undefined">
+<!-- Note we reset 'src' to release decoder resources and cubeb streams to prevent OOM or OpenCubeb() failures. -->
+</video>
+</body>
+</html>
diff --git a/dom/media/test/crashtests/0-timescale.mp4 b/dom/media/test/crashtests/0-timescale.mp4
new file mode 100644
index 0000000000..32df5dc5a5
--- /dev/null
+++ b/dom/media/test/crashtests/0-timescale.mp4
Binary files differ
diff --git a/dom/media/test/crashtests/1012609.html b/dom/media/test/crashtests/1012609.html
new file mode 100644
index 0000000000..1dad783a07
--- /dev/null
+++ b/dom/media/test/crashtests/1012609.html
@@ -0,0 +1,9 @@
+<script>
+try{var r0=new AudioContext();}catch(e){}
+try{var r32=r0.createOscillator();}catch(e){}
+try{var r58=r0.createPeriodicWave(new Float32Array(1997),new Float32Array(1997));}catch(e){}
+try{r32.start(0);}catch(e){}
+try{r32.setPeriodicWave(r58);}catch(e){}
+try{r32.frequency.value=-1;}catch(e){}
+</script>
+
diff --git a/dom/media/test/crashtests/1015662.html b/dom/media/test/crashtests/1015662.html
new file mode 100644
index 0000000000..20407c807d
--- /dev/null
+++ b/dom/media/test/crashtests/1015662.html
@@ -0,0 +1,4 @@
+<!DOCTYPE html>
+<body>
+<video><track src="javascript:5"></track></video>
+</body>
diff --git a/dom/media/test/crashtests/1028458.html b/dom/media/test/crashtests/1028458.html
new file mode 100644
index 0000000000..bb01e3343a
--- /dev/null
+++ b/dom/media/test/crashtests/1028458.html
@@ -0,0 +1,23 @@
+<html class="reftest-wait">
+<audio id="testAudio" controls></audio>
+<script type="text/javascript">
+navigator.mozGetUserMedia({audio: true, fake: true}, function(stream) {
+ stream.getAudioTracks()[0].enabled = false;
+ var testAudio = document.getElementById('testAudio');
+ // Wait some time for good measure
+ var eventReceived = 0;
+ testAudio.addEventListener("timeupdate", function() {
+ if (++eventReceived == 3) {
+ document.querySelector("html").className = "";
+ }
+ })
+ testAudio.srcObject = stream;
+ testAudio.play();
+ }, function(err) {
+ // Don't go orange if we can't get an audio input stream,
+ // as this is not what we are trying to test and can happen on Windows.
+ document.querySelector("html").className = "";
+ });
+</script>
+
+</html>
diff --git a/dom/media/test/crashtests/1041466.html b/dom/media/test/crashtests/1041466.html
new file mode 100644
index 0000000000..0f064d186c
--- /dev/null
+++ b/dom/media/test/crashtests/1041466.html
@@ -0,0 +1,21 @@
+<html>
+<script>
+try{var Context1= new (window.webkitAudioContext || window.AudioContext)()}catch(e){}
+try{var Delay0=Context1.createDelay();}catch(e){}
+try{var ScriptProcessor0=Context1.createScriptProcessor(512,26,7);}catch(e){}
+try{var ChannelSplitter0=Context1.createChannelSplitter(91);}catch(e){}
+try{var Gain1=Context1.createGain();}catch(e){}
+try{var WaveShaper0=Context1.createWaveShaper();}catch(e){}
+try{var Analyser1=Context1.createAnalyser();}catch(e){}
+try{Gain1.connect(Delay0);}catch(e){}
+try{Analyser1.connect(BiquadFilter0);}catch(e){}
+try{Gain1.connect(Context1.destination);}catch(e){}
+try{WaveShaper0.connect(ScriptProcessor0);}catch(e){}
+try{ChannelSplitter0.connect(BiquadFilter1);}catch(e){}
+try{Delay0.connect(Gain1);}catch(e){}
+try{ScriptProcessor0.connect(Context1.destination);}catch(e){}
+try{WaveShaper0.connect(Gain1);}catch(e){}
+try{WaveShaper0.connect(ChannelSplitter0);}catch(e){}
+try{ScriptProcessor0.connect(WaveShaper0);}catch(e){}
+</script>
+</html>
diff --git a/dom/media/test/crashtests/1045650.html b/dom/media/test/crashtests/1045650.html
new file mode 100644
index 0000000000..32acd2ce75
--- /dev/null
+++ b/dom/media/test/crashtests/1045650.html
@@ -0,0 +1,18 @@
+<html><body><script>
+
+var r0=new AudioContext();
+
+var splitter=r0.createChannelSplitter();
+var delay=r0.createDelay();
+var scriptp=r0.createScriptProcessor();
+var cmerger=r0.createChannelMerger();
+var gain=r0.createGain();
+
+splitter.connect(delay,2);
+delay.connect(scriptp);
+scriptp.connect(cmerger);
+cmerger.connect(splitter);
+gain.connect(gain);
+gain.connect(cmerger);
+
+</script></body></html>
diff --git a/dom/media/test/crashtests/1080986.html b/dom/media/test/crashtests/1080986.html
new file mode 100644
index 0000000000..1de3075169
--- /dev/null
+++ b/dom/media/test/crashtests/1080986.html
@@ -0,0 +1,3 @@
+<html>
+<audio autoplay src="1080986.wav"></audio>
+</html>
diff --git a/dom/media/test/crashtests/1080986.wav b/dom/media/test/crashtests/1080986.wav
new file mode 100644
index 0000000000..b96c59b7ec
--- /dev/null
+++ b/dom/media/test/crashtests/1080986.wav
Binary files differ
diff --git a/dom/media/test/crashtests/1122218.html b/dom/media/test/crashtests/1122218.html
new file mode 100644
index 0000000000..984487dd21
--- /dev/null
+++ b/dom/media/test/crashtests/1122218.html
@@ -0,0 +1,24 @@
+<html>
+<head>
+<script>
+function boom() {
+ var r0=new AudioContext();
+
+ var cm=r0.createChannelMerger(20);
+
+ var o1=r0.createOscillator();
+ var o2=r0.createOscillator();
+
+ var pw=r0.createPeriodicWave(new Float32Array(4),new Float32Array(4));
+ o2.setPeriodicWave(pw);
+
+ o1.connect(cm);
+ cm.connect(o2.frequency);
+
+ o1.start();
+ o2.start(0.476);
+}
+</script>
+</head>
+<body onload="boom();"></body>
+</html>
diff --git a/dom/media/test/crashtests/1127188.html b/dom/media/test/crashtests/1127188.html
new file mode 100644
index 0000000000..650f07dd47
--- /dev/null
+++ b/dom/media/test/crashtests/1127188.html
@@ -0,0 +1,3 @@
+<script>
+ (new AudioContext).close();
+</script>
diff --git a/dom/media/test/crashtests/1157994.html b/dom/media/test/crashtests/1157994.html
new file mode 100644
index 0000000000..2e3001302a
--- /dev/null
+++ b/dom/media/test/crashtests/1157994.html
@@ -0,0 +1,21 @@
+<!DOCTYPE html>
+<html>
+<head>
+<script>
+
+function boom()
+{
+ var ac = new AudioContext();
+ // first "suspended" -> "running" transition
+ ac.onstatechange = function() {
+ ac.onstatechange = null;
+ ac.suspend();
+ ac.close();
+ }
+}
+
+</script>
+</head>
+<body onload="boom();"></body>
+</html>
+
diff --git a/dom/media/test/crashtests/1158427.html b/dom/media/test/crashtests/1158427.html
new file mode 100644
index 0000000000..b544a942a1
--- /dev/null
+++ b/dom/media/test/crashtests/1158427.html
@@ -0,0 +1,21 @@
+<!DOCTYPE html>
+<html>
+<head>
+<script>
+
+function boom()
+{
+ dump("before capture\n");
+ document.createElement("audio").mozCaptureStreamUntilEnded();
+ dump("before context\n");
+ new window.AudioContext();
+ dump("before gc\n");
+ SpecialPowers.forceCC();
+ dump("after gc\n");
+}
+
+</script>
+</head>
+<body onload="boom();"></body>
+</html>
+
diff --git a/dom/media/test/crashtests/1180881.html b/dom/media/test/crashtests/1180881.html
new file mode 100644
index 0000000000..d5bf2f5642
--- /dev/null
+++ b/dom/media/test/crashtests/1180881.html
@@ -0,0 +1,8 @@
+<!DOCTYPE html>
+<html>
+<head>
+</head>
+<body>
+<video src="1180881.webm" autoplay></video>
+</body>
+</html>
diff --git a/dom/media/test/crashtests/1180881.webm b/dom/media/test/crashtests/1180881.webm
new file mode 100644
index 0000000000..2fb2be7a7f
--- /dev/null
+++ b/dom/media/test/crashtests/1180881.webm
Binary files differ
diff --git a/dom/media/test/crashtests/1185176.html b/dom/media/test/crashtests/1185176.html
new file mode 100644
index 0000000000..d5e9b68a29
--- /dev/null
+++ b/dom/media/test/crashtests/1185176.html
@@ -0,0 +1,24 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<head>
+<script>
+
+function boom()
+{
+ var ac = new window.AudioContext();
+ var oscillator = ac.createOscillator();
+ oscillator.start(0);
+ oscillator.stop(0.1);
+ setTimeout(function() {
+ oscillator.channelCount = 1;
+ oscillator.channelCountMode = "explicit";
+ oscillator.channelInterpretation = "speakers";
+ document.documentElement.removeAttribute("class");
+ }, 1000);
+}
+
+</script>
+</head>
+<body onload="boom();"></body>
+</html>
+
diff --git a/dom/media/test/crashtests/1185191.html b/dom/media/test/crashtests/1185191.html
new file mode 100644
index 0000000000..4ed5ba2ee2
--- /dev/null
+++ b/dom/media/test/crashtests/1185191.html
@@ -0,0 +1,21 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<head>
+<script>
+
+function boom()
+{
+ var a = new AudioContext();
+ var b = new BroadcastChannel("x");
+ a.addEventListener("statechange", bye, false);
+}
+
+function bye()
+{
+ document.documentElement.removeAttribute("class");
+}
+
+</script>
+</head>
+<body onload="boom();"></body>
+</html>
diff --git a/dom/media/test/crashtests/1185192.html b/dom/media/test/crashtests/1185192.html
new file mode 100644
index 0000000000..46fa311aa2
--- /dev/null
+++ b/dom/media/test/crashtests/1185192.html
@@ -0,0 +1,18 @@
+<!DOCTYPE html>
+<html>
+<head>
+<script>
+
+function boom()
+{
+ new Audio().mozCaptureStreamUntilEnded();
+ var ac = new window.AudioContext();
+ ac.resume();
+ ac.close();
+}
+
+</script>
+</head>
+<body onload="boom();"></body>
+</html>
+
diff --git a/dom/media/test/crashtests/1197935.html b/dom/media/test/crashtests/1197935.html
new file mode 100644
index 0000000000..dd8ad0382d
--- /dev/null
+++ b/dom/media/test/crashtests/1197935.html
@@ -0,0 +1,8 @@
+<!DOCTYPE html>
+<html>
+<head>
+</head>
+<body>
+<video src="1197935.mp4" autoplay></video>
+</body>
+</html>
diff --git a/dom/media/test/crashtests/1197935.mp4 b/dom/media/test/crashtests/1197935.mp4
new file mode 100644
index 0000000000..f00de75627
--- /dev/null
+++ b/dom/media/test/crashtests/1197935.mp4
Binary files differ
diff --git a/dom/media/test/crashtests/1223670.html b/dom/media/test/crashtests/1223670.html
new file mode 100644
index 0000000000..94cad43e23
--- /dev/null
+++ b/dom/media/test/crashtests/1223670.html
@@ -0,0 +1,23 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<head>
+<script>
+
+function boom()
+{
+
+ var ac = new window.AudioContext("publicnotification");
+
+ setTimeout(function() {
+ document.documentElement.removeAttribute("class");
+ var htmlAudio = new Audio();
+ var stream = htmlAudio.mozCaptureStreamUntilEnded();
+ ac.createMediaStreamSource(stream);
+ }, 0);
+}
+
+</script>
+</head>
+<body onload="boom();">
+</body>
+</html>
diff --git a/dom/media/test/crashtests/1236639.html b/dom/media/test/crashtests/1236639.html
new file mode 100644
index 0000000000..5c4634a4d4
--- /dev/null
+++ b/dom/media/test/crashtests/1236639.html
@@ -0,0 +1,9 @@
+<!DOCTYPE html>
+<html>
+<head>
+ <title>Bug 1236639: Crash on audio playback due to invalid XING headers</title>
+</head>
+<body>
+<audio autoplay src="1236639.mp3"></audio>
+</body>
+</html>
diff --git a/dom/media/test/crashtests/1236639.mp3 b/dom/media/test/crashtests/1236639.mp3
new file mode 100644
index 0000000000..8ef96e8cc7
--- /dev/null
+++ b/dom/media/test/crashtests/1236639.mp3
Binary files differ
diff --git a/dom/media/test/crashtests/1257700.html b/dom/media/test/crashtests/1257700.html
new file mode 100644
index 0000000000..5377b17da5
--- /dev/null
+++ b/dom/media/test/crashtests/1257700.html
@@ -0,0 +1,8 @@
+<!DOCTYPE html>
+<html>
+<head>
+</head>
+<body>
+<video src="1257700.webm" autoplay></video>
+</body>
+</html>
diff --git a/dom/media/test/crashtests/1257700.webm b/dom/media/test/crashtests/1257700.webm
new file mode 100644
index 0000000000..63e53c8c0a
--- /dev/null
+++ b/dom/media/test/crashtests/1257700.webm
Binary files differ
diff --git a/dom/media/test/crashtests/1267263.html b/dom/media/test/crashtests/1267263.html
new file mode 100644
index 0000000000..a4d0e621cd
--- /dev/null
+++ b/dom/media/test/crashtests/1267263.html
@@ -0,0 +1,19 @@
+<!DOCTYPE html>
+<html>
+<head>
+<meta charset="UTF-8">
+<script>
+
+function boom()
+{
+ vid.setMediaKeys(null);
+ vid.fastSeek(111);
+}
+
+</script>
+</head>
+
+<body onload="boom();">
+ <video id="vid" src="../../../../layout/reftests/webm-video/frames.webm"></video>
+</body>
+</html>
diff --git a/dom/media/test/crashtests/1270303.html b/dom/media/test/crashtests/1270303.html
new file mode 100644
index 0000000000..23608bb2b8
--- /dev/null
+++ b/dom/media/test/crashtests/1270303.html
@@ -0,0 +1,8 @@
+<!DOCTYPE html>
+<html>
+<head>
+</head>
+<body>
+<video src="1270303.webm" autoplay></video>
+</body>
+</html>
diff --git a/dom/media/test/crashtests/1270303.webm b/dom/media/test/crashtests/1270303.webm
new file mode 100644
index 0000000000..c9b0003d6b
--- /dev/null
+++ b/dom/media/test/crashtests/1270303.webm
Binary files differ
diff --git a/dom/media/test/crashtests/1291702.html b/dom/media/test/crashtests/1291702.html
new file mode 100644
index 0000000000..54f17b45b8
--- /dev/null
+++ b/dom/media/test/crashtests/1291702.html
@@ -0,0 +1,72 @@
+<script>
+Logger={}; Logger.JSError=function(e){};
+try { o0 = new Audio("media/audio/mono-uncompressed-8bit-44100hz.wav") } catch(e) { Logger.JSError(e); }
+try { o1 = o0.mozCaptureStreamUntilEnded() } catch(e) { Logger.JSError(e); }
+try { o2 = new window.AudioContext(); } catch(e) { Logger.JSError(e); }
+try { o3 = o2.createBufferSource(); } catch(e) { Logger.JSError(e); }
+try { o5 = o2.createChannelMerger(3); } catch(e) { Logger.JSError(e); }
+try { o3.start(0) } catch(e) { Logger.JSError(e); }
+try { o2.listener.setPosition(0.0417049336344248, 0.9932504594310304, 32) } catch(e) { Logger.JSError(e); }
+try { o5.disconnect(0) } catch(e) { Logger.JSError(e); }
+try { o0.mozGetMetadata() } catch(e) { Logger.JSError(e); }
+try { o3.channelCount = 1; } catch(e) { Logger.JSError(e); }
+try { o3.buffer = function anonymous() {
+var buffer = o2.createBuffer(1, 512, o2.sampleRate);for(var c=0;c<1;c++) {var data = buffer.getChannelData(c);for(var i=0;i<512;i++) {data[i] = i % 512}}return buffer;
+}(); } catch(e) { Logger.JSError(e); }
+try { o3.loop = false; } catch(e) { Logger.JSError(e); }
+try { o0.preservesPitch = false; } catch(e) { Logger.JSError(e); }
+try { o2.destination.channelCount = 1; } catch(e) { Logger.JSError(e); }
+try { o3.loopStart = 8; } catch(e) { Logger.JSError(e); }
+try { o3.playbackRate.value = 0.46271130895770884; } catch(e) { Logger.JSError(e); }
+try { o2.listener.setVelocity(0.34781960219792546, 4, 2048) } catch(e) { Logger.JSError(e); }
+try { o5.connect(o5, 0, 0) } catch(e) { Logger.JSError(e); }
+try { o3.loopStart = -0.24696638021780326; } catch(e) { Logger.JSError(e); }
+try { o0.mozSetup(1, 44100) } catch(e) { Logger.JSError(e); }
+try { o3.buffer.copyToChannel(function anonymous() {
+var buffer=new Float32Array(256);for(var i=0;i<256;i++){buffer[i]=i / 256}return buffer;
+}(), 1, 2048, 64) } catch(e) { Logger.JSError(e); }
+try { o3.loop = false; } catch(e) { Logger.JSError(e); }
+try { setInterval(function anonymous() {
+try { o0.pause() } catch(e) { Logger.JSError(e); }
+}, 12.902067779658143) } catch(e) { Logger.JSError(e); }
+try { o2.listener.setPosition(256, 256, 16) } catch(e) { Logger.JSError(e); }
+try { o3.playbackRate.setValueCurveAtTime(function anonymous() {
+var buffer=new Float32Array(4);for(var i=0;i<4;i++){buffer[i]=i / 4}return buffer;
+}(), 2, 0.40792575814014437) } catch(e) { Logger.JSError(e); }
+try { o3.playbackRate.value = 0.4997270553139334; } catch(e) { Logger.JSError(e); }
+try { o0.loop = true; } catch(e) { Logger.JSError(e); }
+try { o3.loopStart = 4; } catch(e) { Logger.JSError(e); }
+try { setInterval(function anonymous() {
+try { o3.buffer = function anonymous() {
+var buffer = o2.createBuffer(1, 1, o2.sampleRate);for(var c=0;c<1;c++) {var data = buffer.getChannelData(c);for(var i=0;i<1;i++) {data[i] = Math.sin(Math.sin(i))}}return buffer;
+}() } catch(e) { Logger.JSError(e); }
+}, 54.32078602859342) } catch(e) { Logger.JSError(e); }
+try { o3.connect(o2.destination); } catch(e) { Logger.JSError(e); }
+try { o3.channelCountMode = 'max'; } catch(e) { Logger.JSError(e); }
+try { setInterval(function anonymous() {
+try { o3.channelCount = 1; } catch(e) { Logger.JSError(e); }
+}, 55.448587039802966) } catch(e) { Logger.JSError(e); }
+try { o3.playbackRate.cancelScheduledValues(0.7190983131805198) } catch(e) { Logger.JSError(e); }
+try { o3.playbackRate.cancelScheduledValues(16) } catch(e) { Logger.JSError(e); }
+try { o3.connect(o5, 0, 2) } catch(e) { Logger.JSError(e); }
+try { o3.loopEnd = 0.5864771678080962; } catch(e) { Logger.JSError(e); }
+try { o0.playbackRate = 0.2781783298771312; } catch(e) { Logger.JSError(e); }
+try { o0.loop = false; } catch(e) { Logger.JSError(e); }
+try { setInterval(function anonymous() {
+try { o5.disconnect(0) } catch(e) { Logger.JSError(e); }
+}, 29.75776777646425) } catch(e) { Logger.JSError(e); }
+try { o3.playbackRate.setValueCurveAtTime(function anonymous() {
+var buffer=new Float32Array(8);for(var i=0;i<8;i++){buffer[i]=8 % 8}return buffer;
+}(), 0.4972710112336257, 64) } catch(e) { Logger.JSError(e); }
+try { setInterval(function anonymous() {
+try { o0.controls = false; } catch(e) { Logger.JSError(e); }
+}, 22.550249570567694) } catch(e) { Logger.JSError(e); }
+try { o2.listener.setOrientation(0.6531494410366634, 64, 0.5120918081402992, -64, 0.32912433155093446, 256) } catch(e) { Logger.JSError(e); }
+try { o3.loop = true; } catch(e) { Logger.JSError(e); }
+try { o3.connect(o5, 0, 0) } catch(e) { Logger.JSError(e); }
+try { o3.buffer = function anonymous() {
+var buffer = o2.createBuffer(1, 2048, 48000);for(var c=0;c<1;c++) {var data = buffer.getChannelData(c);for(var i=0;i<2048;i++) {data[i] = Math.sin(Math.sin(2048 * 0.2519529190035427))}}return buffer;
+}(); } catch(e) { Logger.JSError(e); }
+try { o3.disconnect(0) } catch(e) { Logger.JSError(e); }
+</script>
+
diff --git a/dom/media/test/crashtests/1368490.html b/dom/media/test/crashtests/1368490.html
new file mode 100644
index 0000000000..8a2d9f9674
--- /dev/null
+++ b/dom/media/test/crashtests/1368490.html
@@ -0,0 +1,30 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<head>
+ <title> Bug 1368490 : Crash if media recorder source stream reduces number of channels. </title>
+</head>
+<meta charset="utf-8">
+<script type="text/javascript">
+
+function boom() {
+ let audioContext = new window.AudioContext();
+ let oscillator = audioContext.createOscillator();
+ let dst = audioContext.createMediaStreamDestination();
+ oscillator.channelCount = 4;
+ dst.channelCount = 4;
+ oscillator.connect(dst, 0, 0);
+ oscillator.start();
+ mediaRec = new MediaRecorder(dst.stream);
+
+ mediaRec.start(100);
+ setTimeout(() => {
+ dst.channelCount = 1;
+ setTimeout(() => {
+ mediaRec.stop();
+ document.documentElement.removeAttribute("class");
+ }, 100);
+ }, 100);
+}
+</script>
+<body onload="boom();"></body>
+</html>
diff --git a/dom/media/test/crashtests/1378826.html b/dom/media/test/crashtests/1378826.html
new file mode 100644
index 0000000000..e1913cd0f5
--- /dev/null
+++ b/dom/media/test/crashtests/1378826.html
@@ -0,0 +1,46 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<head>
+<title>Bug 1378826 : Removing last video track from recorder stream crashes.</title>
+</head>
+<body>
+<canvas id="canvas"></canvas>
+<script type="text/javascript">
+
+function wait(ms) {
+ return new Promise(resolve => setTimeout(resolve, ms));
+}
+
+function boom() {
+ let canvas = document.getElementById("canvas");
+ let ctx = canvas.getContext('2d');
+ ctx.fillRect(10, 10, 100, 100);
+ let stream = canvas.captureStream();
+ let rec = new MediaRecorder(stream);
+ // At the time of fixing this bug onstop would fire, but this may change in
+ // future. As such defensively listen for onerror too to prevent this test
+ // timing out.
+ let stoppedPromise = new Promise(y => (rec.onstop = y,
+ rec.onerror = e => y));
+ rec.onstart = () => {
+ // Remove the video track from the stream we're recording
+ stream.removeTrack(stream.getTracks()[0]);
+ // Recorder should stop or error in response to the above
+ return stoppedPromise
+ .then(() => {
+ // Little wait to help get bad writes if they're going to happen
+ wait(100)
+ .then(() => {
+ // Didn't crash, finish
+ document.documentElement.removeAttribute("class");
+ });
+ });
+ };
+ rec.start();
+}
+
+window.onload = boom;
+
+</script>
+</body>
+</html>
diff --git a/dom/media/test/crashtests/1384248.html b/dom/media/test/crashtests/1384248.html
new file mode 100644
index 0000000000..5d9c60edda
--- /dev/null
+++ b/dom/media/test/crashtests/1384248.html
@@ -0,0 +1,10 @@
+<html>
+ <head>
+ <script>
+ try { o1 = document.createElement('audio') } catch(e) { }
+ try { o2 = document.implementation.createDocument('', '', null).adoptNode(o1); } catch(e) { };
+ try { o3 = new AudioContext('alarm') } catch(e) { }
+ try { o3.createMediaElementSource(o1) } catch(e) { }
+ </script>
+ </head>
+</html>
diff --git a/dom/media/test/crashtests/1388372.html b/dom/media/test/crashtests/1388372.html
new file mode 100644
index 0000000000..977ebddf53
--- /dev/null
+++ b/dom/media/test/crashtests/1388372.html
@@ -0,0 +1,13 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<script>
+navigator.mediaDevices.getUserMedia({audio: {
+ echoCancellation: false,
+ noiseSuppression: false,
+ autoGainControl: false
+ }, fake: true
+}).then((stream) => {
+ document.documentElement.removeAttribute("class");
+})
+</script>
+</html>
diff --git a/dom/media/test/crashtests/1389304.html b/dom/media/test/crashtests/1389304.html
new file mode 100644
index 0000000000..df419c51d7
--- /dev/null
+++ b/dom/media/test/crashtests/1389304.html
@@ -0,0 +1,32 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: Negative duration.</title>
+</head>
+<body>
+
+<video id="v" controls src="1389304.mp4">
+</video>
+<p id="msg"></p>
+
+<script type="text/javascript">
+
+function log(x) {
+ msg.innerHTML = x + "<br>";
+}
+
+v.play();
+v.onended = function() {
+ log("endded!");
+ let seekable = v.seekable;
+ for (let i = 0; i < seekable.length; ++i) {
+ let start = seekable.start(i);
+ let end = seekable.end(i);
+ log(`[${i}]: start=${start} end=${end}`);
+ }
+}
+
+</script>
+
+</body>
+</html>
diff --git a/dom/media/test/crashtests/1389304.mp4 b/dom/media/test/crashtests/1389304.mp4
new file mode 100644
index 0000000000..25cd746972
--- /dev/null
+++ b/dom/media/test/crashtests/1389304.mp4
Binary files differ
diff --git a/dom/media/test/crashtests/1393272.webm b/dom/media/test/crashtests/1393272.webm
new file mode 100644
index 0000000000..1f1cade6dc
--- /dev/null
+++ b/dom/media/test/crashtests/1393272.webm
Binary files differ
diff --git a/dom/media/test/crashtests/1411322.html b/dom/media/test/crashtests/1411322.html
new file mode 100644
index 0000000000..772b68f0cc
--- /dev/null
+++ b/dom/media/test/crashtests/1411322.html
@@ -0,0 +1,18 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Bug 1411322: Shutdown after getting memory reports from MediaRecorder</title>
+ <script type="text/javascript" src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<audio id="audio"></audio>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+let recorder = new MediaRecorder(audio.mozCaptureStream());
+recorder.start();
+SpecialPowers.getMemoryReports();
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/crashtests/1414444.mp4 b/dom/media/test/crashtests/1414444.mp4
new file mode 100644
index 0000000000..4041852100
--- /dev/null
+++ b/dom/media/test/crashtests/1414444.mp4
Binary files differ
diff --git a/dom/media/test/crashtests/1450845.html b/dom/media/test/crashtests/1450845.html
new file mode 100644
index 0000000000..451d116e83
--- /dev/null
+++ b/dom/media/test/crashtests/1450845.html
@@ -0,0 +1,34 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<head>
+ <title>Bug 1450845: Avoid seek to next frame when already seeking</title>
+ <script>
+ async function boom() {
+ let video = document.getElementById('video');
+
+ // Internally play causes a seek, make sure we don't crash during this
+ video.play();
+ try {
+ await document.getElementById('video').seekToNextFrame();
+ } catch (e) {
+ // We don't mind if the promise was rejected so long as we don't crash
+ }
+ // Didn't crash
+
+ // Stop playback and cause a seek to 0
+ video.pause();
+ video.currentTime = 0;
+ try {
+ await document.getElementById('video').seekToNextFrame();
+ } finally {
+ // Didn't crash
+ document.documentElement.removeAttribute("class");
+ }
+ }
+ window.addEventListener('load', boom)
+ </script>
+</head>
+<body>
+ <video id='video' src='data:video/webm;base64,GkXfowEAAAAAAAAfQoaBAUL3gQFC8oEEQvOBQoWBAhhTgGcBAAAAAAAB6BFNm3RALE27i1OrhBVJqWZTrIHfTbuMU6uEFlSua1OsggEwTbuMU6uEHFO7a1OsggHL7AEAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAVSalmAQAAAAAAAEUqTYCNTGF2ZjU3LjI5LjEwMVdBjUxhdmY1Ny4yOS4xMDFzpJBAb17Yv2oNAF1ZEESuco33RImIQFCAAAAAAAAWVK5rAQAAAAAAADyuAQAAAAAAADPXgQFzxYEBnIEAIrWcg3VuZIaFVl9WUDmDgQEj44OEAfygVeABAAAAAAAAB7CCAUC6gfAfQ7Z1AQAAAAAAAEfngQCjqYEAAICCSYNCABPwDvYAOCQcGFQAAFBh9jAAABML7AAATEnjdRwIJ+gAo5eBACEAhgBAkpwATEAABCasAABekcXgAB'>
+</body>
+</html>
diff --git a/dom/media/test/crashtests/1489160.html b/dom/media/test/crashtests/1489160.html
new file mode 100644
index 0000000000..c4f643700c
--- /dev/null
+++ b/dom/media/test/crashtests/1489160.html
@@ -0,0 +1,10 @@
+<html>
+<head>
+<script>
+audio = new AudioContext()
+audio.close()
+audio.close()
+</script>
+</head>
+</html>
+
diff --git a/dom/media/test/crashtests/1494073.html b/dom/media/test/crashtests/1494073.html
new file mode 100644
index 0000000000..41e7a36554
--- /dev/null
+++ b/dom/media/test/crashtests/1494073.html
@@ -0,0 +1,19 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<head>
+ <title>Bug 1494073: Setting playback rate too high</title>
+ <script>
+ function finish() {
+ document.documentElement.removeAttribute("class");
+ }
+ var src="data:audio/wav;base64,UklGRqsTAABXQVZFZm10IBAAAAABAAMA8FUAANABAQADAAgAZGF0YYcTAAAUCnYAPm6YALYJSgA4IO8A9UuWALiiZQAYWc4AnMNoAKcc9gBmWLcAtL0NAAGU0QDbSgMA777tAEkfrQD14E4A2nb/AERxxAD5c+0AmplbADUo0AC14DQAXWgRAGsl4wB0oGIAkVlkAOWmTwCm/fEAZqErAEHpqgAMXjkA5AlGAMsOOwAUQAsABvRfAJyVUwDTKM8AtIMYAKh6lQDPkq0A9JtqAKiudAD2I3wAHTsPAOy36ACLtz8ATQELAJCXQgDpxksA4BzzABSR7gD7vMEAgo/6AKovngCZNLoA0A5UALhiFABNVRQAjXIuABmhFACj8/sAt+bQAP71dgCbdGoAXuf5AN5zVAACgggAlHiqAGTPswC0x5oAsgxDAEimygCYlMUAS3r2AKFS0gCZzCYAnXPZAEU8cgAf2gQADqF2ACCY+ABSVIEAW17CANxZWQCk0s0AIF1kALYn7gC9U3MAQqRFAHSUegC1UR8Au99GADrbaQCQQksAOQ6kALUmMgAC0YQAS9rjAMNL9QAQKkcAayg7AOjHXgBNZxcAzIUrAAgq+ACVJPMAV5s2APTVsgAsxX4AMP9cAInlqADcltMAv10FAAzJAgDiHQ4AbsW4ACDTowC897AAx1A8AGw3JwBe6msAf/jJAL1PbwCYYRkAscdyACi8+QCXCEwABWnAAOz4QQC0W60AcToLAH4/+wBion0AzzBXAJCuHgAsv+gAvDmdAA1LaAC04xgA/KU6AAurkwCbpwkAzkTeAA9+SABQqJoAPFHNAK96FgDxMNkAaPelAOG5YgCJsS8AtUp8ALhn8gD7sIYAxoR+AMWFHgA3aRIA00/wABwXJwCmQVQAxU7xAGnkjABvBaUAMz8UAL7qjQBokUEAx8eOAAd/ogDKKqcAsx8EAEpW/AADoDAA/D1+AKVqMwDnkmYA/fOAAGpcfwAlm3UA5+t8AM5tMwCdW3cA2UzjABHxWQBKIx4A7bf3AC+RkgARtBAA27gWADcb/ADjO68AS+g0AMutgwAzRawAtbgLAGjfnwCGINIAowqrAFsDmACIyhgAHfz+AByCiABBwdYASnWAANadowA7KFYADTx1ALG1WAAm4csAti5iACQMZgAeg7MAtrnPAMRmsAAsONAAKR6XAEu5TQCrqMYAd0hcABAU8gCLBFEA2C0MAJ7nZwAETAYA6i53APxR2gDRwBsAf392AH/csgAidosA3MaAAKgCrQBwid4A4xUfAHpqnACYWcQAHfIoAO67cQBlKVUAfR6pAEvQyQDFcWoA82ivAGsVXAA5gcQA8YCIAH9igAC6EzIAvrJEACTfdgBM21wAE6W+ABkqTwAL0+wA85kAALdRfQDbetYAT9RxALUxoQDScSoACRY8AIOzSwAUQS4A6yG1ALDLIwCf5vUAUvQTAK8r+QC5zVoAckBEABQfYwBFzoQAEUyXANlojQBGhjMA0WNjAI0qOQC7TIkAMbboAAp5igBw640ArVL3ADsU7QBoR/cAemWCAChwZQDuILoApIhRAOXapADr/IAApX7IAKvnxABvSHgAV1k1ABqwegD4s6IAT4m1AH/wPgBEnc0AwuIVAO5lzgDsHIAA8O8QAPZZeAC68JcAMzVvAAVKUwBGXv0AaOoBACuF6QD9uBQAwGpOAEiF1wCq/1kAXLJ4AGP4xgC8uzMAHgMmAF3WXwCssiUAya68AP7FUwAkZqYAqnfsAJkuUgC+08cANBobALJvTgAtK7oAaIIoALfiPgAW5qAAO1YyAM9+owCFIMMAbL82ABfR7wBJrDgASc3DAB8q7AAi/80AxleNALQ+3wARtIIA3D9uAFWiEwBtshQAoPHFAITU8gA1S4EAB8w6ANZxHgBhLBEAx70EADf8KgADzHIAQ4ONAEyVRgAXxdIAoS0hAHkIhwBUeFgAOP1YAM6KtgAJaJ0AuaUnADdSkQDNaVIAJkoIAFVlkAARLYwAc2jBAEHoeQBMnFIAQS8hAKUu2ABZafMAQRGbAElVMgAdHRgA9ImMAF4/NADLcZMAiaXbAB0rpQB1RTEAGqsMAGYPWgAB4nkAEk/2AOg7lQC5bI8A7x5cALOosgDQV7YA8mwTACI31wCjuIAAegycABfkWwBSqBQApdm/AIqmCgBEp3QANDopAHSFjgDOIKgABWL8AOppagBvyq0AXyT+AGvkZwDaq1MA8J0CAO0qCADs3gcAj/T/AFHxnwBnObIAAHr0AGNgYQCyAeoAerH5ANuJsQAZnSwAqPWRABmF5gBKI5IAkQpVAOQq0gCdZxQArSpTAFC4aACZFzgA2vozAKKrjwADOGYAxejeANjGkgAogccAnJtJAHa0iQDNsMQAvkg0AAxqhQABL88A0+rRAGam/AAZaiwAakBeAIkbewAlIpIAMQjsAHKsxAAVPRIAH5HeABuShQBm4ocAtOdpAIvy0QDAdbAAnRZ2AFNL6QCFcQYACIehACvVNgChty8Ah5A8AFc4iwBQwRwAv3PeAO+FjABRt0kAMfNNAMz2/QDaFEEAzLRaAJ5hlQBihAoAP/CjAPH4bAC/fxwAPapOAO3BCQDgg2gA7IyRABgp/AAO+QgAXrMFAHPiZwDc83IAmMzHALPPaAAlyrkAAk6uAD33pwBo0iAAC56eAOolMACgdmQAKdJeAA0UPgBtua8AIH66AJMoxwCYv7UA4JcKAN5c/wDgWDwAcWbUANvmLwD73kIAu/vCAFUiCQB5CJIAxOlxADKN8AB4H8EAh8gnAC8h8QCdBUgAX7cyADXejABFCWYAm7v7AOENigCsVMYAY8UeABEpaADYLmMAOWcMANk9hAANhM8AAQHaAMfmVwB2p2AA5MbYAOw44gCKrqQADZOHAM1RlwBhnvAAGkToADueDQDmbJ0AFYYTAGE6UgDjbEgAeJhCAOOGgAC0s6QANh6iAMu3kQAMMBQAl7YOANsIXwCkd9oA8mqMANf38wCxC7EAv7gjAKa0HwCgqV4ApgI/AIe1oQBaGE8AcxcBAObgYQBMMWwA1G0gAF+J9ABkAl8A2buoAIlKWAA/ZykAl6S0ABtBMQAkyOEAzFtyAHIMdgCCfMgAG9m4ABOX7wBfi/wAeg74AELcSAC7RgEAP1V0AD/FBwABALUAdGX+APlznwBqD1MAfTHIALA+HgDRP/QAaraOANCTrgBWaVUANUQjAFVulwBFlpkAsU3sAN86OACsleQAcFHoANanugC10xgAjD4TAA0g3gDcmx4A/eNDAIsxFAC6N5UAJ7fDAFRwZQD/+pAALV5yACfwHwBzltQA9CAuAD0nQQDbjogAKLenAHydkQDwD+IABvePAF9zWAAvQBwA4Ae3AJk+hQCB0ugAHXjgAOBbKQC4/ksAP4CtAAELZgDZ6kEAO2TIAPH6ZAC80CwAd9CIANrJ3QBbk7QAcT0AAE58HwDa9QMAOYjoABoHGQArjcwADSF4AFNcRgCnkjIAsG/cAC7iCwCh3qEAramDAAYS3gBO0jcA+2p2ADpCZwATvIsAa0FuABWrbwD+0jUA+PYEAPXTnQCnvXQAYtYoAKfT2wCPsNUA0bqlAG1+sQBnnqEA7hvxABQYewBmeO4Ackj6AJhuRADlYpAA9O9TAPSSeACW+ZAAa0nLAErmuAABaxcA+WJsAAJrYgDjOEkAwldNAC3lLwDy81wAoYZzADYpmwC509wAtFJdAIFkwgDiIEsAkyBvAMmSIgDb/sgAjTw2AJzz6QA+0Z8AwXaTAD/3lwB0Kg8A63iFAFe95wARFK8AFOY5AMAmfwB+Rj4AVleaAJNp0wBDKCUAh08qAEy8BwAX0CUAs+xgABUglAD3+TAA8jt9AM83uwBUVaEAvErrACRYKgB8iGkAb5fUAP71dwDEJIoAx7wfAAIduQALb3kAjm+cABeJCADKk10AdfclABorYAC0WOgA167pAGsx/QC8htoA+7auAATccACYfLsAszNLADwo8ACOsr0AdJmfABl+xgD7MQ4Ax2x6AMpK4gBV7FkA0g/QAFmDRQA01bkAFKhtAH9PEwCEoukAV2UiAC2B+wCuwRsA560TAK92ZAABWWcAovTVAPS/6AC+1C8AXCkSAM2NegBYrWQAmozZAE2CIgBAgY4A9YmBAKY0DgD3ZLAAOgXaAOOGSwDofs8AHlvXAN204wCjH2YAlRVHAA2cxwBDjN8AwumLAC5MEwAKK9YAyELrANJYkACbtlYAKfF2AGpW8ACRmC0AWHqZAB1CzwAGY9sA0N9+AOS4igAHJNkACm+yAA+sXwBKUUIAaipZAJuECwDrgV0AFMJpAAKMjgAtdggAJYIXAHDtegAgFIoASxGoALACPgA2KPEASXlaAGIX7QAjLsYA62u0ABm8HQB5dUsATxadAK6qQwD8VEkAd1ARAFOFzADC26QAfe2GANRbdwC7nK8AEwy4AB2XrgDAHIsAZSA6AEo5AgAM8GUAth6kAINTsACbAcEAxoCEAIZGPAD4GlwA3aV9AK7CDgCjga4ArWf0AHD1VQBd2TgA+eGcABMlUACIulIAOzg4AHP9zACB2VIAkyv0ALSeowCib4MA3fyrAMP7+gBVtSMAzpdQAMFI1AATJq8ALcZ1AGvp0gC/4oMAH3/0AFPf7wB5xTwAHeDZAALL8gDOEuQA+iN+AMs3AgCP3lIAs+6FABBzswBwlFkAxPuuAEg6pgCvKoIAU9EoAGAPigBqilYACnYGADJYkQCkihkA7FTKAISADQCODaoAYE04ALH1iwAUi7MAzURTAORgwQCts3wAc78IAFkTXwB+A5UAJiRXAJFADwAjr7MAdjo6ANNvLACcXagA09XGAOzjAwC0GJ0AV3hVAI1RkwAuDVgAtcfaABkNCQAljBgABQNSABEgDQB5kmwAIre6AFs6+wAnad4AT4iAAMSvawA7Fn8AlLC9AOscDABRf54ATp8wAMzgnwD29YQASW0XAEj2NwDw2U4A9NokAMTdYQBw+5UAgxCpAEwmpwCyDPwANRUYALHKyQB5mGsA1yJ8AJqjDQCaHDEAMtJrAEvLIQCDVUoAwGn3AETZ3wCLIwEAeJNzAMkj8AAUk2wAe+YVAOVR8wDCzpcAu4jkACv7dAA3vOgApBWtAO+tkAD8tu8Ac7noACsS7QAXfvMA7mjTACJc9QCea+gAx2jFALywtACyxdUA7r2wANGY2wD4JvYAg5VOANb0ggDtytkA8WGyAHMOtwA0vqAAwO5NAJSJDAAUaRUAoDvEAIfetwA9PqoA1jC3AA61XADx70MA7fFUAPQtRgBpiV0AP2XDAKNQvgCyW6UAJ9esADvTdQBm6f4AF1lDADTKvgCXNQQA6SFtAEuSVQBsXjUAmGbSANERuABhSpUAD2OJAGLtFABRMFUA0IDAAPc3TwCtToMASSGaAMDpVwCjm40At6keAG6chAAgkRAAz7GGAG8lHQA7qgwA8BucAMnEEQDEC+4A9g4MALMwnwAEFpMAqzf0AK8BoAA6L6sA0l0OACWT/gCVqp8AfumWABzjDQCtsygAHG0JAHebagCmi2EAztDRALUodwCq1+sAEcGyAGmDQQAVnXMAxMu8APC5wACRds0AzdJ4AEUdzwD/LLAAUdAQADW6IQBIUcUAXGxOAHdGZAARzd8AA7Y5ADE7cABegAcA0eDuAMBHYwA6skcARPfCAEYMxwB2/g8APBr5AFNusABVe1kAw4+OAOxwgwB7LiwAOltxAI0eAwALkk4AG9EZAESUzgD9fdQA4YOFAJVkHACkj4MAsWJxABeFJABHocwATeLwAG+OuQDdhzIAabHNAPogKwCTMKQAkanLAAsZLQDFv6YAXjQFAK1LdwDK4XsAcM7eAGCurgA7dq8AgD8dAHK87QDpwi8AwH3WAM/BcgBW7sAAyUeVAGw91wCHzM8Aa5fYABuR9QBhOk8Ay+OSAGYDiwAHCJAAYjifABxdQABItm0Ame2tAE/4XABIC6sA0aPxAAwi2ACitbQAtsWpAA+jPwB2lCgAqaATAMx3JADF2o4A6fhSABNVmwC6giQAYfj6AD0dkwBkmecAU5sTANWMgwD7D+cAp9QWAI6+QQAkZvcAkvGvAMaNOgAtsuEAF+LgADEdxgDAc9MA7+36APqe/wD41wQAIIDKAHsx1gATEKgAnWdaAFgB0wAyMAEAxTsPADTrDACBAuMALov9APMo1wB1wawAlCj7AP3EbAD1lDQA3xUvAAPdwABvPXQAwPeDAGxwkQAaAUEAbJQ2ACGgEQBn8XAAuu81AKaQhgAzfIQAkp8jADQ4TQBSSaIALjCgAJuAjQBP8AsAHL5SAKmH7ACyx5cAP/VjAG71zgCuL+IAV2JLAN+yFgA7QvgAG0HdAFO6AwCx+AwArlawAB83cABUNtcAe9LbACB2gAAZVTUA4DRYAEFV4wByz2QAWUgkALXS9ACCaEYAxjjiAI0qiAArChIAr7twAPvaOwDhyfkAEoYVAIM5KABarEsAol47AIBNdQDfMHQAN8GNADM6XwBbJFMAMAEbALILpgCxRIsA3mxlAGqpZACd9SkA9uwTABtT4ABUZDIA3cxyAGnKygA3zW4Age3TANrX0QBc2KEAAab2AEQksQDabWAAdmSYAOBdUwDsLkgAOa4fAPi45QAjr5gAZvEs"
+ var audio=new Audio(src)
+ audio.onended = finish;
+ audio.onerror = finish;
+ try{audio.play()}catch(e){}
+ try{audio.preload=1}catch(e){}
+ try{audio.muted=1}catch(e){}
+ try{audio.playbackRate=567312.2079031984}catch(e){}
+ </script>
+</head>
+</html>
diff --git a/dom/media/test/crashtests/1517199.html b/dom/media/test/crashtests/1517199.html
new file mode 100644
index 0000000000..10c994f845
--- /dev/null
+++ b/dom/media/test/crashtests/1517199.html
@@ -0,0 +1,17 @@
+<html class="reftest-wait">
+<head>
+<script>
+(async _ => {
+ const video = document.createElement("video");
+ video.mozCaptureStreamUntilEnded();
+ video.src = "test.mp4";
+ video.playbackRate = 2;
+ await new Promise(r => video.onloadedmetadata = r);
+ video.currentTime = video.duration - 1;
+ await video.play();
+ await new Promise(r => video.onended = r);
+ document.documentElement.removeAttribute("class");
+})();
+</script>
+</head>
+</html>
diff --git a/dom/media/test/crashtests/1526044.html b/dom/media/test/crashtests/1526044.html
new file mode 100644
index 0000000000..1914dfc2f5
--- /dev/null
+++ b/dom/media/test/crashtests/1526044.html
@@ -0,0 +1,19 @@
+<script>
+ function start () {
+ try { o1 = new AudioContext({}) } catch (e) { }
+ try { o2 = new DynamicsCompressorNode(o1, {}) } catch (e) { }
+ try { o3 = o2.attack } catch (e) { }
+ try { o4 = new XMLHttpRequest({mozAnon: true, mozSystem: true}) } catch (e) { }
+ try { o4.open('GET', '', false) } catch (e) { }
+ try { o4.send() } catch (e) { }
+ for (let i = 0; i < 20; i++) {
+ try { o5 = o1.createGain() } catch (e) { }
+ try { o1.suspend().then(function () { }) } catch (e) { }
+ try { o5.connect(o3) } catch (e) { }
+ }
+ setTimeout('location.reload()', 200)
+ }
+
+ window.addEventListener('load', start)
+</script>
+
diff --git a/dom/media/test/crashtests/1530897.webm b/dom/media/test/crashtests/1530897.webm
new file mode 100644
index 0000000000..ac2224157a
--- /dev/null
+++ b/dom/media/test/crashtests/1530897.webm
Binary files differ
diff --git a/dom/media/test/crashtests/1538727.html b/dom/media/test/crashtests/1538727.html
new file mode 100644
index 0000000000..6371d54383
--- /dev/null
+++ b/dom/media/test/crashtests/1538727.html
@@ -0,0 +1,14 @@
+<script>
+const canvas = document.createElement('canvas')
+const context = canvas.getContext('2d', {})
+const xhr = new XMLHttpRequest({})
+const stream = canvas.captureStream(new Float64Array([1593177632.1689904])[0])
+recorder = new MediaRecorder(stream)
+recorder.start(100)
+xhr.open('G', '', false)
+xhr.send()
+recorder.stop()
+tracks = stream.getVideoTracks()
+track = tracks[(1051736525 % tracks.length)]
+stream.removeTrack(track)
+</script>
diff --git a/dom/media/test/crashtests/1545133.html b/dom/media/test/crashtests/1545133.html
new file mode 100644
index 0000000000..fb7039aae3
--- /dev/null
+++ b/dom/media/test/crashtests/1545133.html
@@ -0,0 +1,34 @@
+<html class="reftest-wait">
+<head>
+<script>
+const xhr = new XMLHttpRequest()
+
+async function boom () {
+ await new Promise(r => setTimeout(r, 100))
+
+ SpecialPowers.forceCC()
+ SpecialPowers.forceCC()
+ SpecialPowers.forceCC()
+
+ document.documentElement.removeAttribute("class")
+}
+
+function start () {
+ const context = new AudioContext({})
+ const filter = new BiquadFilterNode(context, {})
+ const destination = context.createMediaStreamDestination()
+ const processor = context.createScriptProcessor(8192, 8, 8)
+ processor.connect(filter.Q)
+ processor.disconnect()
+ xhr.open('G', '', false)
+ xhr.send()
+ context.createMediaStreamSource(destination.stream)
+ processor.connect(filter.Q)
+ context.close()
+ context.addEventListener('statechange', boom, true)
+}
+
+document.addEventListener('DOMContentLoaded', start)
+</script>
+</head>
+</html>
diff --git a/dom/media/test/crashtests/1547784.html b/dom/media/test/crashtests/1547784.html
new file mode 100644
index 0000000000..ee270491f1
--- /dev/null
+++ b/dom/media/test/crashtests/1547784.html
@@ -0,0 +1,33 @@
+<html class="reftest-wait">
+<head>
+ <script>
+ const doc = new Document();
+ const video = document.createElementNS('http://www.w3.org/1999/xhtml', 'video');
+ const source = new MediaSource();
+
+ navigator.requestMediaKeySystemAccess('org.w3.clearkey', [{ '': [{ '': '' }] }])
+ .then(keySystemAccess => {
+ return keySystemAccess.createMediaKeys();
+ }).then(_ => {
+ video.src = URL.createObjectURL(source);
+ source.addEventListener('sourceopen', () => {
+ doc.adoptNode(video);
+ });
+ });
+
+ navigator.requestMediaKeySystemAccess('org.w3.clearkey', [{ '': [{ '': '' }] }])
+ .then(keySystemAccess => {
+ return keySystemAccess.createMediaKeys();
+ }).then(mediaKeys => {
+ return video.setMediaKeys(mediaKeys);
+ }).then(() => {
+ video.src = URL.createObjectURL(source);
+ document.documentElement.removeAttribute("class");
+ }).catch(e => {
+ // Catch JS errors caused by raciness in the test. So long as we're
+ // not crashing we're good.
+ document.documentElement.removeAttribute("class");
+ });
+ </script>
+</head>
+</html>
diff --git a/dom/media/test/crashtests/1547899.html b/dom/media/test/crashtests/1547899.html
new file mode 100644
index 0000000000..4ffd90c565
--- /dev/null
+++ b/dom/media/test/crashtests/1547899.html
@@ -0,0 +1,20 @@
+<html>
+<head>
+ <script>
+ function start () {
+ const video = document.getElementById('id_0')
+ const stream_1 = new MediaStream()
+ const stream_2 = video.mozCaptureStreamUntilEnded()
+ const track = stream_2.getTracks()[0]
+
+ video.srcObject = stream_1
+ stream_1.addTrack(track)
+ }
+
+ window.addEventListener('load', start)
+ </script>
+</head>
+<body>
+<video id="id_0" src="data:audio/mpeg;base64,ZQr/+1DEAAAAAAAAAAAAAAAAAAAAAABJbmZvAAAADwAAAAsAAAnKABcXFxcXFxcXFy4uLi4uLi4uLkVFRUVFRUVFRV1dXV1dXV1dXXR0dHR0dHR0dIuLi4uLi4uLi6KioqKioqKiorq6urq6urq6utHR0dHR0dHR0ejo6Ojo6Ojo6P///////////wAAADlMQU1FMy45OHIBpQAAAAAuHQAAFEAkBElCAABAAAAJyuGI2MQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA//tQxAAABmQjXHSRAAH6IfB3OVIDAADLlmjIxWKxWTyIABgmGydGjRox4f8Tg+D4OAgc/BwEDn+UBD+qwH/+8Tny7///KO4IYDgcDgcDgcDgcCgQAAAKKBUFc/nnP30a5O0zyYzMNMkzKlPva2GXgMIBbwDQEAYDQNXGLT2EIgEhoFnQNDb4BAPD9wTFYGJBWJw/8AUWgMBQgIEg4KmAKGP/wRG0csCIWDaQxcFhYvv/8WYMUpEIkXhvnRef//kmsuLPH5OvSoAFKxrJIm26DP/7UsQ">
+</body>
+</html>
diff --git a/dom/media/test/crashtests/1560215.html b/dom/media/test/crashtests/1560215.html
new file mode 100644
index 0000000000..1b76e218d7
--- /dev/null
+++ b/dom/media/test/crashtests/1560215.html
@@ -0,0 +1,20 @@
+<html class="reftest-wait">
+<head>
+ <script>
+ async function start () {
+ const canvas = document.createElement('canvas')
+ const context = canvas.getContext('2d')
+ context.fillStyle = "red"
+ context.fillRect(0, 0, 1, 1)
+ const recorder = new MediaRecorder(
+ canvas.captureStream(), { videoBitsPerSecond: 16 })
+ recorder.start(100)
+ await new Promise(r => recorder.onstart = r)
+ recorder.pause()
+ document.documentElement.removeAttribute("class")
+ }
+
+ window.addEventListener('load', start)
+ </script>
+</head>
+</html>
diff --git a/dom/media/test/crashtests/1569645.html b/dom/media/test/crashtests/1569645.html
new file mode 100644
index 0000000000..b1f1247f26
--- /dev/null
+++ b/dom/media/test/crashtests/1569645.html
@@ -0,0 +1,23 @@
+<html>
+<head>
+ <script>
+ function start () {
+ const canvas = document.getElementById("c")
+ canvas.getContext("2d")
+ const video = canvas.captureStream()
+ const ac = new AudioContext()
+ const dest = ac.createMediaStreamDestination()
+ const recorder = new MediaRecorder(
+ new MediaStream([...video.getTracks(), ...dest.stream.getTracks()]), {
+ 'mimeType': 'audio/ogg'
+ })
+ recorder.start()
+ }
+
+ window.addEventListener('load', start)
+ </script>
+</head>
+<body>
+<canvas id="c"></canvas>
+</body>
+</html>
diff --git a/dom/media/test/crashtests/1575271.html b/dom/media/test/crashtests/1575271.html
new file mode 100644
index 0000000000..ceda986553
--- /dev/null
+++ b/dom/media/test/crashtests/1575271.html
@@ -0,0 +1,25 @@
+<html class="reftest-wait">
+<head>
+<script>
+ async function start () {
+ const canvas = document.createElement("canvas")
+ const context = canvas.getContext("2d")
+ context.fillStyle = "blue"
+ context.fillRect(0, 0, canvas.width, canvas.height)
+ const stream = canvas.captureStream()
+ const track = stream.getTracks()[0]
+ const recorder = new MediaRecorder(stream)
+ recorder.start()
+ await new Promise(r => recorder.onstart = r)
+ recorder.pause()
+ stream.removeTrack(track)
+ recorder.resume()
+ await new Promise(r => recorder.onstop = r)
+ document.documentElement.removeAttribute("class")
+ }
+
+ window.addEventListener('load', start)
+</script>
+</head>
+</html>
+
diff --git a/dom/media/test/crashtests/1577184.html b/dom/media/test/crashtests/1577184.html
new file mode 100644
index 0000000000..a38c4a1265
--- /dev/null
+++ b/dom/media/test/crashtests/1577184.html
@@ -0,0 +1,15 @@
+<html>
+<head>
+<script>
+function start () {
+ const frame = document.createElementNS('http://www.w3.org/1999/xhtml', 'frame')
+ document.documentElement.appendChild(frame)
+ frame.contentWindow.eval('window.top.context=new AudioContext()')
+ document.documentElement.innerHTML = ''
+ context.createMediaElementSource(new Audio(''))
+}
+
+document.addEventListener('DOMContentLoaded', start)
+</script>
+</head>
+</html>
diff --git a/dom/media/test/crashtests/1587248.html b/dom/media/test/crashtests/1587248.html
new file mode 100644
index 0000000000..10c555e2b1
--- /dev/null
+++ b/dom/media/test/crashtests/1587248.html
@@ -0,0 +1,23 @@
+<html class="reftest-wait">
+<head>
+<script>
+function start () {
+ const audio = document.getElementById('id_4')
+ const doc = new Document()
+ const stream = new MediaStream()
+ const track = document.createElementNS('http://www.w3.org/1999/xhtml', 'track')
+ audio.srcObject = stream
+ track.textContent = '�'
+ setTimeout(() => {
+ track.replaceChild(audio, track.childNodes[0])
+ audio.play().then(function (arg4) { })
+ document.documentElement.removeAttribute("class")
+ }, 157)
+ doc.adoptNode(audio)
+}
+
+document.addEventListener('DOMContentLoaded', start)
+</script>
+</head>
+<audio class="" id="id_4" itemscope></audio>
+</html>
diff --git a/dom/media/test/crashtests/1594466.html b/dom/media/test/crashtests/1594466.html
new file mode 100644
index 0000000000..276c5fe3d1
--- /dev/null
+++ b/dom/media/test/crashtests/1594466.html
@@ -0,0 +1,22 @@
+<html>
+<head>
+<script>
+function start() {
+ const ac = new AudioContext();
+ const {stream: audioStream} = ac.createMediaStreamDestination();
+ const [audioTrack] = audioStream.getTracks();
+ const canvas = document.createElement("canvas");
+ const ctx = canvas.getContext("2d");
+ const [videoTrack] = canvas.captureStream().getTracks();
+
+ const rec = new MediaRecorder(new MediaStream([audioTrack, videoTrack]), {
+ mimeType: 'video/webm; codecs="vp8, opus"'
+ });
+ rec.start();
+}
+
+document.addEventListener('DOMContentLoaded', start)
+</script>
+</head>
+</html>
+
diff --git a/dom/media/test/crashtests/1601385.html b/dom/media/test/crashtests/1601385.html
new file mode 100644
index 0000000000..7192809076
--- /dev/null
+++ b/dom/media/test/crashtests/1601385.html
@@ -0,0 +1,12 @@
+<html>
+<head>
+<script>
+const video = document.createElement("video");
+video.srcObject = new MediaStream();
+video.mozCaptureStreamUntilEnded();
+video.src = "sound.ogg";
+video.srcObject = undefined;
+</script>
+</head>
+</html>
+
diff --git a/dom/media/test/crashtests/1601422.html b/dom/media/test/crashtests/1601422.html
new file mode 100644
index 0000000000..9ff3c1a07a
--- /dev/null
+++ b/dom/media/test/crashtests/1601422.html
@@ -0,0 +1,20 @@
+<html class="reftest-wait">
+<head>
+<script>
+(async _ => {
+ try {
+ const video = document.createElement('video')
+ video.preload = 'metadata'
+ video.src = 'sound.ogg'
+ await new Promise(r => video.onloadedmetadata = r)
+ const stream_1 = video.mozCaptureStreamUntilEnded()
+ video.src = ''
+ const stream_2 = video.mozCaptureStreamUntilEnded()
+ } finally {
+ document.documentElement.removeAttribute("class")
+ }
+})()
+</script>
+</head>
+</html>
+
diff --git a/dom/media/test/crashtests/1604941.html b/dom/media/test/crashtests/1604941.html
new file mode 100644
index 0000000000..5a9265ea1b
--- /dev/null
+++ b/dom/media/test/crashtests/1604941.html
@@ -0,0 +1,22 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<head>
+<script>
+
+async function boom()
+{
+ await SpecialPowers.pushPrefEnv({"set": [
+ ["media.cubeb.force_null_context", true],
+ ]});
+ new Audio().mozCaptureStreamUntilEnded();
+ var ac = new window.AudioContext();
+ ac.resume();
+ ac.close();
+ document.documentElement.removeAttribute("class");
+}
+
+</script>
+</head>
+<body onload="boom();"></body>
+</html>
+
diff --git a/dom/media/test/crashtests/1608286.html b/dom/media/test/crashtests/1608286.html
new file mode 100644
index 0000000000..9f52605be6
--- /dev/null
+++ b/dom/media/test/crashtests/1608286.html
@@ -0,0 +1,50 @@
+<html class="reftest-wait">
+<head>
+ <script>
+ function test() {
+ function checkResolve(value) {
+ // Let the test timeout and fail
+ throw new Error("This promise should not resolve");
+ }
+
+ function checkReject(reason) {
+ if (reason.message !== "Browsing context is no longer available") {
+ // Let the test timeout and fail
+ throw new Error("Unexpected rejected promise reason");
+ }
+ // Otherwise, successfully rejected a request not attached to a
+ // window without crashing
+ }
+
+ var i = document.querySelector("iframe");
+ var nav = i.contentWindow.navigator;
+ i.remove();
+
+ // First, check with valid args
+ nav.requestMediaKeySystemAccess(
+ "com.widevine.alpha",
+ [{
+ initDataTypes: ["webm"],
+ videoCapabilities: [{ contentType: 'video/webm; codecs="vp9"' }]
+ }]
+ ).then(
+ checkResolve,
+ (reason) => {
+ checkReject(reason);
+
+ // Then, check with invalid args
+ nav.requestMediaKeySystemAccess("", []).then(
+ checkResolve,
+ (reason) => {
+ checkReject(reason);
+ document.documentElement.removeAttribute("class");
+ }
+ );
+ });
+ }
+ </script>
+</head>
+<body onload="test()">
+ <iframe></iframe>
+</body>
+</html>
diff --git a/dom/media/test/crashtests/1673525.html b/dom/media/test/crashtests/1673525.html
new file mode 100644
index 0000000000..51de202999
--- /dev/null
+++ b/dom/media/test/crashtests/1673525.html
@@ -0,0 +1,15 @@
+<html>
+<head>
+ <script>
+ document.addEventListener("DOMContentLoaded", () => {
+ const audio = document.getElementById("audio");
+ audio.autoplay = true;
+ audio.mozCaptureStream();
+ })
+ </script>
+</head>
+<body>
+ <!-- The data URL is crafted from a fuzzed mp3 so the base64 can be decoded back to an mp3 as needed -->
+ <audio id="audio" src="data:audio/mpeg;base64,//tQxAAAAAAAAAAAAAAAAAAAAAAASW5mbwAAAA8AAAALAAAJygAXFxcXFxcXFxcuLi4uLi4uLi5FRUVFRUVFRUVdXV1dXV1dXV10dHR0dHR0dHSLi4uLi4uLi4uioqKioqKioqK6urq6urq6urrR0dHR0dHR0dHo6Ojo6Ojo6Oj///////////8AAAA5TEFNRTMuOThyAaUAAAAALh0AABRAJARJQgAAQAAACcrhiNjEAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAP/7UMQAAAZkI1x0kQAB+iHwdzlSAwAAy5ZoyMVisVk8iAAYJhsnRo0aMeH/E4Pg+DgIHPwcBA5/lAQ/qsB//vE58u///yjuCGA4HA4HA4HA4HAoEAAACigVBXP55z99GuTtM8mMzDTJMypT72thl4DCAW8A0BAGA0DVxi09hCIBIaBZ0DQ2+AQDw/cExWBiQVicP/AFFoDAUICBIOCpgChj/8ERtHLAiFg2kMXBYWL7//FmDFKRCJF4b50Xn//5JrLizx+Tr0qABSsaySJtugz/+1LEBIALgQmLvLUAMXkorjzSjuWd/+FbPOduga79Dhef0OFYNhp+FEPDPRyRvOEERl/Uwbv9F/UuQN5yt9WPM5xwqlB4+hzmfmoYmcceWPD8s3WA8FhYEPkhR0SgMHw="></audio>
+</body>
+</html>
diff --git a/dom/media/test/crashtests/1673526-1.html b/dom/media/test/crashtests/1673526-1.html
new file mode 100644
index 0000000000..925dbae6ab
--- /dev/null
+++ b/dom/media/test/crashtests/1673526-1.html
@@ -0,0 +1,20 @@
+<html>
+<head>
+<script>
+window.addEventListener('load', async () => {
+ const frame = document.createElement('frame')
+ document.documentElement.appendChild(frame)
+ const pc = new RTCPeerConnection({})
+ await pc.createOffer({ 'offerToReceiveAudio': true })
+ const [{receiver}] = pc.getTransceivers()
+ const track = pc.addTrack(receiver.track)
+ pc.removeTrack(track)
+ const [track2] = (await frame.contentWindow.navigator.mediaDevices.getUserMedia({
+ 'audio': {},
+ 'fake': true
+ })).getTracks()
+ pc.addTrack(track2)
+})
+</script>
+</head>
+</html>
diff --git a/dom/media/test/crashtests/1673526-2.html b/dom/media/test/crashtests/1673526-2.html
new file mode 100644
index 0000000000..ddae9520fc
--- /dev/null
+++ b/dom/media/test/crashtests/1673526-2.html
@@ -0,0 +1,20 @@
+<html>
+<head>
+<script>
+window.addEventListener('load', async () => {
+ const frame = document.createElement('frame')
+ document.documentElement.appendChild(frame)
+ const pc = new RTCPeerConnection({})
+ await pc.createOffer({ 'offerToReceiveAudio': true })
+ const [{sender, receiver}] = pc.getTransceivers()
+ await sender.replaceTrack(receiver.track)
+ await sender.replaceTrack(null)
+ const stream = await frame.contentWindow.navigator.mediaDevices.getUserMedia({
+ 'audio': {},
+ 'fake': true
+ })
+ await sender.replaceTrack(stream.getTracks()[0])
+})
+</script>
+</head>
+</html>
diff --git a/dom/media/test/crashtests/1693043.html b/dom/media/test/crashtests/1693043.html
new file mode 100644
index 0000000000..f73323bbd7
--- /dev/null
+++ b/dom/media/test/crashtests/1693043.html
@@ -0,0 +1,21 @@
+<html class="reftest-wait">
+<head>
+<script>
+async function boom() {
+ const audio = document.createElement("audio");
+ audio.preload = "metadata";
+ audio.src = "sound.ogg";
+ await new Promise(r => audio.onloadedmetadata = r);
+ const s = audio.mozCaptureStream();
+ const recorder = new MediaRecorder(
+ new MediaStream(s.getTracks()),
+ { audioBitsPerSecond: 3994678619 }
+ );
+ recorder.start();
+ document.documentElement.removeAttribute("class");
+}
+</script>
+</head>
+<body onload="boom()">
+</body>
+</html>
diff --git a/dom/media/test/crashtests/1696511.html b/dom/media/test/crashtests/1696511.html
new file mode 100644
index 0000000000..de66fa3a67
--- /dev/null
+++ b/dom/media/test/crashtests/1696511.html
@@ -0,0 +1,22 @@
+<!DOCTYPE html>
+<html class='reftest-wait'>
+<head>
+<script>
+document.addEventListener('DOMContentLoaded', async () => {
+ const canvas = document.createElement('canvas')
+ canvas.height = 27530
+ const context = canvas.getContext('2d')
+ context.strokeRect(5, 5, canvas.width - 5, canvas.height - 5)
+ const stream = canvas.captureStream()
+ const recorder = new MediaRecorder(stream)
+ recorder.start()
+ const stopped = new Promise(r => recorder.onstop = r)
+ await new Promise(r => setTimeout(r, 100))
+ stream.getTracks().forEach(t => t.stop())
+ await stopped
+ document.documentElement.removeAttribute("class")
+})
+</script>
+</head>
+</html>
+
diff --git a/dom/media/test/crashtests/1697521.html b/dom/media/test/crashtests/1697521.html
new file mode 100644
index 0000000000..b44c02e227
--- /dev/null
+++ b/dom/media/test/crashtests/1697521.html
@@ -0,0 +1,19 @@
+<html>
+<head>
+<script>
+async function boom() {
+ const canvas = document.createElement("canvas");
+ canvas.getContext("2d");
+ const recorder = new MediaRecorder(new MediaStream([
+ ...canvas.captureStream().getTracks(),
+ ...canvas.captureStream().getTracks(),
+ ]));
+ recorder.start();
+ recorder.requestData();
+}
+</script>
+</head>
+<body onload="boom()">
+</body>
+</html>
+
diff --git a/dom/media/test/crashtests/1708790.html b/dom/media/test/crashtests/1708790.html
new file mode 100644
index 0000000000..ee97a31c81
--- /dev/null
+++ b/dom/media/test/crashtests/1708790.html
@@ -0,0 +1,22 @@
+<html class="reftest-wait">
+<head>
+<script>
+(async _ => {
+ try {
+ const audio = document.createElement('audio');
+ audio.src = 'sound.ogg';
+ const ac = new AudioContext();
+ const src = ac.createMediaElementSource(audio);
+ src.connect(ac.destination);
+ await audio.play();
+ audio.pause();
+ audio.volume = 0.5;
+ audio.playbackRate = 0.5;
+ audio.preservesPitch = false;
+ } finally {
+ document.documentElement.removeAttribute("class")
+ }
+})()
+</script>
+</head>
+</html>
diff --git a/dom/media/test/crashtests/1709130.html b/dom/media/test/crashtests/1709130.html
new file mode 100644
index 0000000000..2c6d7dc690
--- /dev/null
+++ b/dom/media/test/crashtests/1709130.html
@@ -0,0 +1,19 @@
+<html class="reftest-wait">
+<head>
+<script>
+(async _ => {
+ const video = document.createElement("video");
+ video.mozCaptureStreamUntilEnded();
+ video.src = "test.mp4";
+ await video.play();
+ for (let i = 0; i < 23; i++) {
+ console.log("nop");
+ await video.seekToNextFrame();
+ video.removeAttribute("readonly");
+ }
+ setTimeout("self.close()", 100);
+ document.documentElement.removeAttribute("class");
+})();
+</script>
+</head>
+</html>
diff --git a/dom/media/test/crashtests/1734008.html b/dom/media/test/crashtests/1734008.html
new file mode 100644
index 0000000000..31c8d31dae
--- /dev/null
+++ b/dom/media/test/crashtests/1734008.html
@@ -0,0 +1,22 @@
+<!DOCTYPE html>
+<html>
+<head>
+ <script>
+ window.addEventListener("load", async () => {
+ const object = document.createElementNS("http://www.w3.org/2000/svg", "foreignObject")
+ const video = document.createElementNS("http://www.w3.org/1999/xhtml", "video")
+ document.documentElement.appendChild(video)
+ video.setAttribute("src", "1734008.webm")
+ video.mozCaptureStream()
+ const xhr = new XMLHttpRequest()
+ xhr.open("POST", "FOOBAR", false)
+ xhr.send()
+ const document_0 = new Document()
+ const adopted = document_0.adoptNode(document.documentElement)
+ object.insertBefore(adopted, object.childNodes[(216679474 % object.childNodes.length)])
+ try { await video.play() } catch (e) {}
+ setTimeout("location.reload()", 500)
+ })
+ </script>
+</head>
+</html>
diff --git a/dom/media/test/crashtests/1734008.webm b/dom/media/test/crashtests/1734008.webm
new file mode 100644
index 0000000000..07d2052879
--- /dev/null
+++ b/dom/media/test/crashtests/1734008.webm
Binary files differ
diff --git a/dom/media/test/crashtests/1741677.html b/dom/media/test/crashtests/1741677.html
new file mode 100644
index 0000000000..0c442e830d
--- /dev/null
+++ b/dom/media/test/crashtests/1741677.html
@@ -0,0 +1,15 @@
+<!DOCTYPE html>
+<meta charset="utf-8">
+<html class="reftest-wait">
+<script>
+ const script = `
+ navigator.mediaCapabilities.decodingInfo({
+ type: 'file',
+ audio: { contentType: "audio/flac" }
+ }).then(postMessage("done"));
+ `;
+ let worker = new Worker(URL.createObjectURL(new Blob([script])));
+ // Stop test completion and crash from racing each other.
+ worker.onmessage = () => document.documentElement.removeAttribute("class");
+</script>
+</html>
diff --git a/dom/media/test/crashtests/1748272.html b/dom/media/test/crashtests/1748272.html
new file mode 100644
index 0000000000..53953e7c0e
--- /dev/null
+++ b/dom/media/test/crashtests/1748272.html
@@ -0,0 +1,12 @@
+<!DOCTYPE html>
+<html>
+<head>
+ <script>
+ window.addEventListener("load", () => {
+ const audio = new Audio();
+ const context = new AudioContext({ "sampleRate": 15772.7 })
+ context.createMediaElementSource(audio);
+ });
+ </script>
+</head>
+</html>
diff --git a/dom/media/test/crashtests/1752917.html b/dom/media/test/crashtests/1752917.html
new file mode 100644
index 0000000000..84752a174f
--- /dev/null
+++ b/dom/media/test/crashtests/1752917.html
@@ -0,0 +1,18 @@
+<!DOCTYPE html>
+<html>
+<head>
+ <script>
+ window.addEventListener("load", () => {
+ const context = new AudioContext({})
+ const node = new DelayNode(context, {})
+ const abort = new AbortController()
+ let processor = context.createScriptProcessor(4096, 1, 26)
+ processor.addEventListener("audioprocess", () => {}, { "signal": abort.signal })
+ processor.connect(node.delayTime)
+ processor = undefined
+ SpecialPowers.forceGC()
+ abort.abort(undefined)
+ });
+ </script>
+</head>
+</html>
diff --git a/dom/media/test/crashtests/1762620.html b/dom/media/test/crashtests/1762620.html
new file mode 100644
index 0000000000..b4c705c84f
--- /dev/null
+++ b/dom/media/test/crashtests/1762620.html
@@ -0,0 +1,8 @@
+<audio id='fuzzed' controls>
+ <source src='fuzzed.wav'>
+</audio>
+<script>
+ var m = document.getElementById('fuzzed')
+ m.addEventListener('canplay', m.play, true)
+</script>
+
diff --git a/dom/media/test/crashtests/1765842.html b/dom/media/test/crashtests/1765842.html
new file mode 100644
index 0000000000..8b77ec998a
--- /dev/null
+++ b/dom/media/test/crashtests/1765842.html
@@ -0,0 +1,8 @@
+<!DOCTYPE html>
+<html>
+<head>
+</head>
+<body>
+ <video src="1765842.webm" type="video/webm" autoplay="true" controls></video>
+</body>
+</html>
diff --git a/dom/media/test/crashtests/1765842.webm b/dom/media/test/crashtests/1765842.webm
new file mode 100644
index 0000000000..62eb5d779d
--- /dev/null
+++ b/dom/media/test/crashtests/1765842.webm
Binary files differ
diff --git a/dom/media/test/crashtests/1787281.html b/dom/media/test/crashtests/1787281.html
new file mode 100644
index 0000000000..e5586dbf2f
--- /dev/null
+++ b/dom/media/test/crashtests/1787281.html
@@ -0,0 +1,13 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<head>
+</head>
+<body>
+ <video id=a src="1787281.mp4" autoplay="true" controls></video>
+ <script>
+ a.onended = function() {
+ document.documentElement.className = "";
+ }
+ </script>
+</body>
+</html>
diff --git a/dom/media/test/crashtests/1787281.mp4 b/dom/media/test/crashtests/1787281.mp4
new file mode 100644
index 0000000000..8481cf034c
--- /dev/null
+++ b/dom/media/test/crashtests/1787281.mp4
Binary files differ
diff --git a/dom/media/test/crashtests/1798778.html b/dom/media/test/crashtests/1798778.html
new file mode 100644
index 0000000000..300d84afa1
--- /dev/null
+++ b/dom/media/test/crashtests/1798778.html
@@ -0,0 +1,11 @@
+<html class="reftest-wait">
+<head>
+<audio id=a src="adts-truncated.aac" controls></audio>
+<script>
+ a.play();
+ a.onerror = function() {
+ document.documentElement.removeAttribute("class");
+ }
+</script>
+</head>
+</html>
diff --git a/dom/media/test/crashtests/1830206.html b/dom/media/test/crashtests/1830206.html
new file mode 100644
index 0000000000..69f72176fd
--- /dev/null
+++ b/dom/media/test/crashtests/1830206.html
@@ -0,0 +1,12 @@
+<html class="reftest-wait">
+<video id='a'>
+ <source src='1830206.mp4'>
+</video>
+<script>
+ function done() {
+ document.documentElement.removeAttribute("class");
+ }
+ a.addEventListener('error', done, true)
+ a.addEventListener('ended', done, true)
+ a.play();
+</script>
diff --git a/dom/media/test/crashtests/1830206.mp4 b/dom/media/test/crashtests/1830206.mp4
new file mode 100644
index 0000000000..5a95a4dffa
--- /dev/null
+++ b/dom/media/test/crashtests/1830206.mp4
Binary files differ
diff --git a/dom/media/test/crashtests/1833894.mp4 b/dom/media/test/crashtests/1833894.mp4
new file mode 100644
index 0000000000..a5c20e4846
--- /dev/null
+++ b/dom/media/test/crashtests/1833894.mp4
Binary files differ
diff --git a/dom/media/test/crashtests/1833896.mp4 b/dom/media/test/crashtests/1833896.mp4
new file mode 100644
index 0000000000..6d40590c8a
--- /dev/null
+++ b/dom/media/test/crashtests/1833896.mp4
Binary files differ
diff --git a/dom/media/test/crashtests/1835118.adts b/dom/media/test/crashtests/1835118.adts
new file mode 100644
index 0000000000..cf68dfb274
--- /dev/null
+++ b/dom/media/test/crashtests/1835118.adts
Binary files differ
diff --git a/dom/media/test/crashtests/1835164.html b/dom/media/test/crashtests/1835164.html
new file mode 100644
index 0000000000..702e082954
--- /dev/null
+++ b/dom/media/test/crashtests/1835164.html
@@ -0,0 +1,13 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<audio id=a src=1835164.opus controls autoplay></audio>
+<script>
+a.onloadedmetadata = function() {
+ a.currentTime = 1e40;
+ a.play();
+}
+a.onplaying = function() {
+ document.documentElement.removeAttribute("class");
+}
+</script>
+</html>
diff --git a/dom/media/test/crashtests/1835164.opus b/dom/media/test/crashtests/1835164.opus
new file mode 100644
index 0000000000..124fdbfda1
--- /dev/null
+++ b/dom/media/test/crashtests/1835164.opus
Binary files differ
diff --git a/dom/media/test/crashtests/1839193.adts b/dom/media/test/crashtests/1839193.adts
new file mode 100644
index 0000000000..0bc06397cb
--- /dev/null
+++ b/dom/media/test/crashtests/1839193.adts
Binary files differ
diff --git a/dom/media/test/crashtests/1839193.html b/dom/media/test/crashtests/1839193.html
new file mode 100644
index 0000000000..9892908882
--- /dev/null
+++ b/dom/media/test/crashtests/1839193.html
@@ -0,0 +1,9 @@
+<html class="reftest-wait">
+<audio id=a src=1839193.adts controls></audio>
+<script>
+ a.play();
+ a.onerror = a.onended = () => {
+ document.documentElement.removeAttribute("class")
+ }
+</script>
+</html>
diff --git a/dom/media/test/crashtests/1840002.webm b/dom/media/test/crashtests/1840002.webm
new file mode 100644
index 0000000000..0ad6d513e7
--- /dev/null
+++ b/dom/media/test/crashtests/1840002.webm
Binary files differ
diff --git a/dom/media/test/crashtests/1845350.mp4 b/dom/media/test/crashtests/1845350.mp4
new file mode 100644
index 0000000000..e22af449c2
--- /dev/null
+++ b/dom/media/test/crashtests/1845350.mp4
Binary files differ
diff --git a/dom/media/test/crashtests/1848660.html b/dom/media/test/crashtests/1848660.html
new file mode 100644
index 0000000000..7bebb9c896
--- /dev/null
+++ b/dom/media/test/crashtests/1848660.html
@@ -0,0 +1,10 @@
+<html class="reftest-wait">
+<audio id='a'>
+<source src='1848660.wav'>
+</audio>
+<script>
+function done() {
+ document.documentElement.removeAttribute("class");
+}
+a.addEventListener('error', done, true)
+</script>
diff --git a/dom/media/test/crashtests/1848660.wav b/dom/media/test/crashtests/1848660.wav
new file mode 100644
index 0000000000..e69de29bb2
--- /dev/null
+++ b/dom/media/test/crashtests/1848660.wav
diff --git a/dom/media/test/crashtests/1850453.flac b/dom/media/test/crashtests/1850453.flac
new file mode 100644
index 0000000000..dfca92a634
--- /dev/null
+++ b/dom/media/test/crashtests/1850453.flac
Binary files differ
diff --git a/dom/media/test/crashtests/1850453.html b/dom/media/test/crashtests/1850453.html
new file mode 100644
index 0000000000..d2aa54bc6a
--- /dev/null
+++ b/dom/media/test/crashtests/1850453.html
@@ -0,0 +1,12 @@
+<html class="reftest-wait">
+<audio id=a src=1850453.flac controls></audio>
+<script>
+ a.play();
+ a.onplaying = function() {
+ a.currentTime = 0.5;
+ }
+ a.onseeked = function () {
+ document.documentElement.removeAttribute("class")
+ }
+</script>
+</html>
diff --git a/dom/media/test/crashtests/1859384.mp4 b/dom/media/test/crashtests/1859384.mp4
new file mode 100644
index 0000000000..b0efba5718
--- /dev/null
+++ b/dom/media/test/crashtests/1859384.mp4
Binary files differ
diff --git a/dom/media/test/crashtests/1859600.mp4 b/dom/media/test/crashtests/1859600.mp4
new file mode 100644
index 0000000000..9bfd9fa359
--- /dev/null
+++ b/dom/media/test/crashtests/1859600.mp4
Binary files differ
diff --git a/dom/media/test/crashtests/1860840.mp4 b/dom/media/test/crashtests/1860840.mp4
new file mode 100644
index 0000000000..d3736a232f
--- /dev/null
+++ b/dom/media/test/crashtests/1860840.mp4
Binary files differ
diff --git a/dom/media/test/crashtests/1864450.html b/dom/media/test/crashtests/1864450.html
new file mode 100644
index 0000000000..1d2d758ff8
--- /dev/null
+++ b/dom/media/test/crashtests/1864450.html
@@ -0,0 +1,12 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<script>
+ const ac = new OfflineAudioContext({length: 136, sampleRate: 48000});
+ const filterNode = new BiquadFilterNode(ac);
+ filterNode.frequency.setValueCurveAtTime([0.2, 0.3], 0, 1);
+ filterNode.frequency.linearRampToValueAtTime(0.8, 1);
+ ac.startRendering().then(() => {
+ filterNode.frequency.value;
+ document.documentElement.removeAttribute("class");
+ });
+</script>
diff --git a/dom/media/test/crashtests/1872787.html b/dom/media/test/crashtests/1872787.html
new file mode 100644
index 0000000000..d02a67138e
--- /dev/null
+++ b/dom/media/test/crashtests/1872787.html
@@ -0,0 +1,14 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<title>crashtest for bug 1872787</title>
+<script>
+document.addEventListener("DOMContentLoaded", () => {
+ let a = document.createElementNS("http://www.w3.org/1999/xhtml", "video");
+ document.documentElement.appendChild(a);
+ let b = new AudioContext({"sampleRate": 12000});
+ let _ = new MediaElementAudioSourceNode(b, {"mediaElement": a});
+ a.mozCaptureStreamUntilEnded();
+ // There is nothing to wait for but that no exceptions are thrown.
+ document.documentElement.removeAttribute("class");
+})
+</script>
diff --git a/dom/media/test/crashtests/255ch.wav b/dom/media/test/crashtests/255ch.wav
new file mode 100644
index 0000000000..d9ee36d5bf
--- /dev/null
+++ b/dom/media/test/crashtests/255ch.wav
Binary files differ
diff --git a/dom/media/test/crashtests/459439-1.html b/dom/media/test/crashtests/459439-1.html
new file mode 100644
index 0000000000..7bb0131d51
--- /dev/null
+++ b/dom/media/test/crashtests/459439-1.html
@@ -0,0 +1,36 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<head>
+<script type="text/javascript">
+var i = 0;
+function boom()
+{
+ var div = document.getElementById("div");
+ var audio = document.getElementById("audio");
+
+ audio.onload = null;
+
+ div.textContent = "FAIL";
+ audio.src += "";
+ div.textContent = "PASS?";
+
+ ++i;
+
+ setTimeout(done, 1);
+}
+
+function done()
+{
+ // Note we reset 'src' to release decoder resources and cubeb streams to
+ // prevent OOM or OpenCubeb() failures.
+ var audio = document.getElementById("audio");
+ audio.src = "";
+ document.documentElement.removeAttribute("class");
+}
+</script>
+</head>
+<body>
+<audio id="audio" autoplay src="sound.ogg" oncanplaythrough="setTimeout(boom, 1);"></audio>
+<div id="div"></div>
+</body>
+</html>
diff --git a/dom/media/test/crashtests/466607-1.html b/dom/media/test/crashtests/466607-1.html
new file mode 100644
index 0000000000..e154223efe
--- /dev/null
+++ b/dom/media/test/crashtests/466607-1.html
@@ -0,0 +1,14 @@
+<html>
+<head>
+<script type="text/javascript">
+
+function boom()
+{
+ document.body.appendChild(document.createElementNS("bar", "audio"));
+ document.body.appendChild(document.createElementNS("bar", "video"));
+}
+
+</script>
+</head>
+<body onload="boom();"></body>
+</html>
diff --git a/dom/media/test/crashtests/466945-1.html b/dom/media/test/crashtests/466945-1.html
new file mode 100644
index 0000000000..ac1ba29e36
--- /dev/null
+++ b/dom/media/test/crashtests/466945-1.html
@@ -0,0 +1,25 @@
+<html class="reftest-wait">
+<head>
+<script type="text/javascript">
+
+var s;
+
+function boom()
+{
+ s = document.createElement("span");
+ s.innerHTML = "<video src='data:text/html,' autoplay='autoplay'><\/video>";
+ document.body.appendChild(document.createElement("iframe"));
+ setTimeout(boom2, 0);
+}
+
+function boom2()
+{
+ s.innerHTML = "";
+ document.documentElement.removeAttribute("class");
+}
+
+</script>
+</head>
+
+<body onload="setTimeout(boom, 0);"></body>
+</html>
diff --git a/dom/media/test/crashtests/468763-1.html b/dom/media/test/crashtests/468763-1.html
new file mode 100644
index 0000000000..d21e5bc388
--- /dev/null
+++ b/dom/media/test/crashtests/468763-1.html
@@ -0,0 +1 @@
+<html><head></head><body><video src="nosuchprotocol:"></video></body></html> \ No newline at end of file
diff --git a/dom/media/test/crashtests/474744-1.html b/dom/media/test/crashtests/474744-1.html
new file mode 100644
index 0000000000..8a7c70cf05
--- /dev/null
+++ b/dom/media/test/crashtests/474744-1.html
@@ -0,0 +1,15 @@
+<!DOCTYPE html>
+<html>
+<head>
+<script type="text/javascript">
+
+function boom()
+{
+ document.body.innerHTML += "<video src='aim:yaz'><\/video>";
+}
+
+</script>
+</head>
+<body onload="boom();">
+</body>
+</html>
diff --git a/dom/media/test/crashtests/481136-1.html b/dom/media/test/crashtests/481136-1.html
new file mode 100644
index 0000000000..bd264058d9
--- /dev/null
+++ b/dom/media/test/crashtests/481136-1.html
@@ -0,0 +1,3 @@
+<html>
+<div><object data='sound.ogg'></div>
+</html>
diff --git a/dom/media/test/crashtests/492286-1.xhtml b/dom/media/test/crashtests/492286-1.xhtml
new file mode 100644
index 0000000000..627ac38723
--- /dev/null
+++ b/dom/media/test/crashtests/492286-1.xhtml
@@ -0,0 +1 @@
+<source xmlns="http://www.w3.org/1999/xhtml"/> \ No newline at end of file
diff --git a/dom/media/test/crashtests/493915-1.html b/dom/media/test/crashtests/493915-1.html
new file mode 100644
index 0000000000..2a6ae9bd6c
--- /dev/null
+++ b/dom/media/test/crashtests/493915-1.html
@@ -0,0 +1,18 @@
+<html>
+<head>
+<script type="text/javascript">
+
+function boom()
+{
+ s = document.createElement("span");
+ a = document.createElement("audio");
+ a['src'] = "javascript:4";
+ a['loopend'] = 3;
+ s.appendChild(a);
+}
+
+</script>
+</head>
+
+<body onload="boom();"></body>
+</html>
diff --git a/dom/media/test/crashtests/495794-1.html b/dom/media/test/crashtests/495794-1.html
new file mode 100644
index 0000000000..2db69206a1
--- /dev/null
+++ b/dom/media/test/crashtests/495794-1.html
@@ -0,0 +1,8 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+ <body>
+ <audio src="495794-1.ogg" autoplay onended="this.src=''; document.documentElement.className=undefined"></audio>
+ <!-- Note we reset 'src' to release decoder resources and cubeb streams to prevent OOM or OpenCubeb() failures. -->
+ </body>
+</html>
+
diff --git a/dom/media/test/crashtests/495794-1.ogg b/dom/media/test/crashtests/495794-1.ogg
new file mode 100644
index 0000000000..1c19a64061
--- /dev/null
+++ b/dom/media/test/crashtests/495794-1.ogg
Binary files differ
diff --git a/dom/media/test/crashtests/497734-1.xhtml b/dom/media/test/crashtests/497734-1.xhtml
new file mode 100644
index 0000000000..6df055da39
--- /dev/null
+++ b/dom/media/test/crashtests/497734-1.xhtml
@@ -0,0 +1,21 @@
+<html xmlns="http://www.w3.org/1999/xhtml">
+<head>
+<script type="text/javascript">
+
+function boom()
+{
+
+ div = document.createElementNS("http://www.w3.org/1999/xhtml", "div");
+ div.appendChild(document.getElementById("v"));
+ document.body.appendChild(div);
+}
+
+</script>
+</head>
+
+<body onload="boom();">
+
+<video id="v"><source></source></video>
+
+</body>
+</html>
diff --git a/dom/media/test/crashtests/497734-2.html b/dom/media/test/crashtests/497734-2.html
new file mode 100644
index 0000000000..990ac4af46
--- /dev/null
+++ b/dom/media/test/crashtests/497734-2.html
@@ -0,0 +1,17 @@
+<!DOCTYPE html>
+<html>
+<script>
+
+function boom()
+{
+ audio1 = document.createElement("audio");
+ (audio1).appendChild(document.createElement("source"));
+ (audio1).appendChild(document.createElement("source"));
+ setTimeout(function() {
+ audio2 = document.createElement("audio");
+ audio2.appendChild(audio1);
+ }, 100);
+}
+
+</script>
+<body onload="boom();"></body>
diff --git a/dom/media/test/crashtests/576612-1.html b/dom/media/test/crashtests/576612-1.html
new file mode 100644
index 0000000000..04f993e780
--- /dev/null
+++ b/dom/media/test/crashtests/576612-1.html
@@ -0,0 +1,15 @@
+<html xmlns="http://www.w3.org/1999/xhtml">
+<head>
+<script>
+function boom()
+{
+
+ var v = document.getElementById("v");
+ v.src = "data:text/plain,_";
+ document.documentElement.appendChild(v);
+
+}
+</script>
+</head>
+<body onload="boom();"><video id="v" src="data:video/ogg;codecs=&quot;theora,vorbis&quot;,1"></video></body>
+</html>
diff --git a/dom/media/test/crashtests/691096-1.html b/dom/media/test/crashtests/691096-1.html
new file mode 100644
index 0000000000..3c3ebfc12a
--- /dev/null
+++ b/dom/media/test/crashtests/691096-1.html
@@ -0,0 +1,31 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<head>
+<script>
+var ITERATIONS = 200;
+
+function stoptest (evt)
+{
+ if (evt) {
+ // Note we reset 'src' to release decoder resources and cubeb streams to
+ // prevent OOM or OpenCubeb() failures.
+ evt.target.src = "";
+ }
+ document.documentElement.removeAttribute("class");
+}
+
+function boom()
+{
+ for (var i = 0; i < ITERATIONS; ++i) {
+ a = document.createElementNS("http://www.w3.org/1999/xhtml", "audio");
+ a.addEventListener("loadedmetadata", stoptest);
+ a.setAttributeNS(null, "autoplay", "");
+ a.setAttributeNS(null, "src", "sound.ogg");
+ }
+ setTimeout(stoptest, 250);
+}
+
+</script>
+</head>
+<body onload="boom();"></body>
+</html>
diff --git a/dom/media/test/crashtests/752784-1.html b/dom/media/test/crashtests/752784-1.html
new file mode 100644
index 0000000000..4644eeb89a
--- /dev/null
+++ b/dom/media/test/crashtests/752784-1.html
@@ -0,0 +1,15 @@
+<!DOCTYPE html>
+<html>
+<head>
+<script>
+function boom()
+{
+ document.getElementById("a").mozCaptureStream();
+}
+</script>
+</head>
+
+<body onload="boom();">
+<audio id="a" src="sound.ogg">
+</body>
+</html>
diff --git a/dom/media/test/crashtests/789075-1.html b/dom/media/test/crashtests/789075-1.html
new file mode 100644
index 0000000000..6cd673fb75
--- /dev/null
+++ b/dom/media/test/crashtests/789075-1.html
@@ -0,0 +1,20 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<head>
+</head>
+<body>
+<video src="789075.webm" preload="metadata" id="v">
+</video>
+<!-- Note we reset 'src' to release decoder resources and cubeb streams to prevent OOM or OpenCubeb() failures. -->
+<script type="application/javascript">
+ var video = document.getElementById("v");
+ video.onloadeddata = function () {
+ video.play();
+ };
+ video.onended = function () {
+ video.src="";
+ document.documentElement.className = undefined;
+ };
+</script>
+</body>
+</html>
diff --git a/dom/media/test/crashtests/789075.webm b/dom/media/test/crashtests/789075.webm
new file mode 100644
index 0000000000..602e53fca2
--- /dev/null
+++ b/dom/media/test/crashtests/789075.webm
Binary files differ
diff --git a/dom/media/test/crashtests/795892-1.html b/dom/media/test/crashtests/795892-1.html
new file mode 100644
index 0000000000..d73cea7f2e
--- /dev/null
+++ b/dom/media/test/crashtests/795892-1.html
@@ -0,0 +1,23 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<head>
+<script>
+function boom()
+{
+ var a = document.getElementById("a");
+ a.play();
+ a.onplaying = function () {
+ a.onplaying = null;
+ // Note we reset 'src' to release decoder resources and cubeb streams to
+ // prevent OOM or OpenCubeb() failures.
+ a.src = "";
+ document.documentElement.className = "";
+ }
+}
+</script>
+</head>
+
+<body>
+<video id="a" src="cors.webm" crossorigin preload="metadata" onloadedmetadata="boom();">
+</body>
+</html>
diff --git a/dom/media/test/crashtests/844563.html b/dom/media/test/crashtests/844563.html
new file mode 100644
index 0000000000..a3cb91838d
--- /dev/null
+++ b/dom/media/test/crashtests/844563.html
@@ -0,0 +1,5 @@
+<script>
+var a = document.createElementNS("http://www.w3.org/1999/xhtml", "audio");
+a.preservesPitch = a;
+</script>
+
diff --git a/dom/media/test/crashtests/846612.html b/dom/media/test/crashtests/846612.html
new file mode 100644
index 0000000000..070c375381
--- /dev/null
+++ b/dom/media/test/crashtests/846612.html
@@ -0,0 +1,8 @@
+<script>
+document.addEventListener("DOMContentLoaded", function() {
+ var a = document.querySelector("audio");
+ a.playbackRate = 2;
+ a.play();
+});
+</script>
+<audio src="495794-1.ogg"></audio>
diff --git a/dom/media/test/crashtests/852838.html b/dom/media/test/crashtests/852838.html
new file mode 100644
index 0000000000..0bea29351b
--- /dev/null
+++ b/dom/media/test/crashtests/852838.html
@@ -0,0 +1,11 @@
+<!DOCTYPE html>
+<html>
+ <head>
+ <script>
+ var o0 = new window.AudioContext();
+ var o1 = o0.createBuffer(536870912, 1, 8192);
+ </script>
+ </head>
+ <body>
+ </body>
+</html>
diff --git a/dom/media/test/crashtests/865004.html b/dom/media/test/crashtests/865004.html
new file mode 100644
index 0000000000..4da39071c4
--- /dev/null
+++ b/dom/media/test/crashtests/865004.html
@@ -0,0 +1,19 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<head>
+<meta charset="UTF-8">
+<script>
+
+function boom()
+{
+ var ac = new AudioContext();
+ for (var j = 0; j < 200; ++j) {
+ ac.createScriptProcessor(undefined);
+ }
+ document.documentElement.removeAttribute("class");
+}
+
+</script>
+</head>
+<body onload="boom();"></body>
+</html>
diff --git a/dom/media/test/crashtests/865537-1.html b/dom/media/test/crashtests/865537-1.html
new file mode 100644
index 0000000000..4065eb3608
--- /dev/null
+++ b/dom/media/test/crashtests/865537-1.html
@@ -0,0 +1,13 @@
+<!DOCTYPE HTML>
+<html>
+<body onload="doTest()">
+<base href="../unknown">
+<div id="test3"></div>
+<video id="test4"><source src="white.webm"></video>
+<script>
+function doTest() {
+ test3.appendChild(test4);
+}
+</script>
+</body>
+</html>
diff --git a/dom/media/test/crashtests/865550.html b/dom/media/test/crashtests/865550.html
new file mode 100644
index 0000000000..b8626e8d67
--- /dev/null
+++ b/dom/media/test/crashtests/865550.html
@@ -0,0 +1,22 @@
+<html class="reftest-wait">
+ <head>
+ <script>
+ var i = 0;
+ var interval;
+ function crash() {
+ var o0 = new AudioContext();
+ o1 = o0.createBufferSource();
+ ++i;
+ if (i == 2000) {
+ document.documentElement.removeAttribute("class");
+ clearInterval(interval);
+ }
+ }
+ function start() {
+ interval = setInterval("crash()", 0)
+ }
+ </script>
+ </head>
+ <body onload="start()">
+ </body>
+</html>
diff --git a/dom/media/test/crashtests/868504.html b/dom/media/test/crashtests/868504.html
new file mode 100644
index 0000000000..94090c1c09
--- /dev/null
+++ b/dom/media/test/crashtests/868504.html
@@ -0,0 +1,14 @@
+<!DOCTYPE html>
+<html>
+<head>
+<script>
+
+function boom()
+{
+ new AudioContext().createBufferSource().playbackRate.linearRampToValueAtTime(0, -1);
+}
+
+</script>
+</head>
+<body onload="boom();"></body>
+</html>
diff --git a/dom/media/test/crashtests/874869.html b/dom/media/test/crashtests/874869.html
new file mode 100644
index 0000000000..1fe3dbd2fc
--- /dev/null
+++ b/dom/media/test/crashtests/874869.html
@@ -0,0 +1,15 @@
+<script>
+var Context0= new AudioContext()
+var Panner0=Context0.createPanner();
+var BufferSource3=Context0.createBufferSource();
+Panner0.channelCount=0;
+BufferSource3.connect(Panner0);
+BufferSource3.buffer=function(){
+ var length=1;
+ var Buffer=Context0.createBuffer(1,length,'44200');
+ var bufferData= Buffer.getChannelData(0);
+ for (var i = 0; i < length; ++i) { bufferData[i] = 255;};
+ return Buffer;
+}();
+
+</script> \ No newline at end of file
diff --git a/dom/media/test/crashtests/874915.html b/dom/media/test/crashtests/874915.html
new file mode 100644
index 0000000000..59218b3da3
--- /dev/null
+++ b/dom/media/test/crashtests/874915.html
@@ -0,0 +1,24 @@
+<script>
+var Context0= new AudioContext()
+var BufferSource6=Context0.createBufferSource();
+
+setInterval(function(){
+BufferSource6.buffer=function(){
+ var length=11283;
+ var Buffer=Context0.createBuffer(1,length,Context0.sampleRate);
+ var bufferData= Buffer.getChannelData(0);
+ for (var i = 0; i < length; ++i) { bufferData[i] = Math.sin(i*(624))};
+ return Buffer;
+}();
+},0)
+
+BufferSource6.start(0.15831333969254047,0.23571860056836158,0.529235512483865);
+
+BufferSource6.buffer=function(){
+ var length=48517;
+ var Buffer=Context0.createBuffer(1,length,Context0.sampleRate);
+ var bufferData= Buffer.getChannelData(0);
+ for (var i = 0; i < length; ++i) { bufferData[i] = Math.sin(i*(365))};
+ return Buffer;
+}();
+</script> \ No newline at end of file
diff --git a/dom/media/test/crashtests/874934.html b/dom/media/test/crashtests/874934.html
new file mode 100644
index 0000000000..350ce28101
--- /dev/null
+++ b/dom/media/test/crashtests/874934.html
@@ -0,0 +1,23 @@
+<script>
+var Context0= new AudioContext()
+var BufferSource0=Context0.createBufferSource();
+BufferSource0.start(0.01932738965842873,0.33345631847623736,0.3893404237460345);
+BufferSource0.buffer=function(){
+ var length=35887;
+ var Buffer=Context0.createBuffer(1,length,Context0.sampleRate);
+ var bufferData= Buffer.getChannelData(0);
+ for (var i = 0; i < length; ++i) { bufferData[i] = Math.sin(i*(0.39765272522345185))};
+ return Buffer;
+}();
+
+BufferSource0.buffer=function(){
+ var length=15952;
+ var Buffer=Context0.createBuffer(1,length,Context0.sampleRate);
+ var bufferData= Buffer.getChannelData(0);
+ for (var i = 0; i < length; ++i) { bufferData[i] = Math.sin(i*(85))};
+ return Buffer;
+}();
+
+BufferSource0.playbackRate.value=20.401213286832185;
+
+</script>
diff --git a/dom/media/test/crashtests/874952.html b/dom/media/test/crashtests/874952.html
new file mode 100644
index 0000000000..a9a398fdb2
--- /dev/null
+++ b/dom/media/test/crashtests/874952.html
@@ -0,0 +1,11 @@
+<script>
+var Context0= new AudioContext()
+var ChannelSplitter0=Context0.createChannelSplitter();
+var BiquadFilter0=Context0.createBiquadFilter();
+var WaveShaper0=Context0.createWaveShaper();
+
+ChannelSplitter0.connect(BiquadFilter0,3,0);
+ChannelSplitter0.connect(WaveShaper0);
+BiquadFilter0.disconnect();
+WaveShaper0.connect(ChannelSplitter0);
+</script>
diff --git a/dom/media/test/crashtests/875144.html b/dom/media/test/crashtests/875144.html
new file mode 100644
index 0000000000..bf5d0d0861
--- /dev/null
+++ b/dom/media/test/crashtests/875144.html
@@ -0,0 +1,81 @@
+<script>
+Logger = {}
+Logger.error = function(e) {}
+Logger.comment = function(e) {}
+
+try { o0 = document.createElement('audio'); } catch(e) { Logger.error(Logger.comment(e)); }
+try { (document.body || document.documentElement).appendChild(o0); } catch(e) { Logger.error(Logger.comment(e)); }
+try { o1 = new AudioContext(); } catch(e) { Logger.error(Logger.comment(e)); }
+try { o2 = o1.createGain(); } catch(e) { Logger.error(Logger.comment(e)); }
+try { o3 = o1.createBufferSource(); } catch(e) { Logger.error(Logger.comment(e)); }
+try { o3.buffer = function() { o4 = o1.createBuffer(1, 3, 52970);
+o5 = o4.getChannelData(0);
+for(var i=0; i<3; ++i) {
+o5[i] = Math.sin(i * 63);
+}
+return o4;
+}(); } catch(e) { Logger.error(Logger.comment(e)); }
+try { o3.buffer = function() { o6 = o1.createBuffer(1, 15, 41218);
+o7 = o6.getChannelData(0);
+for(var i=0; i<15; ++i) {
+o7[i] = Math.sin(i * 0);
+}
+return o6;
+}(); } catch(e) { Logger.error(Logger.comment(e)); }
+try { o3.buffer = function() { o8 = o1.createBuffer(1, 0, 49074);
+o9 = o8.getChannelData(0);
+for(var i=0; i<0; ++i) {
+o9[i] = Math.sin(i * 0);
+}
+return o8;
+}(); } catch(e) { Logger.error(Logger.comment(e)); }
+try { o3.buffer = function() { o10 = o1.createBuffer(1, 31, 86527);
+o11 = o10.getChannelData(0);
+for(var i=0; i<31; ++i) {
+o11[i] = Math.sin(i * 127);
+}
+return o10;
+}(); } catch(e) { Logger.error(Logger.comment(e)); }
+try { o3.stop(-1) } catch(e) { Logger.error(Logger.comment(e)); }
+/* [Exception... "An attempt was made to use an object that is not, or is no longer, usable" code: "11" nsresult: "0x8053000b (InvalidStateError)" location: "file:///Users/cdiehl/dev/projects/peach/Peach/Utilities/JS/undefined.js Line: 602"] */
+try { o3.channelCountMode = 'explicit'; } catch(e) { Logger.error(Logger.comment(e)); }
+try { o12 = o1.createBiquadFilter(); } catch(e) { Logger.error(Logger.comment(e)); }
+try { o3.buffer = function() { o13 = o1.createBuffer(1, 63, 28347);
+o14 = o13.getChannelData(0);
+for(var i=0; i<63; ++i) {
+o14[i] = Math.sin(i * 15);
+}
+return o13;
+}(); } catch(e) { Logger.error(Logger.comment(e)); }
+try { o12.channelCount = 1; } catch(e) { Logger.error(Logger.comment(e)); }
+try { o12.connect(GainNode, 65536, 0) } catch(e) { Logger.error(Logger.comment(e)); }
+/* TypeError: Value does not implement interface AudioNode. */
+try { o3.buffer = function() { o15 = o1.createBuffer(1, 1, 72540);
+o16 = o15.getChannelData(0);
+for(var i=0; i<1; ++i) {
+o16[i] = Math.sin(i * 7);
+}
+return o15;
+}(); } catch(e) { Logger.error(Logger.comment(e)); }
+try { o12.getFrequencyResponse(new Float32Array(7), new Float32Array(127), new Float32Array(7)) } catch(e) { Logger.error(Logger.comment(e)); }
+try { o12.getFrequencyResponse(new Float32Array(15), new Float32Array(127), new Float32Array(7)) } catch(e) { Logger.error(Logger.comment(e)); }
+try { o17 = document.createElement('audio'); } catch(e) { Logger.error(Logger.comment(e)); }
+try { (document.body || document.documentElement).appendChild(o0); } catch(e) { Logger.error(Logger.comment(e)); }
+try { o3.buffer = function() { o18 = o1.createBuffer(1, 7, 91261);
+o19 = o18.getChannelData(0);
+for(var i=0; i<7; ++i) {
+o19[i] = Math.sin(i * 7);
+}
+return o18;
+}(); } catch(e) { Logger.error(Logger.comment(e)); }
+try { o12.getFrequencyResponse(new Float32Array(31), new Float32Array(31), new Float32Array(127)) } catch(e) { Logger.error(Logger.comment(e)); }
+try { o20 = o1.createChannelSplitter(1, 2, 4, 16, 32); } catch(e) { Logger.error(Logger.comment(e)); }
+try { o12.channelCountMode = 'explicit'; } catch(e) { Logger.error(Logger.comment(e)); }
+try { o3.buffer = function() { o21 = o1.createBuffer(1, 0, 14451);
+o22 = o21.getChannelData(0);
+for(var i=0; i<0; ++i) {
+o22[i] = Math.sin(i * 63);
+}
+return o21;
+}(); } catch(e) { Logger.error(Logger.comment(e)); }
+</script>
diff --git a/dom/media/test/crashtests/875596.html b/dom/media/test/crashtests/875596.html
new file mode 100644
index 0000000000..7bac09dab7
--- /dev/null
+++ b/dom/media/test/crashtests/875596.html
@@ -0,0 +1,12 @@
+<script>
+try { o1 = new AudioContext(); } catch(e) { }
+try { o6 = o1.createGain(); } catch(e) { }
+try { o8 = o1.createBufferSource(); } catch(e) { }
+try { o6.gain.setValueCurveAtTime(new Float32Array(7), 0, 0) } catch(e) { }
+try { o8.connect(o6, 0, 0) } catch(e) { }
+try { o8.buffer = function() {
+ o19 = o1.createBuffer(1, 1, 76309);
+ for(var i=0; i<1; ++i) { }
+ return o19;
+}(); } catch(e) { }
+</script>
diff --git a/dom/media/test/crashtests/875911.html b/dom/media/test/crashtests/875911.html
new file mode 100644
index 0000000000..fbc52642b4
--- /dev/null
+++ b/dom/media/test/crashtests/875911.html
@@ -0,0 +1,3 @@
+<script>
+ new OfflineAudioContext(1, 10, 48000);
+</script>
diff --git a/dom/media/test/crashtests/876024-1.html b/dom/media/test/crashtests/876024-1.html
new file mode 100644
index 0000000000..5502d8e42d
--- /dev/null
+++ b/dom/media/test/crashtests/876024-1.html
@@ -0,0 +1,5 @@
+<script>
+o1 = new window.AudioContext(2, 8, 44100);
+o4 = o1.createBiquadFilter();
+o4.detune.setValueAtTime(0.0843, 1.7976931348623157e+308);
+</script>
diff --git a/dom/media/test/crashtests/876024-2.html b/dom/media/test/crashtests/876024-2.html
new file mode 100644
index 0000000000..4b9beb7453
--- /dev/null
+++ b/dom/media/test/crashtests/876024-2.html
@@ -0,0 +1,17 @@
+<!DOCTYPE html>
+<html>
+<head>
+<meta charset="UTF-8">
+<script>
+
+function boom()
+{
+ var bufferSource = new AudioContext().createScriptProcessor().context.createBufferSource();
+ bufferSource.start(0, 0, 0);
+ bufferSource.stop(562949953421313);
+}
+
+</script></head>
+
+<body onload="boom();"></body>
+</html>
diff --git a/dom/media/test/crashtests/876118.html b/dom/media/test/crashtests/876118.html
new file mode 100644
index 0000000000..bc0630350a
--- /dev/null
+++ b/dom/media/test/crashtests/876118.html
@@ -0,0 +1,16 @@
+<script>
+var Context0= new AudioContext()
+var BufferSource7=Context0.createBufferSource();
+
+BufferSource7.connect(Context0.destination);
+BufferSource7.buffer=function(){
+ var length=7;
+ var Buffer=Context0.createBuffer(1,length,Context0.sampleRate);
+ var bufferData= Buffer.getChannelData(0);for (var i = 0; i < length; ++i) { bufferData[i] = Math.sin(i*(-1))};
+ return Buffer;
+}();
+
+
+Context0.destination.channelCountMode="explicit";
+Context0.destination.channelCount=519910189000000;
+</script>
diff --git a/dom/media/test/crashtests/876207.html b/dom/media/test/crashtests/876207.html
new file mode 100644
index 0000000000..7bfcb096b2
--- /dev/null
+++ b/dom/media/test/crashtests/876207.html
@@ -0,0 +1,30 @@
+<script>
+try { o1 = new window.OfflineAudioContext(1, 10, 44100); } catch(e) { }
+try { o2 = o1.createChannelSplitter(1); } catch(e) { }
+try { o3 = o1.createAnalyser(); } catch(e) { }
+try { o4 = o1.createWaveShaper(); } catch(e) { }
+try { o5 = o1.createChannelSplitter(6); } catch(e) { }
+try { o6 = o1.createAnalyser(); } catch(e) { }
+try { o7 = o1.createBufferSource(); } catch(e) { }
+try { o7.connect(o1.destination); } catch(e) { }
+try { o7.buffer = function() {
+o8 = o1.createBuffer(1, 3, o1.sampleRate);
+o9 = o8.getChannelData(0);
+for(var i = 0; i < 3; ++i) {
+o9[i] = Math.sin(i * 127);
+}
+return o8;
+}()
+; } catch(e) { }
+try { o7.playbackRate.cancelScheduledValues(0) } catch(e) { }
+try { o7.channelInterpretation = 'speakers'; } catch(e) { }
+try { o1.destination.channelInterpretation = 'speakers'; } catch(e) { }
+try { o5.connect(o1.destination, 1, 1) } catch(e) { }
+try { o1.listener.setOrientation(0, 1073741824, 1073741824, 4194304, 1, 0) } catch(e) { }
+try { o4.curve = new Float32Array(127); } catch(e) { }
+try { o6.getByteFrequencyData(new Uint8Array(12)) } catch(e) { }
+try { o6.disconnect() } catch(e) { }
+try { o1.destination.channelCount = 33554432; } catch(e) { }
+try { o7.playbackRate.setTargetAtTime(0, 8388608, 1) } catch(e) { }
+try { o6.getByteTimeDomainData(new Uint8Array(12)) } catch(e) { }
+</script>
diff --git a/dom/media/test/crashtests/876215.html b/dom/media/test/crashtests/876215.html
new file mode 100644
index 0000000000..07135e3628
--- /dev/null
+++ b/dom/media/test/crashtests/876215.html
@@ -0,0 +1,14 @@
+<script>
+try { o1 = new window.OfflineAudioContext(1, 10, 44100); } catch(e) { }
+try { o6 = o1.createScriptProcessor(0, 0, 2); } catch(e) { }
+try { o8 = o1.createBufferSource(); } catch(e) { }
+try { o8.connect(o1.destination); } catch(e) { }
+try { o8.connect(o6) } catch(e) { }
+try { o3.disconnect() } catch(e) { }
+try { o8.buffer = function() {
+o21 = o1.createBuffer(1, 3, o1.sampleRate);
+o22 = o21.getChannelData(0);
+return o21;
+}()
+; } catch(e) { }
+</script>
diff --git a/dom/media/test/crashtests/876249.html b/dom/media/test/crashtests/876249.html
new file mode 100644
index 0000000000..6f72b40bc1
--- /dev/null
+++ b/dom/media/test/crashtests/876249.html
@@ -0,0 +1,27 @@
+<script>
+var Context0= new AudioContext()
+var BufferSource0=Context0.createBufferSource();
+var BiquadFilter0=Context0.createBiquadFilter();
+BufferSource0.start(0,0.6167310480959713,0.7142638498917222);
+BiquadFilter0.connect(Context0.destination);
+BufferSource0.buffer=function(){
+ var length=86333;
+ var Buffer=Context0.createBuffer(1,length,Context0.sampleRate);
+ var bufferData= Buffer.getChannelData(0);
+ for (var i = 0; i < length; ++i) { bufferData[i] = Math.sin(i*(-1))};
+ return Buffer;
+}();
+
+BufferSource0.connect(BiquadFilter0);
+
+BufferSource0.buffer=function(){
+ var length=21989;
+ var Buffer=Context0.createBuffer(1,length,Context0.sampleRate);
+ var bufferData= Buffer.getChannelData(0);
+ for (var i = 0; i < length; ++i) { bufferData[i] = Math.sin(i*(0))};
+ return Buffer;
+}();
+
+BufferSource0.stop(0.04184641852043569);
+
+</script>
diff --git a/dom/media/test/crashtests/876252.html b/dom/media/test/crashtests/876252.html
new file mode 100644
index 0000000000..cb9cb63ed3
--- /dev/null
+++ b/dom/media/test/crashtests/876252.html
@@ -0,0 +1,23 @@
+<script>
+var Context0= new AudioContext()
+var BufferSource4=Context0.createBufferSource();
+BufferSource4.start(0.05386466556228697,0.397192713804543,0.48810303467325866);
+
+BufferSource4.buffer=function(){
+ var length=109076;
+ var Buffer=Context0.createBuffer(1,length,Context0.sampleRate);
+ var bufferData= Buffer.getChannelData(0);
+ for (var i = 0; i < length; ++i) { bufferData[i] = Math.sin(i*(370))};
+ return Buffer;
+}();
+
+BufferSource4.buffer=function(){
+ var length=19339;
+ var Buffer=Context0.createBuffer(1,length,53362);
+ var bufferData= Buffer.getChannelData(0);
+ for (var i = 0; i < length; ++i) { bufferData[i] = Math.sin(i*(-0.16235407581552863))};
+ return Buffer;
+}();
+
+BufferSource4.stop(0.46482366253621876);
+</script>
diff --git a/dom/media/test/crashtests/876834.html b/dom/media/test/crashtests/876834.html
new file mode 100644
index 0000000000..f4ca6ee558
--- /dev/null
+++ b/dom/media/test/crashtests/876834.html
@@ -0,0 +1,4 @@
+<!DOCTYPE html>
+<script>
+new OfflineAudioContext(0, 0, 3229622);
+</script>
diff --git a/dom/media/test/crashtests/877527.html b/dom/media/test/crashtests/877527.html
new file mode 100644
index 0000000000..c639d501b7
--- /dev/null
+++ b/dom/media/test/crashtests/877527.html
@@ -0,0 +1,37 @@
+<script>
+try { o1 = new window.AudioContext(2, 5, 44100); } catch(e) { }
+try { o2 = o1.createChannelMerger(1); } catch(e) { }
+try { o3 = o1.createDelay(10); } catch(e) { }
+try { o4 = o1.createBuffer(2, 2048, 8000); } catch(e) { }
+try { o5 = o1.createPanner(); } catch(e) { }
+try { o6 = o1.createBufferSource(); } catch(e) { }
+try { o7 = (function() {
+var buf = o1.createBuffer(1, 50000, o1.sampleRate);
+for(var j=0; j<1; ++j) {
+for(var i=0; i<50000; ++i) { buf.getChannelData(j)[i] = Math.sin(i * (9.8));}
+}
+return buf;
+})(); } catch(e) { }
+try { o6.buffer = o7; } catch(e) { }
+try { o6.connect(o5); } catch(e) { }
+try { o5.connect(o1.destination); } catch(e) { }
+try { o1.listener.speedOfSound = 0.0000019073486328125; } catch(e) { }
+try { o6.loop = true; } catch(e) { }
+try { o8 = (function() {
+var buf = o1.createBuffer(2, 1000, o1.sampleRate);
+for(var j=0; j<2; ++j) {
+for(var i=0; i<1000; ++i) { buf.getChannelData(j)[i] = Math.sin(i * (1));}
+}
+return buf;
+})(); } catch(e) { }
+try { o6.buffer = o7; } catch(e) { }
+try { o6.connect(o5); } catch(e) { }
+try { o5.connect(o1.destination); } catch(e) { }
+try { o6.loopEnd = 1.4901161193847656e-8; } catch(e) { }
+try { o6.connect(o1.destination); } catch(e) { }
+try { o6.buffer = o8; } catch(e) { }
+try { o5.setPosition(0.36, o1.destination.context.destination.channelCountMode, o1.destination.context.destination.channelInterpretation) } catch(e) { }
+try { o2.channelCountMode = 'explicit'; } catch(e) { }
+try { o1.listener.speedOfSound = 4; } catch(e) { }
+try { o1.startRendering(); } catch(e) { }
+</script>
diff --git a/dom/media/test/crashtests/877820.html b/dom/media/test/crashtests/877820.html
new file mode 100644
index 0000000000..6d65c1e9d9
--- /dev/null
+++ b/dom/media/test/crashtests/877820.html
@@ -0,0 +1,4 @@
+<script>
+o1 = new window.OfflineAudioContext(2, 5, 0.39);
+window.location.reload();
+</script> \ No newline at end of file
diff --git a/dom/media/test/crashtests/878014.html b/dom/media/test/crashtests/878014.html
new file mode 100644
index 0000000000..4a0ca6dce2
--- /dev/null
+++ b/dom/media/test/crashtests/878014.html
@@ -0,0 +1,31 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<script>
+try { o2 = new window.AudioContext(1, 15, 44100); } catch(e) { }
+try { o5 = o2.createDelay(0.1); } catch(e) { }
+try { o6 = o2.createPanner(); } catch(e) { }
+try { o8 = o2.createBufferSource(); } catch(e) { }
+try { o9 = (function() {
+var buf = o2.createBuffer(1, 12134, o2.sampleRate);
+for(var j=0; j<1; ++j) {
+for(var i=0; i<12134; ++i) { buf.getChannelData(j)[i] = Math.sin(i * (-127));}
+}
+return buf;
+})(); } catch(e) { }
+try { o8.buffer = o9; } catch(e) { }
+try { o8.connect(o6); } catch(e) { }
+try { o6.connect(o5, 0, 0) } catch(e) { }
+try { o8.buffer = (function() {
+var buf = o2.createBuffer(1, 5409, o2.sampleRate);
+for(var j=0; j<1; ++j) {
+for(var i=0; i<5409; ++i) { buf.getChannelData(j)[i] = Math.sin(i * (-1));}
+}
+return buf;
+})(); } catch(e) { }
+setTimeout(function() { try { o5.delayTime.setValueAtTime(0.78, 121560862.56366833); } catch(e) { } setTimeout(done, 0); },128)
+try { o5.delayTime.value = 0.4283; } catch(e) { }
+
+function done() {
+ document.documentElement.removeAttribute("class");
+}
+</script>
diff --git a/dom/media/test/crashtests/878328.html b/dom/media/test/crashtests/878328.html
new file mode 100644
index 0000000000..ec7b39871b
--- /dev/null
+++ b/dom/media/test/crashtests/878328.html
@@ -0,0 +1,5 @@
+<script>
+o1 = new window.AudioContext(2, 7, 44100);
+o5 = o1.createDelay(1);
+o5.delayTime.setValueCurveAtTime(new Float32Array(3), 1.7976931348623157e+308, 0.64);
+</script> \ No newline at end of file
diff --git a/dom/media/test/crashtests/878407.html b/dom/media/test/crashtests/878407.html
new file mode 100644
index 0000000000..ae2918dc85
--- /dev/null
+++ b/dom/media/test/crashtests/878407.html
@@ -0,0 +1,11 @@
+<script>
+o1 = new window.AudioContext();
+o4 = o1.createDelay(0.4468);
+o6 = o1.createPanner();
+o7 = (function() {var buf = o1.createBuffer(1, 1000, o1.sampleRate);for(var j=0; j<1; ++j) {for(var i=0; i<1000; ++i) {buf.getChannelData(j)[i] = Math.sin(i * (-15));}}return buf;})();
+o8 = o1.createBufferSource();
+o8.buffer = o7;
+o8.connect(o6);
+o6.connect(o4)
+o4.delayTime.setValueAtTime(0.6019893103749466289898, 0.26)
+</script>
diff --git a/dom/media/test/crashtests/878478.html b/dom/media/test/crashtests/878478.html
new file mode 100644
index 0000000000..89a47ddb55
--- /dev/null
+++ b/dom/media/test/crashtests/878478.html
@@ -0,0 +1,30 @@
+<script>
+var Context0= new AudioContext()
+var BufferSource0=Context0.createBufferSource();
+
+BufferSource0.loop=true;
+
+BufferSource0.buffer=function(){
+ var channels=3;
+ var length=97252;
+ var Buffer=Context0.createBuffer(channels,length,Context0.sampleRate);
+ for(var y=0;y<channels;y++){
+ var bufferData= Buffer.getChannelData(y);
+ for (var i = 0; i < length; ++i) { bufferData[i] = 1;}
+ };
+ return Buffer;
+}();
+
+BufferSource0.playbackRate.cancelScheduledValues(0.4);
+
+BufferSource0.loopEnd=5e-324;
+
+BufferSource0.buffer=function(){
+ var length=26590;
+ var Buffer=Context0.createBuffer(1,length,Context0.sampleRate);
+ var bufferData= Buffer.getChannelData(0);
+ for (var i = 0; i < length; ++i) { bufferData[i] = 1};
+ return Buffer;
+}();
+
+</script>
diff --git a/dom/media/test/crashtests/880129.html b/dom/media/test/crashtests/880129.html
new file mode 100644
index 0000000000..775e9d80ba
--- /dev/null
+++ b/dom/media/test/crashtests/880129.html
@@ -0,0 +1,9 @@
+<script>
+try { o1 = new window.AudioContext(3, 2, 44100); } catch(e) { }
+try { o6 = o1.createBufferSource(); } catch(e) { }
+try { o15 = o1.createAnalyser(); } catch(e) { }
+try { o15.fftSize = 32; } catch(e) { }
+try { o6.connect(o15,0,0) } catch(e) { }
+try { o27 = o1.createPanner(); } catch(e) { }
+try { o6.buffer = function(){var buffer = o1.createBuffer(2, 1148, o1.sampleRate);for(var c=0; c<2; c++) {for(var i=0; i<1148; i++) {buffer.getChannelData(c)[i] = 0;}}return buffer;}() } catch(e) { }
+</script> \ No newline at end of file
diff --git a/dom/media/test/crashtests/880202.html b/dom/media/test/crashtests/880202.html
new file mode 100644
index 0000000000..dc0fade9db
--- /dev/null
+++ b/dom/media/test/crashtests/880202.html
@@ -0,0 +1,33 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<script>
+var Context0= new window.OfflineAudioContext(15,12119,44100)
+var BufferSource1=Context0.createBufferSource();
+var Gain0=Context0.createGain();
+var Panner0=Context0.createPanner();
+Context0.listener.setPosition(29,135,158);
+
+BufferSource1.connect(Gain0);
+
+BufferSource1.buffer=function(){
+ var channels=3;
+ var length=53325;
+ var Buffer=Context0.createBuffer(channels,length,Context0.sampleRate);
+ for(var y=0;y<channels;y++){
+ var bufferData= Buffer.getChannelData(y);
+ for (var i = 0; i < length; ++i) {
+ bufferData[i] = i*(270);
+ }
+ };
+ return Buffer;
+}();
+
+Gain0.connect(Panner0);
+Panner0.panningModel="equalpower";
+
+
+setTimeout(function(){
+document.documentElement.removeAttribute("class");
+},500)
+
+</script>
diff --git a/dom/media/test/crashtests/880342-1.html b/dom/media/test/crashtests/880342-1.html
new file mode 100644
index 0000000000..7d1aa0c3e3
--- /dev/null
+++ b/dom/media/test/crashtests/880342-1.html
@@ -0,0 +1,208 @@
+<script>
+try { o1 = new window.AudioContext(2, 10, 1019159); } catch(e) { }
+try { o2 = o1.createBufferSource(); } catch(e) { }
+try { o3 = o1.createConvolver(); } catch(e) { }
+try { o4 = o1.createChannelMerger(2); } catch(e) { }
+try { o5 = o1.createAnalyser(); } catch(e) { }
+try { o6 = o1.createPanner(); } catch(e) { }
+try { o7 = o1.createDynamicsCompressor(); } catch(e) { }
+try { o8 = o1.createWaveShaper(); } catch(e) { }
+try { o9 = o1.createChannelSplitter(6); } catch(e) { }
+try { o10 = o1.createScriptProcessor(8192, 2, 3); } catch(e) { }
+try { o1.listener.setPosition(-144115188075855870, 0, -5e-324) } catch(e) { }
+try { o3.buffer = function(){var buffer = o1.createBuffer(2, 741, o1.sampleRate);for(var c=0; c<2; c++) {for(var i=0; i<741; i++) {buffer.getChannelData(c)[i] = 1;}}return buffer;}(); } catch(e) { }
+try { o10.onaudioprocess = function(e) { /* onEvent */ }; } catch(e) { }
+try { o3.connect(o5,0,0) } catch(e) { }
+try { o9.channelCountMode = 'max'; } catch(e) { }
+try { o10.onaudioprocess = function(e) { /* onEvent */ }; } catch(e) { }
+try { o7.channelInterpretation = 'speakers'; } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(2, 1887, 63346);for(var c=0; c<2; c++) {for(var i=0; i<1887; i++) {buffer.getChannelData(c)[i] = 1;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o2.playbackRate.exponentialRampToValueAtTime(3.819464020334534, 32) } catch(e) { }
+try { o5.connect(o9,2,0) } catch(e) { }
+try { o9.channelCountMode = 'explicit'; } catch(e) { }
+try { o1.listener.speedOfSound = 72057594037927940; } catch(e) { }
+try { o1.destination.channelCountMode = 'explicit'; } catch(e) { }
+try { o7.threshold.value = 0.0646435404346891590021684237399313133209944; } catch(e) { }
+try { o7.connect(o10,0,0) } catch(e) { }
+try { o1.destination.channelInterpretation = 'speakers'; } catch(e) { }
+try { o3.connect(o4,0,0) } catch(e) { }
+try { o7.release.value = 23.030355486273447; } catch(e) { }
+try { o9.connect(o9,4,0) } catch(e) { }
+try { o6.channelCountMode = 'max'; } catch(e) { }
+try { o7.threshold.value = 0.6395867363641939418172910336579661816358566284179687500; } catch(e) { }
+try { o2.connect(o9,3,0) } catch(e) { }
+try { o10.onaudioprocess = function(e) { /* onEvent */ }; } catch(e) { }
+try { o7.connect(o10,0,0) } catch(e) { }
+try { o5.connect(o5,0,0) } catch(e) { }
+try { o2.playbackRate.value = 5e-324; } catch(e) { }
+try { o4.channelInterpretation = 'discrete'; } catch(e) { }
+try { o2.playbackRate.value = 1.2211628339508704; } catch(e) { }
+try { o2.channelCountMode = 'clamped-max'; } catch(e) { }
+try { o1.listener.dopplerFactor = 32; } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(22, 1811, 21177);for(var c=0; c<22; c++) {for(var i=0; i<1811; i++) {buffer.getChannelData(c)[i] = 1;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o5.channelCountMode = 'max'; } catch(e) { }
+try { o1.listener.speedOfSound = 1024; } catch(e) { }
+try { o2.playbackRate.value = 9.999999999999998e-91; } catch(e) { }
+try { o1.listener.speedOfSound = 16; } catch(e) { }
+try { o8.curve = new Float32Array(256); } catch(e) { }
+try { o1.listener.setVelocity(0, 128, 4) } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(23, 483, o1.sampleRate);for(var c=0; c<23; c++) {for(var i=0; i<483; i++) {buffer.getChannelData(c)[i] = 0.026;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(3, 945, 43803);for(var c=0; c<3; c++) {for(var i=0; i<945; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(19, 997, 23781);for(var c=0; c<19; c++) {for(var i=0; i<997; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o1.listener.speedOfSound = -8; } catch(e) { }
+try { o3.channelCountMode = 'explicit'; } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(3, 1740, o1.sampleRate);for(var c=0; c<3; c++) {for(var i=0; i<1740; i++) {buffer.getChannelData(c)[i] = 8;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(1, 1057, 30602);for(var c=0; c<1; c++) {for(var i=0; i<1057; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(3, 78, o1.sampleRate);for(var c=0; c<3; c++) {for(var i=0; i<78; i++) {buffer.getChannelData(c)[i] = 10;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o1.listener.setPosition(9.999999999999995e-275, 0.0001, 2) } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(1, 863, o1.sampleRate);for(var c=0; c<1; c++) {for(var i=0; i<863; i++) {buffer.getChannelData(c)[i] = 1;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o8.channelCount = 3; } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(2, 1821, o1.sampleRate);for(var c=0; c<2; c++) {for(var i=0; i<1821; i++) {buffer.getChannelData(c)[i] = 0.3655;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o6.connect(o6,0,0) } catch(e) { }
+try { o1.listener.setOrientation(128, -0.479527496, 128, 1, 1, -16) } catch(e) { }
+try { o4.connect(o8,0,0) } catch(e) { }
+try { o7.reduction.exponentialRampToValueAtTime(1.7976931348623157e+308, 2048) } catch(e) { }
+try { o1.listener.dopplerFactor = 0.5754; } catch(e) { }
+try { o7.release.value = 0; } catch(e) { }
+try { o5.getFloatFrequencyData(new Float32Array(2048)) } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(2, 120, 42419);for(var c=0; c<2; c++) {for(var i=0; i<120; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o3.buffer = function(){var buffer = o1.createBuffer(1, 1620, o1.sampleRate);for(var c=0; c<1; c++) {for(var i=0; i<1620; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}(); } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(1, 823, 77239);for(var c=0; c<1; c++) {for(var i=0; i<823; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o1.destination.channelInterpretation = 'speakers'; } catch(e) { }
+try { o1.listener.setPosition(2048, 0.000522899209, -4503599627370495) } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(3, 84028, 75483);for(var c=0; c<3; c++) {for(var i=0; i<84028; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o2.playbackRate.setTargetAtTime(5e-324, 9.999999999999998e-104, 0) } catch(e) { }
+try { o7.attack.value = 9.999999999999994e-236; } catch(e) { }
+try { o1.listener.dopplerFactor = 1024; } catch(e) { }
+try { o3.connect(o7,0,0) } catch(e) { }
+try { o2.playbackRate.linearRampToValueAtTime(0, 0) } catch(e) { }
+try { o4.connect(o10,0,0) } catch(e) { }
+try { o1.listener.setVelocity(0.682, 32, 1e-76) } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(3, 201, o1.sampleRate);for(var c=0; c<3; c++) {for(var i=0; i<201; i++) {buffer.getChannelData(c)[i] = 4;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o2.connect(o6,0,0) } catch(e) { }
+try { o5.getByteFrequencyData(new Uint8Array(32)) } catch(e) { }
+try { o3.connect(o6,0,0) } catch(e) { }
+try { o7.channelCount = 1; } catch(e) { }
+try { o2.playbackRate.setValueCurveAtTime(new Float32Array(512), 8, 1) } catch(e) { }
+try { o3.connect(o4,0,1) } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(3, 1996, o1.sampleRate);for(var c=0; c<3; c++) {for(var i=0; i<1996; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o1.listener.dopplerFactor = 2; } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(2, 983, 60517);for(var c=0; c<2; c++) {for(var i=0; i<983; i++) {buffer.getChannelData(c)[i] = 1;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o6.coneInnerAngle = 360; } catch(e) { }
+try { o8.curve = new Float32Array(512); } catch(e) { }
+try { o1.listener.setPosition(32, 16, 9.999999999999995e-280) } catch(e) { }
+try { o6.connect(o6,0,0) } catch(e) { }
+try { o1.listener.setPosition(2097151, 512, 5e-324) } catch(e) { }
+try { o7.channelCountMode = 'clamped-max'; } catch(e) { }
+try { o1.destination.channelCountMode = 'max'; } catch(e) { }
+try { o1.listener.setPosition(0, 1000000, 64) } catch(e) { }
+try { o4.connect(o5,0,0) } catch(e) { }
+try { o9.connect(o4,0,1) } catch(e) { }
+try { o7.attack.setValueCurveAtTime(new Float32Array(8), 256, 2048) } catch(e) { }
+try { o9.channelCountMode = 'explicit'; } catch(e) { }
+try { o1.listener.dopplerFactor = 5e-324; } catch(e) { }
+try { o4.connect(o10,0,0) } catch(e) { }
+try { o7.ratio.cancelScheduledValues(1) } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(1, 1590, 9733);for(var c=0; c<1; c++) {for(var i=0; i<1590; i++) {buffer.getChannelData(c)[i] = 2;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o1.destination.channelCount = 2; } catch(e) { }
+try { o1.destination.channelInterpretation = 'speakers'; } catch(e) { }
+try { o5.channelCountMode = 'max'; } catch(e) { }
+try { o9.channelCountMode = 'clamped-max'; } catch(e) { }
+try { o7.attack.setTargetAtTime(0.11499421161482907549622467513472656719386577606201171875000, 1, 1.7976931348623157e+308) } catch(e) { }
+try { o2.playbackRate.value = 1; } catch(e) { }
+try { o3.normalize = false; } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(2, 1866, o1.sampleRate);for(var c=0; c<2; c++) {for(var i=0; i<1866; i++) {buffer.getChannelData(c)[i] = 0.174908422905697580329587026426452212035655975341797;}}return buffer;}(); } catch(e) { }
+try { o2.playbackRate.value = 1.0; } catch(e) { }
+try { o7.threshold.value = 100; } catch(e) { }
+try { o1.listener.setVelocity(961.4441892370145, 9.999999999999999e-157, 1) } catch(e) { }
+try { o5.getByteFrequencyData(new Uint8Array(2)) } catch(e) { }
+try { o6.connect(o10,0,0) } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(3, 1176, o1.sampleRate);for(var c=0; c<3; c++) {for(var i=0; i<1176; i++) {buffer.getChannelData(c)[i] = 9.753993292300242;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o2.playbackRate.linearRampToValueAtTime(0.8687, 32) } catch(e) { }
+try { o5.channelCountMode = 'explicit'; } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(13, 703, 12723);for(var c=0; c<13; c++) {for(var i=0; i<703; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o2.loopEnd = 1.7976931348623157e+308; } catch(e) { }
+try { o3.connect(o7,0,0) } catch(e) { }
+try { o1.destination.channelCountMode = 'clamped-max'; } catch(e) { }
+try { o1.destination.channelInterpretation = 'discrete'; } catch(e) { }
+try { o2.connect(o10,0,0) } catch(e) { }
+try { o3.normalize = false; } catch(e) { }
+try { /* switch-case did not return anything. */ } catch(e) { }
+try { o10.connect(o8,0,0) } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(3, 132, 88507);for(var c=0; c<3; c++) {for(var i=0; i<132; i++) {buffer.getChannelData(c)[i] = 1.7976931348623157e+308;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o2.loopStart = 1.7976931348623157e+308; } catch(e) { }
+try { o1.listener.dopplerFactor = 1024; } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(1, 40, o1.sampleRate);for(var c=0; c<1; c++) {for(var i=0; i<40; i++) {buffer.getChannelData(c)[i] = 1.3785770335346594;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o4.connect(o9,5,0) } catch(e) { }
+try { o8.curve = new Float32Array(16); } catch(e) { }
+try { o10.onaudioprocess = function(e) { /* onEvent */ }; } catch(e) { }
+try { o6.channelCountMode = 'explicit'; } catch(e) { }
+try { o2.playbackRate.value = 0; } catch(e) { }
+try { o4.channelCount = 3; } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(3, 220, o1.sampleRate);for(var c=0; c<3; c++) {for(var i=0; i<220; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o1.destination.channelInterpretation = 'discrete'; } catch(e) { }
+try { o1.listener.setPosition(70368744177664, 2048, 1.7976931348623157e+308) } catch(e) { }
+try { o2.playbackRate.setValueCurveAtTime(new Float32Array(128), 1.7976931348623157e+308, 0.75) } catch(e) { }
+try { o5.fftSize = 118; } catch(e) { }
+try { o5.minDecibels = 1; } catch(e) { }
+try { o7.release.value = 5e-324; } catch(e) { }
+try { o2.channelCount = 3; } catch(e) { }
+try { o2.channelCountMode = 'clamped-max'; } catch(e) { }
+try { o3.normalize = false; } catch(e) { }
+try { o1.listener.speedOfSound = 2; } catch(e) { }
+try { o2.loopStart = 32; } catch(e) { }
+try { o3.channelCountMode = 'max'; } catch(e) { }
+try { o1.listener.setPosition(0.37976588317653304, -274877906943, 64) } catch(e) { }
+try { o8.curve = new Float32Array(4); } catch(e) { }
+try { o1.listener.setVelocity(16, -1024, 0) } catch(e) { }
+try { o5.minDecibels = 10000; } catch(e) { }
+try { o2.playbackRate.value = 5e-324; } catch(e) { }
+try { o3.connect(o9,1,0) } catch(e) { }
+try { o10.onaudioprocess = function(e) { /* onEvent */ }; } catch(e) { }
+try { o5.connect(o8,0,0) } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(3, 554, o1.sampleRate);for(var c=0; c<3; c++) {for(var i=0; i<554; i++) {buffer.getChannelData(c)[i] = 0.000001;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o7.release.value = 0; } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(1, 1289, 29583);for(var c=0; c<1; c++) {for(var i=0; i<1289; i++) {buffer.getChannelData(c)[i] = 5e-324;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o1.listener.setOrientation(1024, 64, 16, 1024, 1, 68719476735) } catch(e) { }
+try { o1.listener.setOrientation(512, 10000000, 274877906944, 9.999999999999998e-113, -8, 3.6191134923595274) } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(1, 1453, 15258);for(var c=0; c<1; c++) {for(var i=0; i<1453; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o8.channelInterpretation = 'discrete'; } catch(e) { }
+try { o5.channelInterpretation = 'discrete'; } catch(e) { }
+try { o7.connect(o9,4,0) } catch(e) { }
+try { o8.connect(o10,0,0) } catch(e) { }
+try { o1.listener.setPosition(18014398509481984, 16, 0.505129302503804056279079759406158700585365295410156250000) } catch(e) { }
+try { o10.connect(o9,5,0) } catch(e) { }
+try { o2.loop = false; } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(11, 1673, o1.sampleRate);for(var c=0; c<11; c++) {for(var i=0; i<1673; i++) {buffer.getChannelData(c)[i] = 0;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(2, 577, o1.sampleRate);for(var c=0; c<2; c++) {for(var i=0; i<577; i++) {buffer.getChannelData(c)[i] = 0.5611;}}return buffer;}() } catch(e) { }
+try { o2.connect(o1.destination); } catch(e) { }
+try { o1.listener.setOrientation(64, 2049, 5e-324, 0.1777, 2, 7) } catch(e) { }
+try { o5.maxDecibels = 128; } catch(e) { }
+try { o2.start(0) } catch(e) { }
+</script> \ No newline at end of file
diff --git a/dom/media/test/crashtests/880342-2.html b/dom/media/test/crashtests/880342-2.html
new file mode 100644
index 0000000000..a8ef88ae70
--- /dev/null
+++ b/dom/media/test/crashtests/880342-2.html
@@ -0,0 +1,8 @@
+<script>
+try { o1 = new window.AudioContext(2, 10, 1019159); } catch(e) { }
+try { o2 = o1.createBufferSource(); } catch(e) { }
+try { o3 = o1.createConvolver(); } catch(e) { }
+try { o3.buffer = function(){var buffer = o1.createBuffer(2, 741, o1.sampleRate);for(var c=0; c<2; c++) {for(var i=0; i<741; i++) {buffer.getChannelData(c)[i] = 1;}}return buffer;}(); } catch(e) { }
+try { o2.buffer = function(){var buffer = o1.createBuffer(2, 120, 42419);for(var c=0; c<2; c++) {for(var i=0; i<120; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}() } catch(e) { }
+try { o3.buffer = function(){var buffer = o1.createBuffer(1, 1620, o1.sampleRate);for(var c=0; c<1; c++) {for(var i=0; i<1620; i++) {buffer.getChannelData(c)[i] = -1;}}return buffer;}(); } catch(e) { }
+</script> \ No newline at end of file
diff --git a/dom/media/test/crashtests/880384.html b/dom/media/test/crashtests/880384.html
new file mode 100644
index 0000000000..798fbf133d
--- /dev/null
+++ b/dom/media/test/crashtests/880384.html
@@ -0,0 +1,8 @@
+<script>
+o1 = new window.AudioContext(3, 256, 44100);
+o2 = o1.createBufferSource();
+o3 = o1.createConvolver();
+o3.normalize = false;
+o3.buffer = function(){var buffer = o1.createBuffer(2, 1051, o1.sampleRate);for(var c=0; c<2; c++) {for(var i=0; i<1051; i++) {buffer.getChannelData(c)[i] = 1;}}return buffer;}();
+o3.buffer = function(){var buffer = o1.createBuffer(2, 1112, o1.sampleRate);for(var c=0; c<2; c++) {for(var i=0; i<1112; i++) {buffer.getChannelData(c)[i] = 1;}}return buffer;}();
+</script>
diff --git a/dom/media/test/crashtests/880404.html b/dom/media/test/crashtests/880404.html
new file mode 100644
index 0000000000..bf34932388
--- /dev/null
+++ b/dom/media/test/crashtests/880404.html
@@ -0,0 +1,6 @@
+<script>
+o1 = new window.OfflineAudioContext(2, 32, 44100);
+o12 = o1.createConvolver();
+o12.buffer = function(){var buffer = o1.createBuffer(1, 78, o1.sampleRate);for(var c=0; c<1; c++) {for(var i=0; i<78; i++) {buffer.getChannelData(c)[i] = 1;}}return buffer;}();
+o1.startRendering();
+</script> \ No newline at end of file
diff --git a/dom/media/test/crashtests/880724.html b/dom/media/test/crashtests/880724.html
new file mode 100644
index 0000000000..b57b5f9964
--- /dev/null
+++ b/dom/media/test/crashtests/880724.html
@@ -0,0 +1,13 @@
+<script>
+o1 = new window.AudioContext();
+o5 = o1.createConvolver();
+o5.buffer = function() {
+ var buffer = o1.createBuffer(2, 3, 33120);
+ for(var c=0; c<2; c++) {
+ for(var i=0; i<3; i++) {
+ buffer.getChannelData(c)[i] = -1;
+ }
+ }
+ return buffer;
+}();
+</script> \ No newline at end of file
diff --git a/dom/media/test/crashtests/881775.html b/dom/media/test/crashtests/881775.html
new file mode 100644
index 0000000000..d55c45d17e
--- /dev/null
+++ b/dom/media/test/crashtests/881775.html
@@ -0,0 +1,25 @@
+<script>
+o1 = new window.AudioContext(2, 16, 44100);
+o2 = o1.createBufferSource();
+o12 = o1.createBiquadFilter();
+o1.destination.channelCountMode = 'max';
+o2.buffer = function(){
+ var buffer = o1.createBuffer(2, 326, 77632);
+ for(var c=0; c<2; c++) {
+ for(var i=0; i<326; i++) {
+ buffer.getChannelData(c)[i] = -1;
+ }
+ }
+ return buffer;
+}();
+o2.connect(o1.destination);
+o2.buffer = function(){
+ var buffer = o1.createBuffer(3, 405, o1.sampleRate);
+ for(var c=0; c<3; c++) {
+ for(var i=0; i<405; i++) {
+ buffer.getChannelData(c)[i] = 1;
+ }
+ }
+ return buffer;
+}();
+</script>
diff --git a/dom/media/test/crashtests/882956.html b/dom/media/test/crashtests/882956.html
new file mode 100644
index 0000000000..ae7b441f99
--- /dev/null
+++ b/dom/media/test/crashtests/882956.html
@@ -0,0 +1,15 @@
+<script>
+o1 = new window.AudioContext(1, 2048, 44100);
+o2 = o1.createBufferSource();
+o1.destination.channelCountMode = 'max';
+o2.connect(o1.destination);
+o2.buffer = function(){
+ var buffer = o1.createBuffer(30, 442, 94933);
+ for(var c=0; c<30; c++) {
+ for(var i=0; i<442; i++) {
+ buffer.getChannelData(c)[i] = 1;
+ }
+ }
+ return buffer;
+}();
+</script>
diff --git a/dom/media/test/crashtests/884459.html b/dom/media/test/crashtests/884459.html
new file mode 100644
index 0000000000..e321d569f2
--- /dev/null
+++ b/dom/media/test/crashtests/884459.html
@@ -0,0 +1,12 @@
+<script>
+var Context0= new window.OfflineAudioContext(14,191531,44100)
+var BufferSource1=Context0.createBufferSource();
+
+setInterval(function(){
+BufferSource1.playbackRate.setTargetAtTime(0xC8F461D3EE6B2,(Context0.currentTime+0.0677539280615747),0.826130285160616);
+BufferSource1.playbackRate.setValueAtTime(35467.63924283907536607336193,0);
+},1)
+
+Context0.startRendering();
+
+</script>
diff --git a/dom/media/test/crashtests/889042.html b/dom/media/test/crashtests/889042.html
new file mode 100644
index 0000000000..9f74979c58
--- /dev/null
+++ b/dom/media/test/crashtests/889042.html
@@ -0,0 +1,4 @@
+<!DOCTYPE html>
+<script>
+ (new AudioContext()).createConvolver().buffer = null;
+</script>
diff --git a/dom/media/test/crashtests/907986-1.html b/dom/media/test/crashtests/907986-1.html
new file mode 100644
index 0000000000..4dec6bc480
--- /dev/null
+++ b/dom/media/test/crashtests/907986-1.html
@@ -0,0 +1,17 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<script>
+var context = new window.OfflineAudioContext(1, 100, 48000);
+context.oncomplete = function(e) {
+ document.documentElement.removeAttribute("class");
+};
+// zero front vector
+context.listener.setOrientation(0, 0, 0, 6.311749985202524e+307, 0, 0);
+var panner = context.createPanner();
+panner.setPosition(3.40282e+38, 4, 3.40282e+38);
+panner.connect(context.destination);
+var source = context.createOscillator();
+source.connect(panner);
+source.start(0);
+context.startRendering();
+</script>
diff --git a/dom/media/test/crashtests/907986-2.html b/dom/media/test/crashtests/907986-2.html
new file mode 100644
index 0000000000..e0626ba2c2
--- /dev/null
+++ b/dom/media/test/crashtests/907986-2.html
@@ -0,0 +1,17 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<script>
+var context = new window.OfflineAudioContext(1, 100, 48000);
+context.oncomplete = function(e) {
+ document.documentElement.removeAttribute("class");
+};
+// zero up vector
+context.listener.setOrientation(0, 6.311749985202524e+307, 0, 0, 0, 0);
+var panner = context.createPanner();
+panner.setPosition(1, 2, 3);
+panner.connect(context.destination);
+var source = context.createOscillator();
+source.connect(panner);
+source.start(0);
+context.startRendering();
+</script>
diff --git a/dom/media/test/crashtests/907986-3.html b/dom/media/test/crashtests/907986-3.html
new file mode 100644
index 0000000000..75b756c363
--- /dev/null
+++ b/dom/media/test/crashtests/907986-3.html
@@ -0,0 +1,17 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<script>
+var context = new window.OfflineAudioContext(1, 100, 48000);
+context.oncomplete = function(e) {
+ document.documentElement.removeAttribute("class");
+};
+// linearly dependent
+context.listener.setOrientation(0, 0, -6.311749985202524e+307, 0, 0, 6.311749985202524e+307);
+var panner = context.createPanner();
+panner.setPosition(1, 2, 3);
+panner.connect(context.destination);
+var source = context.createOscillator();
+source.connect(panner);
+source.start(0);
+context.startRendering();
+</script>
diff --git a/dom/media/test/crashtests/907986-4.html b/dom/media/test/crashtests/907986-4.html
new file mode 100644
index 0000000000..a73500efca
--- /dev/null
+++ b/dom/media/test/crashtests/907986-4.html
@@ -0,0 +1,15 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<script>
+var context = new window.OfflineAudioContext(1, 100, 48000);
+context.oncomplete = function(e) {
+ document.documentElement.removeAttribute("class");
+};
+var panner = context.createPanner();
+panner.setPosition(0, 3, 0); // directly overhead
+panner.connect(context.destination);
+var source = context.createOscillator();
+source.connect(panner);
+source.start(0);
+context.startRendering();
+</script>
diff --git a/dom/media/test/crashtests/910171-1.html b/dom/media/test/crashtests/910171-1.html
new file mode 100644
index 0000000000..9f3ec831be
--- /dev/null
+++ b/dom/media/test/crashtests/910171-1.html
@@ -0,0 +1,17 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<script>
+var context = new window.OfflineAudioContext(1, 4096, 48000);
+context.oncomplete = function(e) {
+ document.documentElement.removeAttribute("class");
+};
+var delay = context.createDelay();
+delay.connect(context.destination);
+delay.delayTime.value = 1.0;
+var buffer = context.createBuffer(1, 2048, context.sampleRate);
+var source = context.createBufferSource();
+source.buffer = buffer;
+source.connect(delay);
+source.start();
+context.startRendering();
+</script>
diff --git a/dom/media/test/crashtests/920987.html b/dom/media/test/crashtests/920987.html
new file mode 100644
index 0000000000..6e8b2992a6
--- /dev/null
+++ b/dom/media/test/crashtests/920987.html
@@ -0,0 +1,6 @@
+<script>
+var ctx = new AudioContext();
+var buffer = ctx.createBuffer(1, 1000, ctx.sampleRate);
+var array = new Float32Array(900);
+buffer.copyToChannel(array, 0, 0xfffffff8);
+</script>
diff --git a/dom/media/test/crashtests/925619-1.html b/dom/media/test/crashtests/925619-1.html
new file mode 100644
index 0000000000..146c531f9b
--- /dev/null
+++ b/dom/media/test/crashtests/925619-1.html
@@ -0,0 +1,14 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<script>
+// 1024 > 89478.5 * 48000 - (1 << 32)
+var context = new window.OfflineAudioContext(1, 1024, 48000);
+context.oncomplete = function(e) {
+ document.documentElement.removeAttribute("class");
+};
+var buffer = context.createBuffer(1, 2048, context.sampleRate);
+var source = context.createBufferSource();
+source.buffer = buffer;
+source.start(89478.5); // 89478.5 is a little greater than 2^32 / 48000.
+context.startRendering();
+</script>
diff --git a/dom/media/test/crashtests/925619-2.html b/dom/media/test/crashtests/925619-2.html
new file mode 100644
index 0000000000..e734b8bcad
--- /dev/null
+++ b/dom/media/test/crashtests/925619-2.html
@@ -0,0 +1,15 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<script>
+var context = new window.OfflineAudioContext(1, 2048, 48000);
+// 1024 > 89478.5 * 48000 - (1 << 32)
+var buffer = context.createBuffer(1, 1024, context.sampleRate);
+var source = context.createBufferSource();
+source.buffer = buffer;
+source.onended = function(e) {
+ document.documentElement.removeAttribute("class");
+};
+source.start(0);
+source.stop(89478.5); // 89478.5 is a little greater than 2^32 / 48000.
+context.startRendering();
+</script>
diff --git a/dom/media/test/crashtests/926619.html b/dom/media/test/crashtests/926619.html
new file mode 100644
index 0000000000..2ead02af4e
--- /dev/null
+++ b/dom/media/test/crashtests/926619.html
@@ -0,0 +1,24 @@
+<!DOCTYPE html>
+<html>
+<head>
+<meta charset="UTF-8">
+<script>
+
+function boom()
+{
+ var ac = new window.AudioContext();
+
+ var _ChannelMergerNode = ac.createChannelMerger(4);
+
+ var _MediaStreamAudioDestinationNode = ac.createMediaStreamDestination();
+ var _MediaStream = _MediaStreamAudioDestinationNode.stream;
+ var _MediaStreamAudioSourceNode = ac.createMediaStreamSource(_MediaStream);
+
+ _ChannelMergerNode.connect(_MediaStreamAudioDestinationNode, 0, 0);
+ _MediaStreamAudioSourceNode.connect(_ChannelMergerNode, 0, 0);
+}
+
+</script>
+</head>
+<body onload="boom();"></body>
+</html>
diff --git a/dom/media/test/crashtests/933151.html b/dom/media/test/crashtests/933151.html
new file mode 100644
index 0000000000..3d45f7af38
--- /dev/null
+++ b/dom/media/test/crashtests/933151.html
@@ -0,0 +1,16 @@
+<!DOCTYPE html>
+<html>
+<head>
+<script>
+
+function boom()
+{
+ var ac = new window.AudioContext();
+ var buffer = ac.createBuffer(1, 24313, 47250);
+ buffer.copyFromChannel(buffer.getChannelData(0), '');
+}
+
+</script>
+</head>
+<body onload="boom();"></body>
+</html>
diff --git a/dom/media/test/crashtests/933156.html b/dom/media/test/crashtests/933156.html
new file mode 100644
index 0000000000..b89445a43d
--- /dev/null
+++ b/dom/media/test/crashtests/933156.html
@@ -0,0 +1,23 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<script>
+function boom()
+{
+ var ac = new window.AudioContext();
+
+ var panner = ac.createPanner();
+ panner.setPosition(8, 0.7643051305237005, 0.10292575673733972);
+
+ var oscillator = ac.createOscillator();
+ oscillator.connect(panner);
+ oscillator.start(0);
+
+ setTimeout(function() {
+ panner.panningModel = 'equalpower';
+ oscillator.stop(0);
+ document.documentElement.removeAttribute("class");
+ }, 0.5);
+}
+boom();
+</script>
+</html>
diff --git a/dom/media/test/crashtests/944851.html b/dom/media/test/crashtests/944851.html
new file mode 100644
index 0000000000..4f663accc4
--- /dev/null
+++ b/dom/media/test/crashtests/944851.html
@@ -0,0 +1,17 @@
+<!DOCTYPE html>
+<html>
+<head>
+<meta charset="UTF-8">
+<script>
+
+var ac = new AudioContext(1, 1354, 44100);
+var shaper = ac.createWaveShaper();
+var biquad = ac.createBiquadFilter();
+shaper.connect(biquad.frequency);
+biquad.getFrequencyResponse(new Float32Array(55785),
+ new Float32Array(62876),
+ new Float32Array(45111));
+
+</script>
+</head>
+</html>
diff --git a/dom/media/test/crashtests/952756.html b/dom/media/test/crashtests/952756.html
new file mode 100644
index 0000000000..ffced2e400
--- /dev/null
+++ b/dom/media/test/crashtests/952756.html
@@ -0,0 +1,19 @@
+<script>
+var Context0= new AudioContext()
+var BufferSource0=Context0.createBufferSource();
+BufferSource0.buffer=function(){
+ var length=35887;
+ var Buffer=Context0.createBuffer(1,length,Context0.sampleRate);
+ var bufferData= Buffer.getChannelData(0);
+ for (var i = 0; i < length; ++i) { bufferData[i] = Math.sin(i*(0.39765272522345185))};
+ return Buffer;
+}();
+BufferSource0.start(0.01932738965842873,0.33345631847623736,0.3893404237460345);
+BufferSource0.buffer=function(){
+ var length=15952;
+ var Buffer=Context0.createBuffer(1,length,Context0.sampleRate);
+ var bufferData= Buffer.getChannelData(0);
+ for (var i = 0; i < length; ++i) { bufferData[i] = Math.sin(i*(85))};
+ return Buffer;
+}();
+</script>
diff --git a/dom/media/test/crashtests/986901.html b/dom/media/test/crashtests/986901.html
new file mode 100644
index 0000000000..343df2c0ed
--- /dev/null
+++ b/dom/media/test/crashtests/986901.html
@@ -0,0 +1,16 @@
+<!DOCTYPE html>
+<html>
+<head>
+<meta charset="UTF-8">
+<script>
+ var ac = new window.AudioContext();
+ var delay1 = ac.createDelay(0.02);
+ var delay2 = ac.createDelay(0.002);
+ var source = ac.createOscillator();
+ source.start(0);
+ source.connect(delay1, 0, 0);
+ delay2.connect(delay1, 0, 0);
+ delay1.connect(delay2, 0, 0);
+</script>
+</head>
+</html>
diff --git a/dom/media/test/crashtests/990794.html b/dom/media/test/crashtests/990794.html
new file mode 100644
index 0000000000..8b40088b1a
--- /dev/null
+++ b/dom/media/test/crashtests/990794.html
@@ -0,0 +1,22 @@
+<!DOCTYPE html>
+<script>
+var ctx = new AudioContext();
+var source = ctx.createOscillator();
+source.start(0);
+
+function appendMerger(src) {
+ const inputCount = 18;
+
+ var merger = ctx.createChannelMerger(32);
+
+ for (var i = 0; i < inputCount; ++i) {
+ src.connect(merger, 0, i);
+ }
+
+ return merger;
+}
+
+for (var i = 0; i < 6; ++i) {
+ source = appendMerger(source);
+}
+</script>
diff --git a/dom/media/test/crashtests/995289.html b/dom/media/test/crashtests/995289.html
new file mode 100644
index 0000000000..c988f41fa8
--- /dev/null
+++ b/dom/media/test/crashtests/995289.html
@@ -0,0 +1,9 @@
+<!DOCTYPE html>
+<script>
+var r0=new AudioContext();
+var r5=r0.createOscillator();
+var r6=r0.createPeriodicWave(new Float32Array(1),new Float32Array(1));
+r5.frequency.value = 4294967295;
+r5.start(0);
+r5.setPeriodicWave(r6);
+</script>
diff --git a/dom/media/test/crashtests/adts-truncated.aac b/dom/media/test/crashtests/adts-truncated.aac
new file mode 100644
index 0000000000..670f696e1d
--- /dev/null
+++ b/dom/media/test/crashtests/adts-truncated.aac
Binary files differ
diff --git a/dom/media/test/crashtests/adts.aac b/dom/media/test/crashtests/adts.aac
new file mode 100644
index 0000000000..208515464a
--- /dev/null
+++ b/dom/media/test/crashtests/adts.aac
Binary files differ
diff --git a/dom/media/test/crashtests/analyser-channels-1.html b/dom/media/test/crashtests/analyser-channels-1.html
new file mode 100644
index 0000000000..2f3133cf13
--- /dev/null
+++ b/dom/media/test/crashtests/analyser-channels-1.html
@@ -0,0 +1,16 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<script>
+ var context = new window.OfflineAudioContext(1, 256, 48000);
+ var analyser = context.createAnalyser();
+ analyser.channelCount = 2;
+ analyser.channelCountMode = "explicit";
+ analyser.fftSize = 32;
+ var source = context.createOscillator();
+ source.connect(analyser);
+ source.start(0);
+ context.startRendering().
+ then(function() {
+ document.documentElement.removeAttribute("class");
+ });
+</script>
diff --git a/dom/media/test/crashtests/audiocontext-after-unload-1.html b/dom/media/test/crashtests/audiocontext-after-unload-1.html
new file mode 100644
index 0000000000..9b4f1181d2
--- /dev/null
+++ b/dom/media/test/crashtests/audiocontext-after-unload-1.html
@@ -0,0 +1,27 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<head>
+<title>Test for bug 1646601</title>
+<script>
+document.addEventListener('DOMContentLoaded', async () => {
+ const frame = document.createElement('iframe');
+ document.body.appendChild(frame);
+ frame.srcdoc = '<html></html>';
+ await new Promise(resolve => frame.onload = resolve);
+ const subwin = frame.contentWindow;
+ const subcontext = subwin.AudioContext;
+ // Construct an AudioContext while the subdocument is fully active to start
+ // a MediaTrackGraph.
+ new subcontext();
+ // Unload the subdocument and wait for completion.
+ // This shuts down the MediaTrackGraph.
+ subwin.location.reload();
+ await new Promise(resolve => frame.onload = resolve);
+ // Test that a new AudioContext on the inactive subdocument does not attempt
+ // to use the shut-down MediaTrackGraph.
+ try { new subcontext() } catch {}
+ document.documentElement.removeAttribute('class');
+});
+</script>
+</head>
+</html>
diff --git a/dom/media/test/crashtests/audiocontext-after-xhr.html b/dom/media/test/crashtests/audiocontext-after-xhr.html
new file mode 100644
index 0000000000..a4dd9990ca
--- /dev/null
+++ b/dom/media/test/crashtests/audiocontext-after-xhr.html
@@ -0,0 +1,13 @@
+<!DOCTYPE html>
+<html>
+<head>
+ <script>
+ document.addEventListener('DOMContentLoaded', () => {
+ const xhr = new XMLHttpRequest()
+ xhr.open('G', '', false)
+ xhr.send()
+ window.ac = new AudioContext()
+ })
+ </script>
+</head>
+</html>
diff --git a/dom/media/test/crashtests/audiocontext-double-suspend.html b/dom/media/test/crashtests/audiocontext-double-suspend.html
new file mode 100644
index 0000000000..98399549bd
--- /dev/null
+++ b/dom/media/test/crashtests/audiocontext-double-suspend.html
@@ -0,0 +1,5 @@
+<script>
+var ac = new AudioContext();
+ac.resume();
+ac.resume();
+</script>
diff --git a/dom/media/test/crashtests/audioworkletnode-after-unload-1.html b/dom/media/test/crashtests/audioworkletnode-after-unload-1.html
new file mode 100644
index 0000000000..7da8d1161a
--- /dev/null
+++ b/dom/media/test/crashtests/audioworkletnode-after-unload-1.html
@@ -0,0 +1,27 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<head>
+<title>Test for bug 1634200 and bug 1655544</title>
+<script>
+document.addEventListener('DOMContentLoaded', async () => {
+ const frame = document.createElement('iframe');
+ document.body.appendChild(frame);
+ frame.srcdoc = '<html></html>';
+ await new Promise(resolve => frame.onload = resolve);
+
+ const subwin = frame.contentWindow;
+ const ctx = new subwin.AudioContext();
+ const url = URL.createObjectURL(
+ new Blob([`registerProcessor("noop",
+ class extends AudioWorkletProcessor {})`]),
+ {type: "application/javascript"});
+ await ctx.audioWorklet.addModule(url);
+
+ frame.remove();
+ new subwin.AudioWorkletNode(ctx, 'noop')
+
+ document.documentElement.removeAttribute('class');
+});
+</script>
+</head>
+</html>
diff --git a/dom/media/test/crashtests/buffer-source-duration-1.html b/dom/media/test/crashtests/buffer-source-duration-1.html
new file mode 100644
index 0000000000..df8d7a37d5
--- /dev/null
+++ b/dom/media/test/crashtests/buffer-source-duration-1.html
@@ -0,0 +1,14 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<script>
+const rate = 44100;
+var context = new window.OfflineAudioContext(1, 512, rate);
+var buffer = context.createBuffer(1, 128, rate);
+var source = context.createBufferSource();
+source.buffer = buffer;
+source.start(0, 0, 86400);
+context.startRendering().
+ then(function() {
+ document.documentElement.removeAttribute("class");
+ });
+</script>
diff --git a/dom/media/test/crashtests/buffer-source-ended-1.html b/dom/media/test/crashtests/buffer-source-ended-1.html
new file mode 100644
index 0000000000..de8546316c
--- /dev/null
+++ b/dom/media/test/crashtests/buffer-source-ended-1.html
@@ -0,0 +1,16 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<script>
+var context = new AudioContext();
+
+var source = context.createBufferSource();
+source.buffer = context.createBuffer(1, 2.0 * context.sampleRate, context.sampleRate);
+source.onended = function(e) {
+ document.documentElement.removeAttribute("class");
+}
+source.start(0.0, 1.0);
+setTimeout(
+ function() {
+ source.buffer = context.createBuffer(1, 1, context.sampleRate);
+ }, 0);
+</script>
diff --git a/dom/media/test/crashtests/buffer-source-resampling-start-1.html b/dom/media/test/crashtests/buffer-source-resampling-start-1.html
new file mode 100644
index 0000000000..55db8591ed
--- /dev/null
+++ b/dom/media/test/crashtests/buffer-source-resampling-start-1.html
@@ -0,0 +1,16 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<script>
+const rate = 44101; // not divisible by 2
+var context = new window.OfflineAudioContext(1, 512, rate);
+var buffer = context.createBuffer(1, 128, rate);
+buffer.getChannelData(0)[0] = 1.0;
+var source = context.createBufferSource();
+source.buffer = buffer;
+source.playbackRate.value = rate / (Math.pow(2, 30) * 1.0000001);
+source.start(512 / rate);
+context.startRendering().
+ then(function() {
+ document.documentElement.removeAttribute("class");
+ });
+</script>
diff --git a/dom/media/test/crashtests/buffer-source-slow-resampling-1.html b/dom/media/test/crashtests/buffer-source-slow-resampling-1.html
new file mode 100644
index 0000000000..5d8a50442b
--- /dev/null
+++ b/dom/media/test/crashtests/buffer-source-slow-resampling-1.html
@@ -0,0 +1,34 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<script>
+const blockSize = 128;
+// The sample rate is a prime number so that the resampler is not expected to
+// simplify in/out fractions.
+const rate = 44101;
+var context = new window.OfflineAudioContext(1, 3 * blockSize, rate);
+// Non-zero buffer, so it can't be optimized away.
+var buffer = context.createBuffer(1, 128, rate);
+buffer.getChannelData(0)[0] = 1.0;
+var source = context.createBufferSource();
+source.buffer = buffer;
+source.loop = true;
+// Initialize the resampler with a slow input rate.
+// With the current (Mar 2017) implementation, very slow rates give the
+// resampler a very large denominator.
+source.playbackRate.setValueAtTime(rate / 0x7fffffff, 0.0);
+// Change to a moderate input rate.
+// With the current implementation, skip_frac_num increases by den_rate for
+// each output sample and so one block before the change in playback rate is
+// enough for high skip_frac_num at the time of the change.
+const changeBlock = 1;
+const changeBlockSeconds = changeBlock * blockSize / rate;
+// With the current speex_resampler_set_rate_frac() implementation, the
+// moderate resampler denominator is still large enough to trigger overflow of
+// 32-bit unsigned integer arithmetic.
+source.playbackRate.setValueAtTime(rate / (rate + 1), changeBlockSeconds);
+source.start(0);
+context.startRendering().
+ then(function() {
+ document.documentElement.removeAttribute("class");
+ });
+</script>
diff --git a/dom/media/test/crashtests/channel-count-in-metadata-different-than-in-content.mp4 b/dom/media/test/crashtests/channel-count-in-metadata-different-than-in-content.mp4
new file mode 100644
index 0000000000..92bf3722f2
--- /dev/null
+++ b/dom/media/test/crashtests/channel-count-in-metadata-different-than-in-content.mp4
Binary files differ
diff --git a/dom/media/test/crashtests/convolver-memory-report-1.html b/dom/media/test/crashtests/convolver-memory-report-1.html
new file mode 100644
index 0000000000..a49a281d1c
--- /dev/null
+++ b/dom/media/test/crashtests/convolver-memory-report-1.html
@@ -0,0 +1,25 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<head>
+ <title>Bug 1481745: Exercise ConvolverNode memory reporting</title>
+ <script type="text/javascript" src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script>
+let context = new AudioContext();
+let response = new AudioBuffer({length: 128,
+ sampleRate: context.sampleRate});
+response.getChannelData(0)[response.length - 1] = 1;
+let convolver = new ConvolverNode(context,
+ {disableNormalization: true,
+ buffer: response});
+convolver.connect(context.destination);
+let osc = new OscillatorNode(context);
+osc.connect(convolver);
+osc.start();
+osc.stop(128/context.sampleRate);
+osc.onended = (e) => {
+ SpecialPowers.getMemoryReports();
+ document.documentElement.removeAttribute("class");
+};
+ </script>
+</head>
+</html>
diff --git a/dom/media/test/crashtests/copyFromChannel-2.html b/dom/media/test/crashtests/copyFromChannel-2.html
new file mode 100644
index 0000000000..8d3d5a2124
--- /dev/null
+++ b/dom/media/test/crashtests/copyFromChannel-2.html
@@ -0,0 +1,16 @@
+<!DOCTYPE html>
+<html>
+<head>
+ <title>Crashtest for bug 1548816</title>
+ <script>
+let cx = new OfflineAudioContext({numberOfChannels: 1,
+ length: 1, sampleRate: 44100});
+let buffer = new AudioBuffer({numberOfChannels: 13,
+ length: 22050, sampleRate: 44100});
+buffer.getChannelData(12)[0] = 1.0;
+let o2248 = new AudioBufferSourceNode(cx, {buffer: buffer});
+let array = new Float32Array(52428);
+buffer.copyFromChannel(array, 12);
+ </script>
+</head>
+</html>
diff --git a/dom/media/test/crashtests/cors.webm b/dom/media/test/crashtests/cors.webm
new file mode 100644
index 0000000000..72b0297233
--- /dev/null
+++ b/dom/media/test/crashtests/cors.webm
Binary files differ
diff --git a/dom/media/test/crashtests/cors.webm^headers^ b/dom/media/test/crashtests/cors.webm^headers^
new file mode 100644
index 0000000000..cb762eff80
--- /dev/null
+++ b/dom/media/test/crashtests/cors.webm^headers^
@@ -0,0 +1 @@
+Access-Control-Allow-Origin: *
diff --git a/dom/media/test/crashtests/crashtests.list b/dom/media/test/crashtests/crashtests.list
new file mode 100644
index 0000000000..637b161e7f
--- /dev/null
+++ b/dom/media/test/crashtests/crashtests.list
@@ -0,0 +1,182 @@
+skip-if(OSX) load 1185191.html # this needs to run near the beginning of the test suite
+load 0-timescale.html # bug 1229166
+skip-if(Android) pref(media.autoplay.default,0) load 459439-1.html # bug 888557
+load 466607-1.html
+load 466945-1.html
+load 468763-1.html
+load 474744-1.html
+HTTP load 481136-1.html # needs to be HTTP to recognize the ogg as an audio file?
+load 492286-1.xhtml
+load 493915-1.html
+pref(media.autoplay.default,0) load 495794-1.html
+load 497734-1.xhtml
+load 497734-2.html
+load 576612-1.html
+load 752784-1.html
+skip-if(Android) load 789075-1.html # bug 1374405 for android
+skip-if(Android&&AndroidVersion=='22') HTTP load 795892-1.html # bug 1358718
+load 844563.html
+load 846612.html
+load 852838.html
+load 865004.html
+load 865537-1.html
+load 865550.html
+load 868504.html
+load 874869.html
+load 874915.html
+load 874934.html
+load 874952.html
+load 875144.html
+load 875596.html
+load 875911.html
+load 876024-1.html
+load 876024-2.html
+load 876118.html
+load 876207.html
+load 876215.html
+load 876249.html
+load 876252.html
+load 876834.html
+load 877527.html
+load 877820.html
+load 878014.html
+load 878328.html
+load 878407.html
+load 878478.html
+load 880129.html
+load 880202.html
+load 880342-1.html
+load 880342-2.html
+load 880384.html
+load 880404.html
+load 880724.html
+load 881775.html
+load 882956.html
+load 884459.html
+load 889042.html
+load 907986-1.html
+load 907986-2.html
+load 907986-3.html
+load 907986-4.html
+load 910171-1.html
+load 920987.html
+load 925619-1.html
+load 925619-2.html
+load 926619.html
+load 933151.html
+load 933156.html
+load 944851.html
+load 952756.html
+load 986901.html
+load 990794.html
+load 995289.html
+load 1012609.html
+load 1015662.html
+skip-if(Android) test-pref(media.navigator.permission.disabled,true) load 1028458.html # bug 1048863
+load 1041466.html
+load 1045650.html
+load 1080986.html
+skip-if(Android&&AndroidVersion=='21') load 1180881.html # bug 1409365
+load 1197935.html
+load 1122218.html
+load 1127188.html
+load 1157994.html
+load 1158427.html
+load 1185176.html
+load 1185192.html
+skip-if(Android) load 1257700.html # bug 1575666
+load 1267263.html
+load 1270303.html
+load 1368490.html
+load 1291702.html
+load 1378826.html
+load 1384248.html
+load 1389304.html
+load 1393272.webm
+load 1411322.html
+load 1450845.html
+load 1489160.html
+load disconnect-wrong-destination.html
+load analyser-channels-1.html
+load audiocontext-after-unload-1.html
+load audiocontext-after-xhr.html
+load audiocontext-double-suspend.html
+skip-if(Android) load audioworkletnode-after-unload-1.html # Needs secure context
+load buffer-source-duration-1.html
+load buffer-source-ended-1.html
+load buffer-source-resampling-start-1.html
+load buffer-source-slow-resampling-1.html
+load convolver-memory-report-1.html
+load copyFromChannel-2.html
+load empty-buffer-source.html
+HTTP load media-element-source-seek-1.html
+load offline-buffer-source-ended-1.html
+load oscillator-ended-1.html
+load oscillator-ended-2.html
+skip-if(Android&&AndroidVersion=='22') load video-replay-after-audio-end.html # bug 1315125, bug 1358876
+# This needs to run at the end to avoid leaking busted state into other tests.
+skip-if(Android||ThreadSanitizer) load 691096-1.html # Bug 1365451
+load 1236639.html
+test-pref(media.navigator.permission.disabled,true) test-pref(media.devices.insecure.enabled,true) test-pref(media.getusermedia.insecure.enabled,true) load 1388372.html
+load 1414444.mp4
+load 1494073.html
+skip-if(Android) load 1526044.html # Bug 1528391
+load 1530897.webm
+skip-if(Android&&AndroidVersion<21) load encrypted-track-with-bad-sample-description-index.mp4 # Bug 1533211, unkip after bug 1550912
+load encrypted-track-without-tenc.mp4 # Bug 1533215
+asserts-if(Android,0-1) load encrypted-track-with-sample-missing-cenc-aux.mp4 # Bug 1533625, bug 1588967
+load 1538727.html
+load empty-samples.webm # Bug 1540580
+test-pref(media.autoplay.default,0) load 1545133.html
+load track-with-zero-dimensions.mp4 # Bug 1542539
+load 1560215.html
+skip-if(Android) load 1547784.html # Skip on Android as clearkey is not supported
+load 1547899.html
+load 1569645.html
+load 1575271.html
+load 1577184.html
+pref(media.autoplay.default,0) load 1587248.html
+load 1594466.html
+load 1601385.html
+load 1601422.html
+load 1604941.html
+pref(media.autoplay.default,0) load 1673525.html
+skip-if(!winWidget) load 1608286.html
+load channel-count-in-metadata-different-than-in-content.mp4 # Bug 1584959
+load mp4_box_emptyrange.mp4 # Bug 1667480
+load 1673526-1.html
+load 1673526-2.html
+load 1693043.html
+load 1696511.html
+load 1697521.html
+load 1708790.html
+skip-if(cocoaWidget) load 1709130.html # video failed decoding on MacOS, 1709684
+skip-if(appleSilicon) load 1517199.html
+load 1734008.html
+load 1741677.html
+load 1748272.html
+load 1752917.html
+load 1762620.html
+load 1765842.html
+load adts.aac # Bug 1770073
+load 1787281.html
+load 1798778.html
+load 1830206.html
+load 1833896.mp4
+load 1833894.mp4
+load 1835164.html
+load 1840002.webm
+load 1845350.mp4
+load noextradata-8ch.html
+load invalidfmt.html
+load 1848660.html
+load 1835118.adts
+load trimming_needed_and_last_sample_invalid_duration.ogg
+load 1839193.html
+load 1850453.html
+test-pref(media.wmf.hevc.enabled,1) load 1859384.mp4
+test-pref(media.wmf.hevc.enabled,1) load 1859600.mp4
+test-pref(media.wmf.hevc.enabled,1) load 1860840.mp4
+load 1864450.html
+load 1872787.html
+load small-timebase.html
diff --git a/dom/media/test/crashtests/disconnect-wrong-destination.html b/dom/media/test/crashtests/disconnect-wrong-destination.html
new file mode 100644
index 0000000000..515ca3c877
--- /dev/null
+++ b/dom/media/test/crashtests/disconnect-wrong-destination.html
@@ -0,0 +1,13 @@
+<script>
+ var oc = new OfflineAudioContext(1, 1, 44100);
+ var splitter = oc.createChannelSplitter(2);
+ var merger0 = oc.createChannelMerger(2);
+ var merger1 = oc.createChannelMerger(2);
+ splitter.connect(merger0, 0);
+ splitter.connect(merger0, 1);
+ splitter.connect(merger1, 0);
+ splitter.connect(merger1, 1);
+
+ splitter.disconnect(merger0, 0);
+ splitter.disconnect(merger1, 1);
+</script>
diff --git a/dom/media/test/crashtests/doppler-1.html b/dom/media/test/crashtests/doppler-1.html
new file mode 100644
index 0000000000..2af3c8f460
--- /dev/null
+++ b/dom/media/test/crashtests/doppler-1.html
@@ -0,0 +1,23 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<script>
+var context = new window.AudioContext();
+var source = context.createBufferSource();
+source.buffer = context.createBuffer(1, 1, context.sampleRate);
+source.onended =
+ function(e) {
+ setTimeout(
+ function() {
+ var panner = context.createPanner();
+ source.connect(panner);
+ panner.setVelocity(1.0, 0.0, 0.0);
+ setTimeout(
+ function() {
+ document.documentElement.removeAttribute("class");
+ },
+ 0);
+ },
+ 0);
+ };
+source.start(0);
+</script>
diff --git a/dom/media/test/crashtests/empty-buffer-source.html b/dom/media/test/crashtests/empty-buffer-source.html
new file mode 100644
index 0000000000..2ce48a9ec1
--- /dev/null
+++ b/dom/media/test/crashtests/empty-buffer-source.html
@@ -0,0 +1,14 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<head>
+ <title>Bug 1636540: AudioBufferSourceNode with empty buffer</title>
+ <script>
+const offline = new OfflineAudioContext({length: 128, sampleRate: 16384});
+const buffer = new AudioBuffer({length: 1, sampleRate: 21725});
+const node = new AudioBufferSourceNode(offline, {buffer: buffer});
+node.start(5/offline.sampleRate);
+offline.startRendering().then(
+ () => document.documentElement.removeAttribute("class"));
+ </script>
+</head>
+</html>
diff --git a/dom/media/test/crashtests/empty-samples.webm b/dom/media/test/crashtests/empty-samples.webm
new file mode 100644
index 0000000000..e69de29bb2
--- /dev/null
+++ b/dom/media/test/crashtests/empty-samples.webm
diff --git a/dom/media/test/crashtests/encrypted-track-with-bad-sample-description-index.mp4 b/dom/media/test/crashtests/encrypted-track-with-bad-sample-description-index.mp4
new file mode 100644
index 0000000000..32303f0357
--- /dev/null
+++ b/dom/media/test/crashtests/encrypted-track-with-bad-sample-description-index.mp4
Binary files differ
diff --git a/dom/media/test/crashtests/encrypted-track-with-sample-missing-cenc-aux.mp4 b/dom/media/test/crashtests/encrypted-track-with-sample-missing-cenc-aux.mp4
new file mode 100644
index 0000000000..875c5dca76
--- /dev/null
+++ b/dom/media/test/crashtests/encrypted-track-with-sample-missing-cenc-aux.mp4
Binary files differ
diff --git a/dom/media/test/crashtests/encrypted-track-without-tenc.mp4 b/dom/media/test/crashtests/encrypted-track-without-tenc.mp4
new file mode 100644
index 0000000000..188faebf1b
--- /dev/null
+++ b/dom/media/test/crashtests/encrypted-track-without-tenc.mp4
Binary files differ
diff --git a/dom/media/test/crashtests/invalidfmt.html b/dom/media/test/crashtests/invalidfmt.html
new file mode 100644
index 0000000000..10c951dcf8
--- /dev/null
+++ b/dom/media/test/crashtests/invalidfmt.html
@@ -0,0 +1,8 @@
+<html class="reftest-wait">
+ <audio
+ onerror="document.documentElement.removeAttribute('class')"
+ autoplay
+ controls
+ src="fmttoosmall.wav"
+ ></audio>
+</html>
diff --git a/dom/media/test/crashtests/invalidfmt.wav b/dom/media/test/crashtests/invalidfmt.wav
new file mode 100644
index 0000000000..2939c836b9
--- /dev/null
+++ b/dom/media/test/crashtests/invalidfmt.wav
Binary files differ
diff --git a/dom/media/test/crashtests/media-element-source-seek-1.html b/dom/media/test/crashtests/media-element-source-seek-1.html
new file mode 100644
index 0000000000..5c3aed5ae7
--- /dev/null
+++ b/dom/media/test/crashtests/media-element-source-seek-1.html
@@ -0,0 +1,27 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<script>
+var audioElement = document.createElement("audio");
+audioElement.autoplay = true;
+audioElement.src = "sound.ogg";
+audioElement.onplaying =
+ function() {
+ audioElement.onplaying = null;
+ setTimeout(
+ function() {
+ audioElement.onseeked =
+ function() {
+ // Note we reset 'src' to release decoder resources and cubeb
+ // streams to prevent OOM or OpenCubeb() failures.
+ audioElement.src = "";
+ document.documentElement.removeAttribute("class");
+ };
+ audioElement.currentTime = 0;
+ }, 100);
+ };
+
+var context = new window.AudioContext();
+var source = context.createMediaElementSource(audioElement);
+source.connect(context.destination);
+</script>
+</html>
diff --git a/dom/media/test/crashtests/mp4_box_emptyrange.mp4 b/dom/media/test/crashtests/mp4_box_emptyrange.mp4
new file mode 100644
index 0000000000..83057533a0
--- /dev/null
+++ b/dom/media/test/crashtests/mp4_box_emptyrange.mp4
Binary files differ
diff --git a/dom/media/test/crashtests/noextradata-8ch.wav b/dom/media/test/crashtests/noextradata-8ch.wav
new file mode 100644
index 0000000000..40587a7de9
--- /dev/null
+++ b/dom/media/test/crashtests/noextradata-8ch.wav
Binary files differ
diff --git a/dom/media/test/crashtests/offline-buffer-source-ended-1.html b/dom/media/test/crashtests/offline-buffer-source-ended-1.html
new file mode 100644
index 0000000000..0631021126
--- /dev/null
+++ b/dom/media/test/crashtests/offline-buffer-source-ended-1.html
@@ -0,0 +1,15 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<script>
+var context = new window.OfflineAudioContext(1, 12001, 12000);
+
+var source = context.createBufferSource();
+source.buffer = context.createBuffer(1, 12000, context.sampleRate);
+source.onended = function(e) {
+ document.documentElement.removeAttribute("class");
+}
+source.connect(context.destination);
+source.start(0);
+
+context.startRendering();
+</script>
diff --git a/dom/media/test/crashtests/oscillator-ended-1.html b/dom/media/test/crashtests/oscillator-ended-1.html
new file mode 100644
index 0000000000..831111261c
--- /dev/null
+++ b/dom/media/test/crashtests/oscillator-ended-1.html
@@ -0,0 +1,15 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<script>
+function createContext() {
+ var context = new window.AudioContext();
+ var source = context.createOscillator();
+ source.onended = function(e) {
+ document.documentElement.removeAttribute("class");
+ };
+ source.connect(context.destination);
+ source.start(0.49);
+ source.stop(0.5);
+}
+createContext();
+</script>
diff --git a/dom/media/test/crashtests/oscillator-ended-2.html b/dom/media/test/crashtests/oscillator-ended-2.html
new file mode 100644
index 0000000000..ee9b8cf300
--- /dev/null
+++ b/dom/media/test/crashtests/oscillator-ended-2.html
@@ -0,0 +1,15 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<script>
+function createContext() {
+ var context = new window.AudioContext();
+ var source = context.createOscillator();
+ source.onended = function(e) {
+ document.documentElement.removeAttribute("class");
+ };
+ source.connect(context.destination);
+ source.start(60);
+ source.stop(0.5);
+}
+createContext();
+</script>
diff --git a/dom/media/test/crashtests/small-timebase.html b/dom/media/test/crashtests/small-timebase.html
new file mode 100644
index 0000000000..fffc518e4a
--- /dev/null
+++ b/dom/media/test/crashtests/small-timebase.html
@@ -0,0 +1,10 @@
+<!DOCTYPE html>
+<html class="reftest-wait">
+<video id=a src=small-timebase.mp4 controls autoplay></video>
+<script>
+a.onplaying = function() {
+ dump("OK");
+ document.documentElement.removeAttribute("class");
+}
+</script>
+</html>
diff --git a/dom/media/test/crashtests/small-timebase.mp4 b/dom/media/test/crashtests/small-timebase.mp4
new file mode 100644
index 0000000000..f2fba95cce
--- /dev/null
+++ b/dom/media/test/crashtests/small-timebase.mp4
Binary files differ
diff --git a/dom/media/test/crashtests/sound.ogg b/dom/media/test/crashtests/sound.ogg
new file mode 100644
index 0000000000..edda4e9128
--- /dev/null
+++ b/dom/media/test/crashtests/sound.ogg
Binary files differ
diff --git a/dom/media/test/crashtests/test.mp4 b/dom/media/test/crashtests/test.mp4
new file mode 100644
index 0000000000..44d7ee9cbc
--- /dev/null
+++ b/dom/media/test/crashtests/test.mp4
Binary files differ
diff --git a/dom/media/test/crashtests/track-with-zero-dimensions.mp4 b/dom/media/test/crashtests/track-with-zero-dimensions.mp4
new file mode 100644
index 0000000000..3f4a1317f3
--- /dev/null
+++ b/dom/media/test/crashtests/track-with-zero-dimensions.mp4
Binary files differ
diff --git a/dom/media/test/crashtests/trimming_needed_and_last_sample_invalid_duration.ogg b/dom/media/test/crashtests/trimming_needed_and_last_sample_invalid_duration.ogg
new file mode 100644
index 0000000000..593c489ec1
--- /dev/null
+++ b/dom/media/test/crashtests/trimming_needed_and_last_sample_invalid_duration.ogg
Binary files differ
diff --git a/dom/media/test/crashtests/video-crash.webm b/dom/media/test/crashtests/video-crash.webm
new file mode 100644
index 0000000000..9532113d87
--- /dev/null
+++ b/dom/media/test/crashtests/video-crash.webm
Binary files differ
diff --git a/dom/media/test/crashtests/video-replay-after-audio-end.html b/dom/media/test/crashtests/video-replay-after-audio-end.html
new file mode 100644
index 0000000000..9ffd6078de
--- /dev/null
+++ b/dom/media/test/crashtests/video-replay-after-audio-end.html
@@ -0,0 +1,43 @@
+<html class="reftest-wait">
+<head>
+ <title> Bug 1242774 : video crashed if pause and play again after audio track ends </title>
+</head>
+<body>
+<script type="text/javascript">
+function assert(value, msg) {
+ if (!value) {
+ dump("### Error : " + msg + "\n");
+ }
+}
+
+var AUDIO_END_TIME = 4.5;
+var video = document.createElement('video');
+video.src = "video-crash.webm";
+video.play();
+
+video.ontimeupdate = function () {
+ assert(AUDIO_END_TIME < video.duration,
+ "AUDIO_END_TIME should be smaller than the duration!");
+
+ if (video.currentTime > AUDIO_END_TIME) {
+ dump("### Pause video during silent part.\n");
+ video.ontimeupdate = null;
+ video.pause();
+ }
+
+ video.onpause = function () {
+ video.onpause = null;
+ setTimeout(function() {
+ dump("### Re-play after pausing during silent part.\n");
+ video.play();
+ video.onended = function () {
+ video.onended = null;
+ dump("### Video is ended.\n");
+ document.documentElement.removeAttribute("class");
+ }
+ }, 1000);
+ }
+}
+</script>
+</body>
+</html> \ No newline at end of file
diff --git a/dom/media/test/dash/dash-manifest-garbled-webm.mpd b/dom/media/test/dash/dash-manifest-garbled-webm.mpd
new file mode 100644
index 0000000000..aa78ded3ec
--- /dev/null
+++ b/dom/media/test/dash/dash-manifest-garbled-webm.mpd
@@ -0,0 +1,35 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<MPD
+ xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
+ xmlns="urn:mpeg:DASH:schema:MPD:2011"
+ xsi:schemaLocation="urn:mpeg:DASH:schema:MPD:2011"
+ type="static"
+ mediaPresentationDuration="PT3.958S"
+ minBufferTime="PT1S"
+ profiles="urn:webm:dash:profile:webm-on-demand:2012">
+ <BaseURL>./</BaseURL>
+ <Period id="0" start="PT0S" duration="PT3.958S" >
+ <AdaptationSet id="0" mimeType="video/webm" codecs="vp8" lang="eng" subsegmentAlignment="true" subsegmentStartsWithSAP="1" bitstreamSwitching="true">
+ <Representation id="0" bandwidth="54207" width="320" height="180">
+ <BaseURL>garbled.webm</BaseURL>
+ <SegmentBase indexRange="35090-35123">
+ <Initialization range="0-228" />
+ </SegmentBase>
+ </Representation>
+ <Representation id="1" bandwidth="78006" width="428" height="240">
+ <BaseURL>dash-webm-video-428x240.webm</BaseURL>
+ <SegmentBase indexRange="50173-50206">
+ <Initialization range="0-228" />
+ </SegmentBase>
+ </Representation>
+ </AdaptationSet>
+ <AdaptationSet id="1" mimeType="audio/webm" codecs="vorbis" lang="eng" audioSamplingRate="48000" subsegmentStartsWithSAP="1">
+ <Representation id="2" bandwidth="57264">
+ <BaseURL>dash-webm-audio-128k.webm</BaseURL>
+ <SegmentBase indexRange="41927-41946">
+ <Initialization range="0-4521" />
+ </SegmentBase>
+ </Representation>
+ </AdaptationSet>
+ </Period>
+</MPD>
diff --git a/dom/media/test/dash/dash-manifest-garbled.mpd b/dom/media/test/dash/dash-manifest-garbled.mpd
new file mode 100644
index 0000000000..ac8eadbddc
--- /dev/null
+++ b/dom/media/test/dash/dash-manifest-garbled.mpd
@@ -0,0 +1 @@
+PD94bWwgdmVyc2lvbj0iMS4wIiA/PjxNUEQgbWVkaWFQcmVzZW50YXRpb25EdXJhdGlvbj0iUFQxOS41MVMiIG1pbkJ1ZmZlclRpbWU9IlBUMVMiIHByb2ZpbGVzPSJ1cm46d2VibTpkYXNoOnByb2ZpbGU6d2VibS1vbi1kZW1hbmQ6MjAxMiIgdHlwZT0ic3RhdGljIiB4bWxucz0idXJuOm1wZWc6REFTSDpzY2hlbWE6TVBEOjIwMTEiIHhtbG5zOnhzaT0iaHR0cDovL3d3dy53My5vcmcvMjAwMS9YTUxTY2hlbWEtaW5zdGFuY2UiIHhzaTpzY2hlbWFMb2NhdGlvbj0idXJuOm1wZWc6REFTSDpzY2hlbWE6TVBEOjIwMTEiPjxCYXNlVVJMPmh0dHA6Ly93d3cuZ29vZ2xlLmNvbTwvQmFzZVVSTD48UGVyaW9kIGR1cmF0aW9uPSJQVDE5LjUxUyIgaWQ9IjAiIHN0YXJ0PSJQVDBTIj48QWRhcHRhdGlvblNldCBhdWRpb1NhbXBsaW5nUmF0ZT0iNDgwMDAiIGNvZGVjcz0idm9yYmlzIiBpZD0iMSIgbGFuZz0iZW5nIiBtaW1lVHlwZT0iYXVkaW8vd2VibSIgc3Vic2VnbWVudFN0YXJ0c1dpdGhTQVA9IjEiPjxSZXByZXNlbnRhdGlvbiBiYW5kd2lkdGg9IjIwMTA5IiBpZD0iMiI+PEJhc2VVUkwvPjxTZWdtZW50QmFzZSBpbmRleFJhbmdlPSIzMTk3ODAtMzIwNjEyIj48SW5pdGlhbGl6YXRpb24gcmFuZ2U9IjAtMjA4NzAiLz48L1NlZ21lbnRCYXNlPjwvUmVwcmVzZW50YXRpb24+PC9BZGFwdGF0aW9uU2V0PjwvUGVyaW9kPjwvTVBEPg
diff --git a/dom/media/test/dash/dash-manifest-sjs.mpd b/dom/media/test/dash/dash-manifest-sjs.mpd
new file mode 100644
index 0000000000..c7ecba3c69
--- /dev/null
+++ b/dom/media/test/dash/dash-manifest-sjs.mpd
@@ -0,0 +1,35 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<MPD
+ xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
+ xmlns="urn:mpeg:DASH:schema:MPD:2011"
+ xsi:schemaLocation="urn:mpeg:DASH:schema:MPD:2011"
+ type="static"
+ mediaPresentationDuration="PT3.958S"
+ minBufferTime="PT1S"
+ profiles="urn:webm:dash:profile:webm-on-demand:2012">
+ <BaseURL>./dash_detect_stream_switch.sjs?name=</BaseURL>
+ <Period id="0" start="PT0S" duration="PT3.958S" >
+ <AdaptationSet id="0" mimeType="video/webm" codecs="vp8" lang="eng" subsegmentAlignment="true" subsegmentStartsWithSAP="1" bitstreamSwitching="true">
+ <Representation id="0" bandwidth="54207" width="320" height="180">
+ <BaseURL>dash-webm-video-320x180.webm</BaseURL>
+ <SegmentBase indexRange="35090-35123">
+ <Initialization range="0-228" />
+ </SegmentBase>
+ </Representation>
+ <Representation id="1" bandwidth="78006" width="428" height="240">
+ <BaseURL>dash-webm-video-428x240.webm</BaseURL>
+ <SegmentBase indexRange="50173-50206">
+ <Initialization range="0-228" />
+ </SegmentBase>
+ </Representation>
+ </AdaptationSet>
+ <AdaptationSet id="1" mimeType="audio/webm" codecs="vorbis" lang="eng" audioSamplingRate="48000" subsegmentStartsWithSAP="1">
+ <Representation id="2" bandwidth="57264">
+ <BaseURL>dash-webm-audio-128k.webm</BaseURL>
+ <SegmentBase indexRange="41927-41946">
+ <Initialization range="0-4521" />
+ </SegmentBase>
+ </Representation>
+ </AdaptationSet>
+ </Period>
+</MPD>
diff --git a/dom/media/test/dash/dash-manifest.mpd b/dom/media/test/dash/dash-manifest.mpd
new file mode 100644
index 0000000000..98c7a90480
--- /dev/null
+++ b/dom/media/test/dash/dash-manifest.mpd
@@ -0,0 +1,35 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<MPD
+ xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
+ xmlns="urn:mpeg:DASH:schema:MPD:2011"
+ xsi:schemaLocation="urn:mpeg:DASH:schema:MPD:2011"
+ type="static"
+ mediaPresentationDuration="PT3.958S"
+ minBufferTime="PT1S"
+ profiles="urn:webm:dash:profile:webm-on-demand:2012">
+ <BaseURL>./</BaseURL>
+ <Period id="0" start="PT0S" duration="PT3.958S" >
+ <AdaptationSet id="0" mimeType="video/webm" codecs="vp8" lang="eng" subsegmentAlignment="true" subsegmentStartsWithSAP="1" bitstreamSwitching="true">
+ <Representation id="0" bandwidth="54207" width="320" height="180">
+ <BaseURL>dash-webm-video-320x180.webm</BaseURL>
+ <SegmentBase indexRange="35090-35123">
+ <Initialization range="0-228" />
+ </SegmentBase>
+ </Representation>
+ <Representation id="1" bandwidth="78006" width="428" height="240">
+ <BaseURL>dash-webm-video-428x240.webm</BaseURL>
+ <SegmentBase indexRange="50173-50206">
+ <Initialization range="0-228" />
+ </SegmentBase>
+ </Representation>
+ </AdaptationSet>
+ <AdaptationSet id="1" mimeType="audio/webm" codecs="vorbis" lang="eng" audioSamplingRate="48000" subsegmentStartsWithSAP="1">
+ <Representation id="2" bandwidth="57264">
+ <BaseURL>dash-webm-audio-128k.webm</BaseURL>
+ <SegmentBase indexRange="41927-41946">
+ <Initialization range="0-4521" />
+ </SegmentBase>
+ </Representation>
+ </AdaptationSet>
+ </Period>
+</MPD>
diff --git a/dom/media/test/dash/dash-webm-audio-128k.webm b/dom/media/test/dash/dash-webm-audio-128k.webm
new file mode 100644
index 0000000000..f56c042053
--- /dev/null
+++ b/dom/media/test/dash/dash-webm-audio-128k.webm
Binary files differ
diff --git a/dom/media/test/dash/dash-webm-video-320x180.webm b/dom/media/test/dash/dash-webm-video-320x180.webm
new file mode 100644
index 0000000000..282e6a2cc3
--- /dev/null
+++ b/dom/media/test/dash/dash-webm-video-320x180.webm
Binary files differ
diff --git a/dom/media/test/dash/dash-webm-video-428x240.webm b/dom/media/test/dash/dash-webm-video-428x240.webm
new file mode 100644
index 0000000000..23f2c89616
--- /dev/null
+++ b/dom/media/test/dash/dash-webm-video-428x240.webm
Binary files differ
diff --git a/dom/media/test/dash/garbled.webm b/dom/media/test/dash/garbled.webm
new file mode 100644
index 0000000000..ac8eadbddc
--- /dev/null
+++ b/dom/media/test/dash/garbled.webm
@@ -0,0 +1 @@
+PD94bWwgdmVyc2lvbj0iMS4wIiA/PjxNUEQgbWVkaWFQcmVzZW50YXRpb25EdXJhdGlvbj0iUFQxOS41MVMiIG1pbkJ1ZmZlclRpbWU9IlBUMVMiIHByb2ZpbGVzPSJ1cm46d2VibTpkYXNoOnByb2ZpbGU6d2VibS1vbi1kZW1hbmQ6MjAxMiIgdHlwZT0ic3RhdGljIiB4bWxucz0idXJuOm1wZWc6REFTSDpzY2hlbWE6TVBEOjIwMTEiIHhtbG5zOnhzaT0iaHR0cDovL3d3dy53My5vcmcvMjAwMS9YTUxTY2hlbWEtaW5zdGFuY2UiIHhzaTpzY2hlbWFMb2NhdGlvbj0idXJuOm1wZWc6REFTSDpzY2hlbWE6TVBEOjIwMTEiPjxCYXNlVVJMPmh0dHA6Ly93d3cuZ29vZ2xlLmNvbTwvQmFzZVVSTD48UGVyaW9kIGR1cmF0aW9uPSJQVDE5LjUxUyIgaWQ9IjAiIHN0YXJ0PSJQVDBTIj48QWRhcHRhdGlvblNldCBhdWRpb1NhbXBsaW5nUmF0ZT0iNDgwMDAiIGNvZGVjcz0idm9yYmlzIiBpZD0iMSIgbGFuZz0iZW5nIiBtaW1lVHlwZT0iYXVkaW8vd2VibSIgc3Vic2VnbWVudFN0YXJ0c1dpdGhTQVA9IjEiPjxSZXByZXNlbnRhdGlvbiBiYW5kd2lkdGg9IjIwMTA5IiBpZD0iMiI+PEJhc2VVUkwvPjxTZWdtZW50QmFzZSBpbmRleFJhbmdlPSIzMTk3ODAtMzIwNjEyIj48SW5pdGlhbGl6YXRpb24gcmFuZ2U9IjAtMjA4NzAiLz48L1NlZ21lbnRCYXNlPjwvUmVwcmVzZW50YXRpb24+PC9BZGFwdGF0aW9uU2V0PjwvUGVyaW9kPjwvTVBEPg
diff --git a/dom/media/test/dash_detect_stream_switch.sjs b/dom/media/test/dash_detect_stream_switch.sjs
new file mode 100644
index 0000000000..b8480a1274
--- /dev/null
+++ b/dom/media/test/dash_detect_stream_switch.sjs
@@ -0,0 +1,143 @@
+/* -*- Mode: JavaScript; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
+/* vim: set ts=8 sts=2 et sw=2 tw=80: */
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+/* dash_detect_stream_switch.sjs
+ *
+ * Parses requests for DASH manifests and ensures stream switching takes place
+ * by verifying the subsegments downloaded and the streams they belong to.
+ * If unexpected subsegments (byte ranges) are requested, the script will
+ * will respond with a 404.
+ */
+
+var DEBUG = false;
+
+function parseQuery(request, key) {
+ var params = request.queryString.split("&");
+ if (DEBUG) {
+ dump('DASH-SJS: request params = "' + params + '"\n');
+ }
+ for (var j = 0; j < params.length; ++j) {
+ var p = params[j];
+ if (p == key) {
+ return true;
+ }
+ if (p.indexOf(key + "=") === 0) {
+ return p.substring(key.length + 1);
+ }
+ if (!p.includes("=") && key === "") {
+ return p;
+ }
+ }
+ return false;
+}
+
+function handleRequest(request, response) {
+ try {
+ var name = parseQuery(request, "name");
+ var range = request.hasHeader("Range")
+ ? request.getHeader("Range")
+ : undefined;
+
+ // Should not get request for 1st subsegment from 2nd stream, nor 2nd
+ // subsegment from 1st stream.
+ if (
+ (name == "dash-webm-video-320x180.webm" &&
+ range == "bytes=25514-32767") ||
+ (name == "dash-webm-video-428x240.webm" && range == "bytes=228-35852")
+ ) {
+ throw new Error(
+ "Should not request " + name + " with byte-range " + range
+ );
+ } else {
+ var rangeSplit = range.split("=");
+ if (rangeSplit.length != 2) {
+ throw new Error(
+ "DASH-SJS: ERROR: invalid number of tokens (" +
+ rangeSplit.length +
+ ") delimited by '=' in 'Range' header."
+ );
+ }
+ var offsets = rangeSplit[1].split("-");
+ if (offsets.length != 2) {
+ throw new Error(
+ "DASH-SJS: ERROR: invalid number of tokens (" +
+ offsets.length +
+ ") delimited by '-' in 'Range' header."
+ );
+ }
+ var startOffset = parseInt(offsets[0]);
+ var endOffset = parseInt(offsets[1]);
+ var file = Services.dirsvc.get("CurWorkD", Ci.nsIFile);
+ var fis = Cc["@mozilla.org/network/file-input-stream;1"].createInstance(
+ Ci.nsIFileInputStream
+ );
+ var bis = Cc["@mozilla.org/binaryinputstream;1"].createInstance(
+ Ci.nsIBinaryInputStream
+ );
+
+ var paths = "tests/dom/media/test/" + name;
+ var split = paths.split("/");
+ for (var i = 0; i < split.length; ++i) {
+ file.append(split[i]);
+ }
+
+ fis.init(file, -1, -1, false);
+ // Exception: start offset should be within file bounds.
+ if (startOffset > file.fileSize) {
+ throw new Error(
+ "Starting offset [" +
+ startOffset +
+ "] is after end of file [" +
+ file.fileSize +
+ "]."
+ );
+ }
+ // End offset may be too large in the MPD. Real world HTTP servers just
+ // return what data they can; do the same here - reduce the end offset.
+ if (endOffset >= file.fileSize) {
+ if (DEBUG) {
+ dump(
+ "DASH-SJS: reducing endOffset [" +
+ endOffset +
+ "] to fileSize [" +
+ (file.fileSize - 1) +
+ "]\n"
+ );
+ }
+ endOffset = file.fileSize - 1;
+ }
+ fis.seek(Ci.nsISeekableStream.NS_SEEK_SET, startOffset);
+ bis.setInputStream(fis);
+
+ var byteLengthToRead = endOffset + 1 - startOffset;
+ var totalBytesExpected = byteLengthToRead + startOffset;
+ if (DEBUG) {
+ dump(
+ "DASH-SJS: byteLengthToRead = " +
+ byteLengthToRead +
+ " byteLengthToRead+startOffset = " +
+ totalBytesExpected +
+ " fileSize = " +
+ file.fileSize +
+ "\n"
+ );
+ }
+
+ var bytes = bis.readBytes(byteLengthToRead);
+ response.setStatusLine(request.httpVersion, 206, "Partial Content");
+ response.setHeader("Content-Length", "" + bytes.length, false);
+ response.setHeader("Content-Type", "application/dash+xml", false);
+ var contentRange =
+ "bytes " + startOffset + "-" + endOffset + "/" + file.fileSize;
+ response.setHeader("Content-Range", contentRange, false);
+ response.write(bytes, bytes.length);
+ bis.close();
+ }
+ } catch (e) {
+ dump("DASH-SJS-ERROR: " + e + "\n");
+ response.setStatusLine(request.httpVersion, 404, "Not found");
+ }
+}
diff --git a/dom/media/test/detodos-recorder-test.opus b/dom/media/test/detodos-recorder-test.opus
new file mode 100644
index 0000000000..88b2eab0f8
--- /dev/null
+++ b/dom/media/test/detodos-recorder-test.opus
Binary files differ
diff --git a/dom/media/test/detodos-recorder-test.opus^headers^ b/dom/media/test/detodos-recorder-test.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/detodos-recorder-test.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/detodos-short.opus b/dom/media/test/detodos-short.opus
new file mode 100644
index 0000000000..8bda283fc5
--- /dev/null
+++ b/dom/media/test/detodos-short.opus
Binary files differ
diff --git a/dom/media/test/detodos-short.opus^headers^ b/dom/media/test/detodos-short.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/detodos-short.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/detodos-short.webm b/dom/media/test/detodos-short.webm
new file mode 100644
index 0000000000..45af2675a6
--- /dev/null
+++ b/dom/media/test/detodos-short.webm
Binary files differ
diff --git a/dom/media/test/detodos-short.webm^headers^ b/dom/media/test/detodos-short.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/detodos-short.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/detodos.opus b/dom/media/test/detodos.opus
new file mode 100644
index 0000000000..6c7ba88a66
--- /dev/null
+++ b/dom/media/test/detodos.opus
Binary files differ
diff --git a/dom/media/test/detodos.opus^headers^ b/dom/media/test/detodos.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/detodos.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/detodos.webm b/dom/media/test/detodos.webm
new file mode 100644
index 0000000000..39cfa7f537
--- /dev/null
+++ b/dom/media/test/detodos.webm
Binary files differ
diff --git a/dom/media/test/detodos.webm^headers^ b/dom/media/test/detodos.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/detodos.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/dirac.ogg b/dom/media/test/dirac.ogg
new file mode 100644
index 0000000000..2986cf1e80
--- /dev/null
+++ b/dom/media/test/dirac.ogg
Binary files differ
diff --git a/dom/media/test/dirac.ogg^headers^ b/dom/media/test/dirac.ogg^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/dirac.ogg^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/dynamic_resource.sjs b/dom/media/test/dynamic_resource.sjs
new file mode 100644
index 0000000000..98c02e552d
--- /dev/null
+++ b/dom/media/test/dynamic_resource.sjs
@@ -0,0 +1,53 @@
+function parseQuery(request, key) {
+ var params = request.queryString.split("&");
+ for (var j = 0; j < params.length; ++j) {
+ var p = params[j];
+ if (p == key) {
+ return true;
+ }
+ if (p.indexOf(key + "=") == 0) {
+ return p.substring(key.length + 1);
+ }
+ if (!p.includes("=") && key == "") {
+ return p;
+ }
+ }
+ return false;
+}
+
+// Return resource1 file content for the first request with a given key.
+// All subsequent requests return resource2. Both must be video/ogg.
+function handleRequest(request, response) {
+ var key = parseQuery(request, "key");
+ var resource1 = parseQuery(request, "res1");
+ var resource2 = parseQuery(request, "res2");
+
+ var resource = getState(key) == "2" ? resource2 : resource1;
+ setState(key, "2");
+
+ var file = Services.dirsvc.get("CurWorkD", Ci.nsIFile);
+ var fis = Cc["@mozilla.org/network/file-input-stream;1"].createInstance(
+ Ci.nsIFileInputStream
+ );
+ var bis = Cc["@mozilla.org/binaryinputstream;1"].createInstance(
+ Ci.nsIBinaryInputStream
+ );
+ var paths = "tests/dom/media/test/" + resource;
+ var split = paths.split("/");
+ for (var i = 0; i < split.length; ++i) {
+ file.append(split[i]);
+ }
+ fis.init(file, -1, -1, false);
+ dump("file=" + file + "\n");
+ bis.setInputStream(fis);
+ var bytes = bis.readBytes(bis.available());
+ response.setStatusLine(request.httpVersion, 206, "Partial Content");
+ response.setHeader(
+ "Content-Range",
+ "bytes 0-" + (bytes.length - 1) + "/" + bytes.length
+ );
+ response.setHeader("Content-Length", "" + bytes.length, false);
+ response.setHeader("Content-Type", "video/ogg", false);
+ response.write(bytes, bytes.length);
+ bis.close();
+}
diff --git a/dom/media/test/eme.js b/dom/media/test/eme.js
new file mode 100644
index 0000000000..927c99876a
--- /dev/null
+++ b/dom/media/test/eme.js
@@ -0,0 +1,479 @@
+/* import-globals-from manifest.js */
+
+const CLEARKEY_KEYSYSTEM = "org.w3.clearkey";
+
+const gCencMediaKeySystemConfig = [
+ {
+ initDataTypes: ["cenc"],
+ videoCapabilities: [{ contentType: "video/mp4" }],
+ audioCapabilities: [{ contentType: "audio/mp4" }],
+ },
+];
+
+function bail(message) {
+ return function (err) {
+ if (err) {
+ message += "; " + String(err);
+ }
+ ok(false, message);
+ if (err) {
+ info(String(err));
+ }
+ SimpleTest.finish();
+ };
+}
+
+function ArrayBufferToString(arr) {
+ var str = "";
+ var view = new Uint8Array(arr);
+ for (var i = 0; i < view.length; i++) {
+ str += String.fromCharCode(view[i]);
+ }
+ return str;
+}
+
+function StringToArrayBuffer(str) {
+ var arr = new ArrayBuffer(str.length);
+ var view = new Uint8Array(arr);
+ for (var i = 0; i < str.length; i++) {
+ view[i] = str.charCodeAt(i);
+ }
+ return arr;
+}
+
+function StringToHex(str) {
+ var res = "";
+ for (var i = 0; i < str.length; ++i) {
+ res += ("0" + str.charCodeAt(i).toString(16)).slice(-2);
+ }
+ return res;
+}
+
+function Base64ToHex(str) {
+ var bin = window.atob(str.replace(/-/g, "+").replace(/_/g, "/"));
+ var res = "";
+ for (var i = 0; i < bin.length; i++) {
+ res += ("0" + bin.charCodeAt(i).toString(16)).substr(-2);
+ }
+ return res;
+}
+
+function HexToBase64(hex) {
+ var bin = "";
+ for (var i = 0; i < hex.length; i += 2) {
+ bin += String.fromCharCode(parseInt(hex.substr(i, 2), 16));
+ }
+ return window
+ .btoa(bin)
+ .replace(/=/g, "")
+ .replace(/\+/g, "-")
+ .replace(/\//g, "_");
+}
+
+function TimeRangesToString(trs) {
+ var l = trs.length;
+ if (l === 0) {
+ return "-";
+ }
+ var s = "";
+ var i = 0;
+ for (;;) {
+ s += trs.start(i) + "-" + trs.end(i);
+ if (++i === l) {
+ return s;
+ }
+ s += ",";
+ }
+}
+
+function SourceBufferToString(sb) {
+ return (
+ "SourceBuffer{" +
+ "AppendMode=" +
+ (sb.AppendMode || "-") +
+ ", updating=" +
+ (sb.updating ? "true" : "false") +
+ ", buffered=" +
+ TimeRangesToString(sb.buffered) +
+ ", audioTracks=" +
+ (sb.audioTracks ? sb.audioTracks.length : "-") +
+ ", videoTracks=" +
+ (sb.videoTracks ? sb.videoTracks.length : "-") +
+ "}"
+ );
+}
+
+function SourceBufferListToString(sbl) {
+ return "SourceBufferList[" + sbl.map(SourceBufferToString).join(", ") + "]";
+}
+
+function GenerateClearKeyLicense(licenseRequest, keyStore) {
+ var msgStr = ArrayBufferToString(licenseRequest);
+ var msg = JSON.parse(msgStr);
+
+ var keys = [];
+ for (var i = 0; i < msg.kids.length; i++) {
+ var id64 = msg.kids[i];
+ var idHex = Base64ToHex(msg.kids[i]).toLowerCase();
+ var key = keyStore[idHex];
+
+ if (key) {
+ keys.push({
+ kty: "oct",
+ kid: id64,
+ k: HexToBase64(key),
+ });
+ }
+ }
+
+ return new TextEncoder().encode(
+ JSON.stringify({
+ keys,
+ type: msg.type || "temporary",
+ })
+ );
+}
+
+function UpdateSessionFunc(test, token, sessionType, resolve, reject) {
+ return function (ev) {
+ var license = GenerateClearKeyLicense(ev.message, test.keys);
+ Log(
+ token,
+ "sending update message to CDM: " + new TextDecoder().decode(license)
+ );
+ ev.target
+ .update(license)
+ .then(function () {
+ Log(token, "MediaKeySession update ok!");
+ resolve(ev.target);
+ })
+ .catch(function (reason) {
+ reject(`${token} MediaKeySession update failed: ${reason}`);
+ });
+ };
+}
+
+function MaybeCrossOriginURI(test, uri) {
+ if (test.crossOrigin) {
+ return "https://example.com:443/tests/dom/media/test/allowed.sjs?" + uri;
+ }
+ return uri;
+}
+
+function AppendTrack(test, ms, track, token) {
+ return new Promise(function (resolve, reject) {
+ var sb;
+ var curFragment = 0;
+ var fragments = track.fragments;
+ var fragmentFile;
+
+ function addNextFragment() {
+ if (curFragment >= fragments.length) {
+ Log(token, track.name + ": end of track");
+ resolve();
+ return;
+ }
+
+ fragmentFile = MaybeCrossOriginURI(test, fragments[curFragment++]);
+
+ var req = new XMLHttpRequest();
+ req.open("GET", fragmentFile);
+ req.responseType = "arraybuffer";
+
+ req.addEventListener("load", function () {
+ Log(
+ token,
+ track.name + ": fetch of " + fragmentFile + " complete, appending"
+ );
+ sb.appendBuffer(new Uint8Array(req.response));
+ });
+
+ req.addEventListener("error", function () {
+ reject(`${token} - ${track.name}: error fetching ${fragmentFile}`);
+ });
+ req.addEventListener("abort", function () {
+ reject(`${token} - ${track.name}: aborted fetching ${fragmentFile}`);
+ });
+
+ Log(
+ token,
+ track.name +
+ ": addNextFragment() fetching next fragment " +
+ fragmentFile
+ );
+ req.send(null);
+ }
+
+ Log(token, track.name + ": addSourceBuffer(" + track.type + ")");
+ sb = ms.addSourceBuffer(track.type);
+ sb.addEventListener("updateend", function () {
+ Log(
+ token,
+ track.name +
+ ": updateend for " +
+ fragmentFile +
+ ", " +
+ SourceBufferToString(sb)
+ );
+ addNextFragment();
+ });
+
+ addNextFragment();
+ });
+}
+
+//Returns a promise that is resolved when the media element is ready to have
+//its play() function called; when it's loaded MSE fragments.
+function LoadTest(test, elem, token, endOfStream = true) {
+ if (!test.tracks) {
+ ok(false, token + " test does not have a tracks list");
+ return Promise.reject();
+ }
+
+ var ms = new MediaSource();
+ elem.src = URL.createObjectURL(ms);
+ elem.crossOrigin = test.crossOrigin || false;
+
+ return new Promise(function (resolve, reject) {
+ ms.addEventListener(
+ "sourceopen",
+ function () {
+ Log(token, "sourceopen");
+ Promise.all(
+ test.tracks.map(function (track) {
+ return AppendTrack(test, ms, track, token);
+ })
+ )
+ .then(function () {
+ Log(token, "Tracks loaded, calling MediaSource.endOfStream()");
+ if (endOfStream) {
+ ms.endOfStream();
+ }
+ resolve();
+ })
+ .catch(reject);
+ },
+ { once: true }
+ );
+ });
+}
+
+function EMEPromise() {
+ var self = this;
+ self.promise = new Promise(function (resolve, reject) {
+ self.resolve = resolve;
+ self.reject = reject;
+ });
+}
+
+/*
+ * Create a new MediaKeys object.
+ * Return a promise which will be resolved with a new MediaKeys object,
+ * or will be rejected with a string that describes the failure.
+ */
+function CreateMediaKeys(v, test, token) {
+ let p = new EMEPromise();
+
+ function streamType(type) {
+ var x = test.tracks.find(o => o.name == type);
+ return x ? x.type : undefined;
+ }
+
+ function onencrypted(ev) {
+ var options = { initDataTypes: [ev.initDataType] };
+ if (streamType("video")) {
+ options.videoCapabilities = [{ contentType: streamType("video") }];
+ }
+ if (streamType("audio")) {
+ options.audioCapabilities = [{ contentType: streamType("audio") }];
+ }
+ navigator.requestMediaKeySystemAccess(CLEARKEY_KEYSYSTEM, [options]).then(
+ keySystemAccess => {
+ keySystemAccess
+ .createMediaKeys()
+ .then(p.resolve, () =>
+ p.reject(`${token} Failed to create MediaKeys object.`)
+ );
+ },
+ () => p.reject(`${token} Failed to request key system access.`)
+ );
+ }
+
+ v.addEventListener("encrypted", onencrypted, { once: true });
+ return p.promise;
+}
+
+/*
+ * Create a new MediaKeys object and provide it to the media element.
+ * Return a promise which will be resolved if succeeded, or will be rejected
+ * with a string that describes the failure.
+ */
+function CreateAndSetMediaKeys(v, test, token) {
+ let p = new EMEPromise();
+
+ CreateMediaKeys(v, test, token).then(mediaKeys => {
+ v.setMediaKeys(mediaKeys).then(p.resolve, () =>
+ p.reject(`${token} Failed to set MediaKeys on <video> element.`)
+ );
+ }, p.reject);
+
+ return p.promise;
+}
+
+/*
+ * Collect the init data from 'encrypted' events.
+ * Return a promise which will be resolved with the init data when collection
+ * is completed (specified by test.sessionCount).
+ */
+function LoadInitData(v, test, token) {
+ let p = new EMEPromise();
+ let initDataQueue = [];
+
+ // Call SimpleTest._originalSetTimeout() to bypass the flaky timeout checker.
+ let timer = SimpleTest._originalSetTimeout.call(
+ window,
+ () => {
+ p.reject(`${token} Timed out in waiting for the init data.`);
+ },
+ 60000
+ );
+
+ function onencrypted(ev) {
+ initDataQueue.push(ev);
+ Log(
+ token,
+ `got encrypted(${ev.initDataType}, ` +
+ `${StringToHex(ArrayBufferToString(ev.initData))}) event.`
+ );
+ if (test.sessionCount == initDataQueue.length) {
+ p.resolve(initDataQueue);
+ clearTimeout(timer);
+ }
+ }
+
+ v.addEventListener("encrypted", onencrypted);
+ return p.promise;
+}
+
+/*
+ * Generate a license request and update the session.
+ * Return a promsise which will be resolved with the updated session
+ * or rejected with a string that describes the failure.
+ */
+function MakeRequest(test, token, ev, session, sessionType) {
+ sessionType = sessionType || "temporary";
+ let p = new EMEPromise();
+ let str =
+ `session[${session.sessionId}].generateRequest(` +
+ `${ev.initDataType}, ${StringToHex(ArrayBufferToString(ev.initData))})`;
+
+ session.addEventListener(
+ "message",
+ UpdateSessionFunc(test, token, sessionType, p.resolve, p.reject)
+ );
+
+ Log(token, str);
+ session.generateRequest(ev.initDataType, ev.initData).catch(reason => {
+ // Reject the promise if generateRequest() failed.
+ // Otherwise it will be resolved in UpdateSessionFunc().
+ p.reject(`${token}: ${str} failed; ${reason}`);
+ });
+
+ return p.promise;
+}
+
+/*
+ * Process the init data by calling MakeRequest().
+ * Return a promise which will be resolved with the updated sessions
+ * when all init data are processed or rejected if any failure.
+ */
+function ProcessInitData(v, test, token, initData, sessionType) {
+ return Promise.all(
+ initData.map(ev => {
+ let session = v.mediaKeys.createSession(sessionType);
+ return MakeRequest(test, token, ev, session, sessionType);
+ })
+ );
+}
+
+/*
+ * Clean up the |v| element.
+ */
+function CleanUpMedia(v) {
+ v.setMediaKeys(null);
+ v.remove();
+ v.removeAttribute("src");
+ v.load();
+}
+
+/*
+ * Close all sessions and clean up the |v| element.
+ */
+function CloseSessions(v, sessions) {
+ return Promise.all(sessions.map(s => s.close())).then(CleanUpMedia(v));
+}
+
+/*
+ * Set up media keys and source buffers for the media element.
+ * Return a promise resolved when all key sessions are updated or rejected
+ * if any failure.
+ */
+function SetupEME(v, test, token) {
+ let p = new EMEPromise();
+
+ v.onerror = function () {
+ p.reject(`${token} got an error event.`);
+ };
+
+ Promise.all([
+ LoadInitData(v, test, token),
+ CreateAndSetMediaKeys(v, test, token),
+ LoadTest(test, v, token),
+ ])
+ .then(values => {
+ let initData = values[0];
+ return ProcessInitData(v, test, token, initData);
+ })
+ .then(p.resolve, p.reject);
+
+ return p.promise;
+}
+
+function fetchWithXHR(uri, onLoadFunction) {
+ var p = new Promise(function (resolve, reject) {
+ var xhr = new XMLHttpRequest();
+ xhr.open("GET", uri, true);
+ xhr.responseType = "arraybuffer";
+ xhr.addEventListener("load", function () {
+ is(
+ xhr.status,
+ 200,
+ "fetchWithXHR load uri='" + uri + "' status=" + xhr.status
+ );
+ resolve(xhr.response);
+ });
+ xhr.send();
+ });
+
+ if (onLoadFunction) {
+ p.then(onLoadFunction);
+ }
+
+ return p;
+}
+
+function once(target, name, cb) {
+ var p = new Promise(function (resolve, reject) {
+ target.addEventListener(
+ name,
+ function (arg) {
+ resolve(arg);
+ },
+ { once: true }
+ );
+ });
+ if (cb) {
+ p.then(cb);
+ }
+ return p;
+}
diff --git a/dom/media/test/eme_standalone.js b/dom/media/test/eme_standalone.js
new file mode 100644
index 0000000000..202259b7fa
--- /dev/null
+++ b/dom/media/test/eme_standalone.js
@@ -0,0 +1,286 @@
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this file,
+ * You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+// This file offers standalone (no dependencies on other files) EME test
+// helpers. The intention is that this file can be used to provide helpers
+// while not coupling tests as tightly to `dom/media/test/manifest.js` or other
+// files. This allows these helpers to be used in different tests across the
+// codebase without imports becoming a mess.
+
+// A helper class to assist in setting up EME on media.
+//
+// Usage
+// 1. First configure the EME helper so it can have the information needed
+// to setup EME correctly. This is done by setting
+// - keySystem via `SetKeySystem`.
+// - initDataTypes via `SetInitDataTypes`.
+// - audioCapabilities and/or videoCapabilities via `SetAudioCapabilities`
+// and/or `SetVideoCapabilities`.
+// - keyIds and keys via `AddKeyIdAndKey`.
+// - onerror should be set to a function that will handle errors from the
+// helper. This function should take one argument, the error.
+// 2. Use the helper to configure a media element via `ConfigureEme`.
+// 3. One the promise from `ConfigureEme` has resolved the media element should
+// be configured and can be played. Errors that happen after this point are
+// reported via `onerror`.
+var EmeHelper = class EmeHelper {
+ // Members used to configure EME.
+ _keySystem;
+ _initDataTypes;
+ _audioCapabilities = [];
+ _videoCapabilities = [];
+
+ // Map of keyIds to keys.
+ _keyMap = new Map();
+
+ // Will be called if an error occurs during event handling. Users of the
+ // class should set a handler to be notified of errors.
+ onerror;
+
+ /**
+ * Get the clearkey key system string.
+ * @return The clearkey key system string.
+ */
+ static GetClearkeyKeySystemString() {
+ return "org.w3.clearkey";
+ }
+
+ // Begin conversion helpers.
+
+ /**
+ * Helper to convert Uint8Array into base64 using base64url alphabet, without
+ * padding.
+ * @param uint8Array An array of bytes to convert to base64.
+ * @return A base 64 encoded string
+ */
+ static Uint8ArrayToBase64(uint8Array) {
+ return new TextDecoder()
+ .decode(uint8Array)
+ .replace(/\+/g, "-") // Replace chars for base64url.
+ .replace(/\//g, "_")
+ .replace(/=*$/, ""); // Remove padding for base64url.
+ }
+
+ /**
+ * Helper to convert a hex string into base64 using base64url alphabet,
+ * without padding.
+ * @param hexString A string of hex characters.
+ * @return A base 64 encoded string
+ */
+ static HexToBase64(hexString) {
+ return btoa(
+ hexString
+ .match(/\w{2}/g) // Take chars two by two.
+ // Map to characters.
+ .map(hexByte => String.fromCharCode(parseInt(hexByte, 16)))
+ .join("")
+ )
+ .replace(/\+/g, "-") // Replace chars for base64url.
+ .replace(/\//g, "_")
+ .replace(/=*$/, ""); // Remove padding for base64url.
+ }
+
+ /**
+ * Helper to convert a base64 string (base64 or base64url) into a hex string.
+ * @param base64String A base64 encoded string. This can be base64url.
+ * @return A hex string (lower case);
+ */
+ static Base64ToHex(base64String) {
+ let binString = atob(base64String.replace(/-/g, "+").replace(/_/g, "/"));
+ let hexString = "";
+ for (let i = 0; i < binString.length; i++) {
+ // Covert to hex char. The "0" + and substr code are used to ensure we
+ // always get 2 chars, even for outputs the would normally be only one.
+ // E.g. for charcode 14 we'd get output 'e', and want to buffer that
+ // to '0e'.
+ hexString += ("0" + binString.charCodeAt(i).toString(16)).substr(-2);
+ }
+ // EMCA spec says that the num -> string conversion is lower case, so our
+ // hex string should already be lower case.
+ // https://tc39.es/ecma262/#sec-number.prototype.tostring
+ return hexString;
+ }
+
+ // End conversion helpers.
+
+ // Begin setters that setup the helper.
+ // These should be used to configure the helper prior to calling
+ // `ConfigureEme`.
+
+ /**
+ * Sets the key system that will be used by the EME helper.
+ * @param keySystem The key system to use. Probably "org.w3.clearkey", which
+ * can be fetched via `GetClearkeyKeySystemString`.
+ */
+ SetKeySystem(keySystem) {
+ this._keySystem = keySystem;
+ }
+
+ /**
+ * Sets the init data types that will be used by the EME helper. This is used
+ * when calling `navigator.requestMediaKeySystemAccess`.
+ * @param initDataTypes A list containing the init data types to be set by
+ * the helper. This will usually be ["cenc"] or ["webm"], see
+ * https://www.w3.org/TR/eme-initdata-registry/ for more info on what these
+ * mean.
+ */
+ SetInitDataTypes(initDataTypes) {
+ this._initDataTypes = initDataTypes;
+ }
+
+ /**
+ * Sets the audio capabilities that will be used by the EME helper. These are
+ * used when calling `navigator.requestMediaKeySystemAccess`.
+ * See https://developer.mozilla.org/en-US/docs/Web/API/Navigator/requestMediaKeySystemAccess
+ * for more info on these.
+ * @param audioCapabilities A list containing audio capabilities. E.g.
+ * [{ contentType: 'audio/webm; codecs="opus"' }].
+ */
+ SetAudioCapabilities(audioCapabilities) {
+ this._audioCapabilities = audioCapabilities;
+ }
+
+ /**
+ * Sets the video capabilities that will be used by the EME helper. These are
+ * used when calling `navigator.requestMediaKeySystemAccess`.
+ * See https://developer.mozilla.org/en-US/docs/Web/API/Navigator/requestMediaKeySystemAccess
+ * for more info on these.
+ * @param videoCapabilities A list containing video capabilities. E.g.
+ * [{ contentType: 'video/webm; codecs="vp9"' }]
+ */
+ SetVideoCapabilities(videoCapabilities) {
+ this._videoCapabilities = videoCapabilities;
+ }
+
+ /**
+ * Adds a key id and key pair to the key map. These should both be hex
+ * strings. E.g.
+ * emeHelper.AddKeyIdAndKey(
+ * "2cdb0ed6119853e7850671c3e9906c3c",
+ * "808b9adac384de1e4f56140f4ad76194"
+ * );
+ * This function will store the keyId and key in lower case to ensure
+ * consistency internally.
+ * @param keyId The key id used to lookup the following key.
+ * @param key The key associated with the earlier key id.
+ */
+ AddKeyIdAndKey(keyId, key) {
+ this._keyMap.set(keyId.toLowerCase(), key.toLowerCase());
+ }
+
+ /**
+ * Removes a key id and its associate key from the key map.
+ * @param keyId The key id to remove.
+ */
+ RemoveKeyIdAndKey(keyId) {
+ this._keyMap.delete(keyId);
+ }
+
+ // End setters that setup the helper.
+
+ /**
+ * Internal handler for `session.onmessage`. When calling this either do so
+ * from inside an arrow function or using `bind` to ensure `this` points to
+ * an EmeHelper instance (rather than a session).
+ * @param messageEvent The message event passed to `session.onmessage`.
+ */
+ _SessionMessageHandler(messageEvent) {
+ // This handles a session message and generates a clearkey license based
+ // on the information in this._keyMap. This is done by populating the
+ // appropriate keys on the session based on the keyIds surfaced in the
+ // session message (a license request).
+ let request = JSON.parse(new TextDecoder().decode(messageEvent.message));
+
+ let keys = [];
+ for (const keyId of request.kids) {
+ let id64 = keyId;
+ let idHex = EmeHelper.Base64ToHex(keyId);
+ let key = this._keyMap.get(idHex);
+
+ if (key) {
+ keys.push({
+ kty: "oct",
+ kid: id64,
+ k: EmeHelper.HexToBase64(key),
+ });
+ }
+ }
+
+ let license = new TextEncoder().encode(
+ JSON.stringify({
+ keys,
+ type: request.type || "temporary",
+ })
+ );
+
+ let session = messageEvent.target;
+ session.update(license).catch(error => {
+ if (this.onerror) {
+ this.onerror(error);
+ } else {
+ console.log(
+ `EmeHelper got an error, but no onerror handler was registered! Logging to console, error: ${error}`
+ );
+ }
+ });
+ }
+
+ /**
+ * Configures EME on a media element using the parameters already set on the
+ * instance of EmeHelper.
+ * @param htmlMediaElement - A media element to configure EME on.
+ * @return A promise that will be resolved once the media element is
+ * configured. This promise will be rejected with an error if configuration
+ * fails.
+ */
+ async ConfigureEme(htmlMediaElement) {
+ if (!this._keySystem) {
+ throw new Error("EmeHelper needs _keySystem to configure media");
+ }
+ if (!this._initDataTypes) {
+ throw new Error("EmeHelper needs _initDataTypes to configure media");
+ }
+ if (!this._audioCapabilities.length && !this._videoCapabilities.length) {
+ throw new Error(
+ "EmeHelper needs _audioCapabilities or _videoCapabilities to configure media"
+ );
+ }
+ const options = [
+ {
+ initDataTypes: this._initDataTypes,
+ audioCapabilities: this._audioCapabilities,
+ videoCapabilities: this._videoCapabilities,
+ },
+ ];
+ let access = await window.navigator.requestMediaKeySystemAccess(
+ this._keySystem,
+ options
+ );
+ let mediaKeys = await access.createMediaKeys();
+ await htmlMediaElement.setMediaKeys(mediaKeys);
+
+ htmlMediaElement.onencrypted = async encryptedEvent => {
+ let session = htmlMediaElement.mediaKeys.createSession();
+ // Use arrow notation so that `this` is the EmeHelper in the message
+ // handler. If we do `session.onmessage = this._SessionMessageHandler`
+ // then `this` will be the session in the callback.
+ session.onmessage = messageEvent =>
+ this._SessionMessageHandler(messageEvent);
+ try {
+ await session.generateRequest(
+ encryptedEvent.initDataType,
+ encryptedEvent.initData
+ );
+ } catch (error) {
+ if (this.onerror) {
+ this.onerror(error);
+ } else {
+ console.log(
+ `EmeHelper got an error, but no onerror handler was registered! Logging to console, error: ${error}`
+ );
+ }
+ }
+ };
+ }
+};
diff --git a/dom/media/test/empty_size.mp3 b/dom/media/test/empty_size.mp3
new file mode 100644
index 0000000000..0c208a2959
--- /dev/null
+++ b/dom/media/test/empty_size.mp3
Binary files differ
diff --git a/dom/media/test/file_access_controls.html b/dom/media/test/file_access_controls.html
new file mode 100644
index 0000000000..2f7bc360ed
--- /dev/null
+++ b/dom/media/test/file_access_controls.html
@@ -0,0 +1,160 @@
+<html>
+<head>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body onload="setTimeout(load, 0);">
+<script>
+
+// Page URL: http://example.org/tests/dom/media/test/file_access_controls.html
+
+var gResource = getPlayableVideo(gSmallTests).name;
+
+var gTests = [
+ {
+ // Test 0
+ url: "redirect.sjs?domain=example.com&file="+ gResource,
+ result: "error",
+ description: "Won't load when redirected to different domain",
+ },{
+ // Test 1
+ url: "redirect.sjs?domain=example.com&allowed&file=" + gResource,
+ result: "loadeddata",
+ description: "Can load when redirected to different domain with allow-origin",
+ },{
+ // Test 2
+ url: "redirect.sjs?domain=test1.example.org&file=" + gResource,
+ result: "error",
+ description: "Won't load when redirected to subdomain",
+ },{
+ // Test 3
+ url: "redirect.sjs?domain=test1.example.org&allowed&file=" + gResource,
+ result: "loadeddata",
+ description: "Can load when redirected to subdomain with allow-origin",
+ },{
+ // Test 4
+ url: "redirect.sjs?domain=example.org&file=" + gResource,
+ result: "loadeddata",
+ description: "Can load when redirected to same domain",
+ },{
+ // Test 5
+ url: "http://example.org/tests/dom/media/test/" + gResource,
+ result: "loadeddata",
+ description: "Can load from same domain"
+ },{
+ // Test 6
+ url: "http://example.org:8000/tests/dom/media/test/" + gResource,
+ result: "error",
+ description: "Won't load from different port on same domain"
+ },{
+ // Test 7
+ url: "http://example.org:8000/tests/dom/media/test/allowed.sjs?" + gResource,
+ result: "loadeddata",
+ description: "Can load from different port on same domain with allow-origin",
+ },{
+ // Test 8
+ url: "http://example.com/tests/dom/media/test/" + gResource,
+ result: "error",
+ description: "Won't load cross domain",
+ },{
+ // Test 9
+ url: "http://example.com/tests/dom/media/test/allowed.sjs?" + gResource,
+ result: "loadeddata",
+ description: "Can load cross domain with allow-origin",
+ },{
+ // Test 10
+ url: "http://test1.example.org/tests/dom/media/test/allowed.sjs?" + gResource,
+ result: "loadeddata",
+ description: "Can load from subdomain with allow-origin",
+ },{
+ // Test 11
+ url: "http://test1.example.org/tests/dom/media/test/" + gResource,
+ result: "error",
+ description: "Won't load from subdomain",
+ }
+];
+
+var gTestNum = 0;
+var gVideo = null;
+var gTestedRemoved = false;
+
+function eventHandler(event) {
+ //dump((gTestNum - 1) + ": " + event.type + "\n");
+ var video = event.target;
+ opener.postMessage({"result": (event.type == video.expectedResult),
+ "message": video.testDescription + (gTestedRemoved ? " (element not in document)" : " (element in document)")},
+ "http://mochi.test:8888");
+ // Make sure any extra events cause an error
+ video.expectedResult = "<none>";
+ nextTest();
+}
+
+function createVideo() {
+ var v = document.createElement('video');
+ v.addEventListener('loadeddata', eventHandler);
+ v.addEventListener('error', eventHandler);
+ v.crossOrigin = 'anonymous';
+ return v;
+}
+
+function load() {
+ opener.postMessage({"result": (window.location.href == "http://example.org/tests/dom/media/test/file_access_controls.html"),
+ "message": "We must be on a example.org:80"},
+ "http://mochi.test:8888");
+
+ nextTest();
+}
+
+function nextTest() {
+ //dump("nextTest() called, gTestNum="+gTestNum+" gTestedRemoved="+gTestedRemoved+"\n");
+ if (gTestNum == gTests.length) {
+ //dump("gTestNum == gTests.length\n");
+ if (!gTestedRemoved) {
+ // Repeat all tests with element removed from doc, should get same result.
+ gTestedRemoved = true;
+ gTestNum = 0;
+ } else {
+ //dump("Exiting...\n");
+ // We're done, exit the test.
+ done();
+ window.close();
+ return;
+ }
+ }
+
+ if (gVideo) {
+ gVideo.remove();
+ gVideo.removeAttribute("src");
+ gVideo.load();
+ }
+
+ gVideo = null;
+ SpecialPowers.forceGC();
+
+ gVideo = createVideo();
+ gVideo.expectedResult = gTests[gTestNum].result;
+ gVideo.testDescription = gTests[gTestNum].description;
+ // Uniquify the resource URL to ensure that the resources loaded by earlier or subsequent tests
+ // don't overlap with the resources we load here, which are loaded with non-default preferences set.
+ // We also want to make sure that an HTTP fetch actually happens for each testcase.
+ var url = gTests[gTestNum].url;
+ var random = Math.floor(Math.random()*1000000000);
+ url += (url.search(/\?/) < 0 ? "?" : "&") + "rand=" + random;
+ gVideo.src = url;
+ //dump("Starting test " + gTestNum + " at " + gVideo.src + " expecting:" + gVideo.expectedResult + "\n");
+ if (!gTestedRemoved) {
+ document.body.appendChild(gVideo);
+ // Will cause load() to be invoked.
+ } else {
+ gVideo.load();
+ }
+ gTestNum++;
+}
+
+function done() {
+ opener.postMessage({"done": "true"}, "http://mochi.test:8888");
+}
+
+</script>
+</body>
+</html>
+
diff --git a/dom/media/test/file_eme_createMediaKeys.html b/dom/media/test/file_eme_createMediaKeys.html
new file mode 100644
index 0000000000..3ff782b1bf
--- /dev/null
+++ b/dom/media/test/file_eme_createMediaKeys.html
@@ -0,0 +1,47 @@
+<!DOCTYPE html>
+<html>
+<head>
+<title>Eme createMediaKeys helper page</title>
+<script>
+// This script waits for a message then attempts to requestMediaKeySystemAccess
+// then createMediaKeys. On success posts 'successCreatingMediaKeys' to the
+// source of the message, on failure posts 'failureCreatingMediaKeys' and a
+// description of the failure to the source of the message.
+
+async function createMediaKeys() {
+ const clearKeyOptions = [
+ {
+ initDataTypes: ["webm"],
+ videoCapabilities: [{ contentType: 'video/webm; codecs="vp9"' }],
+ },
+ ];
+
+ let access = await navigator.requestMediaKeySystemAccess(
+ "org.w3.clearkey",
+ clearKeyOptions
+ );
+
+ return access.createMediaKeys();
+}
+function setupMessageListener() {
+ window.onmessage = async event => {
+ // We don't bother checking the message data since it should always be
+ // telling us to create media keys.
+ try {
+ let keys = await createMediaKeys();
+ if (!keys) {
+ event.source.postMessage("failureCreatingMediaKeys no keys", "*");
+ return;
+ }
+ event.source.postMessage("successCreatingMediaKeys", "*");
+ } catch (e) {
+ event.source.postMessage(`failureCreatingMediaKeys ${e}`, "*");
+ }
+ };
+}
+window.onload = setupMessageListener;
+</script>
+</head>
+<body>
+</body>
+</html>
diff --git a/dom/media/test/file_playback_and_bfcache.html b/dom/media/test/file_playback_and_bfcache.html
new file mode 100644
index 0000000000..4798e1487e
--- /dev/null
+++ b/dom/media/test/file_playback_and_bfcache.html
@@ -0,0 +1,57 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <script>
+ function init() {
+ if (location.search == "") {
+ let bc1 = new BroadcastChannel("bc1");
+ bc1.onmessage = function(e) {
+ if (e.data == "loadNext") {
+ location.href = location.href + "?page2";
+ } else if (e.data == "forward") {
+ bc1.close();
+ history.forward();
+ }
+ };
+ window.onpageshow = function() {
+ bc1.postMessage("pageshow");
+ };
+ } else {
+ document.body.innerHTML = "<video controls src='owl.mp3' autoplay>";
+ let bc2 = new BroadcastChannel("bc2");
+ bc2.onmessage = function(e) {
+ if (e.data == "back") {
+ history.back();
+ } else if (e.data == "statistics") {
+ bc2.postMessage({ currentTime: document.body.firstChild.currentTime,
+ duration: document.body.firstChild.duration });
+ bc2.close();
+ window.close();
+ }
+ }
+ window.onpageshow = function(e) {
+ bc2.postMessage({ event: "pageshow", persisted: e.persisted});
+ if (!e.persisted) {
+ // The initial statistics is sent once we know the duration and
+ // have loaded all the data.
+ let mediaElement = document.body.firstChild;
+ mediaElement.onpause = function() {
+ mediaElement.onpause = null;
+ mediaElement.currentTime = 0;
+ mediaElement.onplay = function() {
+ setTimeout(function() {
+ bc2.postMessage({ currentTime: mediaElement.currentTime,
+ duration: mediaElement.duration });
+ }, 500);
+ }
+ mediaElement.play();
+ }
+ }
+ };
+ }
+ }
+ </script>
+</head>
+<body onload="init()">
+</body>
+</html>
diff --git a/dom/media/test/flac-noheader-s16.flac b/dom/media/test/flac-noheader-s16.flac
new file mode 100644
index 0000000000..01152142a9
--- /dev/null
+++ b/dom/media/test/flac-noheader-s16.flac
Binary files differ
diff --git a/dom/media/test/flac-noheader-s16.flac^headers^ b/dom/media/test/flac-noheader-s16.flac^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/flac-noheader-s16.flac^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/flac-s24.flac b/dom/media/test/flac-s24.flac
new file mode 100644
index 0000000000..1ba5e27a15
--- /dev/null
+++ b/dom/media/test/flac-s24.flac
Binary files differ
diff --git a/dom/media/test/flac-s24.flac^headers^ b/dom/media/test/flac-s24.flac^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/flac-s24.flac^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/flac-sample-cenc.mp4 b/dom/media/test/flac-sample-cenc.mp4
new file mode 100644
index 0000000000..c89190387a
--- /dev/null
+++ b/dom/media/test/flac-sample-cenc.mp4
Binary files differ
diff --git a/dom/media/test/flac-sample-cenc.mp4^headers^ b/dom/media/test/flac-sample-cenc.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/flac-sample-cenc.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/flac-sample.mp4 b/dom/media/test/flac-sample.mp4
new file mode 100644
index 0000000000..d39c94a7b2
--- /dev/null
+++ b/dom/media/test/flac-sample.mp4
Binary files differ
diff --git a/dom/media/test/flac-sample.mp4^headers^ b/dom/media/test/flac-sample.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/flac-sample.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/force_octet_stream.mp4 b/dom/media/test/force_octet_stream.mp4
new file mode 100644
index 0000000000..802da047bc
--- /dev/null
+++ b/dom/media/test/force_octet_stream.mp4
Binary files differ
diff --git a/dom/media/test/force_octet_stream.mp4^headers^ b/dom/media/test/force_octet_stream.mp4^headers^
new file mode 100644
index 0000000000..d554cbedb5
--- /dev/null
+++ b/dom/media/test/force_octet_stream.mp4^headers^
@@ -0,0 +1,2 @@
+Cache-Control: no-store
+Content-Type: application/octet-stream
diff --git a/dom/media/test/fragment_noplay.js b/dom/media/test/fragment_noplay.js
new file mode 100644
index 0000000000..247641763f
--- /dev/null
+++ b/dom/media/test/fragment_noplay.js
@@ -0,0 +1,19 @@
+function test_fragment_noplay(v, start, end, is, ok, finish) {
+ function onLoadedMetadata() {
+ var s = start == null ? 0 : start;
+ var e = end == null ? v.duration : end;
+ var a = s - 0.15;
+ var b = s + 0.15;
+ ok(
+ v.currentTime >= a && v.currentTime <= b,
+ "loadedmetadata currentTime is " + a + " < " + v.currentTime + " < " + b
+ );
+ ok(
+ v.mozFragmentEnd == e,
+ "mozFragmentEnd (" + v.mozFragmentEnd + ") == end Time (" + e + ")"
+ );
+ finish();
+ }
+
+ v.addEventListener("loadedmetadata", onLoadedMetadata);
+}
diff --git a/dom/media/test/fragment_play.js b/dom/media/test/fragment_play.js
new file mode 100644
index 0000000000..ec0fe7952a
--- /dev/null
+++ b/dom/media/test/fragment_play.js
@@ -0,0 +1,92 @@
+function test_fragment_play(v, start, end, is, ok, finish) {
+ var completed = false;
+ var loadedMetadataRaised = false;
+ var seekedRaised = false;
+ var pausedRaised = false;
+
+ function onLoadedMetadata() {
+ var s = start == null ? 0 : start;
+ var e = end == null ? v.duration : end;
+ ok(
+ v.currentTime == s,
+ "loadedmetadata currentTime is " + v.currentTime + " != " + s
+ );
+ ok(
+ v.mozFragmentEnd == e,
+ "mozFragmentEnd (" + v.mozFragmentEnd + ") == end Time (" + e + ")"
+ );
+ loadedMetadataRaised = true;
+ v.play();
+ }
+
+ function onSeeked() {
+ if (completed) {
+ return;
+ }
+
+ var s = start == null ? 0 : start;
+ ok(
+ v.currentTime - s < 0.1,
+ "seeked currentTime is " +
+ v.currentTime +
+ " != " +
+ s +
+ " (fuzzy compare +-0.1)"
+ );
+
+ seekedRaised = true;
+ }
+
+ function onTimeUpdate() {
+ if (completed) {
+ return;
+ }
+
+ v._lastTimeUpdate = v.currentTime;
+ }
+
+ function onPause() {
+ if (completed) {
+ return;
+ }
+
+ var e = end == null ? v.duration : end;
+ var a = e - 0.05;
+ var b = e + 0.05;
+ ok(
+ v.currentTime >= a && v.currentTime <= b,
+ "paused currentTime is " +
+ a +
+ " < " +
+ v.currentTime +
+ " < " +
+ b +
+ " ? " +
+ v._lastTimeUpdate
+ );
+ pausedRaised = true;
+ v.play();
+ }
+
+ function onEnded() {
+ if (completed) {
+ return;
+ }
+
+ completed = true;
+ ok(loadedMetadataRaised, "loadedmetadata event");
+ if (start) {
+ ok(seekedRaised, "seeked event");
+ }
+ if (end) {
+ ok(pausedRaised, "paused event: " + end + " " + v.duration);
+ }
+ finish();
+ }
+
+ v.addEventListener("ended", onEnded);
+ v.addEventListener("loadedmetadata", onLoadedMetadata);
+ v.addEventListener("seeked", onSeeked);
+ v.addEventListener("pause", onPause);
+ v.addEventListener("timeupdate", onTimeUpdate);
+}
diff --git a/dom/media/test/gUM_support.js b/dom/media/test/gUM_support.js
new file mode 100644
index 0000000000..80d218cae7
--- /dev/null
+++ b/dom/media/test/gUM_support.js
@@ -0,0 +1,103 @@
+// Support script for test that use getUserMedia. This allows explicit
+// configuration of prefs which affect gUM. See also
+// `testing/mochitest/runtests.py` for how the harness configures values.
+
+// Setup preconditions for tests using getUserMedia. This functions helps
+// manage different prefs that affect gUM calls in tests and makes explicit
+// the expected state before test runs.
+async function pushGetUserMediaTestPrefs({
+ fakeAudio = false,
+ fakeVideo = false,
+ loopbackAudio = false,
+ loopbackVideo = false,
+}) {
+ // Make sure we have sensical arguments
+ if (!fakeAudio && !loopbackAudio) {
+ throw new Error(
+ "pushGetUserMediaTestPrefs: Should have fake or loopback audio!"
+ );
+ } else if (fakeAudio && loopbackAudio) {
+ throw new Error(
+ "pushGetUserMediaTestPrefs: Should not have both fake and loopback audio!"
+ );
+ }
+ if (!fakeVideo && !loopbackVideo) {
+ throw new Error(
+ "pushGetUserMediaTestPrefs: Should have fake or loopback video!"
+ );
+ } else if (fakeVideo && loopbackVideo) {
+ throw new Error(
+ "pushGetUserMediaTestPrefs: Should not have both fake and loopback video!"
+ );
+ }
+
+ let testPrefs = [];
+ if (fakeAudio) {
+ // Unset the loopback device so it doesn't take precedence
+ testPrefs.push(["media.audio_loopback_dev", ""]);
+ // Setup fake streams pref
+ testPrefs.push(["media.navigator.streams.fake", true]);
+ }
+ if (loopbackAudio) {
+ // If audio loopback is requested we expect the test harness to have set
+ // the loopback device pref, make sure it's set
+ let audioLoopDev = SpecialPowers.getCharPref(
+ "media.audio_loopback_dev",
+ ""
+ );
+ if (!audioLoopDev) {
+ throw new Error(
+ "pushGetUserMediaTestPrefs: Loopback audio requested but " +
+ "media.audio_loopback_dev does not appear to be set!"
+ );
+ }
+ }
+ if (fakeVideo) {
+ // Unset the loopback device so it doesn't take precedence
+ testPrefs.push(["media.video_loopback_dev", ""]);
+ // Setup fake streams pref
+ testPrefs.push(["media.navigator.streams.fake", true]);
+ }
+ if (loopbackVideo) {
+ // If video loopback is requested we expect the test harness to have set
+ // the loopback device pref, make sure it's set
+ let videoLoopDev = SpecialPowers.getCharPref(
+ "media.video_loopback_dev",
+ ""
+ );
+ if (!videoLoopDev) {
+ throw new Error(
+ "pushGetUserMediaTestPrefs: Loopback video requested but " +
+ "media.video_loopback_dev does not appear to be set!"
+ );
+ }
+ }
+ // Prevent presentation of the gUM permission prompt.
+ testPrefs.push(["media.navigator.permission.disabled", true]);
+ return SpecialPowers.pushPrefEnv({ set: testPrefs });
+}
+
+// Setup preconditions for tests using getUserMedia. This function will
+// configure prefs to select loopback device(s) if it can find loopback device
+// names already set in the prefs. If no loopback device name can be found then
+// prefs are setup such that a fake device is used.
+async function setupGetUserMediaTestPrefs() {
+ let prefRequests = {};
+ let audioLoopDev = SpecialPowers.getCharPref("media.audio_loopback_dev", "");
+ if (audioLoopDev) {
+ prefRequests.fakeAudio = false;
+ prefRequests.loopbackAudio = true;
+ } else {
+ prefRequests.fakeAudio = true;
+ prefRequests.loopbackAudio = false;
+ }
+ let videoLoopDev = SpecialPowers.getCharPref("media.video_loopback_dev", "");
+ if (videoLoopDev) {
+ prefRequests.fakeVideo = false;
+ prefRequests.loopbackVideo = true;
+ } else {
+ prefRequests.fakeVideo = true;
+ prefRequests.loopbackVideo = false;
+ }
+ return pushGetUserMediaTestPrefs(prefRequests);
+}
diff --git a/dom/media/test/gizmo-frag.mp4 b/dom/media/test/gizmo-frag.mp4
new file mode 100644
index 0000000000..f6980663c2
--- /dev/null
+++ b/dom/media/test/gizmo-frag.mp4
Binary files differ
diff --git a/dom/media/test/gizmo-noaudio.mp4 b/dom/media/test/gizmo-noaudio.mp4
new file mode 100644
index 0000000000..24732a4064
--- /dev/null
+++ b/dom/media/test/gizmo-noaudio.mp4
Binary files differ
diff --git a/dom/media/test/gizmo-noaudio.mp4^headers^ b/dom/media/test/gizmo-noaudio.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/gizmo-noaudio.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/gizmo-noaudio.webm b/dom/media/test/gizmo-noaudio.webm
new file mode 100644
index 0000000000..9f412cb6e3
--- /dev/null
+++ b/dom/media/test/gizmo-noaudio.webm
Binary files differ
diff --git a/dom/media/test/gizmo-noaudio.webm^headers^ b/dom/media/test/gizmo-noaudio.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/gizmo-noaudio.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/gizmo-short.mp4 b/dom/media/test/gizmo-short.mp4
new file mode 100644
index 0000000000..f8caec741e
--- /dev/null
+++ b/dom/media/test/gizmo-short.mp4
Binary files differ
diff --git a/dom/media/test/gizmo-short.mp4^headers^ b/dom/media/test/gizmo-short.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/gizmo-short.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/gizmo.mp4 b/dom/media/test/gizmo.mp4
new file mode 100644
index 0000000000..87efad5ade
--- /dev/null
+++ b/dom/media/test/gizmo.mp4
Binary files differ
diff --git a/dom/media/test/gizmo.mp4^headers^ b/dom/media/test/gizmo.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/gizmo.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/gizmo.webm b/dom/media/test/gizmo.webm
new file mode 100644
index 0000000000..518531a93f
--- /dev/null
+++ b/dom/media/test/gizmo.webm
Binary files differ
diff --git a/dom/media/test/gizmo.webm^headers^ b/dom/media/test/gizmo.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/gizmo.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/gzipped_mp4.sjs b/dom/media/test/gzipped_mp4.sjs
new file mode 100644
index 0000000000..b08362d75e
--- /dev/null
+++ b/dom/media/test/gzipped_mp4.sjs
@@ -0,0 +1,25 @@
+function getGzippedFileBytes() {
+ var file;
+ getObjectState("SERVER_ROOT", function (serverRoot) {
+ file = serverRoot.getFile("tests/dom/media/test/short.mp4.gz");
+ });
+ var fileInputStream = Cc[
+ "@mozilla.org/network/file-input-stream;1"
+ ].createInstance(Ci.nsIFileInputStream);
+ var binaryInputStream = Cc["@mozilla.org/binaryinputstream;1"].createInstance(
+ Ci.nsIBinaryInputStream
+ );
+ fileInputStream.init(file, -1, -1, 0);
+ binaryInputStream.setInputStream(fileInputStream);
+ return binaryInputStream.readBytes(binaryInputStream.available());
+}
+
+function handleRequest(request, response) {
+ var bytes = getGzippedFileBytes();
+ response.setHeader("Content-Length", String(bytes.length), false);
+ response.setHeader("Content-Type", "video/mp4", false);
+ response.setHeader("Access-Control-Allow-Origin", "*", false);
+ response.setHeader("Content-Encoding", "gzip", false);
+ response.setHeader("Cache-Control", "no-cache", false);
+ response.write(bytes, bytes.length);
+}
diff --git a/dom/media/test/hevc_white_frame.mp4 b/dom/media/test/hevc_white_frame.mp4
new file mode 100644
index 0000000000..27d642346e
--- /dev/null
+++ b/dom/media/test/hevc_white_frame.mp4
Binary files differ
diff --git a/dom/media/test/hevc_white_frame.mp4^headers^ b/dom/media/test/hevc_white_frame.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/hevc_white_frame.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/hevc_white_red_frames.mp4 b/dom/media/test/hevc_white_red_frames.mp4
new file mode 100644
index 0000000000..5da0aecd17
--- /dev/null
+++ b/dom/media/test/hevc_white_red_frames.mp4
Binary files differ
diff --git a/dom/media/test/hevc_white_red_frames.mp4^headers^ b/dom/media/test/hevc_white_red_frames.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/hevc_white_red_frames.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/hls/400x300_prog_index.m3u8 b/dom/media/test/hls/400x300_prog_index.m3u8
new file mode 100644
index 0000000000..3252eb178f
--- /dev/null
+++ b/dom/media/test/hls/400x300_prog_index.m3u8
@@ -0,0 +1,10 @@
+#EXTM3U
+#EXT-X-TARGETDURATION:10
+#EXT-X-VERSION:3
+#EXT-X-MEDIA-SEQUENCE:0
+#EXT-X-PLAYLIST-TYPE:VOD
+#EXTINF:9.97667,
+400x300_seg0.ts
+#EXTINF:9.97667,
+400x300_seg1.ts
+#EXT-X-ENDLIST
diff --git a/dom/media/test/hls/400x300_prog_index_5s.m3u8 b/dom/media/test/hls/400x300_prog_index_5s.m3u8
new file mode 100644
index 0000000000..8e9d19f764
--- /dev/null
+++ b/dom/media/test/hls/400x300_prog_index_5s.m3u8
@@ -0,0 +1,8 @@
+#EXTM3U
+#EXT-X-TARGETDURATION:4.00
+#EXT-X-VERSION:3
+#EXT-X-MEDIA-SEQUENCE:0
+#EXT-X-PLAYLIST-TYPE:VOD
+#EXTINF:4.00,
+400x300_seg0_5s.ts
+#EXT-X-ENDLIST
diff --git a/dom/media/test/hls/400x300_seg0.ts b/dom/media/test/hls/400x300_seg0.ts
new file mode 100644
index 0000000000..b17b0a88ff
--- /dev/null
+++ b/dom/media/test/hls/400x300_seg0.ts
Binary files differ
diff --git a/dom/media/test/hls/400x300_seg0_5s.ts b/dom/media/test/hls/400x300_seg0_5s.ts
new file mode 100644
index 0000000000..9504d5d889
--- /dev/null
+++ b/dom/media/test/hls/400x300_seg0_5s.ts
Binary files differ
diff --git a/dom/media/test/hls/400x300_seg1.ts b/dom/media/test/hls/400x300_seg1.ts
new file mode 100644
index 0000000000..d751091e68
--- /dev/null
+++ b/dom/media/test/hls/400x300_seg1.ts
Binary files differ
diff --git a/dom/media/test/hls/416x243_prog_index_5s.m3u8 b/dom/media/test/hls/416x243_prog_index_5s.m3u8
new file mode 100644
index 0000000000..ae518e7890
--- /dev/null
+++ b/dom/media/test/hls/416x243_prog_index_5s.m3u8
@@ -0,0 +1,8 @@
+#EXTM3U
+#EXT-X-TARGETDURATION:4.04
+#EXT-X-VERSION:4
+#EXT-X-MEDIA-SEQUENCE:0
+#EXT-X-PLAYLIST-TYPE:VOD
+#EXTINF:4.04,
+416x243_seg0_5s.ts
+#EXT-X-ENDLIST
diff --git a/dom/media/test/hls/416x243_seg0_5s.ts b/dom/media/test/hls/416x243_seg0_5s.ts
new file mode 100644
index 0000000000..48e5473276
--- /dev/null
+++ b/dom/media/test/hls/416x243_seg0_5s.ts
Binary files differ
diff --git a/dom/media/test/hls/640x480_prog_index.m3u8 b/dom/media/test/hls/640x480_prog_index.m3u8
new file mode 100644
index 0000000000..ac1212e2e7
--- /dev/null
+++ b/dom/media/test/hls/640x480_prog_index.m3u8
@@ -0,0 +1,10 @@
+#EXTM3U
+#EXT-X-TARGETDURATION:10
+#EXT-X-VERSION:3
+#EXT-X-MEDIA-SEQUENCE:0
+#EXT-X-PLAYLIST-TYPE:VOD
+#EXTINF:9.97667,
+640x480_seg0.ts
+#EXTINF:9.97667,
+640x480_seg1.ts
+#EXT-X-ENDLIST
diff --git a/dom/media/test/hls/640x480_seg0.ts b/dom/media/test/hls/640x480_seg0.ts
new file mode 100644
index 0000000000..9bf0f0454a
--- /dev/null
+++ b/dom/media/test/hls/640x480_seg0.ts
Binary files differ
diff --git a/dom/media/test/hls/640x480_seg1.ts b/dom/media/test/hls/640x480_seg1.ts
new file mode 100644
index 0000000000..c1ed938f44
--- /dev/null
+++ b/dom/media/test/hls/640x480_seg1.ts
Binary files differ
diff --git a/dom/media/test/hls/960x720_prog_index.m3u8 b/dom/media/test/hls/960x720_prog_index.m3u8
new file mode 100644
index 0000000000..8ff18b089c
--- /dev/null
+++ b/dom/media/test/hls/960x720_prog_index.m3u8
@@ -0,0 +1,10 @@
+#EXTM3U
+#EXT-X-TARGETDURATION:10
+#EXT-X-VERSION:3
+#EXT-X-MEDIA-SEQUENCE:0
+#EXT-X-PLAYLIST-TYPE:VOD
+#EXTINF:9.97667,
+960x720_seg0.ts
+#EXTINF:9.97667,
+960x720_seg1.ts
+#EXT-X-ENDLIST
diff --git a/dom/media/test/hls/960x720_seg0.ts b/dom/media/test/hls/960x720_seg0.ts
new file mode 100644
index 0000000000..031bfe30d6
--- /dev/null
+++ b/dom/media/test/hls/960x720_seg0.ts
Binary files differ
diff --git a/dom/media/test/hls/960x720_seg1.ts b/dom/media/test/hls/960x720_seg1.ts
new file mode 100644
index 0000000000..63f15cb0cb
--- /dev/null
+++ b/dom/media/test/hls/960x720_seg1.ts
Binary files differ
diff --git a/dom/media/test/hls/bipbop_16x9_single.m3u8 b/dom/media/test/hls/bipbop_16x9_single.m3u8
new file mode 100644
index 0000000000..dce6a76c7b
--- /dev/null
+++ b/dom/media/test/hls/bipbop_16x9_single.m3u8
@@ -0,0 +1,5 @@
+#EXTM3U
+
+#EXT-X-STREAM-INF:BANDWIDTH=263851,CODECS="mp4a.40.2, avc1.4d400d"
+416x243_prog_index_5s.m3u8
+
diff --git a/dom/media/test/hls/bipbop_4x3_single.m3u8 b/dom/media/test/hls/bipbop_4x3_single.m3u8
new file mode 100644
index 0000000000..8f354ff011
--- /dev/null
+++ b/dom/media/test/hls/bipbop_4x3_single.m3u8
@@ -0,0 +1,4 @@
+#EXTM3U
+
+#EXT-X-STREAM-INF:PROGRAM-ID=1,BANDWIDTH=232370,CODECS="mp4a.40.2, avc1.4d4015"
+400x300_prog_index_5s.m3u8 \ No newline at end of file
diff --git a/dom/media/test/hls/bipbop_4x3_variant.m3u8 b/dom/media/test/hls/bipbop_4x3_variant.m3u8
new file mode 100644
index 0000000000..8a9a100dba
--- /dev/null
+++ b/dom/media/test/hls/bipbop_4x3_variant.m3u8
@@ -0,0 +1,10 @@
+#EXTM3U
+
+#EXT-X-STREAM-INF:PROGRAM-ID=1,BANDWIDTH=232370,CODECS="mp4a.40.2, avc1.4d4015"
+400x300_prog_index.m3u8
+
+#EXT-X-STREAM-INF:PROGRAM-ID=1,BANDWIDTH=649879,CODECS="mp4a.40.2, avc1.4d401e"
+640x480_prog_index.m3u8
+
+#EXT-X-STREAM-INF:PROGRAM-ID=1,BANDWIDTH=649879,CODECS="mp4a.40.2, avc1.4d401e"
+960x720_prog_index.m3u8
diff --git a/dom/media/test/huge-id3.mp3 b/dom/media/test/huge-id3.mp3
new file mode 100644
index 0000000000..41cb93d805
--- /dev/null
+++ b/dom/media/test/huge-id3.mp3
Binary files differ
diff --git a/dom/media/test/huge-id3.mp3^headers^ b/dom/media/test/huge-id3.mp3^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/huge-id3.mp3^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/id3tags.mp3 b/dom/media/test/id3tags.mp3
new file mode 100644
index 0000000000..bad506cf18
--- /dev/null
+++ b/dom/media/test/id3tags.mp3
Binary files differ
diff --git a/dom/media/test/id3tags.mp3^headers^ b/dom/media/test/id3tags.mp3^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/id3tags.mp3^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/id3v1afterlongid3v2.mp3 b/dom/media/test/id3v1afterlongid3v2.mp3
new file mode 100644
index 0000000000..8de7bde0d3
--- /dev/null
+++ b/dom/media/test/id3v1afterlongid3v2.mp3
Binary files differ
diff --git a/dom/media/test/invalid-cmap-s0c0.opus b/dom/media/test/invalid-cmap-s0c0.opus
new file mode 100644
index 0000000000..0b99587865
--- /dev/null
+++ b/dom/media/test/invalid-cmap-s0c0.opus
Binary files differ
diff --git a/dom/media/test/invalid-cmap-s0c0.opus^headers^ b/dom/media/test/invalid-cmap-s0c0.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/invalid-cmap-s0c0.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/invalid-cmap-s0c2.opus b/dom/media/test/invalid-cmap-s0c2.opus
new file mode 100644
index 0000000000..a921894fee
--- /dev/null
+++ b/dom/media/test/invalid-cmap-s0c2.opus
Binary files differ
diff --git a/dom/media/test/invalid-cmap-s0c2.opus^headers^ b/dom/media/test/invalid-cmap-s0c2.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/invalid-cmap-s0c2.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/invalid-cmap-s1c2.opus b/dom/media/test/invalid-cmap-s1c2.opus
new file mode 100644
index 0000000000..95a84f523c
--- /dev/null
+++ b/dom/media/test/invalid-cmap-s1c2.opus
Binary files differ
diff --git a/dom/media/test/invalid-cmap-s1c2.opus^headers^ b/dom/media/test/invalid-cmap-s1c2.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/invalid-cmap-s1c2.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/invalid-cmap-short.opus b/dom/media/test/invalid-cmap-short.opus
new file mode 100644
index 0000000000..fcd7eb506a
--- /dev/null
+++ b/dom/media/test/invalid-cmap-short.opus
Binary files differ
diff --git a/dom/media/test/invalid-cmap-short.opus^headers^ b/dom/media/test/invalid-cmap-short.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/invalid-cmap-short.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/invalid-discard_on_multi_blocks.webm b/dom/media/test/invalid-discard_on_multi_blocks.webm
new file mode 100644
index 0000000000..f39ab5bcb8
--- /dev/null
+++ b/dom/media/test/invalid-discard_on_multi_blocks.webm
Binary files differ
diff --git a/dom/media/test/invalid-discard_on_multi_blocks.webm^headers^ b/dom/media/test/invalid-discard_on_multi_blocks.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/invalid-discard_on_multi_blocks.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/invalid-excess_discard.webm b/dom/media/test/invalid-excess_discard.webm
new file mode 100644
index 0000000000..5b34aca1a7
--- /dev/null
+++ b/dom/media/test/invalid-excess_discard.webm
Binary files differ
diff --git a/dom/media/test/invalid-excess_discard.webm^headers^ b/dom/media/test/invalid-excess_discard.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/invalid-excess_discard.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/invalid-excess_neg_discard.webm b/dom/media/test/invalid-excess_neg_discard.webm
new file mode 100644
index 0000000000..2bfad6ed21
--- /dev/null
+++ b/dom/media/test/invalid-excess_neg_discard.webm
Binary files differ
diff --git a/dom/media/test/invalid-excess_neg_discard.webm^headers^ b/dom/media/test/invalid-excess_neg_discard.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/invalid-excess_neg_discard.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/invalid-m0c0.opus b/dom/media/test/invalid-m0c0.opus
new file mode 100644
index 0000000000..86555f60eb
--- /dev/null
+++ b/dom/media/test/invalid-m0c0.opus
Binary files differ
diff --git a/dom/media/test/invalid-m0c0.opus^headers^ b/dom/media/test/invalid-m0c0.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/invalid-m0c0.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/invalid-m0c3.opus b/dom/media/test/invalid-m0c3.opus
new file mode 100644
index 0000000000..2c681a8c03
--- /dev/null
+++ b/dom/media/test/invalid-m0c3.opus
Binary files differ
diff --git a/dom/media/test/invalid-m0c3.opus^headers^ b/dom/media/test/invalid-m0c3.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/invalid-m0c3.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/invalid-m1c0.opus b/dom/media/test/invalid-m1c0.opus
new file mode 100644
index 0000000000..c7728f79ee
--- /dev/null
+++ b/dom/media/test/invalid-m1c0.opus
Binary files differ
diff --git a/dom/media/test/invalid-m1c0.opus^headers^ b/dom/media/test/invalid-m1c0.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/invalid-m1c0.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/invalid-m1c9.opus b/dom/media/test/invalid-m1c9.opus
new file mode 100644
index 0000000000..1ef6e9f9cf
--- /dev/null
+++ b/dom/media/test/invalid-m1c9.opus
Binary files differ
diff --git a/dom/media/test/invalid-m1c9.opus^headers^ b/dom/media/test/invalid-m1c9.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/invalid-m1c9.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/invalid-m2c0.opus b/dom/media/test/invalid-m2c0.opus
new file mode 100644
index 0000000000..5c3f97e2ab
--- /dev/null
+++ b/dom/media/test/invalid-m2c0.opus
Binary files differ
diff --git a/dom/media/test/invalid-m2c0.opus^headers^ b/dom/media/test/invalid-m2c0.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/invalid-m2c0.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/invalid-m2c1.opus b/dom/media/test/invalid-m2c1.opus
new file mode 100644
index 0000000000..5ecb95ee25
--- /dev/null
+++ b/dom/media/test/invalid-m2c1.opus
Binary files differ
diff --git a/dom/media/test/invalid-m2c1.opus^headers^ b/dom/media/test/invalid-m2c1.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/invalid-m2c1.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/invalid-neg_discard.webm b/dom/media/test/invalid-neg_discard.webm
new file mode 100644
index 0000000000..3f665c0b59
--- /dev/null
+++ b/dom/media/test/invalid-neg_discard.webm
Binary files differ
diff --git a/dom/media/test/invalid-neg_discard.webm^headers^ b/dom/media/test/invalid-neg_discard.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/invalid-neg_discard.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/invalid-preskip.webm b/dom/media/test/invalid-preskip.webm
new file mode 100644
index 0000000000..99b4f2ca71
--- /dev/null
+++ b/dom/media/test/invalid-preskip.webm
Binary files differ
diff --git a/dom/media/test/invalid-preskip.webm^headers^ b/dom/media/test/invalid-preskip.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/invalid-preskip.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/make-headers.sh b/dom/media/test/make-headers.sh
new file mode 100644
index 0000000000..35d9bd90f8
--- /dev/null
+++ b/dom/media/test/make-headers.sh
@@ -0,0 +1,18 @@
+# This Source Code Form is subject to the terms of the Mozilla Public
+# License, v. 2.0. If a copy of the MPL was not distributed with this
+# file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
+# Script to generate ^header^ files for all media files we use.
+# This is to ensure that our media files are not cached by necko,
+# so that our detection as to whether the server supports byte range
+# requests is not interferred with by Necko's cache. See bug 977398
+# for details. Necko will fix this in bug 977314.
+
+FILES=(`ls *.ogg *.ogv *.webm *.mp3 *.opus *.mp4 *.m4s *.wav`)
+
+rm -f *.ogg^headers^ *.ogv^headers^ *.webm^headers^ *.mp3^headers^ *.opus^headers^ *.mp4^headers^ *.m4s^headers^ *.wav^headers^
+
+for i in "${FILES[@]}"
+do
+ echo "Cache-Control: no-store" >> $i^headers^
+done
diff --git a/dom/media/test/manifest.js b/dom/media/test/manifest.js
new file mode 100644
index 0000000000..bb4557d498
--- /dev/null
+++ b/dom/media/test/manifest.js
@@ -0,0 +1,2556 @@
+const { AppConstants } = SpecialPowers.ChromeUtils.importESModule(
+ "resource://gre/modules/AppConstants.sys.mjs"
+);
+
+// In each list of tests below, test file types that are not supported should
+// be ignored. To make sure tests respect that, we include a file of type
+// "bogus/duh" in each list.
+
+// Make sure to not touch navigator in here, since we want to push prefs that
+// will affect the APIs it exposes, but the set of exposed APIs is determined
+// when Navigator.prototype is created. So if we touch navigator before pushing
+// the prefs, the APIs it exposes will not take those prefs into account. We
+// work around this by using a navigator object from a different global for our
+// UA string testing.
+var gManifestNavigatorSource = document.documentElement.appendChild(
+ document.createElement("iframe")
+);
+gManifestNavigatorSource.style.display = "none";
+function manifestNavigator() {
+ return gManifestNavigatorSource.contentWindow.navigator;
+}
+
+// Similarly, use a <video> element from a different global for canPlayType or
+// other feature testing. If we used one from our global and did so before our
+// prefs are pushed, then we'd instantiate HTMLMediaElement.prototype before the
+// prefs are pushed and APIs we expect to be on that object would not be there.
+function manifestVideo() {
+ return gManifestNavigatorSource.contentDocument.createElement("video");
+}
+
+// Need to get the server url composed with ip:port instead of mochi.test.
+// Since we will provide the url to Exoplayer which cannot recognize the domain
+// name "mochi.test".
+let serverUrl = SpecialPowers.Services.prefs.getCharPref(
+ "media.hls.server.url"
+);
+var gHLSTests = [
+ {
+ name: serverUrl + "/bipbop_4x3_variant.m3u8",
+ type: "audio/x-mpegurl",
+ duration: 20.0,
+ },
+];
+
+// These are small test files, good for just seeing if something loads. We
+// really only need one test file per backend here.
+var gSmallTests = [
+ { name: "small-shot.ogg", type: "audio/ogg", duration: 0.276 },
+ { name: "small-shot.m4a", type: "audio/mp4", duration: 0.29 },
+ { name: "small-shot.mp3", type: "audio/mpeg", duration: 0.27 },
+ { name: "small-shot-mp3.mp4", type: "audio/mp4; codecs=mp3", duration: 0.34 },
+ { name: "small-shot.flac", type: "audio/flac", duration: 0.197 },
+ { name: "r11025_s16_c1-short.wav", type: "audio/x-wav", duration: 0.37 },
+ {
+ name: "320x240.ogv",
+ type: "video/ogg",
+ width: 320,
+ height: 240,
+ duration: 0.266,
+ contentDuration: 0.133,
+ },
+ {
+ name: "seek-short.webm",
+ type: "video/webm",
+ width: 320,
+ height: 240,
+ duration: 0.23,
+ },
+ {
+ name: "vp9-short.webm",
+ type: "video/webm",
+ width: 320,
+ height: 240,
+ duration: 0.2,
+ },
+ {
+ name: "detodos-short.opus",
+ type: "audio/ogg; codecs=opus",
+ duration: 0.22,
+ },
+ {
+ name: "gizmo-short.mp4",
+ type: "video/mp4",
+ width: 560,
+ height: 320,
+ duration: 0.27,
+ },
+ { name: "flac-s24.flac", type: "audio/flac", duration: 4.04 },
+ { name: "bogus.duh", type: "bogus/duh" },
+];
+
+var gFrameCountTests = [
+ { name: "bipbop.mp4", type: "video/mp4", totalFrameCount: 297 },
+ { name: "gizmo.mp4", type: "video/mp4", totalFrameCount: 166 },
+ { name: "seek-short.webm", type: "video/webm", totalFrameCount: 8 },
+ { name: "seek.webm", type: "video/webm", totalFrameCount: 120 },
+ { name: "320x240.ogv", type: "video/ogg", totalFrameCount: 8 },
+ { name: "av1.mp4", type: "video/mp4", totalFrameCount: 24 },
+];
+
+gSmallTests = gSmallTests.concat([
+ { name: "sample.3gp", type: "video/3gpp", duration: 4.933 },
+ { name: "sample.3g2", type: "video/3gpp2", duration: 4.933 },
+]);
+
+// Used by test_bug654550.html, for videoStats preference
+var gVideoTests = [
+ {
+ name: "320x240.ogv",
+ type: "video/ogg",
+ width: 320,
+ height: 240,
+ duration: 0.266,
+ },
+ {
+ name: "seek-short.webm",
+ type: "video/webm",
+ width: 320,
+ height: 240,
+ duration: 0.23,
+ },
+ { name: "bogus.duh", type: "bogus/duh" },
+];
+
+// Temp hack for trackIDs and captureStream() -- bug 1215769
+var gLongerTests = [
+ {
+ name: "seek.webm",
+ type: "video/webm",
+ width: 320,
+ height: 240,
+ duration: 3.966,
+ },
+ {
+ name: "gizmo.mp4",
+ type: "video/mp4",
+ width: 560,
+ height: 320,
+ duration: 5.56,
+ },
+];
+
+// Used by test_progress to ensure we get the correct progress information
+// during resource download.
+var gProgressTests = [
+ { name: "r11025_u8_c1.wav", type: "audio/x-wav", duration: 1.0, size: 11069 },
+ { name: "big-short.wav", type: "audio/x-wav", duration: 1.11, size: 12366 },
+ { name: "seek-short.ogv", type: "video/ogg", duration: 1.03, size: 79921 },
+ {
+ name: "320x240.ogv",
+ type: "video/ogg",
+ width: 320,
+ height: 240,
+ duration: 0.266,
+ size: 28942,
+ },
+ { name: "seek-short.webm", type: "video/webm", duration: 0.23, size: 19267 },
+ { name: "gizmo-short.mp4", type: "video/mp4", duration: 0.27, size: 29905 },
+ { name: "bogus.duh", type: "bogus/duh" },
+];
+
+// Used by test_played.html
+var gPlayedTests = [
+ { name: "big-short.wav", type: "audio/x-wav", duration: 1.11 },
+ { name: "seek-short.ogv", type: "video/ogg", duration: 1.03 },
+ { name: "seek-short.webm", type: "video/webm", duration: 0.23 },
+ { name: "gizmo-short.mp4", type: "video/mp4", duration: 0.27 },
+ { name: "owl-short.mp3", type: "audio/mpeg", duration: 0.52 },
+ { name: "very-short.mp3", type: "audio/mpeg", duration: 0.07 },
+ // Disable vbr.mp3 to see if it reduces the error of AUDCLNT_E_CPUUSAGE_EXCEEDED.
+ // See bug 1110922 comment 26.
+ //{ name:"vbr.mp3", type:"audio/mpeg", duration:10.0 },
+ { name: "bug495794.ogg", type: "audio/ogg", duration: 0.3 },
+];
+
+if (
+ manifestNavigator().userAgent.includes("Windows") &&
+ manifestVideo().canPlayType('video/mp4; codecs="avc1.64000c"')
+) {
+ gPlayedTests = gPlayedTests.concat(
+ { name: "red-46x48.mp4", type: "video/mp4", duration: 1.0 },
+ { name: "red-48x46.mp4", type: "video/mp4", duration: 1.0 }
+ );
+}
+
+// Used by test_mozLoadFrom. Need one test file per decoder backend, plus
+// anything for testing clone-specific bugs.
+var cloneKey = Math.floor(Math.random() * 100000000);
+var gCloneTests = [
+ // short-video is more like 1s, so if you load this twice you'll get an unexpected duration
+ {
+ name:
+ "dynamic_resource.sjs?key=" +
+ cloneKey +
+ "&res1=320x240.ogv&res2=short-video.ogv",
+ type: "video/ogg",
+ duration: 0.266,
+ },
+];
+
+// Used by test_play_twice. Need one test file per decoder backend, plus
+// anything for testing bugs that occur when replying a played file.
+var gReplayTests = gSmallTests.concat([
+ { name: "bug533822.ogg", type: "audio/ogg" },
+]);
+
+// Used by test_paused_after_ended. Need one test file per decoder backend, plus
+// anything for testing bugs that occur when replying a played file.
+var gPausedAfterEndedTests = gSmallTests.concat([
+ { name: "r11025_u8_c1.wav", type: "audio/x-wav", duration: 1.0 },
+ { name: "small-shot.ogg", type: "video/ogg", duration: 0.276 },
+]);
+
+// Test the mozHasAudio property, and APIs that detect different kinds of
+// tracks
+var gTrackTests = [
+ {
+ name: "big-short.wav",
+ type: "audio/x-wav",
+ duration: 1.11,
+ size: 12366,
+ hasAudio: true,
+ hasVideo: false,
+ },
+ {
+ name: "320x240.ogv",
+ type: "video/ogg",
+ width: 320,
+ height: 240,
+ duration: 0.266,
+ size: 28942,
+ hasAudio: false,
+ hasVideo: true,
+ },
+ {
+ name: "short-video.ogv",
+ type: "video/ogg",
+ duration: 1.081,
+ hasAudio: true,
+ hasVideo: true,
+ },
+ {
+ name: "seek-short.webm",
+ type: "video/webm",
+ duration: 0.23,
+ size: 19267,
+ hasAudio: false,
+ hasVideo: true,
+ },
+ {
+ name: "flac-s24.flac",
+ type: "audio/flac",
+ duration: 4.04,
+ hasAudio: true,
+ hasVideo: false,
+ },
+ { name: "bogus.duh", type: "bogus/duh" },
+];
+
+var gClosingConnectionsTest = [
+ { name: "seek-short.ogv", type: "video/ogg", duration: 1.03 },
+];
+
+// Used by any media recorder test. Need one test file per decoder backend
+// currently supported by the media encoder.
+var gMediaRecorderTests = [
+ // Duration should be greater than 500ms because we will record 2
+ // time slices (250ms per slice)
+ {
+ name: "detodos-recorder-test.opus",
+ type: "audio/ogg; codecs=opus",
+ duration: 0.62,
+ },
+];
+
+// Used by video media recorder tests
+var gMediaRecorderVideoTests = [
+ {
+ name: "seek-short.webm",
+ type: "video/webm",
+ width: 320,
+ height: 240,
+ duration: 0.23,
+ },
+];
+
+// These are files that we want to make sure we can play through. We can
+// also check metadata. Put files of the same type together in this list so if
+// something crashes we have some idea of which backend is responsible.
+// Used by test_playback, which expects no error event and one ended event.
+var gPlayTests = [
+ // Test playback of a WebM file with vp9 video
+ { name: "vp9cake-short.webm", type: "video/webm", duration: 1.0 },
+ // 8-bit samples
+ { name: "r11025_u8_c1.wav", type: "audio/x-wav", duration: 1.0 },
+ // 8-bit samples, file is truncated
+ { name: "r11025_u8_c1_trunc.wav", type: "audio/x-wav", duration: 1.8 },
+ // file has trailing non-PCM data
+ { name: "r11025_s16_c1_trailing.wav", type: "audio/x-wav", duration: 1.0 },
+ // file with list chunk
+ { name: "r16000_u8_c1_list.wav", type: "audio/x-wav", duration: 4.2 },
+ // file with 2 extra bytes of metadata
+ {
+ name: "16bit_wave_extrametadata.wav",
+ type: "audio/x-wav",
+ duration: 1.108,
+ },
+ // IEEE float wave file
+ { name: "wavedata_float.wav", type: "audio/x-wav", duration: 1.0 },
+ // 24-bit samples
+ { name: "wavedata_s24.wav", type: "audio/x-wav", duration: 1.0 },
+ // aLaw compressed wave file
+ { name: "wavedata_alaw.wav", type: "audio/x-wav", duration: 1.0 },
+ // uLaw compressed wave file
+ { name: "wavedata_ulaw.wav", type: "audio/x-wav", duration: 1.0 },
+ // Data length 0xFFFFFFFF
+ { name: "bug1301226.wav", type: "audio/x-wav", duration: 0.003673 },
+ // Data length 0xFFFFFFFF and odd chunk lengths.
+ { name: "bug1301226-odd.wav", type: "audio/x-wav", duration: 0.003673 },
+
+ // Ogg stream without eof marker
+ { name: "bug461281.ogg", type: "application/ogg", duration: 2.208 },
+
+ // oggz-chop stream
+ { name: "bug482461.ogv", type: "video/ogg", duration: 4.34 },
+ // Theora only oggz-chop stream
+ { name: "bug482461-theora.ogv", type: "video/ogg", duration: 4.138 },
+ // With first frame a "duplicate" (empty) frame.
+ {
+ name: "bug500311.ogv",
+ type: "video/ogg",
+ duration: 1.96,
+ contentDuration: 1.958,
+ },
+ // Small audio file
+ { name: "small-shot.ogg", type: "audio/ogg", duration: 0.276 },
+ // More audio in file than video.
+ { name: "short-video.ogv", type: "video/ogg", duration: 1.081 },
+ // First Theora data packet is zero bytes.
+ { name: "bug504613.ogv", type: "video/ogg", duration: Number.NaN },
+ // Multiple audio streams.
+ { name: "bug516323.ogv", type: "video/ogg", duration: 4.208 },
+ // oggz-chop with non-keyframe as first frame
+ {
+ name: "bug556821.ogv",
+ type: "video/ogg",
+ duration: 2.936,
+ contentDuration: 2.903,
+ },
+
+ // Encoded with vorbis beta1, includes unusually sized codebooks
+ { name: "beta-phrasebook.ogg", type: "audio/ogg", duration: 4.01 },
+ // Small file, only 1 frame with audio only.
+ { name: "bug520493.ogg", type: "audio/ogg", duration: 0.458 },
+ // Small file with vorbis comments with 0 length values and names.
+ { name: "bug520500.ogg", type: "audio/ogg", duration: 0.123 },
+
+ // Various weirdly formed Ogg files
+ {
+ name: "bug499519.ogv",
+ type: "video/ogg",
+ duration: 0.24,
+ contentDuration: 0.22,
+ },
+ { name: "bug506094.ogv", type: "video/ogg", duration: 0 },
+ { name: "bug498855-1.ogv", type: "video/ogg", duration: 0.24 },
+ { name: "bug498855-2.ogv", type: "video/ogg", duration: 0.24 },
+ { name: "bug498855-3.ogv", type: "video/ogg", duration: 0.24 },
+ {
+ name: "bug504644.ogv",
+ type: "video/ogg",
+ duration: 1.6,
+ contentDuration: 1.52,
+ },
+ {
+ name: "chain.ogv",
+ type: "video/ogg",
+ duration: Number.NaN,
+ contentDuration: 0.266,
+ },
+ {
+ name: "bug523816.ogv",
+ type: "video/ogg",
+ duration: 0.766,
+ contentDuration: 0,
+ },
+ { name: "bug495129.ogv", type: "video/ogg", duration: 2.41 },
+ {
+ name: "bug498380.ogv",
+ type: "video/ogg",
+ duration: 0.7663,
+ contentDuration: 0,
+ },
+ { name: "bug495794.ogg", type: "audio/ogg", duration: 0.3 },
+ { name: "bug557094.ogv", type: "video/ogg", duration: 0.24 },
+ { name: "multiple-bos.ogg", type: "video/ogg", duration: 0.431 },
+ { name: "audio-overhang.ogg", type: "video/ogg", duration: 2.3 },
+ { name: "video-overhang.ogg", type: "video/ogg", duration: 3.966 },
+
+ // bug461281.ogg with the middle second chopped out.
+ { name: "audio-gaps.ogg", type: "audio/ogg", duration: 2.208 },
+
+ // Test playback/metadata work after a redirect
+ {
+ name: "redirect.sjs?domain=mochi.test:8888&file=320x240.ogv",
+ type: "video/ogg",
+ duration: 0.266,
+ },
+
+ // Test playback of a webm file
+ { name: "seek-short.webm", type: "video/webm", duration: 0.23 },
+
+ // Test playback of a webm file with 'matroska' doctype
+ { name: "bug1377278.webm", type: "video/webm", duration: 4.0 },
+
+ // Test playback of a WebM file with non-zero start time.
+ { name: "split.webm", type: "video/webm", duration: 1.967 },
+
+ // Test playback of a WebM file with resolution changes.
+ { name: "resolution-change.webm", type: "video/webm", duration: 6.533 },
+
+ // The following webm files test cases where the webm metadata dimensions do
+ // not match those in the stream. See bug 1695033 for more info.
+
+ // Reference file with correct dimensions (webm metadata matches stream
+ // resolution).
+ { name: "bipbop_short_vp8.webm", type: "video/webm", duration: 1.011 },
+
+ // The webm resolution is greater in both dimensions than the in stream
+ // resolution.
+ {
+ name: "bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm",
+ type: "video/webm",
+ duration: 1.011,
+ },
+
+ // The webm resolution is correct for height, but is narrower than the stream
+ // resolution.
+ {
+ name: "bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm",
+ type: "video/webm",
+ duration: 1.011,
+ },
+
+ // The webm resolution is smaller in both dimensions than the in stream
+ // resolution.
+ {
+ name: "bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm",
+ type: "video/webm",
+ duration: 1.011,
+ },
+
+ // End of webm dimension clashing files.
+
+ // A really short, low sample rate, single channel file. This tests whether
+ // we can handle playing files when only push very little audio data to the
+ // hardware.
+ { name: "spacestorm-1000Hz-100ms.ogg", type: "audio/ogg", duration: 0.099 },
+
+ // Opus data in an ogg container
+ {
+ name: "detodos-short.opus",
+ type: "audio/ogg; codecs=opus",
+ duration: 0.22,
+ contentDuration: 0.2135,
+ },
+ // Opus data in a webm container
+ {
+ name: "detodos-short.webm",
+ type: "audio/webm; codecs=opus",
+ duration: 0.26,
+ contentDuration: 0.2535,
+ },
+ // Opus in webm channel mapping=2 sample file
+ {
+ name: "opus-mapping2.webm",
+ type: "audio/webm; codecs=opus",
+ duration: 10.01,
+ contentDuration: 9.99,
+ },
+ { name: "bug1066943.webm", type: "audio/webm; codecs=opus", duration: 1.383 },
+
+ // Multichannel Opus in an ogg container
+ { name: "test-1-mono.opus", type: "audio/ogg; codecs=opus", duration: 1.044 },
+ {
+ name: "test-2-stereo.opus",
+ type: "audio/ogg; codecs=opus",
+ duration: 2.925,
+ },
+ { name: "test-3-LCR.opus", type: "audio/ogg; codecs=opus", duration: 4.214 },
+ { name: "test-4-quad.opus", type: "audio/ogg; codecs=opus", duration: 6.234 },
+ { name: "test-5-5.0.opus", type: "audio/ogg; codecs=opus", duration: 7.558 },
+ { name: "test-6-5.1.opus", type: "audio/ogg; codecs=opus", duration: 10.333 },
+ { name: "test-7-6.1.opus", type: "audio/ogg; codecs=opus", duration: 11.69 },
+ { name: "test-8-7.1.opus", type: "audio/ogg; codecs=opus", duration: 13.478 },
+
+ {
+ name: "gizmo-short.mp4",
+ type: "video/mp4",
+ duration: 0.27,
+ contentDuration: 0.267,
+ },
+ // Test playback of a MP4 file with a non-zero start time (and audio starting
+ // a second later).
+ { name: "bipbop-lateaudio.mp4", type: "video/mp4" },
+ // Opus in MP4 channel mapping=0 sample file (content shorter due to preskip)
+ {
+ name: "opus-sample.mp4",
+ type: "audio/mp4; codecs=opus",
+ duration: 10.92,
+ contentDuration: 10.09,
+ },
+ // Opus in MP4 channel mapping=2 sample file
+ { name: "opus-mapping2.mp4", type: "audio/mp4; codecs=opus", duration: 10.0 },
+
+ { name: "small-shot.m4a", type: "audio/mp4", duration: 0.29 },
+ { name: "small-shot.mp3", type: "audio/mpeg", duration: 0.27 },
+ { name: "owl.mp3", type: "audio/mpeg", duration: 3.343 },
+ // owl.mp3 as above, but with something funny going on in the ID3v2 tag
+ // that caused DirectShow to fail.
+ { name: "owl-funny-id3.mp3", type: "audio/mpeg", duration: 3.343 },
+ // owl.mp3 as above, but with something even funnier going on in the ID3v2 tag
+ // that caused DirectShow to fail.
+ { name: "owl-funnier-id3.mp3", type: "audio/mpeg", duration: 3.343 },
+ // One second of silence with ~140KB of ID3 tags. Usually when the first MP3
+ // frame is at such a high offset into the file, MP3FrameParser will give up
+ // and report that the stream is not MP3. However, it does not count ID3 tags
+ // in that offset. This test case makes sure that ID3 exclusion holds.
+ { name: "huge-id3.mp3", type: "audio/mpeg", duration: 1.0 },
+ // Half a second file of a sine with a large ID3v2 tag, followed by an ID3v1
+ // tag. The ID3v1 tags should be at the end of the file, but software usually
+ // play it anyway.
+ { name: "id3v1afterlongid3v2.mp3", type: "audio/mpeg", duration: 0.5 },
+ // An VBR file with a padding value that is greater than an mp3 packet, and
+ // also subsequent packets after the theoretical EOF computed from metadata,
+ // to test padding trimming edge cases.
+ {
+ name: "padding-spanning-multiple-packets.mp3",
+ type: "audio/mpeg",
+ },
+ // A truncated VBR MP3 with just enough frames to keep most decoders happy.
+ // The Xing header reports the length of the file to be around 10 seconds, but
+ // there is really only one second worth of data. We want MP3FrameParser to
+ // trust the header, so this should be reported as 10 seconds.
+ {
+ name: "vbr-head.mp3",
+ type: "audio/mpeg",
+ duration: 10.0,
+ contentDuration: 1.019,
+ },
+
+ // A flac file where the STREAMINFO block was removed.
+ // It is necessary to parse the file to find an audio frame instead.
+ { name: "flac-noheader-s16.flac", type: "audio/flac", duration: 4.0 },
+ { name: "flac-s24.flac", type: "audio/flac", duration: 4.04 },
+ {
+ name: "flac-sample.mp4",
+ type: "audio/mp4; codecs=flac",
+ duration: 4.95,
+ contentDuration: 5.03,
+ },
+ // Ogg with theora video and flac audio.
+ {
+ name: "A4.ogv",
+ type: "video/ogg",
+ width: 320,
+ height: 240,
+ duration: 3.13,
+ },
+ // A file that has no codec delay at the container level, but has a delay at
+ // the codec level.
+ {
+ name: "no-container-codec-delay.webm",
+ type: "video/webm",
+ },
+ // A file that has a codec delay at a container level of 0, but as a delay at
+ // the codec level that is non-zero.
+ { name: "invalid-preskip.webm", type: "audio/webm; codecs=opus" },
+
+ // Invalid file
+ { name: "bogus.duh", type: "bogus/duh", duration: Number.NaN },
+];
+
+const win32 =
+ SpecialPowers.Services.appinfo.OS == "WINNT" &&
+ !SpecialPowers.Services.appinfo.is64Bit;
+if (!win32) {
+ gPlayTests.push({ name: "av1.mp4", type: "video/mp4", duration: 1.0 });
+}
+
+// AAC files with different sample rates. We add these here as some are added
+// conditionally.
+gPlayTests.push(
+ {
+ name: "bipbop_audio_aac_8k.mp4",
+ type: "audio/mp4",
+ duration: 1.06,
+ },
+ {
+ name: "bipbop_audio_aac_22.05k.mp4",
+ type: "audio/mp4",
+ duration: 1.06,
+ },
+ {
+ name: "bipbop_audio_aac_44.1k.mp4",
+ type: "audio/mp4",
+ duration: 1.06,
+ },
+ {
+ name: "bipbop_audio_aac_48k.mp4",
+ type: "audio/mp4",
+ duration: 1.06,
+ }
+);
+if (AppConstants.platform != "win") {
+ // Windows WMF decoder doesn't do >48K everywhere. See bug 1698639.
+ gPlayTests.push(
+ {
+ name: "bipbop_audio_aac_88.2k.mp4",
+ type: "audio/mp4",
+ duration: 1.06,
+ },
+ {
+ name: "bipbop_audio_aac_96k.mp4",
+ type: "audio/mp4",
+ duration: 1.06,
+ }
+ );
+}
+
+// ambisonics.mp4 causes intermittents, so we conditionally add it until we fix
+// the root cause.
+const skipAmbisonics =
+ // Bug 1484451 - skip on mac debug
+ (AppConstants.platform == "macosx" && AppConstants.DEBUG) ||
+ // Bug 1483259 - skip on linux64 opt
+ (AppConstants.platform == "linux" &&
+ !AppConstants.DEBUG &&
+ SpecialPowers.Services.appinfo.is64Bit);
+if (!skipAmbisonics) {
+ // Ambisonics AAC, requires AAC extradata to be set when creating decoder (see bug 1431169)
+ // Also test 4.0 decoding.
+ gPlayTests.push({
+ name: "ambisonics.mp4",
+ type: "audio/mp4",
+ duration: 16.48,
+ });
+}
+
+var gSeekToNextFrameTests = [
+ // Test playback of a WebM file with vp9 video
+ { name: "vp9-short.webm", type: "video/webm", duration: 0.2 },
+ { name: "vp9cake-short.webm", type: "video/webm", duration: 1.0 },
+ // oggz-chop stream
+ { name: "bug482461.ogv", type: "video/ogg", duration: 4.34 },
+ // Theora only oggz-chop stream
+ { name: "bug482461-theora.ogv", type: "video/ogg", duration: 4.138 },
+ // With first frame a "duplicate" (empty) frame.
+ { name: "bug500311.ogv", type: "video/ogg", duration: 1.96 },
+
+ // More audio in file than video.
+ { name: "short-video.ogv", type: "video/ogg", duration: 1.081 },
+ // First Theora data packet is zero bytes.
+ { name: "bug504613.ogv", type: "video/ogg", duration: Number.NaN },
+ // Multiple audio streams.
+ { name: "bug516323.ogv", type: "video/ogg", duration: 4.208 },
+ // oggz-chop with non-keyframe as first frame
+ { name: "bug556821.ogv", type: "video/ogg", duration: 2.936 },
+ // Various weirdly formed Ogg files
+ { name: "bug498855-1.ogv", type: "video/ogg", duration: 0.24 },
+ { name: "bug498855-2.ogv", type: "video/ogg", duration: 0.24 },
+ { name: "bug498855-3.ogv", type: "video/ogg", duration: 0.24 },
+ { name: "bug504644.ogv", type: "video/ogg", duration: 1.6 },
+
+ { name: "bug523816.ogv", type: "video/ogg", duration: 0.766 },
+
+ { name: "bug498380.ogv", type: "video/ogg", duration: 0.2 },
+ { name: "bug557094.ogv", type: "video/ogg", duration: 0.24 },
+ { name: "multiple-bos.ogg", type: "video/ogg", duration: 0.431 },
+ // Test playback/metadata work after a redirect
+ {
+ name: "redirect.sjs?domain=mochi.test:8888&file=320x240.ogv",
+ type: "video/ogg",
+ duration: 0.266,
+ },
+ // Test playback of a webm file
+ { name: "seek-short.webm", type: "video/webm", duration: 0.23 },
+ // Test playback of a WebM file with non-zero start time.
+ { name: "split.webm", type: "video/webm", duration: 1.967 },
+
+ { name: "gizmo-short.mp4", type: "video/mp4", duration: 0.27 },
+
+ // Test playback of a MP4 file with a non-zero start time (and audio starting
+ // a second later).
+ { name: "bipbop-lateaudio.mp4", type: "video/mp4" },
+];
+
+// A file for each type we can support.
+var gSnifferTests = [
+ { name: "big.wav", type: "audio/x-wav", duration: 9.278982, size: 102444 },
+ {
+ name: "320x240.ogv",
+ type: "video/ogg",
+ width: 320,
+ height: 240,
+ duration: 0.233,
+ size: 28942,
+ },
+ { name: "seek.webm", type: "video/webm", duration: 3.966, size: 215529 },
+ { name: "gizmo.mp4", type: "video/mp4", duration: 5.56, size: 383631 },
+ // A mp3 file with id3 tags.
+ { name: "id3tags.mp3", type: "audio/mpeg", duration: 0.28, size: 3530 },
+ { name: "bogus.duh", type: "bogus/duh" },
+];
+
+// Files that contain resolution changes
+var gResolutionChangeTests = [
+ { name: "resolution-change.webm", type: "video/webm", duration: 6.533 },
+];
+
+// Files we must reject as invalid.
+var gInvalidTests = [
+ { name: "invalid-m0c0.opus", type: "audio/ogg; codecs=opus" },
+ { name: "invalid-m0c3.opus", type: "audio/ogg; codecs=opus" },
+ { name: "invalid-m1c0.opus", type: "audio/ogg; codecs=opus" },
+ { name: "invalid-m1c9.opus", type: "audio/ogg; codecs=opus" },
+ { name: "invalid-m2c0.opus", type: "audio/ogg; codecs=opus" },
+ { name: "invalid-m2c1.opus", type: "audio/ogg; codecs=opus" },
+ { name: "invalid-cmap-short.opus", type: "audio/ogg; codecs=opus" },
+ { name: "invalid-cmap-s0c0.opus", type: "audio/ogg; codecs=opus" },
+ { name: "invalid-cmap-s0c2.opus", type: "audio/ogg; codecs=opus" },
+ { name: "invalid-cmap-s1c2.opus", type: "audio/ogg; codecs=opus" },
+];
+
+var gInvalidPlayTests = [
+ { name: "invalid-excess_discard.webm", type: "audio/webm; codecs=opus" },
+ { name: "invalid-excess_neg_discard.webm", type: "audio/webm; codecs=opus" },
+ { name: "invalid-neg_discard.webm", type: "audio/webm; codecs=opus" },
+ {
+ name: "invalid-discard_on_multi_blocks.webm",
+ type: "audio/webm; codecs=opus",
+ },
+];
+
+// Files to check different cases of ogg skeleton information.
+// sample-fisbone-skeleton4.ogv
+// - Skeleton v4, w/ Content-Type,Role,Name,Language,Title for both theora/vorbis
+// sample-fisbone-wrong-header.ogv
+// - Skeleton v4, wrong message field sequence for vorbis
+// multiple-bos-more-header-fields.ogg
+// - Skeleton v3, w/ Content-Type,Role,Name,Language,Title for both theora/vorbis
+// seek-short.ogv
+// - No skeleton, but theora
+// audio-gaps-short.ogg
+// - No skeleton, but vorbis
+var gMultitrackInfoOggPlayList = [
+ { name: "sample-fisbone-skeleton4.ogv", type: "video/ogg", duration: 1.0 },
+ { name: "sample-fisbone-wrong-header.ogv", type: "video/ogg", duration: 1.0 },
+ {
+ name: "multiple-bos-more-header-fileds.ogg",
+ type: "video/ogg",
+ duration: 0.431,
+ },
+ { name: "seek-short.ogv", type: "video/ogg", duration: 1.03 },
+ { name: "audio-gaps-short.ogg", type: "audio/ogg", duration: 0.5 },
+];
+// Pre-parsed results of gMultitrackInfoOggPlayList.
+var gOggTrackInfoResults = {
+ "sample-fisbone-skeleton4.ogv": {
+ audio_id: " audio_1",
+ audio_kind: "main",
+ audio_language: " en-US",
+ audio_label: " Audio track for test",
+ video_id: " video_1",
+ video_kind: "main",
+ video_language: " fr",
+ video_label: " Video track for test",
+ },
+ "sample-fisbone-wrong-header.ogv": {
+ audio_id: "1",
+ audio_kind: "main",
+ audio_language: "",
+ audio_label: "",
+ video_id: " video_1",
+ video_kind: "main",
+ video_language: " fr",
+ video_label: " Video track for test",
+ },
+ "multiple-bos-more-header-fileds.ogg": {
+ audio_id: "1",
+ audio_kind: "main",
+ audio_language: "",
+ audio_label: "",
+ video_id: "2",
+ video_kind: "main",
+ video_language: "",
+ video_label: "",
+ },
+ "seek-short.ogv": {
+ video_id: "2",
+ video_kind: "main",
+ video_language: "",
+ video_label: "",
+ },
+ "audio-gaps-short.ogg": {
+ audio_id: "1",
+ audio_kind: "main",
+ audio_language: "",
+ audio_label: "",
+ },
+};
+
+// Returns a promise that resolves to a function that converts
+// relative paths to absolute, to test loading files from file: URIs.
+// Optionally checks whether the file actually exists on disk at the location
+// we've specified.
+function makeAbsolutePathConverter() {
+ const url = SimpleTest.getTestFileURL("chromeHelper.js");
+ const script = SpecialPowers.loadChromeScript(url);
+ return new Promise((resolve, reject) => {
+ script.addMessageListener("media-test:cwd", cwd => {
+ if (!cwd) {
+ ok(false, "Failed to find path to test files");
+ }
+
+ resolve((path, mustExist) => {
+ // android mochitest doesn't support file://
+ if (manifestNavigator().appVersion.includes("Android")) {
+ return path;
+ }
+
+ const { Ci, Cc } = SpecialPowers;
+ var f = Cc["@mozilla.org/file/local;1"].createInstance(Ci.nsIFile);
+ f.initWithPath(cwd);
+ var split = path.split("/");
+ for (var i = 0; i < split.length; ++i) {
+ f.append(split[i]);
+ }
+ if (mustExist && !f.exists()) {
+ ok(false, "We expected '" + path + "' to exist, but it doesn't!");
+ }
+ return f.path;
+ });
+ });
+ script.sendAsyncMessage("media-test:getcwd");
+ });
+}
+
+// Returns true if two TimeRanges are equal, false otherwise
+function range_equals(r1, r2) {
+ if (r1.length != r2.length) {
+ return false;
+ }
+ for (var i = 0; i < r1.length; i++) {
+ if (r1.start(i) != r2.start(i) || r1.end(i) != r2.end(i)) {
+ return false;
+ }
+ }
+ return true;
+}
+
+// These are URIs to files that we use to check that we don't leak any state
+// or other information such that script can determine stuff about a user's
+// environment. Used by test_info_leak.
+function makeInfoLeakTests() {
+ return makeAbsolutePathConverter().then(fileUriToSrc => [
+ {
+ type: "video/ogg",
+ src: fileUriToSrc("tests/dom/media/test/320x240.ogv", true),
+ },
+ {
+ type: "video/ogg",
+ src: fileUriToSrc("tests/dom/media/test/404.ogv", false),
+ },
+ {
+ type: "audio/x-wav",
+ src: fileUriToSrc("tests/dom/media/test/r11025_s16_c1.wav", true),
+ },
+ {
+ type: "audio/x-wav",
+ src: fileUriToSrc("tests/dom/media/test/404.wav", false),
+ },
+ {
+ type: "audio/ogg",
+ src: fileUriToSrc("tests/dom/media/test/bug461281.ogg", true),
+ },
+ {
+ type: "audio/ogg",
+ src: fileUriToSrc("tests/dom/media/test/404.ogg", false),
+ },
+ {
+ type: "video/webm",
+ src: fileUriToSrc("tests/dom/media/test/seek.webm", true),
+ },
+ {
+ type: "video/webm",
+ src: fileUriToSrc("tests/dom/media/test/404.webm", false),
+ },
+ {
+ type: "video/ogg",
+ src: "http://localhost/404.ogv",
+ },
+ {
+ type: "audio/x-wav",
+ src: "http://localhost/404.wav",
+ },
+ {
+ type: "video/webm",
+ src: "http://localhost/404.webm",
+ },
+ {
+ type: "video/ogg",
+ src: "http://example.com/tests/dom/media/test/test_info_leak.html",
+ },
+ {
+ type: "audio/ogg",
+ src: "http://example.com/tests/dom/media/test/test_info_leak.html",
+ },
+ ]);
+}
+
+// These are files that must fire an error during load or playback, and do not
+// cause a crash. Used by test_playback_errors, which expects one error event
+// and no ended event. Put files of the same type together in this list so if
+// something crashes we have some idea of which backend is responsible.
+var gErrorTests = [
+ { name: "bogus.wav", type: "audio/x-wav" },
+ { name: "bogus.ogv", type: "video/ogg" },
+ { name: "448636.ogv", type: "video/ogg" },
+ { name: "bug504843.ogv", type: "video/ogg" },
+ { name: "bug501279.ogg", type: "audio/ogg" },
+ { name: "bug604067.webm", type: "video/webm" },
+ { name: "bug1535980.webm", type: "video/webm" },
+ { name: "bug1799787.webm", type: "video/webm" },
+ { name: "bogus.duh", type: "bogus/duh" },
+];
+
+// Playing this file errors out after receiving "loadedmetadata", we still want
+// to check the duration in "onerror" and make sure it is still available.
+var gDurationTests = [{ name: "bug604067.webm", duration: 6.076 }];
+
+// These are files that have nontrivial duration and are useful for seeking within.
+var gSeekTests = [
+ { name: "r11025_s16_c1.wav", type: "audio/x-wav", duration: 1.0 },
+ { name: "audio.wav", type: "audio/x-wav", duration: 0.031247 },
+ { name: "seek.ogv", type: "video/ogg", duration: 3.966 },
+ { name: "320x240.ogv", type: "video/ogg", duration: 0.266 },
+ { name: "seek.webm", type: "video/webm", duration: 3.966 },
+ { name: "sine.webm", type: "audio/webm", duration: 4.001 },
+ { name: "bug516323.indexed.ogv", type: "video/ogg", duration: 4.208333 },
+ { name: "split.webm", type: "video/webm", duration: 1.967 },
+ { name: "detodos.opus", type: "audio/ogg; codecs=opus", duration: 2.9135 },
+ { name: "gizmo.mp4", type: "video/mp4", duration: 5.56 },
+ {
+ name: "/tests/dom/media/webaudio/test/half-a-second-1ch-44100-aac-afconvert.mp4",
+ type: 'audio/mp4; codecs="mp4a.40.2"',
+ duration: 0.5,
+ },
+ { name: "owl.mp3", type: "audio/mpeg", duration: 3.343 },
+ { name: "bogus.duh", type: "bogus/duh", duration: 123 },
+
+ // Bug 1242338: hit a numerical problem while seeking to the duration.
+ { name: "bug482461-theora.ogv", type: "video/ogg", duration: 4.138 },
+];
+
+var gFastSeekTests = [
+ {
+ name: "gizmo.mp4",
+ type: "video/mp4",
+ keyframes: [0, 1.0, 2.0, 3.0, 4.0, 5.0],
+ },
+ // Note: Not all keyframes in the file are actually referenced in the Cues in this file.
+ { name: "seek.webm", type: "video/webm", keyframes: [0, 0.8, 1.6, 2.4, 3.2] },
+ // Note: the sync points are the points on both the audio and video streams
+ // before the keyframes. You can't just assume that the keyframes are the sync
+ // points, as the audio required for that sync point may be before the keyframe.
+ {
+ name: "bug516323.indexed.ogv",
+ type: "video/ogg",
+ keyframes: [0, 0.46, 3.06],
+ },
+];
+
+// These files are WebMs without cues. They're seekable within their buffered
+// ranges. If work renders WebMs fully seekable these files should be moved
+// into gSeekTests
+var gCuelessWebMTests = [
+ { name: "no-cues.webm", type: "video/webm", duration: 3.967 },
+];
+
+// These are files that are non seekable, due to problems with the media,
+// for example broken or missing indexes.
+var gUnseekableTests = [{ name: "bogus.duh", type: "bogus/duh" }];
+
+var androidVersion = -1; // non-Android platforms
+if (
+ manifestNavigator().userAgent.includes("Mobile") ||
+ manifestNavigator().userAgent.includes("Tablet")
+) {
+ androidVersion = SpecialPowers.Services.sysinfo.getProperty("version");
+}
+
+function getAndroidVersion() {
+ return androidVersion;
+}
+
+// These are files suitable for using with a "new Audio" constructor.
+var gAudioTests = [
+ { name: "adts.aac", type: "audio/mp4", duration: 1.37 },
+ { name: "r11025_s16_c1.wav", type: "audio/x-wav", duration: 1.0 },
+ { name: "sound.ogg", type: "audio/ogg" },
+ { name: "owl.mp3", type: "audio/mpeg", duration: 3.343 },
+ { name: "small-shot.m4a", type: "audio/mp4", duration: 0.29 },
+ { name: "bogus.duh", type: "bogus/duh", duration: 123 },
+ { name: "empty_size.mp3", type: "audio/mpeg", duration: 2.235 },
+];
+
+// These files ensure our handling of 404 errors is consistent across the
+// various backends.
+var g404Tests = [
+ { name: "404.wav", type: "audio/x-wav" },
+ { name: "404.ogv", type: "video/ogg" },
+ { name: "404.oga", type: "audio/ogg" },
+ { name: "404.webm", type: "video/webm" },
+ { name: "bogus.duh", type: "bogus/duh" },
+];
+
+// These are files suitable for testing various decoder failures that are
+// expected to fire MEDIA_ERR_DECODE. Used by test_decode_error, which expects
+// an error and emptied event, and no loadedmetadata or ended event.
+var gDecodeErrorTests = [
+ // Valid files with unsupported codecs
+ { name: "r11025_msadpcm_c1.wav", type: "audio/x-wav" },
+ { name: "dirac.ogg", type: "video/ogg" },
+ // Invalid files
+ { name: "bogus.wav", type: "audio/x-wav" },
+ { name: "bogus.ogv", type: "video/ogg" },
+
+ { name: "bogus.duh", type: "bogus/duh" },
+];
+
+// These are files that are used for media fragments tests
+var gFragmentTests = [
+ { name: "big.wav", type: "audio/x-wav", duration: 9.278982, size: 102444 },
+];
+
+// Used by test_chaining.html. The |links| attributes is the number of links in
+// this file that we should be able to play.
+var gChainingTests = [
+ // Vorbis and Opus chained file. They have user comments |index=n| where `n`
+ // is the index of this segment in the file, 0 indexed.
+ { name: "chain.ogg", type: "audio/ogg", links: 4 },
+ { name: "chain.opus", type: "audio/ogg; codec=opus", links: 4 },
+ // Those files are chained files with a different number of channels in each
+ // part. This is not supported and should stop playing after the first part.
+ { name: "variable-channel.ogg", type: "audio/ogg", links: 1 },
+ { name: "variable-channel.opus", type: "audio/ogg; codec=opus", links: 1 },
+ // Those files are chained files with a different sample rate in each
+ // part. This is not supported and should stop playing after the first part.
+ { name: "variable-samplerate.ogg", type: "audio/ogg", links: 1 },
+ // Opus decoding in Firefox outputs 48 kHz PCM despite having a different
+ // original sample rate, so we can safely play Opus chained media that have
+ // different samplerate accross links.
+ { name: "variable-samplerate.opus", type: "audio/ogg; codec=opus", links: 2 },
+ // A chained video file. We don't support those, so only one link should be
+ // reported.
+ { name: "chained-video.ogv", type: "video/ogg", links: 1 },
+ // A file that consist in 4 links of audio, then another link that has video.
+ // We should stop right after the 4 audio links.
+ { name: "chained-audio-video.ogg", type: "video/ogg", links: 4 },
+ // An opus file that has two links, with a different preskip value for each
+ // link. We should be able to play both links.
+ { name: "variable-preskip.opus", type: "audio/ogg; codec=opus", links: 2 },
+ { name: "bogus.duh", type: "bogus/duh" },
+];
+
+// Videos with an aspect ratio. Used for testing that displaying frames
+// on a canvas works correctly in the case of non-standard aspect ratios.
+// See bug 874897 for an example.
+var gAspectRatioTests = [
+ { name: "VID_0001.ogg", type: "video/ogg", duration: 19.966 },
+];
+
+// These are files with non-trivial tag sets.
+// Used by test_metadata.html.
+var gMetadataTests = [
+ // Ogg Vorbis files
+ {
+ name: "short-video.ogv",
+ tags: {
+ TITLE: "Lepidoptera",
+ ARTIST: "Epoq",
+ ALBUM: "Kahvi Collective",
+ DATE: "2002",
+ COMMENT: "http://www.kahvi.org",
+ },
+ },
+ {
+ name: "bug516323.ogv",
+ tags: {
+ GENRE: "Open Movie",
+ ENCODER: "Audacity",
+ TITLE: "Elephants Dream",
+ ARTIST: "Silvia Pfeiffer",
+ COMMENTS: "Audio Description",
+ },
+ },
+ {
+ name: "bug516323.indexed.ogv",
+ tags: {
+ GENRE: "Open Movie",
+ ENCODER: "Audacity",
+ TITLE: "Elephants Dream",
+ ARTIST: "Silvia Pfeiffer",
+ COMMENTS: "Audio Description",
+ },
+ },
+ {
+ name: "detodos.opus",
+ tags: {
+ title: "De todos. Para todos.",
+ artist: "Mozilla.org",
+ },
+ },
+ { name: "sound.ogg", tags: {} },
+ {
+ name: "small-shot.ogg",
+ tags: {
+ title: "Pew SFX",
+ },
+ },
+ {
+ name: "badtags.ogg",
+ tags: {
+ // We list only the valid tags here, and verify
+ // the invalid ones are filtered out.
+ title: "Invalid comments test file",
+ empty: "",
+ "": "empty",
+ "{- [(`!@\"#$%^&')] -}": "valid tag name, surprisingly",
+ // The file also includes the following invalid tags.
+ // "A description with no separator is a common problem.",
+ // "雨":"Likely, but an invalid key (non-ascii).",
+ // "not\nval\x1fid":"invalid tag name",
+ // "not~valid":"this isn't a valid name either",
+ // "not-utf-8":"invalid sequences: \xff\xfe\xfa\xfb\0eol"
+ },
+ },
+ {
+ name: "wave_metadata.wav",
+ tags: {
+ name: "Track Title",
+ artist: "Artist Name",
+ comments: "Comments",
+ },
+ },
+ {
+ name: "wave_metadata_utf8.wav",
+ tags: {
+ name: "歌曲名稱",
+ artist: "作曲者",
+ comments: "註解",
+ },
+ },
+ {
+ name: "wave_metadata_unknown_tag.wav",
+ tags: {
+ name: "Track Title",
+ comments: "Comments",
+ },
+ },
+ {
+ name: "wave_metadata_bad_len.wav",
+ tags: {
+ name: "Track Title",
+ artist: "Artist Name",
+ comments: "Comments",
+ },
+ },
+ {
+ name: "wave_metadata_bad_no_null.wav",
+ tags: {
+ name: "Track Title",
+ artist: "Artist Name",
+ comments: "Comments!!",
+ },
+ },
+ {
+ name: "wave_metadata_bad_utf8.wav",
+ tags: {
+ name: "歌曲名稱",
+ comments: "註解",
+ },
+ },
+ { name: "wavedata_u8.wav", tags: {} },
+];
+
+// Now Fennec doesn't support flac, so only test it on non-android platforms.
+if (getAndroidVersion() < 0) {
+ gMetadataTests = gMetadataTests.concat([
+ {
+ name: "flac-s24.flac",
+ tags: {
+ ALBUM: "Seascapes",
+ TITLE: "(La Mer) - II. Jeux de vagues. Allegro",
+ COMPOSER: "Debussy, Claude",
+ TRACKNUMBER: "2/9",
+ DISCNUMBER: "1/1",
+ encoder: "Lavf57.41.100",
+ },
+ },
+ ]);
+}
+
+// Test files for Encrypted Media Extensions
+var gEMETests = [
+ {
+ name: "vp9 in mp4",
+ tracks: [
+ {
+ name: "video",
+ type: 'video/mp4; codecs="vp9.0"',
+ fragments: ["short-vp9-encrypted-video.mp4"],
+ },
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="mp4a.40.2"',
+ fragments: ["short-aac-encrypted-audio.mp4"],
+ },
+ ],
+ keys: {
+ "2cdb0ed6119853e7850671c3e9906c3c": "808B9ADAC384DE1E4F56140F4AD76194",
+ },
+ sessionType: "temporary",
+ sessionCount: 2,
+ duration: 0.47,
+ },
+ {
+ name: "video-only with 2 keys",
+ tracks: [
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc1.4d4015"',
+ fragments: [
+ "bipbop-cenc-videoinit.mp4",
+ "bipbop-cenc-video1.m4s",
+ "bipbop-cenc-video2.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d037e571d037e571d037e571d03": "7e5733337e5733337e5733337e573333",
+ "7e571d047e571d047e571d047e571d04": "7e5744447e5744447e5744447e574444",
+ },
+ sessionType: "temporary",
+ sessionCount: 1,
+ duration: 1.6,
+ },
+ {
+ name: "video-only with 2 keys, CORS",
+ tracks: [
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc1.4d4015"',
+ fragments: [
+ "bipbop-cenc-videoinit.mp4",
+ "bipbop-cenc-video1.m4s",
+ "bipbop-cenc-video2.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d037e571d037e571d037e571d03": "7e5733337e5733337e5733337e573333",
+ "7e571d047e571d047e571d047e571d04": "7e5744447e5744447e5744447e574444",
+ },
+ sessionType: "temporary",
+ sessionCount: 1,
+ crossOrigin: true,
+ duration: 1.6,
+ },
+ {
+ name: "audio&video tracks, both with all keys",
+ tracks: [
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="mp4a.40.2"',
+ fragments: [
+ "bipbop-cenc-audioinit.mp4",
+ "bipbop-cenc-audio1.m4s",
+ "bipbop-cenc-audio2.m4s",
+ "bipbop-cenc-audio3.m4s",
+ ],
+ },
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc1.4d4015"',
+ fragments: [
+ "bipbop-cenc-videoinit.mp4",
+ "bipbop-cenc-video1.m4s",
+ "bipbop-cenc-video2.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d037e571d037e571d037e571d03": "7e5733337e5733337e5733337e573333",
+ "7e571d047e571d047e571d047e571d04": "7e5744447e5744447e5744447e574444",
+ },
+ sessionType: "temporary",
+ sessionCount: 2,
+ duration: 1.6,
+ },
+ {
+ name: "audio&video tracks, both with all keys, CORS",
+ tracks: [
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="mp4a.40.2"',
+ fragments: [
+ "bipbop-cenc-audioinit.mp4",
+ "bipbop-cenc-audio1.m4s",
+ "bipbop-cenc-audio2.m4s",
+ "bipbop-cenc-audio3.m4s",
+ ],
+ },
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc1.4d4015"',
+ fragments: [
+ "bipbop-cenc-videoinit.mp4",
+ "bipbop-cenc-video1.m4s",
+ "bipbop-cenc-video2.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d037e571d037e571d037e571d03": "7e5733337e5733337e5733337e573333",
+ "7e571d047e571d047e571d047e571d04": "7e5744447e5744447e5744447e574444",
+ },
+ sessionType: "temporary",
+ sessionCount: 2,
+ crossOrigin: true,
+ duration: 1.6,
+ },
+ {
+ name: "400x300 audio&video tracks, each with its key",
+ tracks: [
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="mp4a.40.2"',
+ fragments: [
+ "bipbop_300_215kbps-cenc-audio-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-audio-key1-1.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-2.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-3.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-4.m4s",
+ ],
+ },
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc1.4d4015"',
+ fragments: [
+ "bipbop_300_215kbps-cenc-video-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-video-key1-1.m4s",
+ "bipbop_300_215kbps-cenc-video-key1-2.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311",
+ "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421",
+ },
+ sessionType: "temporary",
+ sessionCount: 2,
+ duration: 1.6,
+ },
+ {
+ name: "640x480@624kbps audio&video tracks, each with its key",
+ tracks: [
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="mp4a.40.2"',
+ fragments: [
+ "bipbop_480_624kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-4.m4s",
+ ],
+ },
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc1.4d401e"',
+ fragments: [
+ "bipbop_480_624kbps-cenc-video-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-video-key1-2.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311",
+ "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421",
+ },
+ sessionType: "temporary",
+ sessionCount: 2,
+ duration: 1.6,
+ },
+ {
+ name: "640x480@959kbps audio&video tracks, each with its key",
+ tracks: [
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="mp4a.40.2"',
+ fragments: [
+ "bipbop_480_959kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480_959kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-4.m4s",
+ ],
+ },
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc1.4d401e"',
+ fragments: [
+ "bipbop_480_959kbps-cenc-video-key1-init.mp4",
+ "bipbop_480_959kbps-cenc-video-key1-1.m4s",
+ "bipbop_480_959kbps-cenc-video-key1-2.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311",
+ "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421",
+ },
+ sessionType: "temporary",
+ sessionCount: 2,
+ duration: 1.6,
+ },
+ {
+ name: "640x480 then 400x300, same key (1st) per track",
+ tracks: [
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="mp4a.40.2"',
+ fragments: [
+ "bipbop_480_624kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-4.m4s",
+ ],
+ },
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc1.4d401e,avc1.4d4015"',
+ fragments: [
+ "bipbop_480_624kbps-cenc-video-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key1-1.m4s",
+ "bipbop_300_215kbps-cenc-video-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-video-key1-2.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311",
+ "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421",
+ },
+ sessionType: "temporary",
+ sessionCount: 3,
+ duration: 1.6,
+ },
+ {
+ name: "640x480 then 400x300, same key (2nd) per track",
+ tracks: [
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="mp4a.40.2"',
+ fragments: [
+ "bipbop_480_624kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480_624kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-4.m4s",
+ ],
+ },
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc1.4d401e,avc1.4d4015"',
+ fragments: [
+ "bipbop_480_624kbps-cenc-video-key2-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key2-1.m4s",
+ "bipbop_300_215kbps-cenc-video-key2-init.mp4",
+ "bipbop_300_215kbps-cenc-video-key2-2.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d037e571d037e571d037e571d12": "7e5733337e5733337e5733337e573312",
+ "7e571d047e571d047e571d047e571d22": "7e5744447e5744447e5744447e574422",
+ },
+ sessionType: "temporary",
+ sessionCount: 3,
+ duration: 1.6,
+ },
+ {
+ name: "640x480 with 1st keys then 400x300 with 2nd keys",
+ tracks: [
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="mp4a.40.2"',
+ fragments: [
+ "bipbop_480_624kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-4.m4s",
+ ],
+ },
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc1.4d401e,avc1.4d4015"',
+ fragments: [
+ "bipbop_480_624kbps-cenc-video-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key1-1.m4s",
+ "bipbop_300_215kbps-cenc-video-key2-init.mp4",
+ "bipbop_300_215kbps-cenc-video-key2-2.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311",
+ "7e571d037e571d037e571d037e571d12": "7e5733337e5733337e5733337e573312",
+ "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421",
+ },
+ sessionType: "temporary",
+ sessionCount: 3,
+ duration: 1.6,
+ },
+ {
+ name: "400x300 with 1st keys then 640x480 with 2nd keys",
+ tracks: [
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="mp4a.40.2"',
+ fragments: [
+ "bipbop_300_215kbps-cenc-audio-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-audio-key1-1.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-2.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-3.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-4.m4s",
+ ],
+ },
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc1.4d4015,avc1.4d401e"',
+ fragments: [
+ "bipbop_300_215kbps-cenc-video-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-video-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-video-key2-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key2-2.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311",
+ "7e571d037e571d037e571d037e571d12": "7e5733337e5733337e5733337e573312",
+ "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421",
+ },
+ sessionType: "temporary",
+ sessionCount: 3,
+ duration: 1.6,
+ },
+ {
+ name: "640x480@959kbps with 1st keys then 640x480@624kbps with 2nd keys",
+ tracks: [
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="mp4a.40.2"',
+ fragments: [
+ "bipbop_480_959kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480_959kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-4.m4s",
+ ],
+ },
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc1.4d401e"',
+ fragments: [
+ "bipbop_480_959kbps-cenc-video-key1-init.mp4",
+ "bipbop_480_959kbps-cenc-video-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-video-key2-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key2-2.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311",
+ "7e571d037e571d037e571d037e571d12": "7e5733337e5733337e5733337e573312",
+ "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421",
+ },
+ sessionType: "temporary",
+ sessionCount: 3,
+ duration: 1.6,
+ },
+ {
+ name: "640x480@624kbps with 1st keys then 640x480@959kbps with 2nd keys",
+ tracks: [
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="mp4a.40.2"',
+ fragments: [
+ "bipbop_480_624kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-4.m4s",
+ ],
+ },
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc1.4d401e"',
+ fragments: [
+ "bipbop_480_624kbps-cenc-video-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key1-1.m4s",
+ "bipbop_480_959kbps-cenc-video-key2-init.mp4",
+ "bipbop_480_959kbps-cenc-video-key2-2.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311",
+ "7e571d037e571d037e571d037e571d12": "7e5733337e5733337e5733337e573312",
+ "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421",
+ },
+ sessionType: "temporary",
+ sessionCount: 3,
+ duration: 1.6,
+ },
+ {
+ name: "400x300 with presentation size 533x300",
+ tracks: [
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="mp4a.40.2"',
+ fragments: [
+ "bipbop_300wp_227kbps-cenc-audio-key1-init.mp4",
+ "bipbop_300wp_227kbps-cenc-audio-key1-1.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-2.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-3.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-4.m4s",
+ ],
+ },
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc1.4d4015"',
+ fragments: [
+ "bipbop_300wp_227kbps-cenc-video-key1-init.mp4",
+ "bipbop_300wp_227kbps-cenc-video-key1-1.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key1-2.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311",
+ "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421",
+ },
+ sessionType: "temporary",
+ sessionCount: 2,
+ duration: 1.6,
+ },
+ {
+ name: "400x300 as-is then 400x300 presented as 533x300",
+ tracks: [
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="mp4a.40.2"',
+ fragments: [
+ "bipbop_300_215kbps-cenc-audio-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-audio-key1-1.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-2.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-3.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-4.m4s",
+ ],
+ },
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc1.4d4015"',
+ fragments: [
+ "bipbop_300_215kbps-cenc-video-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-video-key1-1.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key1-init.mp4",
+ "bipbop_300wp_227kbps-cenc-video-key1-2.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311",
+ "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421",
+ },
+ sessionType: "temporary",
+ sessionCount: 3,
+ duration: 1.6,
+ },
+ {
+ name: "400x225",
+ tracks: [
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="mp4a.40.2"',
+ fragments: [
+ "bipbop_225w_175kbps-cenc-audio-key1-init.mp4",
+ "bipbop_225w_175kbps-cenc-audio-key1-1.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-2.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-3.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-4.m4s",
+ ],
+ },
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc1.64000d"',
+ fragments: [
+ "bipbop_225w_175kbps-cenc-video-key1-init.mp4",
+ "bipbop_225w_175kbps-cenc-video-key1-1.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311",
+ "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421",
+ },
+ sessionType: "temporary",
+ sessionCount: 2,
+ duration: 1.6,
+ },
+ {
+ name: "640x360",
+ tracks: [
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="mp4a.40.2"',
+ fragments: [
+ "bipbop_360w_253kbps-cenc-audio-key1-init.mp4",
+ "bipbop_360w_253kbps-cenc-audio-key1-1.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-2.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-3.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-4.m4s",
+ ],
+ },
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc1.64001e"',
+ fragments: [
+ "bipbop_360w_253kbps-cenc-video-key1-init.mp4",
+ "bipbop_360w_253kbps-cenc-video-key1-1.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311",
+ "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421",
+ },
+ sessionType: "temporary",
+ sessionCount: 2,
+ duration: 1.6,
+ },
+ {
+ name: "400x225 then 640x360",
+ tracks: [
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="mp4a.40.2"',
+ fragments: [
+ "bipbop_225w_175kbps-cenc-audio-key1-init.mp4",
+ "bipbop_225w_175kbps-cenc-audio-key1-1.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-2.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-3.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-4.m4s",
+ ],
+ },
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc1.64000d,avc1.64001e"',
+ fragments: [
+ "bipbop_225w_175kbps-cenc-video-key1-init.mp4",
+ "bipbop_225w_175kbps-cenc-video-key1-1.m4s",
+ "bipbop_360w_253kbps-cenc-video-key2-init.mp4",
+ "bipbop_360w_253kbps-cenc-video-key2-1.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311",
+ "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421",
+ "7e571d037e571d037e571d037e571d12": "7e5733337e5733337e5733337e573312",
+ },
+ sessionType: "temporary",
+ sessionCount: 3,
+ duration: 1.6,
+ },
+ {
+ name: "640x360 then 640x480",
+ tracks: [
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="mp4a.40.2"',
+ fragments: [
+ "bipbop_360w_253kbps-cenc-audio-key1-init.mp4",
+ "bipbop_360w_253kbps-cenc-audio-key1-1.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-2.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-3.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-4.m4s",
+ ],
+ },
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc1.64001e,avc1.4d401e"',
+ fragments: [
+ "bipbop_360w_253kbps-cenc-video-key1-init.mp4",
+ "bipbop_360w_253kbps-cenc-video-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-video-key2-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key2-2.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d037e571d037e571d037e571d11": "7e5733337e5733337e5733337e573311",
+ "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421",
+ "7e571d037e571d037e571d037e571d12": "7e5733337e5733337e5733337e573312",
+ },
+ sessionType: "temporary",
+ sessionCount: 3,
+ duration: 1.6,
+ },
+ {
+ // File generated with shaka packager:
+ // packager-osx --enable_raw_key_encryption --keys label=:key_id=7e571d047e571d047e571d047e571d21:key=7e5744447e5744447e5744447e574421 --segment_duration 1 --clear_lead 0 in=test-flac.mp4,stream=audio,output=flac-sample-cenc.mp4
+ name: "flac in mp4 clearkey",
+ tracks: [
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="flac"',
+ fragments: ["flac-sample-cenc.mp4"],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421",
+ },
+ sessionType: "temporary",
+ sessionCount: 1,
+ duration: 2.05,
+ },
+ {
+ // File generated with shaka packager:
+ // packager-osx --enable_raw_key_encryption --keys label=:key_id=7e571d047e571d047e571d047e571d21:key=7e5744447e5744447e5744447e574421 --segment_duration 1 --clear_lead 0 in=test-opus.mp4,stream=audio,output=opus-sample-cenc.mp4
+ name: "opus in mp4 clearkey",
+ tracks: [
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="opus"',
+ fragments: ["opus-sample-cenc.mp4"],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421",
+ },
+ sessionType: "temporary",
+ sessionCount: 1,
+ duration: 1.98,
+ },
+ {
+ name: "WebM vorbis audio & vp8 video clearkey",
+ tracks: [
+ {
+ name: "audio",
+ type: 'audio/webm; codecs="vorbis"',
+ fragments: ["bipbop_360w_253kbps-clearkey-audio.webm"],
+ },
+ {
+ name: "video",
+ type: 'video/webm; codecs="vp8"',
+ fragments: ["bipbop_360w_253kbps-clearkey-video-vp8.webm"],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ f1f3ee1790527e9de47217d43835f76a: "97b9ddc459c8d5ff23c1f2754c95abe8",
+ "8b5df745ad84145b5617c33116e35a67": "bddfd35dd9be033ee73bc18bc1885056",
+ },
+ sessionType: "temporary",
+ sessionCount: 2,
+ duration: 1.6,
+ },
+ {
+ name: "WebM vorbis audio & vp9 video clearkey",
+ tracks: [
+ {
+ name: "audio",
+ type: 'audio/webm; codecs="vorbis"',
+ fragments: ["bipbop_360w_253kbps-clearkey-audio.webm"],
+ },
+ {
+ name: "video",
+ type: 'video/webm; codecs="vp9"',
+ fragments: ["bipbop_360w_253kbps-clearkey-video-vp9.webm"],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ f1f3ee1790527e9de47217d43835f76a: "97b9ddc459c8d5ff23c1f2754c95abe8",
+ eedf63a94fa7c398ee094f123a4ee709: "973b679a746c82f3acdb856b30e9378e",
+ },
+ sessionType: "temporary",
+ sessionCount: 2,
+ duration: 1.6,
+ },
+ {
+ name: "WebM vorbis audio & vp9 video clearkey with subsample encryption",
+ tracks: [
+ {
+ name: "audio",
+ type: 'audio/webm; codecs="vorbis"',
+ fragments: ["sintel-short-clearkey-subsample-encrypted-audio.webm"],
+ },
+ {
+ name: "video",
+ type: 'video/webm; codecs="vp9"',
+ fragments: ["sintel-short-clearkey-subsample-encrypted-video.webm"],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "2cdb0ed6119853e7850671c3e9906c3c": "808B9ADAC384DE1E4F56140F4AD76194",
+ },
+ sessionType: "temporary",
+ sessionCount: 2,
+ duration: 2.0,
+ },
+ {
+ // Files adapted from testcase for bug 1560092. See bug 1630381 for a
+ // detailed explanation on how they were adapted.
+ name: "avc3 h264 video in mp4 using clearkey cenc encryption",
+ tracks: [
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc3.640015"',
+ fragments: [
+ "big-buck-bunny-cenc-avc3-init.mp4",
+ "big-buck-bunny-cenc-avc3-1.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "10000000100010001000100000000001": "3A2A1B68DD2BD9B2EEB25E84C4776668",
+ },
+ sessionType: "temporary",
+ sessionCount: 1,
+ duration: 2.08,
+ },
+ // The following cbcs files are created using shaka-packager using commands like
+ // ./packager-win.exe 'in=bipbop_2s.mp4,stream=audio,init_segment=bipbop_cbcs_1_9_audio_init.mp4,segment_template=bipbop_cbcs_1_9_audio_$Number$.m4s' \
+ // 'in=bipbop_2s.mp4,stream=video,init_segment=bipbop_cbcs_1_9_video_init.mp4,segment_template=bipbop_cbcs_1_9_video_$Number$.m4s' --protection_scheme cbcs \
+ // --enable_raw_key_encryption --keys label=:key_id=7e571d047e571d047e571d047e571d21:key=7e5744447e5744447e5744447e574421 --iv 11223344556677889900112233445566 \
+ // --clear_lead 0 --crypt_byte_block 1 --skip_byte_block 9
+ // See bug 1726202 for more details on their creation.
+ {
+ name: "mp4 h264 + aac clearkey cbcs 1:9 pattern",
+ tracks: [
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc1.4d4015"',
+ fragments: [
+ "bipbop_cbcs_1_9_video_init.mp4",
+ "bipbop_cbcs_1_9_video_1.m4s",
+ ],
+ },
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="mp4a.40.2"',
+ fragments: [
+ "bipbop_cbcs_1_9_audio_init.mp4",
+ "bipbop_cbcs_1_9_audio_1.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421",
+ },
+ sessionType: "temporary",
+ sessionCount: 2,
+ duration: 2.04,
+ },
+ {
+ name: "mp4 h264 + aac clearkey cbcs 5:5 pattern",
+ tracks: [
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc1.4d4015"',
+ fragments: [
+ "bipbop_cbcs_5_5_video_init.mp4",
+ "bipbop_cbcs_5_5_video_1.m4s",
+ ],
+ },
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="mp4a.40.2"',
+ fragments: [
+ "bipbop_cbcs_5_5_audio_init.mp4",
+ "bipbop_cbcs_5_5_audio_1.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421",
+ },
+ sessionType: "temporary",
+ sessionCount: 2,
+ duration: 2.04,
+ },
+ {
+ name: "mp4 h264 + aac clearkey cbcs 10:0 pattern",
+ tracks: [
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc1.4d4015"',
+ fragments: [
+ "bipbop_cbcs_10_0_video_init.mp4",
+ "bipbop_cbcs_10_0_video_1.m4s",
+ ],
+ },
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="mp4a.40.2"',
+ fragments: [
+ "bipbop_cbcs_10_0_audio_init.mp4",
+ "bipbop_cbcs_10_0_audio_1.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421",
+ },
+ sessionType: "temporary",
+ sessionCount: 2,
+ duration: 2.04,
+ },
+ {
+ name: "mp4 h264 + aac clearkey cbcs 7:7 pattern",
+ tracks: [
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc1.4d4015"',
+ fragments: [
+ "bipbop_cbcs_7_7_video_init.mp4",
+ "bipbop_cbcs_7_7_video_1.m4s",
+ ],
+ },
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="mp4a.40.2"',
+ fragments: [
+ "bipbop_cbcs_7_7_audio_init.mp4",
+ "bipbop_cbcs_7_7_audio_1.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421",
+ },
+ sessionType: "temporary",
+ sessionCount: 2,
+ duration: 2.04,
+ },
+ {
+ name: "mp4 h264 + aac clearkey cbcs 9:8 pattern",
+ tracks: [
+ {
+ name: "video",
+ type: 'video/mp4; codecs="avc1.4d4015"',
+ fragments: [
+ "bipbop_cbcs_9_8_video_init.mp4",
+ "bipbop_cbcs_9_8_video_1.m4s",
+ ],
+ },
+ {
+ name: "audio",
+ type: 'audio/mp4; codecs="mp4a.40.2"',
+ fragments: [
+ "bipbop_cbcs_9_8_audio_init.mp4",
+ "bipbop_cbcs_9_8_audio_1.m4s",
+ ],
+ },
+ ],
+ keys: {
+ // "keyid" : "key"
+ "7e571d047e571d047e571d047e571d21": "7e5744447e5744447e5744447e574421",
+ },
+ sessionType: "temporary",
+ sessionCount: 2,
+ duration: 2.04,
+ },
+];
+
+var gEMENonMSEFailTests = [
+ {
+ name: "short-cenc.mp4",
+ audioType: 'audio/mp4; codecs="mp4a.40.2"',
+ videoType: 'video/mp4; codecs="avc1.64000d"',
+ duration: 0.47,
+ },
+];
+
+// Test files that are supposed to loop seamlessly when played back.
+var gSeamlessLoopingTests = [
+ // MP4 files dont't loop seamlessly yet, the seeking logic seeks to 0, not the
+ // actual first packet, resulting in incorrect decoding.
+ // See bug 1817989
+ // { name: "sin-441-1s-44100-fdk_aac.mp4", type: "audio/mp4" },
+ // { name: "sin-441-1s-44100-afconvert.mp4", type: "audio/mp4" },
+ // { name: "sin-441-1s-44100.ogg", type: "audio/vorbis" },
+ // { name: "sin-441-1s-44100.opus", type: "audio/opus" },
+ { name: "sin-441-1s-44100-lame.mp3", type: "audio/mpeg" },
+ { name: "sin-441-1s-44100.flac", type: "audio/flac" },
+];
+
+// These are files that are used for video decode suspend in
+// background tabs tests.
+var gDecodeSuspendTests = [
+ { name: "gizmo.mp4", type: "video/mp4", duration: 5.56 },
+ { name: "gizmo-noaudio.mp4", type: "video/mp4", duration: 5.56 },
+ { name: "gizmo.webm", type: 'video/webm; codecs="vp9,opus"', duration: 5.56 },
+ {
+ name: "gizmo-noaudio.webm",
+ type: 'video/webm; codecs="vp9"',
+ duration: 5.56,
+ },
+];
+
+// These are video files with hardware-decodable formats and longer
+// durations that are looped while we check telemetry for macOS video
+// low power mode.
+var gVideoLowPowerTests = [
+ { name: "seek.ogv", type: "video/ogg", duration: 3.966 },
+ { name: "gizmo.mp4", type: "video/mp4", duration: 5.56 },
+];
+
+function checkMetadata(msg, e, test) {
+ if (test.width) {
+ is(e.videoWidth, test.width, msg + " video width");
+ }
+ if (test.height) {
+ is(e.videoHeight, test.height, msg + " video height");
+ }
+ if (test.duration) {
+ ok(
+ Math.abs(e.duration - test.duration) < 0.1,
+ msg + " duration (" + e.duration + ") should be around " + test.duration
+ );
+ }
+ is(
+ !!test.keys,
+ SpecialPowers.do_lookupGetter(e, "isEncrypted").apply(e),
+ msg + " isEncrypted should be true if we have decryption keys"
+ );
+}
+
+// Returns the first test from candidates array which we can play with the
+// installed video backends.
+function getPlayableVideo(candidates) {
+ var resources = getPlayableVideos(candidates);
+ if (resources.length) {
+ return resources[0];
+ }
+ return null;
+}
+
+function getPlayableVideos(candidates) {
+ var v = manifestVideo();
+ return candidates.filter(function (x) {
+ return /^video/.test(x.type) && v.canPlayType(x.type);
+ });
+}
+
+function getPlayableAudio(candidates) {
+ var v = manifestVideo();
+ var resources = candidates.filter(function (x) {
+ return /^audio/.test(x.type) && v.canPlayType(x.type);
+ });
+ if (resources.length) {
+ return resources[0];
+ }
+ return null;
+}
+
+// Returns the type of element that should be created for the given mimetype.
+function getMajorMimeType(mimetype) {
+ if (/^video/.test(mimetype)) {
+ return "video";
+ }
+ return "audio";
+}
+
+// Force releasing decoder to avoid timeout in waiting for decoding resource.
+function removeNodeAndSource(n) {
+ n.remove();
+ // reset |srcObject| first since it takes precedence over |src|.
+ n.srcObject = null;
+ n.removeAttribute("src");
+ n.load();
+ while (n.firstChild) {
+ n.firstChild.remove();
+ }
+}
+
+function once(target, name, cb) {
+ var p = new Promise(function (resolve, reject) {
+ target.addEventListener(
+ name,
+ function () {
+ resolve();
+ },
+ { once: true }
+ );
+ });
+ if (cb) {
+ p.then(cb);
+ }
+ return p;
+}
+
+/**
+ * @param {HTMLMediaElement} video target of interest.
+ * @param {string} eventName the event to wait on.
+ * @returns {Promise} A promise that is resolved when event happens.
+ */
+function nextEvent(video, eventName) {
+ return new Promise(function (resolve, reject) {
+ let f = function (event) {
+ video.removeEventListener(eventName, f);
+ resolve(event);
+ };
+ video.addEventListener(eventName, f);
+ });
+}
+
+function TimeStamp(token) {
+ function pad(x) {
+ return x < 10 ? "0" + x : x;
+ }
+ var now = new Date();
+ var ms = now.getMilliseconds();
+ var time =
+ "[" +
+ pad(now.getHours()) +
+ ":" +
+ pad(now.getMinutes()) +
+ ":" +
+ pad(now.getSeconds()) +
+ "." +
+ ms +
+ "]" +
+ // eslint-disable-next-line no-nested-ternary
+ (ms < 10 ? " " : ms < 100 ? " " : "");
+ return token ? time + " " + token : time;
+}
+
+function Log(token, msg) {
+ info(TimeStamp(token) + " " + msg);
+}
+
+// Number of tests to run in parallel.
+var PARALLEL_TESTS = 2;
+
+// Disable parallel test for media engine tests, see more info in bug 1840914.
+if (
+ SpecialPowers.Services.prefs.getIntPref("media.wmf.media-engine.enabled", 0)
+) {
+ PARALLEL_TESTS = 1;
+}
+
+// Prefs to set before running tests. Use this to improve coverage of
+// conditions that might not otherwise be encountered on the test data.
+var gTestPrefs = [
+ ["media.recorder.max_memory", 1024],
+ ["media.audio-max-decode-error", 0],
+ ["media.video-max-decode-error", 0],
+];
+
+// When true, we'll loop forever on whatever test we run. Use this to debug
+// intermittent test failures.
+const DEBUG_TEST_LOOP_FOREVER = false;
+
+// Manages a run of media tests. Runs them in chunks in order to limit
+// the number of media elements/threads running in parallel. This limits peak
+// memory use, particularly on Linux x86 where thread stacks use 10MB of
+// virtual address space.
+// Usage:
+// 1. Create a new MediaTestManager object.
+// 2. Create a test startTest function. This takes a test object and a token,
+// and performs anything necessary to start the test. The test object is an
+// element in one of the g*Tests above. Your startTest function must call
+// MediaTestManager.start(token) if it starts a test. The test object is
+// guaranteed to be playable by our supported decoders; you don't need to
+// check canPlayType.
+// 3. When your tests finishes, call MediaTestManager.finished(), passing
+// the token back to the manager. The manager may either start the next run
+// or end the mochitest if all the tests are done.
+function MediaTestManager() {
+ // Set a very large timeout to prevent Mochitest timeout.
+ // Instead MediaTestManager will manage timeout of each test.
+ SimpleTest.requestLongerTimeout(1000);
+
+ // Return how many seconds elapsed since |begin|.
+ function elapsedTime(begin) {
+ var end = new Date();
+ return (end.getTime() - begin.getTime()) / 1000;
+ }
+ // Sets up a MediaTestManager to runs through the 'tests' array, which needs
+ // to be one of, or have the same fields as, the g*Test arrays of tests. Uses
+ // the user supplied 'startTest' function to initialize the test. This
+ // function must accept two arguments, the test entry from the 'tests' array,
+ // and a token. Call MediaTestManager.started(token) if you start the test,
+ // and MediaTestManager.finished(token) when the test finishes. You don't have
+ // to start every test, but if you call started() you *must* call finish()
+ // else you'll timeout.
+ this.runTests = function (tests, startTest) {
+ this.startTime = new Date();
+ SimpleTest.info(
+ "Started " +
+ this.startTime +
+ " (" +
+ this.startTime.getTime() / 1000 +
+ "s)"
+ );
+ this.testNum = 0;
+ this.tests = tests;
+ this.startTest = startTest;
+ this.tokens = [];
+ this.isShutdown = false;
+ this.numTestsRunning = 0;
+ this.handlers = {};
+ this.timers = {};
+
+ // Always wait for explicit finish.
+ SimpleTest.waitForExplicitFinish();
+ SpecialPowers.pushPrefEnv({ set: gTestPrefs }, () => {
+ this.nextTest();
+ });
+
+ SimpleTest.registerCleanupFunction(() => {
+ if (this.tokens.length) {
+ info("Test timed out. Remaining tests=" + this.tokens);
+ }
+ for (var token of this.tokens) {
+ var handler = this.handlers[token];
+ if (handler && handler.ontimeout) {
+ handler.ontimeout();
+ }
+ }
+ });
+ };
+
+ // Registers that the test corresponding to 'token' has been started.
+ // Don't call more than once per token.
+ this.started = function (token, handler) {
+ this.tokens.push(token);
+ this.numTestsRunning++;
+ this.handlers[token] = handler;
+
+ var onTimeout = async () => {
+ ok(false, "Test timed out!");
+ info(`${token} timed out!`);
+ await dumpDebugInfoForToken(token);
+ this.finished(token);
+ };
+ // Default timeout to 180s for each test.
+ // Call SimpleTest._originalSetTimeout() to bypass the flaky timeout checker.
+ this.timers[token] = SimpleTest._originalSetTimeout.call(
+ window,
+ onTimeout,
+ 180000
+ );
+
+ is(
+ this.numTestsRunning,
+ this.tokens.length,
+ "[started " +
+ token +
+ " t=" +
+ elapsedTime(this.startTime) +
+ "] Length of array should match number of running tests"
+ );
+ };
+
+ // Registers that the test corresponding to 'token' has finished. Call when
+ // you've finished your test. If all tests are complete this will finish the
+ // run, otherwise it may start up the next run. It's ok to call multiple times
+ // per token.
+ this.finished = function (token) {
+ var i = this.tokens.indexOf(token);
+ if (i != -1) {
+ // Remove the element from the list of running tests.
+ this.tokens.splice(i, 1);
+ }
+
+ if (this.timers[token]) {
+ // Cancel the timer when the test finishes.
+ clearTimeout(this.timers[token]);
+ this.timers[token] = null;
+ }
+
+ info("[finished " + token + "] remaining= " + this.tokens);
+ this.numTestsRunning--;
+ is(
+ this.numTestsRunning,
+ this.tokens.length,
+ "[finished " +
+ token +
+ " t=" +
+ elapsedTime(this.startTime) +
+ "] Length of array should match number of running tests"
+ );
+ if (this.tokens.length < PARALLEL_TESTS) {
+ this.nextTest();
+ }
+ };
+
+ // Starts the next batch of tests, or finishes if they're all done.
+ // Don't call this directly, call finished(token) when you're done.
+ this.nextTest = function () {
+ while (
+ this.testNum < this.tests.length &&
+ this.tokens.length < PARALLEL_TESTS
+ ) {
+ var test = this.tests[this.testNum];
+ var token = (test.name ? test.name + "-" : "") + this.testNum;
+ this.testNum++;
+
+ if (DEBUG_TEST_LOOP_FOREVER && this.testNum == this.tests.length) {
+ this.testNum = 0;
+ }
+
+ // Ensure we can play the resource type.
+ if (
+ test.type &&
+ !document.createElement("video").canPlayType(test.type)
+ ) {
+ continue;
+ }
+
+ // Do the init. This should start the test.
+ this.startTest(test, token);
+ }
+
+ if (
+ this.testNum == this.tests.length &&
+ !DEBUG_TEST_LOOP_FOREVER &&
+ !this.tokens.length &&
+ !this.isShutdown
+ ) {
+ this.isShutdown = true;
+ if (this.onFinished) {
+ this.onFinished();
+ }
+ var onCleanup = () => {
+ var end = new Date();
+ SimpleTest.info(
+ "Finished at " + end + " (" + end.getTime() / 1000 + "s)"
+ );
+ SimpleTest.info("Running time: " + elapsedTime(this.startTime) + "s");
+ SimpleTest.finish();
+ };
+ mediaTestCleanup(onCleanup);
+ }
+ };
+}
+
+// Ensures we've got no active video or audio elements in the document, and
+// forces a GC to release the address space reserved by the decoders' threads'
+// stacks.
+function mediaTestCleanup(callback) {
+ var V = document.getElementsByTagName("video");
+ for (let i = 0; i < V.length; i++) {
+ removeNodeAndSource(V[i]);
+ V[i] = null;
+ }
+ var A = document.getElementsByTagName("audio");
+ for (let i = 0; i < A.length; i++) {
+ removeNodeAndSource(A[i]);
+ A[i] = null;
+ }
+ SpecialPowers.exactGC(callback);
+}
+
+async function dumpDebugInfoForToken(token) {
+ for (let v of document.getElementsByTagName("video")) {
+ if (token === v.token) {
+ info(JSON.stringify(await SpecialPowers.wrap(v).mozRequestDebugInfo()));
+ return;
+ }
+ }
+ for (let a of document.getElementsByTagName("audio")) {
+ if (token === a.token) {
+ info(JSON.stringify(await SpecialPowers.wrap(a).mozRequestDebugInfo()));
+ return;
+ }
+ }
+}
+
+// Could be undefined in a page opened by the parent test page
+// like file_access_controls.html.
+if ("SimpleTest" in window) {
+ SimpleTest.requestFlakyTimeout("untriaged");
+
+ // Register timeout function to dump debugging logs.
+ SimpleTest.registerTimeoutFunction(async function () {
+ for (const v of document.getElementsByTagName("video")) {
+ SimpleTest.info(
+ JSON.stringify(await SpecialPowers.wrap(v).mozRequestDebugInfo())
+ );
+ }
+ for (const a of document.getElementsByTagName("audio")) {
+ SimpleTest.info(
+ JSON.stringify(await SpecialPowers.wrap(a).mozRequestDebugInfo())
+ );
+ }
+ });
+}
diff --git a/dom/media/test/midflight-redirect.sjs b/dom/media/test/midflight-redirect.sjs
new file mode 100644
index 0000000000..9a2101251c
--- /dev/null
+++ b/dom/media/test/midflight-redirect.sjs
@@ -0,0 +1,87 @@
+function parseQuery(query, key) {
+ for (let p of query.split("&")) {
+ if (p == key) {
+ return true;
+ }
+ if (p.startsWith(key + "=")) {
+ return p.substring(key.length + 1);
+ }
+ }
+ return false;
+}
+
+// Return the first few bytes in a short byte range response. When Firefox
+// requests subsequent bytes in a second range request, respond with a
+// redirect. Requests after the first redirected are serviced as expected.
+function handleRequest(request, response) {
+ var query = request.queryString;
+ var resource = parseQuery(query, "resource");
+ var type = parseQuery(query, "type") || "application/octet-stream";
+ var redirected = parseQuery(query, "redirected") || false;
+ var useCors = parseQuery(query, "cors") || false;
+
+ var file = Services.dirsvc.get("CurWorkD", Ci.nsIFile);
+ var fis = Cc["@mozilla.org/network/file-input-stream;1"].createInstance(
+ Ci.nsIFileInputStream
+ );
+ var bis = Cc["@mozilla.org/binaryinputstream;1"].createInstance(
+ Ci.nsIBinaryInputStream
+ );
+ var paths = "tests/dom/media/test/" + resource;
+ var split = paths.split("/");
+ for (var i = 0; i < split.length; ++i) {
+ file.append(split[i]);
+ }
+ fis.init(file, -1, -1, false);
+
+ bis.setInputStream(fis);
+ var bytes = bis.readBytes(bis.available());
+ let [from, to] = request
+ .getHeader("range")
+ .split("=")[1]
+ .split("-")
+ .map(s => parseInt(s));
+
+ if (!redirected && from > 0) {
+ var origin =
+ request.host == "mochi.test" ? "example.org" : "mochi.test:8888";
+ response.setStatusLine(request.httpVersion, 303, "See Other");
+ let url =
+ "http://" +
+ origin +
+ "/tests/dom/media/test/midflight-redirect.sjs?redirected&" +
+ query;
+ response.setHeader("Location", url);
+ response.setHeader("Content-Type", "text/html");
+ return;
+ }
+
+ if (isNaN(to)) {
+ to = bytes.length - 1;
+ }
+
+ if (from == 0 && !redirected) {
+ to =
+ parseInt(parseQuery(query, "redirectAt")) || Math.floor(bytes.length / 4);
+ }
+ to = Math.min(to, bytes.length - 1);
+
+ // Note: 'to' is the first index *excluded*, so we need (to + 1)
+ // in the substring end here.
+ let byterange = bytes.substring(from, to + 1);
+
+ let contentRange = "bytes " + from + "-" + to + "/" + bytes.length;
+ let contentLength = byterange.length.toString();
+
+ response.setStatusLine(request.httpVersion, 206, "Partial Content");
+ response.setHeader("Content-Range", contentRange);
+ response.setHeader("Content-Length", contentLength, false);
+ response.setHeader("Content-Type", type, false);
+ response.setHeader("Accept-Ranges", "bytes", false);
+ response.setHeader("Cache-Control", "no-cache, no-store, must-revalidate");
+ if (redirected && useCors) {
+ response.setHeader("Access-Control-Allow-Origin", "*");
+ }
+ response.write(byterange, byterange.length);
+ bis.close();
+}
diff --git a/dom/media/test/mochitest.toml b/dom/media/test/mochitest.toml
new file mode 100644
index 0000000000..de8c690255
--- /dev/null
+++ b/dom/media/test/mochitest.toml
@@ -0,0 +1,951 @@
+# Media tests should be backend independent, i.e., not conditioned on ogg,
+# wave etc. (The only exception is the can_play_type tests, which
+# necessarily depend on the backend(s) configured.) As far as possible, each
+# test should work with any resource type. This makes it easy to add new
+# backends and reduces the amount of test duplication.
+
+# For each supported backend, resources that can be played by that backend
+# should be added to the lists in manifest.js. Media tests that aren't
+# testing for a bug in handling a specific resource type should pick one of
+# the lists in manifest.js and run the test for each resource in the list
+# that is supported in the current build (the canPlayType API is useful for
+# this).
+
+# To test whether a valid resource can simply be played through correctly,
+# and optionally that its metadata is read correctly, just add it to
+# gPlayTests in manifest.js. To test whether an invalid resource correctly
+# throws an error (and does not cause a crash or hang), just add it to
+# gErrorTests in manifest.js.
+
+# To test for a specific bug in handling a specific resource type, make the
+# test first check canPlayType for the type, and if it's not supported, just
+# do ok(true, "Type not supported") and stop the test.
+
+[DEFAULT]
+subsuite = "media"
+tags = "media-gpu"
+prefs = [
+ "gfx.core-animation.low-power-telemetry-frames=60", # only for test_video_low_power_telemetry.html
+ "media.setsinkid.enabled=true", # for test_setSinkId_after_loop.html
+]
+support-files = [
+ "16bit_wave_extrametadata.wav",
+ "16bit_wave_extrametadata.wav^headers^",
+ "320x240.ogv",
+ "320x240.ogv^headers^",
+ "448636.ogv",
+ "448636.ogv^headers^",
+ "A4.ogv",
+ "A4.ogv^headers^",
+ "VID_0001.ogg",
+ "VID_0001.ogg^headers^",
+ "allowed.sjs",
+ "ambisonics.mp4",
+ "ambisonics.mp4^headers^",
+ "audio-gaps.ogg",
+ "audio-gaps.ogg^headers^",
+ "audio-gaps-short.ogg",
+ "audio-gaps-short.ogg^headers^",
+ "audio-overhang.ogg",
+ "audio-overhang.ogg^headers^",
+ "audio.wav",
+ "audio.wav^headers^",
+ "av1.mp4",
+ "av1.mp4^headers^",
+ "background_video.js",
+ "badtags.ogg",
+ "badtags.ogg^headers^",
+ "bear-640x360-v_frag-cenc-key_rotation.mp4",
+ "bear-640x360-a_frag-cenc-key_rotation.mp4",
+ "beta-phrasebook.ogg",
+ "beta-phrasebook.ogg^headers^",
+ "big.wav",
+ "big.wav^headers^",
+ "big-buck-bunny-cenc-avc3-1.m4s",
+ "big-buck-bunny-cenc-avc3-1.m4s^headers^",
+ "big-buck-bunny-cenc-avc3-init.mp4",
+ "big-buck-bunny-cenc-avc3-init.mp4^headers^",
+ "big-short.wav",
+ "big-short.wav^headers^",
+ "bipbop.mp4",
+ "bipbop-cenc-audio1.m4s",
+ "bipbop-cenc-audio1.m4s^headers^",
+ "bipbop-cenc-audio2.m4s",
+ "bipbop-cenc-audio2.m4s^headers^",
+ "bipbop-cenc-audio3.m4s",
+ "bipbop-cenc-audio3.m4s^headers^",
+ "bipbop-cenc-audioinit.mp4",
+ "bipbop-cenc-audioinit.mp4^headers^",
+ "bipbop-cenc-video1.m4s",
+ "bipbop-cenc-video1.m4s^headers^",
+ "bipbop-cenc-video2.m4s",
+ "bipbop-cenc-video2.m4s^headers^",
+ "bipbop-cenc-videoinit.mp4",
+ "bipbop-cenc-videoinit.mp4^headers^",
+ "bipbop-cenc-video-10s.mp4",
+ "bipbop-cenc-video-10s.mp4^headers^",
+ "bipbop-clearkey-keyrotation-clear-lead-audio.mp4",
+ "bipbop-clearkey-keyrotation-clear-lead-audio.mp4^headers^",
+ "bipbop-clearkey-keyrotation-clear-lead-video.mp4",
+ "bipbop-clearkey-keyrotation-clear-lead-video.mp4^headers^",
+ "bipbop_225w_175kbps.mp4",
+ "bipbop_225w_175kbps.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-1.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-2.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-3.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-4.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-init.mp4",
+ "bipbop_225w_175kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-1.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-2.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-3.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-4.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-init.mp4",
+ "bipbop_225w_175kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-video-key1-1.m4s",
+ "bipbop_225w_175kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-video-key1-init.mp4",
+ "bipbop_225w_175kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-video-key2-1.m4s",
+ "bipbop_225w_175kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-video-key2-init.mp4",
+ "bipbop_225w_175kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-1.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-2.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-3.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-4.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-1.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-2.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-3.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-4.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-init.mp4",
+ "bipbop_300_215kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-1.m4s",
+ "bipbop_300_215kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-2.m4s",
+ "bipbop_300_215kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-1.m4s",
+ "bipbop_300_215kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-2.m4s",
+ "bipbop_300_215kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-init.mp4",
+ "bipbop_300_215kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-1.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-2.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-3.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-4.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-init.mp4",
+ "bipbop_300wp_227kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-1.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-2.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-3.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-4.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-init.mp4",
+ "bipbop_300wp_227kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-1.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-2.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-init.mp4",
+ "bipbop_300wp_227kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-1.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-2.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-init.mp4",
+ "bipbop_300wp_227kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-1.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-2.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-3.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-4.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-init.mp4",
+ "bipbop_360w_253kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-1.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-2.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-3.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-4.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-init.mp4",
+ "bipbop_360w_253kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-video-key1-1.m4s",
+ "bipbop_360w_253kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-video-key1-init.mp4",
+ "bipbop_360w_253kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-video-key2-1.m4s",
+ "bipbop_360w_253kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-video-key2-init.mp4",
+ "bipbop_360w_253kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-clearkey-audio.webm",
+ "bipbop_360w_253kbps-clearkey-audio.webm^headers^",
+ "bipbop_360w_253kbps-clearkey-video-vp8.webm",
+ "bipbop_360w_253kbps-clearkey-video-vp8.webm^headers^",
+ "bipbop_360w_253kbps-clearkey-video-vp9.webm",
+ "bipbop_360w_253kbps-clearkey-video-vp9.webm^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480_624kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-2.m4s",
+ "bipbop_480_624kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-1.m4s",
+ "bipbop_480_624kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-2.m4s",
+ "bipbop_480_624kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480_959kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480_959kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-1.m4s",
+ "bipbop_480_959kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-2.m4s",
+ "bipbop_480_959kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-init.mp4",
+ "bipbop_480_959kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-1.m4s",
+ "bipbop_480_959kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-2.m4s",
+ "bipbop_480_959kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-init.mp4",
+ "bipbop_480_959kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480wp_663kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480wp_663kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-1.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-2.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-init.mp4",
+ "bipbop_480wp_663kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-1.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-2.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-init.mp4",
+ "bipbop_480wp_663kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_audio_aac_8k.mp4",
+ "bipbop_audio_aac_8k.mp4^headers^",
+ "bipbop_audio_aac_22.05k.mp4",
+ "bipbop_audio_aac_22.05k.mp4^headers^",
+ "bipbop_audio_aac_44.1k.mp4",
+ "bipbop_audio_aac_44.1k.mp4^headers^",
+ "bipbop_audio_aac_48k.mp4",
+ "bipbop_audio_aac_48k.mp4^headers^",
+ "bipbop_audio_aac_88.2k.mp4",
+ "bipbop_audio_aac_88.2k.mp4^headers^",
+ "bipbop_audio_aac_96k.mp4",
+ "bipbop_audio_aac_96k.mp4^headers^",
+ "bipbop_cbcs_1_9_audio_1.m4s",
+ "bipbop_cbcs_1_9_audio_1.m4s^headers^",
+ "bipbop_cbcs_1_9_audio_init.mp4",
+ "bipbop_cbcs_1_9_audio_init.mp4^headers^",
+ "bipbop_cbcs_1_9_video_1.m4s",
+ "bipbop_cbcs_1_9_video_1.m4s^headers^",
+ "bipbop_cbcs_1_9_video_init.mp4",
+ "bipbop_cbcs_1_9_video_init.mp4^headers^",
+ "bipbop_cbcs_5_5_audio_1.m4s",
+ "bipbop_cbcs_5_5_audio_1.m4s^headers^",
+ "bipbop_cbcs_5_5_audio_init.mp4",
+ "bipbop_cbcs_5_5_audio_init.mp4^headers^",
+ "bipbop_cbcs_5_5_video_1.m4s",
+ "bipbop_cbcs_5_5_video_1.m4s^headers^",
+ "bipbop_cbcs_5_5_video_init.mp4",
+ "bipbop_cbcs_5_5_video_init.mp4^headers^",
+ "bipbop_cbcs_7_7_audio_1.m4s",
+ "bipbop_cbcs_7_7_audio_1.m4s^headers^",
+ "bipbop_cbcs_7_7_audio_init.mp4",
+ "bipbop_cbcs_7_7_audio_init.mp4^headers^",
+ "bipbop_cbcs_7_7_video_1.m4s",
+ "bipbop_cbcs_7_7_video_1.m4s^headers^",
+ "bipbop_cbcs_7_7_video_init.mp4",
+ "bipbop_cbcs_7_7_video_init.mp4^headers^",
+ "bipbop_cbcs_9_8_audio_1.m4s",
+ "bipbop_cbcs_9_8_audio_1.m4s^headers^",
+ "bipbop_cbcs_9_8_audio_init.mp4",
+ "bipbop_cbcs_9_8_audio_init.mp4^headers^",
+ "bipbop_cbcs_9_8_video_1.m4s",
+ "bipbop_cbcs_9_8_video_1.m4s^headers^",
+ "bipbop_cbcs_9_8_video_init.mp4",
+ "bipbop_cbcs_9_8_video_init.mp4^headers^",
+ "bipbop_cbcs_10_0_audio_1.m4s",
+ "bipbop_cbcs_10_0_audio_1.m4s^headers^",
+ "bipbop_cbcs_10_0_audio_init.mp4",
+ "bipbop_cbcs_10_0_audio_init.mp4^headers^",
+ "bipbop_cbcs_10_0_video_1.m4s",
+ "bipbop_cbcs_10_0_video_1.m4s^headers^",
+ "bipbop_cbcs_10_0_video_init.mp4",
+ "bipbop_cbcs_10_0_video_init.mp4^headers^",
+ "bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm",
+ "bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm",
+ "bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm",
+ "bipbop_short_vp8.webm",
+ "bipbop_short_vp8.webm^headers^",
+ "bipbop-lateaudio.mp4",
+ "bipbop-lateaudio.mp4^headers^",
+ "black100x100-aspect3to2.ogv",
+ "black100x100-aspect3to2.ogv^headers^",
+ "bogus.duh",
+ "bogus.ogv",
+ "bogus.ogv^headers^",
+ "bogus.wav",
+ "bogus.wav^headers^",
+ "bug461281.ogg",
+ "bug461281.ogg^headers^",
+ "bug482461-theora.ogv",
+ "bug482461-theora.ogv^headers^",
+ "bug482461.ogv",
+ "bug482461.ogv^headers^",
+ "bug495129.ogv",
+ "bug495129.ogv^headers^",
+ "bug495794.ogg",
+ "bug495794.ogg^headers^",
+ "bug498380.ogv",
+ "bug498380.ogv^headers^",
+ "bug498855-1.ogv",
+ "bug498855-1.ogv^headers^",
+ "bug498855-2.ogv",
+ "bug498855-2.ogv^headers^",
+ "bug498855-3.ogv",
+ "bug498855-3.ogv^headers^",
+ "bug499519.ogv",
+ "bug499519.ogv^headers^",
+ "bug500311.ogv",
+ "bug500311.ogv^headers^",
+ "bug501279.ogg",
+ "bug501279.ogg^headers^",
+ "bug504613.ogv",
+ "bug504613.ogv^headers^",
+ "bug504644.ogv",
+ "bug504644.ogv^headers^",
+ "bug504843.ogv",
+ "bug504843.ogv^headers^",
+ "bug506094.ogv",
+ "bug506094.ogv^headers^",
+ "bug516323.indexed.ogv",
+ "bug516323.indexed.ogv^headers^",
+ "bug516323.ogv",
+ "bug516323.ogv^headers^",
+ "bug520493.ogg",
+ "bug520493.ogg^headers^",
+ "bug520500.ogg",
+ "bug520500.ogg^headers^",
+ "bug520908.ogv",
+ "bug520908.ogv^headers^",
+ "bug523816.ogv",
+ "bug523816.ogv^headers^",
+ "bug533822.ogg",
+ "bug533822.ogg^headers^",
+ "bug556821.ogv",
+ "bug556821.ogv^headers^",
+ "bug557094.ogv",
+ "bug557094.ogv^headers^",
+ "bug604067.webm",
+ "bug604067.webm^headers^",
+ "bug1066943.webm",
+ "bug1066943.webm^headers^",
+ "bug1301226.wav",
+ "bug1301226.wav^headers^",
+ "bug1301226-odd.wav",
+ "bug1301226-odd.wav^headers^",
+ "bug1377278.webm",
+ "bug1377278.webm^headers^",
+ "bug1535980.webm",
+ "bug1535980.webm^headers^",
+ "bunny.webm",
+ "can_play_type_dash.js",
+ "can_play_type_ogg.js",
+ "can_play_type_wave.js",
+ "can_play_type_webm.js",
+ "cancellable_request.sjs",
+ "chain.ogg",
+ "chain.ogg^headers^",
+ "chain.ogv",
+ "chain.ogv^headers^",
+ "chain.opus",
+ "chain.opus^headers^",
+ "chained-audio-video.ogg",
+ "chained-audio-video.ogg^headers^",
+ "chained-video.ogv",
+ "chained-video.ogv^headers^",
+ "chromeHelper.js",
+ "cloneElementVisually_helpers.js",
+ "contentType.sjs",
+ "detodos.opus",
+ "detodos.opus^headers^",
+ "detodos.webm",
+ "detodos.webm^headers^",
+ "detodos-short.webm",
+ "detodos-short.webm^headers^",
+ "detodos-recorder-test.opus",
+ "detodos-recorder-test.opus^headers^",
+ "detodos-short.opus",
+ "detodos-short.opus^headers^",
+ "dirac.ogg",
+ "dirac.ogg^headers^",
+ "dynamic_resource.sjs",
+ "eme_standalone.js",
+ "eme.js",
+ "empty_size.mp3",
+ "file_access_controls.html",
+ "file_eme_createMediaKeys.html",
+ "flac-s24.flac",
+ "flac-s24.flac^headers^",
+ "flac-noheader-s16.flac",
+ "flac-noheader-s16.flac^headers^",
+ "flac-sample.mp4",
+ "flac-sample.mp4^headers^",
+ "flac-sample-cenc.mp4",
+ "flac-sample-cenc.mp4^headers^",
+ "fragment_noplay.js",
+ "fragment_play.js",
+ "gizmo.mp4",
+ "gizmo.mp4^headers^",
+ "gizmo-noaudio.mp4",
+ "gizmo-noaudio.mp4^headers^",
+ "gizmo-short.mp4",
+ "gizmo-short.mp4^headers^",
+ "gizmo.webm",
+ "gizmo.webm^headers^",
+ "gizmo-noaudio.webm",
+ "gizmo-noaudio.webm^headers^",
+ "gUM_support.js",
+ "gzipped_mp4.sjs",
+ "huge-id3.mp3",
+ "huge-id3.mp3^headers^",
+ "id3tags.mp3",
+ "id3tags.mp3^headers^",
+ "id3v1afterlongid3v2.mp3",
+ "invalid-cmap-s0c0.opus",
+ "invalid-cmap-s0c0.opus^headers^",
+ "invalid-cmap-s0c2.opus",
+ "invalid-cmap-s0c2.opus^headers^",
+ "invalid-cmap-s1c2.opus",
+ "invalid-cmap-s1c2.opus^headers^",
+ "invalid-cmap-short.opus",
+ "invalid-cmap-short.opus^headers^",
+ "invalid-discard_on_multi_blocks.webm",
+ "invalid-discard_on_multi_blocks.webm^headers^",
+ "invalid-excess_discard.webm",
+ "invalid-excess_discard.webm^headers^",
+ "invalid-excess_neg_discard.webm",
+ "invalid-excess_neg_discard.webm^headers^",
+ "invalid-m0c0.opus",
+ "invalid-m0c0.opus^headers^",
+ "invalid-m0c3.opus",
+ "invalid-m0c3.opus^headers^",
+ "invalid-m1c0.opus",
+ "invalid-m1c0.opus^headers^",
+ "invalid-m1c9.opus",
+ "invalid-m1c9.opus^headers^",
+ "invalid-m2c0.opus",
+ "invalid-m2c0.opus^headers^",
+ "invalid-m2c1.opus",
+ "invalid-m2c1.opus^headers^",
+ "invalid-neg_discard.webm",
+ "invalid-neg_discard.webm^headers^",
+ "invalid-preskip.webm",
+ "invalid-preskip.webm^headers^",
+ "manifest.js",
+ "midflight-redirect.sjs",
+ "multiple-bos.ogg",
+ "multiple-bos.ogg^headers^",
+ "multiple-bos-more-header-fileds.ogg",
+ "multiple-bos-more-header-fileds.ogg^headers^",
+ "multi_id3v2.mp3",
+ "no-container-codec-delay.webm",
+ "no-cues.webm",
+ "no-cues.webm^headers^",
+ "notags.mp3",
+ "notags.mp3^headers^",
+ "opus-mapping2.mp4",
+ "opus-mapping2.mp4^headers^",
+ "opus-mapping2.webm",
+ "opus-mapping2.webm^headers^",
+ "opus-sample.mp4",
+ "opus-sample.mp4^headers^",
+ "opus-sample-cenc.mp4",
+ "opus-sample-cenc.mp4^headers^",
+ "owl-funnier-id3.mp3",
+ "owl-funnier-id3.mp3^headers^",
+ "owl-funny-id3.mp3",
+ "owl-funny-id3.mp3^headers^",
+ "owl.mp3",
+ "owl.mp3^headers^",
+ "owl-short.mp3",
+ "owl-short.mp3^headers^",
+ "pixel_aspect_ratio.mp4",
+ "play_promise.js",
+ "poster-test.jpg",
+ "r11025_msadpcm_c1.wav",
+ "r11025_msadpcm_c1.wav^headers^",
+ "r11025_s16_c1.wav",
+ "r11025_s16_c1.wav^headers^",
+ "r11025_s16_c1_trailing.wav",
+ "r11025_s16_c1_trailing.wav^headers^",
+ "r11025_s16_c1-short.wav",
+ "r11025_s16_c1-short.wav^headers^",
+ "r11025_u8_c1.wav",
+ "r11025_u8_c1.wav^headers^",
+ "r11025_u8_c1_trunc.wav",
+ "r11025_u8_c1_trunc.wav^headers^",
+ "r16000_u8_c1_list.wav",
+ "r16000_u8_c1_list.wav^headers^",
+ "reactivate_helper.html",
+ "red-46x48.mp4",
+ "red-46x48.mp4^headers^",
+ "red-48x46.mp4",
+ "red-48x46.mp4^headers^",
+ "redirect.sjs",
+ "referer.sjs",
+ "resolution-change.webm",
+ "resolution-change.webm^headers^",
+ "sample.3gp",
+ "sample.3g2",
+ "sample-encrypted-sgpdstbl-sbgptraf.mp4",
+ "sample-encrypted-sgpdstbl-sbgptraf.mp4^headers^",
+ "sample-fisbone-skeleton4.ogv",
+ "sample-fisbone-skeleton4.ogv^headers^",
+ "sample-fisbone-wrong-header.ogv",
+ "sample-fisbone-wrong-header.ogv^headers^",
+ "seek.ogv",
+ "seek.ogv^headers^",
+ "seek-short.ogv",
+ "seek-short.ogv^headers^",
+ "seek.webm",
+ "seek.webm^headers^",
+ "seek-short.webm",
+ "seek-short.webm^headers^",
+ "seek_support.js",
+ "seekLies.sjs",
+ "seek_with_sound.ogg^headers^",
+ "short-aac-encrypted-audio.mp4",
+ "short-aac-encrypted-audio.mp4^headers^",
+ "short-audio-fragmented-cenc-without-pssh.mp4",
+ "short-audio-fragmented-cenc-without-pssh.mp4^headers^",
+ "short-cenc.mp4",
+ "short-video.ogv",
+ "short.mp4",
+ "short.mp4.gz",
+ "short.mp4^headers^",
+ "shorter_audio_than_video_3s.webm",
+ "shorter_audio_than_video_3s.webm^headers^",
+ # source file generated with:
+ # > sox -V -r 44100 -n -b 16 -c 1 sin-441-1s-44100.wav synth 1 sin 441 vol -5dB
+ # then encoded:
+ # ffmpeg -i sin-441-1s-44100.wav sin-441-1s-44100-libfdk_aac.mp4
+ "sin-441-1s-44100-fdk_aac.mp4",
+ # afconvert -s 3 -f mp4f -d aac sin-4411-1s-441100.wav
+ "sin-441-1s-44100-afconvert.mp4",
+ # ffmpeg -i sin-441-1s-44100.wav sin-441-1s-44100-libfdk_lame.mp3
+ "sin-441-1s-44100-lame.mp3",
+ # ffmpeg -i sin-441-1s-44100.wav sin-441-1s-44100.ogg
+ "sin-441-1s-44100.ogg",
+ # ffmpeg -i sin-441-1s-44100.wav sin-441-1s-44100.opus
+ "sin-441-1s-44100.opus",
+ # ffmpeg -i sin-441-1s-44100.wav sin-441-1s-44100.flac
+ "sin-441-1s-44100.flac",
+ "sine.webm",
+ "sine.webm^headers^",
+ "sintel-short-clearkey-subsample-encrypted-audio.webm",
+ "sintel-short-clearkey-subsample-encrypted-audio.webm^headers^",
+ "sintel-short-clearkey-subsample-encrypted-video.webm",
+ "sintel-short-clearkey-subsample-encrypted-video.webm^headers^",
+ "short-video.ogv^headers^",
+ "short-vp9-encrypted-video.mp4",
+ "short-vp9-encrypted-video.mp4^headers^",
+ "small-shot-mp3.mp4",
+ "small-shot-mp3.mp4^headers^",
+ "small-shot.m4a",
+ "small-shot.mp3",
+ "small-shot.mp3^headers^",
+ "small-shot.ogg",
+ "small-shot.ogg^headers^",
+ "small-shot.flac",
+ "sound.ogg",
+ "sound.ogg^headers^",
+ "spacestorm-1000Hz-100ms.ogg",
+ "spacestorm-1000Hz-100ms.ogg^headers^",
+ "split.webm",
+ "split.webm^headers^",
+ "street.mp4",
+ "street.mp4^headers^",
+ "test-1-mono.opus",
+ "test-1-mono.opus^headers^",
+ "test-2-stereo.opus",
+ "test-2-stereo.opus^headers^",
+ "test-3-LCR.opus",
+ "test-3-LCR.opus^headers^",
+ "test-4-quad.opus",
+ "test-4-quad.opus^headers^",
+ "test-5-5.0.opus",
+ "test-5-5.0.opus^headers^",
+ "test-6-5.1.opus",
+ "test-6-5.1.opus^headers^",
+ "test-7-6.1.opus",
+ "test-7-6.1.opus^headers^",
+ "test-8-7.1.opus",
+ "test-8-7.1.opus^headers^",
+ "test-stereo-phase-inversion-180.opus",
+ "test-stereo-phase-inversion-180.opus^headers^",
+ "variable-channel.ogg",
+ "variable-channel.ogg^headers^",
+ "variable-channel.opus",
+ "variable-channel.opus^headers^",
+ "variable-preskip.opus",
+ "variable-preskip.opus^headers^",
+ "variable-samplerate.ogg",
+ "variable-samplerate.ogg^headers^",
+ "variable-samplerate.opus",
+ "variable-samplerate.opus^headers^",
+ "vbr-head.mp3",
+ "vbr-head.mp3^headers^",
+ "vbr.mp3",
+ "vbr.mp3^headers^",
+ "very-short.mp3",
+ "video-overhang.ogg",
+ "video-overhang.ogg^headers^",
+ "vp9-superframes.webm",
+ "vp9-superframes.webm^headers^",
+ "vp9.webm",
+ "vp9.webm^headers^",
+ "vp9-short.webm",
+ "vp9-short.webm^headers^",
+ "vp9cake.webm",
+ "vp9cake.webm^headers^",
+ "vp9cake-short.webm",
+ "vp9cake-short.webm^headers^",
+ "wave_metadata.wav",
+ "wave_metadata.wav^headers^",
+ "wave_metadata_bad_len.wav",
+ "wave_metadata_bad_len.wav^headers^",
+ "wave_metadata_bad_no_null.wav",
+ "wave_metadata_bad_no_null.wav^headers^",
+ "wave_metadata_bad_utf8.wav",
+ "wave_metadata_bad_utf8.wav^headers^",
+ "wave_metadata_unknown_tag.wav",
+ "wave_metadata_unknown_tag.wav^headers^",
+ "wave_metadata_utf8.wav",
+ "wave_metadata_utf8.wav^headers^",
+ "wavedata_alaw.wav",
+ "wavedata_alaw.wav^headers^",
+ "wavedata_float.wav",
+ "wavedata_float.wav^headers^",
+ "wavedata_s24.wav",
+ "wavedata_s24.wav^headers^",
+ "wavedata_s16.wav",
+ "wavedata_s16.wav^headers^",
+ "wavedata_u8.wav",
+ "wavedata_u8.wav^headers^",
+ "wavedata_ulaw.wav",
+ "wavedata_ulaw.wav^headers^",
+ "white-short.webm",
+ "white-3s-black-1s.webm",
+ "white-3s-black-1s.webm^headers^",
+ "!/dom/canvas/test/captureStream_common.js",
+ "!/dom/html/test/reflect.js",
+ "!/dom/media/webrtc/tests/mochitests/head.js",
+ "hls/bipbop_16x9_single.m3u8",
+ "hls/bipbop_4x3_single.m3u8",
+ "hls/bipbop_4x3_variant.m3u8",
+ "hls/400x300_prog_index.m3u8",
+ "hls/400x300_prog_index_5s.m3u8",
+ "hls/416x243_prog_index_5s.m3u8",
+ "hls/640x480_prog_index.m3u8",
+ "hls/960x720_prog_index.m3u8",
+ "hls/400x300_seg0.ts",
+ "hls/400x300_seg0_5s.ts",
+ "hls/400x300_seg1.ts",
+ "hls/416x243_seg0_5s.ts",
+ "hls/640x480_seg0.ts",
+ "hls/640x480_seg1.ts",
+ "hls/960x720_seg0.ts",
+ "hls/960x720_seg1.ts",
+ "sync.webm",
+ "two-xing-header-no-content-length.mp3",
+ "two-xing-header-no-content-length.mp3^headers^",
+ "single-xing-header-no-content-length.mp3",
+ "single-xing-header-no-content-length.mp3^headers^",
+ "padding-spanning-multiple-packets.mp3",
+]
+
+["test_capture_stream_av_sync.html"]
+skip-if = [
+ "os == 'android'", # 1712598 (canvas error)
+ "os == 'mac'", # 1517199 (timeout-on-osx)
+ "os == 'linux'", # 1713397, 1719881 (high intermittent failure on linux tsan), 1776937
+ "apple_silicon", # Disabled due to bleedover with other tests when run in regular suites; passes in "failures" jobs
+]
+
+["test_chaining.html"]
+
+["test_cloneElementVisually_ended_video.html"]
+tags = "cloneelementvisually"
+
+["test_cloneElementVisually_mediastream.html"]
+tags = "cloneelementvisually"
+
+["test_cloneElementVisually_mediastream_multitrack.html"]
+tags = "cloneelementvisually"
+
+["test_cloneElementVisually_no_suspend.html"]
+tags = "cloneelementvisually"
+
+["test_cloneElementVisually_paused.html"]
+tags = "cloneelementvisually"
+
+["test_cloneElementVisually_poster.html"]
+tags = "cloneelementvisually"
+
+["test_cloneElementVisually_resource_change.html"]
+skip-if = ["os == 'linux' && !(debug || asan || tsan)"] # Bug 1559308 - lower frequency intermittent
+tags = "cloneelementvisually"
+
+["test_clone_media_element.html"]
+skip-if = ["os == 'android'"] # bug 1108558, android(bug 1232305)
+
+["test_fastSeek-forwards.html"]
+
+["test_fastSeek.html"]
+
+["test_hls_player_independency.html"]
+# There's a limit for creating decoder when API lever < 18(Bug 1278574)
+# We could skip the test in that case as we cannot play 2 video at a time.
+skip-if = ["os != 'android'"]
+tags = "hls"
+
+["test_hw_video_decoding.html"]
+skip-if = ["!mda_gpu"]
+scheme = "https"
+
+["test_imagecapture.html"]
+scheme = "https"
+
+["test_invalid_reject_play.html"]
+
+["test_media_selection.html"]
+
+["test_media_sniffer.html"]
+
+["test_mediastream_as_eventarget.html"]
+
+["test_mediatrack_consuming_mediastream.html"]
+scheme = "https"
+tags = "mtg"
+
+["test_mediatrack_replay_from_end.html"]
+
+["test_midflight_redirect_blocked.html"]
+
+["test_mixed_principals.html"]
+skip-if = [
+ "os == 'android'", # bug 1309814, android(bug 1232305)
+ "apple_silicon", # bug 1707737
+]
+
+["test_mp3_broadcast.html"]
+
+["test_multiple_mediastreamtracks.html"]
+scheme = "https"
+
+["test_play_events_2.html"]
+
+["test_playback.html"]
+skip-if = ["os == 'android'"] # bug 1316177
+
+["test_playback_and_bfcache.html"]
+support-files = ["file_playback_and_bfcache.html"]
+
+["test_playback_hls.html"]
+# HLS is only supported on Fennec with API level >= 16
+# TODO: This test is similar to test_playback.html, will remove the
+# redundant code once test_playback.html is enabled on Fennec.
+skip-if = ["os != 'android'"]
+tags = "hls"
+
+["test_playback_rate.html"]
+
+["test_played.html"]
+skip-if = [
+ "os == 'android' && is_emulator", # Times out on android-em, Bug 1613946
+ "os == 'mac'", # Bug 1110922
+ "os == 'win'", # Bug 1110922
+ "os == 'android' && processor == 'aarch64'", # Bug 1110922
+]
+
+["test_replay_metadata.html"]
+
+["test_reset_events_async.html"]
+
+["test_resolution_change.html"]
+tags = "capturestream"
+
+["test_resume.html"]
+skip-if = ["true"] # bug 1021673
+
+["test_seamless_looping.html"]
+skip-if = [
+ "apple_catalina && debug", # Bug 1796318
+]
+
+["test_seamless_looping_cancel_looping_future_frames.html"]
+
+["test_seamless_looping_duration.html"]
+
+["test_seamless_looping_media_element_state.html"]
+
+["test_seamless_looping_not_keep_painting_old_video_frames.html"]
+skip-if = ["os == 'android'"] # Android has black frames issue.
+
+["test_seamless_looping_resume_video_decoding.html"]
+
+["test_seamless_looping_seek_current_time.html"]
+
+["test_seamless_looping_shorter_audio_than_video.html"]
+skip-if = ["os == 'android' && verify"] # Crash
+
+["test_seamless_looping_video.html"]
+skip-if = ["os == 'android'"] # Android has black frames issue.
+
+["test_setSinkId_after_loop.html"]
+scheme = "https"
+
+["test_temporary_file_blob_video_plays.html"]
+skip-if = ["os == 'android'"] # bug 1533534 # android(bug 1232305)
+
+["test_videoPlaybackQuality_totalFrames.html"]
+skip-if = ["os == 'win'"] # bug 1374189
+
+["test_video_dimensions.html"]
+
+["test_video_low_power_telemetry.html"]
+run-if = ["os == 'mac'"] # This telemetry is macOS-specific.
+disabled = "https://bugzilla.mozilla.org/show_bug.cgi?id=1849680"
+
+["test_video_to_canvas.html"]
+skip-if = ["os == 'android'"] # android(bug 1232305), bugs 1320418,1347953,1347954,1348140,1348386
+
+["test_vp9_superframes.html"]
diff --git a/dom/media/test/mochitest_background_video.toml b/dom/media/test/mochitest_background_video.toml
new file mode 100644
index 0000000000..762a9a8648
--- /dev/null
+++ b/dom/media/test/mochitest_background_video.toml
@@ -0,0 +1,800 @@
+# Media tests should be backend independent, i.e., not conditioned on ogg,
+# wave etc. (The only exception is the can_play_type tests, which
+# necessarily depend on the backend(s) configured.) As far as possible, each
+# test should work with any resource type. This makes it easy to add new
+# backends and reduces the amount of test duplication.
+
+# For each supported backend, resources that can be played by that backend
+# should be added to the lists in manifest.js. Media tests that aren't
+# testing for a bug in handling a specific resource type should pick one of
+# the lists in manifest.js and run the test for each resource in the list
+# that is supported in the current build (the canPlayType API is useful for
+# this).
+
+# To test whether a valid resource can simply be played through correctly,
+# and optionally that its metadata is read correctly, just add it to
+# gPlayTests in manifest.js. To test whether an invalid resource correctly
+# throws an error (and does not cause a crash or hang), just add it to
+# gErrorTests in manifest.js.
+
+# To test for a specific bug in handling a specific resource type, make the
+# test first check canPlayType for the type, and if it's not supported, just
+# do ok(true, "Type not supported") and stop the test.
+
+[DEFAULT]
+subsuite = "media"
+tags = "suspend media-gpu"
+support-files = [
+ "16bit_wave_extrametadata.wav",
+ "16bit_wave_extrametadata.wav^headers^",
+ "320x240.ogv",
+ "320x240.ogv^headers^",
+ "448636.ogv",
+ "448636.ogv^headers^",
+ "A4.ogv",
+ "A4.ogv^headers^",
+ "VID_0001.ogg",
+ "VID_0001.ogg^headers^",
+ "allowed.sjs",
+ "ambisonics.mp4",
+ "ambisonics.mp4^headers^",
+ "audio-gaps.ogg",
+ "audio-gaps.ogg^headers^",
+ "audio-gaps-short.ogg",
+ "audio-gaps-short.ogg^headers^",
+ "audio-overhang.ogg",
+ "audio-overhang.ogg^headers^",
+ "audio.wav",
+ "audio.wav^headers^",
+ "av1.mp4",
+ "av1.mp4^headers^",
+ "background_video.js",
+ "badtags.ogg",
+ "badtags.ogg^headers^",
+ "bear-640x360-v_frag-cenc-key_rotation.mp4",
+ "bear-640x360-a_frag-cenc-key_rotation.mp4",
+ "beta-phrasebook.ogg",
+ "beta-phrasebook.ogg^headers^",
+ "big.wav",
+ "big.wav^headers^",
+ "big-buck-bunny-cenc-avc3-1.m4s",
+ "big-buck-bunny-cenc-avc3-1.m4s^headers^",
+ "big-buck-bunny-cenc-avc3-init.mp4",
+ "big-buck-bunny-cenc-avc3-init.mp4^headers^",
+ "big-short.wav",
+ "big-short.wav^headers^",
+ "bipbop.mp4",
+ "bipbop-cenc-audio1.m4s",
+ "bipbop-cenc-audio1.m4s^headers^",
+ "bipbop-cenc-audio2.m4s",
+ "bipbop-cenc-audio2.m4s^headers^",
+ "bipbop-cenc-audio3.m4s",
+ "bipbop-cenc-audio3.m4s^headers^",
+ "bipbop-cenc-audioinit.mp4",
+ "bipbop-cenc-audioinit.mp4^headers^",
+ "bipbop-cenc-video1.m4s",
+ "bipbop-cenc-video1.m4s^headers^",
+ "bipbop-cenc-video2.m4s",
+ "bipbop-cenc-video2.m4s^headers^",
+ "bipbop-cenc-videoinit.mp4",
+ "bipbop-cenc-videoinit.mp4^headers^",
+ "bipbop-cenc-video-10s.mp4",
+ "bipbop-cenc-video-10s.mp4^headers^",
+ "bipbop-clearkey-keyrotation-clear-lead-audio.mp4",
+ "bipbop-clearkey-keyrotation-clear-lead-audio.mp4^headers^",
+ "bipbop-clearkey-keyrotation-clear-lead-video.mp4",
+ "bipbop-clearkey-keyrotation-clear-lead-video.mp4^headers^",
+ "bipbop_225w_175kbps.mp4",
+ "bipbop_225w_175kbps.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-1.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-2.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-3.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-4.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-init.mp4",
+ "bipbop_225w_175kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-1.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-2.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-3.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-4.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-init.mp4",
+ "bipbop_225w_175kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-video-key1-1.m4s",
+ "bipbop_225w_175kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-video-key1-init.mp4",
+ "bipbop_225w_175kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-video-key2-1.m4s",
+ "bipbop_225w_175kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-video-key2-init.mp4",
+ "bipbop_225w_175kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-1.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-2.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-3.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-4.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-1.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-2.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-3.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-4.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-init.mp4",
+ "bipbop_300_215kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-1.m4s",
+ "bipbop_300_215kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-2.m4s",
+ "bipbop_300_215kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-1.m4s",
+ "bipbop_300_215kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-2.m4s",
+ "bipbop_300_215kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-init.mp4",
+ "bipbop_300_215kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-1.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-2.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-3.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-4.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-init.mp4",
+ "bipbop_300wp_227kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-1.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-2.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-3.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-4.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-init.mp4",
+ "bipbop_300wp_227kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-1.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-2.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-init.mp4",
+ "bipbop_300wp_227kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-1.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-2.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-init.mp4",
+ "bipbop_300wp_227kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-1.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-2.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-3.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-4.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-init.mp4",
+ "bipbop_360w_253kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-1.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-2.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-3.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-4.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-init.mp4",
+ "bipbop_360w_253kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-video-key1-1.m4s",
+ "bipbop_360w_253kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-video-key1-init.mp4",
+ "bipbop_360w_253kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-video-key2-1.m4s",
+ "bipbop_360w_253kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-video-key2-init.mp4",
+ "bipbop_360w_253kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-clearkey-audio.webm",
+ "bipbop_360w_253kbps-clearkey-audio.webm^headers^",
+ "bipbop_360w_253kbps-clearkey-video-vp8.webm",
+ "bipbop_360w_253kbps-clearkey-video-vp8.webm^headers^",
+ "bipbop_360w_253kbps-clearkey-video-vp9.webm",
+ "bipbop_360w_253kbps-clearkey-video-vp9.webm^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480_624kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-2.m4s",
+ "bipbop_480_624kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-1.m4s",
+ "bipbop_480_624kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-2.m4s",
+ "bipbop_480_624kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480_959kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480_959kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-1.m4s",
+ "bipbop_480_959kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-2.m4s",
+ "bipbop_480_959kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-init.mp4",
+ "bipbop_480_959kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-1.m4s",
+ "bipbop_480_959kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-2.m4s",
+ "bipbop_480_959kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-init.mp4",
+ "bipbop_480_959kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480wp_663kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480wp_663kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-1.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-2.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-init.mp4",
+ "bipbop_480wp_663kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-1.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-2.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-init.mp4",
+ "bipbop_480wp_663kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_audio_aac_8k.mp4",
+ "bipbop_audio_aac_8k.mp4^headers^",
+ "bipbop_audio_aac_22.05k.mp4",
+ "bipbop_audio_aac_22.05k.mp4^headers^",
+ "bipbop_audio_aac_44.1k.mp4",
+ "bipbop_audio_aac_44.1k.mp4^headers^",
+ "bipbop_audio_aac_48k.mp4",
+ "bipbop_audio_aac_48k.mp4^headers^",
+ "bipbop_audio_aac_88.2k.mp4",
+ "bipbop_audio_aac_88.2k.mp4^headers^",
+ "bipbop_audio_aac_96k.mp4",
+ "bipbop_audio_aac_96k.mp4^headers^",
+ "bipbop_cbcs_1_9_audio_1.m4s",
+ "bipbop_cbcs_1_9_audio_1.m4s^headers^",
+ "bipbop_cbcs_1_9_audio_init.mp4",
+ "bipbop_cbcs_1_9_audio_init.mp4^headers^",
+ "bipbop_cbcs_1_9_video_1.m4s",
+ "bipbop_cbcs_1_9_video_1.m4s^headers^",
+ "bipbop_cbcs_1_9_video_init.mp4",
+ "bipbop_cbcs_1_9_video_init.mp4^headers^",
+ "bipbop_cbcs_5_5_audio_1.m4s",
+ "bipbop_cbcs_5_5_audio_1.m4s^headers^",
+ "bipbop_cbcs_5_5_audio_init.mp4",
+ "bipbop_cbcs_5_5_audio_init.mp4^headers^",
+ "bipbop_cbcs_5_5_video_1.m4s",
+ "bipbop_cbcs_5_5_video_1.m4s^headers^",
+ "bipbop_cbcs_5_5_video_init.mp4",
+ "bipbop_cbcs_5_5_video_init.mp4^headers^",
+ "bipbop_cbcs_7_7_audio_1.m4s",
+ "bipbop_cbcs_7_7_audio_1.m4s^headers^",
+ "bipbop_cbcs_7_7_audio_init.mp4",
+ "bipbop_cbcs_7_7_audio_init.mp4^headers^",
+ "bipbop_cbcs_7_7_video_1.m4s",
+ "bipbop_cbcs_7_7_video_1.m4s^headers^",
+ "bipbop_cbcs_7_7_video_init.mp4",
+ "bipbop_cbcs_7_7_video_init.mp4^headers^",
+ "bipbop_cbcs_9_8_audio_1.m4s",
+ "bipbop_cbcs_9_8_audio_1.m4s^headers^",
+ "bipbop_cbcs_9_8_audio_init.mp4",
+ "bipbop_cbcs_9_8_audio_init.mp4^headers^",
+ "bipbop_cbcs_9_8_video_1.m4s",
+ "bipbop_cbcs_9_8_video_1.m4s^headers^",
+ "bipbop_cbcs_9_8_video_init.mp4",
+ "bipbop_cbcs_9_8_video_init.mp4^headers^",
+ "bipbop_cbcs_10_0_audio_1.m4s",
+ "bipbop_cbcs_10_0_audio_1.m4s^headers^",
+ "bipbop_cbcs_10_0_audio_init.mp4",
+ "bipbop_cbcs_10_0_audio_init.mp4^headers^",
+ "bipbop_cbcs_10_0_video_1.m4s",
+ "bipbop_cbcs_10_0_video_1.m4s^headers^",
+ "bipbop_cbcs_10_0_video_init.mp4",
+ "bipbop_cbcs_10_0_video_init.mp4^headers^",
+ "bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm",
+ "bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm",
+ "bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm",
+ "bipbop_short_vp8.webm",
+ "bipbop_short_vp8.webm^headers^",
+ "bipbop-lateaudio.mp4",
+ "bipbop-lateaudio.mp4^headers^",
+ "black100x100-aspect3to2.ogv",
+ "black100x100-aspect3to2.ogv^headers^",
+ "bogus.duh",
+ "bogus.ogv",
+ "bogus.ogv^headers^",
+ "bogus.wav",
+ "bogus.wav^headers^",
+ "bug461281.ogg",
+ "bug461281.ogg^headers^",
+ "bug482461-theora.ogv",
+ "bug482461-theora.ogv^headers^",
+ "bug482461.ogv",
+ "bug482461.ogv^headers^",
+ "bug495129.ogv",
+ "bug495129.ogv^headers^",
+ "bug495794.ogg",
+ "bug495794.ogg^headers^",
+ "bug498380.ogv",
+ "bug498380.ogv^headers^",
+ "bug498855-1.ogv",
+ "bug498855-1.ogv^headers^",
+ "bug498855-2.ogv",
+ "bug498855-2.ogv^headers^",
+ "bug498855-3.ogv",
+ "bug498855-3.ogv^headers^",
+ "bug499519.ogv",
+ "bug499519.ogv^headers^",
+ "bug500311.ogv",
+ "bug500311.ogv^headers^",
+ "bug501279.ogg",
+ "bug501279.ogg^headers^",
+ "bug504613.ogv",
+ "bug504613.ogv^headers^",
+ "bug504644.ogv",
+ "bug504644.ogv^headers^",
+ "bug504843.ogv",
+ "bug504843.ogv^headers^",
+ "bug506094.ogv",
+ "bug506094.ogv^headers^",
+ "bug516323.indexed.ogv",
+ "bug516323.indexed.ogv^headers^",
+ "bug516323.ogv",
+ "bug516323.ogv^headers^",
+ "bug520493.ogg",
+ "bug520493.ogg^headers^",
+ "bug520500.ogg",
+ "bug520500.ogg^headers^",
+ "bug520908.ogv",
+ "bug520908.ogv^headers^",
+ "bug523816.ogv",
+ "bug523816.ogv^headers^",
+ "bug533822.ogg",
+ "bug533822.ogg^headers^",
+ "bug556821.ogv",
+ "bug556821.ogv^headers^",
+ "bug557094.ogv",
+ "bug557094.ogv^headers^",
+ "bug604067.webm",
+ "bug604067.webm^headers^",
+ "bug1066943.webm",
+ "bug1066943.webm^headers^",
+ "bug1301226.wav",
+ "bug1301226.wav^headers^",
+ "bug1301226-odd.wav",
+ "bug1301226-odd.wav^headers^",
+ "bug1377278.webm",
+ "bug1377278.webm^headers^",
+ "bunny.webm",
+ "can_play_type_dash.js",
+ "can_play_type_ogg.js",
+ "can_play_type_wave.js",
+ "can_play_type_webm.js",
+ "cancellable_request.sjs",
+ "chain.ogg",
+ "chain.ogg^headers^",
+ "chain.ogv",
+ "chain.ogv^headers^",
+ "chain.opus",
+ "chain.opus^headers^",
+ "chained-audio-video.ogg",
+ "chained-audio-video.ogg^headers^",
+ "chained-video.ogv",
+ "chained-video.ogv^headers^",
+ "chromeHelper.js",
+ "cloneElementVisually_helpers.js",
+ "contentType.sjs",
+ "detodos.opus",
+ "detodos.opus^headers^",
+ "detodos.webm",
+ "detodos.webm^headers^",
+ "detodos-short.webm",
+ "detodos-short.webm^headers^",
+ "detodos-recorder-test.opus",
+ "detodos-recorder-test.opus^headers^",
+ "detodos-short.opus",
+ "detodos-short.opus^headers^",
+ "dirac.ogg",
+ "dirac.ogg^headers^",
+ "dynamic_resource.sjs",
+ "eme_standalone.js",
+ "eme.js",
+ "empty_size.mp3",
+ "file_access_controls.html",
+ "file_eme_createMediaKeys.html",
+ "flac-s24.flac",
+ "flac-s24.flac^headers^",
+ "flac-noheader-s16.flac",
+ "flac-noheader-s16.flac^headers^",
+ "flac-sample.mp4",
+ "flac-sample.mp4^headers^",
+ "flac-sample-cenc.mp4",
+ "flac-sample-cenc.mp4^headers^",
+ "fragment_noplay.js",
+ "fragment_play.js",
+ "gizmo.mp4",
+ "gizmo.mp4^headers^",
+ "gizmo-noaudio.mp4",
+ "gizmo-noaudio.mp4^headers^",
+ "gizmo-short.mp4",
+ "gizmo-short.mp4^headers^",
+ "gizmo.webm",
+ "gizmo.webm^headers^",
+ "gizmo-noaudio.webm",
+ "gizmo-noaudio.webm^headers^",
+ "gUM_support.js",
+ "gzipped_mp4.sjs",
+ "huge-id3.mp3",
+ "huge-id3.mp3^headers^",
+ "id3tags.mp3",
+ "id3tags.mp3^headers^",
+ "invalid-cmap-s0c0.opus",
+ "invalid-cmap-s0c0.opus^headers^",
+ "invalid-cmap-s0c2.opus",
+ "invalid-cmap-s0c2.opus^headers^",
+ "invalid-cmap-s1c2.opus",
+ "invalid-cmap-s1c2.opus^headers^",
+ "invalid-cmap-short.opus",
+ "invalid-cmap-short.opus^headers^",
+ "invalid-discard_on_multi_blocks.webm",
+ "invalid-discard_on_multi_blocks.webm^headers^",
+ "invalid-excess_discard.webm",
+ "invalid-excess_discard.webm^headers^",
+ "invalid-excess_neg_discard.webm",
+ "invalid-excess_neg_discard.webm^headers^",
+ "invalid-m0c0.opus",
+ "invalid-m0c0.opus^headers^",
+ "invalid-m0c3.opus",
+ "invalid-m0c3.opus^headers^",
+ "invalid-m1c0.opus",
+ "invalid-m1c0.opus^headers^",
+ "invalid-m1c9.opus",
+ "invalid-m1c9.opus^headers^",
+ "invalid-m2c0.opus",
+ "invalid-m2c0.opus^headers^",
+ "invalid-m2c1.opus",
+ "invalid-m2c1.opus^headers^",
+ "invalid-neg_discard.webm",
+ "invalid-neg_discard.webm^headers^",
+ "invalid-preskip.webm",
+ "invalid-preskip.webm^headers^",
+ "manifest.js",
+ "midflight-redirect.sjs",
+ "multiple-bos.ogg",
+ "multiple-bos.ogg^headers^",
+ "multiple-bos-more-header-fileds.ogg",
+ "multiple-bos-more-header-fileds.ogg^headers^",
+ "multi_id3v2.mp3",
+ "no-container-codec-delay.webm",
+ "no-cues.webm",
+ "no-cues.webm^headers^",
+ "notags.mp3",
+ "notags.mp3^headers^",
+ "opus-mapping2.mp4",
+ "opus-mapping2.mp4^headers^",
+ "opus-mapping2.webm",
+ "opus-mapping2.webm^headers^",
+ "opus-sample.mp4",
+ "opus-sample.mp4^headers^",
+ "opus-sample-cenc.mp4",
+ "opus-sample-cenc.mp4^headers^",
+ "owl-funnier-id3.mp3",
+ "owl-funnier-id3.mp3^headers^",
+ "owl-funny-id3.mp3",
+ "owl-funny-id3.mp3^headers^",
+ "owl.mp3",
+ "owl.mp3^headers^",
+ "owl-short.mp3",
+ "owl-short.mp3^headers^",
+ "pixel_aspect_ratio.mp4",
+ "play_promise.js",
+ "poster-test.jpg",
+ "r11025_msadpcm_c1.wav",
+ "r11025_msadpcm_c1.wav^headers^",
+ "r11025_s16_c1.wav",
+ "r11025_s16_c1.wav^headers^",
+ "r11025_s16_c1_trailing.wav",
+ "r11025_s16_c1_trailing.wav^headers^",
+ "r11025_s16_c1-short.wav",
+ "r11025_s16_c1-short.wav^headers^",
+ "r11025_u8_c1.wav",
+ "r11025_u8_c1.wav^headers^",
+ "r11025_u8_c1_trunc.wav",
+ "r11025_u8_c1_trunc.wav^headers^",
+ "r16000_u8_c1_list.wav",
+ "r16000_u8_c1_list.wav^headers^",
+ "reactivate_helper.html",
+ "red-46x48.mp4",
+ "red-46x48.mp4^headers^",
+ "red-48x46.mp4",
+ "red-48x46.mp4^headers^",
+ "redirect.sjs",
+ "referer.sjs",
+ "resolution-change.webm",
+ "resolution-change.webm^headers^",
+ "sample.3gp",
+ "sample.3g2",
+ "sample-encrypted-sgpdstbl-sbgptraf.mp4",
+ "sample-encrypted-sgpdstbl-sbgptraf.mp4^headers^",
+ "sample-fisbone-skeleton4.ogv",
+ "sample-fisbone-skeleton4.ogv^headers^",
+ "sample-fisbone-wrong-header.ogv",
+ "sample-fisbone-wrong-header.ogv^headers^",
+ "seek.ogv",
+ "seek.ogv^headers^",
+ "seek-short.ogv",
+ "seek-short.ogv^headers^",
+ "seek.webm",
+ "seek.webm^headers^",
+ "seek-short.webm",
+ "seek-short.webm^headers^",
+ "seek_support.js",
+ "seekLies.sjs",
+ "seek_with_sound.ogg^headers^",
+ "short-cenc.mp4",
+ "sine.webm",
+ "sine.webm^headers^",
+ "sintel-short-clearkey-subsample-encrypted-audio.webm",
+ "sintel-short-clearkey-subsample-encrypted-audio.webm^headers^",
+ "sintel-short-clearkey-subsample-encrypted-video.webm",
+ "sintel-short-clearkey-subsample-encrypted-video.webm^headers^",
+ "short.mp4",
+ "short.mp4.gz",
+ "short.mp4^headers^",
+ "short-aac-encrypted-audio.mp4",
+ "short-aac-encrypted-audio.mp4^headers^",
+ "short-audio-fragmented-cenc-without-pssh.mp4",
+ "short-audio-fragmented-cenc-without-pssh.mp4^headers^",
+ "short-video.ogv",
+ "short-video.ogv^headers^",
+ "short-vp9-encrypted-video.mp4",
+ "short-vp9-encrypted-video.mp4^headers^",
+ "small-shot-mp3.mp4",
+ "small-shot-mp3.mp4^headers^",
+ "small-shot.m4a",
+ "small-shot.mp3",
+ "small-shot.mp3^headers^",
+ "small-shot.ogg",
+ "small-shot.ogg^headers^",
+ "small-shot.flac",
+ "sound.ogg",
+ "sound.ogg^headers^",
+ "spacestorm-1000Hz-100ms.ogg",
+ "spacestorm-1000Hz-100ms.ogg^headers^",
+ "split.webm",
+ "split.webm^headers^",
+ "street.mp4",
+ "street.mp4^headers^",
+ "test-1-mono.opus",
+ "test-1-mono.opus^headers^",
+ "test-2-stereo.opus",
+ "test-2-stereo.opus^headers^",
+ "test-3-LCR.opus",
+ "test-3-LCR.opus^headers^",
+ "test-4-quad.opus",
+ "test-4-quad.opus^headers^",
+ "test-5-5.0.opus",
+ "test-5-5.0.opus^headers^",
+ "test-6-5.1.opus",
+ "test-6-5.1.opus^headers^",
+ "test-7-6.1.opus",
+ "test-7-6.1.opus^headers^",
+ "test-8-7.1.opus",
+ "test-8-7.1.opus^headers^",
+ "test-stereo-phase-inversion-180.opus",
+ "test-stereo-phase-inversion-180.opus^headers^",
+ "variable-channel.ogg",
+ "variable-channel.ogg^headers^",
+ "variable-channel.opus",
+ "variable-channel.opus^headers^",
+ "variable-preskip.opus",
+ "variable-preskip.opus^headers^",
+ "variable-samplerate.ogg",
+ "variable-samplerate.ogg^headers^",
+ "variable-samplerate.opus",
+ "variable-samplerate.opus^headers^",
+ "vbr-head.mp3",
+ "vbr-head.mp3^headers^",
+ "vbr.mp3",
+ "vbr.mp3^headers^",
+ "very-short.mp3",
+ "video-overhang.ogg",
+ "video-overhang.ogg^headers^",
+ "vp9-superframes.webm",
+ "vp9-superframes.webm^headers^",
+ "vp9.webm",
+ "vp9.webm^headers^",
+ "vp9-short.webm",
+ "vp9-short.webm^headers^",
+ "vp9cake.webm",
+ "vp9cake.webm^headers^",
+ "vp9cake-short.webm",
+ "vp9cake-short.webm^headers^",
+ "wave_metadata.wav",
+ "wave_metadata.wav^headers^",
+ "wave_metadata_bad_len.wav",
+ "wave_metadata_bad_len.wav^headers^",
+ "wave_metadata_bad_no_null.wav",
+ "wave_metadata_bad_no_null.wav^headers^",
+ "wave_metadata_bad_utf8.wav",
+ "wave_metadata_bad_utf8.wav^headers^",
+ "wave_metadata_unknown_tag.wav",
+ "wave_metadata_unknown_tag.wav^headers^",
+ "wave_metadata_utf8.wav",
+ "wave_metadata_utf8.wav^headers^",
+ "wavedata_alaw.wav",
+ "wavedata_alaw.wav^headers^",
+ "wavedata_float.wav",
+ "wavedata_float.wav^headers^",
+ "wavedata_s24.wav",
+ "wavedata_s24.wav^headers^",
+ "wavedata_s16.wav",
+ "wavedata_s16.wav^headers^",
+ "wavedata_u8.wav",
+ "wavedata_u8.wav^headers^",
+ "wavedata_ulaw.wav",
+ "wavedata_ulaw.wav^headers^",
+ "!/dom/canvas/test/captureStream_common.js",
+ "!/dom/html/test/reflect.js",
+ "!/dom/media/webrtc/tests/mochitests/head.js",
+ "hls/bipbop_16x9_single.m3u8",
+ "hls/bipbop_4x3_single.m3u8",
+ "hls/bipbop_4x3_variant.m3u8",
+ "hls/400x300_prog_index.m3u8",
+ "hls/400x300_prog_index_5s.m3u8",
+ "hls/416x243_prog_index_5s.m3u8",
+ "hls/640x480_prog_index.m3u8",
+ "hls/960x720_prog_index.m3u8",
+ "hls/400x300_seg0.ts",
+ "hls/400x300_seg0_5s.ts",
+ "hls/400x300_seg1.ts",
+ "hls/416x243_seg0_5s.ts",
+ "hls/640x480_seg0.ts",
+ "hls/640x480_seg1.ts",
+ "hls/960x720_seg0.ts",
+ "hls/960x720_seg1.ts",
+ "sync.webm",
+]
+
+["test_background_video_cancel_suspend_taint.html"]
+skip-if = [
+ "os == 'android'", # bug 1346705
+ "os == 'mac' && debug", # Bug 1670131 - after splitting manifest perma fail
+]
+
+["test_background_video_cancel_suspend_visible.html"]
+
+["test_background_video_drawimage_with_suspended_video.html"]
+skip-if = ["os == 'android'"] # bug 1346705
+
+["test_background_video_ended_event.html"]
+skip-if = ["os == 'android'"] # bug 1346705
+
+["test_background_video_no_suspend_disabled.html"]
+
+["test_background_video_no_suspend_not_in_tree.html"]
+
+["test_background_video_no_suspend_short_vid.html"]
+
+["test_background_video_resume_after_end_show_last_frame.html"]
+skip-if = ["os == 'android'"] # bug 1346705
+
+["test_background_video_resume_looping_video_without_audio.html"]
+
+["test_background_video_suspend.html"]
+skip-if = ["os == 'android'"] #Bug 1304480
+
+["test_background_video_suspend_ends.html"]
+
+["test_background_video_suspend_ready_state.html"]
+
+["test_background_video_tainted_by_capturestream.html"]
+
+["test_background_video_tainted_by_createimagebitmap.html"]
+
+["test_background_video_tainted_by_drawimage.html"]
+skip-if = ["os == 'android'"] # bug 1346705
diff --git a/dom/media/test/mochitest_bugs.toml b/dom/media/test/mochitest_bugs.toml
new file mode 100644
index 0000000000..9d4a0dea2f
--- /dev/null
+++ b/dom/media/test/mochitest_bugs.toml
@@ -0,0 +1,812 @@
+# Media tests should be backend independent, i.e., not conditioned on ogg,
+# wave etc. (The only exception is the can_play_type tests, which
+# necessarily depend on the backend(s) configured.) As far as possible, each
+# test should work with any resource type. This makes it easy to add new
+# backends and reduces the amount of test duplication.
+
+# For each supported backend, resources that can be played by that backend
+# should be added to the lists in manifest.js. Media tests that aren't
+# testing for a bug in handling a specific resource type should pick one of
+# the lists in manifest.js and run the test for each resource in the list
+# that is supported in the current build (the canPlayType API is useful for
+# this).
+
+# To test whether a valid resource can simply be played through correctly,
+# and optionally that its metadata is read correctly, just add it to
+# gPlayTests in manifest.js. To test whether an invalid resource correctly
+# throws an error (and does not cause a crash or hang), just add it to
+# gErrorTests in manifest.js.
+
+# To test for a specific bug in handling a specific resource type, make the
+# test first check canPlayType for the type, and if it's not supported, just
+# do ok(true, "Type not supported") and stop the test.
+
+[DEFAULT]
+subsuite = "media"
+tags = "media-gpu"
+support-files = [
+ "16bit_wave_extrametadata.wav",
+ "16bit_wave_extrametadata.wav^headers^",
+ "320x240.ogv",
+ "320x240.ogv^headers^",
+ "448636.ogv",
+ "448636.ogv^headers^",
+ "A4.ogv",
+ "A4.ogv^headers^",
+ "VID_0001.ogg",
+ "VID_0001.ogg^headers^",
+ "allowed.sjs",
+ "ambisonics.mp4",
+ "ambisonics.mp4^headers^",
+ "audio-gaps.ogg",
+ "audio-gaps.ogg^headers^",
+ "audio-gaps-short.ogg",
+ "audio-gaps-short.ogg^headers^",
+ "audio-overhang.ogg",
+ "audio-overhang.ogg^headers^",
+ "audio.wav",
+ "audio.wav^headers^",
+ "av1.mp4",
+ "av1.mp4^headers^",
+ "background_video.js",
+ "badtags.ogg",
+ "badtags.ogg^headers^",
+ "bear-640x360-v_frag-cenc-key_rotation.mp4",
+ "bear-640x360-a_frag-cenc-key_rotation.mp4",
+ "beta-phrasebook.ogg",
+ "beta-phrasebook.ogg^headers^",
+ "big.wav",
+ "big.wav^headers^",
+ "big-buck-bunny-cenc-avc3-1.m4s",
+ "big-buck-bunny-cenc-avc3-1.m4s^headers^",
+ "big-buck-bunny-cenc-avc3-init.mp4",
+ "big-buck-bunny-cenc-avc3-init.mp4^headers^",
+ "big-short.wav",
+ "big-short.wav^headers^",
+ "bipbop.mp4",
+ "bipbop-cenc-audio1.m4s",
+ "bipbop-cenc-audio1.m4s^headers^",
+ "bipbop-cenc-audio2.m4s",
+ "bipbop-cenc-audio2.m4s^headers^",
+ "bipbop-cenc-audio3.m4s",
+ "bipbop-cenc-audio3.m4s^headers^",
+ "bipbop-cenc-audioinit.mp4",
+ "bipbop-cenc-audioinit.mp4^headers^",
+ "bipbop-cenc-video1.m4s",
+ "bipbop-cenc-video1.m4s^headers^",
+ "bipbop-cenc-video2.m4s",
+ "bipbop-cenc-video2.m4s^headers^",
+ "bipbop-cenc-videoinit.mp4",
+ "bipbop-cenc-videoinit.mp4^headers^",
+ "bipbop-cenc-video-10s.mp4",
+ "bipbop-cenc-video-10s.mp4^headers^",
+ "bipbop-clearkey-keyrotation-clear-lead-audio.mp4",
+ "bipbop-clearkey-keyrotation-clear-lead-audio.mp4^headers^",
+ "bipbop-clearkey-keyrotation-clear-lead-video.mp4",
+ "bipbop-clearkey-keyrotation-clear-lead-video.mp4^headers^",
+ "bipbop_225w_175kbps.mp4",
+ "bipbop_225w_175kbps.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-1.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-2.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-3.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-4.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-init.mp4",
+ "bipbop_225w_175kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-1.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-2.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-3.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-4.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-init.mp4",
+ "bipbop_225w_175kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-video-key1-1.m4s",
+ "bipbop_225w_175kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-video-key1-init.mp4",
+ "bipbop_225w_175kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-video-key2-1.m4s",
+ "bipbop_225w_175kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-video-key2-init.mp4",
+ "bipbop_225w_175kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-1.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-2.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-3.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-4.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-1.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-2.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-3.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-4.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-init.mp4",
+ "bipbop_300_215kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-1.m4s",
+ "bipbop_300_215kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-2.m4s",
+ "bipbop_300_215kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-1.m4s",
+ "bipbop_300_215kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-2.m4s",
+ "bipbop_300_215kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-init.mp4",
+ "bipbop_300_215kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-1.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-2.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-3.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-4.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-init.mp4",
+ "bipbop_300wp_227kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-1.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-2.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-3.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-4.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-init.mp4",
+ "bipbop_300wp_227kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-1.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-2.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-init.mp4",
+ "bipbop_300wp_227kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-1.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-2.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-init.mp4",
+ "bipbop_300wp_227kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-1.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-2.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-3.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-4.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-init.mp4",
+ "bipbop_360w_253kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-1.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-2.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-3.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-4.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-init.mp4",
+ "bipbop_360w_253kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-video-key1-1.m4s",
+ "bipbop_360w_253kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-video-key1-init.mp4",
+ "bipbop_360w_253kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-video-key2-1.m4s",
+ "bipbop_360w_253kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-video-key2-init.mp4",
+ "bipbop_360w_253kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-clearkey-audio.webm",
+ "bipbop_360w_253kbps-clearkey-audio.webm^headers^",
+ "bipbop_360w_253kbps-clearkey-video-vp8.webm",
+ "bipbop_360w_253kbps-clearkey-video-vp8.webm^headers^",
+ "bipbop_360w_253kbps-clearkey-video-vp9.webm",
+ "bipbop_360w_253kbps-clearkey-video-vp9.webm^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480_624kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-2.m4s",
+ "bipbop_480_624kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-1.m4s",
+ "bipbop_480_624kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-2.m4s",
+ "bipbop_480_624kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480_959kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480_959kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-1.m4s",
+ "bipbop_480_959kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-2.m4s",
+ "bipbop_480_959kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-init.mp4",
+ "bipbop_480_959kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-1.m4s",
+ "bipbop_480_959kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-2.m4s",
+ "bipbop_480_959kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-init.mp4",
+ "bipbop_480_959kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480wp_663kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480wp_663kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-1.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-2.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-init.mp4",
+ "bipbop_480wp_663kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-1.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-2.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-init.mp4",
+ "bipbop_480wp_663kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_audio_aac_8k.mp4",
+ "bipbop_audio_aac_8k.mp4^headers^",
+ "bipbop_audio_aac_22.05k.mp4",
+ "bipbop_audio_aac_22.05k.mp4^headers^",
+ "bipbop_audio_aac_44.1k.mp4",
+ "bipbop_audio_aac_44.1k.mp4^headers^",
+ "bipbop_audio_aac_48k.mp4",
+ "bipbop_audio_aac_48k.mp4^headers^",
+ "bipbop_audio_aac_88.2k.mp4",
+ "bipbop_audio_aac_88.2k.mp4^headers^",
+ "bipbop_audio_aac_96k.mp4",
+ "bipbop_audio_aac_96k.mp4^headers^",
+ "bipbop_cbcs_1_9_audio_1.m4s",
+ "bipbop_cbcs_1_9_audio_1.m4s^headers^",
+ "bipbop_cbcs_1_9_audio_init.mp4",
+ "bipbop_cbcs_1_9_audio_init.mp4^headers^",
+ "bipbop_cbcs_1_9_video_1.m4s",
+ "bipbop_cbcs_1_9_video_1.m4s^headers^",
+ "bipbop_cbcs_1_9_video_init.mp4",
+ "bipbop_cbcs_1_9_video_init.mp4^headers^",
+ "bipbop_cbcs_5_5_audio_1.m4s",
+ "bipbop_cbcs_5_5_audio_1.m4s^headers^",
+ "bipbop_cbcs_5_5_audio_init.mp4",
+ "bipbop_cbcs_5_5_audio_init.mp4^headers^",
+ "bipbop_cbcs_5_5_video_1.m4s",
+ "bipbop_cbcs_5_5_video_1.m4s^headers^",
+ "bipbop_cbcs_5_5_video_init.mp4",
+ "bipbop_cbcs_5_5_video_init.mp4^headers^",
+ "bipbop_cbcs_7_7_audio_1.m4s",
+ "bipbop_cbcs_7_7_audio_1.m4s^headers^",
+ "bipbop_cbcs_7_7_audio_init.mp4",
+ "bipbop_cbcs_7_7_audio_init.mp4^headers^",
+ "bipbop_cbcs_7_7_video_1.m4s",
+ "bipbop_cbcs_7_7_video_1.m4s^headers^",
+ "bipbop_cbcs_7_7_video_init.mp4",
+ "bipbop_cbcs_7_7_video_init.mp4^headers^",
+ "bipbop_cbcs_9_8_audio_1.m4s",
+ "bipbop_cbcs_9_8_audio_1.m4s^headers^",
+ "bipbop_cbcs_9_8_audio_init.mp4",
+ "bipbop_cbcs_9_8_audio_init.mp4^headers^",
+ "bipbop_cbcs_9_8_video_1.m4s",
+ "bipbop_cbcs_9_8_video_1.m4s^headers^",
+ "bipbop_cbcs_9_8_video_init.mp4",
+ "bipbop_cbcs_9_8_video_init.mp4^headers^",
+ "bipbop_cbcs_10_0_audio_1.m4s",
+ "bipbop_cbcs_10_0_audio_1.m4s^headers^",
+ "bipbop_cbcs_10_0_audio_init.mp4",
+ "bipbop_cbcs_10_0_audio_init.mp4^headers^",
+ "bipbop_cbcs_10_0_video_1.m4s",
+ "bipbop_cbcs_10_0_video_1.m4s^headers^",
+ "bipbop_cbcs_10_0_video_init.mp4",
+ "bipbop_cbcs_10_0_video_init.mp4^headers^",
+ "bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm",
+ "bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm",
+ "bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm",
+ "bipbop_short_vp8.webm",
+ "bipbop_short_vp8.webm^headers^",
+ "bipbop-lateaudio.mp4",
+ "bipbop-lateaudio.mp4^headers^",
+ "black100x100-aspect3to2.ogv",
+ "black100x100-aspect3to2.ogv^headers^",
+ "bogus.duh",
+ "bogus.ogv",
+ "bogus.ogv^headers^",
+ "bogus.wav",
+ "bogus.wav^headers^",
+ "bug461281.ogg",
+ "bug461281.ogg^headers^",
+ "bug482461-theora.ogv",
+ "bug482461-theora.ogv^headers^",
+ "bug482461.ogv",
+ "bug482461.ogv^headers^",
+ "bug495129.ogv",
+ "bug495129.ogv^headers^",
+ "bug495794.ogg",
+ "bug495794.ogg^headers^",
+ "bug498380.ogv",
+ "bug498380.ogv^headers^",
+ "bug498855-1.ogv",
+ "bug498855-1.ogv^headers^",
+ "bug498855-2.ogv",
+ "bug498855-2.ogv^headers^",
+ "bug498855-3.ogv",
+ "bug498855-3.ogv^headers^",
+ "bug499519.ogv",
+ "bug499519.ogv^headers^",
+ "bug500311.ogv",
+ "bug500311.ogv^headers^",
+ "bug501279.ogg",
+ "bug501279.ogg^headers^",
+ "bug504613.ogv",
+ "bug504613.ogv^headers^",
+ "bug504644.ogv",
+ "bug504644.ogv^headers^",
+ "bug504843.ogv",
+ "bug504843.ogv^headers^",
+ "bug506094.ogv",
+ "bug506094.ogv^headers^",
+ "bug516323.indexed.ogv",
+ "bug516323.indexed.ogv^headers^",
+ "bug516323.ogv",
+ "bug516323.ogv^headers^",
+ "bug520493.ogg",
+ "bug520493.ogg^headers^",
+ "bug520500.ogg",
+ "bug520500.ogg^headers^",
+ "bug520908.ogv",
+ "bug520908.ogv^headers^",
+ "bug523816.ogv",
+ "bug523816.ogv^headers^",
+ "bug533822.ogg",
+ "bug533822.ogg^headers^",
+ "bug556821.ogv",
+ "bug556821.ogv^headers^",
+ "bug557094.ogv",
+ "bug557094.ogv^headers^",
+ "bug604067.webm",
+ "bug604067.webm^headers^",
+ "bug1066943.webm",
+ "bug1066943.webm^headers^",
+ "bug1301226.wav",
+ "bug1301226.wav^headers^",
+ "bug1301226-odd.wav",
+ "bug1301226-odd.wav^headers^",
+ "bug1377278.webm",
+ "bug1377278.webm^headers^",
+ "bug1799787.webm",
+ "bug1799787.webm^headers^",
+ "bunny.webm",
+ "can_play_type_dash.js",
+ "can_play_type_ogg.js",
+ "can_play_type_wave.js",
+ "can_play_type_webm.js",
+ "cancellable_request.sjs",
+ "chain.ogg",
+ "chain.ogg^headers^",
+ "chain.ogv",
+ "chain.ogv^headers^",
+ "chain.opus",
+ "chain.opus^headers^",
+ "chained-audio-video.ogg",
+ "chained-audio-video.ogg^headers^",
+ "chained-video.ogv",
+ "chained-video.ogv^headers^",
+ "chromeHelper.js",
+ "cloneElementVisually_helpers.js",
+ "contentType.sjs",
+ "detodos.opus",
+ "detodos.opus^headers^",
+ "detodos.webm",
+ "detodos.webm^headers^",
+ "detodos-short.webm",
+ "detodos-short.webm^headers^",
+ "detodos-recorder-test.opus",
+ "detodos-recorder-test.opus^headers^",
+ "detodos-short.opus",
+ "detodos-short.opus^headers^",
+ "dirac.ogg",
+ "dirac.ogg^headers^",
+ "dynamic_resource.sjs",
+ "eme_standalone.js",
+ "eme.js",
+ "empty_size.mp3",
+ "file_access_controls.html",
+ "file_eme_createMediaKeys.html",
+ "flac-s24.flac",
+ "flac-s24.flac^headers^",
+ "flac-noheader-s16.flac",
+ "flac-noheader-s16.flac^headers^",
+ "flac-sample.mp4",
+ "flac-sample.mp4^headers^",
+ "flac-sample-cenc.mp4",
+ "flac-sample-cenc.mp4^headers^",
+ "fragment_noplay.js",
+ "fragment_play.js",
+ "gizmo.mp4",
+ "gizmo.mp4^headers^",
+ "gizmo-noaudio.mp4",
+ "gizmo-noaudio.mp4^headers^",
+ "gizmo-short.mp4",
+ "gizmo-short.mp4^headers^",
+ "gizmo.webm",
+ "gizmo.webm^headers^",
+ "gizmo-noaudio.webm",
+ "gizmo-noaudio.webm^headers^",
+ "gUM_support.js",
+ "gzipped_mp4.sjs",
+ "huge-id3.mp3",
+ "huge-id3.mp3^headers^",
+ "id3tags.mp3",
+ "id3tags.mp3^headers^",
+ "invalid-cmap-s0c0.opus",
+ "invalid-cmap-s0c0.opus^headers^",
+ "invalid-cmap-s0c2.opus",
+ "invalid-cmap-s0c2.opus^headers^",
+ "invalid-cmap-s1c2.opus",
+ "invalid-cmap-s1c2.opus^headers^",
+ "invalid-cmap-short.opus",
+ "invalid-cmap-short.opus^headers^",
+ "invalid-discard_on_multi_blocks.webm",
+ "invalid-discard_on_multi_blocks.webm^headers^",
+ "invalid-excess_discard.webm",
+ "invalid-excess_discard.webm^headers^",
+ "invalid-excess_neg_discard.webm",
+ "invalid-excess_neg_discard.webm^headers^",
+ "invalid-m0c0.opus",
+ "invalid-m0c0.opus^headers^",
+ "invalid-m0c3.opus",
+ "invalid-m0c3.opus^headers^",
+ "invalid-m1c0.opus",
+ "invalid-m1c0.opus^headers^",
+ "invalid-m1c9.opus",
+ "invalid-m1c9.opus^headers^",
+ "invalid-m2c0.opus",
+ "invalid-m2c0.opus^headers^",
+ "invalid-m2c1.opus",
+ "invalid-m2c1.opus^headers^",
+ "invalid-neg_discard.webm",
+ "invalid-neg_discard.webm^headers^",
+ "invalid-preskip.webm",
+ "invalid-preskip.webm^headers^",
+ "manifest.js",
+ "midflight-redirect.sjs",
+ "multiple-bos.ogg",
+ "multiple-bos.ogg^headers^",
+ "multiple-bos-more-header-fileds.ogg",
+ "multiple-bos-more-header-fileds.ogg^headers^",
+ "multi_id3v2.mp3",
+ "no-container-codec-delay.webm",
+ "no-cues.webm",
+ "no-cues.webm^headers^",
+ "notags.mp3",
+ "notags.mp3^headers^",
+ "opus-mapping2.mp4",
+ "opus-mapping2.mp4^headers^",
+ "opus-mapping2.webm",
+ "opus-mapping2.webm^headers^",
+ "opus-sample.mp4",
+ "opus-sample.mp4^headers^",
+ "opus-sample-cenc.mp4",
+ "opus-sample-cenc.mp4^headers^",
+ "owl-funnier-id3.mp3",
+ "owl-funnier-id3.mp3^headers^",
+ "owl-funny-id3.mp3",
+ "owl-funny-id3.mp3^headers^",
+ "owl.mp3",
+ "owl.mp3^headers^",
+ "owl-short.mp3",
+ "owl-short.mp3^headers^",
+ "pixel_aspect_ratio.mp4",
+ "play_promise.js",
+ "poster-test.jpg",
+ "r11025_msadpcm_c1.wav",
+ "r11025_msadpcm_c1.wav^headers^",
+ "r11025_s16_c1.wav",
+ "r11025_s16_c1.wav^headers^",
+ "r11025_s16_c1_trailing.wav",
+ "r11025_s16_c1_trailing.wav^headers^",
+ "r11025_s16_c1-short.wav",
+ "r11025_s16_c1-short.wav^headers^",
+ "r11025_u8_c1.wav",
+ "r11025_u8_c1.wav^headers^",
+ "r11025_u8_c1_trunc.wav",
+ "r11025_u8_c1_trunc.wav^headers^",
+ "r16000_u8_c1_list.wav",
+ "r16000_u8_c1_list.wav^headers^",
+ "reactivate_helper.html",
+ "red-46x48.mp4",
+ "red-46x48.mp4^headers^",
+ "red-48x46.mp4",
+ "red-48x46.mp4^headers^",
+ "redirect.sjs",
+ "referer.sjs",
+ "resolution-change.webm",
+ "resolution-change.webm^headers^",
+ "sample.3gp",
+ "sample.3g2",
+ "sample-encrypted-sgpdstbl-sbgptraf.mp4",
+ "sample-encrypted-sgpdstbl-sbgptraf.mp4^headers^",
+ "sample-fisbone-skeleton4.ogv",
+ "sample-fisbone-skeleton4.ogv^headers^",
+ "sample-fisbone-wrong-header.ogv",
+ "sample-fisbone-wrong-header.ogv^headers^",
+ "seek.ogv",
+ "seek.ogv^headers^",
+ "seek-short.ogv",
+ "seek-short.ogv^headers^",
+ "seek.webm",
+ "seek.webm^headers^",
+ "seek-short.webm",
+ "seek-short.webm^headers^",
+ "seek_support.js",
+ "seekLies.sjs",
+ "seek_with_sound.ogg^headers^",
+ "short-cenc.mp4",
+ "sine.webm",
+ "sine.webm^headers^",
+ "sintel-short-clearkey-subsample-encrypted-audio.webm",
+ "sintel-short-clearkey-subsample-encrypted-audio.webm^headers^",
+ "sintel-short-clearkey-subsample-encrypted-video.webm",
+ "sintel-short-clearkey-subsample-encrypted-video.webm^headers^",
+ "short.mp4",
+ "short.mp4.gz",
+ "short.mp4^headers^",
+ "short-aac-encrypted-audio.mp4",
+ "short-aac-encrypted-audio.mp4^headers^",
+ "short-audio-fragmented-cenc-without-pssh.mp4",
+ "short-audio-fragmented-cenc-without-pssh.mp4^headers^",
+ "short-video.ogv",
+ "short-video.ogv^headers^",
+ "short-vp9-encrypted-video.mp4",
+ "short-vp9-encrypted-video.mp4^headers^",
+ "small-shot-mp3.mp4",
+ "small-shot-mp3.mp4^headers^",
+ "small-shot.m4a",
+ "small-shot.mp3",
+ "small-shot.mp3^headers^",
+ "small-shot.ogg",
+ "small-shot.ogg^headers^",
+ "small-shot.flac",
+ "sound.ogg",
+ "sound.ogg^headers^",
+ "spacestorm-1000Hz-100ms.ogg",
+ "spacestorm-1000Hz-100ms.ogg^headers^",
+ "split.webm",
+ "split.webm^headers^",
+ "street.mp4",
+ "street.mp4^headers^",
+ "test-1-mono.opus",
+ "test-1-mono.opus^headers^",
+ "test-2-stereo.opus",
+ "test-2-stereo.opus^headers^",
+ "test-3-LCR.opus",
+ "test-3-LCR.opus^headers^",
+ "test-4-quad.opus",
+ "test-4-quad.opus^headers^",
+ "test-5-5.0.opus",
+ "test-5-5.0.opus^headers^",
+ "test-6-5.1.opus",
+ "test-6-5.1.opus^headers^",
+ "test-7-6.1.opus",
+ "test-7-6.1.opus^headers^",
+ "test-8-7.1.opus",
+ "test-8-7.1.opus^headers^",
+ "test-stereo-phase-inversion-180.opus",
+ "test-stereo-phase-inversion-180.opus^headers^",
+ "variable-channel.ogg",
+ "variable-channel.ogg^headers^",
+ "variable-channel.opus",
+ "variable-channel.opus^headers^",
+ "variable-preskip.opus",
+ "variable-preskip.opus^headers^",
+ "variable-samplerate.ogg",
+ "variable-samplerate.ogg^headers^",
+ "variable-samplerate.opus",
+ "variable-samplerate.opus^headers^",
+ "vbr-head.mp3",
+ "vbr-head.mp3^headers^",
+ "vbr.mp3",
+ "vbr.mp3^headers^",
+ "very-short.mp3",
+ "video-overhang.ogg",
+ "video-overhang.ogg^headers^",
+ "vp9-superframes.webm",
+ "vp9-superframes.webm^headers^",
+ "vp9.webm",
+ "vp9.webm^headers^",
+ "vp9-short.webm",
+ "vp9-short.webm^headers^",
+ "vp9cake.webm",
+ "vp9cake.webm^headers^",
+ "vp9cake-short.webm",
+ "vp9cake-short.webm^headers^",
+ "wave_metadata.wav",
+ "wave_metadata.wav^headers^",
+ "wave_metadata_bad_len.wav",
+ "wave_metadata_bad_len.wav^headers^",
+ "wave_metadata_bad_no_null.wav",
+ "wave_metadata_bad_no_null.wav^headers^",
+ "wave_metadata_bad_utf8.wav",
+ "wave_metadata_bad_utf8.wav^headers^",
+ "wave_metadata_unknown_tag.wav",
+ "wave_metadata_unknown_tag.wav^headers^",
+ "wave_metadata_utf8.wav",
+ "wave_metadata_utf8.wav^headers^",
+ "wavedata_alaw.wav",
+ "wavedata_alaw.wav^headers^",
+ "wavedata_float.wav",
+ "wavedata_float.wav^headers^",
+ "wavedata_s24.wav",
+ "wavedata_s24.wav^headers^",
+ "wavedata_s16.wav",
+ "wavedata_s16.wav^headers^",
+ "wavedata_u8.wav",
+ "wavedata_u8.wav^headers^",
+ "wavedata_ulaw.wav",
+ "wavedata_ulaw.wav^headers^",
+ "!/dom/canvas/test/captureStream_common.js",
+ "!/dom/html/test/reflect.js",
+ "!/dom/media/webrtc/tests/mochitests/head.js",
+ "hls/bipbop_16x9_single.m3u8",
+ "hls/bipbop_4x3_single.m3u8",
+ "hls/bipbop_4x3_variant.m3u8",
+ "hls/400x300_prog_index.m3u8",
+ "hls/400x300_prog_index_5s.m3u8",
+ "hls/416x243_prog_index_5s.m3u8",
+ "hls/640x480_prog_index.m3u8",
+ "hls/960x720_prog_index.m3u8",
+ "hls/400x300_seg0.ts",
+ "hls/400x300_seg0_5s.ts",
+ "hls/400x300_seg1.ts",
+ "hls/416x243_seg0_5s.ts",
+ "hls/640x480_seg0.ts",
+ "hls/640x480_seg1.ts",
+ "hls/960x720_seg0.ts",
+ "hls/960x720_seg1.ts",
+ "sync.webm",
+]
+
+["test_bug448534.html"]
+
+["test_bug463162.xhtml"]
+
+["test_bug465498.html"]
+
+["test_bug495145.html"]
+skip-if = ["os == 'win'"] #Bug 1404373
+
+["test_bug495300.html"]
+
+["test_bug654550.html"]
+
+["test_bug686942.html"]
+
+["test_bug726904.html"]
+
+["test_bug874897.html"]
+
+["test_bug879717.html"]
+skip-if = ["os == 'android'"] # bug 1285441, android(bug 1232305)
+tags = "capturestream"
+
+["test_bug895305.html"]
+skip-if = ["android_version == '25' && debug"] # android(bug 1232305)
+
+["test_bug919265.html"]
+skip-if = ["android_version == '25' && debug"] # android(bug 1232305)
+
+["test_bug1113600.html"]
+skip-if = [
+ "os == 'win' && os_version == '10.0' && debug", # Bug 1713410
+ "os == 'mac'", # Bug 1198168
+]
+
+["test_bug1120222.html"]
+tags = "capturestream"
+
+["test_bug1242338.html"]
+
+["test_bug1248229.html"]
+tags = "capturestream"
+
+["test_bug1512958.html"]
+tags = "mtg capturestream"
+
+["test_bug1553262.html"]
+tags = "mtg capturestream"
diff --git a/dom/media/test/mochitest_compat.toml b/dom/media/test/mochitest_compat.toml
new file mode 100644
index 0000000000..c477661a19
--- /dev/null
+++ b/dom/media/test/mochitest_compat.toml
@@ -0,0 +1,1033 @@
+# This file lists tests which are compatible for testing Windows Media
+# Foundation media engine playback. These tests would still be testing on
+# different platforms, but we will use these tests to do additional testing on
+# Windows to test the media engine playback.
+
+# --------------------------------------------------------------------------
+
+# Media tests should be backend independent, i.e., not conditioned on ogg,
+# wave etc. (The only exception is the can_play_type tests, which
+# necessarily depend on the backend(s) configured.) As far as possible, each
+# test should work with any resource type. This makes it easy to add new
+# backends and reduces the amount of test duplication.
+
+# For each supported backend, resources that can be played by that backend
+# should be added to the lists in manifest.js. Media tests that aren't
+# testing for a bug in handling a specific resource type should pick one of
+# the lists in manifest.js and run the test for each resource in the list
+# that is supported in the current build (the canPlayType API is useful for
+# this).
+
+# To test whether a valid resource can simply be played through correctly,
+# and optionally that its metadata is read correctly, just add it to
+# gPlayTests in manifest.js. To test whether an invalid resource correctly
+# throws an error (and does not cause a crash or hang), just add it to
+# gErrorTests in manifest.js.
+
+# To test for a specific bug in handling a specific resource type, make the
+# test first check canPlayType for the type, and if it's not supported, just
+# do ok(true, "Type not supported") and stop the test.
+
+[DEFAULT]
+subsuite = "media"
+tags = "media-engine-compatible media-gpu"
+prefs = ["media.wmf.hevc.enabled=1"] # for test_hevc_playback
+support-files = [
+ "16bit_wave_extrametadata.wav",
+ "16bit_wave_extrametadata.wav^headers^",
+ "320x240.ogv",
+ "320x240.ogv^headers^",
+ "448636.ogv",
+ "448636.ogv^headers^",
+ "A4.ogv",
+ "A4.ogv^headers^",
+ "VID_0001.ogg",
+ "VID_0001.ogg^headers^",
+ "adts.aac",
+ "adts.aac^headers^",
+ "allowed.sjs",
+ "ambisonics.mp4",
+ "ambisonics.mp4^headers^",
+ "audio-gaps.ogg",
+ "audio-gaps.ogg^headers^",
+ "audio-gaps-short.ogg",
+ "audio-gaps-short.ogg^headers^",
+ "audio-overhang.ogg",
+ "audio-overhang.ogg^headers^",
+ "audio.wav",
+ "audio.wav^headers^",
+ "av1.mp4",
+ "av1.mp4^headers^",
+ "background_video.js",
+ "badtags.ogg",
+ "badtags.ogg^headers^",
+ "bear-640x360-v_frag-cenc-key_rotation.mp4",
+ "bear-640x360-a_frag-cenc-key_rotation.mp4",
+ "beta-phrasebook.ogg",
+ "beta-phrasebook.ogg^headers^",
+ "big.wav",
+ "big.wav^headers^",
+ "big-buck-bunny-cenc-avc3-1.m4s",
+ "big-buck-bunny-cenc-avc3-1.m4s^headers^",
+ "big-buck-bunny-cenc-avc3-init.mp4",
+ "big-buck-bunny-cenc-avc3-init.mp4^headers^",
+ "big-short.wav",
+ "big-short.wav^headers^",
+ "bipbop.mp4",
+ "bipbop-cenc-audio1.m4s",
+ "bipbop-cenc-audio1.m4s^headers^",
+ "bipbop-cenc-audio2.m4s",
+ "bipbop-cenc-audio2.m4s^headers^",
+ "bipbop-cenc-audio3.m4s",
+ "bipbop-cenc-audio3.m4s^headers^",
+ "bipbop-cenc-audioinit.mp4",
+ "bipbop-cenc-audioinit.mp4^headers^",
+ "bipbop-cenc-video1.m4s",
+ "bipbop-cenc-video1.m4s^headers^",
+ "bipbop-cenc-video2.m4s",
+ "bipbop-cenc-video2.m4s^headers^",
+ "bipbop-cenc-videoinit.mp4",
+ "bipbop-cenc-videoinit.mp4^headers^",
+ "bipbop-cenc-video-10s.mp4",
+ "bipbop-cenc-video-10s.mp4^headers^",
+ "bipbop-clearkey-keyrotation-clear-lead-audio.mp4",
+ "bipbop-clearkey-keyrotation-clear-lead-audio.mp4^headers^",
+ "bipbop-clearkey-keyrotation-clear-lead-video.mp4",
+ "bipbop-clearkey-keyrotation-clear-lead-video.mp4^headers^",
+ "bipbop_225w_175kbps.mp4",
+ "bipbop_225w_175kbps.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-1.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-2.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-3.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-4.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-init.mp4",
+ "bipbop_225w_175kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-1.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-2.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-3.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-4.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-init.mp4",
+ "bipbop_225w_175kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-video-key1-1.m4s",
+ "bipbop_225w_175kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-video-key1-init.mp4",
+ "bipbop_225w_175kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-video-key2-1.m4s",
+ "bipbop_225w_175kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-video-key2-init.mp4",
+ "bipbop_225w_175kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-1.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-2.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-3.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-4.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-1.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-2.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-3.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-4.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-init.mp4",
+ "bipbop_300_215kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-1.m4s",
+ "bipbop_300_215kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-2.m4s",
+ "bipbop_300_215kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-1.m4s",
+ "bipbop_300_215kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-2.m4s",
+ "bipbop_300_215kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-init.mp4",
+ "bipbop_300_215kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-1.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-2.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-3.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-4.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-init.mp4",
+ "bipbop_300wp_227kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-1.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-2.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-3.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-4.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-init.mp4",
+ "bipbop_300wp_227kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-1.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-2.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-init.mp4",
+ "bipbop_300wp_227kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-1.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-2.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-init.mp4",
+ "bipbop_300wp_227kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-1.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-2.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-3.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-4.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-init.mp4",
+ "bipbop_360w_253kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-1.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-2.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-3.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-4.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-init.mp4",
+ "bipbop_360w_253kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-video-key1-1.m4s",
+ "bipbop_360w_253kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-video-key1-init.mp4",
+ "bipbop_360w_253kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-video-key2-1.m4s",
+ "bipbop_360w_253kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-video-key2-init.mp4",
+ "bipbop_360w_253kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-clearkey-audio.webm",
+ "bipbop_360w_253kbps-clearkey-audio.webm^headers^",
+ "bipbop_360w_253kbps-clearkey-video-vp8.webm",
+ "bipbop_360w_253kbps-clearkey-video-vp8.webm^headers^",
+ "bipbop_360w_253kbps-clearkey-video-vp9.webm",
+ "bipbop_360w_253kbps-clearkey-video-vp9.webm^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480_624kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-2.m4s",
+ "bipbop_480_624kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-1.m4s",
+ "bipbop_480_624kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-2.m4s",
+ "bipbop_480_624kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480_959kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480_959kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-1.m4s",
+ "bipbop_480_959kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-2.m4s",
+ "bipbop_480_959kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-init.mp4",
+ "bipbop_480_959kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-1.m4s",
+ "bipbop_480_959kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-2.m4s",
+ "bipbop_480_959kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-init.mp4",
+ "bipbop_480_959kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480wp_663kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480wp_663kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-1.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-2.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-init.mp4",
+ "bipbop_480wp_663kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-1.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-2.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-init.mp4",
+ "bipbop_480wp_663kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_audio_aac_8k.mp4",
+ "bipbop_audio_aac_8k.mp4^headers^",
+ "bipbop_audio_aac_22.05k.mp4",
+ "bipbop_audio_aac_22.05k.mp4^headers^",
+ "bipbop_audio_aac_44.1k.mp4",
+ "bipbop_audio_aac_44.1k.mp4^headers^",
+ "bipbop_audio_aac_48k.mp4",
+ "bipbop_audio_aac_48k.mp4^headers^",
+ "bipbop_audio_aac_88.2k.mp4",
+ "bipbop_audio_aac_88.2k.mp4^headers^",
+ "bipbop_audio_aac_96k.mp4",
+ "bipbop_audio_aac_96k.mp4^headers^",
+ "bipbop_cbcs_1_9_audio_1.m4s",
+ "bipbop_cbcs_1_9_audio_1.m4s^headers^",
+ "bipbop_cbcs_1_9_audio_init.mp4",
+ "bipbop_cbcs_1_9_audio_init.mp4^headers^",
+ "bipbop_cbcs_1_9_video_1.m4s",
+ "bipbop_cbcs_1_9_video_1.m4s^headers^",
+ "bipbop_cbcs_1_9_video_init.mp4",
+ "bipbop_cbcs_1_9_video_init.mp4^headers^",
+ "bipbop_cbcs_5_5_audio_1.m4s",
+ "bipbop_cbcs_5_5_audio_1.m4s^headers^",
+ "bipbop_cbcs_5_5_audio_init.mp4",
+ "bipbop_cbcs_5_5_audio_init.mp4^headers^",
+ "bipbop_cbcs_5_5_video_1.m4s",
+ "bipbop_cbcs_5_5_video_1.m4s^headers^",
+ "bipbop_cbcs_5_5_video_init.mp4",
+ "bipbop_cbcs_5_5_video_init.mp4^headers^",
+ "bipbop_cbcs_7_7_audio_1.m4s",
+ "bipbop_cbcs_7_7_audio_1.m4s^headers^",
+ "bipbop_cbcs_7_7_audio_init.mp4",
+ "bipbop_cbcs_7_7_audio_init.mp4^headers^",
+ "bipbop_cbcs_7_7_video_1.m4s",
+ "bipbop_cbcs_7_7_video_1.m4s^headers^",
+ "bipbop_cbcs_7_7_video_init.mp4",
+ "bipbop_cbcs_7_7_video_init.mp4^headers^",
+ "bipbop_cbcs_9_8_audio_1.m4s",
+ "bipbop_cbcs_9_8_audio_1.m4s^headers^",
+ "bipbop_cbcs_9_8_audio_init.mp4",
+ "bipbop_cbcs_9_8_audio_init.mp4^headers^",
+ "bipbop_cbcs_9_8_video_1.m4s",
+ "bipbop_cbcs_9_8_video_1.m4s^headers^",
+ "bipbop_cbcs_9_8_video_init.mp4",
+ "bipbop_cbcs_9_8_video_init.mp4^headers^",
+ "bipbop_cbcs_10_0_audio_1.m4s",
+ "bipbop_cbcs_10_0_audio_1.m4s^headers^",
+ "bipbop_cbcs_10_0_audio_init.mp4",
+ "bipbop_cbcs_10_0_audio_init.mp4^headers^",
+ "bipbop_cbcs_10_0_video_1.m4s",
+ "bipbop_cbcs_10_0_video_1.m4s^headers^",
+ "bipbop_cbcs_10_0_video_init.mp4",
+ "bipbop_cbcs_10_0_video_init.mp4^headers^",
+ "bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm",
+ "bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm",
+ "bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm",
+ "bipbop_short_vp8.webm",
+ "bipbop_short_vp8.webm^headers^",
+ "bipbop-lateaudio.mp4",
+ "bipbop-lateaudio.mp4^headers^",
+ "black100x100-aspect3to2.ogv",
+ "black100x100-aspect3to2.ogv^headers^",
+ "bogus.duh",
+ "bogus.ogv",
+ "bogus.ogv^headers^",
+ "bogus.wav",
+ "bogus.wav^headers^",
+ "bug461281.ogg",
+ "bug461281.ogg^headers^",
+ "bug482461-theora.ogv",
+ "bug482461-theora.ogv^headers^",
+ "bug482461.ogv",
+ "bug482461.ogv^headers^",
+ "bug495129.ogv",
+ "bug495129.ogv^headers^",
+ "bug495794.ogg",
+ "bug495794.ogg^headers^",
+ "bug498380.ogv",
+ "bug498380.ogv^headers^",
+ "bug498855-1.ogv",
+ "bug498855-1.ogv^headers^",
+ "bug498855-2.ogv",
+ "bug498855-2.ogv^headers^",
+ "bug498855-3.ogv",
+ "bug498855-3.ogv^headers^",
+ "bug499519.ogv",
+ "bug499519.ogv^headers^",
+ "bug500311.ogv",
+ "bug500311.ogv^headers^",
+ "bug501279.ogg",
+ "bug501279.ogg^headers^",
+ "bug504613.ogv",
+ "bug504613.ogv^headers^",
+ "bug504644.ogv",
+ "bug504644.ogv^headers^",
+ "bug504843.ogv",
+ "bug504843.ogv^headers^",
+ "bug506094.ogv",
+ "bug506094.ogv^headers^",
+ "bug516323.indexed.ogv",
+ "bug516323.indexed.ogv^headers^",
+ "bug516323.ogv",
+ "bug516323.ogv^headers^",
+ "bug520493.ogg",
+ "bug520493.ogg^headers^",
+ "bug520500.ogg",
+ "bug520500.ogg^headers^",
+ "bug520908.ogv",
+ "bug520908.ogv^headers^",
+ "bug523816.ogv",
+ "bug523816.ogv^headers^",
+ "bug533822.ogg",
+ "bug533822.ogg^headers^",
+ "bug556821.ogv",
+ "bug556821.ogv^headers^",
+ "bug557094.ogv",
+ "bug557094.ogv^headers^",
+ "bug604067.webm",
+ "bug604067.webm^headers^",
+ "bug1066943.webm",
+ "bug1066943.webm^headers^",
+ "bug1301226.wav",
+ "bug1301226.wav^headers^",
+ "bug1301226-odd.wav",
+ "bug1301226-odd.wav^headers^",
+ "bug1377278.webm",
+ "bug1377278.webm^headers^",
+ "bunny.webm",
+ "can_play_type_dash.js",
+ "can_play_type_ogg.js",
+ "can_play_type_wave.js",
+ "can_play_type_webm.js",
+ "cancellable_request.sjs",
+ "chain.ogg",
+ "chain.ogg^headers^",
+ "chain.ogv",
+ "chain.ogv^headers^",
+ "chain.opus",
+ "chain.opus^headers^",
+ "chained-audio-video.ogg",
+ "chained-audio-video.ogg^headers^",
+ "chained-video.ogv",
+ "chained-video.ogv^headers^",
+ "chromeHelper.js",
+ "cloneElementVisually_helpers.js",
+ "contentType.sjs",
+ "detodos.opus",
+ "detodos.opus^headers^",
+ "detodos.webm",
+ "detodos.webm^headers^",
+ "detodos-short.webm",
+ "detodos-short.webm^headers^",
+ "detodos-recorder-test.opus",
+ "detodos-recorder-test.opus^headers^",
+ "detodos-short.opus",
+ "detodos-short.opus^headers^",
+ "dirac.ogg",
+ "dirac.ogg^headers^",
+ "dynamic_resource.sjs",
+ "eme_standalone.js",
+ "eme.js",
+ "empty_size.mp3",
+ "file_access_controls.html",
+ "file_eme_createMediaKeys.html",
+ "flac-s24.flac",
+ "flac-s24.flac^headers^",
+ "flac-noheader-s16.flac",
+ "flac-noheader-s16.flac^headers^",
+ "flac-sample.mp4",
+ "flac-sample.mp4^headers^",
+ "flac-sample-cenc.mp4",
+ "flac-sample-cenc.mp4^headers^",
+ "fragment_noplay.js",
+ "fragment_play.js",
+ "gizmo.mp4",
+ "gizmo.mp4^headers^",
+ "gizmo-noaudio.mp4",
+ "gizmo-noaudio.mp4^headers^",
+ "gizmo-short.mp4",
+ "gizmo-short.mp4^headers^",
+ "gizmo.webm",
+ "gizmo.webm^headers^",
+ "gizmo-noaudio.webm",
+ "gizmo-noaudio.webm^headers^",
+ "gUM_support.js",
+ "gzipped_mp4.sjs",
+ "hevc_white_frame.mp4",
+ "hevc_white_frame.mp4^headers^",
+ # This HEVC file is created by following steps.
+ # 1. generate two videos
+ # - a white frame video (0-2 seconds) at 1024x768 resolution
+ # ffmpeg -t 2 -f lavfi -i color=size=1024x768:rate=30:color=white -c:v libx265 -pix_fmt yuv420p -strict experimental -metadata:s:v:0 rotate=0 -metadata:s:v:0 width=1024 -metadata:s:v:0 height=768 -y white_frames.mp4
+ # - a red frame video (2-4 seconds) at 3480x2160 resolution with inband SPS
+ # ffmpeg -t 2 -f lavfi -i color=size=3480x2160:rate=30:color=red -c:v libx265 -x265-params "sps-id=1" -strict experimental -metadata:s:v:0 rotate=0 -metadata:s:v:0 width=3480 -metadata:s:v:0 height=2160 -y red_frames.mp4
+ # 2. covert them to annexB
+ # ffmpeg -i XXX.mp4 -codec copy -bsf:v h264_mp4toannexb XXX.ts
+ # 3. combine them together
+ # ffmpeg -i "concat:white_annexb.ts|red_annexb.ts" -c copy -strict -2 -y hevc_white_red_frames.mp4
+ "hevc_white_red_frames.mp4",
+ "hevc_white_red_frames.mp4^headers^",
+ "huge-id3.mp3",
+ "huge-id3.mp3^headers^",
+ "id3tags.mp3",
+ "id3tags.mp3^headers^",
+ "invalid-cmap-s0c0.opus",
+ "invalid-cmap-s0c0.opus^headers^",
+ "invalid-cmap-s0c2.opus",
+ "invalid-cmap-s0c2.opus^headers^",
+ "invalid-cmap-s1c2.opus",
+ "invalid-cmap-s1c2.opus^headers^",
+ "invalid-cmap-short.opus",
+ "invalid-cmap-short.opus^headers^",
+ "invalid-discard_on_multi_blocks.webm",
+ "invalid-discard_on_multi_blocks.webm^headers^",
+ "invalid-excess_discard.webm",
+ "invalid-excess_discard.webm^headers^",
+ "invalid-excess_neg_discard.webm",
+ "invalid-excess_neg_discard.webm^headers^",
+ "invalid-m0c0.opus",
+ "invalid-m0c0.opus^headers^",
+ "invalid-m0c3.opus",
+ "invalid-m0c3.opus^headers^",
+ "invalid-m1c0.opus",
+ "invalid-m1c0.opus^headers^",
+ "invalid-m1c9.opus",
+ "invalid-m1c9.opus^headers^",
+ "invalid-m2c0.opus",
+ "invalid-m2c0.opus^headers^",
+ "invalid-m2c1.opus",
+ "invalid-m2c1.opus^headers^",
+ "invalid-neg_discard.webm",
+ "invalid-neg_discard.webm^headers^",
+ "invalid-preskip.webm",
+ "invalid-preskip.webm^headers^",
+ "manifest.js",
+ "midflight-redirect.sjs",
+ "multiple-bos.ogg",
+ "multiple-bos.ogg^headers^",
+ "multiple-bos-more-header-fileds.ogg",
+ "multiple-bos-more-header-fileds.ogg^headers^",
+ "multi_id3v2.mp3",
+ "no-container-codec-delay.webm",
+ "no-cues.webm",
+ "no-cues.webm^headers^",
+ "notags.mp3",
+ "notags.mp3^headers^",
+ "opus-mapping2.mp4",
+ "opus-mapping2.mp4^headers^",
+ "opus-mapping2.webm",
+ "opus-mapping2.webm^headers^",
+ "opus-sample.mp4",
+ "opus-sample.mp4^headers^",
+ "opus-sample-cenc.mp4",
+ "opus-sample-cenc.mp4^headers^",
+ "owl-funnier-id3.mp3",
+ "owl-funnier-id3.mp3^headers^",
+ "owl-funny-id3.mp3",
+ "owl-funny-id3.mp3^headers^",
+ "owl.mp3",
+ "owl.mp3^headers^",
+ "owl-short.mp3",
+ "owl-short.mp3^headers^",
+ "pixel_aspect_ratio.mp4",
+ "play_promise.js",
+ "poster-test.jpg",
+ "r11025_msadpcm_c1.wav",
+ "r11025_msadpcm_c1.wav^headers^",
+ "r11025_s16_c1.wav",
+ "r11025_s16_c1.wav^headers^",
+ "r11025_s16_c1_trailing.wav",
+ "r11025_s16_c1_trailing.wav^headers^",
+ "r11025_s16_c1-short.wav",
+ "r11025_s16_c1-short.wav^headers^",
+ "r11025_u8_c1.wav",
+ "r11025_u8_c1.wav^headers^",
+ "r11025_u8_c1_trunc.wav",
+ "r11025_u8_c1_trunc.wav^headers^",
+ "r16000_u8_c1_list.wav",
+ "r16000_u8_c1_list.wav^headers^",
+ "reactivate_helper.html",
+ "red-46x48.mp4",
+ "red-46x48.mp4^headers^",
+ "red-48x46.mp4",
+ "red-48x46.mp4^headers^",
+ "redirect.sjs",
+ "referer.sjs",
+ "resolution-change.webm",
+ "resolution-change.webm^headers^",
+ "sample.3gp",
+ "sample.3g2",
+ "sample-encrypted-sgpdstbl-sbgptraf.mp4",
+ "sample-encrypted-sgpdstbl-sbgptraf.mp4^headers^",
+ "sample-fisbone-skeleton4.ogv",
+ "sample-fisbone-skeleton4.ogv^headers^",
+ "sample-fisbone-wrong-header.ogv",
+ "sample-fisbone-wrong-header.ogv^headers^",
+ "seek.ogv",
+ "seek.ogv^headers^",
+ "seek-short.ogv",
+ "seek-short.ogv^headers^",
+ "seek.webm",
+ "seek.webm^headers^",
+ "seek-short.webm",
+ "seek-short.webm^headers^",
+ "seek_support.js",
+ "seekLies.sjs",
+ "seek_with_sound.ogg^headers^",
+ "short-cenc.mp4",
+ "sine.webm",
+ "sine.webm^headers^",
+ "sintel-short-clearkey-subsample-encrypted-audio.webm",
+ "sintel-short-clearkey-subsample-encrypted-audio.webm^headers^",
+ "sintel-short-clearkey-subsample-encrypted-video.webm",
+ "sintel-short-clearkey-subsample-encrypted-video.webm^headers^",
+ "short.mp4",
+ "short.mp4.gz",
+ "short.mp4^headers^",
+ "short-aac-encrypted-audio.mp4",
+ "short-aac-encrypted-audio.mp4^headers^",
+ "short-audio-fragmented-cenc-without-pssh.mp4",
+ "short-audio-fragmented-cenc-without-pssh.mp4^headers^",
+ "short-video.ogv",
+ "short-video.ogv^headers^",
+ "short-vp9-encrypted-video.mp4",
+ "short-vp9-encrypted-video.mp4^headers^",
+ "small-shot-mp3.mp4",
+ "small-shot-mp3.mp4^headers^",
+ "small-shot.m4a",
+ "small-shot.mp3",
+ "small-shot.mp3^headers^",
+ "small-shot.ogg",
+ "small-shot.ogg^headers^",
+ "small-shot.flac",
+ "sound.ogg",
+ "sound.ogg^headers^",
+ "spacestorm-1000Hz-100ms.ogg",
+ "spacestorm-1000Hz-100ms.ogg^headers^",
+ "split.webm",
+ "split.webm^headers^",
+ "street.mp4",
+ "street.mp4^headers^",
+ "test-1-mono.opus",
+ "test-1-mono.opus^headers^",
+ "test-2-stereo.opus",
+ "test-2-stereo.opus^headers^",
+ "test-3-LCR.opus",
+ "test-3-LCR.opus^headers^",
+ "test-4-quad.opus",
+ "test-4-quad.opus^headers^",
+ "test-5-5.0.opus",
+ "test-5-5.0.opus^headers^",
+ "test-6-5.1.opus",
+ "test-6-5.1.opus^headers^",
+ "test-7-6.1.opus",
+ "test-7-6.1.opus^headers^",
+ "test-8-7.1.opus",
+ "test-8-7.1.opus^headers^",
+ "test-stereo-phase-inversion-180.opus",
+ "test-stereo-phase-inversion-180.opus^headers^",
+ "variable-channel.ogg",
+ "variable-channel.ogg^headers^",
+ "variable-channel.opus",
+ "variable-channel.opus^headers^",
+ "variable-preskip.opus",
+ "variable-preskip.opus^headers^",
+ "variable-samplerate.ogg",
+ "variable-samplerate.ogg^headers^",
+ "variable-samplerate.opus",
+ "variable-samplerate.opus^headers^",
+ "vbr-head.mp3",
+ "vbr-head.mp3^headers^",
+ "vbr.mp3",
+ "vbr.mp3^headers^",
+ "very-short.mp3",
+ "video-overhang.ogg",
+ "video-overhang.ogg^headers^",
+ "vp9-superframes.webm",
+ "vp9-superframes.webm^headers^",
+ "vp9.webm",
+ "vp9.webm^headers^",
+ "vp9-short.webm",
+ "vp9-short.webm^headers^",
+ "vp9cake.webm",
+ "vp9cake.webm^headers^",
+ "vp9cake-short.webm",
+ "vp9cake-short.webm^headers^",
+ "wave_metadata.wav",
+ "wave_metadata.wav^headers^",
+ "wave_metadata_bad_len.wav",
+ "wave_metadata_bad_len.wav^headers^",
+ "wave_metadata_bad_no_null.wav",
+ "wave_metadata_bad_no_null.wav^headers^",
+ "wave_metadata_bad_utf8.wav",
+ "wave_metadata_bad_utf8.wav^headers^",
+ "wave_metadata_unknown_tag.wav",
+ "wave_metadata_unknown_tag.wav^headers^",
+ "wave_metadata_utf8.wav",
+ "wave_metadata_utf8.wav^headers^",
+ "wavedata_alaw.wav",
+ "wavedata_alaw.wav^headers^",
+ "wavedata_float.wav",
+ "wavedata_float.wav^headers^",
+ "wavedata_s24.wav",
+ "wavedata_s24.wav^headers^",
+ "wavedata_s16.wav",
+ "wavedata_s16.wav^headers^",
+ "wavedata_u8.wav",
+ "wavedata_u8.wav^headers^",
+ "wavedata_ulaw.wav",
+ "wavedata_ulaw.wav^headers^",
+ "!/dom/canvas/test/captureStream_common.js",
+ "!/dom/html/test/reflect.js",
+ "!/dom/media/webaudio/test/half-a-second-1ch-44100-aac-afconvert.mp4",
+ "!/dom/media/webrtc/tests/mochitests/head.js",
+ "hls/bipbop_16x9_single.m3u8",
+ "hls/bipbop_4x3_single.m3u8",
+ "hls/bipbop_4x3_variant.m3u8",
+ "hls/400x300_prog_index.m3u8",
+ "hls/400x300_prog_index_5s.m3u8",
+ "hls/416x243_prog_index_5s.m3u8",
+ "hls/640x480_prog_index.m3u8",
+ "hls/960x720_prog_index.m3u8",
+ "hls/400x300_seg0.ts",
+ "hls/400x300_seg0_5s.ts",
+ "hls/400x300_seg1.ts",
+ "hls/416x243_seg0_5s.ts",
+ "hls/640x480_seg0.ts",
+ "hls/640x480_seg1.ts",
+ "hls/960x720_seg0.ts",
+ "hls/960x720_seg1.ts",
+ "sync.webm",
+]
+
+["test_VideoPlaybackQuality.html"]
+
+["test_VideoPlaybackQuality_disabled.html"]
+
+["test_access_control.html"]
+
+["test_arraybuffer.html"]
+
+["test_aspectratio_mp4.html"]
+
+["test_audio1.html"]
+
+["test_audio2.html"]
+
+["test_audioDocumentTitle.html"]
+skip-if = ["true"] # bug 475110 - disabled since we don't play Wave files standalone
+
+["test_buffered.html"]
+
+["test_bug1431810_opus_downmix_to_mono.html"]
+
+["test_can_play_type.html"]
+skip-if = ["(android_version == '25' && debug)"] # android(bug 1232305)
+
+["test_can_play_type_mpeg.html"]
+
+["test_can_play_type_no_ogg.html"]
+skip-if = ["(android_version == '25' && debug)"] # android(bug 1232305)
+
+["test_can_play_type_ogg.html"]
+skip-if = ["(android_version == '25' && debug)"] # android(bug 1232305)
+
+["test_can_play_type_wave.html"]
+
+["test_can_play_type_webm.html"]
+
+["test_closing_connections.html"]
+
+["test_constants.html"]
+
+["test_controls.html"]
+
+["test_cueless_webm_seek-1.html"]
+
+["test_cueless_webm_seek-2.html"]
+
+["test_cueless_webm_seek-3.html"]
+
+["test_currentTime.html"]
+
+["test_debug_data_helpers.html"]
+
+["test_decode_error.html"]
+
+["test_decode_error_crossorigin.html"]
+
+["test_decoder_disable.html"]
+
+["test_defaultMuted.html"]
+
+["test_delay_load.html"]
+
+["test_duration_after_error.html"]
+
+["test_empty_resource.html"]
+
+["test_error_in_video_document.html"]
+
+["test_error_on_404.html"]
+
+["test_fragment_noplay.html"]
+
+["test_fragment_play.html"]
+
+["test_hevc_playback.html"]
+run-if = ["os == 'win'"]
+skip-if = ["!mda_gpu"]
+scheme = "https"
+
+["test_hevc_support.html"]
+skip-if = ["!mda_gpu"]
+scheme = "https"
+
+["test_info_leak.html"]
+
+["test_invalid_reject.html"]
+
+["test_invalid_seek.html"]
+
+["test_load.html"]
+
+["test_load_candidates.html"]
+
+["test_load_same_resource.html"]
+
+["test_load_source.html"]
+
+["test_load_source_empty_type.html"]
+
+["test_loop.html"]
+skip-if = ["wmfme"] # Bug 1781539
+
+["test_looping_eventsOrder.html"]
+
+["test_mediatrack_consuming_mediaresource.html"]
+
+["test_mediatrack_events.html"]
+scheme = "https"
+
+["test_mediatrack_parsing_ogg.html"]
+
+["test_metadata.html"]
+
+["test_mozHasAudio.html"]
+
+["test_mp3_with_multiple_ID3v2.html"]
+
+["test_networkState.html"]
+
+["test_new_audio.html"]
+
+["test_no_load_event.html"]
+
+["test_not_reset_playbackRate_when_removing_nonloaded_media_from_document.html"]
+
+["test_paused.html"]
+
+["test_paused_after_ended.html"]
+
+["test_periodic_timeupdate.html"]
+
+["test_play_events.html"]
+
+["test_play_promise_1.html"]
+tags = "promise-play"
+
+["test_play_promise_10.html"]
+tags = "promise-play"
+
+["test_play_promise_11.html"]
+tags = "promise-play"
+
+["test_play_promise_12.html"]
+tags = "promise-play"
+
+["test_play_promise_13.html"]
+tags = "promise-play"
+
+["test_play_promise_14.html"]
+tags = "promise-play"
+
+["test_play_promise_15.html"]
+tags = "promise-play"
+
+["test_play_promise_16.html"]
+tags = "promise-play"
+
+["test_play_promise_17.html"]
+tags = "promise-play"
+
+["test_play_promise_18.html"]
+tags = "promise-play"
+
+["test_play_promise_2.html"]
+tags = "promise-play"
+
+["test_play_promise_3.html"]
+tags = "promise-play"
+
+["test_play_promise_4.html"]
+tags = "promise-play"
+
+["test_play_promise_5.html"]
+tags = "promise-play"
+
+["test_play_promise_6.html"]
+tags = "promise-play"
+
+["test_play_promise_7.html"]
+tags = "promise-play"
+
+["test_play_promise_8.html"]
+tags = "promise-play"
+
+["test_play_promise_9.html"]
+tags = "promise-play"
+
+["test_play_twice.html"]
+skip-if = ["appname == 'seamonkey'"] # Seamonkey: Bug 598252, bug 1307337, bug 1143695
+# If encountering intermittents in test_playback.html please consider disabling
+# the individual faulting file via `manifest.js` as disabling the whole test on
+# a platform removes a lot of coverage.
+
+["test_playback_errors.html"]
+
+["test_playback_rate_playpause.html"]
+
+["test_playback_reactivate.html"]
+
+["test_preload_actions.html"]
+
+["test_preload_attribute.html"]
+
+["test_preload_suspend.html"]
+
+["test_preserve_playbackrate_after_ui_play.html"]
+
+["test_progress.html"]
+
+["test_reactivate.html"]
+skip-if = ["true"] # see bug 1319725
+
+["test_readyState.html"]
+
+["test_referer.html"]
+skip-if = ["android_version == '25' && debug"] # android(bug 1232305)
+
+["test_reset_src.html"]
+skip-if = ["verify && debug && os == 'win'"]
+
+["test_source.html"]
+
+["test_source_null.html"]
+
+["test_source_write.html"]
+
+["test_standalone.html"]
+
+["test_suspend_media_by_inactive_docshell.html"]
+
+["test_timeupdate_small_files.html"]
+
+["test_unseekable.html"]
+
+["test_videoDocumentTitle.html"]
+
+["test_video_gzip_encoding.html"]
+
+["test_video_in_audio_element.html"]
+
+["test_video_stats_resistfingerprinting.html"]
+tags = "resistfingerprinting"
+
+["test_volume.html"]
+# The tests below contain backend-specific tests. Write backend independent
+# tests rather than adding to this list.
diff --git a/dom/media/test/mochitest_eme.toml b/dom/media/test/mochitest_eme.toml
new file mode 100644
index 0000000000..696e0e947e
--- /dev/null
+++ b/dom/media/test/mochitest_eme.toml
@@ -0,0 +1,850 @@
+# Media tests should be backend independent, i.e., not conditioned on ogg,
+# wave etc. (The only exception is the can_play_type tests, which
+# necessarily depend on the backend(s) configured.) As far as possible, each
+# test should work with any resource type. This makes it easy to add new
+# backends and reduces the amount of test duplication.
+
+# For each supported backend, resources that can be played by that backend
+# should be added to the lists in manifest.js. Media tests that aren't
+# testing for a bug in handling a specific resource type should pick one of
+# the lists in manifest.js and run the test for each resource in the list
+# that is supported in the current build (the canPlayType API is useful for
+# this).
+
+# To test whether a valid resource can simply be played through correctly,
+# and optionally that its metadata is read correctly, just add it to
+# gPlayTests in manifest.js. To test whether an invalid resource correctly
+# throws an error (and does not cause a crash or hang), just add it to
+# gErrorTests in manifest.js.
+
+# To test for a specific bug in handling a specific resource type, make the
+# test first check canPlayType for the type, and if it's not supported, just
+# do ok(true, "Type not supported") and stop the test.
+
+[DEFAULT]
+subsuite = "media"
+skip-if = ["os == 'linux' && (asan || debug)"] # Bug 1476870: common fatal error (shutdown hang) on asan/debug
+support-files = [
+ "16bit_wave_extrametadata.wav",
+ "16bit_wave_extrametadata.wav^headers^",
+ "320x240.ogv",
+ "320x240.ogv^headers^",
+ "448636.ogv",
+ "448636.ogv^headers^",
+ "A4.ogv",
+ "A4.ogv^headers^",
+ "VID_0001.ogg",
+ "VID_0001.ogg^headers^",
+ "allowed.sjs",
+ "ambisonics.mp4",
+ "ambisonics.mp4^headers^",
+ "audio-gaps.ogg",
+ "audio-gaps.ogg^headers^",
+ "audio-gaps-short.ogg",
+ "audio-gaps-short.ogg^headers^",
+ "audio-overhang.ogg",
+ "audio-overhang.ogg^headers^",
+ "audio.wav",
+ "audio.wav^headers^",
+ "av1.mp4",
+ "av1.mp4^headers^",
+ "background_video.js",
+ "badtags.ogg",
+ "badtags.ogg^headers^",
+ "bear-640x360-v_frag-cenc-key_rotation.mp4",
+ "bear-640x360-a_frag-cenc-key_rotation.mp4",
+ "beta-phrasebook.ogg",
+ "beta-phrasebook.ogg^headers^",
+ "big.wav",
+ "big.wav^headers^",
+ "big-buck-bunny-cenc-avc3-1.m4s",
+ "big-buck-bunny-cenc-avc3-1.m4s^headers^",
+ "big-buck-bunny-cenc-avc3-init.mp4",
+ "big-buck-bunny-cenc-avc3-init.mp4^headers^",
+ "big-short.wav",
+ "big-short.wav^headers^",
+ "bipbop.mp4",
+ "bipbop-cenc-audio1.m4s",
+ "bipbop-cenc-audio1.m4s^headers^",
+ "bipbop-cenc-audio2.m4s",
+ "bipbop-cenc-audio2.m4s^headers^",
+ "bipbop-cenc-audio3.m4s",
+ "bipbop-cenc-audio3.m4s^headers^",
+ "bipbop-cenc-audioinit.mp4",
+ "bipbop-cenc-audioinit.mp4^headers^",
+ "bipbop-cenc-video1.m4s",
+ "bipbop-cenc-video1.m4s^headers^",
+ "bipbop-cenc-video2.m4s",
+ "bipbop-cenc-video2.m4s^headers^",
+ "bipbop-cenc-videoinit.mp4",
+ "bipbop-cenc-videoinit.mp4^headers^",
+ "bipbop-cenc-video-10s.mp4",
+ "bipbop-cenc-video-10s.mp4^headers^",
+ "bipbop-clearkey-keyrotation-clear-lead-audio.mp4",
+ "bipbop-clearkey-keyrotation-clear-lead-audio.mp4^headers^",
+ "bipbop-clearkey-keyrotation-clear-lead-video.mp4",
+ "bipbop-clearkey-keyrotation-clear-lead-video.mp4^headers^",
+ "bipbop_225w_175kbps.mp4",
+ "bipbop_225w_175kbps.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-1.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-2.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-3.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-4.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-init.mp4",
+ "bipbop_225w_175kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-1.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-2.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-3.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-4.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-init.mp4",
+ "bipbop_225w_175kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-video-key1-1.m4s",
+ "bipbop_225w_175kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-video-key1-init.mp4",
+ "bipbop_225w_175kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-video-key2-1.m4s",
+ "bipbop_225w_175kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-video-key2-init.mp4",
+ "bipbop_225w_175kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-1.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-2.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-3.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-4.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-1.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-2.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-3.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-4.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-init.mp4",
+ "bipbop_300_215kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-1.m4s",
+ "bipbop_300_215kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-2.m4s",
+ "bipbop_300_215kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-1.m4s",
+ "bipbop_300_215kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-2.m4s",
+ "bipbop_300_215kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-init.mp4",
+ "bipbop_300_215kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-1.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-2.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-3.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-4.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-init.mp4",
+ "bipbop_300wp_227kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-1.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-2.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-3.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-4.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-init.mp4",
+ "bipbop_300wp_227kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-1.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-2.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-init.mp4",
+ "bipbop_300wp_227kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-1.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-2.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-init.mp4",
+ "bipbop_300wp_227kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-1.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-2.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-3.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-4.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-init.mp4",
+ "bipbop_360w_253kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-1.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-2.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-3.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-4.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-init.mp4",
+ "bipbop_360w_253kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-video-key1-1.m4s",
+ "bipbop_360w_253kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-video-key1-init.mp4",
+ "bipbop_360w_253kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-video-key2-1.m4s",
+ "bipbop_360w_253kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-video-key2-init.mp4",
+ "bipbop_360w_253kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-clearkey-audio.webm",
+ "bipbop_360w_253kbps-clearkey-audio.webm^headers^",
+ "bipbop_360w_253kbps-clearkey-video-vp8.webm",
+ "bipbop_360w_253kbps-clearkey-video-vp8.webm^headers^",
+ "bipbop_360w_253kbps-clearkey-video-vp9.webm",
+ "bipbop_360w_253kbps-clearkey-video-vp9.webm^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480_624kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-2.m4s",
+ "bipbop_480_624kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-1.m4s",
+ "bipbop_480_624kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-2.m4s",
+ "bipbop_480_624kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480_959kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480_959kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-1.m4s",
+ "bipbop_480_959kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-2.m4s",
+ "bipbop_480_959kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-init.mp4",
+ "bipbop_480_959kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-1.m4s",
+ "bipbop_480_959kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-2.m4s",
+ "bipbop_480_959kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-init.mp4",
+ "bipbop_480_959kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480wp_663kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480wp_663kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-1.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-2.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-init.mp4",
+ "bipbop_480wp_663kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-1.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-2.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-init.mp4",
+ "bipbop_480wp_663kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_audio_aac_8k.mp4",
+ "bipbop_audio_aac_8k.mp4^headers^",
+ "bipbop_audio_aac_22.05k.mp4",
+ "bipbop_audio_aac_22.05k.mp4^headers^",
+ "bipbop_audio_aac_44.1k.mp4",
+ "bipbop_audio_aac_44.1k.mp4^headers^",
+ "bipbop_audio_aac_48k.mp4",
+ "bipbop_audio_aac_48k.mp4^headers^",
+ "bipbop_audio_aac_88.2k.mp4",
+ "bipbop_audio_aac_88.2k.mp4^headers^",
+ "bipbop_audio_aac_96k.mp4",
+ "bipbop_audio_aac_96k.mp4^headers^",
+ "bipbop_cbcs_1_9_audio_1.m4s",
+ "bipbop_cbcs_1_9_audio_1.m4s^headers^",
+ "bipbop_cbcs_1_9_audio_init.mp4",
+ "bipbop_cbcs_1_9_audio_init.mp4^headers^",
+ "bipbop_cbcs_1_9_video_1.m4s",
+ "bipbop_cbcs_1_9_video_1.m4s^headers^",
+ "bipbop_cbcs_1_9_video_init.mp4",
+ "bipbop_cbcs_1_9_video_init.mp4^headers^",
+ "bipbop_cbcs_5_5_audio_1.m4s",
+ "bipbop_cbcs_5_5_audio_1.m4s^headers^",
+ "bipbop_cbcs_5_5_audio_init.mp4",
+ "bipbop_cbcs_5_5_audio_init.mp4^headers^",
+ "bipbop_cbcs_5_5_video_1.m4s",
+ "bipbop_cbcs_5_5_video_1.m4s^headers^",
+ "bipbop_cbcs_5_5_video_init.mp4",
+ "bipbop_cbcs_5_5_video_init.mp4^headers^",
+ "bipbop_cbcs_7_7_audio_1.m4s",
+ "bipbop_cbcs_7_7_audio_1.m4s^headers^",
+ "bipbop_cbcs_7_7_audio_init.mp4",
+ "bipbop_cbcs_7_7_audio_init.mp4^headers^",
+ "bipbop_cbcs_7_7_video_1.m4s",
+ "bipbop_cbcs_7_7_video_1.m4s^headers^",
+ "bipbop_cbcs_7_7_video_init.mp4",
+ "bipbop_cbcs_7_7_video_init.mp4^headers^",
+ "bipbop_cbcs_9_8_audio_1.m4s",
+ "bipbop_cbcs_9_8_audio_1.m4s^headers^",
+ "bipbop_cbcs_9_8_audio_init.mp4",
+ "bipbop_cbcs_9_8_audio_init.mp4^headers^",
+ "bipbop_cbcs_9_8_video_1.m4s",
+ "bipbop_cbcs_9_8_video_1.m4s^headers^",
+ "bipbop_cbcs_9_8_video_init.mp4",
+ "bipbop_cbcs_9_8_video_init.mp4^headers^",
+ "bipbop_cbcs_10_0_audio_1.m4s",
+ "bipbop_cbcs_10_0_audio_1.m4s^headers^",
+ "bipbop_cbcs_10_0_audio_init.mp4",
+ "bipbop_cbcs_10_0_audio_init.mp4^headers^",
+ "bipbop_cbcs_10_0_video_1.m4s",
+ "bipbop_cbcs_10_0_video_1.m4s^headers^",
+ "bipbop_cbcs_10_0_video_init.mp4",
+ "bipbop_cbcs_10_0_video_init.mp4^headers^",
+ "bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm",
+ "bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm",
+ "bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm",
+ "bipbop_short_vp8.webm",
+ "bipbop_short_vp8.webm^headers^",
+ "bipbop-lateaudio.mp4",
+ "bipbop-lateaudio.mp4^headers^",
+ "black100x100-aspect3to2.ogv",
+ "black100x100-aspect3to2.ogv^headers^",
+ "bogus.duh",
+ "bogus.ogv",
+ "bogus.ogv^headers^",
+ "bogus.wav",
+ "bogus.wav^headers^",
+ "bug461281.ogg",
+ "bug461281.ogg^headers^",
+ "bug482461-theora.ogv",
+ "bug482461-theora.ogv^headers^",
+ "bug482461.ogv",
+ "bug482461.ogv^headers^",
+ "bug495129.ogv",
+ "bug495129.ogv^headers^",
+ "bug495794.ogg",
+ "bug495794.ogg^headers^",
+ "bug498380.ogv",
+ "bug498380.ogv^headers^",
+ "bug498855-1.ogv",
+ "bug498855-1.ogv^headers^",
+ "bug498855-2.ogv",
+ "bug498855-2.ogv^headers^",
+ "bug498855-3.ogv",
+ "bug498855-3.ogv^headers^",
+ "bug499519.ogv",
+ "bug499519.ogv^headers^",
+ "bug500311.ogv",
+ "bug500311.ogv^headers^",
+ "bug501279.ogg",
+ "bug501279.ogg^headers^",
+ "bug504613.ogv",
+ "bug504613.ogv^headers^",
+ "bug504644.ogv",
+ "bug504644.ogv^headers^",
+ "bug504843.ogv",
+ "bug504843.ogv^headers^",
+ "bug506094.ogv",
+ "bug506094.ogv^headers^",
+ "bug516323.indexed.ogv",
+ "bug516323.indexed.ogv^headers^",
+ "bug516323.ogv",
+ "bug516323.ogv^headers^",
+ "bug520493.ogg",
+ "bug520493.ogg^headers^",
+ "bug520500.ogg",
+ "bug520500.ogg^headers^",
+ "bug520908.ogv",
+ "bug520908.ogv^headers^",
+ "bug523816.ogv",
+ "bug523816.ogv^headers^",
+ "bug533822.ogg",
+ "bug533822.ogg^headers^",
+ "bug556821.ogv",
+ "bug556821.ogv^headers^",
+ "bug557094.ogv",
+ "bug557094.ogv^headers^",
+ "bug604067.webm",
+ "bug604067.webm^headers^",
+ "bug1066943.webm",
+ "bug1066943.webm^headers^",
+ "bug1301226.wav",
+ "bug1301226.wav^headers^",
+ "bug1301226-odd.wav",
+ "bug1301226-odd.wav^headers^",
+ "bug1377278.webm",
+ "bug1377278.webm^headers^",
+ "bunny.webm",
+ "can_play_type_dash.js",
+ "can_play_type_ogg.js",
+ "can_play_type_wave.js",
+ "can_play_type_webm.js",
+ "cancellable_request.sjs",
+ "chain.ogg",
+ "chain.ogg^headers^",
+ "chain.ogv",
+ "chain.ogv^headers^",
+ "chain.opus",
+ "chain.opus^headers^",
+ "chained-audio-video.ogg",
+ "chained-audio-video.ogg^headers^",
+ "chained-video.ogv",
+ "chained-video.ogv^headers^",
+ "chromeHelper.js",
+ "cloneElementVisually_helpers.js",
+ "contentType.sjs",
+ "detodos.opus",
+ "detodos.opus^headers^",
+ "detodos.webm",
+ "detodos.webm^headers^",
+ "detodos-short.webm",
+ "detodos-short.webm^headers^",
+ "detodos-recorder-test.opus",
+ "detodos-recorder-test.opus^headers^",
+ "detodos-short.opus",
+ "detodos-short.opus^headers^",
+ "dirac.ogg",
+ "dirac.ogg^headers^",
+ "dynamic_resource.sjs",
+ "eme_standalone.js",
+ "eme.js",
+ "empty_size.mp3",
+ "file_access_controls.html",
+ "file_eme_createMediaKeys.html",
+ "flac-s24.flac",
+ "flac-s24.flac^headers^",
+ "flac-noheader-s16.flac",
+ "flac-noheader-s16.flac^headers^",
+ "flac-sample.mp4",
+ "flac-sample.mp4^headers^",
+ "flac-sample-cenc.mp4",
+ "flac-sample-cenc.mp4^headers^",
+ "fragment_noplay.js",
+ "fragment_play.js",
+ "gizmo.mp4",
+ "gizmo.mp4^headers^",
+ "gizmo-noaudio.mp4",
+ "gizmo-noaudio.mp4^headers^",
+ "gizmo-short.mp4",
+ "gizmo-short.mp4^headers^",
+ "gizmo.webm",
+ "gizmo.webm^headers^",
+ "gizmo-noaudio.webm",
+ "gizmo-noaudio.webm^headers^",
+ "gUM_support.js",
+ "gzipped_mp4.sjs",
+ "huge-id3.mp3",
+ "huge-id3.mp3^headers^",
+ "id3tags.mp3",
+ "id3tags.mp3^headers^",
+ "invalid-cmap-s0c0.opus",
+ "invalid-cmap-s0c0.opus^headers^",
+ "invalid-cmap-s0c2.opus",
+ "invalid-cmap-s0c2.opus^headers^",
+ "invalid-cmap-s1c2.opus",
+ "invalid-cmap-s1c2.opus^headers^",
+ "invalid-cmap-short.opus",
+ "invalid-cmap-short.opus^headers^",
+ "invalid-discard_on_multi_blocks.webm",
+ "invalid-discard_on_multi_blocks.webm^headers^",
+ "invalid-excess_discard.webm",
+ "invalid-excess_discard.webm^headers^",
+ "invalid-excess_neg_discard.webm",
+ "invalid-excess_neg_discard.webm^headers^",
+ "invalid-m0c0.opus",
+ "invalid-m0c0.opus^headers^",
+ "invalid-m0c3.opus",
+ "invalid-m0c3.opus^headers^",
+ "invalid-m1c0.opus",
+ "invalid-m1c0.opus^headers^",
+ "invalid-m1c9.opus",
+ "invalid-m1c9.opus^headers^",
+ "invalid-m2c0.opus",
+ "invalid-m2c0.opus^headers^",
+ "invalid-m2c1.opus",
+ "invalid-m2c1.opus^headers^",
+ "invalid-neg_discard.webm",
+ "invalid-neg_discard.webm^headers^",
+ "invalid-preskip.webm",
+ "invalid-preskip.webm^headers^",
+ "manifest.js",
+ "midflight-redirect.sjs",
+ "multiple-bos.ogg",
+ "multiple-bos.ogg^headers^",
+ "multiple-bos-more-header-fileds.ogg",
+ "multiple-bos-more-header-fileds.ogg^headers^",
+ "multi_id3v2.mp3",
+ "no-container-codec-delay.webm",
+ "no-cues.webm",
+ "no-cues.webm^headers^",
+ "notags.mp3",
+ "notags.mp3^headers^",
+ "opus-mapping2.mp4",
+ "opus-mapping2.mp4^headers^",
+ "opus-mapping2.webm",
+ "opus-mapping2.webm^headers^",
+ "opus-sample.mp4",
+ "opus-sample.mp4^headers^",
+ "opus-sample-cenc.mp4",
+ "opus-sample-cenc.mp4^headers^",
+ "owl-funnier-id3.mp3",
+ "owl-funnier-id3.mp3^headers^",
+ "owl-funny-id3.mp3",
+ "owl-funny-id3.mp3^headers^",
+ "owl.mp3",
+ "owl.mp3^headers^",
+ "owl-short.mp3",
+ "owl-short.mp3^headers^",
+ "pixel_aspect_ratio.mp4",
+ "play_promise.js",
+ "poster-test.jpg",
+ "r11025_msadpcm_c1.wav",
+ "r11025_msadpcm_c1.wav^headers^",
+ "r11025_s16_c1.wav",
+ "r11025_s16_c1.wav^headers^",
+ "r11025_s16_c1_trailing.wav",
+ "r11025_s16_c1_trailing.wav^headers^",
+ "r11025_s16_c1-short.wav",
+ "r11025_s16_c1-short.wav^headers^",
+ "r11025_u8_c1.wav",
+ "r11025_u8_c1.wav^headers^",
+ "r11025_u8_c1_trunc.wav",
+ "r11025_u8_c1_trunc.wav^headers^",
+ "r16000_u8_c1_list.wav",
+ "r16000_u8_c1_list.wav^headers^",
+ "reactivate_helper.html",
+ "red-46x48.mp4",
+ "red-46x48.mp4^headers^",
+ "red-48x46.mp4",
+ "red-48x46.mp4^headers^",
+ "redirect.sjs",
+ "referer.sjs",
+ "resolution-change.webm",
+ "resolution-change.webm^headers^",
+ "sample.3gp",
+ "sample.3g2",
+ "sample-encrypted-sgpdstbl-sbgptraf.mp4",
+ "sample-encrypted-sgpdstbl-sbgptraf.mp4^headers^",
+ "sample-fisbone-skeleton4.ogv",
+ "sample-fisbone-skeleton4.ogv^headers^",
+ "sample-fisbone-wrong-header.ogv",
+ "sample-fisbone-wrong-header.ogv^headers^",
+ "seek.ogv",
+ "seek.ogv^headers^",
+ "seek-short.ogv",
+ "seek-short.ogv^headers^",
+ "seek.webm",
+ "seek.webm^headers^",
+ "seek-short.webm",
+ "seek-short.webm^headers^",
+ "seek_support.js",
+ "seekLies.sjs",
+ "seek_with_sound.ogg^headers^",
+ "short-cenc.mp4",
+ "sine.webm",
+ "sine.webm^headers^",
+ "sintel-short-clearkey-subsample-encrypted-audio.webm",
+ "sintel-short-clearkey-subsample-encrypted-audio.webm^headers^",
+ "sintel-short-clearkey-subsample-encrypted-video.webm",
+ "sintel-short-clearkey-subsample-encrypted-video.webm^headers^",
+ "short.mp4",
+ "short.mp4.gz",
+ "short.mp4^headers^",
+ "short-aac-encrypted-audio.mp4",
+ "short-aac-encrypted-audio.mp4^headers^",
+ "short-audio-fragmented-cenc-without-pssh.mp4",
+ "short-audio-fragmented-cenc-without-pssh.mp4^headers^",
+ "short-video.ogv",
+ "short-video.ogv^headers^",
+ "short-vp9-encrypted-video.mp4",
+ "short-vp9-encrypted-video.mp4^headers^",
+ "small-shot-mp3.mp4",
+ "small-shot-mp3.mp4^headers^",
+ "small-shot.m4a",
+ "small-shot.mp3",
+ "small-shot.mp3^headers^",
+ "small-shot.ogg",
+ "small-shot.ogg^headers^",
+ "small-shot.flac",
+ "sound.ogg",
+ "sound.ogg^headers^",
+ "spacestorm-1000Hz-100ms.ogg",
+ "spacestorm-1000Hz-100ms.ogg^headers^",
+ "split.webm",
+ "split.webm^headers^",
+ "street.mp4",
+ "street.mp4^headers^",
+ "test-1-mono.opus",
+ "test-1-mono.opus^headers^",
+ "test-2-stereo.opus",
+ "test-2-stereo.opus^headers^",
+ "test-3-LCR.opus",
+ "test-3-LCR.opus^headers^",
+ "test-4-quad.opus",
+ "test-4-quad.opus^headers^",
+ "test-5-5.0.opus",
+ "test-5-5.0.opus^headers^",
+ "test-6-5.1.opus",
+ "test-6-5.1.opus^headers^",
+ "test-7-6.1.opus",
+ "test-7-6.1.opus^headers^",
+ "test-8-7.1.opus",
+ "test-8-7.1.opus^headers^",
+ "test-stereo-phase-inversion-180.opus",
+ "test-stereo-phase-inversion-180.opus^headers^",
+ "variable-channel.ogg",
+ "variable-channel.ogg^headers^",
+ "variable-channel.opus",
+ "variable-channel.opus^headers^",
+ "variable-preskip.opus",
+ "variable-preskip.opus^headers^",
+ "variable-samplerate.ogg",
+ "variable-samplerate.ogg^headers^",
+ "variable-samplerate.opus",
+ "variable-samplerate.opus^headers^",
+ "vbr-head.mp3",
+ "vbr-head.mp3^headers^",
+ "vbr.mp3",
+ "vbr.mp3^headers^",
+ "very-short.mp3",
+ "video-overhang.ogg",
+ "video-overhang.ogg^headers^",
+ "vp9-superframes.webm",
+ "vp9-superframes.webm^headers^",
+ "vp9.webm",
+ "vp9.webm^headers^",
+ "vp9-short.webm",
+ "vp9-short.webm^headers^",
+ "vp9cake.webm",
+ "vp9cake.webm^headers^",
+ "vp9cake-short.webm",
+ "vp9cake-short.webm^headers^",
+ "wave_metadata.wav",
+ "wave_metadata.wav^headers^",
+ "wave_metadata_bad_len.wav",
+ "wave_metadata_bad_len.wav^headers^",
+ "wave_metadata_bad_no_null.wav",
+ "wave_metadata_bad_no_null.wav^headers^",
+ "wave_metadata_bad_utf8.wav",
+ "wave_metadata_bad_utf8.wav^headers^",
+ "wave_metadata_unknown_tag.wav",
+ "wave_metadata_unknown_tag.wav^headers^",
+ "wave_metadata_utf8.wav",
+ "wave_metadata_utf8.wav^headers^",
+ "wavedata_alaw.wav",
+ "wavedata_alaw.wav^headers^",
+ "wavedata_float.wav",
+ "wavedata_float.wav^headers^",
+ "wavedata_s24.wav",
+ "wavedata_s24.wav^headers^",
+ "wavedata_s16.wav",
+ "wavedata_s16.wav^headers^",
+ "wavedata_u8.wav",
+ "wavedata_u8.wav^headers^",
+ "wavedata_ulaw.wav",
+ "wavedata_ulaw.wav^headers^",
+ "!/dom/canvas/test/captureStream_common.js",
+ "!/dom/html/test/reflect.js",
+ "!/dom/media/webrtc/tests/mochitests/head.js",
+ "hls/bipbop_16x9_single.m3u8",
+ "hls/bipbop_4x3_single.m3u8",
+ "hls/bipbop_4x3_variant.m3u8",
+ "hls/400x300_prog_index.m3u8",
+ "hls/400x300_prog_index_5s.m3u8",
+ "hls/416x243_prog_index_5s.m3u8",
+ "hls/640x480_prog_index.m3u8",
+ "hls/960x720_prog_index.m3u8",
+ "hls/400x300_seg0.ts",
+ "hls/400x300_seg0_5s.ts",
+ "hls/400x300_seg1.ts",
+ "hls/416x243_seg0_5s.ts",
+ "hls/640x480_seg0.ts",
+ "hls/640x480_seg1.ts",
+ "hls/960x720_seg0.ts",
+ "hls/960x720_seg1.ts",
+ "sync.webm",
+]
+
+["test_eme_autoplay.html"]
+skip-if = ["os == 'android'"] # bug 1149374
+scheme = "https"
+
+["test_eme_canvas_blocked.html"]
+skip-if = [
+ "os == 'android'", # bug 1149374
+ "apple_silicon", # bug 1707737
+]
+scheme = "https"
+
+["test_eme_detach_reattach_same_mediakeys_during_playback.html"]
+skip-if = ["os == 'android'"] # bug 1149374
+scheme = "https"
+
+["test_eme_initDataTypes.html"]
+skip-if = ["os == 'android'"]
+scheme = "https"
+
+["test_eme_missing_pssh.html"]
+skip-if = ["os == 'android'"]
+scheme = "https"
+
+["test_eme_non_mse_fails.html"]
+skip-if = ["os == 'android'"] # bug 1149374
+scheme = "https"
+
+["test_eme_playback.html"]
+skip-if = [
+ "os == 'android'", # bug 1149374
+ "apple_silicon", # bug 1707737
+]
+
+["test_eme_protection_query.html"]
+skip-if = ["os == 'android'"] # bug 1149374
+scheme = "https"
+
+["test_eme_pssh_in_moof.html"]
+skip-if = ["os == 'android'"] # bug 1149374
+scheme = "https"
+
+["test_eme_sample_groups_playback.html"]
+skip-if = ["os == 'android'"] # bug 1149374
+scheme = "https"
+
+["test_eme_special_key_system.html"]
+skip-if = ["os == 'android'"] # bug 1149374
+scheme = "https"
+
+["test_eme_stream_capture_blocked_case1.html"]
+tags = "mtg capturestream"
+skip-if = ["os == 'android'"] # bug 1149374
+scheme = "https"
+
+["test_eme_stream_capture_blocked_case2.html"]
+tags = "mtg capturestream"
+skip-if = [
+ "os == 'android'", # bug 1149374
+ "apple_silicon", # bug 1707737
+]
+scheme = "https"
+
+["test_eme_stream_capture_blocked_case3.html"]
+tags = "mtg capturestream"
+skip-if = [
+ "os == 'android'", # bug 1149374
+ "apple_silicon", # bug 1707737
+]
+scheme = "https"
+
+["test_eme_unsetMediaKeys_then_capture.html"]
+skip-if = [
+ "xorigin",
+ "os == 'android'", # bug 1149374
+ "apple_silicon", # bug 1707737
+]
+scheme = "https"
+
+["test_eme_waitingforkey.html"]
+skip-if = [
+ "xorigin",
+ "os == 'android'", # bug 1149374
+ "apple_silicon", # bug 1707737
+]
+scheme = "https"
+
+["test_eme_wv_privacy.html"]
+scheme = "https"
diff --git a/dom/media/test/mochitest_eme_compat.toml b/dom/media/test/mochitest_eme_compat.toml
new file mode 100644
index 0000000000..43a5b510fc
--- /dev/null
+++ b/dom/media/test/mochitest_eme_compat.toml
@@ -0,0 +1,793 @@
+# Media tests should be backend independent, i.e., not conditioned on ogg,
+# wave etc. (The only exception is the can_play_type tests, which
+# necessarily depend on the backend(s) configured.) As far as possible, each
+# test should work with any resource type. This makes it easy to add new
+# backends and reduces the amount of test duplication.
+
+# For each supported backend, resources that can be played by that backend
+# should be added to the lists in manifest.js. Media tests that aren't
+# testing for a bug in handling a specific resource type should pick one of
+# the lists in manifest.js and run the test for each resource in the list
+# that is supported in the current build (the canPlayType API is useful for
+# this).
+
+# To test whether a valid resource can simply be played through correctly,
+# and optionally that its metadata is read correctly, just add it to
+# gPlayTests in manifest.js. To test whether an invalid resource correctly
+# throws an error (and does not cause a crash or hang), just add it to
+# gErrorTests in manifest.js.
+
+# To test for a specific bug in handling a specific resource type, make the
+# test first check canPlayType for the type, and if it's not supported, just
+# do ok(true, "Type not supported") and stop the test.
+
+[DEFAULT]
+subsuite = "media"
+tags = "media-engine-compatible"
+skip-if = ["os == 'linux' && (asan || debug)"] # Bug 1476870: common fatal error (shutdown hang) on asan/debug
+support-files = [
+ "16bit_wave_extrametadata.wav",
+ "16bit_wave_extrametadata.wav^headers^",
+ "320x240.ogv",
+ "320x240.ogv^headers^",
+ "448636.ogv",
+ "448636.ogv^headers^",
+ "A4.ogv",
+ "A4.ogv^headers^",
+ "VID_0001.ogg",
+ "VID_0001.ogg^headers^",
+ "allowed.sjs",
+ "ambisonics.mp4",
+ "ambisonics.mp4^headers^",
+ "audio-gaps.ogg",
+ "audio-gaps.ogg^headers^",
+ "audio-gaps-short.ogg",
+ "audio-gaps-short.ogg^headers^",
+ "audio-overhang.ogg",
+ "audio-overhang.ogg^headers^",
+ "audio.wav",
+ "audio.wav^headers^",
+ "av1.mp4",
+ "av1.mp4^headers^",
+ "background_video.js",
+ "badtags.ogg",
+ "badtags.ogg^headers^",
+ "bear-640x360-v_frag-cenc-key_rotation.mp4",
+ "bear-640x360-a_frag-cenc-key_rotation.mp4",
+ "beta-phrasebook.ogg",
+ "beta-phrasebook.ogg^headers^",
+ "big.wav",
+ "big.wav^headers^",
+ "big-buck-bunny-cenc-avc3-1.m4s",
+ "big-buck-bunny-cenc-avc3-1.m4s^headers^",
+ "big-buck-bunny-cenc-avc3-init.mp4",
+ "big-buck-bunny-cenc-avc3-init.mp4^headers^",
+ "big-short.wav",
+ "big-short.wav^headers^",
+ "bipbop.mp4",
+ "bipbop-cenc-audio1.m4s",
+ "bipbop-cenc-audio1.m4s^headers^",
+ "bipbop-cenc-audio2.m4s",
+ "bipbop-cenc-audio2.m4s^headers^",
+ "bipbop-cenc-audio3.m4s",
+ "bipbop-cenc-audio3.m4s^headers^",
+ "bipbop-cenc-audioinit.mp4",
+ "bipbop-cenc-audioinit.mp4^headers^",
+ "bipbop-cenc-video1.m4s",
+ "bipbop-cenc-video1.m4s^headers^",
+ "bipbop-cenc-video2.m4s",
+ "bipbop-cenc-video2.m4s^headers^",
+ "bipbop-cenc-videoinit.mp4",
+ "bipbop-cenc-videoinit.mp4^headers^",
+ "bipbop-cenc-video-10s.mp4",
+ "bipbop-cenc-video-10s.mp4^headers^",
+ "bipbop-clearkey-keyrotation-clear-lead-audio.mp4",
+ "bipbop-clearkey-keyrotation-clear-lead-audio.mp4^headers^",
+ "bipbop-clearkey-keyrotation-clear-lead-video.mp4",
+ "bipbop-clearkey-keyrotation-clear-lead-video.mp4^headers^",
+ "bipbop_225w_175kbps.mp4",
+ "bipbop_225w_175kbps.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-1.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-2.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-3.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-4.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-init.mp4",
+ "bipbop_225w_175kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-1.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-2.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-3.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-4.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-init.mp4",
+ "bipbop_225w_175kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-video-key1-1.m4s",
+ "bipbop_225w_175kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-video-key1-init.mp4",
+ "bipbop_225w_175kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-video-key2-1.m4s",
+ "bipbop_225w_175kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-video-key2-init.mp4",
+ "bipbop_225w_175kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-1.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-2.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-3.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-4.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-1.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-2.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-3.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-4.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-init.mp4",
+ "bipbop_300_215kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-1.m4s",
+ "bipbop_300_215kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-2.m4s",
+ "bipbop_300_215kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-1.m4s",
+ "bipbop_300_215kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-2.m4s",
+ "bipbop_300_215kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-init.mp4",
+ "bipbop_300_215kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-1.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-2.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-3.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-4.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-init.mp4",
+ "bipbop_300wp_227kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-1.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-2.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-3.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-4.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-init.mp4",
+ "bipbop_300wp_227kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-1.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-2.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-init.mp4",
+ "bipbop_300wp_227kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-1.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-2.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-init.mp4",
+ "bipbop_300wp_227kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-1.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-2.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-3.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-4.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-init.mp4",
+ "bipbop_360w_253kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-1.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-2.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-3.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-4.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-init.mp4",
+ "bipbop_360w_253kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-video-key1-1.m4s",
+ "bipbop_360w_253kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-video-key1-init.mp4",
+ "bipbop_360w_253kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-video-key2-1.m4s",
+ "bipbop_360w_253kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-video-key2-init.mp4",
+ "bipbop_360w_253kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-clearkey-audio.webm",
+ "bipbop_360w_253kbps-clearkey-audio.webm^headers^",
+ "bipbop_360w_253kbps-clearkey-video-vp8.webm",
+ "bipbop_360w_253kbps-clearkey-video-vp8.webm^headers^",
+ "bipbop_360w_253kbps-clearkey-video-vp9.webm",
+ "bipbop_360w_253kbps-clearkey-video-vp9.webm^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480_624kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-2.m4s",
+ "bipbop_480_624kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-1.m4s",
+ "bipbop_480_624kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-2.m4s",
+ "bipbop_480_624kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480_959kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480_959kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-1.m4s",
+ "bipbop_480_959kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-2.m4s",
+ "bipbop_480_959kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-init.mp4",
+ "bipbop_480_959kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-1.m4s",
+ "bipbop_480_959kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-2.m4s",
+ "bipbop_480_959kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-init.mp4",
+ "bipbop_480_959kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480wp_663kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480wp_663kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-1.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-2.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-init.mp4",
+ "bipbop_480wp_663kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-1.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-2.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-init.mp4",
+ "bipbop_480wp_663kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_audio_aac_8k.mp4",
+ "bipbop_audio_aac_8k.mp4^headers^",
+ "bipbop_audio_aac_22.05k.mp4",
+ "bipbop_audio_aac_22.05k.mp4^headers^",
+ "bipbop_audio_aac_44.1k.mp4",
+ "bipbop_audio_aac_44.1k.mp4^headers^",
+ "bipbop_audio_aac_48k.mp4",
+ "bipbop_audio_aac_48k.mp4^headers^",
+ "bipbop_audio_aac_88.2k.mp4",
+ "bipbop_audio_aac_88.2k.mp4^headers^",
+ "bipbop_audio_aac_96k.mp4",
+ "bipbop_audio_aac_96k.mp4^headers^",
+ "bipbop_cbcs_1_9_audio_1.m4s",
+ "bipbop_cbcs_1_9_audio_1.m4s^headers^",
+ "bipbop_cbcs_1_9_audio_init.mp4",
+ "bipbop_cbcs_1_9_audio_init.mp4^headers^",
+ "bipbop_cbcs_1_9_video_1.m4s",
+ "bipbop_cbcs_1_9_video_1.m4s^headers^",
+ "bipbop_cbcs_1_9_video_init.mp4",
+ "bipbop_cbcs_1_9_video_init.mp4^headers^",
+ "bipbop_cbcs_5_5_audio_1.m4s",
+ "bipbop_cbcs_5_5_audio_1.m4s^headers^",
+ "bipbop_cbcs_5_5_audio_init.mp4",
+ "bipbop_cbcs_5_5_audio_init.mp4^headers^",
+ "bipbop_cbcs_5_5_video_1.m4s",
+ "bipbop_cbcs_5_5_video_1.m4s^headers^",
+ "bipbop_cbcs_5_5_video_init.mp4",
+ "bipbop_cbcs_5_5_video_init.mp4^headers^",
+ "bipbop_cbcs_7_7_audio_1.m4s",
+ "bipbop_cbcs_7_7_audio_1.m4s^headers^",
+ "bipbop_cbcs_7_7_audio_init.mp4",
+ "bipbop_cbcs_7_7_audio_init.mp4^headers^",
+ "bipbop_cbcs_7_7_video_1.m4s",
+ "bipbop_cbcs_7_7_video_1.m4s^headers^",
+ "bipbop_cbcs_7_7_video_init.mp4",
+ "bipbop_cbcs_7_7_video_init.mp4^headers^",
+ "bipbop_cbcs_9_8_audio_1.m4s",
+ "bipbop_cbcs_9_8_audio_1.m4s^headers^",
+ "bipbop_cbcs_9_8_audio_init.mp4",
+ "bipbop_cbcs_9_8_audio_init.mp4^headers^",
+ "bipbop_cbcs_9_8_video_1.m4s",
+ "bipbop_cbcs_9_8_video_1.m4s^headers^",
+ "bipbop_cbcs_9_8_video_init.mp4",
+ "bipbop_cbcs_9_8_video_init.mp4^headers^",
+ "bipbop_cbcs_10_0_audio_1.m4s",
+ "bipbop_cbcs_10_0_audio_1.m4s^headers^",
+ "bipbop_cbcs_10_0_audio_init.mp4",
+ "bipbop_cbcs_10_0_audio_init.mp4^headers^",
+ "bipbop_cbcs_10_0_video_1.m4s",
+ "bipbop_cbcs_10_0_video_1.m4s^headers^",
+ "bipbop_cbcs_10_0_video_init.mp4",
+ "bipbop_cbcs_10_0_video_init.mp4^headers^",
+ "bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm",
+ "bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm",
+ "bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm",
+ "bipbop_short_vp8.webm",
+ "bipbop_short_vp8.webm^headers^",
+ "bipbop-lateaudio.mp4",
+ "bipbop-lateaudio.mp4^headers^",
+ "black100x100-aspect3to2.ogv",
+ "black100x100-aspect3to2.ogv^headers^",
+ "bogus.duh",
+ "bogus.ogv",
+ "bogus.ogv^headers^",
+ "bogus.wav",
+ "bogus.wav^headers^",
+ "bug461281.ogg",
+ "bug461281.ogg^headers^",
+ "bug482461-theora.ogv",
+ "bug482461-theora.ogv^headers^",
+ "bug482461.ogv",
+ "bug482461.ogv^headers^",
+ "bug495129.ogv",
+ "bug495129.ogv^headers^",
+ "bug495794.ogg",
+ "bug495794.ogg^headers^",
+ "bug498380.ogv",
+ "bug498380.ogv^headers^",
+ "bug498855-1.ogv",
+ "bug498855-1.ogv^headers^",
+ "bug498855-2.ogv",
+ "bug498855-2.ogv^headers^",
+ "bug498855-3.ogv",
+ "bug498855-3.ogv^headers^",
+ "bug499519.ogv",
+ "bug499519.ogv^headers^",
+ "bug500311.ogv",
+ "bug500311.ogv^headers^",
+ "bug501279.ogg",
+ "bug501279.ogg^headers^",
+ "bug504613.ogv",
+ "bug504613.ogv^headers^",
+ "bug504644.ogv",
+ "bug504644.ogv^headers^",
+ "bug504843.ogv",
+ "bug504843.ogv^headers^",
+ "bug506094.ogv",
+ "bug506094.ogv^headers^",
+ "bug516323.indexed.ogv",
+ "bug516323.indexed.ogv^headers^",
+ "bug516323.ogv",
+ "bug516323.ogv^headers^",
+ "bug520493.ogg",
+ "bug520493.ogg^headers^",
+ "bug520500.ogg",
+ "bug520500.ogg^headers^",
+ "bug520908.ogv",
+ "bug520908.ogv^headers^",
+ "bug523816.ogv",
+ "bug523816.ogv^headers^",
+ "bug533822.ogg",
+ "bug533822.ogg^headers^",
+ "bug556821.ogv",
+ "bug556821.ogv^headers^",
+ "bug557094.ogv",
+ "bug557094.ogv^headers^",
+ "bug604067.webm",
+ "bug604067.webm^headers^",
+ "bug1066943.webm",
+ "bug1066943.webm^headers^",
+ "bug1301226.wav",
+ "bug1301226.wav^headers^",
+ "bug1301226-odd.wav",
+ "bug1301226-odd.wav^headers^",
+ "bug1377278.webm",
+ "bug1377278.webm^headers^",
+ "bunny.webm",
+ "can_play_type_dash.js",
+ "can_play_type_ogg.js",
+ "can_play_type_wave.js",
+ "can_play_type_webm.js",
+ "cancellable_request.sjs",
+ "chain.ogg",
+ "chain.ogg^headers^",
+ "chain.ogv",
+ "chain.ogv^headers^",
+ "chain.opus",
+ "chain.opus^headers^",
+ "chained-audio-video.ogg",
+ "chained-audio-video.ogg^headers^",
+ "chained-video.ogv",
+ "chained-video.ogv^headers^",
+ "chromeHelper.js",
+ "cloneElementVisually_helpers.js",
+ "contentType.sjs",
+ "detodos.opus",
+ "detodos.opus^headers^",
+ "detodos.webm",
+ "detodos.webm^headers^",
+ "detodos-short.webm",
+ "detodos-short.webm^headers^",
+ "detodos-recorder-test.opus",
+ "detodos-recorder-test.opus^headers^",
+ "detodos-short.opus",
+ "detodos-short.opus^headers^",
+ "dirac.ogg",
+ "dirac.ogg^headers^",
+ "dynamic_resource.sjs",
+ "eme_standalone.js",
+ "eme.js",
+ "empty_size.mp3",
+ "file_access_controls.html",
+ "file_eme_createMediaKeys.html",
+ "flac-s24.flac",
+ "flac-s24.flac^headers^",
+ "flac-noheader-s16.flac",
+ "flac-noheader-s16.flac^headers^",
+ "flac-sample.mp4",
+ "flac-sample.mp4^headers^",
+ "flac-sample-cenc.mp4",
+ "flac-sample-cenc.mp4^headers^",
+ "fragment_noplay.js",
+ "fragment_play.js",
+ "gizmo.mp4",
+ "gizmo.mp4^headers^",
+ "gizmo-noaudio.mp4",
+ "gizmo-noaudio.mp4^headers^",
+ "gizmo-short.mp4",
+ "gizmo-short.mp4^headers^",
+ "gizmo.webm",
+ "gizmo.webm^headers^",
+ "gizmo-noaudio.webm",
+ "gizmo-noaudio.webm^headers^",
+ "gUM_support.js",
+ "gzipped_mp4.sjs",
+ "huge-id3.mp3",
+ "huge-id3.mp3^headers^",
+ "id3tags.mp3",
+ "id3tags.mp3^headers^",
+ "invalid-cmap-s0c0.opus",
+ "invalid-cmap-s0c0.opus^headers^",
+ "invalid-cmap-s0c2.opus",
+ "invalid-cmap-s0c2.opus^headers^",
+ "invalid-cmap-s1c2.opus",
+ "invalid-cmap-s1c2.opus^headers^",
+ "invalid-cmap-short.opus",
+ "invalid-cmap-short.opus^headers^",
+ "invalid-discard_on_multi_blocks.webm",
+ "invalid-discard_on_multi_blocks.webm^headers^",
+ "invalid-excess_discard.webm",
+ "invalid-excess_discard.webm^headers^",
+ "invalid-excess_neg_discard.webm",
+ "invalid-excess_neg_discard.webm^headers^",
+ "invalid-m0c0.opus",
+ "invalid-m0c0.opus^headers^",
+ "invalid-m0c3.opus",
+ "invalid-m0c3.opus^headers^",
+ "invalid-m1c0.opus",
+ "invalid-m1c0.opus^headers^",
+ "invalid-m1c9.opus",
+ "invalid-m1c9.opus^headers^",
+ "invalid-m2c0.opus",
+ "invalid-m2c0.opus^headers^",
+ "invalid-m2c1.opus",
+ "invalid-m2c1.opus^headers^",
+ "invalid-neg_discard.webm",
+ "invalid-neg_discard.webm^headers^",
+ "invalid-preskip.webm",
+ "invalid-preskip.webm^headers^",
+ "manifest.js",
+ "midflight-redirect.sjs",
+ "multiple-bos.ogg",
+ "multiple-bos.ogg^headers^",
+ "multiple-bos-more-header-fileds.ogg",
+ "multiple-bos-more-header-fileds.ogg^headers^",
+ "multi_id3v2.mp3",
+ "no-container-codec-delay.webm",
+ "no-cues.webm",
+ "no-cues.webm^headers^",
+ "notags.mp3",
+ "notags.mp3^headers^",
+ "opus-mapping2.mp4",
+ "opus-mapping2.mp4^headers^",
+ "opus-mapping2.webm",
+ "opus-mapping2.webm^headers^",
+ "opus-sample.mp4",
+ "opus-sample.mp4^headers^",
+ "opus-sample-cenc.mp4",
+ "opus-sample-cenc.mp4^headers^",
+ "owl-funnier-id3.mp3",
+ "owl-funnier-id3.mp3^headers^",
+ "owl-funny-id3.mp3",
+ "owl-funny-id3.mp3^headers^",
+ "owl.mp3",
+ "owl.mp3^headers^",
+ "owl-short.mp3",
+ "owl-short.mp3^headers^",
+ "pixel_aspect_ratio.mp4",
+ "play_promise.js",
+ "poster-test.jpg",
+ "r11025_msadpcm_c1.wav",
+ "r11025_msadpcm_c1.wav^headers^",
+ "r11025_s16_c1.wav",
+ "r11025_s16_c1.wav^headers^",
+ "r11025_s16_c1_trailing.wav",
+ "r11025_s16_c1_trailing.wav^headers^",
+ "r11025_s16_c1-short.wav",
+ "r11025_s16_c1-short.wav^headers^",
+ "r11025_u8_c1.wav",
+ "r11025_u8_c1.wav^headers^",
+ "r11025_u8_c1_trunc.wav",
+ "r11025_u8_c1_trunc.wav^headers^",
+ "r16000_u8_c1_list.wav",
+ "r16000_u8_c1_list.wav^headers^",
+ "reactivate_helper.html",
+ "red-46x48.mp4",
+ "red-46x48.mp4^headers^",
+ "red-48x46.mp4",
+ "red-48x46.mp4^headers^",
+ "redirect.sjs",
+ "referer.sjs",
+ "resolution-change.webm",
+ "resolution-change.webm^headers^",
+ "sample.3gp",
+ "sample.3g2",
+ "sample-encrypted-sgpdstbl-sbgptraf.mp4",
+ "sample-encrypted-sgpdstbl-sbgptraf.mp4^headers^",
+ "sample-fisbone-skeleton4.ogv",
+ "sample-fisbone-skeleton4.ogv^headers^",
+ "sample-fisbone-wrong-header.ogv",
+ "sample-fisbone-wrong-header.ogv^headers^",
+ "seek.ogv",
+ "seek.ogv^headers^",
+ "seek-short.ogv",
+ "seek-short.ogv^headers^",
+ "seek.webm",
+ "seek.webm^headers^",
+ "seek-short.webm",
+ "seek-short.webm^headers^",
+ "seek_support.js",
+ "seekLies.sjs",
+ "seek_with_sound.ogg^headers^",
+ "short-cenc.mp4",
+ "sine.webm",
+ "sine.webm^headers^",
+ "sintel-short-clearkey-subsample-encrypted-audio.webm",
+ "sintel-short-clearkey-subsample-encrypted-audio.webm^headers^",
+ "sintel-short-clearkey-subsample-encrypted-video.webm",
+ "sintel-short-clearkey-subsample-encrypted-video.webm^headers^",
+ "short.mp4",
+ "short.mp4.gz",
+ "short.mp4^headers^",
+ "short-aac-encrypted-audio.mp4",
+ "short-aac-encrypted-audio.mp4^headers^",
+ "short-audio-fragmented-cenc-without-pssh.mp4",
+ "short-audio-fragmented-cenc-without-pssh.mp4^headers^",
+ "short-video.ogv",
+ "short-video.ogv^headers^",
+ "short-vp9-encrypted-video.mp4",
+ "short-vp9-encrypted-video.mp4^headers^",
+ "small-shot-mp3.mp4",
+ "small-shot-mp3.mp4^headers^",
+ "small-shot.m4a",
+ "small-shot.mp3",
+ "small-shot.mp3^headers^",
+ "small-shot.ogg",
+ "small-shot.ogg^headers^",
+ "small-shot.flac",
+ "sound.ogg",
+ "sound.ogg^headers^",
+ "spacestorm-1000Hz-100ms.ogg",
+ "spacestorm-1000Hz-100ms.ogg^headers^",
+ "split.webm",
+ "split.webm^headers^",
+ "street.mp4",
+ "street.mp4^headers^",
+ "test-1-mono.opus",
+ "test-1-mono.opus^headers^",
+ "test-2-stereo.opus",
+ "test-2-stereo.opus^headers^",
+ "test-3-LCR.opus",
+ "test-3-LCR.opus^headers^",
+ "test-4-quad.opus",
+ "test-4-quad.opus^headers^",
+ "test-5-5.0.opus",
+ "test-5-5.0.opus^headers^",
+ "test-6-5.1.opus",
+ "test-6-5.1.opus^headers^",
+ "test-7-6.1.opus",
+ "test-7-6.1.opus^headers^",
+ "test-8-7.1.opus",
+ "test-8-7.1.opus^headers^",
+ "test-stereo-phase-inversion-180.opus",
+ "test-stereo-phase-inversion-180.opus^headers^",
+ "variable-channel.ogg",
+ "variable-channel.ogg^headers^",
+ "variable-channel.opus",
+ "variable-channel.opus^headers^",
+ "variable-preskip.opus",
+ "variable-preskip.opus^headers^",
+ "variable-samplerate.ogg",
+ "variable-samplerate.ogg^headers^",
+ "variable-samplerate.opus",
+ "variable-samplerate.opus^headers^",
+ "vbr-head.mp3",
+ "vbr-head.mp3^headers^",
+ "vbr.mp3",
+ "vbr.mp3^headers^",
+ "very-short.mp3",
+ "video-overhang.ogg",
+ "video-overhang.ogg^headers^",
+ "vp9-superframes.webm",
+ "vp9-superframes.webm^headers^",
+ "vp9.webm",
+ "vp9.webm^headers^",
+ "vp9-short.webm",
+ "vp9-short.webm^headers^",
+ "vp9cake.webm",
+ "vp9cake.webm^headers^",
+ "vp9cake-short.webm",
+ "vp9cake-short.webm^headers^",
+ "wave_metadata.wav",
+ "wave_metadata.wav^headers^",
+ "wave_metadata_bad_len.wav",
+ "wave_metadata_bad_len.wav^headers^",
+ "wave_metadata_bad_no_null.wav",
+ "wave_metadata_bad_no_null.wav^headers^",
+ "wave_metadata_bad_utf8.wav",
+ "wave_metadata_bad_utf8.wav^headers^",
+ "wave_metadata_unknown_tag.wav",
+ "wave_metadata_unknown_tag.wav^headers^",
+ "wave_metadata_utf8.wav",
+ "wave_metadata_utf8.wav^headers^",
+ "wavedata_alaw.wav",
+ "wavedata_alaw.wav^headers^",
+ "wavedata_float.wav",
+ "wavedata_float.wav^headers^",
+ "wavedata_s24.wav",
+ "wavedata_s24.wav^headers^",
+ "wavedata_s16.wav",
+ "wavedata_s16.wav^headers^",
+ "wavedata_u8.wav",
+ "wavedata_u8.wav^headers^",
+ "wavedata_ulaw.wav",
+ "wavedata_ulaw.wav^headers^",
+ "!/dom/canvas/test/captureStream_common.js",
+ "!/dom/html/test/reflect.js",
+ "!/dom/media/webrtc/tests/mochitests/head.js",
+ "hls/bipbop_16x9_single.m3u8",
+ "hls/bipbop_4x3_single.m3u8",
+ "hls/bipbop_4x3_variant.m3u8",
+ "hls/400x300_prog_index.m3u8",
+ "hls/400x300_prog_index_5s.m3u8",
+ "hls/416x243_prog_index_5s.m3u8",
+ "hls/640x480_prog_index.m3u8",
+ "hls/960x720_prog_index.m3u8",
+ "hls/400x300_seg0.ts",
+ "hls/400x300_seg0_5s.ts",
+ "hls/400x300_seg1.ts",
+ "hls/416x243_seg0_5s.ts",
+ "hls/640x480_seg0.ts",
+ "hls/640x480_seg1.ts",
+ "hls/960x720_seg0.ts",
+ "hls/960x720_seg1.ts",
+ "sync.webm",
+]
+
+["test_eme_createMediaKeys_iframes.html"]
+skip-if = ["os == 'android'"] # bug 1149374
+scheme = "https"
+
+["test_eme_detach_media_keys.html"]
+skip-if = ["os == 'android'"]
+scheme = "https"
+
+["test_eme_getstatusforpolicy.html"]
+skip-if = ["os == 'android'"] # bug 1149374
+scheme = "https"
+
+["test_eme_requestKeySystemAccess.html"]
+skip-if = ["os == 'android'"] # bug 1149374
+scheme = "https"
+
+["test_eme_requestMediaKeySystemAccess_with_app_approval.html"]
+skip-if = ["os == 'android'"] # bug 1149374
+scheme = "https"
+
+["test_eme_request_notifications.html"]
+skip-if = ["os == 'android'"]
+scheme = "https"
+
+["test_eme_session_callable_value.html"]
+scheme = "https"
+
+["test_eme_setMediaKeys_before_attach_MediaSource.html"]
+skip-if = ["os == 'android'"] # bug 1149374
+scheme = "https"
diff --git a/dom/media/test/mochitest_media_engine.toml b/dom/media/test/mochitest_media_engine.toml
new file mode 100644
index 0000000000..c9ad881aec
--- /dev/null
+++ b/dom/media/test/mochitest_media_engine.toml
@@ -0,0 +1,10 @@
+[DEFAULT]
+subsuite = "media"
+tags = "media-engine-compatible"
+skip-if = ["!wmfme"]
+
+["test_eme_mfcdm_generate_request.html"]
+
+["test_eme_wideinve_l1_installation.html"]
+
+["test_eme_mfcdm_getstatusforpolicy.html"]
diff --git a/dom/media/test/mochitest_media_recorder.toml b/dom/media/test/mochitest_media_recorder.toml
new file mode 100644
index 0000000000..428f99601d
--- /dev/null
+++ b/dom/media/test/mochitest_media_recorder.toml
@@ -0,0 +1,856 @@
+# Media tests should be backend independent, i.e., not conditioned on ogg,
+# wave etc. (The only exception is the can_play_type tests, which
+# necessarily depend on the backend(s) configured.) As far as possible, each
+# test should work with any resource type. This makes it easy to add new
+# backends and reduces the amount of test duplication.
+
+# For each supported backend, resources that can be played by that backend
+# should be added to the lists in manifest.js. Media tests that aren't
+# testing for a bug in handling a specific resource type should pick one of
+# the lists in manifest.js and run the test for each resource in the list
+# that is supported in the current build (the canPlayType API is useful for
+# this).
+
+# To test whether a valid resource can simply be played through correctly,
+# and optionally that its metadata is read correctly, just add it to
+# gPlayTests in manifest.js. To test whether an invalid resource correctly
+# throws an error (and does not cause a crash or hang), just add it to
+# gErrorTests in manifest.js.
+
+# To test for a specific bug in handling a specific resource type, make the
+# test first check canPlayType for the type, and if it's not supported, just
+# do ok(true, "Type not supported") and stop the test.
+
+[DEFAULT]
+subsuite = "media"
+tags = "mtg"
+support-files = [
+ "16bit_wave_extrametadata.wav",
+ "16bit_wave_extrametadata.wav^headers^",
+ "320x240.ogv",
+ "320x240.ogv^headers^",
+ "448636.ogv",
+ "448636.ogv^headers^",
+ "A4.ogv",
+ "A4.ogv^headers^",
+ "VID_0001.ogg",
+ "VID_0001.ogg^headers^",
+ "allowed.sjs",
+ "ambisonics.mp4",
+ "ambisonics.mp4^headers^",
+ "audio-gaps.ogg",
+ "audio-gaps.ogg^headers^",
+ "audio-gaps-short.ogg",
+ "audio-gaps-short.ogg^headers^",
+ "audio-overhang.ogg",
+ "audio-overhang.ogg^headers^",
+ "audio.wav",
+ "audio.wav^headers^",
+ "av1.mp4",
+ "av1.mp4^headers^",
+ "background_video.js",
+ "badtags.ogg",
+ "badtags.ogg^headers^",
+ "bear-640x360-v_frag-cenc-key_rotation.mp4",
+ "bear-640x360-a_frag-cenc-key_rotation.mp4",
+ "beta-phrasebook.ogg",
+ "beta-phrasebook.ogg^headers^",
+ "big.wav",
+ "big.wav^headers^",
+ "big-buck-bunny-cenc-avc3-1.m4s",
+ "big-buck-bunny-cenc-avc3-1.m4s^headers^",
+ "big-buck-bunny-cenc-avc3-init.mp4",
+ "big-buck-bunny-cenc-avc3-init.mp4^headers^",
+ "big-short.wav",
+ "big-short.wav^headers^",
+ "bipbop.mp4",
+ "bipbop-cenc-audio1.m4s",
+ "bipbop-cenc-audio1.m4s^headers^",
+ "bipbop-cenc-audio2.m4s",
+ "bipbop-cenc-audio2.m4s^headers^",
+ "bipbop-cenc-audio3.m4s",
+ "bipbop-cenc-audio3.m4s^headers^",
+ "bipbop-cenc-audioinit.mp4",
+ "bipbop-cenc-audioinit.mp4^headers^",
+ "bipbop-cenc-video1.m4s",
+ "bipbop-cenc-video1.m4s^headers^",
+ "bipbop-cenc-video2.m4s",
+ "bipbop-cenc-video2.m4s^headers^",
+ "bipbop-cenc-videoinit.mp4",
+ "bipbop-cenc-videoinit.mp4^headers^",
+ "bipbop-cenc-video-10s.mp4",
+ "bipbop-cenc-video-10s.mp4^headers^",
+ "bipbop-clearkey-keyrotation-clear-lead-audio.mp4",
+ "bipbop-clearkey-keyrotation-clear-lead-audio.mp4^headers^",
+ "bipbop-clearkey-keyrotation-clear-lead-video.mp4",
+ "bipbop-clearkey-keyrotation-clear-lead-video.mp4^headers^",
+ "bipbop_225w_175kbps.mp4",
+ "bipbop_225w_175kbps.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-1.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-2.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-3.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-4.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-init.mp4",
+ "bipbop_225w_175kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-1.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-2.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-3.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-4.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-init.mp4",
+ "bipbop_225w_175kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-video-key1-1.m4s",
+ "bipbop_225w_175kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-video-key1-init.mp4",
+ "bipbop_225w_175kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-video-key2-1.m4s",
+ "bipbop_225w_175kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-video-key2-init.mp4",
+ "bipbop_225w_175kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-1.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-2.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-3.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-4.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-1.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-2.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-3.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-4.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-init.mp4",
+ "bipbop_300_215kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-1.m4s",
+ "bipbop_300_215kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-2.m4s",
+ "bipbop_300_215kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-1.m4s",
+ "bipbop_300_215kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-2.m4s",
+ "bipbop_300_215kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-init.mp4",
+ "bipbop_300_215kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-1.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-2.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-3.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-4.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-init.mp4",
+ "bipbop_300wp_227kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-1.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-2.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-3.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-4.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-init.mp4",
+ "bipbop_300wp_227kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-1.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-2.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-init.mp4",
+ "bipbop_300wp_227kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-1.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-2.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-init.mp4",
+ "bipbop_300wp_227kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-1.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-2.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-3.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-4.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-init.mp4",
+ "bipbop_360w_253kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-1.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-2.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-3.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-4.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-init.mp4",
+ "bipbop_360w_253kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-video-key1-1.m4s",
+ "bipbop_360w_253kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-video-key1-init.mp4",
+ "bipbop_360w_253kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-video-key2-1.m4s",
+ "bipbop_360w_253kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-video-key2-init.mp4",
+ "bipbop_360w_253kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-clearkey-audio.webm",
+ "bipbop_360w_253kbps-clearkey-audio.webm^headers^",
+ "bipbop_360w_253kbps-clearkey-video-vp8.webm",
+ "bipbop_360w_253kbps-clearkey-video-vp8.webm^headers^",
+ "bipbop_360w_253kbps-clearkey-video-vp9.webm",
+ "bipbop_360w_253kbps-clearkey-video-vp9.webm^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480_624kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-2.m4s",
+ "bipbop_480_624kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-1.m4s",
+ "bipbop_480_624kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-2.m4s",
+ "bipbop_480_624kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480_959kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480_959kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-1.m4s",
+ "bipbop_480_959kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-2.m4s",
+ "bipbop_480_959kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-init.mp4",
+ "bipbop_480_959kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-1.m4s",
+ "bipbop_480_959kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-2.m4s",
+ "bipbop_480_959kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-init.mp4",
+ "bipbop_480_959kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480wp_663kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480wp_663kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-1.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-2.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-init.mp4",
+ "bipbop_480wp_663kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-1.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-2.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-init.mp4",
+ "bipbop_480wp_663kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_audio_aac_8k.mp4",
+ "bipbop_audio_aac_8k.mp4^headers^",
+ "bipbop_audio_aac_22.05k.mp4",
+ "bipbop_audio_aac_22.05k.mp4^headers^",
+ "bipbop_audio_aac_44.1k.mp4",
+ "bipbop_audio_aac_44.1k.mp4^headers^",
+ "bipbop_audio_aac_48k.mp4",
+ "bipbop_audio_aac_48k.mp4^headers^",
+ "bipbop_audio_aac_88.2k.mp4",
+ "bipbop_audio_aac_88.2k.mp4^headers^",
+ "bipbop_audio_aac_96k.mp4",
+ "bipbop_audio_aac_96k.mp4^headers^",
+ "bipbop_cbcs_1_9_audio_1.m4s",
+ "bipbop_cbcs_1_9_audio_1.m4s^headers^",
+ "bipbop_cbcs_1_9_audio_init.mp4",
+ "bipbop_cbcs_1_9_audio_init.mp4^headers^",
+ "bipbop_cbcs_1_9_video_1.m4s",
+ "bipbop_cbcs_1_9_video_1.m4s^headers^",
+ "bipbop_cbcs_1_9_video_init.mp4",
+ "bipbop_cbcs_1_9_video_init.mp4^headers^",
+ "bipbop_cbcs_5_5_audio_1.m4s",
+ "bipbop_cbcs_5_5_audio_1.m4s^headers^",
+ "bipbop_cbcs_5_5_audio_init.mp4",
+ "bipbop_cbcs_5_5_audio_init.mp4^headers^",
+ "bipbop_cbcs_5_5_video_1.m4s",
+ "bipbop_cbcs_5_5_video_1.m4s^headers^",
+ "bipbop_cbcs_5_5_video_init.mp4",
+ "bipbop_cbcs_5_5_video_init.mp4^headers^",
+ "bipbop_cbcs_7_7_audio_1.m4s",
+ "bipbop_cbcs_7_7_audio_1.m4s^headers^",
+ "bipbop_cbcs_7_7_audio_init.mp4",
+ "bipbop_cbcs_7_7_audio_init.mp4^headers^",
+ "bipbop_cbcs_7_7_video_1.m4s",
+ "bipbop_cbcs_7_7_video_1.m4s^headers^",
+ "bipbop_cbcs_7_7_video_init.mp4",
+ "bipbop_cbcs_7_7_video_init.mp4^headers^",
+ "bipbop_cbcs_9_8_audio_1.m4s",
+ "bipbop_cbcs_9_8_audio_1.m4s^headers^",
+ "bipbop_cbcs_9_8_audio_init.mp4",
+ "bipbop_cbcs_9_8_audio_init.mp4^headers^",
+ "bipbop_cbcs_9_8_video_1.m4s",
+ "bipbop_cbcs_9_8_video_1.m4s^headers^",
+ "bipbop_cbcs_9_8_video_init.mp4",
+ "bipbop_cbcs_9_8_video_init.mp4^headers^",
+ "bipbop_cbcs_10_0_audio_1.m4s",
+ "bipbop_cbcs_10_0_audio_1.m4s^headers^",
+ "bipbop_cbcs_10_0_audio_init.mp4",
+ "bipbop_cbcs_10_0_audio_init.mp4^headers^",
+ "bipbop_cbcs_10_0_video_1.m4s",
+ "bipbop_cbcs_10_0_video_1.m4s^headers^",
+ "bipbop_cbcs_10_0_video_init.mp4",
+ "bipbop_cbcs_10_0_video_init.mp4^headers^",
+ "bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm",
+ "bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm",
+ "bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm",
+ "bipbop_short_vp8.webm",
+ "bipbop_short_vp8.webm^headers^",
+ "bipbop-lateaudio.mp4",
+ "bipbop-lateaudio.mp4^headers^",
+ "black100x100-aspect3to2.ogv",
+ "black100x100-aspect3to2.ogv^headers^",
+ "bogus.duh",
+ "bogus.ogv",
+ "bogus.ogv^headers^",
+ "bogus.wav",
+ "bogus.wav^headers^",
+ "bug461281.ogg",
+ "bug461281.ogg^headers^",
+ "bug482461-theora.ogv",
+ "bug482461-theora.ogv^headers^",
+ "bug482461.ogv",
+ "bug482461.ogv^headers^",
+ "bug495129.ogv",
+ "bug495129.ogv^headers^",
+ "bug495794.ogg",
+ "bug495794.ogg^headers^",
+ "bug498380.ogv",
+ "bug498380.ogv^headers^",
+ "bug498855-1.ogv",
+ "bug498855-1.ogv^headers^",
+ "bug498855-2.ogv",
+ "bug498855-2.ogv^headers^",
+ "bug498855-3.ogv",
+ "bug498855-3.ogv^headers^",
+ "bug499519.ogv",
+ "bug499519.ogv^headers^",
+ "bug500311.ogv",
+ "bug500311.ogv^headers^",
+ "bug501279.ogg",
+ "bug501279.ogg^headers^",
+ "bug504613.ogv",
+ "bug504613.ogv^headers^",
+ "bug504644.ogv",
+ "bug504644.ogv^headers^",
+ "bug504843.ogv",
+ "bug504843.ogv^headers^",
+ "bug506094.ogv",
+ "bug506094.ogv^headers^",
+ "bug516323.indexed.ogv",
+ "bug516323.indexed.ogv^headers^",
+ "bug516323.ogv",
+ "bug516323.ogv^headers^",
+ "bug520493.ogg",
+ "bug520493.ogg^headers^",
+ "bug520500.ogg",
+ "bug520500.ogg^headers^",
+ "bug520908.ogv",
+ "bug520908.ogv^headers^",
+ "bug523816.ogv",
+ "bug523816.ogv^headers^",
+ "bug533822.ogg",
+ "bug533822.ogg^headers^",
+ "bug556821.ogv",
+ "bug556821.ogv^headers^",
+ "bug557094.ogv",
+ "bug557094.ogv^headers^",
+ "bug604067.webm",
+ "bug604067.webm^headers^",
+ "bug1066943.webm",
+ "bug1066943.webm^headers^",
+ "bug1301226.wav",
+ "bug1301226.wav^headers^",
+ "bug1301226-odd.wav",
+ "bug1301226-odd.wav^headers^",
+ "bug1377278.webm",
+ "bug1377278.webm^headers^",
+ "bunny.webm",
+ "can_play_type_dash.js",
+ "can_play_type_ogg.js",
+ "can_play_type_wave.js",
+ "can_play_type_webm.js",
+ "cancellable_request.sjs",
+ "chain.ogg",
+ "chain.ogg^headers^",
+ "chain.ogv",
+ "chain.ogv^headers^",
+ "chain.opus",
+ "chain.opus^headers^",
+ "chained-audio-video.ogg",
+ "chained-audio-video.ogg^headers^",
+ "chained-video.ogv",
+ "chained-video.ogv^headers^",
+ "chromeHelper.js",
+ "cloneElementVisually_helpers.js",
+ "contentType.sjs",
+ "detodos.opus",
+ "detodos.opus^headers^",
+ "detodos.webm",
+ "detodos.webm^headers^",
+ "detodos-short.webm",
+ "detodos-short.webm^headers^",
+ "detodos-recorder-test.opus",
+ "detodos-recorder-test.opus^headers^",
+ "detodos-short.opus",
+ "detodos-short.opus^headers^",
+ "dirac.ogg",
+ "dirac.ogg^headers^",
+ "dynamic_resource.sjs",
+ "eme_standalone.js",
+ "eme.js",
+ "empty_size.mp3",
+ "file_access_controls.html",
+ "file_eme_createMediaKeys.html",
+ "flac-s24.flac",
+ "flac-s24.flac^headers^",
+ "flac-noheader-s16.flac",
+ "flac-noheader-s16.flac^headers^",
+ "flac-sample.mp4",
+ "flac-sample.mp4^headers^",
+ "flac-sample-cenc.mp4",
+ "flac-sample-cenc.mp4^headers^",
+ "fragment_noplay.js",
+ "fragment_play.js",
+ "gizmo.mp4",
+ "gizmo.mp4^headers^",
+ "gizmo-noaudio.mp4",
+ "gizmo-noaudio.mp4^headers^",
+ "gizmo-short.mp4",
+ "gizmo-short.mp4^headers^",
+ "gizmo.webm",
+ "gizmo.webm^headers^",
+ "gizmo-noaudio.webm",
+ "gizmo-noaudio.webm^headers^",
+ "gUM_support.js",
+ "gzipped_mp4.sjs",
+ "huge-id3.mp3",
+ "huge-id3.mp3^headers^",
+ "id3tags.mp3",
+ "id3tags.mp3^headers^",
+ "invalid-cmap-s0c0.opus",
+ "invalid-cmap-s0c0.opus^headers^",
+ "invalid-cmap-s0c2.opus",
+ "invalid-cmap-s0c2.opus^headers^",
+ "invalid-cmap-s1c2.opus",
+ "invalid-cmap-s1c2.opus^headers^",
+ "invalid-cmap-short.opus",
+ "invalid-cmap-short.opus^headers^",
+ "invalid-discard_on_multi_blocks.webm",
+ "invalid-discard_on_multi_blocks.webm^headers^",
+ "invalid-excess_discard.webm",
+ "invalid-excess_discard.webm^headers^",
+ "invalid-excess_neg_discard.webm",
+ "invalid-excess_neg_discard.webm^headers^",
+ "invalid-m0c0.opus",
+ "invalid-m0c0.opus^headers^",
+ "invalid-m0c3.opus",
+ "invalid-m0c3.opus^headers^",
+ "invalid-m1c0.opus",
+ "invalid-m1c0.opus^headers^",
+ "invalid-m1c9.opus",
+ "invalid-m1c9.opus^headers^",
+ "invalid-m2c0.opus",
+ "invalid-m2c0.opus^headers^",
+ "invalid-m2c1.opus",
+ "invalid-m2c1.opus^headers^",
+ "invalid-neg_discard.webm",
+ "invalid-neg_discard.webm^headers^",
+ "invalid-preskip.webm",
+ "invalid-preskip.webm^headers^",
+ "manifest.js",
+ "midflight-redirect.sjs",
+ "multiple-bos.ogg",
+ "multiple-bos.ogg^headers^",
+ "multiple-bos-more-header-fileds.ogg",
+ "multiple-bos-more-header-fileds.ogg^headers^",
+ "multi_id3v2.mp3",
+ "no-container-codec-delay.webm",
+ "no-cues.webm",
+ "no-cues.webm^headers^",
+ "notags.mp3",
+ "notags.mp3^headers^",
+ "opus-mapping2.mp4",
+ "opus-mapping2.mp4^headers^",
+ "opus-mapping2.webm",
+ "opus-mapping2.webm^headers^",
+ "opus-sample.mp4",
+ "opus-sample.mp4^headers^",
+ "opus-sample-cenc.mp4",
+ "opus-sample-cenc.mp4^headers^",
+ "owl-funnier-id3.mp3",
+ "owl-funnier-id3.mp3^headers^",
+ "owl-funny-id3.mp3",
+ "owl-funny-id3.mp3^headers^",
+ "owl.mp3",
+ "owl.mp3^headers^",
+ "owl-short.mp3",
+ "owl-short.mp3^headers^",
+ "pixel_aspect_ratio.mp4",
+ "play_promise.js",
+ "poster-test.jpg",
+ "r11025_msadpcm_c1.wav",
+ "r11025_msadpcm_c1.wav^headers^",
+ "r11025_s16_c1.wav",
+ "r11025_s16_c1.wav^headers^",
+ "r11025_s16_c1_trailing.wav",
+ "r11025_s16_c1_trailing.wav^headers^",
+ "r11025_s16_c1-short.wav",
+ "r11025_s16_c1-short.wav^headers^",
+ "r11025_u8_c1.wav",
+ "r11025_u8_c1.wav^headers^",
+ "r11025_u8_c1_trunc.wav",
+ "r11025_u8_c1_trunc.wav^headers^",
+ "r16000_u8_c1_list.wav",
+ "r16000_u8_c1_list.wav^headers^",
+ "reactivate_helper.html",
+ "red-46x48.mp4",
+ "red-46x48.mp4^headers^",
+ "red-48x46.mp4",
+ "red-48x46.mp4^headers^",
+ "redirect.sjs",
+ "referer.sjs",
+ "resolution-change.webm",
+ "resolution-change.webm^headers^",
+ "sample.3gp",
+ "sample.3g2",
+ "sample-encrypted-sgpdstbl-sbgptraf.mp4",
+ "sample-encrypted-sgpdstbl-sbgptraf.mp4^headers^",
+ "sample-fisbone-skeleton4.ogv",
+ "sample-fisbone-skeleton4.ogv^headers^",
+ "sample-fisbone-wrong-header.ogv",
+ "sample-fisbone-wrong-header.ogv^headers^",
+ "seek.ogv",
+ "seek.ogv^headers^",
+ "seek-short.ogv",
+ "seek-short.ogv^headers^",
+ "seek.webm",
+ "seek.webm^headers^",
+ "seek-short.webm",
+ "seek-short.webm^headers^",
+ "seek_support.js",
+ "seekLies.sjs",
+ "seek_with_sound.ogg^headers^",
+ "short-cenc.mp4",
+ "sine.webm",
+ "sine.webm^headers^",
+ "sintel-short-clearkey-subsample-encrypted-audio.webm",
+ "sintel-short-clearkey-subsample-encrypted-audio.webm^headers^",
+ "sintel-short-clearkey-subsample-encrypted-video.webm",
+ "sintel-short-clearkey-subsample-encrypted-video.webm^headers^",
+ "short.mp4",
+ "short.mp4.gz",
+ "short.mp4^headers^",
+ "short-aac-encrypted-audio.mp4",
+ "short-aac-encrypted-audio.mp4^headers^",
+ "short-audio-fragmented-cenc-without-pssh.mp4",
+ "short-audio-fragmented-cenc-without-pssh.mp4^headers^",
+ "short-video.ogv",
+ "short-video.ogv^headers^",
+ "short-vp9-encrypted-video.mp4",
+ "short-vp9-encrypted-video.mp4^headers^",
+ "small-shot-mp3.mp4",
+ "small-shot-mp3.mp4^headers^",
+ "small-shot.m4a",
+ "small-shot.mp3",
+ "small-shot.mp3^headers^",
+ "small-shot.ogg",
+ "small-shot.ogg^headers^",
+ "small-shot.flac",
+ "sound.ogg",
+ "sound.ogg^headers^",
+ "spacestorm-1000Hz-100ms.ogg",
+ "spacestorm-1000Hz-100ms.ogg^headers^",
+ "split.webm",
+ "split.webm^headers^",
+ "street.mp4",
+ "street.mp4^headers^",
+ "test-1-mono.opus",
+ "test-1-mono.opus^headers^",
+ "test-2-stereo.opus",
+ "test-2-stereo.opus^headers^",
+ "test-3-LCR.opus",
+ "test-3-LCR.opus^headers^",
+ "test-4-quad.opus",
+ "test-4-quad.opus^headers^",
+ "test-5-5.0.opus",
+ "test-5-5.0.opus^headers^",
+ "test-6-5.1.opus",
+ "test-6-5.1.opus^headers^",
+ "test-7-6.1.opus",
+ "test-7-6.1.opus^headers^",
+ "test-8-7.1.opus",
+ "test-8-7.1.opus^headers^",
+ "test-stereo-phase-inversion-180.opus",
+ "test-stereo-phase-inversion-180.opus^headers^",
+ "variable-channel.ogg",
+ "variable-channel.ogg^headers^",
+ "variable-channel.opus",
+ "variable-channel.opus^headers^",
+ "variable-preskip.opus",
+ "variable-preskip.opus^headers^",
+ "variable-samplerate.ogg",
+ "variable-samplerate.ogg^headers^",
+ "variable-samplerate.opus",
+ "variable-samplerate.opus^headers^",
+ "vbr-head.mp3",
+ "vbr-head.mp3^headers^",
+ "vbr.mp3",
+ "vbr.mp3^headers^",
+ "very-short.mp3",
+ "video-overhang.ogg",
+ "video-overhang.ogg^headers^",
+ "vp9-superframes.webm",
+ "vp9-superframes.webm^headers^",
+ "vp9.webm",
+ "vp9.webm^headers^",
+ "vp9-short.webm",
+ "vp9-short.webm^headers^",
+ "vp9cake.webm",
+ "vp9cake.webm^headers^",
+ "vp9cake-short.webm",
+ "vp9cake-short.webm^headers^",
+ "wave_metadata.wav",
+ "wave_metadata.wav^headers^",
+ "wave_metadata_bad_len.wav",
+ "wave_metadata_bad_len.wav^headers^",
+ "wave_metadata_bad_no_null.wav",
+ "wave_metadata_bad_no_null.wav^headers^",
+ "wave_metadata_bad_utf8.wav",
+ "wave_metadata_bad_utf8.wav^headers^",
+ "wave_metadata_unknown_tag.wav",
+ "wave_metadata_unknown_tag.wav^headers^",
+ "wave_metadata_utf8.wav",
+ "wave_metadata_utf8.wav^headers^",
+ "wavedata_alaw.wav",
+ "wavedata_alaw.wav^headers^",
+ "wavedata_float.wav",
+ "wavedata_float.wav^headers^",
+ "wavedata_s24.wav",
+ "wavedata_s24.wav^headers^",
+ "wavedata_s16.wav",
+ "wavedata_s16.wav^headers^",
+ "wavedata_u8.wav",
+ "wavedata_u8.wav^headers^",
+ "wavedata_ulaw.wav",
+ "wavedata_ulaw.wav^headers^",
+ "!/dom/canvas/test/captureStream_common.js",
+ "!/dom/html/test/reflect.js",
+ "!/dom/media/webrtc/tests/mochitests/head.js",
+ "hls/bipbop_16x9_single.m3u8",
+ "hls/bipbop_4x3_single.m3u8",
+ "hls/bipbop_4x3_variant.m3u8",
+ "hls/400x300_prog_index.m3u8",
+ "hls/400x300_prog_index_5s.m3u8",
+ "hls/416x243_prog_index_5s.m3u8",
+ "hls/640x480_prog_index.m3u8",
+ "hls/960x720_prog_index.m3u8",
+ "hls/400x300_seg0.ts",
+ "hls/400x300_seg0_5s.ts",
+ "hls/400x300_seg1.ts",
+ "hls/416x243_seg0_5s.ts",
+ "hls/640x480_seg0.ts",
+ "hls/640x480_seg1.ts",
+ "hls/960x720_seg0.ts",
+ "hls/960x720_seg1.ts",
+ "sync.webm",
+]
+prefs = ["media.recorder.video.frame_drops=false"]
+
+["test_mediacapabilities_resistfingerprinting.html"]
+
+["test_mediarecorder_avoid_recursion.html"]
+skip-if = ["os == 'win' && !debug"]
+scheme = "https"
+
+["test_mediarecorder_bitrate.html"]
+skip-if = ["os == 'android'"] # bug 1297432, android(bug 1232305)
+
+["test_mediarecorder_creation.html"]
+tags = "mtg capturestream"
+
+["test_mediarecorder_creation_fail.html"]
+
+["test_mediarecorder_fires_start_event_once_when_erroring.html"]
+
+["test_mediarecorder_multipletracks.html"]
+
+["test_mediarecorder_onerror_pause.html"]
+scheme = "https"
+
+["test_mediarecorder_pause_resume_video.html"]
+skip-if = ["os == 'android'"] # android(bug 1232305)
+
+["test_mediarecorder_playback_can_repeat.html"]
+
+["test_mediarecorder_principals.html"]
+skip-if = [
+ "os == 'win' && os_version == '10.0'", # Bug 1453375
+ "os == 'android'", # Bug 1694645
+]
+
+["test_mediarecorder_record_4ch_audiocontext.html"]
+skip-if = ["os == 'linux' && bits == 64"] #Bug 1598101
+
+["test_mediarecorder_record_addtracked_stream.html"]
+skip-if = ["os == 'android'"] # Bug 1408241
+tags = "mtg capturestream"
+
+["test_mediarecorder_record_audiocontext.html"]
+
+["test_mediarecorder_record_audiocontext_mlk.html"]
+
+["test_mediarecorder_record_audionode.html"]
+
+["test_mediarecorder_record_canvas_captureStream.html"]
+skip-if = ["os == 'android'"] # android(bug 1232305)
+
+["test_mediarecorder_record_changing_video_resolution.html"]
+skip-if = [
+ "os == 'android'", # android(bug 1232305)
+ "os == 'linux' && (tsan || asan || debug)", # 1770504 (high intermittent failure on Linux tsan)
+]
+
+["test_mediarecorder_record_downsize_resolution.html"]
+skip-if = ["os == 'android'"] # android(bug 1232305)
+
+["test_mediarecorder_record_getdata_afterstart.html"]
+tags = "mtg capturestream"
+
+["test_mediarecorder_record_gum_video_timeslice.html"]
+scheme = "https"
+
+["test_mediarecorder_record_gum_video_timeslice_mixed.html"]
+scheme = "https"
+
+["test_mediarecorder_record_immediate_stop.html"]
+tags = "mtg capturestream"
+
+["test_mediarecorder_record_no_timeslice.html"]
+tags = "mtg capturestream"
+
+["test_mediarecorder_record_session.html"]
+tags = "mtg capturestream"
+
+["test_mediarecorder_record_startstopstart.html"]
+
+["test_mediarecorder_record_timeslice.html"]
+tags = "mtg capturestream"
+
+["test_mediarecorder_record_upsize_resolution.html"]
+skip-if = ["os == 'android'"] # android(bug 1232305)
+
+["test_mediarecorder_reload_crash.html"]
+tags = "mtg capturestream"
+
+["test_mediarecorder_state_event_order.html"]
+tags = "mtg capturestream"
+
+["test_mediarecorder_state_transition.html"]
+tags = "mtg capturestream"
+
+["test_mediarecorder_webm_support.html"]
diff --git a/dom/media/test/mochitest_seek.toml b/dom/media/test/mochitest_seek.toml
new file mode 100644
index 0000000000..0c90bd1bfb
--- /dev/null
+++ b/dom/media/test/mochitest_seek.toml
@@ -0,0 +1,824 @@
+# Media tests should be backend independent, i.e., not conditioned on ogg,
+# wave etc. (The only exception is the can_play_type tests, which
+# necessarily depend on the backend(s) configured.) As far as possible, each
+# test should work with any resource type. This makes it easy to add new
+# backends and reduces the amount of test duplication.
+
+# For each supported backend, resources that can be played by that backend
+# should be added to the lists in manifest.js. Media tests that aren't
+# testing for a bug in handling a specific resource type should pick one of
+# the lists in manifest.js and run the test for each resource in the list
+# that is supported in the current build (the canPlayType API is useful for
+# this).
+
+# To test whether a valid resource can simply be played through correctly,
+# and optionally that its metadata is read correctly, just add it to
+# gPlayTests in manifest.js. To test whether an invalid resource correctly
+# throws an error (and does not cause a crash or hang), just add it to
+# gErrorTests in manifest.js.
+
+# To test for a specific bug in handling a specific resource type, make the
+# test first check canPlayType for the type, and if it's not supported, just
+# do ok(true, "Type not supported") and stop the test.
+
+[DEFAULT]
+subsuite = "media"
+tags = "media-gpu"
+support-files = [
+ "16bit_wave_extrametadata.wav",
+ "16bit_wave_extrametadata.wav^headers^",
+ "320x240.ogv",
+ "320x240.ogv^headers^",
+ "448636.ogv",
+ "448636.ogv^headers^",
+ "A4.ogv",
+ "A4.ogv^headers^",
+ "VID_0001.ogg",
+ "VID_0001.ogg^headers^",
+ "allowed.sjs",
+ "ambisonics.mp4",
+ "ambisonics.mp4^headers^",
+ "audio-gaps.ogg",
+ "audio-gaps.ogg^headers^",
+ "audio-gaps-short.ogg",
+ "audio-gaps-short.ogg^headers^",
+ "audio-overhang.ogg",
+ "audio-overhang.ogg^headers^",
+ "audio.wav",
+ "audio.wav^headers^",
+ "av1.mp4",
+ "av1.mp4^headers^",
+ "background_video.js",
+ "badtags.ogg",
+ "badtags.ogg^headers^",
+ "bear-640x360-v_frag-cenc-key_rotation.mp4",
+ "bear-640x360-a_frag-cenc-key_rotation.mp4",
+ "beta-phrasebook.ogg",
+ "beta-phrasebook.ogg^headers^",
+ "big.wav",
+ "big.wav^headers^",
+ "big-buck-bunny-cenc-avc3-1.m4s",
+ "big-buck-bunny-cenc-avc3-1.m4s^headers^",
+ "big-buck-bunny-cenc-avc3-init.mp4",
+ "big-buck-bunny-cenc-avc3-init.mp4^headers^",
+ "big-short.wav",
+ "big-short.wav^headers^",
+ "bipbop.mp4",
+ "bipbop-cenc-audio1.m4s",
+ "bipbop-cenc-audio1.m4s^headers^",
+ "bipbop-cenc-audio2.m4s",
+ "bipbop-cenc-audio2.m4s^headers^",
+ "bipbop-cenc-audio3.m4s",
+ "bipbop-cenc-audio3.m4s^headers^",
+ "bipbop-cenc-audioinit.mp4",
+ "bipbop-cenc-audioinit.mp4^headers^",
+ "bipbop-cenc-video1.m4s",
+ "bipbop-cenc-video1.m4s^headers^",
+ "bipbop-cenc-video2.m4s",
+ "bipbop-cenc-video2.m4s^headers^",
+ "bipbop-cenc-videoinit.mp4",
+ "bipbop-cenc-videoinit.mp4^headers^",
+ "bipbop-cenc-video-10s.mp4",
+ "bipbop-cenc-video-10s.mp4^headers^",
+ "bipbop-clearkey-keyrotation-clear-lead-audio.mp4",
+ "bipbop-clearkey-keyrotation-clear-lead-audio.mp4^headers^",
+ "bipbop-clearkey-keyrotation-clear-lead-video.mp4",
+ "bipbop-clearkey-keyrotation-clear-lead-video.mp4^headers^",
+ "bipbop_225w_175kbps.mp4",
+ "bipbop_225w_175kbps.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-1.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-2.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-3.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-4.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-init.mp4",
+ "bipbop_225w_175kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-1.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-2.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-3.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-4.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-init.mp4",
+ "bipbop_225w_175kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-video-key1-1.m4s",
+ "bipbop_225w_175kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-video-key1-init.mp4",
+ "bipbop_225w_175kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-video-key2-1.m4s",
+ "bipbop_225w_175kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-video-key2-init.mp4",
+ "bipbop_225w_175kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-1.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-2.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-3.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-4.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-1.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-2.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-3.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-4.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-init.mp4",
+ "bipbop_300_215kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-1.m4s",
+ "bipbop_300_215kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-2.m4s",
+ "bipbop_300_215kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-1.m4s",
+ "bipbop_300_215kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-2.m4s",
+ "bipbop_300_215kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-init.mp4",
+ "bipbop_300_215kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-1.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-2.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-3.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-4.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-init.mp4",
+ "bipbop_300wp_227kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-1.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-2.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-3.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-4.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-init.mp4",
+ "bipbop_300wp_227kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-1.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-2.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-init.mp4",
+ "bipbop_300wp_227kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-1.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-2.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-init.mp4",
+ "bipbop_300wp_227kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-1.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-2.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-3.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-4.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-init.mp4",
+ "bipbop_360w_253kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-1.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-2.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-3.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-4.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-init.mp4",
+ "bipbop_360w_253kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-video-key1-1.m4s",
+ "bipbop_360w_253kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-video-key1-init.mp4",
+ "bipbop_360w_253kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-video-key2-1.m4s",
+ "bipbop_360w_253kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-video-key2-init.mp4",
+ "bipbop_360w_253kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-clearkey-audio.webm",
+ "bipbop_360w_253kbps-clearkey-audio.webm^headers^",
+ "bipbop_360w_253kbps-clearkey-video-vp8.webm",
+ "bipbop_360w_253kbps-clearkey-video-vp8.webm^headers^",
+ "bipbop_360w_253kbps-clearkey-video-vp9.webm",
+ "bipbop_360w_253kbps-clearkey-video-vp9.webm^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480_624kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-2.m4s",
+ "bipbop_480_624kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-1.m4s",
+ "bipbop_480_624kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-2.m4s",
+ "bipbop_480_624kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480_959kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480_959kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-1.m4s",
+ "bipbop_480_959kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-2.m4s",
+ "bipbop_480_959kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-init.mp4",
+ "bipbop_480_959kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-1.m4s",
+ "bipbop_480_959kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-2.m4s",
+ "bipbop_480_959kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-init.mp4",
+ "bipbop_480_959kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480wp_663kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480wp_663kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-1.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-2.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-init.mp4",
+ "bipbop_480wp_663kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-1.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-2.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-init.mp4",
+ "bipbop_480wp_663kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_audio_aac_8k.mp4",
+ "bipbop_audio_aac_8k.mp4^headers^",
+ "bipbop_audio_aac_22.05k.mp4",
+ "bipbop_audio_aac_22.05k.mp4^headers^",
+ "bipbop_audio_aac_44.1k.mp4",
+ "bipbop_audio_aac_44.1k.mp4^headers^",
+ "bipbop_audio_aac_48k.mp4",
+ "bipbop_audio_aac_48k.mp4^headers^",
+ "bipbop_audio_aac_88.2k.mp4",
+ "bipbop_audio_aac_88.2k.mp4^headers^",
+ "bipbop_audio_aac_96k.mp4",
+ "bipbop_audio_aac_96k.mp4^headers^",
+ "bipbop_cbcs_1_9_audio_1.m4s",
+ "bipbop_cbcs_1_9_audio_1.m4s^headers^",
+ "bipbop_cbcs_1_9_audio_init.mp4",
+ "bipbop_cbcs_1_9_audio_init.mp4^headers^",
+ "bipbop_cbcs_1_9_video_1.m4s",
+ "bipbop_cbcs_1_9_video_1.m4s^headers^",
+ "bipbop_cbcs_1_9_video_init.mp4",
+ "bipbop_cbcs_1_9_video_init.mp4^headers^",
+ "bipbop_cbcs_5_5_audio_1.m4s",
+ "bipbop_cbcs_5_5_audio_1.m4s^headers^",
+ "bipbop_cbcs_5_5_audio_init.mp4",
+ "bipbop_cbcs_5_5_audio_init.mp4^headers^",
+ "bipbop_cbcs_5_5_video_1.m4s",
+ "bipbop_cbcs_5_5_video_1.m4s^headers^",
+ "bipbop_cbcs_5_5_video_init.mp4",
+ "bipbop_cbcs_5_5_video_init.mp4^headers^",
+ "bipbop_cbcs_7_7_audio_1.m4s",
+ "bipbop_cbcs_7_7_audio_1.m4s^headers^",
+ "bipbop_cbcs_7_7_audio_init.mp4",
+ "bipbop_cbcs_7_7_audio_init.mp4^headers^",
+ "bipbop_cbcs_7_7_video_1.m4s",
+ "bipbop_cbcs_7_7_video_1.m4s^headers^",
+ "bipbop_cbcs_7_7_video_init.mp4",
+ "bipbop_cbcs_7_7_video_init.mp4^headers^",
+ "bipbop_cbcs_9_8_audio_1.m4s",
+ "bipbop_cbcs_9_8_audio_1.m4s^headers^",
+ "bipbop_cbcs_9_8_audio_init.mp4",
+ "bipbop_cbcs_9_8_audio_init.mp4^headers^",
+ "bipbop_cbcs_9_8_video_1.m4s",
+ "bipbop_cbcs_9_8_video_1.m4s^headers^",
+ "bipbop_cbcs_9_8_video_init.mp4",
+ "bipbop_cbcs_9_8_video_init.mp4^headers^",
+ "bipbop_cbcs_10_0_audio_1.m4s",
+ "bipbop_cbcs_10_0_audio_1.m4s^headers^",
+ "bipbop_cbcs_10_0_audio_init.mp4",
+ "bipbop_cbcs_10_0_audio_init.mp4^headers^",
+ "bipbop_cbcs_10_0_video_1.m4s",
+ "bipbop_cbcs_10_0_video_1.m4s^headers^",
+ "bipbop_cbcs_10_0_video_init.mp4",
+ "bipbop_cbcs_10_0_video_init.mp4^headers^",
+ "bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm",
+ "bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm",
+ "bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm",
+ "bipbop_short_vp8.webm",
+ "bipbop_short_vp8.webm^headers^",
+ "bipbop-lateaudio.mp4",
+ "bipbop-lateaudio.mp4^headers^",
+ "black100x100-aspect3to2.ogv",
+ "black100x100-aspect3to2.ogv^headers^",
+ "bogus.duh",
+ "bogus.ogv",
+ "bogus.ogv^headers^",
+ "bogus.wav",
+ "bogus.wav^headers^",
+ "bug461281.ogg",
+ "bug461281.ogg^headers^",
+ "bug482461-theora.ogv",
+ "bug482461-theora.ogv^headers^",
+ "bug482461.ogv",
+ "bug482461.ogv^headers^",
+ "bug495129.ogv",
+ "bug495129.ogv^headers^",
+ "bug495794.ogg",
+ "bug495794.ogg^headers^",
+ "bug498380.ogv",
+ "bug498380.ogv^headers^",
+ "bug498855-1.ogv",
+ "bug498855-1.ogv^headers^",
+ "bug498855-2.ogv",
+ "bug498855-2.ogv^headers^",
+ "bug498855-3.ogv",
+ "bug498855-3.ogv^headers^",
+ "bug499519.ogv",
+ "bug499519.ogv^headers^",
+ "bug500311.ogv",
+ "bug500311.ogv^headers^",
+ "bug501279.ogg",
+ "bug501279.ogg^headers^",
+ "bug504613.ogv",
+ "bug504613.ogv^headers^",
+ "bug504644.ogv",
+ "bug504644.ogv^headers^",
+ "bug504843.ogv",
+ "bug504843.ogv^headers^",
+ "bug506094.ogv",
+ "bug506094.ogv^headers^",
+ "bug516323.indexed.ogv",
+ "bug516323.indexed.ogv^headers^",
+ "bug516323.ogv",
+ "bug516323.ogv^headers^",
+ "bug520493.ogg",
+ "bug520493.ogg^headers^",
+ "bug520500.ogg",
+ "bug520500.ogg^headers^",
+ "bug520908.ogv",
+ "bug520908.ogv^headers^",
+ "bug523816.ogv",
+ "bug523816.ogv^headers^",
+ "bug533822.ogg",
+ "bug533822.ogg^headers^",
+ "bug556821.ogv",
+ "bug556821.ogv^headers^",
+ "bug557094.ogv",
+ "bug557094.ogv^headers^",
+ "bug604067.webm",
+ "bug604067.webm^headers^",
+ "bug1066943.webm",
+ "bug1066943.webm^headers^",
+ "bug1301226.wav",
+ "bug1301226.wav^headers^",
+ "bug1301226-odd.wav",
+ "bug1301226-odd.wav^headers^",
+ "bug1377278.webm",
+ "bug1377278.webm^headers^",
+ "bunny.webm",
+ "can_play_type_dash.js",
+ "can_play_type_ogg.js",
+ "can_play_type_wave.js",
+ "can_play_type_webm.js",
+ "cancellable_request.sjs",
+ "chain.ogg",
+ "chain.ogg^headers^",
+ "chain.ogv",
+ "chain.ogv^headers^",
+ "chain.opus",
+ "chain.opus^headers^",
+ "chained-audio-video.ogg",
+ "chained-audio-video.ogg^headers^",
+ "chained-video.ogv",
+ "chained-video.ogv^headers^",
+ "chromeHelper.js",
+ "cloneElementVisually_helpers.js",
+ "contentType.sjs",
+ "detodos.opus",
+ "detodos.opus^headers^",
+ "detodos.webm",
+ "detodos.webm^headers^",
+ "detodos-short.webm",
+ "detodos-short.webm^headers^",
+ "detodos-recorder-test.opus",
+ "detodos-recorder-test.opus^headers^",
+ "detodos-short.opus",
+ "detodos-short.opus^headers^",
+ "dirac.ogg",
+ "dirac.ogg^headers^",
+ "dynamic_resource.sjs",
+ "eme_standalone.js",
+ "eme.js",
+ "empty_size.mp3",
+ "file_access_controls.html",
+ "file_eme_createMediaKeys.html",
+ "flac-s24.flac",
+ "flac-s24.flac^headers^",
+ "flac-noheader-s16.flac",
+ "flac-noheader-s16.flac^headers^",
+ "flac-sample.mp4",
+ "flac-sample.mp4^headers^",
+ "flac-sample-cenc.mp4",
+ "flac-sample-cenc.mp4^headers^",
+ "fragment_noplay.js",
+ "fragment_play.js",
+ "gizmo.mp4",
+ "gizmo.mp4^headers^",
+ "gizmo-noaudio.mp4",
+ "gizmo-noaudio.mp4^headers^",
+ "gizmo-short.mp4",
+ "gizmo-short.mp4^headers^",
+ "gizmo.webm",
+ "gizmo.webm^headers^",
+ "gizmo-noaudio.webm",
+ "gizmo-noaudio.webm^headers^",
+ "gUM_support.js",
+ "gzipped_mp4.sjs",
+ "huge-id3.mp3",
+ "huge-id3.mp3^headers^",
+ "id3tags.mp3",
+ "id3tags.mp3^headers^",
+ "invalid-cmap-s0c0.opus",
+ "invalid-cmap-s0c0.opus^headers^",
+ "invalid-cmap-s0c2.opus",
+ "invalid-cmap-s0c2.opus^headers^",
+ "invalid-cmap-s1c2.opus",
+ "invalid-cmap-s1c2.opus^headers^",
+ "invalid-cmap-short.opus",
+ "invalid-cmap-short.opus^headers^",
+ "invalid-discard_on_multi_blocks.webm",
+ "invalid-discard_on_multi_blocks.webm^headers^",
+ "invalid-excess_discard.webm",
+ "invalid-excess_discard.webm^headers^",
+ "invalid-excess_neg_discard.webm",
+ "invalid-excess_neg_discard.webm^headers^",
+ "invalid-m0c0.opus",
+ "invalid-m0c0.opus^headers^",
+ "invalid-m0c3.opus",
+ "invalid-m0c3.opus^headers^",
+ "invalid-m1c0.opus",
+ "invalid-m1c0.opus^headers^",
+ "invalid-m1c9.opus",
+ "invalid-m1c9.opus^headers^",
+ "invalid-m2c0.opus",
+ "invalid-m2c0.opus^headers^",
+ "invalid-m2c1.opus",
+ "invalid-m2c1.opus^headers^",
+ "invalid-neg_discard.webm",
+ "invalid-neg_discard.webm^headers^",
+ "invalid-preskip.webm",
+ "invalid-preskip.webm^headers^",
+ "manifest.js",
+ "midflight-redirect.sjs",
+ "multiple-bos.ogg",
+ "multiple-bos.ogg^headers^",
+ "multiple-bos-more-header-fileds.ogg",
+ "multiple-bos-more-header-fileds.ogg^headers^",
+ "multi_id3v2.mp3",
+ "no-container-codec-delay.webm",
+ "no-cues.webm",
+ "no-cues.webm^headers^",
+ "notags.mp3",
+ "notags.mp3^headers^",
+ "opus-mapping2.mp4",
+ "opus-mapping2.mp4^headers^",
+ "opus-mapping2.webm",
+ "opus-mapping2.webm^headers^",
+ "opus-sample.mp4",
+ "opus-sample.mp4^headers^",
+ "opus-sample-cenc.mp4",
+ "opus-sample-cenc.mp4^headers^",
+ "owl-funnier-id3.mp3",
+ "owl-funnier-id3.mp3^headers^",
+ "owl-funny-id3.mp3",
+ "owl-funny-id3.mp3^headers^",
+ "owl.mp3",
+ "owl.mp3^headers^",
+ "owl-short.mp3",
+ "owl-short.mp3^headers^",
+ "pixel_aspect_ratio.mp4",
+ "play_promise.js",
+ "poster-test.jpg",
+ "r11025_msadpcm_c1.wav",
+ "r11025_msadpcm_c1.wav^headers^",
+ "r11025_s16_c1.wav",
+ "r11025_s16_c1.wav^headers^",
+ "r11025_s16_c1_trailing.wav",
+ "r11025_s16_c1_trailing.wav^headers^",
+ "r11025_s16_c1-short.wav",
+ "r11025_s16_c1-short.wav^headers^",
+ "r11025_u8_c1.wav",
+ "r11025_u8_c1.wav^headers^",
+ "r11025_u8_c1_trunc.wav",
+ "r11025_u8_c1_trunc.wav^headers^",
+ "r16000_u8_c1_list.wav",
+ "r16000_u8_c1_list.wav^headers^",
+ "reactivate_helper.html",
+ "red-46x48.mp4",
+ "red-46x48.mp4^headers^",
+ "red-48x46.mp4",
+ "red-48x46.mp4^headers^",
+ "redirect.sjs",
+ "referer.sjs",
+ "resolution-change.webm",
+ "resolution-change.webm^headers^",
+ "sample.3gp",
+ "sample.3g2",
+ "sample-encrypted-sgpdstbl-sbgptraf.mp4",
+ "sample-encrypted-sgpdstbl-sbgptraf.mp4^headers^",
+ "sample-fisbone-skeleton4.ogv",
+ "sample-fisbone-skeleton4.ogv^headers^",
+ "sample-fisbone-wrong-header.ogv",
+ "sample-fisbone-wrong-header.ogv^headers^",
+ "seek.ogv",
+ "seek.ogv^headers^",
+ "seek-short.ogv",
+ "seek-short.ogv^headers^",
+ "seek.webm",
+ "seek.webm^headers^",
+ "seek-short.webm",
+ "seek-short.webm^headers^",
+ "seek_support.js",
+ "seekLies.sjs",
+ "seek_with_sound.ogg^headers^",
+ "short-cenc.mp4",
+ "sine.webm",
+ "sine.webm^headers^",
+ "sintel-short-clearkey-subsample-encrypted-audio.webm",
+ "sintel-short-clearkey-subsample-encrypted-audio.webm^headers^",
+ "sintel-short-clearkey-subsample-encrypted-video.webm",
+ "sintel-short-clearkey-subsample-encrypted-video.webm^headers^",
+ "short.mp4",
+ "short.mp4.gz",
+ "short.mp4^headers^",
+ "short-aac-encrypted-audio.mp4",
+ "short-aac-encrypted-audio.mp4^headers^",
+ "short-audio-fragmented-cenc-without-pssh.mp4",
+ "short-audio-fragmented-cenc-without-pssh.mp4^headers^",
+ "short-video.ogv",
+ "short-video.ogv^headers^",
+ "short-vp9-encrypted-video.mp4",
+ "short-vp9-encrypted-video.mp4^headers^",
+ "small-shot-mp3.mp4",
+ "small-shot-mp3.mp4^headers^",
+ "small-shot.m4a",
+ "small-shot.mp3",
+ "small-shot.mp3^headers^",
+ "small-shot.ogg",
+ "small-shot.ogg^headers^",
+ "small-shot.flac",
+ "sound.ogg",
+ "sound.ogg^headers^",
+ "spacestorm-1000Hz-100ms.ogg",
+ "spacestorm-1000Hz-100ms.ogg^headers^",
+ "split.webm",
+ "split.webm^headers^",
+ "street.mp4",
+ "street.mp4^headers^",
+ "test-1-mono.opus",
+ "test-1-mono.opus^headers^",
+ "test-2-stereo.opus",
+ "test-2-stereo.opus^headers^",
+ "test-3-LCR.opus",
+ "test-3-LCR.opus^headers^",
+ "test-4-quad.opus",
+ "test-4-quad.opus^headers^",
+ "test-5-5.0.opus",
+ "test-5-5.0.opus^headers^",
+ "test-6-5.1.opus",
+ "test-6-5.1.opus^headers^",
+ "test-7-6.1.opus",
+ "test-7-6.1.opus^headers^",
+ "test-8-7.1.opus",
+ "test-8-7.1.opus^headers^",
+ "test-stereo-phase-inversion-180.opus",
+ "test-stereo-phase-inversion-180.opus^headers^",
+ "variable-channel.ogg",
+ "variable-channel.ogg^headers^",
+ "variable-channel.opus",
+ "variable-channel.opus^headers^",
+ "variable-preskip.opus",
+ "variable-preskip.opus^headers^",
+ "variable-samplerate.ogg",
+ "variable-samplerate.ogg^headers^",
+ "variable-samplerate.opus",
+ "variable-samplerate.opus^headers^",
+ "vbr-head.mp3",
+ "vbr-head.mp3^headers^",
+ "vbr.mp3",
+ "vbr.mp3^headers^",
+ "very-short.mp3",
+ "video-overhang.ogg",
+ "video-overhang.ogg^headers^",
+ "vp9-superframes.webm",
+ "vp9-superframes.webm^headers^",
+ "vp9.webm",
+ "vp9.webm^headers^",
+ "vp9-short.webm",
+ "vp9-short.webm^headers^",
+ "vp9cake.webm",
+ "vp9cake.webm^headers^",
+ "vp9cake-short.webm",
+ "vp9cake-short.webm^headers^",
+ "wave_metadata.wav",
+ "wave_metadata.wav^headers^",
+ "wave_metadata_bad_len.wav",
+ "wave_metadata_bad_len.wav^headers^",
+ "wave_metadata_bad_no_null.wav",
+ "wave_metadata_bad_no_null.wav^headers^",
+ "wave_metadata_bad_utf8.wav",
+ "wave_metadata_bad_utf8.wav^headers^",
+ "wave_metadata_unknown_tag.wav",
+ "wave_metadata_unknown_tag.wav^headers^",
+ "wave_metadata_utf8.wav",
+ "wave_metadata_utf8.wav^headers^",
+ "wavedata_alaw.wav",
+ "wavedata_alaw.wav^headers^",
+ "wavedata_float.wav",
+ "wavedata_float.wav^headers^",
+ "wavedata_s24.wav",
+ "wavedata_s24.wav^headers^",
+ "wavedata_s16.wav",
+ "wavedata_s16.wav^headers^",
+ "wavedata_u8.wav",
+ "wavedata_u8.wav^headers^",
+ "wavedata_ulaw.wav",
+ "wavedata_ulaw.wav^headers^",
+ "!/dom/canvas/test/captureStream_common.js",
+ "!/dom/html/test/reflect.js",
+ "!/dom/media/webrtc/tests/mochitests/head.js",
+ "hls/bipbop_16x9_single.m3u8",
+ "hls/bipbop_4x3_single.m3u8",
+ "hls/bipbop_4x3_variant.m3u8",
+ "hls/400x300_prog_index.m3u8",
+ "hls/400x300_prog_index_5s.m3u8",
+ "hls/416x243_prog_index_5s.m3u8",
+ "hls/640x480_prog_index.m3u8",
+ "hls/960x720_prog_index.m3u8",
+ "hls/400x300_seg0.ts",
+ "hls/400x300_seg0_5s.ts",
+ "hls/400x300_seg1.ts",
+ "hls/416x243_seg0_5s.ts",
+ "hls/640x480_seg0.ts",
+ "hls/640x480_seg1.ts",
+ "hls/960x720_seg0.ts",
+ "hls/960x720_seg1.ts",
+ "sync.webm",
+]
+
+["test_seek-1.html"]
+skip-if = ["os == 'android'"] # bug 1322806, android(bug 1232305)
+
+["test_seek-10.html"]
+skip-if = ["os == 'android'"] # android(bug 1232305)
+
+["test_seek-11.html"]
+skip-if = ["os == 'android'"] # bug 1323133, android(bug 1232305)
+
+["test_seek-12.html"]
+skip-if = ["os == 'android'"] # bug 1321081, android(bug 1232305)
+
+["test_seek-13.html"]
+skip-if = ["os == 'android'"] # bug 1299174, android(bug 1232305)
+
+["test_seek-14.html"]
+skip-if = ["os == 'android'"] # android(bug 1232305)
+
+["test_seek-2.html"]
+skip-if = ["os == 'android'"] # bug 1309778, android(bug 1232305)
+
+["test_seek-3.html"]
+skip-if = ["os == 'android'"] # bug 1321082, android(bug 1232305)
+
+["test_seek-4.html"]
+skip-if = ["os == 'android'"] # android(bug 1232305)
+
+["test_seek-5.html"]
+skip-if = ["os == 'android'"] # android(bug 1232305)
+
+["test_seek-6.html"]
+skip-if = ["os == 'android'"] # bug 1336629, bug 1324482, android(bug 1232305)
+
+["test_seek-7.html"]
+skip-if = ["os == 'android'"] # android(bug 1232305)
+
+["test_seek-8.html"]
+skip-if = ["os == 'android'"] # bug 1310584, android(bug 1232305)
+
+["test_seek-9.html"]
+skip-if = ["os == 'android'"] # bug 1332019, android(bug 1232305)
+
+["test_seekLies.html"]
+
+["test_seekToNextFrame.html"]
+skip-if = ["os == 'android'"] # bug 1329391, android(bug 1232305)
+tags = "seektonextframe"
+
+["test_seek_duration.html"]
+
+["test_seek_negative.html"]
+
+["test_seek_nosrc.html"]
+
+["test_seek_out_of_range.html"]
+skip-if = ["os == 'android'"] # bug 1299382, android(bug 1232305)
+
+["test_seek_promise_bug1344357.html"]
+skip-if = ["os == 'android'"] # bug 1299382, android(bug 1232305)
+
+["test_seekable1.html"]
+skip-if = ["os == 'android'"] # android(bug 1232305)
diff --git a/dom/media/test/mochitest_stream.toml b/dom/media/test/mochitest_stream.toml
new file mode 100644
index 0000000000..043f8471d4
--- /dev/null
+++ b/dom/media/test/mochitest_stream.toml
@@ -0,0 +1,792 @@
+# Media tests should be backend independent, i.e., not conditioned on ogg,
+# wave etc. (The only exception is the can_play_type tests, which
+# necessarily depend on the backend(s) configured.) As far as possible, each
+# test should work with any resource type. This makes it easy to add new
+# backends and reduces the amount of test duplication.
+
+# For each supported backend, resources that can be played by that backend
+# should be added to the lists in manifest.js. Media tests that aren't
+# testing for a bug in handling a specific resource type should pick one of
+# the lists in manifest.js and run the test for each resource in the list
+# that is supported in the current build (the canPlayType API is useful for
+# this).
+
+# To test whether a valid resource can simply be played through correctly,
+# and optionally that its metadata is read correctly, just add it to
+# gPlayTests in manifest.js. To test whether an invalid resource correctly
+# throws an error (and does not cause a crash or hang), just add it to
+# gErrorTests in manifest.js.
+
+# To test for a specific bug in handling a specific resource type, make the
+# test first check canPlayType for the type, and if it's not supported, just
+# do ok(true, "Type not supported") and stop the test.
+
+[DEFAULT]
+subsuite = "media"
+tags = "mtg capturestream"
+support-files = [
+ "16bit_wave_extrametadata.wav",
+ "16bit_wave_extrametadata.wav^headers^",
+ "320x240.ogv",
+ "320x240.ogv^headers^",
+ "448636.ogv",
+ "448636.ogv^headers^",
+ "A4.ogv",
+ "A4.ogv^headers^",
+ "VID_0001.ogg",
+ "VID_0001.ogg^headers^",
+ "allowed.sjs",
+ "ambisonics.mp4",
+ "ambisonics.mp4^headers^",
+ "audio-gaps.ogg",
+ "audio-gaps.ogg^headers^",
+ "audio-gaps-short.ogg",
+ "audio-gaps-short.ogg^headers^",
+ "audio-overhang.ogg",
+ "audio-overhang.ogg^headers^",
+ "audio.wav",
+ "audio.wav^headers^",
+ "av1.mp4",
+ "av1.mp4^headers^",
+ "background_video.js",
+ "badtags.ogg",
+ "badtags.ogg^headers^",
+ "bear-640x360-v_frag-cenc-key_rotation.mp4",
+ "bear-640x360-a_frag-cenc-key_rotation.mp4",
+ "beta-phrasebook.ogg",
+ "beta-phrasebook.ogg^headers^",
+ "big.wav",
+ "big.wav^headers^",
+ "big-buck-bunny-cenc-avc3-1.m4s",
+ "big-buck-bunny-cenc-avc3-1.m4s^headers^",
+ "big-buck-bunny-cenc-avc3-init.mp4",
+ "big-buck-bunny-cenc-avc3-init.mp4^headers^",
+ "big-short.wav",
+ "big-short.wav^headers^",
+ "bipbop.mp4",
+ "bipbop-cenc-audio1.m4s",
+ "bipbop-cenc-audio1.m4s^headers^",
+ "bipbop-cenc-audio2.m4s",
+ "bipbop-cenc-audio2.m4s^headers^",
+ "bipbop-cenc-audio3.m4s",
+ "bipbop-cenc-audio3.m4s^headers^",
+ "bipbop-cenc-audioinit.mp4",
+ "bipbop-cenc-audioinit.mp4^headers^",
+ "bipbop-cenc-video1.m4s",
+ "bipbop-cenc-video1.m4s^headers^",
+ "bipbop-cenc-video2.m4s",
+ "bipbop-cenc-video2.m4s^headers^",
+ "bipbop-cenc-videoinit.mp4",
+ "bipbop-cenc-videoinit.mp4^headers^",
+ "bipbop-cenc-video-10s.mp4",
+ "bipbop-cenc-video-10s.mp4^headers^",
+ "bipbop-clearkey-keyrotation-clear-lead-audio.mp4",
+ "bipbop-clearkey-keyrotation-clear-lead-audio.mp4^headers^",
+ "bipbop-clearkey-keyrotation-clear-lead-video.mp4",
+ "bipbop-clearkey-keyrotation-clear-lead-video.mp4^headers^",
+ "bipbop_225w_175kbps.mp4",
+ "bipbop_225w_175kbps.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-1.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-2.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-3.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-4.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key1-init.mp4",
+ "bipbop_225w_175kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-1.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-2.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-3.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-4.m4s",
+ "bipbop_225w_175kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-audio-key2-init.mp4",
+ "bipbop_225w_175kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-video-key1-1.m4s",
+ "bipbop_225w_175kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-video-key1-init.mp4",
+ "bipbop_225w_175kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_225w_175kbps-cenc-video-key2-1.m4s",
+ "bipbop_225w_175kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_225w_175kbps-cenc-video-key2-init.mp4",
+ "bipbop_225w_175kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-1.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-2.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-3.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-4.m4s",
+ "bipbop_300_215kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-1.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-2.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-3.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-4.m4s",
+ "bipbop_300_215kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_300_215kbps-cenc-audio-key2-init.mp4",
+ "bipbop_300_215kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-1.m4s",
+ "bipbop_300_215kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-2.m4s",
+ "bipbop_300_215kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key1-init.mp4",
+ "bipbop_300_215kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-1.m4s",
+ "bipbop_300_215kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-2.m4s",
+ "bipbop_300_215kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_300_215kbps-cenc-video-key2-init.mp4",
+ "bipbop_300_215kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-1.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-2.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-3.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-4.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key1-init.mp4",
+ "bipbop_300wp_227kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-1.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-2.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-3.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-4.m4s",
+ "bipbop_300wp_227kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-audio-key2-init.mp4",
+ "bipbop_300wp_227kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-1.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-2.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key1-init.mp4",
+ "bipbop_300wp_227kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-1.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-2.m4s",
+ "bipbop_300wp_227kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_300wp_227kbps-cenc-video-key2-init.mp4",
+ "bipbop_300wp_227kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-1.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-2.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-3.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-4.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key1-init.mp4",
+ "bipbop_360w_253kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-1.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-2.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-3.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-4.m4s",
+ "bipbop_360w_253kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-audio-key2-init.mp4",
+ "bipbop_360w_253kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-video-key1-1.m4s",
+ "bipbop_360w_253kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-video-key1-init.mp4",
+ "bipbop_360w_253kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_360w_253kbps-cenc-video-key2-1.m4s",
+ "bipbop_360w_253kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_360w_253kbps-cenc-video-key2-init.mp4",
+ "bipbop_360w_253kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_360w_253kbps-clearkey-audio.webm",
+ "bipbop_360w_253kbps-clearkey-audio.webm^headers^",
+ "bipbop_360w_253kbps-clearkey-video-vp8.webm",
+ "bipbop_360w_253kbps-clearkey-video-vp8.webm^headers^",
+ "bipbop_360w_253kbps-clearkey-video-vp9.webm",
+ "bipbop_360w_253kbps-clearkey-video-vp9.webm^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480_624kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480_624kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480_624kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480_624kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-1.m4s",
+ "bipbop_480_624kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-2.m4s",
+ "bipbop_480_624kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key1-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-1.m4s",
+ "bipbop_480_624kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-2.m4s",
+ "bipbop_480_624kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480_624kbps-cenc-video-key2-init.mp4",
+ "bipbop_480_624kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480_959kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480_959kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480_959kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480_959kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480_959kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-1.m4s",
+ "bipbop_480_959kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-2.m4s",
+ "bipbop_480_959kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key1-init.mp4",
+ "bipbop_480_959kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-1.m4s",
+ "bipbop_480_959kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-2.m4s",
+ "bipbop_480_959kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480_959kbps-cenc-video-key2-init.mp4",
+ "bipbop_480_959kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480wp_663kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480wp_663kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480wp_663kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-1.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-2.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key1-init.mp4",
+ "bipbop_480wp_663kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-1.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-2.m4s",
+ "bipbop_480wp_663kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480wp_663kbps-cenc-video-key2-init.mp4",
+ "bipbop_480wp_663kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-3.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-4.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-audio-key1-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-3.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-4.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-audio-key2-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key1-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key1-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key1-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-video-key1-init.mp4^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-1.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key2-1.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-2.m4s",
+ "bipbop_480wp_1001kbps-cenc-video-key2-2.m4s^headers^",
+ "bipbop_480wp_1001kbps-cenc-video-key2-init.mp4",
+ "bipbop_480wp_1001kbps-cenc-video-key2-init.mp4^headers^",
+ "bipbop_audio_aac_8k.mp4",
+ "bipbop_audio_aac_8k.mp4^headers^",
+ "bipbop_audio_aac_22.05k.mp4",
+ "bipbop_audio_aac_22.05k.mp4^headers^",
+ "bipbop_audio_aac_44.1k.mp4",
+ "bipbop_audio_aac_44.1k.mp4^headers^",
+ "bipbop_audio_aac_48k.mp4",
+ "bipbop_audio_aac_48k.mp4^headers^",
+ "bipbop_audio_aac_88.2k.mp4",
+ "bipbop_audio_aac_88.2k.mp4^headers^",
+ "bipbop_audio_aac_96k.mp4",
+ "bipbop_audio_aac_96k.mp4^headers^",
+ "bipbop_cbcs_1_9_audio_1.m4s",
+ "bipbop_cbcs_1_9_audio_1.m4s^headers^",
+ "bipbop_cbcs_1_9_audio_init.mp4",
+ "bipbop_cbcs_1_9_audio_init.mp4^headers^",
+ "bipbop_cbcs_1_9_video_1.m4s",
+ "bipbop_cbcs_1_9_video_1.m4s^headers^",
+ "bipbop_cbcs_1_9_video_init.mp4",
+ "bipbop_cbcs_1_9_video_init.mp4^headers^",
+ "bipbop_cbcs_5_5_audio_1.m4s",
+ "bipbop_cbcs_5_5_audio_1.m4s^headers^",
+ "bipbop_cbcs_5_5_audio_init.mp4",
+ "bipbop_cbcs_5_5_audio_init.mp4^headers^",
+ "bipbop_cbcs_5_5_video_1.m4s",
+ "bipbop_cbcs_5_5_video_1.m4s^headers^",
+ "bipbop_cbcs_5_5_video_init.mp4",
+ "bipbop_cbcs_5_5_video_init.mp4^headers^",
+ "bipbop_cbcs_7_7_audio_1.m4s",
+ "bipbop_cbcs_7_7_audio_1.m4s^headers^",
+ "bipbop_cbcs_7_7_audio_init.mp4",
+ "bipbop_cbcs_7_7_audio_init.mp4^headers^",
+ "bipbop_cbcs_7_7_video_1.m4s",
+ "bipbop_cbcs_7_7_video_1.m4s^headers^",
+ "bipbop_cbcs_7_7_video_init.mp4",
+ "bipbop_cbcs_7_7_video_init.mp4^headers^",
+ "bipbop_cbcs_9_8_audio_1.m4s",
+ "bipbop_cbcs_9_8_audio_1.m4s^headers^",
+ "bipbop_cbcs_9_8_audio_init.mp4",
+ "bipbop_cbcs_9_8_audio_init.mp4^headers^",
+ "bipbop_cbcs_9_8_video_1.m4s",
+ "bipbop_cbcs_9_8_video_1.m4s^headers^",
+ "bipbop_cbcs_9_8_video_init.mp4",
+ "bipbop_cbcs_9_8_video_init.mp4^headers^",
+ "bipbop_cbcs_10_0_audio_1.m4s",
+ "bipbop_cbcs_10_0_audio_1.m4s^headers^",
+ "bipbop_cbcs_10_0_audio_init.mp4",
+ "bipbop_cbcs_10_0_audio_init.mp4^headers^",
+ "bipbop_cbcs_10_0_video_1.m4s",
+ "bipbop_cbcs_10_0_video_1.m4s^headers^",
+ "bipbop_cbcs_10_0_video_init.mp4",
+ "bipbop_cbcs_10_0_video_init.mp4^headers^",
+ "bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm",
+ "bipbop_short_pixel_metadata_bigger_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm",
+ "bipbop_short_pixel_metadata_narrower_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm^headers^",
+ "bipbop_short_pixel_metadata_smaller_than_in_stream_vp8.webm",
+ "bipbop_short_vp8.webm",
+ "bipbop_short_vp8.webm^headers^",
+ "bipbop-lateaudio.mp4",
+ "bipbop-lateaudio.mp4^headers^",
+ "black100x100-aspect3to2.ogv",
+ "black100x100-aspect3to2.ogv^headers^",
+ "bogus.duh",
+ "bogus.ogv",
+ "bogus.ogv^headers^",
+ "bogus.wav",
+ "bogus.wav^headers^",
+ "bug461281.ogg",
+ "bug461281.ogg^headers^",
+ "bug482461-theora.ogv",
+ "bug482461-theora.ogv^headers^",
+ "bug482461.ogv",
+ "bug482461.ogv^headers^",
+ "bug495129.ogv",
+ "bug495129.ogv^headers^",
+ "bug495794.ogg",
+ "bug495794.ogg^headers^",
+ "bug498380.ogv",
+ "bug498380.ogv^headers^",
+ "bug498855-1.ogv",
+ "bug498855-1.ogv^headers^",
+ "bug498855-2.ogv",
+ "bug498855-2.ogv^headers^",
+ "bug498855-3.ogv",
+ "bug498855-3.ogv^headers^",
+ "bug499519.ogv",
+ "bug499519.ogv^headers^",
+ "bug500311.ogv",
+ "bug500311.ogv^headers^",
+ "bug501279.ogg",
+ "bug501279.ogg^headers^",
+ "bug504613.ogv",
+ "bug504613.ogv^headers^",
+ "bug504644.ogv",
+ "bug504644.ogv^headers^",
+ "bug504843.ogv",
+ "bug504843.ogv^headers^",
+ "bug506094.ogv",
+ "bug506094.ogv^headers^",
+ "bug516323.indexed.ogv",
+ "bug516323.indexed.ogv^headers^",
+ "bug516323.ogv",
+ "bug516323.ogv^headers^",
+ "bug520493.ogg",
+ "bug520493.ogg^headers^",
+ "bug520500.ogg",
+ "bug520500.ogg^headers^",
+ "bug520908.ogv",
+ "bug520908.ogv^headers^",
+ "bug523816.ogv",
+ "bug523816.ogv^headers^",
+ "bug533822.ogg",
+ "bug533822.ogg^headers^",
+ "bug556821.ogv",
+ "bug556821.ogv^headers^",
+ "bug557094.ogv",
+ "bug557094.ogv^headers^",
+ "bug604067.webm",
+ "bug604067.webm^headers^",
+ "bug1066943.webm",
+ "bug1066943.webm^headers^",
+ "bug1301226.wav",
+ "bug1301226.wav^headers^",
+ "bug1301226-odd.wav",
+ "bug1301226-odd.wav^headers^",
+ "bug1377278.webm",
+ "bug1377278.webm^headers^",
+ "bunny.webm",
+ "can_play_type_dash.js",
+ "can_play_type_ogg.js",
+ "can_play_type_wave.js",
+ "can_play_type_webm.js",
+ "cancellable_request.sjs",
+ "chain.ogg",
+ "chain.ogg^headers^",
+ "chain.ogv",
+ "chain.ogv^headers^",
+ "chain.opus",
+ "chain.opus^headers^",
+ "chained-audio-video.ogg",
+ "chained-audio-video.ogg^headers^",
+ "chained-video.ogv",
+ "chained-video.ogv^headers^",
+ "chromeHelper.js",
+ "cloneElementVisually_helpers.js",
+ "contentType.sjs",
+ "detodos.opus",
+ "detodos.opus^headers^",
+ "detodos.webm",
+ "detodos.webm^headers^",
+ "detodos-short.webm",
+ "detodos-short.webm^headers^",
+ "detodos-recorder-test.opus",
+ "detodos-recorder-test.opus^headers^",
+ "detodos-short.opus",
+ "detodos-short.opus^headers^",
+ "dirac.ogg",
+ "dirac.ogg^headers^",
+ "dynamic_resource.sjs",
+ "eme_standalone.js",
+ "eme.js",
+ "empty_size.mp3",
+ "file_access_controls.html",
+ "file_eme_createMediaKeys.html",
+ "flac-s24.flac",
+ "flac-s24.flac^headers^",
+ "flac-noheader-s16.flac",
+ "flac-noheader-s16.flac^headers^",
+ "flac-sample.mp4",
+ "flac-sample.mp4^headers^",
+ "flac-sample-cenc.mp4",
+ "flac-sample-cenc.mp4^headers^",
+ "fragment_noplay.js",
+ "fragment_play.js",
+ "gizmo.mp4",
+ "gizmo.mp4^headers^",
+ "gizmo-noaudio.mp4",
+ "gizmo-noaudio.mp4^headers^",
+ "gizmo-short.mp4",
+ "gizmo-short.mp4^headers^",
+ "gizmo.webm",
+ "gizmo.webm^headers^",
+ "gizmo-noaudio.webm",
+ "gizmo-noaudio.webm^headers^",
+ "gUM_support.js",
+ "gzipped_mp4.sjs",
+ "huge-id3.mp3",
+ "huge-id3.mp3^headers^",
+ "id3tags.mp3",
+ "id3tags.mp3^headers^",
+ "invalid-cmap-s0c0.opus",
+ "invalid-cmap-s0c0.opus^headers^",
+ "invalid-cmap-s0c2.opus",
+ "invalid-cmap-s0c2.opus^headers^",
+ "invalid-cmap-s1c2.opus",
+ "invalid-cmap-s1c2.opus^headers^",
+ "invalid-cmap-short.opus",
+ "invalid-cmap-short.opus^headers^",
+ "invalid-discard_on_multi_blocks.webm",
+ "invalid-discard_on_multi_blocks.webm^headers^",
+ "invalid-excess_discard.webm",
+ "invalid-excess_discard.webm^headers^",
+ "invalid-excess_neg_discard.webm",
+ "invalid-excess_neg_discard.webm^headers^",
+ "invalid-m0c0.opus",
+ "invalid-m0c0.opus^headers^",
+ "invalid-m0c3.opus",
+ "invalid-m0c3.opus^headers^",
+ "invalid-m1c0.opus",
+ "invalid-m1c0.opus^headers^",
+ "invalid-m1c9.opus",
+ "invalid-m1c9.opus^headers^",
+ "invalid-m2c0.opus",
+ "invalid-m2c0.opus^headers^",
+ "invalid-m2c1.opus",
+ "invalid-m2c1.opus^headers^",
+ "invalid-neg_discard.webm",
+ "invalid-neg_discard.webm^headers^",
+ "invalid-preskip.webm",
+ "invalid-preskip.webm^headers^",
+ "manifest.js",
+ "midflight-redirect.sjs",
+ "multiple-bos.ogg",
+ "multiple-bos.ogg^headers^",
+ "multiple-bos-more-header-fileds.ogg",
+ "multiple-bos-more-header-fileds.ogg^headers^",
+ "multi_id3v2.mp3",
+ "no-container-codec-delay.webm",
+ "no-cues.webm",
+ "no-cues.webm^headers^",
+ "notags.mp3",
+ "notags.mp3^headers^",
+ "opus-mapping2.mp4",
+ "opus-mapping2.mp4^headers^",
+ "opus-mapping2.webm",
+ "opus-mapping2.webm^headers^",
+ "opus-sample.mp4",
+ "opus-sample.mp4^headers^",
+ "opus-sample-cenc.mp4",
+ "opus-sample-cenc.mp4^headers^",
+ "owl-funnier-id3.mp3",
+ "owl-funnier-id3.mp3^headers^",
+ "owl-funny-id3.mp3",
+ "owl-funny-id3.mp3^headers^",
+ "owl.mp3",
+ "owl.mp3^headers^",
+ "owl-short.mp3",
+ "owl-short.mp3^headers^",
+ "pixel_aspect_ratio.mp4",
+ "play_promise.js",
+ "poster-test.jpg",
+ "r11025_msadpcm_c1.wav",
+ "r11025_msadpcm_c1.wav^headers^",
+ "r11025_s16_c1.wav",
+ "r11025_s16_c1.wav^headers^",
+ "r11025_s16_c1_trailing.wav",
+ "r11025_s16_c1_trailing.wav^headers^",
+ "r11025_s16_c1-short.wav",
+ "r11025_s16_c1-short.wav^headers^",
+ "r11025_u8_c1.wav",
+ "r11025_u8_c1.wav^headers^",
+ "r11025_u8_c1_trunc.wav",
+ "r11025_u8_c1_trunc.wav^headers^",
+ "r16000_u8_c1_list.wav",
+ "r16000_u8_c1_list.wav^headers^",
+ "reactivate_helper.html",
+ "red-46x48.mp4",
+ "red-46x48.mp4^headers^",
+ "red-48x46.mp4",
+ "red-48x46.mp4^headers^",
+ "redirect.sjs",
+ "referer.sjs",
+ "resolution-change.webm",
+ "resolution-change.webm^headers^",
+ "sample.3gp",
+ "sample.3g2",
+ "sample-encrypted-sgpdstbl-sbgptraf.mp4",
+ "sample-encrypted-sgpdstbl-sbgptraf.mp4^headers^",
+ "sample-fisbone-skeleton4.ogv",
+ "sample-fisbone-skeleton4.ogv^headers^",
+ "sample-fisbone-wrong-header.ogv",
+ "sample-fisbone-wrong-header.ogv^headers^",
+ "seek.ogv",
+ "seek.ogv^headers^",
+ "seek-short.ogv",
+ "seek-short.ogv^headers^",
+ "seek.webm",
+ "seek.webm^headers^",
+ "seek-short.webm",
+ "seek-short.webm^headers^",
+ "seek_support.js",
+ "seekLies.sjs",
+ "seek_with_sound.ogg^headers^",
+ "short-cenc.mp4",
+ "sine.webm",
+ "sine.webm^headers^",
+ "sintel-short-clearkey-subsample-encrypted-audio.webm",
+ "sintel-short-clearkey-subsample-encrypted-audio.webm^headers^",
+ "sintel-short-clearkey-subsample-encrypted-video.webm",
+ "sintel-short-clearkey-subsample-encrypted-video.webm^headers^",
+ "short.mp4",
+ "short.mp4.gz",
+ "short.mp4^headers^",
+ "short-aac-encrypted-audio.mp4",
+ "short-aac-encrypted-audio.mp4^headers^",
+ "short-audio-fragmented-cenc-without-pssh.mp4",
+ "short-audio-fragmented-cenc-without-pssh.mp4^headers^",
+ "short-video.ogv",
+ "short-video.ogv^headers^",
+ "short-vp9-encrypted-video.mp4",
+ "short-vp9-encrypted-video.mp4^headers^",
+ "small-shot-mp3.mp4",
+ "small-shot-mp3.mp4^headers^",
+ "small-shot.m4a",
+ "small-shot.mp3",
+ "small-shot.mp3^headers^",
+ "small-shot.ogg",
+ "small-shot.ogg^headers^",
+ "small-shot.flac",
+ "sound.ogg",
+ "sound.ogg^headers^",
+ "spacestorm-1000Hz-100ms.ogg",
+ "spacestorm-1000Hz-100ms.ogg^headers^",
+ "split.webm",
+ "split.webm^headers^",
+ "street.mp4",
+ "street.mp4^headers^",
+ "test-1-mono.opus",
+ "test-1-mono.opus^headers^",
+ "test-2-stereo.opus",
+ "test-2-stereo.opus^headers^",
+ "test-3-LCR.opus",
+ "test-3-LCR.opus^headers^",
+ "test-4-quad.opus",
+ "test-4-quad.opus^headers^",
+ "test-5-5.0.opus",
+ "test-5-5.0.opus^headers^",
+ "test-6-5.1.opus",
+ "test-6-5.1.opus^headers^",
+ "test-7-6.1.opus",
+ "test-7-6.1.opus^headers^",
+ "test-8-7.1.opus",
+ "test-8-7.1.opus^headers^",
+ "test-stereo-phase-inversion-180.opus",
+ "test-stereo-phase-inversion-180.opus^headers^",
+ "variable-channel.ogg",
+ "variable-channel.ogg^headers^",
+ "variable-channel.opus",
+ "variable-channel.opus^headers^",
+ "variable-preskip.opus",
+ "variable-preskip.opus^headers^",
+ "variable-samplerate.ogg",
+ "variable-samplerate.ogg^headers^",
+ "variable-samplerate.opus",
+ "variable-samplerate.opus^headers^",
+ "vbr-head.mp3",
+ "vbr-head.mp3^headers^",
+ "vbr.mp3",
+ "vbr.mp3^headers^",
+ "very-short.mp3",
+ "video-overhang.ogg",
+ "video-overhang.ogg^headers^",
+ "vp9-superframes.webm",
+ "vp9-superframes.webm^headers^",
+ "vp9.webm",
+ "vp9.webm^headers^",
+ "vp9-short.webm",
+ "vp9-short.webm^headers^",
+ "vp9cake.webm",
+ "vp9cake.webm^headers^",
+ "vp9cake-short.webm",
+ "vp9cake-short.webm^headers^",
+ "wave_metadata.wav",
+ "wave_metadata.wav^headers^",
+ "wave_metadata_bad_len.wav",
+ "wave_metadata_bad_len.wav^headers^",
+ "wave_metadata_bad_no_null.wav",
+ "wave_metadata_bad_no_null.wav^headers^",
+ "wave_metadata_bad_utf8.wav",
+ "wave_metadata_bad_utf8.wav^headers^",
+ "wave_metadata_unknown_tag.wav",
+ "wave_metadata_unknown_tag.wav^headers^",
+ "wave_metadata_utf8.wav",
+ "wave_metadata_utf8.wav^headers^",
+ "wavedata_alaw.wav",
+ "wavedata_alaw.wav^headers^",
+ "wavedata_float.wav",
+ "wavedata_float.wav^headers^",
+ "wavedata_s24.wav",
+ "wavedata_s24.wav^headers^",
+ "wavedata_s16.wav",
+ "wavedata_s16.wav^headers^",
+ "wavedata_u8.wav",
+ "wavedata_u8.wav^headers^",
+ "wavedata_ulaw.wav",
+ "wavedata_ulaw.wav^headers^",
+ "!/dom/canvas/test/captureStream_common.js",
+ "!/dom/html/test/reflect.js",
+ "!/dom/media/webrtc/tests/mochitests/head.js",
+ "hls/bipbop_16x9_single.m3u8",
+ "hls/bipbop_4x3_single.m3u8",
+ "hls/bipbop_4x3_variant.m3u8",
+ "hls/400x300_prog_index.m3u8",
+ "hls/400x300_prog_index_5s.m3u8",
+ "hls/416x243_prog_index_5s.m3u8",
+ "hls/640x480_prog_index.m3u8",
+ "hls/960x720_prog_index.m3u8",
+ "hls/400x300_seg0.ts",
+ "hls/400x300_seg0_5s.ts",
+ "hls/400x300_seg1.ts",
+ "hls/416x243_seg0_5s.ts",
+ "hls/640x480_seg0.ts",
+ "hls/640x480_seg1.ts",
+ "hls/960x720_seg0.ts",
+ "hls/960x720_seg1.ts",
+ "sync.webm",
+]
+
+["test_streams_capture_origin.html"]
+
+["test_streams_element_capture.html"]
+skip-if = ["os == 'android'"] # bug 1372457, bug 1526207 for drawImage
+
+["test_streams_element_capture_mediatrack.html"]
+
+["test_streams_element_capture_playback.html"]
+
+["test_streams_element_capture_reset.html"]
+
+["test_streams_element_capture_twice.html"]
+
+["test_streams_firstframe.html"]
+
+["test_streams_gc.html"]
+
+["test_streams_individual_pause.html"]
+skip-if = ["os == 'mac' && debug"] # Bug 1756880 - temp due to low frequency shutdown hang
+scheme = "https"
+tags = "mtg"
+
+["test_streams_srcObject.html"]
+skip-if = [
+ "os == 'android'", # bug 1300443, android(bug 1232305)
+ "os == 'mac' && debug", # Bug 1756880 - temp due to high frequency shutdown hang
+]
+
+["test_streams_tracks.html"]
+skip-if = ["os == 'android'"] # android(bug 1232305)
diff --git a/dom/media/test/multi_id3v2.mp3 b/dom/media/test/multi_id3v2.mp3
new file mode 100644
index 0000000000..253f19a9b6
--- /dev/null
+++ b/dom/media/test/multi_id3v2.mp3
Binary files differ
diff --git a/dom/media/test/multiple-bos-more-header-fileds.ogg b/dom/media/test/multiple-bos-more-header-fileds.ogg
new file mode 100644
index 0000000000..c9721cb98e
--- /dev/null
+++ b/dom/media/test/multiple-bos-more-header-fileds.ogg
Binary files differ
diff --git a/dom/media/test/multiple-bos-more-header-fileds.ogg^headers^ b/dom/media/test/multiple-bos-more-header-fileds.ogg^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/multiple-bos-more-header-fileds.ogg^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/multiple-bos.ogg b/dom/media/test/multiple-bos.ogg
new file mode 100644
index 0000000000..193200868e
--- /dev/null
+++ b/dom/media/test/multiple-bos.ogg
Binary files differ
diff --git a/dom/media/test/multiple-bos.ogg^headers^ b/dom/media/test/multiple-bos.ogg^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/multiple-bos.ogg^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/no-container-codec-delay.webm b/dom/media/test/no-container-codec-delay.webm
new file mode 100644
index 0000000000..9c1ef8e6d7
--- /dev/null
+++ b/dom/media/test/no-container-codec-delay.webm
Binary files differ
diff --git a/dom/media/test/no-cues.webm b/dom/media/test/no-cues.webm
new file mode 100644
index 0000000000..8ed761099e
--- /dev/null
+++ b/dom/media/test/no-cues.webm
Binary files differ
diff --git a/dom/media/test/no-cues.webm^headers^ b/dom/media/test/no-cues.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/no-cues.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/notags.mp3 b/dom/media/test/notags.mp3
new file mode 100644
index 0000000000..7f298131aa
--- /dev/null
+++ b/dom/media/test/notags.mp3
Binary files differ
diff --git a/dom/media/test/notags.mp3^headers^ b/dom/media/test/notags.mp3^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/notags.mp3^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/opus-mapping2.mp4 b/dom/media/test/opus-mapping2.mp4
new file mode 100644
index 0000000000..72401a9c0b
--- /dev/null
+++ b/dom/media/test/opus-mapping2.mp4
Binary files differ
diff --git a/dom/media/test/opus-mapping2.mp4^headers^ b/dom/media/test/opus-mapping2.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/opus-mapping2.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/opus-mapping2.webm b/dom/media/test/opus-mapping2.webm
new file mode 100644
index 0000000000..4379f2534a
--- /dev/null
+++ b/dom/media/test/opus-mapping2.webm
Binary files differ
diff --git a/dom/media/test/opus-mapping2.webm^headers^ b/dom/media/test/opus-mapping2.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/opus-mapping2.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/opus-sample-cenc.mp4 b/dom/media/test/opus-sample-cenc.mp4
new file mode 100644
index 0000000000..22bb787540
--- /dev/null
+++ b/dom/media/test/opus-sample-cenc.mp4
Binary files differ
diff --git a/dom/media/test/opus-sample-cenc.mp4^headers^ b/dom/media/test/opus-sample-cenc.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/opus-sample-cenc.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/opus-sample.mp4 b/dom/media/test/opus-sample.mp4
new file mode 100644
index 0000000000..80329ce14b
--- /dev/null
+++ b/dom/media/test/opus-sample.mp4
Binary files differ
diff --git a/dom/media/test/opus-sample.mp4^headers^ b/dom/media/test/opus-sample.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/opus-sample.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/owl-funnier-id3.mp3 b/dom/media/test/owl-funnier-id3.mp3
new file mode 100644
index 0000000000..05ec507530
--- /dev/null
+++ b/dom/media/test/owl-funnier-id3.mp3
Binary files differ
diff --git a/dom/media/test/owl-funnier-id3.mp3^headers^ b/dom/media/test/owl-funnier-id3.mp3^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/owl-funnier-id3.mp3^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/owl-funny-id3.mp3 b/dom/media/test/owl-funny-id3.mp3
new file mode 100644
index 0000000000..6533755a32
--- /dev/null
+++ b/dom/media/test/owl-funny-id3.mp3
Binary files differ
diff --git a/dom/media/test/owl-funny-id3.mp3^headers^ b/dom/media/test/owl-funny-id3.mp3^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/owl-funny-id3.mp3^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/owl-short.mp3 b/dom/media/test/owl-short.mp3
new file mode 100644
index 0000000000..9b31531f22
--- /dev/null
+++ b/dom/media/test/owl-short.mp3
Binary files differ
diff --git a/dom/media/test/owl-short.mp3^headers^ b/dom/media/test/owl-short.mp3^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/owl-short.mp3^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/owl.mp3 b/dom/media/test/owl.mp3
new file mode 100644
index 0000000000..9fafa32f93
--- /dev/null
+++ b/dom/media/test/owl.mp3
Binary files differ
diff --git a/dom/media/test/owl.mp3^headers^ b/dom/media/test/owl.mp3^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/owl.mp3^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/padding-spanning-multiple-packets.mp3 b/dom/media/test/padding-spanning-multiple-packets.mp3
new file mode 100644
index 0000000000..d7ab9dc3f5
--- /dev/null
+++ b/dom/media/test/padding-spanning-multiple-packets.mp3
Binary files differ
diff --git a/dom/media/test/pixel_aspect_ratio.mp4 b/dom/media/test/pixel_aspect_ratio.mp4
new file mode 100644
index 0000000000..fce12cc03e
--- /dev/null
+++ b/dom/media/test/pixel_aspect_ratio.mp4
Binary files differ
diff --git a/dom/media/test/play_promise.js b/dom/media/test/play_promise.js
new file mode 100644
index 0000000000..7051fedc19
--- /dev/null
+++ b/dom/media/test/play_promise.js
@@ -0,0 +1,3 @@
+function getNotSupportedFile(name) {
+ return name + ".bad";
+}
diff --git a/dom/media/test/poster-test.jpg b/dom/media/test/poster-test.jpg
new file mode 100644
index 0000000000..595a5315f8
--- /dev/null
+++ b/dom/media/test/poster-test.jpg
Binary files differ
diff --git a/dom/media/test/r11025_msadpcm_c1.wav b/dom/media/test/r11025_msadpcm_c1.wav
new file mode 100644
index 0000000000..2e883ba5ed
--- /dev/null
+++ b/dom/media/test/r11025_msadpcm_c1.wav
Binary files differ
diff --git a/dom/media/test/r11025_msadpcm_c1.wav^headers^ b/dom/media/test/r11025_msadpcm_c1.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/r11025_msadpcm_c1.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/r11025_s16_c1-short.wav b/dom/media/test/r11025_s16_c1-short.wav
new file mode 100644
index 0000000000..e08d5bbdc0
--- /dev/null
+++ b/dom/media/test/r11025_s16_c1-short.wav
Binary files differ
diff --git a/dom/media/test/r11025_s16_c1-short.wav^headers^ b/dom/media/test/r11025_s16_c1-short.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/r11025_s16_c1-short.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/r11025_s16_c1.wav b/dom/media/test/r11025_s16_c1.wav
new file mode 100644
index 0000000000..ab2e08befb
--- /dev/null
+++ b/dom/media/test/r11025_s16_c1.wav
Binary files differ
diff --git a/dom/media/test/r11025_s16_c1.wav^headers^ b/dom/media/test/r11025_s16_c1.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/r11025_s16_c1.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/r11025_s16_c1_trailing.wav b/dom/media/test/r11025_s16_c1_trailing.wav
new file mode 100644
index 0000000000..af53beaf25
--- /dev/null
+++ b/dom/media/test/r11025_s16_c1_trailing.wav
Binary files differ
diff --git a/dom/media/test/r11025_s16_c1_trailing.wav^headers^ b/dom/media/test/r11025_s16_c1_trailing.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/r11025_s16_c1_trailing.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/r11025_u8_c1.wav b/dom/media/test/r11025_u8_c1.wav
new file mode 100644
index 0000000000..97dc453b9e
--- /dev/null
+++ b/dom/media/test/r11025_u8_c1.wav
Binary files differ
diff --git a/dom/media/test/r11025_u8_c1.wav^headers^ b/dom/media/test/r11025_u8_c1.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/r11025_u8_c1.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/r11025_u8_c1_trunc.wav b/dom/media/test/r11025_u8_c1_trunc.wav
new file mode 100644
index 0000000000..4d2db39777
--- /dev/null
+++ b/dom/media/test/r11025_u8_c1_trunc.wav
Binary files differ
diff --git a/dom/media/test/r11025_u8_c1_trunc.wav^headers^ b/dom/media/test/r11025_u8_c1_trunc.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/r11025_u8_c1_trunc.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/r16000_u8_c1_list.wav b/dom/media/test/r16000_u8_c1_list.wav
new file mode 100644
index 0000000000..afde32e9a3
--- /dev/null
+++ b/dom/media/test/r16000_u8_c1_list.wav
Binary files differ
diff --git a/dom/media/test/r16000_u8_c1_list.wav^headers^ b/dom/media/test/r16000_u8_c1_list.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/r16000_u8_c1_list.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/rdd_process_xpcom/RddProcessTest.cpp b/dom/media/test/rdd_process_xpcom/RddProcessTest.cpp
new file mode 100644
index 0000000000..fad7d6ee2e
--- /dev/null
+++ b/dom/media/test/rdd_process_xpcom/RddProcessTest.cpp
@@ -0,0 +1,69 @@
+/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
+/* vim: set ts=8 sts=2 et sw=2 tw=80: */
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+#if defined(ENABLE_TESTS)
+# include "mozilla/RDDProcessManager.h"
+# include "mozilla/RDDChild.h"
+# include "mozilla/RddProcessTest.h"
+# include "mozilla/dom/Promise.h"
+
+namespace mozilla {
+
+NS_IMETHODIMP
+RddProcessTest::TestTelemetryProbes(JSContext* aCx,
+ mozilla::dom::Promise** aOutPromise) {
+ NS_ENSURE_ARG(aOutPromise);
+ *aOutPromise = nullptr;
+ nsIGlobalObject* global = xpc::CurrentNativeGlobal(aCx);
+ if (NS_WARN_IF(!global)) {
+ return NS_ERROR_FAILURE;
+ }
+
+ ErrorResult erv;
+ RefPtr<dom::Promise> promise = dom::Promise::Create(global, erv);
+ if (NS_WARN_IF(erv.Failed())) {
+ return erv.StealNSResult();
+ }
+
+ RDDProcessManager* rddProc = RDDProcessManager::Get();
+ MOZ_ASSERT(rddProc, "No RddProcessManager?");
+
+ rddProc->LaunchRDDProcess()->Then(
+ GetMainThreadSerialEventTarget(), __func__,
+ [promise, rddProc]() {
+ RDDChild* child = rddProc->GetRDDChild();
+ if (!rddProc) {
+ promise->MaybeReject(NS_ERROR_NOT_AVAILABLE);
+ }
+ MOZ_ASSERT(rddProc, "No RDD Proc?");
+
+ if (!child) {
+ promise->MaybeReject(NS_ERROR_NOT_AVAILABLE);
+ }
+ MOZ_ASSERT(child, "No RDD Child?");
+
+ Unused << child->SendTestTelemetryProbes();
+ promise->MaybeResolve((int32_t)rddProc->RDDProcessPid());
+ },
+ [promise](nsresult aError) {
+ MOZ_ASSERT_UNREACHABLE("RddProcessTest; failure to get RDD child");
+ promise->MaybeReject(aError);
+ });
+
+ promise.forget(aOutPromise);
+ return NS_OK;
+}
+
+NS_IMETHODIMP
+RddProcessTest::StopProcess() {
+ RDDProcessManager::RDDProcessShutdown();
+ return NS_OK;
+}
+
+NS_IMPL_ISUPPORTS(RddProcessTest, nsIRddProcessTest)
+
+} // namespace mozilla
+#endif // defined(ENABLE_TESTS)
diff --git a/dom/media/test/rdd_process_xpcom/RddProcessTest.h b/dom/media/test/rdd_process_xpcom/RddProcessTest.h
new file mode 100644
index 0000000000..b0281df45b
--- /dev/null
+++ b/dom/media/test/rdd_process_xpcom/RddProcessTest.h
@@ -0,0 +1,28 @@
+/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
+/* vim: set ts=8 sts=2 et sw=2 tw=80: */
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+#ifndef _include_dom_media_RddProcessTest_h_
+#define _include_dom_media_RddProcessTest_h_
+
+#if defined(ENABLE_TESTS)
+# include "nsIRddProcessTest.h"
+
+namespace mozilla {
+
+class RddProcessTest final : public nsIRddProcessTest {
+ public:
+ NS_DECL_THREADSAFE_ISUPPORTS
+ NS_DECL_NSIRDDPROCESSTEST
+
+ RddProcessTest() = default;
+
+ private:
+ ~RddProcessTest() = default;
+};
+
+} // namespace mozilla
+#endif // defined(ENABLE_TESTS)
+
+#endif // _include_dom_media_RddProcessTest_h_
diff --git a/dom/media/test/rdd_process_xpcom/components.conf b/dom/media/test/rdd_process_xpcom/components.conf
new file mode 100644
index 0000000000..507559851b
--- /dev/null
+++ b/dom/media/test/rdd_process_xpcom/components.conf
@@ -0,0 +1,15 @@
+# -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
+# vim: set filetype=python:
+# This Source Code Form is subject to the terms of the Mozilla Public
+# License, v. 2.0. If a copy of the MPL was not distributed with this
+# file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
+Classes = [
+ {
+ 'cid': '{12f7d302-5368-412d-bdc9-26d151518e6c}',
+ 'contract_ids': ['@mozilla.org/rdd-process-test;1'],
+ 'type': 'mozilla::RddProcessTest',
+ 'headers': ['mozilla/RddProcessTest.h'],
+ 'processes': ProcessSelector.MAIN_PROCESS_ONLY,
+ },
+]
diff --git a/dom/media/test/rdd_process_xpcom/moz.build b/dom/media/test/rdd_process_xpcom/moz.build
new file mode 100644
index 0000000000..87dbb39bc1
--- /dev/null
+++ b/dom/media/test/rdd_process_xpcom/moz.build
@@ -0,0 +1,21 @@
+# -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
+# vim: set filetype=python:
+# This Source Code Form is subject to the terms of the Mozilla Public
+# License, v. 2.0. If a copy of the MPL was not distributed with this
+# file, You can obtain one at http://mozilla.org/MPL/2.0/.
+
+EXPORTS.mozilla += ["RddProcessTest.h"]
+
+UNIFIED_SOURCES += ["RddProcessTest.cpp"]
+
+XPCOM_MANIFESTS += ["components.conf"]
+
+XPIDL_MODULE = "rdd_process_xpcom_test"
+
+XPIDL_SOURCES += [
+ "nsIRddProcessTest.idl",
+]
+
+include("/ipc/chromium/chromium-config.mozbuild")
+
+FINAL_LIBRARY = "xul"
diff --git a/dom/media/test/rdd_process_xpcom/nsIRddProcessTest.idl b/dom/media/test/rdd_process_xpcom/nsIRddProcessTest.idl
new file mode 100644
index 0000000000..a10fdbd249
--- /dev/null
+++ b/dom/media/test/rdd_process_xpcom/nsIRddProcessTest.idl
@@ -0,0 +1,25 @@
+/* -*- Mode: C++; c-basic-offset: 2; indent-tabs-mode: nil; tab-width: 8 -*- */
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+#include "nsISupports.idl"
+
+[scriptable, uuid(12f7d302-5368-412d-bdc9-26d151518e6c)]
+interface nsIRddProcessTest : nsISupports
+{
+ /**
+ * ** Test-only Method **
+ *
+ * Sending Telemetry probes
+ */
+ [implicit_jscontext]
+ Promise testTelemetryProbes();
+
+ /**
+ * ** Test-only Method **
+ *
+ * Stop existing RDD process
+ */
+ void stopProcess();
+};
diff --git a/dom/media/test/reactivate_helper.html b/dom/media/test/reactivate_helper.html
new file mode 100644
index 0000000000..1834131559
--- /dev/null
+++ b/dom/media/test/reactivate_helper.html
@@ -0,0 +1,57 @@
+<!DOCTYPE HTML>
+<html>
+<body>
+<script>
+var loadsWaiting = 0;
+var elements = [];
+
+function checkAllLoaded() {
+ --loadsWaiting;
+ if (loadsWaiting == 0) {
+ parent.loadedAll(elements);
+ }
+}
+
+function loadedData(event) {
+ var e = event.target;
+ parent.ok(!elements.includes(e), "Element already loaded: " + e._name);
+ parent.info("Loaded " + e._name);
+ elements.push(e);
+ // Reset "onerror" handler to avoid triggering another error in removeNodeAndSource().
+ e.onerror = null;
+ checkAllLoaded();
+
+}
+
+function error(event) {
+ var e = event.target;
+ parent.info("Error " + e._name);
+ // Don't wait for the element encounting errors.
+ checkAllLoaded();
+}
+
+for (var i = 0; i < parent.gSmallTests.length; ++i) {
+ var test = parent.gSmallTests[i];
+ var elemType = /^audio/.test(test.type) ? "audio" : "video";
+ // Associate these elements with the subframe's document
+ var e = document.createElement(elemType);
+ e.preload = "metadata";
+ if (e.canPlayType(test.type)) {
+ e.src = test.name;
+ e._name = test.name;
+ e.onloadeddata = loadedData;
+ e.onerror = error;
+ e.load();
+ ++loadsWaiting;
+ parent.info("Loading " + e._name);
+ }
+}
+
+if (loadsWaiting == 0) {
+ parent.todo(false, "Can't play anything");
+} else {
+ parent.SimpleTest.waitForExplicitFinish();
+}
+</script>
+</body>
+</html>
diff --git a/dom/media/test/red-46x48.mp4 b/dom/media/test/red-46x48.mp4
new file mode 100644
index 0000000000..0760cc1c16
--- /dev/null
+++ b/dom/media/test/red-46x48.mp4
Binary files differ
diff --git a/dom/media/test/red-46x48.mp4^headers^ b/dom/media/test/red-46x48.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/red-46x48.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/red-48x46.mp4 b/dom/media/test/red-48x46.mp4
new file mode 100644
index 0000000000..d83de4027d
--- /dev/null
+++ b/dom/media/test/red-48x46.mp4
Binary files differ
diff --git a/dom/media/test/red-48x46.mp4^headers^ b/dom/media/test/red-48x46.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/red-48x46.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/redirect.sjs b/dom/media/test/redirect.sjs
new file mode 100644
index 0000000000..6873fd52f3
--- /dev/null
+++ b/dom/media/test/redirect.sjs
@@ -0,0 +1,35 @@
+function parseQuery(request, key) {
+ var params = request.queryString.split("&");
+ for (var j = 0; j < params.length; ++j) {
+ var p = params[j];
+ if (p == key) {
+ return true;
+ }
+ if (p.indexOf(key + "=") == 0) {
+ return p.substring(key.length + 1);
+ }
+ if (!p.includes("=") && key == "") {
+ return p;
+ }
+ }
+ return false;
+}
+
+// Return file content for the first request with a given key.
+// All subsequent requests return a redirect to a different-origin resource.
+function handleRequest(request, response) {
+ var domain = parseQuery(request, "domain");
+ var file = parseQuery(request, "file");
+ var allowed = parseQuery(request, "allowed");
+
+ response.setStatusLine(request.httpVersion, 303, "See Other");
+ response.setHeader(
+ "Location",
+ "http://" +
+ domain +
+ "/tests/dom/media/test/" +
+ (allowed ? "allowed.sjs?" : "") +
+ file
+ );
+ response.setHeader("Content-Type", "text/html");
+}
diff --git a/dom/media/test/referer.sjs b/dom/media/test/referer.sjs
new file mode 100644
index 0000000000..0c699c4ae1
--- /dev/null
+++ b/dom/media/test/referer.sjs
@@ -0,0 +1,49 @@
+function parseQuery(request, key) {
+ var params = request.queryString.split("&");
+ for (var j = 0; j < params.length; ++j) {
+ var p = params[j];
+ if (p == key) {
+ return true;
+ }
+ if (p.indexOf(key + "=") == 0) {
+ return p.substring(key.length + 1);
+ }
+ if (!p.includes("=") && key == "") {
+ return p;
+ }
+ }
+ return false;
+}
+
+function handleRequest(request, response) {
+ var referer = request.hasHeader("Referer")
+ ? request.getHeader("Referer")
+ : undefined;
+ if (
+ referer == "http://mochi.test:8888/tests/dom/media/test/test_referer.html"
+ ) {
+ var name = parseQuery(request, "name");
+ var type = parseQuery(request, "type");
+ var file = Services.dirsvc.get("CurWorkD", Ci.nsIFile);
+ var fis = Cc["@mozilla.org/network/file-input-stream;1"].createInstance(
+ Ci.nsIFileInputStream
+ );
+ var bis = Cc["@mozilla.org/binaryinputstream;1"].createInstance(
+ Ci.nsIBinaryInputStream
+ );
+ var paths = "tests/dom/media/test/" + name;
+ var split = paths.split("/");
+ for (var i = 0; i < split.length; ++i) {
+ file.append(split[i]);
+ }
+ fis.init(file, -1, -1, false);
+ bis.setInputStream(fis);
+ var bytes = bis.readBytes(bis.available());
+ response.setHeader("Content-Length", "" + bytes.length, false);
+ response.setHeader("Content-Type", type, false);
+ response.write(bytes, bytes.length);
+ bis.close();
+ } else {
+ response.setStatusLine(request.httpVersion, 404, "Not found");
+ }
+}
diff --git a/dom/media/test/reftest/av1hdr2020.mp4 b/dom/media/test/reftest/av1hdr2020.mp4
new file mode 100644
index 0000000000..295bec8a3c
--- /dev/null
+++ b/dom/media/test/reftest/av1hdr2020.mp4
Binary files differ
diff --git a/dom/media/test/reftest/av1hdr2020.png b/dom/media/test/reftest/av1hdr2020.png
new file mode 100644
index 0000000000..c5d3344a80
--- /dev/null
+++ b/dom/media/test/reftest/av1hdr2020.png
Binary files differ
diff --git a/dom/media/test/reftest/bipbop_300_215kbps.mp4.lastframe-ref.html b/dom/media/test/reftest/bipbop_300_215kbps.mp4.lastframe-ref.html
new file mode 100644
index 0000000000..575acb107d
--- /dev/null
+++ b/dom/media/test/reftest/bipbop_300_215kbps.mp4.lastframe-ref.html
@@ -0,0 +1,4 @@
+<!DOCTYPE HTML>
+<img style="position:absolute; left:0; top:0"
+src=""
+>
diff --git a/dom/media/test/reftest/bipbop_300_215kbps.mp4.lastframe.html b/dom/media/test/reftest/bipbop_300_215kbps.mp4.lastframe.html
new file mode 100644
index 0000000000..600b04a4f0
--- /dev/null
+++ b/dom/media/test/reftest/bipbop_300_215kbps.mp4.lastframe.html
@@ -0,0 +1,19 @@
+<!DOCTYPE HTML>
+<html class="reftest-wait">
+<head>
+<script type="text/javascript">
+function doTest() {
+ var video = document.getElementById("v1");
+ video.src = "../bipbop_300_215kbps.mp4";
+ video.play();
+ video.addEventListener("ended", function() {
+ document.documentElement.removeAttribute('class');
+ }, {once: true});
+}
+window.addEventListener("MozReftestInvalidate", doTest);
+</script>
+</head>
+<body>
+<video id="v1" style="position:absolute; left:0; top:0"></video>
+</body>
+</html>
diff --git a/dom/media/test/reftest/color_quads/720p.png b/dom/media/test/reftest/color_quads/720p.png
new file mode 100644
index 0000000000..cf3f2408c1
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.gbrp.av1.mp4 b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.gbrp.av1.mp4
new file mode 100644
index 0000000000..fa1801013a
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.gbrp.av1.mp4
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.gbrp.av1.webm b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.gbrp.av1.webm
new file mode 100644
index 0000000000..96fd46f848
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.gbrp.av1.webm
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.gbrp.h264.mp4 b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.gbrp.h264.mp4
new file mode 100644
index 0000000000..c5f08a56f2
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.gbrp.h264.mp4
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.gbrp.vp9.mp4 b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.gbrp.vp9.mp4
new file mode 100644
index 0000000000..c64f7eb8d2
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.gbrp.vp9.mp4
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.gbrp.vp9.webm b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.gbrp.vp9.webm
new file mode 100644
index 0000000000..d4341d22ed
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.gbrp.vp9.webm
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p.av1.mp4 b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p.av1.mp4
new file mode 100644
index 0000000000..ac89827620
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p.av1.mp4
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p.av1.webm b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p.av1.webm
new file mode 100644
index 0000000000..1544ef996e
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p.av1.webm
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p.h264.mp4 b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p.h264.mp4
new file mode 100644
index 0000000000..041d7b4477
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p.h264.mp4
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p.vp9.mp4 b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p.vp9.mp4
new file mode 100644
index 0000000000..e9cdbb5f51
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p.vp9.mp4
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p.vp9.webm b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p.vp9.webm
new file mode 100644
index 0000000000..6d0b53f002
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p.vp9.webm
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p10.av1.mp4 b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p10.av1.mp4
new file mode 100644
index 0000000000..f30c731bf6
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p10.av1.mp4
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p10.av1.webm b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p10.av1.webm
new file mode 100644
index 0000000000..0e733057d8
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p10.av1.webm
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p10.h264.mp4 b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p10.h264.mp4
new file mode 100644
index 0000000000..9a4783f2db
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p10.h264.mp4
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p10.vp9.mp4 b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p10.vp9.mp4
new file mode 100644
index 0000000000..00565d66d3
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p10.vp9.mp4
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p10.vp9.webm b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p10.vp9.webm
new file mode 100644
index 0000000000..08159b5233
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.pc.yuv420p10.vp9.webm
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.gbrp.av1.mp4 b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.gbrp.av1.mp4
new file mode 100644
index 0000000000..3a50d11691
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.gbrp.av1.mp4
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.gbrp.av1.webm b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.gbrp.av1.webm
new file mode 100644
index 0000000000..c18de4bf4f
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.gbrp.av1.webm
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.gbrp.h264.mp4 b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.gbrp.h264.mp4
new file mode 100644
index 0000000000..9d735cf9ed
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.gbrp.h264.mp4
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.gbrp.vp9.mp4 b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.gbrp.vp9.mp4
new file mode 100644
index 0000000000..79dcb0d72d
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.gbrp.vp9.mp4
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.gbrp.vp9.webm b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.gbrp.vp9.webm
new file mode 100644
index 0000000000..31af47cba1
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.gbrp.vp9.webm
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p.av1.mp4 b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p.av1.mp4
new file mode 100644
index 0000000000..dfcac969b9
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p.av1.mp4
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p.av1.webm b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p.av1.webm
new file mode 100644
index 0000000000..00f46b0597
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p.av1.webm
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p.h264.mp4 b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p.h264.mp4
new file mode 100644
index 0000000000..ca8b8a1572
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p.h264.mp4
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p.vp9.mp4 b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p.vp9.mp4
new file mode 100644
index 0000000000..19ff1b4265
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p.vp9.mp4
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p.vp9.webm b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p.vp9.webm
new file mode 100644
index 0000000000..6fd0aa6756
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p.vp9.webm
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p10.av1.mp4 b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p10.av1.mp4
new file mode 100644
index 0000000000..e95fd5c9fb
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p10.av1.mp4
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p10.av1.webm b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p10.av1.webm
new file mode 100644
index 0000000000..f28e65f034
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p10.av1.webm
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p10.h264.mp4 b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p10.h264.mp4
new file mode 100644
index 0000000000..1fe6824cb4
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p10.h264.mp4
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p10.vp9.mp4 b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p10.vp9.mp4
new file mode 100644
index 0000000000..532ae19359
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p10.vp9.mp4
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p10.vp9.webm b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p10.vp9.webm
new file mode 100644
index 0000000000..e4aad890f5
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/720p.png.bt709.bt709.tv.yuv420p10.vp9.webm
Binary files differ
diff --git a/dom/media/test/reftest/color_quads/reftest.list b/dom/media/test/reftest/color_quads/reftest.list
new file mode 100644
index 0000000000..6071733f76
--- /dev/null
+++ b/dom/media/test/reftest/color_quads/reftest.list
@@ -0,0 +1,69 @@
+# Reference image generated via https://jdashg.github.io/misc/colors/color-quads-16-127-235.html
+# Test videos encoded via ../gen_combos.py --write color_quads/720p.png
+
+# We're sort of testing two things here:
+# 1. Does a av1.webm video into the actual values we expect?
+# 2. Do other similar videos decode the same was as av1.webm?
+# We have this split because while each platform/compositor has its own inaccuracies,
+# each platform/compositor will have the *same* inaccuracies regardless of video.
+# So, we just need to first check if e.g. av1.webm decodes to what we expect,
+# and then we have generally trivially compare other codecs/containers to that.
+
+defaults pref(media.av1.enabled,true)
+
+# -
+# yuv420p
+
+fuzzy(16-51,5234-5622) fuzzy-if(swgl,32-38,1600-91746) fuzzy-if(useDrawSnapshot,16-16,11600-11600) fuzzy-if(OSX,16-73,5212-5622) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.yuv420p.av1.webm ../reftest_img.html?src=color_quads/720p.png
+fuzzy-if(winWidget&&swgl,0-20,0-5620) fuzzy-if(winWidget&&!swgl,0-1,0-78) fuzzy-if(Android,254-255,273680-273807) fuzzy-if(OSX,0-35,0-1947) fuzzy-if(OSX&&swgl,0-67,0-5451) fuzzy-if(appleSilicon,30-48,1760-187409) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.yuv420p.vp9.webm ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.yuv420p.av1.webm
+fuzzy-if(winWidget,0-1,0-78) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.yuv420p.av1.mp4 ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.yuv420p.av1.webm
+skip-if(winWidget&&isCoverageBuild) fuzzy(0-16,75-1941) fuzzy-if(Android,28-255,273680-359920) fuzzy-if(OSX,30-32,187326-187407) fuzzy-if(appleSilicon,30-48,1835-187409) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.yuv420p.h264.mp4 ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.yuv420p.av1.webm
+fuzzy-if(winWidget&&swgl,0-20,0-5620) fuzzy-if(winWidget&&!swgl,0-1,0-78) fuzzy-if(Android,254-255,273680-273807) fuzzy-if(OSX,0-35,0-1947) fuzzy-if(OSX&&swgl,0-67,0-5451) fuzzy-if(appleSilicon,30-48,1760-187409) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.yuv420p.vp9.mp4 ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.yuv420p.av1.webm
+
+skip-if(Android) fuzzy(16-48,8107-8818) fuzzy-if(winWidget&&swgl,31-38,8240-184080) fuzzy-if(appleSilicon,33-38,8819-11705) fuzzy-if(useDrawSnapshot,20-20,187200-187200) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.yuv420p.av1.webm ../reftest_img.html?src=color_quads/720p.png
+skip-if(Android) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.yuv420p.av1.mp4 ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.yuv420p.av1.webm
+# On Windows & sw render, we noticed that the comparison image captured from AV1 is not equal to its displayed video frame, so we would need to compare other codecs directly to PNG file. That should be fixed in bug 1748540.
+skip-if(Android) skip-if(winWidget&&swgl) fuzzy-if(Android,255-255,273726-273726) fuzzy-if(OSX,0-16,0-1718) fuzzy-if(OSX&&swgl,0-20,0-2423) fuzzy-if(appleSilicon,0-16,0-1874) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.yuv420p.vp9.webm ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.yuv420p.av1.webm
+skip-if(Android) skip-if(winWidget&&swgl) fuzzy-if(Android,255-255,273726-273726) fuzzy-if(OSX,2-36,184281-187407) fuzzy-if(winWidget,0-21,0-360000) fuzzy-if(appleSilicon,36-49,187329-187407) fuzzy-if(useDrawSnapshot,0-1,0-10) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.yuv420p.h264.mp4 ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.yuv420p.av1.webm
+skip-if(Android) skip-if(winWidget&&swgl) fuzzy-if(Android,255-255,273726-273726) fuzzy-if(OSX,0-16,0-1718) fuzzy-if(OSX&&swgl,0-20,0-2423) fuzzy-if(appleSilicon,0-16,0-1874) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.yuv420p.vp9.mp4 ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.yuv420p.av1.webm
+skip-if(Android) skip-if(!(winWidget&&swgl)) fuzzy(0-35,0-8506) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.yuv420p.vp9.webm ../reftest_img.html?src=color_quads/720p.png
+skip-if(Android) skip-if(!(winWidget&&swgl)) fuzzy(0-35,0-8506) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.yuv420p.vp9.mp4 ../reftest_img.html?src=color_quads/720p.png
+
+# -
+# yuv420p10
+
+skip-if(Android) fuzzy(33-49,1870-2579) fuzzy-if(swgl,34-52,180421-270528) fuzzy-if(useDrawSnapshot,16-16,183840-183840) fuzzy-if(OSX,60-74,270329-271024) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.yuv420p10.av1.webm ../reftest_img.html?src=color_quads/720p.png
+skip-if(Android) fuzzy-if(OSX,0-12,0-187770) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.yuv420p10.vp9.webm ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.yuv420p10.av1.webm
+skip-if(Android) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.yuv420p10.av1.mp4 ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.yuv420p10.av1.webm
+#[2] skip-if(Android) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.yuv420p10.h264.mp4 ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.yuv420p10.av1.webm
+skip-if(Android) fuzzy-if(OSX,0-12,0-187770) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.yuv420p10.vp9.mp4 ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.yuv420p10.av1.webm
+
+skip-if(Android) fuzzy(33-49,174620-270059) fuzzy-if(swgl&&!winWidget,36-52,11553-11555) fuzzy-if(swgl&&winWidget,36-52,11554-187200) fuzzy-if(swgl&&OSX,34-50,11465-270059) fuzzy-if(useDrawSnapshot,20-20,186800-186800) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.yuv420p10.av1.webm ../reftest_img.html?src=color_quads/720p.png
+skip-if(Android) fuzzy-if(OSX,0-12,0-274122) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.yuv420p10.vp9.webm ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.yuv420p10.av1.webm
+skip-if(Android) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.yuv420p10.av1.mp4 ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.yuv420p10.av1.webm
+#[2] skip-if(Android) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.yuv420p10.h264.mp4 ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.yuv420p10.av1.webm
+skip-if(Android) fuzzy-if(OSX,0-12,0-274122) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.yuv420p10.vp9.mp4 ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.yuv420p10.av1.webm
+
+# Android is really broken in a variety of ways for p10.
+#[2]: yuv420p10 broken in h264.mp4: https://bugzilla.mozilla.org/show_bug.cgi?id=1711812
+
+
+# -
+# gbrp
+# Note: tv-gbrp doesn't really make sense, and we should consider dropping it.
+# Specifically, we should probably do (gbrp, ...(tv,pc)x(yuv,yuv10)) instead of (tv,pc)x(gbrp,yuv,yuv10)
+# That said, we should probably test a couple combos, at least. (But then again, why not all!)
+
+skip-if(winWidget&&swgl) fuzzy(0-1,0-3600) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.gbrp.av1.webm ../reftest_img.html?src=color_quads/720p.png
+skip-if(winWidget&&swgl) fuzzy(0-1,0-7200) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.gbrp.av1.webm ../reftest_img.html?src=color_quads/720p.png
+
+== ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.gbrp.av1.mp4 ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.gbrp.av1.webm
+== ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.gbrp.av1.mp4 ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.gbrp.av1.webm
+
+# Our h264.mp4 doesn't handle gbrp, but *also* doesn't error properly.
+== ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.gbrp.h264.mp4 ../reftest_video.html?src=timeout
+== ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.gbrp.h264.mp4 ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.gbrp.h264.mp4
+
+fuzzy(0-1,0-3600) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.gbrp.vp9.webm ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.gbrp.vp9.webm
+== ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.gbrp.vp9.mp4 ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.gbrp.vp9.webm
+fuzzy(0-1,0-3600) == ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.pc.gbrp.vp9.mp4 ../reftest_video.html?src=color_quads/720p.png.bt709.bt709.tv.gbrp.vp9.webm
diff --git a/dom/media/test/reftest/frame_order.mp4 b/dom/media/test/reftest/frame_order.mp4
new file mode 100644
index 0000000000..87b65f4386
--- /dev/null
+++ b/dom/media/test/reftest/frame_order.mp4
Binary files differ
diff --git a/dom/media/test/reftest/frame_order_mp4-ref.html b/dom/media/test/reftest/frame_order_mp4-ref.html
new file mode 100644
index 0000000000..0c59debcfa
--- /dev/null
+++ b/dom/media/test/reftest/frame_order_mp4-ref.html
@@ -0,0 +1,13 @@
+<!DOCTYPE HTML>
+<head>
+</head>
+<body>
+<!--
+ This ref image is generated by capturing video frame on 0.3s from
+ 'frame_order.mp4'
+-->
+<image style="position:absolute; left:0; top:0" src=""></video>
+<script type="text/javascript">
+</script>
+</body>
+</html>
diff --git a/dom/media/test/reftest/frame_order_mp4.html b/dom/media/test/reftest/frame_order_mp4.html
new file mode 100644
index 0000000000..e77cf05e9e
--- /dev/null
+++ b/dom/media/test/reftest/frame_order_mp4.html
@@ -0,0 +1,37 @@
+<!DOCTYPE HTML>
+<html class="reftest-wait">
+<head>
+</head>
+<body>
+<video id="v" style="position:absolute; left:0; top:0"></video>
+<canvas id="canvas" style="position:absolute; left:0; top:0"></video>
+<script type="text/javascript">
+/**
+ * Do seek multiple times and check the video frame on 0.3s.
+ */
+async function testFrameOrderAfterSeeking() {
+ const video = document.getElementById("v");
+ video.src = "frame_order.mp4";
+ await new Promise(r => video.oncanplay = r);
+ // The issue won't happen on the first seek, because the decoder hasn't been
+ // created yet.
+ video.currentTime = 0.1;
+ await new Promise(r => video.onseeked = r);
+ video.currentTime = 0.3;
+ await new Promise(r => video.onseeked = r);
+ // Since our media pipeline sends the frame to imageBridge, then fires
+ // a seeked event, the target frame may not be shown on the screen.
+ // So using canvas to access the target frame in the imageContainer in
+ // videoElement.
+ const canvas = document.getElementById("canvas");
+ canvas.width = video.videoWidth;
+ canvas.height = video.videoHeight;
+ const ctx = canvas.getContext("2d");
+ ctx.drawImage(video, 0, 0, video.videoWidth, video.videoHeight);
+ document.documentElement.removeAttribute('class');
+};
+
+window.addEventListener("MozReftestInvalidate", testFrameOrderAfterSeeking);
+</script>
+</body>
+</html>
diff --git a/dom/media/test/reftest/gen_combos.py b/dom/media/test/reftest/gen_combos.py
new file mode 100644
index 0000000000..f7e7d50fe1
--- /dev/null
+++ b/dom/media/test/reftest/gen_combos.py
@@ -0,0 +1,257 @@
+#!/usr/bin/env python3
+
+# E.g. `./gen_combos.py [--write] color_quads/720p.png`
+
+import concurrent.futures
+import pathlib
+import subprocess
+import sys
+
+ARGS = sys.argv
+SRC_PATH = pathlib.Path(ARGS.pop())
+assert SRC_PATH.exists(), "gen_combos.py [--flags] <src file path>"
+DIR = SRC_PATH.parent
+
+
+# crossCombine([{a:false},{a:5}], [{},{b:5}])
+# [{a:false}, {a:true}, {a:false,b:5}, {a:true,b:5}]
+def cross_combine(*args):
+ args = list(args)
+
+ def cross_combine2(listA, listB):
+ listC = []
+ for a in listA:
+ for b in listB:
+ c = dict()
+ c.update(a)
+ c.update(b)
+ listC.append(c)
+ return listC
+
+ res = [dict()]
+ while True:
+ try:
+ next = args.pop(0)
+ except IndexError:
+ break
+ res = cross_combine2(res, next)
+ return res
+
+
+def keyed_combiner(key, vals):
+ res = []
+ for v in vals:
+ d = dict()
+ d[key] = v
+ res.append(d)
+ return res
+
+
+# -
+
+
+def eprint(*args, **kwargs):
+ print(*args, file=sys.stderr, **kwargs)
+
+
+# -
+
+OGG = []
+WEBM_CODECS = ["av1", "vp9"]
+
+if "--all" in ARGS:
+ OGG = cross_combine(
+ [{"ext": "ogg"}], keyed_combiner("vcodec", ["theora", "vp8", "vp9"])
+ )
+ WEBM_CODECS += ["vp8"]
+
+MP4 = cross_combine([{"ext": "mp4"}], keyed_combiner("vcodec", ["av1", "h264", "vp9"]))
+
+WEBM = cross_combine([{"ext": "webm"}], keyed_combiner("vcodec", WEBM_CODECS))
+
+# -
+
+FORMAT_LIST = set(
+ [
+ "yuv420p",
+ "yuv420p10",
+ # 'yuv420p12',
+ # 'yuv420p16be',
+ # 'yuv420p16le',
+ "gbrp",
+ ]
+)
+
+if "--all" in ARGS:
+ FORMAT_LIST |= set(
+ [
+ "yuv420p",
+ "yuv420p10",
+ "yuv420p12",
+ "yuv420p16be",
+ "yuv420p16le",
+ "yuv422p",
+ "yuv422p10",
+ "yuv422p12",
+ "yuv422p16be",
+ "yuv422p16le",
+ "yuv444p",
+ "yuv444p10",
+ "yuv444p12",
+ "yuv444p16be",
+ "yuv444p16le",
+ "yuv411p",
+ "yuv410p",
+ "yuyv422",
+ "uyvy422",
+ "rgb24",
+ "bgr24",
+ "rgb8",
+ "bgr8",
+ "rgb444be",
+ "rgb444le",
+ "bgr444be",
+ "bgr444le",
+ # 'nv12', # Encoding not different than yuv420p?
+ # 'nv21', # Encoding not different than yuv420p?
+ "gbrp",
+ "gbrp9be",
+ "gbrp9le",
+ "gbrp10be",
+ "gbrp10le",
+ "gbrp12be",
+ "gbrp12le",
+ "gbrp14be",
+ "gbrp14le",
+ "gbrp16be",
+ "gbrp16le",
+ ]
+ )
+
+FORMATS = keyed_combiner("format", list(FORMAT_LIST))
+
+RANGE = keyed_combiner("range", ["tv", "pc"])
+
+CSPACE_LIST = set(
+ [
+ "bt709",
+ # 'bt2020',
+ ]
+)
+
+if "--all" in ARGS:
+ CSPACE_LIST |= set(
+ [
+ "bt709",
+ "bt2020",
+ "bt601-6-525", # aka smpte170m NTSC
+ "bt601-6-625", # aka bt470bg PAL
+ ]
+ )
+CSPACE_LIST = list(CSPACE_LIST)
+
+# -
+
+COMBOS = cross_combine(
+ WEBM + MP4 + OGG,
+ FORMATS,
+ RANGE,
+ keyed_combiner("src_cspace", CSPACE_LIST),
+ keyed_combiner("dst_cspace", CSPACE_LIST),
+)
+
+# -
+
+print(f"{len(COMBOS)} combinations...")
+
+todo = []
+for c in COMBOS:
+ dst_name = ".".join(
+ [
+ SRC_PATH.name,
+ c["src_cspace"],
+ c["dst_cspace"],
+ c["range"],
+ c["format"],
+ c["vcodec"],
+ c["ext"],
+ ]
+ )
+
+ src_cspace = c["src_cspace"]
+
+ vf = f"scale=out_range={c['range']}"
+ vf += f",colorspace=all={c['dst_cspace']}"
+ vf += f":iall={src_cspace}"
+ args = [
+ "ffmpeg",
+ "-y",
+ # For input:
+ "-color_primaries",
+ src_cspace,
+ "-color_trc",
+ src_cspace,
+ "-colorspace",
+ src_cspace,
+ "-i",
+ SRC_PATH.as_posix(),
+ # For output:
+ "-bitexact", # E.g. don't use true random uuids
+ "-vf",
+ vf,
+ "-pix_fmt",
+ c["format"],
+ "-vcodec",
+ c["vcodec"],
+ "-crf",
+ "1", # Not-quite-lossless
+ (DIR / dst_name).as_posix(),
+ ]
+ if "-v" in ARGS or "-vv" in ARGS:
+ print("$ " + " ".join(args))
+ else:
+ print(" " + args[-1])
+
+ todo.append(args)
+
+# -
+
+with open(DIR / "reftest.list", "r") as f:
+ reftest_list_text = f.read()
+
+for args in todo:
+ vid_name = pathlib.Path(args[-1]).name
+ if vid_name not in reftest_list_text:
+ print(f"WARNING: Not in reftest.list: {vid_name}")
+
+# -
+
+if "--write" not in ARGS:
+ print("Use --write to write. Exiting...")
+ exit(0)
+
+# -
+
+
+def run_cmd(args):
+ dest = None
+ if "-vv" not in ARGS:
+ dest = subprocess.DEVNULL
+ try:
+ subprocess.run(args, stderr=dest)
+ except FileNotFoundError:
+ print("FileNotFoundError, is ffmpeg not in your PATH?")
+ raise
+
+
+with concurrent.futures.ThreadPoolExecutor() as pool:
+ fs = []
+ for cur_args in todo:
+ f = pool.submit(run_cmd, cur_args)
+ fs.append(f)
+
+ done = 0
+ for f in concurrent.futures.as_completed(fs):
+ f.result() # Raise if it raised
+ done += 1
+ sys.stdout.write(f"\rEncoded {done}/{len(todo)}")
diff --git a/dom/media/test/reftest/generateREF.html b/dom/media/test/reftest/generateREF.html
new file mode 100644
index 0000000000..4e26066973
--- /dev/null
+++ b/dom/media/test/reftest/generateREF.html
@@ -0,0 +1,104 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+<script type="application/javascript">
+</script>
+</head>
+<body>
+<p id="out"></p>
+<video id="v1" style="position:absolute; left:0; top:0"></video>
+<canvas id="canvas"></canvas>
+<script type="application/javascript">
+// READ ME FIRST.
+// The script is trying to make a reftest sample for reftest.
+
+// STEP1. Uncomment the method below that you want to use. If you want to dump
+// Nth frame, modify the parameter to the number of frame you want to dump.
+//window.onload = function() { setTimeout(dumpFirstFrame, 0); };
+//window.onload = function() { setTimeout(dumpLastFrame, 0); };
+window.onload = function() { setTimeout(function(){dumpNthFrame(15);}, 0); };
+
+// STEP2. Set the source of video that you want to capture
+const videoSrc = '';
+
+// STEP3. Ensure the pref `media.seekToNextFrame.enabled` is on
+// STEP4. In a terminal, navigate to the containing folder and start a server with "python -m SimpleHTTPServer 8000"
+// STEP5. Open "http://localhost:8000/generateREF.html" in the browser
+// STEP6. Copy the base64 image url to your xxx-ref.html
+
+function drawVideoToInnerHTML(v) {
+ // This allows to dump content via canvas when the source is cross-origin.
+ v.crossorigin = "anonymous";
+ var canvas = document.getElementById("canvas");
+ canvas.width = v.videoWidth;
+ canvas.height = v.videoHeight;
+ var ctx = canvas.getContext("2d");
+ ctx.drawImage(v, 0, 0, v.videoWidth, v.videoHeight);
+ var dataURL = canvas.toDataURL();
+ document.getElementById("out").innerHTML=dataURL;
+}
+
+function dumpFirstFrame() {
+ var video = document.getElementById("v1");
+ video.src = videoSrc;
+ video.preload = "metadata";
+
+ video.addEventListener("loadeddata", function() {
+ drawVideoToInnerHTML(video);
+ });
+}
+
+function dumpNthFrame(n) {
+ var video = document.getElementById("v1");
+ video.src = videoSrc;
+ video.preload = "metadata";
+ const totalFrames = n;
+
+ function checkNthFrame() {
+ console.log((totalFrames-n+1)+"th Frame time is " + video.currentTime);
+ n--;
+ if (n == 0) {
+ drawVideoToInnerHTML(video);
+ } else {
+ video.seekToNextFrame();
+ }
+ }
+ video.addEventListener("loadeddata", checkNthFrame);
+ video.addEventListener("seeked", checkNthFrame);
+}
+
+function dumpLastFrame() {
+ var video = document.getElementById("v1");
+ video.src = videoSrc;
+ video.preload = "metadata";
+ video.seenEnded = false;
+
+ // Seek to the end
+ video.addEventListener("loadeddata", function() {
+ video.currentTime = video.duration;
+ video.onseeked = () => {
+ video.onseeked = null;
+ callSeekToNextFrame();
+ };
+ });
+
+ function callSeekToNextFrame() {
+ video.seekToNextFrame().then(
+ () => {
+ if (!video.seenEnded)
+ callSeekToNextFrame();
+ },
+ () => {
+ // Reach the end, do nothing.
+ }
+ );
+ }
+
+ video.addEventListener("ended", function() {
+ video.seenEnded = true;
+ drawVideoToInnerHTML(video);
+ });
+}
+</script>
+</body>
+</html>
diff --git a/dom/media/test/reftest/gizmo.mp4.55thframe-ref.html b/dom/media/test/reftest/gizmo.mp4.55thframe-ref.html
new file mode 100644
index 0000000000..28a93cc268
--- /dev/null
+++ b/dom/media/test/reftest/gizmo.mp4.55thframe-ref.html
@@ -0,0 +1,7 @@
+<!DOCTYPE HTML>
+<img style="position:absolute; left:0; top:0"
+src="
+
+
+"
+>
diff --git a/dom/media/test/reftest/gizmo.mp4.seek.html b/dom/media/test/reftest/gizmo.mp4.seek.html
new file mode 100644
index 0000000000..e4c1fe9515
--- /dev/null
+++ b/dom/media/test/reftest/gizmo.mp4.seek.html
@@ -0,0 +1,36 @@
+<!DOCTYPE HTML>
+<html class="reftest-wait">
+<!--This testing should match the 55th frame of gizmo.mp4. The
+55th frame's time is 1.8s, so seek to a time which is a little
+greater than 1.8s, the display frame should be the 55th frame.
+-->
+<head>
+<script type="text/javascript">
+function doTest() {
+ var video = document.getElementById("v1");
+ video.src = "../gizmo.mp4";
+ video.preload = "metadata";
+
+ video.currentTime = 1.801;
+
+ video.addEventListener("seeked", function() {
+ // Since the our media pipeline send the frame to imageBridge, then fire
+ // seeked event, the target frame may not be shown on the screen.
+ // So using canvas to access the target frame in the imageContainer in
+ // videoElement.
+ var canvas = document.getElementById("canvas");
+ canvas.width = video.videoWidth;
+ canvas.height = video.videoHeight;
+ var ctx = canvas.getContext("2d");
+ ctx.drawImage(video, 0, 0, video.videoWidth, video.videoHeight);
+ document.documentElement.removeAttribute('class');
+ });
+}
+window.addEventListener("MozReftestInvalidate", doTest);
+</script>
+</head>
+<body>
+<video id="v1" style="position:absolute; left:0; top:0"></video>
+<canvas id="canvas" style="position:absolute; left:0; top:0"></video>
+</body>
+</html>
diff --git a/dom/media/test/reftest/image-10bits-rendering-720-90-ref.html b/dom/media/test/reftest/image-10bits-rendering-720-90-ref.html
new file mode 100644
index 0000000000..5e9a8e9b4c
--- /dev/null
+++ b/dom/media/test/reftest/image-10bits-rendering-720-90-ref.html
@@ -0,0 +1,4 @@
+<!DOCTYPE HTML>
+<img style="position:absolute; left:0; top:0; filter:hue-rotate(90deg);"
+ src="vp9hdr2020.png"
+>
diff --git a/dom/media/test/reftest/image-10bits-rendering-720-90-video.html b/dom/media/test/reftest/image-10bits-rendering-720-90-video.html
new file mode 100644
index 0000000000..a9ba8a9f2e
--- /dev/null
+++ b/dom/media/test/reftest/image-10bits-rendering-720-90-video.html
@@ -0,0 +1,22 @@
+<!DOCTYPE HTML>
+<html class="reftest-wait">
+<head>
+<script type="text/javascript">
+function doTest() {
+ var video = document.getElementById("v1");
+ video.src = "vp9hdr2020.webm";
+ video.preload = "metadata";
+ video.addEventListener("loadeddata", function() {
+ document.documentElement.removeAttribute('class');
+ });
+}
+window.addEventListener("MozReftestInvalidate", doTest);
+</script>
+</head>
+<body>
+<video id="v1" style="position:absolute; left:0; top:0; filter:hue-rotate(90deg);"></video>
+<script>
+//doTest();
+</script>
+</body>
+</html>
diff --git a/dom/media/test/reftest/image-10bits-rendering-720-ref.html b/dom/media/test/reftest/image-10bits-rendering-720-ref.html
new file mode 100644
index 0000000000..1ae393031a
--- /dev/null
+++ b/dom/media/test/reftest/image-10bits-rendering-720-ref.html
@@ -0,0 +1,4 @@
+<!DOCTYPE HTML>
+<img style="position:absolute; left:0; top:0"
+ src="vp9hdr2020.png"
+>
diff --git a/dom/media/test/reftest/image-10bits-rendering-720-video.html b/dom/media/test/reftest/image-10bits-rendering-720-video.html
new file mode 100644
index 0000000000..93d2651ffc
--- /dev/null
+++ b/dom/media/test/reftest/image-10bits-rendering-720-video.html
@@ -0,0 +1,19 @@
+<!DOCTYPE HTML>
+<html class="reftest-wait">
+<head>
+<script type="text/javascript">
+function doTest() {
+ var video = document.getElementById("v1");
+ video.src = "vp9hdr2020.webm";
+ video.preload = "metadata";
+ video.addEventListener("loadeddata", function() {
+ document.documentElement.removeAttribute('class');
+ });
+}
+window.addEventListener("MozReftestInvalidate", doTest);
+</script>
+</head>
+<body>
+<video id="v1" style="position:absolute; left:0; top:0"></video>
+</body>
+</html>
diff --git a/dom/media/test/reftest/image-10bits-rendering-720.video.html b/dom/media/test/reftest/image-10bits-rendering-720.video.html
new file mode 100644
index 0000000000..93d2651ffc
--- /dev/null
+++ b/dom/media/test/reftest/image-10bits-rendering-720.video.html
@@ -0,0 +1,19 @@
+<!DOCTYPE HTML>
+<html class="reftest-wait">
+<head>
+<script type="text/javascript">
+function doTest() {
+ var video = document.getElementById("v1");
+ video.src = "vp9hdr2020.webm";
+ video.preload = "metadata";
+ video.addEventListener("loadeddata", function() {
+ document.documentElement.removeAttribute('class');
+ });
+}
+window.addEventListener("MozReftestInvalidate", doTest);
+</script>
+</head>
+<body>
+<video id="v1" style="position:absolute; left:0; top:0"></video>
+</body>
+</html>
diff --git a/dom/media/test/reftest/image-10bits-rendering-90-ref.html b/dom/media/test/reftest/image-10bits-rendering-90-ref.html
new file mode 100644
index 0000000000..38f032f0fd
--- /dev/null
+++ b/dom/media/test/reftest/image-10bits-rendering-90-ref.html
@@ -0,0 +1,4 @@
+<!DOCTYPE HTML>
+<img style="position:absolute; left:0; top:0; filter:hue-rotate(90deg);"
+ src="av1hdr2020.png"
+>
diff --git a/dom/media/test/reftest/image-10bits-rendering-90-video.html b/dom/media/test/reftest/image-10bits-rendering-90-video.html
new file mode 100644
index 0000000000..94edcaefd8
--- /dev/null
+++ b/dom/media/test/reftest/image-10bits-rendering-90-video.html
@@ -0,0 +1,22 @@
+<!DOCTYPE HTML>
+<html class="reftest-wait">
+<head>
+<script type="text/javascript">
+function doTest() {
+ var video = document.getElementById("v1");
+ video.src = "av1hdr2020.mp4";
+ video.preload = "metadata";
+ video.addEventListener("loadeddata", function() {
+ document.documentElement.removeAttribute('class');
+ });
+}
+window.addEventListener("MozReftestInvalidate", doTest);
+</script>
+</head>
+<body>
+<video id="v1" style="position:absolute; left:0; top:0; filter:hue-rotate(90deg);"></video>
+<script>
+//doTest();
+</script>
+</body>
+</html>
diff --git a/dom/media/test/reftest/image-10bits-rendering-ref.html b/dom/media/test/reftest/image-10bits-rendering-ref.html
new file mode 100644
index 0000000000..de7fbbfc95
--- /dev/null
+++ b/dom/media/test/reftest/image-10bits-rendering-ref.html
@@ -0,0 +1,4 @@
+<!DOCTYPE HTML>
+<img style="position:absolute; left:0; top:0"
+ src="av1hdr2020.png"
+>
diff --git a/dom/media/test/reftest/image-10bits-rendering-video.html b/dom/media/test/reftest/image-10bits-rendering-video.html
new file mode 100644
index 0000000000..faaf9c29ef
--- /dev/null
+++ b/dom/media/test/reftest/image-10bits-rendering-video.html
@@ -0,0 +1,22 @@
+<!DOCTYPE HTML>
+<html class="reftest-wait">
+<head>
+<script type="text/javascript">
+function doTest() {
+ var video = document.getElementById("v1");
+ video.src = "av1hdr2020.mp4";
+ video.preload = "metadata";
+ video.addEventListener("loadeddata", function() {
+ document.documentElement.removeAttribute('class');
+ });
+}
+//window.addEventListener("MozReftestInvalidate", doTest);
+</script>
+</head>
+<body>
+<video id="v1" style="position:absolute; left:0; top:0"></video>
+<script>
+doTest();
+</script>
+</body>
+</html>
diff --git a/dom/media/test/reftest/incorrect_display_in_bytestream_vp8-ref.html b/dom/media/test/reftest/incorrect_display_in_bytestream_vp8-ref.html
new file mode 100644
index 0000000000..b08a8d365b
--- /dev/null
+++ b/dom/media/test/reftest/incorrect_display_in_bytestream_vp8-ref.html
@@ -0,0 +1,13 @@
+<!DOCTYPE HTML>
+<head>
+</head>
+<body>
+<!--
+ This ref image is generated by the first frame of the testing video in webm_aspect_ratio_vp8.html
+-->
+<image style="position:absolute; left:0; top:0" src="
+"></video>
+<script type="text/javascript">
+</script>
+</body>
+</html>
diff --git a/dom/media/test/reftest/incorrect_display_in_bytestream_vp8.html b/dom/media/test/reftest/incorrect_display_in_bytestream_vp8.html
new file mode 100644
index 0000000000..271343cfe2
--- /dev/null
+++ b/dom/media/test/reftest/incorrect_display_in_bytestream_vp8.html
@@ -0,0 +1,33 @@
+<!DOCTYPE HTML>
+<html class="reftest-wait">
+<head>
+</head>
+<body>
+<video id="v" style="position:absolute; left:0; top:0"></video>
+<canvas id="canvas" style="position:absolute; left:0; top:0"></video>
+<script type="text/javascript">
+/**
+ * The display information in the VP8 byte stream is different from the display
+ * information in the container, and it's the incorrect one, which doesn't honer
+ * the original DAR. This test is used to check whether we will display the
+ * video frame in correct DAR.
+ */
+async function testDisplayRatioForVP8() {
+ const video = document.getElementById("v");
+ video.src = "incorrect_display_in_bytestream_vp8.webm";
+ await new Promise(r => video.oncanplay = r);
+ // Since our media pipeline sends the frame to imageBridge, the target frame
+ // may not be shown on the screen yet. So using canvas to access the target
+ // frame in the imageContainer in videoElement.
+ const canvas = document.getElementById("canvas");
+ canvas.width = video.videoWidth;
+ canvas.height = video.videoHeight;
+ const ctx = canvas.getContext("2d");
+ ctx.drawImage(video, 0, 0, video.videoWidth, video.videoHeight);
+ document.documentElement.removeAttribute('class');
+};
+
+window.addEventListener("MozReftestInvalidate", testDisplayRatioForVP8);
+</script>
+</body>
+</html>
diff --git a/dom/media/test/reftest/incorrect_display_in_bytestream_vp8.webm b/dom/media/test/reftest/incorrect_display_in_bytestream_vp8.webm
new file mode 100644
index 0000000000..e4143b7f38
--- /dev/null
+++ b/dom/media/test/reftest/incorrect_display_in_bytestream_vp8.webm
Binary files differ
diff --git a/dom/media/test/reftest/incorrect_display_in_bytestream_vp9-ref.html b/dom/media/test/reftest/incorrect_display_in_bytestream_vp9-ref.html
new file mode 100644
index 0000000000..6517c77f63
--- /dev/null
+++ b/dom/media/test/reftest/incorrect_display_in_bytestream_vp9-ref.html
@@ -0,0 +1,12 @@
+<!DOCTYPE HTML>
+<head>
+</head>
+<body>
+<!--
+ This ref image is generated by the first frame of the testing video in webm_aspect_ratio_vp8.html
+-->
+<image style="position:absolute; left:0; top:0" src=""></video>
+<script type="text/javascript">
+</script>
+</body>
+</html>
diff --git a/dom/media/test/reftest/incorrect_display_in_bytestream_vp9.html b/dom/media/test/reftest/incorrect_display_in_bytestream_vp9.html
new file mode 100644
index 0000000000..7d0e9404d5
--- /dev/null
+++ b/dom/media/test/reftest/incorrect_display_in_bytestream_vp9.html
@@ -0,0 +1,33 @@
+<!DOCTYPE HTML>
+<html class="reftest-wait">
+<head>
+</head>
+<body>
+<video id="v" style="position:absolute; left:0; top:0"></video>
+<canvas id="canvas" style="position:absolute; left:0; top:0"></video>
+<script type="text/javascript">
+/**
+ * The display information in the VP9 byte stream is different from the display
+ * information in the container, and it's the incorrect one, which doesn't honer
+ * the original DAR. This test is used to check whether we will display the
+ * video frame in correct DAR.
+ */
+async function testDisplayRatioForVP9() {
+ const video = document.getElementById("v");
+ video.src = "incorrect_display_in_bytestream_vp9.webm";
+ await new Promise(r => video.oncanplay = r);
+ // Since our media pipeline sends the frame to imageBridge, the target frame
+ // may not be shown on the screen yet. So using canvas to access the target
+ // frame in the imageContainer in videoElement.
+ const canvas = document.getElementById("canvas");
+ canvas.width = video.videoWidth;
+ canvas.height = video.videoHeight;
+ const ctx = canvas.getContext("2d");
+ ctx.drawImage(video, 0, 0, video.videoWidth, video.videoHeight);
+ document.documentElement.removeAttribute('class');
+};
+
+window.addEventListener("MozReftestInvalidate", testDisplayRatioForVP9);
+</script>
+</body>
+</html>
diff --git a/dom/media/test/reftest/incorrect_display_in_bytestream_vp9.webm b/dom/media/test/reftest/incorrect_display_in_bytestream_vp9.webm
new file mode 100644
index 0000000000..499e3acd94
--- /dev/null
+++ b/dom/media/test/reftest/incorrect_display_in_bytestream_vp9.webm
Binary files differ
diff --git a/dom/media/test/reftest/reftest.list b/dom/media/test/reftest/reftest.list
new file mode 100644
index 0000000000..6bbd7b9a53
--- /dev/null
+++ b/dom/media/test/reftest/reftest.list
@@ -0,0 +1,15 @@
+skip-if(Android) fuzzy-if(OSX,0-80,0-76800) fuzzy-if(appleSilicon,0-80,0-76800) fuzzy-if(winWidget,0-63,0-76799) fuzzy-if(gtkWidget&&layersGPUAccelerated,0-70,0-2032) HTTP(..) == short.mp4.firstframe.html short.mp4.firstframe-ref.html
+skip-if(Android) fuzzy-if(OSX,0-87,0-76797) fuzzy-if(appleSilicon,0-87,0-76797) fuzzy-if(winWidget,0-60,0-76797) fuzzy-if(gtkWidget&&layersGPUAccelerated,0-60,0-6070) HTTP(..) == short.mp4.lastframe.html short.mp4.lastframe-ref.html
+skip-if(Android) skip-if(cocoaWidget) skip-if(winWidget) fuzzy-if(gtkWidget&&layersGPUAccelerated,0-57,0-4282) fuzzy-if(OSX,55-80,4173-4417) HTTP(..) == bipbop_300_215kbps.mp4.lastframe.html bipbop_300_215kbps.mp4.lastframe-ref.html
+skip-if(Android) fuzzy-if(OSX,0-25,0-175921) fuzzy-if(appleSilicon,34-34,40100-40100) fuzzy-if(winWidget,0-71,0-179198) HTTP(..) == gizmo.mp4.seek.html gizmo.mp4.55thframe-ref.html
+# Bug 1758718
+skip-if(Android) skip-if(MinGW) skip-if(OSX) fuzzy(0-10,0-778236) == image-10bits-rendering-video.html image-10bits-rendering-ref.html
+skip-if(Android) skip-if(MinGW) fuzzy(0-10,0-778536) fuzzy-if(appleSilicon,0-37,0-699614) == image-10bits-rendering-90-video.html image-10bits-rendering-90-ref.html
+# Bug 1758718
+skip-if(Android) fuzzy(0-27,0-573106) skip-if(OSX) == image-10bits-rendering-720-video.html image-10bits-rendering-720-ref.html
+skip-if(Android) fuzzy(0-31,0-573249) fuzzy-if(appleSilicon,0-37,0-543189) == image-10bits-rendering-720-90-video.html image-10bits-rendering-720-90-ref.html
+skip-if(Android) fuzzy(0-84,0-771156) fails-if(useDrawSnapshot) == uneven_frame_duration_video.html uneven_frame_duration_video-ref.html # Skip on Windows 7 as the resolution of the video is too high for test machines and will fail in the decoder.
+# Set media.dormant-on-pause-timeout-ms to avoid decoders becoming dormant and busting test, skip on android as test is too noisy and unstable
+skip-if(Android) pref(media.dormant-on-pause-timeout-ms,-1) fuzzy(0-20,0-500) == frame_order_mp4.html frame_order_mp4-ref.html
+skip-if(Android) fuzzy(0-30,0-270000) == incorrect_display_in_bytestream_vp8.html incorrect_display_in_bytestream_vp8-ref.html
+skip-if(Android) fuzzy(0-22,0-377335) == incorrect_display_in_bytestream_vp9.html incorrect_display_in_bytestream_vp9-ref.html
diff --git a/dom/media/test/reftest/reftest_img.html b/dom/media/test/reftest/reftest_img.html
new file mode 100644
index 0000000000..3e55c264ec
--- /dev/null
+++ b/dom/media/test/reftest/reftest_img.html
@@ -0,0 +1,20 @@
+<!DOCTYPE HTML>
+<html class="reftest-wait">
+<head>
+<meta charset='utf-8'>
+</head>
+<body>
+<img id="e_img" style="position:absolute; left:0; top:0; max-width:100%">
+<script>
+(async () => {
+ const params = new URLSearchParams(window.location.search);
+ const src = params.get('src');
+ src.defined;
+
+ e_img.src = src;
+ await e_img.decode()
+ document.documentElement.removeAttribute('class');
+})();
+</script>
+</body>
+</html>
diff --git a/dom/media/test/reftest/reftest_video.html b/dom/media/test/reftest/reftest_video.html
new file mode 100644
index 0000000000..cf9223e7ae
--- /dev/null
+++ b/dom/media/test/reftest/reftest_video.html
@@ -0,0 +1,64 @@
+<!DOCTYPE HTML>
+<html class="reftest-wait">
+<head>
+<meta charset='utf-8'>
+</head>
+<body>
+<video id="e_video" style="position:absolute; left:0; top:0; max-width:100%">
+<script>
+
+const TIMEOUT_MS = 2000;
+
+// -
+
+function sleepPromise(ms) {
+ return new Promise(go => {
+ setTimeout(go, ms);
+ });
+}
+
+(async () => {
+ await sleepPromise(TIMEOUT_MS);
+ if (!document.documentElement.hasAttribute('class')) return;
+
+ const div = document.body.appendChild(document.createElement('div'));
+ div.textContent = `Timed out after ${TIMEOUT_MS}ms`;
+ console.log(div.textContent);
+
+ document.documentElement.removeAttribute('class');
+})();
+
+// -
+// Test
+
+(async () => {
+ const params = new URLSearchParams(window.location.search);
+ const src = params.get('src');
+ src.defined;
+ if (src == 'none') {
+ console.log('Show blank.');
+ document.documentElement.removeAttribute('class');
+ return;
+ }
+ if (src == 'timeout') {
+ console.log('Deliberate timeout.');
+ return;
+ }
+
+ e_video.src = src;
+ e_video.muted = true;
+ const p = e_video.play();
+ p.defined;
+ try {
+ await p;
+ console.log('e_video.play() accepted');
+ } catch (e) {
+ const div = document.body.appendChild(document.createElement('div'));
+ div.textContent = `Error: ${JSON.stringify(e)}`;
+ console.log(div.textContent);
+ }
+ document.documentElement.removeAttribute('class');
+})();
+</script>
+</body>
+</html>
diff --git a/dom/media/test/reftest/short.mp4.firstframe-ref.html b/dom/media/test/reftest/short.mp4.firstframe-ref.html
new file mode 100644
index 0000000000..d80f2f985f
--- /dev/null
+++ b/dom/media/test/reftest/short.mp4.firstframe-ref.html
@@ -0,0 +1,4 @@
+<!DOCTYPE HTML>
+<img style="position:absolute; left:0; top:0"
+ src=""
+>
diff --git a/dom/media/test/reftest/short.mp4.firstframe.html b/dom/media/test/reftest/short.mp4.firstframe.html
new file mode 100644
index 0000000000..759bdc5ede
--- /dev/null
+++ b/dom/media/test/reftest/short.mp4.firstframe.html
@@ -0,0 +1,19 @@
+<!DOCTYPE HTML>
+<html class="reftest-wait">
+<head>
+<script type="text/javascript">
+function doTest() {
+ var video = document.getElementById("v1");
+ video.src = "../short.mp4";
+ video.preload = "metadata";
+ video.addEventListener("loadeddata", function() {
+ document.documentElement.removeAttribute('class');
+ });
+}
+window.addEventListener("MozReftestInvalidate", doTest);
+</script>
+</head>
+<body>
+<video id="v1" style="position:absolute; left:0; top:0"></video>
+</body>
+</html>
diff --git a/dom/media/test/reftest/short.mp4.lastframe-ref.html b/dom/media/test/reftest/short.mp4.lastframe-ref.html
new file mode 100644
index 0000000000..7474b9039e
--- /dev/null
+++ b/dom/media/test/reftest/short.mp4.lastframe-ref.html
@@ -0,0 +1,4 @@
+<!DOCTYPE HTML>
+<img style="position:absolute; left:0; top:0"
+src=""
+>
diff --git a/dom/media/test/reftest/short.mp4.lastframe.html b/dom/media/test/reftest/short.mp4.lastframe.html
new file mode 100644
index 0000000000..abd27c5c8e
--- /dev/null
+++ b/dom/media/test/reftest/short.mp4.lastframe.html
@@ -0,0 +1,42 @@
+<!DOCTYPE HTML>
+<html class="reftest-wait">
+<head>
+<script type="text/javascript">
+function doTest() {
+ var video = document.getElementById("v1");
+ video.src = "../short.mp4";
+ video.preload = "metadata";
+ video.seenEnded = false;
+ // Seek to the end
+ video.addEventListener("loadeddata", function() {
+ video.currentTime = video.duration;
+ video.onseeked = () => {
+ video.onseeked = null;
+ callSeekToNextFrame();
+ };
+ });
+
+ function callSeekToNextFrame() {
+ video.seekToNextFrame().then(
+ () => {
+ if (!video.seenEnded)
+ callSeekToNextFrame();
+ },
+ () => {
+ // Reach the end, do nothing.
+ }
+ );
+ }
+
+ video.addEventListener("ended", function() {
+ video.seenEnded = true;
+ document.documentElement.removeAttribute('class');
+ });
+}
+window.addEventListener("MozReftestInvalidate", doTest);
+</script>
+</head>
+<body>
+<video id="v1" style="position:absolute; left:0; top:0"></video>
+</body>
+</html>
diff --git a/dom/media/test/reftest/uneven_frame_duration_video-ref.html b/dom/media/test/reftest/uneven_frame_duration_video-ref.html
new file mode 100644
index 0000000000..6f1d73cc71
--- /dev/null
+++ b/dom/media/test/reftest/uneven_frame_duration_video-ref.html
@@ -0,0 +1,7 @@
+<!DOCTYPE HTML>
+<img style="position:absolute; left:0; top:0"
+src="uneven_frame_durations_3.8s_frame.png"
+>
+<!-- Test frame created via
+`ffmpeg.exe -i uneven_frame_durations.mp4 -ss 3.8 -frames 1 uneven_frame_durations_3.8s_frame.png`
+-->
diff --git a/dom/media/test/reftest/uneven_frame_duration_video.html b/dom/media/test/reftest/uneven_frame_duration_video.html
new file mode 100644
index 0000000000..73ca34b8bd
--- /dev/null
+++ b/dom/media/test/reftest/uneven_frame_duration_video.html
@@ -0,0 +1,39 @@
+<!DOCTYPE HTML>
+<html class="reftest-wait">
+<!--Tests the fix for bug 1576990. We want to make sure the frames do not
+ become stuck due to decoders (in particular the Windows decoder) estimating
+ frame durations incorrectly.
+-->
+<head>
+<script type="text/javascript">
+function doTest() {
+ var video = document.getElementById("v1");
+ // Video from bug 1576990.
+ video.src = "uneven_frame_durations.mp4";
+ video.preload = "metadata";
+
+ // This frame differs depending on if the Windows' decoder is estimating
+ // frame duration or if we use Gecko durations.
+ video.currentTime = 3.8;
+
+ video.addEventListener("seeked", function() {
+ // Since our media pipeline sends the frame to imageBridge, then fires
+ // a seeked event, the target frame may not be shown on the screen.
+ // So using canvas to access the target frame in the imageContainer in
+ // videoElement.
+ var canvas = document.getElementById("canvas");
+ canvas.width = video.videoWidth;
+ canvas.height = video.videoHeight;
+ var ctx = canvas.getContext("2d");
+ ctx.drawImage(video, 0, 0, video.videoWidth, video.videoHeight);
+ document.documentElement.removeAttribute("class");
+ });
+}
+window.addEventListener("MozReftestInvalidate", doTest);
+</script>
+</head>
+<body>
+<video id="v1" style="position:absolute; left:0; top:0"></video>
+<canvas id="canvas" style="position:absolute; left:0; top:0"></video>
+</body>
+</html>
diff --git a/dom/media/test/reftest/uneven_frame_durations.mp4 b/dom/media/test/reftest/uneven_frame_durations.mp4
new file mode 100644
index 0000000000..a5a0c16a0f
--- /dev/null
+++ b/dom/media/test/reftest/uneven_frame_durations.mp4
Binary files differ
diff --git a/dom/media/test/reftest/uneven_frame_durations_3.8s_frame.png b/dom/media/test/reftest/uneven_frame_durations_3.8s_frame.png
new file mode 100644
index 0000000000..889174e115
--- /dev/null
+++ b/dom/media/test/reftest/uneven_frame_durations_3.8s_frame.png
Binary files differ
diff --git a/dom/media/test/reftest/vp9hdr2020.png b/dom/media/test/reftest/vp9hdr2020.png
new file mode 100644
index 0000000000..afb68d9e0a
--- /dev/null
+++ b/dom/media/test/reftest/vp9hdr2020.png
Binary files differ
diff --git a/dom/media/test/reftest/vp9hdr2020.webm b/dom/media/test/reftest/vp9hdr2020.webm
new file mode 100644
index 0000000000..516f62093a
--- /dev/null
+++ b/dom/media/test/reftest/vp9hdr2020.webm
Binary files differ
diff --git a/dom/media/test/resolution-change.webm b/dom/media/test/resolution-change.webm
new file mode 100644
index 0000000000..29aad93b96
--- /dev/null
+++ b/dom/media/test/resolution-change.webm
Binary files differ
diff --git a/dom/media/test/resolution-change.webm^headers^ b/dom/media/test/resolution-change.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/resolution-change.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/sample-encrypted-sgpdstbl-sbgptraf.mp4 b/dom/media/test/sample-encrypted-sgpdstbl-sbgptraf.mp4
new file mode 100644
index 0000000000..720339bdc2
--- /dev/null
+++ b/dom/media/test/sample-encrypted-sgpdstbl-sbgptraf.mp4
Binary files differ
diff --git a/dom/media/test/sample-encrypted-sgpdstbl-sbgptraf.mp4^headers^ b/dom/media/test/sample-encrypted-sgpdstbl-sbgptraf.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/sample-encrypted-sgpdstbl-sbgptraf.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/sample-fisbone-skeleton4.ogv b/dom/media/test/sample-fisbone-skeleton4.ogv
new file mode 100644
index 0000000000..8afe0be7a4
--- /dev/null
+++ b/dom/media/test/sample-fisbone-skeleton4.ogv
Binary files differ
diff --git a/dom/media/test/sample-fisbone-skeleton4.ogv^headers^ b/dom/media/test/sample-fisbone-skeleton4.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/sample-fisbone-skeleton4.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/sample-fisbone-wrong-header.ogv b/dom/media/test/sample-fisbone-wrong-header.ogv
new file mode 100644
index 0000000000..46c3933da5
--- /dev/null
+++ b/dom/media/test/sample-fisbone-wrong-header.ogv
Binary files differ
diff --git a/dom/media/test/sample-fisbone-wrong-header.ogv^headers^ b/dom/media/test/sample-fisbone-wrong-header.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/sample-fisbone-wrong-header.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/sample.3g2 b/dom/media/test/sample.3g2
new file mode 100644
index 0000000000..769cb01dbd
--- /dev/null
+++ b/dom/media/test/sample.3g2
Binary files differ
diff --git a/dom/media/test/sample.3gp b/dom/media/test/sample.3gp
new file mode 100644
index 0000000000..4a3d8ea66f
--- /dev/null
+++ b/dom/media/test/sample.3gp
Binary files differ
diff --git a/dom/media/test/seek-short.ogv b/dom/media/test/seek-short.ogv
new file mode 100644
index 0000000000..a5ca6951d0
--- /dev/null
+++ b/dom/media/test/seek-short.ogv
Binary files differ
diff --git a/dom/media/test/seek-short.ogv^headers^ b/dom/media/test/seek-short.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/seek-short.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/seek-short.webm b/dom/media/test/seek-short.webm
new file mode 100644
index 0000000000..36abd1570e
--- /dev/null
+++ b/dom/media/test/seek-short.webm
Binary files differ
diff --git a/dom/media/test/seek-short.webm^headers^ b/dom/media/test/seek-short.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/seek-short.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/seek.ogv b/dom/media/test/seek.ogv
new file mode 100644
index 0000000000..ac7ece3519
--- /dev/null
+++ b/dom/media/test/seek.ogv
Binary files differ
diff --git a/dom/media/test/seek.ogv^headers^ b/dom/media/test/seek.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/seek.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/seek.webm b/dom/media/test/seek.webm
new file mode 100644
index 0000000000..72b0297233
--- /dev/null
+++ b/dom/media/test/seek.webm
Binary files differ
diff --git a/dom/media/test/seek.webm^headers^ b/dom/media/test/seek.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/seek.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/seekLies.sjs b/dom/media/test/seekLies.sjs
new file mode 100644
index 0000000000..4fc528a0a5
--- /dev/null
+++ b/dom/media/test/seekLies.sjs
@@ -0,0 +1,22 @@
+function handleRequest(request, response) {
+ var file = Services.dirsvc.get("CurWorkD", Ci.nsIFile);
+ var fis = Cc["@mozilla.org/network/file-input-stream;1"].createInstance(
+ Ci.nsIFileInputStream
+ );
+ var bis = Cc["@mozilla.org/binaryinputstream;1"].createInstance(
+ Ci.nsIBinaryInputStream
+ );
+ var paths = "tests/dom/media/test/seek.ogv";
+ var split = paths.split("/");
+ for (var i = 0; i < split.length; ++i) {
+ file.append(split[i]);
+ }
+ fis.init(file, -1, -1, false);
+ bis.setInputStream(fis);
+ var bytes = bis.readBytes(bis.available());
+ response.setHeader("Content-Length", "" + bytes.length, false);
+ response.setHeader("Content-Type", "video/ogg", false);
+ response.setHeader("Accept-Ranges", "bytes", false);
+ response.write(bytes, bytes.length);
+ bis.close();
+}
diff --git a/dom/media/test/seek_support.js b/dom/media/test/seek_support.js
new file mode 100644
index 0000000000..ca6d563f1a
--- /dev/null
+++ b/dom/media/test/seek_support.js
@@ -0,0 +1,61 @@
+// This file expects manifest.js to be included in the same scope.
+/* import-globals-from manifest.js */
+// This file expects SEEK_TEST_NUMBER to be defined by the test.
+/* global SEEK_TEST_NUMBER */
+var manager = new MediaTestManager();
+
+function createTestArray() {
+ var tests = [];
+ var tmpVid = document.createElement("video");
+
+ for (var testNum = 0; testNum < gSeekTests.length; testNum++) {
+ var test = gSeekTests[testNum];
+ if (!tmpVid.canPlayType(test.type)) {
+ continue;
+ }
+
+ var t = {};
+ t.name = test.name;
+ t.type = test.type;
+ t.duration = test.duration;
+ t.number = SEEK_TEST_NUMBER;
+ tests.push(t);
+ }
+ return tests;
+}
+
+function startTest(test, token) {
+ var video = document.createElement("video");
+ video.token = token += "-seek" + test.number + ".js";
+ manager.started(video.token);
+ video.src = test.name;
+ video.preload = "metadata";
+ document.body.appendChild(video);
+ var name = test.name + " seek test " + test.number;
+ var localIs = (function (n) {
+ return function (a, b, msg) {
+ is(a, b, n + ": " + msg);
+ };
+ })(name);
+ var localOk = (function (n) {
+ return function (a, msg) {
+ ok(a, n + ": " + msg);
+ };
+ })(name);
+ var localFinish = (function (v, m) {
+ return function () {
+ v.onerror = null;
+ removeNodeAndSource(v);
+ dump("SEEK-TEST: Finished " + name + " token: " + v.token + "\n");
+ m.finished(v.token);
+ };
+ })(video, manager);
+ dump("SEEK-TEST: Started " + name + "\n");
+ window["test_seek" + test.number](
+ video,
+ test.duration / 2,
+ localIs,
+ localOk,
+ localFinish
+ );
+}
diff --git a/dom/media/test/seek_with_sound.ogg b/dom/media/test/seek_with_sound.ogg
new file mode 100644
index 0000000000..c86d9946bd
--- /dev/null
+++ b/dom/media/test/seek_with_sound.ogg
Binary files differ
diff --git a/dom/media/test/seek_with_sound.ogg^headers^ b/dom/media/test/seek_with_sound.ogg^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/seek_with_sound.ogg^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/short-aac-encrypted-audio.mp4 b/dom/media/test/short-aac-encrypted-audio.mp4
new file mode 100644
index 0000000000..c1ba8d37e4
--- /dev/null
+++ b/dom/media/test/short-aac-encrypted-audio.mp4
Binary files differ
diff --git a/dom/media/test/short-aac-encrypted-audio.mp4^headers^ b/dom/media/test/short-aac-encrypted-audio.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/short-aac-encrypted-audio.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/short-audio-fragmented-cenc-without-pssh.mp4 b/dom/media/test/short-audio-fragmented-cenc-without-pssh.mp4
new file mode 100644
index 0000000000..bc58623999
--- /dev/null
+++ b/dom/media/test/short-audio-fragmented-cenc-without-pssh.mp4
Binary files differ
diff --git a/dom/media/test/short-audio-fragmented-cenc-without-pssh.mp4^headers^ b/dom/media/test/short-audio-fragmented-cenc-without-pssh.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/short-audio-fragmented-cenc-without-pssh.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/short-cenc-pssh-in-moof.mp4 b/dom/media/test/short-cenc-pssh-in-moof.mp4
new file mode 100644
index 0000000000..aa44c3b9a1
--- /dev/null
+++ b/dom/media/test/short-cenc-pssh-in-moof.mp4
Binary files differ
diff --git a/dom/media/test/short-cenc.mp4 b/dom/media/test/short-cenc.mp4
new file mode 100644
index 0000000000..aa44c3b9a1
--- /dev/null
+++ b/dom/media/test/short-cenc.mp4
Binary files differ
diff --git a/dom/media/test/short-cenc.xml b/dom/media/test/short-cenc.xml
new file mode 100644
index 0000000000..9658c3e32f
--- /dev/null
+++ b/dom/media/test/short-cenc.xml
@@ -0,0 +1,37 @@
+<?xml version="1.0" encoding="UTF-8"?>
+
+<!--
+ This XML file describes the encryption applied to short-cenc.mp4. To generate
+ short-cenc, run the following command:
+
+ MP4Box -crypt short-cenc.xml -out short-cenc.mp4 short.mp4
+-->
+
+<GPACDRM type="CENC AES-CTR">
+
+ <DRMInfo type="pssh" version="1">
+ <!--
+ SystemID specified in
+ https://dvcs.w3.org/hg/html-media/raw-file/tip/encrypted-media/cenc-format.html
+ -->
+ <BS ID128="1077efecc0b24d02ace33c1e52e2fb4b" />
+ <!-- Number of KeyIDs = 2 -->
+ <BS bits="32" value="2" />
+ <!-- KeyID -->
+ <BS ID128="0x7e571d017e571d017e571d017e571d01" />
+ <BS ID128="0x7e571d027e571d027e571d027e571d02" />
+ </DRMInfo>
+
+ <CrypTrack trackID="1" isEncrypted="1" IV_size="16" saiSavedBox="senc"
+ first_IV="0x00000000000000000000000000000000">
+ <key KID="0x7e571d017e571d017e571d017e571d01"
+ value="0x7e5711117e5711117e5711117e571111" />
+ </CrypTrack>
+
+ <CrypTrack trackID="2" isEncrypted="1" IV_size="16" saiSavedBox="senc"
+ first_IV="0x00000000000000000000000000000000">
+ <key KID="0x7e571d027e571d027e571d027e571d02"
+ value="0x7e5722227e5722227e5722227e572222" />
+ </CrypTrack>
+
+</GPACDRM>
diff --git a/dom/media/test/short-video.ogv b/dom/media/test/short-video.ogv
new file mode 100644
index 0000000000..68dee3cf2b
--- /dev/null
+++ b/dom/media/test/short-video.ogv
Binary files differ
diff --git a/dom/media/test/short-video.ogv^headers^ b/dom/media/test/short-video.ogv^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/short-video.ogv^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/short-vp9-encrypted-video.mp4 b/dom/media/test/short-vp9-encrypted-video.mp4
new file mode 100644
index 0000000000..23d1083d18
--- /dev/null
+++ b/dom/media/test/short-vp9-encrypted-video.mp4
Binary files differ
diff --git a/dom/media/test/short-vp9-encrypted-video.mp4^headers^ b/dom/media/test/short-vp9-encrypted-video.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/short-vp9-encrypted-video.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/short.mp4 b/dom/media/test/short.mp4
new file mode 100644
index 0000000000..802da047bc
--- /dev/null
+++ b/dom/media/test/short.mp4
Binary files differ
diff --git a/dom/media/test/short.mp4.gz b/dom/media/test/short.mp4.gz
new file mode 100644
index 0000000000..efb95e38e3
--- /dev/null
+++ b/dom/media/test/short.mp4.gz
Binary files differ
diff --git a/dom/media/test/short.mp4^headers^ b/dom/media/test/short.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/short.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/shorter_audio_than_video_3s.webm b/dom/media/test/shorter_audio_than_video_3s.webm
new file mode 100644
index 0000000000..5f8f49b4ac
--- /dev/null
+++ b/dom/media/test/shorter_audio_than_video_3s.webm
Binary files differ
diff --git a/dom/media/test/shorter_audio_than_video_3s.webm^headers^ b/dom/media/test/shorter_audio_than_video_3s.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/shorter_audio_than_video_3s.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/sin-441-1s-44100-afconvert.mp4 b/dom/media/test/sin-441-1s-44100-afconvert.mp4
new file mode 100644
index 0000000000..2e2763823d
--- /dev/null
+++ b/dom/media/test/sin-441-1s-44100-afconvert.mp4
Binary files differ
diff --git a/dom/media/test/sin-441-1s-44100-fdk_aac.mp4 b/dom/media/test/sin-441-1s-44100-fdk_aac.mp4
new file mode 100644
index 0000000000..34b54a1849
--- /dev/null
+++ b/dom/media/test/sin-441-1s-44100-fdk_aac.mp4
Binary files differ
diff --git a/dom/media/test/sin-441-1s-44100-lame.mp3 b/dom/media/test/sin-441-1s-44100-lame.mp3
new file mode 100644
index 0000000000..f0c0164c0a
--- /dev/null
+++ b/dom/media/test/sin-441-1s-44100-lame.mp3
Binary files differ
diff --git a/dom/media/test/sin-441-1s-44100.flac b/dom/media/test/sin-441-1s-44100.flac
new file mode 100644
index 0000000000..960044638c
--- /dev/null
+++ b/dom/media/test/sin-441-1s-44100.flac
Binary files differ
diff --git a/dom/media/test/sin-441-1s-44100.ogg b/dom/media/test/sin-441-1s-44100.ogg
new file mode 100644
index 0000000000..16c197b633
--- /dev/null
+++ b/dom/media/test/sin-441-1s-44100.ogg
Binary files differ
diff --git a/dom/media/test/sin-441-1s-44100.opus b/dom/media/test/sin-441-1s-44100.opus
new file mode 100644
index 0000000000..795034521d
--- /dev/null
+++ b/dom/media/test/sin-441-1s-44100.opus
Binary files differ
diff --git a/dom/media/test/sine.webm b/dom/media/test/sine.webm
new file mode 100644
index 0000000000..3913ffa874
--- /dev/null
+++ b/dom/media/test/sine.webm
Binary files differ
diff --git a/dom/media/test/sine.webm^headers^ b/dom/media/test/sine.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/sine.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/single-xing-header-no-content-length.mp3 b/dom/media/test/single-xing-header-no-content-length.mp3
new file mode 100644
index 0000000000..aef54e220a
--- /dev/null
+++ b/dom/media/test/single-xing-header-no-content-length.mp3
Binary files differ
diff --git a/dom/media/test/single-xing-header-no-content-length.mp3^headers^ b/dom/media/test/single-xing-header-no-content-length.mp3^headers^
new file mode 100644
index 0000000000..abfeb4ce28
--- /dev/null
+++ b/dom/media/test/single-xing-header-no-content-length.mp3^headers^
@@ -0,0 +1,3 @@
+HTTP 200 OK
+Content-Length: invalid
+Cache-Control: no-store
diff --git a/dom/media/test/sintel-short-clearkey-subsample-encrypted-audio.webm b/dom/media/test/sintel-short-clearkey-subsample-encrypted-audio.webm
new file mode 100644
index 0000000000..7497096ba5
--- /dev/null
+++ b/dom/media/test/sintel-short-clearkey-subsample-encrypted-audio.webm
Binary files differ
diff --git a/dom/media/test/sintel-short-clearkey-subsample-encrypted-audio.webm^headers^ b/dom/media/test/sintel-short-clearkey-subsample-encrypted-audio.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/sintel-short-clearkey-subsample-encrypted-audio.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/sintel-short-clearkey-subsample-encrypted-video.webm b/dom/media/test/sintel-short-clearkey-subsample-encrypted-video.webm
new file mode 100644
index 0000000000..0f7609439d
--- /dev/null
+++ b/dom/media/test/sintel-short-clearkey-subsample-encrypted-video.webm
Binary files differ
diff --git a/dom/media/test/sintel-short-clearkey-subsample-encrypted-video.webm^headers^ b/dom/media/test/sintel-short-clearkey-subsample-encrypted-video.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/sintel-short-clearkey-subsample-encrypted-video.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/small-shot-mp3.mp4 b/dom/media/test/small-shot-mp3.mp4
new file mode 100644
index 0000000000..61fe0ac719
--- /dev/null
+++ b/dom/media/test/small-shot-mp3.mp4
Binary files differ
diff --git a/dom/media/test/small-shot-mp3.mp4^headers^ b/dom/media/test/small-shot-mp3.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/small-shot-mp3.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/small-shot.flac b/dom/media/test/small-shot.flac
new file mode 100644
index 0000000000..0da7c9044e
--- /dev/null
+++ b/dom/media/test/small-shot.flac
Binary files differ
diff --git a/dom/media/test/small-shot.m4a b/dom/media/test/small-shot.m4a
new file mode 100644
index 0000000000..51a23c5b49
--- /dev/null
+++ b/dom/media/test/small-shot.m4a
Binary files differ
diff --git a/dom/media/test/small-shot.mp3 b/dom/media/test/small-shot.mp3
new file mode 100644
index 0000000000..f9397a5106
--- /dev/null
+++ b/dom/media/test/small-shot.mp3
Binary files differ
diff --git a/dom/media/test/small-shot.mp3^headers^ b/dom/media/test/small-shot.mp3^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/small-shot.mp3^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/small-shot.ogg b/dom/media/test/small-shot.ogg
new file mode 100644
index 0000000000..1a41623f81
--- /dev/null
+++ b/dom/media/test/small-shot.ogg
Binary files differ
diff --git a/dom/media/test/small-shot.ogg^headers^ b/dom/media/test/small-shot.ogg^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/small-shot.ogg^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/sound.ogg b/dom/media/test/sound.ogg
new file mode 100644
index 0000000000..edda4e9128
--- /dev/null
+++ b/dom/media/test/sound.ogg
Binary files differ
diff --git a/dom/media/test/sound.ogg^headers^ b/dom/media/test/sound.ogg^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/sound.ogg^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/spacestorm-1000Hz-100ms.ogg b/dom/media/test/spacestorm-1000Hz-100ms.ogg
new file mode 100644
index 0000000000..994041e1b0
--- /dev/null
+++ b/dom/media/test/spacestorm-1000Hz-100ms.ogg
Binary files differ
diff --git a/dom/media/test/spacestorm-1000Hz-100ms.ogg^headers^ b/dom/media/test/spacestorm-1000Hz-100ms.ogg^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/spacestorm-1000Hz-100ms.ogg^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/split.webm b/dom/media/test/split.webm
new file mode 100644
index 0000000000..9207017fb6
--- /dev/null
+++ b/dom/media/test/split.webm
Binary files differ
diff --git a/dom/media/test/split.webm^headers^ b/dom/media/test/split.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/split.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/street.mp4 b/dom/media/test/street.mp4
new file mode 100644
index 0000000000..837d23b383
--- /dev/null
+++ b/dom/media/test/street.mp4
Binary files differ
diff --git a/dom/media/test/street.mp4^headers^ b/dom/media/test/street.mp4^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/street.mp4^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/sync.webm b/dom/media/test/sync.webm
new file mode 100644
index 0000000000..91ce6320b5
--- /dev/null
+++ b/dom/media/test/sync.webm
Binary files differ
diff --git a/dom/media/test/test-1-mono.opus b/dom/media/test/test-1-mono.opus
new file mode 100644
index 0000000000..d5198e9ceb
--- /dev/null
+++ b/dom/media/test/test-1-mono.opus
Binary files differ
diff --git a/dom/media/test/test-1-mono.opus^headers^ b/dom/media/test/test-1-mono.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/test-1-mono.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/test-2-stereo.opus b/dom/media/test/test-2-stereo.opus
new file mode 100644
index 0000000000..7115cac243
--- /dev/null
+++ b/dom/media/test/test-2-stereo.opus
Binary files differ
diff --git a/dom/media/test/test-2-stereo.opus^headers^ b/dom/media/test/test-2-stereo.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/test-2-stereo.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/test-3-LCR.opus b/dom/media/test/test-3-LCR.opus
new file mode 100644
index 0000000000..145536f3e7
--- /dev/null
+++ b/dom/media/test/test-3-LCR.opus
Binary files differ
diff --git a/dom/media/test/test-3-LCR.opus^headers^ b/dom/media/test/test-3-LCR.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/test-3-LCR.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/test-4-quad.opus b/dom/media/test/test-4-quad.opus
new file mode 100644
index 0000000000..731b867b2e
--- /dev/null
+++ b/dom/media/test/test-4-quad.opus
Binary files differ
diff --git a/dom/media/test/test-4-quad.opus^headers^ b/dom/media/test/test-4-quad.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/test-4-quad.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/test-5-5.0.opus b/dom/media/test/test-5-5.0.opus
new file mode 100644
index 0000000000..7eb2faa812
--- /dev/null
+++ b/dom/media/test/test-5-5.0.opus
Binary files differ
diff --git a/dom/media/test/test-5-5.0.opus^headers^ b/dom/media/test/test-5-5.0.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/test-5-5.0.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/test-6-5.1.opus b/dom/media/test/test-6-5.1.opus
new file mode 100644
index 0000000000..526515eb7b
--- /dev/null
+++ b/dom/media/test/test-6-5.1.opus
Binary files differ
diff --git a/dom/media/test/test-6-5.1.opus^headers^ b/dom/media/test/test-6-5.1.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/test-6-5.1.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/test-7-6.1.opus b/dom/media/test/test-7-6.1.opus
new file mode 100644
index 0000000000..8b6a7ce329
--- /dev/null
+++ b/dom/media/test/test-7-6.1.opus
Binary files differ
diff --git a/dom/media/test/test-7-6.1.opus^headers^ b/dom/media/test/test-7-6.1.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/test-7-6.1.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/test-8-7.1.opus b/dom/media/test/test-8-7.1.opus
new file mode 100644
index 0000000000..e56176a30f
--- /dev/null
+++ b/dom/media/test/test-8-7.1.opus
Binary files differ
diff --git a/dom/media/test/test-8-7.1.opus^headers^ b/dom/media/test/test-8-7.1.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/test-8-7.1.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/test-stereo-phase-inversion-180.opus b/dom/media/test/test-stereo-phase-inversion-180.opus
new file mode 100644
index 0000000000..ce70290002
--- /dev/null
+++ b/dom/media/test/test-stereo-phase-inversion-180.opus
Binary files differ
diff --git a/dom/media/test/test-stereo-phase-inversion-180.opus^headers^ b/dom/media/test/test-stereo-phase-inversion-180.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/test-stereo-phase-inversion-180.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/test_VideoPlaybackQuality.html b/dom/media/test/test_VideoPlaybackQuality.html
new file mode 100644
index 0000000000..67636f9ea6
--- /dev/null
+++ b/dom/media/test/test_VideoPlaybackQuality.html
@@ -0,0 +1,61 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test basic functionality of VideoPlaybackQuality</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+SimpleTest.waitForExplicitFinish();
+
+function test() {
+ var video = document.createElement("video");
+ ok(video.getVideoPlaybackQuality, "getVideoPlaybackQuality should be exposed with pref set");
+
+ var vpq = video.getVideoPlaybackQuality();
+ ok(vpq, "getVideoPlaybackQuality should return an object");
+ ok(vpq.creationTime <= performance.now(), "creationTime should be in the past");
+ is(vpq.totalVideoFrames, 0, "totalVideoFrames should be 0");
+ is(vpq.droppedVideoFrames, 0, "droppedVideoFrames should be 0");
+
+ var vpq2 = video.getVideoPlaybackQuality();
+ ok(vpq !== vpq2, "getVideoPlaybackQuality should return a new object");
+ ok(vpq.creationTime <= vpq2.creationTime, "VideoPlaybackQuality objects should have increasing creationTime");
+
+ var audio = document.createElement("audio");
+ ok(!audio.getVideoPlaybackQuality, "getVideoPlaybackQuality should not be available on Audio elements");
+
+ video.src = "seek.webm";
+ video.play();
+ video.addEventListener("ended", function () {
+ vpq = video.getVideoPlaybackQuality();
+ ok(vpq.creationTime <= performance.now(), "creationTime should be in the past");
+ ok(vpq.totalVideoFrames > 0, "totalVideoFrames should be > 0");
+ ok(vpq.droppedVideoFrames >= 0, "droppedVideoFrames should be >= 0");
+ ok(vpq.droppedVideoFrames <= vpq.totalVideoFrames, "droppedVideoFrames should be <= totalVideoFrames");
+
+ SpecialPowers.pushPrefEnv({"set": [["media.video_stats.enabled", false]]}, function () {
+ vpq = video.getVideoPlaybackQuality();
+ is(vpq.creationTime, 0, "creationTime should be 0");
+ is(vpq.totalVideoFrames, 0, "totalVideoFrames should be 0");
+ is(vpq.droppedVideoFrames, 0, "droppedVideoFrames should be 0");
+
+ SimpleTest.finish();
+ });
+ });
+}
+
+addLoadEvent(function() {
+ SpecialPowers.pushPrefEnv({"set":
+ [
+ ["media.mediasource.enabled", true],
+ ]
+ }, test);
+});
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_VideoPlaybackQuality_disabled.html b/dom/media/test/test_VideoPlaybackQuality_disabled.html
new file mode 100644
index 0000000000..1c41b79d8b
--- /dev/null
+++ b/dom/media/test/test_VideoPlaybackQuality_disabled.html
@@ -0,0 +1,37 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test basic functionality of VideoPlaybackQuality</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+SimpleTest.waitForExplicitFinish();
+
+function test() {
+ var video = document.createElement("video");
+ ok(!video.getVideoPlaybackQuality, "getVideoPlaybackQuality should be hidden behind a pref");
+ var accessThrows = false;
+ try {
+ video.getVideoPlaybackQuality();
+ } catch (e) {
+ accessThrows = true;
+ }
+ ok(accessThrows, "getVideoPlaybackQuality should be hidden behind a pref");
+ SimpleTest.finish();
+}
+
+addLoadEvent(function() {
+ SpecialPowers.pushPrefEnv({"set":
+ [
+ ["media.mediasource.enabled", false],
+ ]
+ }, test);
+});
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_access_control.html b/dom/media/test/test_access_control.html
new file mode 100644
index 0000000000..bad43dff22
--- /dev/null
+++ b/dom/media/test/test_access_control.html
@@ -0,0 +1,62 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=451958
+-->
+<head>
+ <title>Test for Bug 451958</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=451958">Mozilla Bug 451958</a>
+<p id="display"></p>
+<div id="content" style="display: none">
+</div>
+<pre id="test">
+<script type="application/javascript">
+
+/** Test for Bug 451958 **/
+
+function openWindow() {
+ window.open("http://example.org:80/tests/dom/media/test/file_access_controls.html", "", "width=500,height=500");
+}
+
+function run() {
+ SpecialPowers.pushPrefEnv({
+ "set": [
+ ["dom.security.https_first", false],
+ ],
+ }, openWindow);
+}
+
+function done() {
+ mediaTestCleanup();
+ SimpleTest.finish();
+}
+
+addLoadEvent(run);
+SimpleTest.waitForExplicitFinish();
+
+
+window.addEventListener("message", receiveMessage);
+
+function receiveMessage(event)
+{
+ if (event.origin !== "http://example.org") {
+ ok(false, "Received message from wrong domain");
+ return;
+ }
+
+ if (event.data.done == "true") {
+ done();
+ return;
+ }
+
+ ok(event.data.result, event.data.message);
+}
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_arraybuffer.html b/dom/media/test/test_arraybuffer.html
new file mode 100644
index 0000000000..9ef84c53dc
--- /dev/null
+++ b/dom/media/test/test_arraybuffer.html
@@ -0,0 +1,83 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=1462967
+-->
+<head>
+ <title>Test for Bug 1457661</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=1457661">Mozilla Bug 1457661</a>
+
+<p id="display"></p>
+<div id="content" style="display: none">
+
+</div>
+<pre id="test">
+<script type="application/javascript">
+
+// Test for Bug 1457661; Ensure that readyState properly move to 4 when using arrayBuffer obtained from XMLHttpRequest
+
+var manager = new MediaTestManager;
+
+function getBlob(url, callback) {
+ const req = new XMLHttpRequest();
+ req.addEventListener("load", function() {
+ callback(req.response);
+ });
+ req.open("GET", url, true);
+ req.responseType = "arraybuffer";
+ req.send();
+}
+
+function startTest(test, token) {
+ manager.started(token);
+ // Fetch the media resource using XHR so we can be sure the entire
+ // resource is loaded before we test buffered ranges. This ensures
+ // we have deterministic behaviour.
+ getBlob(test.name, function(arr) {
+ const v = document.createElement("video");
+ const events = [ "suspend", "play", "canplay", "canplaythrough", "loadstart", "loadedmetadata",
+ "loadeddata", "playing", "ended", "error", "stalled", "emptied", "abort",
+ "waiting", "pause", "durationchange", "seeking", "seeked" ];
+ function logEvent(e) {
+ info(test.name + ": got " + e.type + " event");
+ }
+ events.forEach(function(e) {
+ v.addEventListener(e, logEvent);
+ });
+ once(v, "stalled", function(e) {
+ // Resource fetch algorithm in local mode should never fire stalled event.
+ // https://html.spec.whatwg.org/multipage/media.html#concept-media-load-resource
+ ok(false, test.name + ": got stalled");
+ removeNodeAndSource(v);
+ manager.finished(token);
+ });
+ once(v, "canplaythrough", function(e) {
+ ok(true, test.name + ": got canplaythrough");
+ is(v.readyState, v.HAVE_ENOUGH_DATA, test.name + ": readyState is HAVE_ENOUGH_DATA");
+ removeNodeAndSource(v);
+ manager.finished(token);
+ });
+ const blob = new Blob([arr], {type:test.type});
+ const blobUrl = URL.createObjectURL(blob);
+
+ document.body.appendChild(v);
+ v.preload = "auto";
+ v.src = blobUrl;
+ v.load();
+ });
+}
+
+SimpleTest.waitForExplicitFinish();
+// Note: No need to set media test prefs, since we're using XHR to fetch
+// media data.
+manager.runTests(gSmallTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_aspectratio_mp4.html b/dom/media/test/test_aspectratio_mp4.html
new file mode 100644
index 0000000000..5e01875439
--- /dev/null
+++ b/dom/media/test/test_aspectratio_mp4.html
@@ -0,0 +1,46 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=975978
+-->
+
+<head>
+ <title>Media test: default video size</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=975978">Mozilla Bug 975978</a>
+
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+SimpleTest.waitForExplicitFinish();
+
+// MP4 video with display size is difference to decode frame size.
+// The display size is recorded in TrackHeaderBox 'tkhd' of this mp4 video.
+var resource =
+ { name:"pixel_aspect_ratio.mp4", type:"video/mp4", width:525, height:288 };
+
+var v = document.createElement("video");
+v.onloadedmetadata = function() {
+ is(v.videoWidth, resource.width, "Intrinsic width should match video width");
+ is(v.videoHeight, resource.height, "Intrinsic height should match video height");
+ SimpleTest.finish();
+}
+v.addEventListener("error", function(ev) {
+ if (v.readyState < v.HAVE_METADATA) {
+ info("Video element returns with readyState " + v.readyState + " error.code " + v.error.code);
+ todo(false, "This platform doesn't support to retrieve MP4 metadata.");
+ SimpleTest.finish();
+ }
+});
+
+v.src = resource.name;
+v.preload = "auto";
+
+document.body.appendChild(v);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_audio1.html b/dom/media/test/test_audio1.html
new file mode 100644
index 0000000000..a5d9f12ab9
--- /dev/null
+++ b/dom/media/test/test_audio1.html
@@ -0,0 +1,33 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: Audio Constructor Test 1</title>
+ <script type="text/javascript" src="/MochiKit/MochiKit.js"></script>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function startTest(test, token) {
+ var a1 = new Audio();
+ if (!a1.canPlayType(test.type))
+ return;
+ manager.started(token);
+
+ is(a1.getAttribute("preload"), "auto", "preload:auto automatically set");
+ is(a1.src, "", "Src set?");
+ a1 = null;
+ manager.finished(token);
+}
+
+manager.runTests(gAudioTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_audio2.html b/dom/media/test/test_audio2.html
new file mode 100644
index 0000000000..fde9adc791
--- /dev/null
+++ b/dom/media/test/test_audio2.html
@@ -0,0 +1,33 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: Audio Constructor Test 2</title>
+ <script type="text/javascript" src="/MochiKit/MochiKit.js"></script>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var tmpAudio = new Audio();
+
+var manager = new MediaTestManager;
+
+function startTest(test, token) {
+ if (!tmpAudio.canPlayType(test.type))
+ return;
+ manager.started(token);
+ var a1 = new Audio(test.name);
+ is(a1.getAttribute("preload"), "auto", "Preload automatically set to auto");
+ ok(a1.src.endsWith("/" + test.name), "src OK");
+ manager.finished(token);
+}
+
+manager.runTests(gAudioTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_audioDocumentTitle.html b/dom/media/test/test_audioDocumentTitle.html
new file mode 100644
index 0000000000..2913debb39
--- /dev/null
+++ b/dom/media/test/test_audioDocumentTitle.html
@@ -0,0 +1,56 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=463830
+-->
+<head>
+ <title>Test for Bug 463830</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=463830">Mozilla Bug 463830</a>
+<p id="display"></p>
+<iframe id="i"></iframe>
+<div id="content" style="display: none">
+
+</div>
+<pre id="test">
+<script type="application/javascript">
+
+/** Test for Bug 463830 **/
+
+var gTests = [
+ { file: "r11025_s16_c1.wav", title: "r11025_s16_c1.wav" }
+];
+
+var gTestNum = 0;
+
+addLoadEvent(runTest);
+
+var title;
+var i = document.getElementById("i");
+
+function runTest() {
+ if (gTestNum == gTests.length) {
+ SimpleTest.finish();
+ return;
+ }
+ if (gTestNum == 0) {
+ i.addEventListener("load", function() {
+ is(i.contentDocument.title, title, "Doc title incorrect");
+ setTimeout(runTest, 0);
+ });
+ }
+
+ title = gTests[gTestNum].title;
+ i.src = gTests[gTestNum].file;
+ gTestNum++;
+}
+
+SimpleTest.waitForExplicitFinish();
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_background_video_cancel_suspend_taint.html b/dom/media/test/test_background_video_cancel_suspend_taint.html
new file mode 100644
index 0000000000..fc48be524b
--- /dev/null
+++ b/dom/media/test/test_background_video_cancel_suspend_taint.html
@@ -0,0 +1,70 @@
+<!DOCTYPE html>
+<meta charset="utf-8">
+<title>Test Background Video Suspend Cancels (Element Taint)</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<script src="manifest.js"></script>
+<script src="background_video.js"></script>
+<link rel="stylesheet" href="/tests/SimpleTest/test.css"/>
+<script type="text/javascript">
+"use strict";
+
+var manager = new MediaTestManager;
+
+/**
+ * @param {HTMLMediaElement} video Video element under test.
+ * @returns {Promise} Promise that is resolved when video decode resumes.
+ */
+function testSuspendTimerCanceledWhenTainted(video) {
+ function ended() {
+ video.removeEventListener("mozcancelvideosuspendtimer", canceled);
+ ok(false, `${video.token} ended before suspend cancels`);
+ this.ended_resolve();
+ }
+
+ function canceled() {
+ video.removeEventListener("ended", ended);
+ ok(true, `${video.token} suspend cancels`);
+ this.canceled_resolve();
+ }
+
+ let p = Promise.race([
+ new Promise((resolve) => {
+ video.ended_resolve = resolve;
+ video.addEventListener('ended', ended, { 'once': true });
+ }),
+ new Promise((resolve) => {
+ video.canceled_resolve = resolve;
+ video.addEventListener('mozcancelvideosuspendtimer', canceled, { 'once': true });
+ })
+ ]);
+
+ Log(video.token, "Mark tainted");
+
+ let c = document.createElement('canvas');
+ let g = c.getContext('2d');
+ g.drawImage(video, 0, 0, c.width, c.height);
+ ok(video.hasSuspendTaint(), 'video used with drawImage is tainted.');
+
+ return p;
+}
+
+startTest({
+ desc: 'Test Background Video Suspend Cancels (Element Taint)',
+ prefs: [
+ [ "media.test.video-suspend", true ],
+ [ "media.suspend-background-video.enabled", true ],
+ [ "media.suspend-background-video.delay-ms", 10000 ]
+ ],
+ tests: gDecodeSuspendTests,
+ runTest: (test, token) => {
+ let v = appendVideoToDoc(test.name, token);
+ manager.started(token);
+
+ waitUntilVisible(v)
+ .then(() => waitUntilPlaying(v))
+ .then(() => testSuspendTimerStartedWhenHidden(v))
+ .then(() => testSuspendTimerCanceledWhenTainted(v))
+ .then(() => { manager.finished(token); });
+ }
+});
+</script>
diff --git a/dom/media/test/test_background_video_cancel_suspend_visible.html b/dom/media/test/test_background_video_cancel_suspend_visible.html
new file mode 100644
index 0000000000..73aaa4783e
--- /dev/null
+++ b/dom/media/test/test_background_video_cancel_suspend_visible.html
@@ -0,0 +1,69 @@
+<!DOCTYPE html>
+<meta charset="utf-8">
+<title>Test Background Video Suspend Cancels (Visibility)</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<script src="manifest.js"></script>
+<script src="background_video.js"></script>
+<link rel="stylesheet" href="/tests/SimpleTest/test.css"/>
+<script type="text/javascript">
+"use strict";
+
+var manager = new MediaTestManager;
+
+/**
+ * Check that making the element visible before suspend timer delay causes the
+ * the suspend timer to be canceled.
+ * @param {HTMLMediaElement} video Video element under test.
+ * @returns {Promise} Promise that is resolved when video decode resumes.
+ */
+function testSuspendTimerCanceledWhenShown(video) {
+ function ended() {
+ video.removeEventListener("mozcancelvideosuspendtimer", canceled);
+ ok(false, `${video.token} ended before suspend cancels`);
+ this.ended_resolve();
+ }
+
+ function canceled() {
+ video.removeEventListener("ended", ended);
+ ok(true, `${video.token} suspend cancels`);
+ this.canceled_resolve();
+ }
+
+ let p = Promise.race([
+ new Promise((resolve) => {
+ video.ended_resolve = resolve;
+ video.addEventListener('ended', ended, { 'once': true });
+ }),
+ new Promise((resolve) => {
+ video.canceled_resolve = resolve;
+ video.addEventListener('mozcancelvideosuspendtimer', canceled, { 'once': true });
+ })
+ ]);
+
+ Log(video.token, "Set visible");
+ video.setVisible(true);
+
+ return p;
+}
+
+startTest({
+ desc: 'Test Background Video Suspend Cancels (Visibility)',
+ prefs: [
+ [ "media.test.video-suspend", true ],
+ [ "media.suspend-background-video.enabled", true ],
+ [ "media.suspend-background-video.delay-ms", 10000 ]
+ ],
+ tests: gDecodeSuspendTests,
+ runTest: (test, token) => {
+ let v = appendVideoToDoc(test.name, token);
+ manager.started(token);
+
+ waitUntilPlaying(v)
+ .then(() => testSuspendTimerStartedWhenHidden(v))
+ .then(() => testSuspendTimerCanceledWhenShown(v))
+ .then(() => {
+ ok(true, `${v.token} finished`);
+ manager.finished(token);
+ });
+ }
+});
diff --git a/dom/media/test/test_background_video_drawimage_with_suspended_video.html b/dom/media/test/test_background_video_drawimage_with_suspended_video.html
new file mode 100644
index 0000000000..1b1b4264f2
--- /dev/null
+++ b/dom/media/test/test_background_video_drawimage_with_suspended_video.html
@@ -0,0 +1,76 @@
+<!DOCTYPE html>
+<meta charset="utf-8">
+<title>Test Background Video Displays Video Frame via drawImage When Suspended</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<script src="manifest.js"></script>
+<script src="background_video.js"></script>
+<link rel="stylesheet" href="/tests/SimpleTest/test.css"/>
+<style>
+video, canvas {
+ border: 1px solid black;
+}
+</style>
+<script type="text/javascript">
+"use strict";
+
+var manager = new MediaTestManager;
+
+function drawVideoToCanvas(v) {
+ console.log('drawVideoToCanvas');
+ let c = document.createElement('canvas');
+ c.width = 4;
+ c.height = 4;
+ c.style.width = 64;
+ c.style.height = 64;
+ document.body.appendChild(c);
+
+ let gfx = c.getContext('2d');
+ if (!gfx) {
+ throw Error("Unable to obtain context '2d' from canvas");
+ }
+
+ gfx.drawImage(v, 0, 0, 4, 4);
+ let imageData = gfx.getImageData(0, 0, 4, 4);
+ let pixels = imageData.data;
+
+ // Check that pixels aren't all the same colour.
+ // Implements by checking against rgb of the first pixel.
+ let rr = pixels[0],
+ gg = pixels[1],
+ bb = pixels[2],
+ allSame = true;
+
+ for (let i = 0; i < 4*4; i++) {
+ let r = pixels[4*i+0];
+ let g = pixels[4*i+1];
+ let b = pixels[4*i+2];
+ if (r != rr || g != gg || b != bb) {
+ allSame = false;
+ break;
+ }
+ }
+
+ ok(!allSame, "Pixels aren't all the same color");
+}
+
+startTest({
+ desc: 'Test Background Video Displays Video Frame via drawImage When Suspended',
+ prefs: [
+ [ "media.test.video-suspend", true ],
+ [ "media.suspend-background-video.enabled", true ],
+ [ "media.suspend-background-video.delay-ms", 500 ]
+ ],
+ tests: gDecodeSuspendTests,
+ runTest: (test, token) => {
+ let v = appendVideoToDoc(test.name, token);
+ manager.started(token);
+
+ waitUntilPlaying(v)
+ .then(() => testVideoSuspendsWhenHidden(v))
+ .then(() => {
+ drawVideoToCanvas(v);
+ manager.finished(token);
+ });
+ }
+});
+</script>
diff --git a/dom/media/test/test_background_video_ended_event.html b/dom/media/test/test_background_video_ended_event.html
new file mode 100644
index 0000000000..7ef8ec81f6
--- /dev/null
+++ b/dom/media/test/test_background_video_ended_event.html
@@ -0,0 +1,48 @@
+<!DOCTYPE html>
+<meta charset="utf-8">
+<title>Test Background Video Suspends</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<script src="manifest.js"></script>
+<script src="background_video.js"></script>
+<link rel="stylesheet" href="/tests/SimpleTest/test.css"/>
+<script type="text/javascript">
+"use strict";
+
+var manager = new MediaTestManager;
+
+startTest({
+ desc: "Test background video doesn't fire the 'ended' event twice.",
+ prefs: [
+ [ "media.test.video-suspend", true ],
+ [ "media.suspend-background-video.enabled", true ],
+ [ "media.suspend-background-video.delay-ms", 100 ]
+ ],
+ tests: gDecodeSuspendTests,
+ runTest: (test, token) => {
+ let v = appendVideoToDoc(test.name, token);
+ manager.started(token);
+
+ let count = 0;
+ v.addEventListener("ended", function() {
+ is(++count, 1, `${token} should get only one 'ended' event.`);
+ });
+
+ /*
+ * This test checks that, after a video element had finished its playback,
+ * resuming video decoder doesn't dispatch 2nd ended event.
+ * This issue was found in bug 1349097.
+ */
+ waitUntilPlaying(v)
+ .then(() => testVideoSuspendsWhenHidden(v))
+ .then(() => waitUntilEnded(v))
+ .then(() => testVideoResumesWhenShown(v))
+ .then(() => {
+ // Wait for a while and finish the test. We should get only one 'ended' event.
+ setTimeout(function() {
+ removeNodeAndSource(v);
+ manager.finished(token);
+ }, 3000);
+ });
+ }
+});
+</script>
diff --git a/dom/media/test/test_background_video_no_suspend_disabled.html b/dom/media/test/test_background_video_no_suspend_disabled.html
new file mode 100644
index 0000000000..0d6ede5c7e
--- /dev/null
+++ b/dom/media/test/test_background_video_no_suspend_disabled.html
@@ -0,0 +1,36 @@
+<!DOCTYPE html>
+<meta charset="utf-8">
+<title>Test Background Video Doesn't Suspend When Feature Disabled</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<script src="manifest.js"></script>
+<script src="background_video.js"></script>
+<link rel="stylesheet" href="/tests/SimpleTest/test.css"/>
+<script>
+"use strict";
+
+var manager = new MediaTestManager;
+
+startTest({
+ desc: "Test Background Video Doesn't Suspend When Feature Disabled.",
+ prefs: [
+ [ 'media.test.video-suspend', true ],
+ [ 'media.suspend-background-video.enabled', false ],
+ [ 'media.suspend-background-video.delay-ms', 0 ]
+ ],
+ tests: gDecodeSuspendTests,
+ runTest: (test, token) => {
+ let v = appendVideoToDoc(test.name, token);
+ manager.started(token);
+
+ /* This test checks that suspend doesn't occur when the feature is disabled */
+ waitUntilPlaying(v)
+ .then(() => checkVideoDoesntSuspend(v))
+ .then(() => {
+ ok(true, 'Video ended before decode was suspended');
+ manager.finished(token); })
+ .catch((e) => {
+ ok(false, 'Test Failed: ' + e.toString());
+ manager.finished(token); });
+ }
+});
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_background_video_no_suspend_not_in_tree.html b/dom/media/test/test_background_video_no_suspend_not_in_tree.html
new file mode 100644
index 0000000000..327fda4403
--- /dev/null
+++ b/dom/media/test/test_background_video_no_suspend_not_in_tree.html
@@ -0,0 +1,56 @@
+<!DOCTYPE html>
+<meta charset="utf-8">
+<title>Test Background Video Doesn't Suspend When Timeout Is Longer Than Video</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<script src="manifest.js"></script>
+<script src="background_video.js"></script>
+<link rel="stylesheet" href="/tests/SimpleTest/test.css"/>
+<script>
+"use strict";
+
+var manager = new MediaTestManager;
+
+var MIN_DELAY = 100;
+
+/**
+ * @param {string} url video src.
+ * @returns {HTMLMediaElement} The created video element.
+ */
+function createVideoNotAppendToDoc(url, token, width, height) {
+ // Default size of (160, 120) is used by other media tests.
+ if (width === undefined) { width = 160; }
+ if (height === undefined) { height = 3*width/4; }
+
+ let v = document.createElement('video');
+ v.token = token;
+ v.width = width;
+ v.height = height;
+ v.src = url;
+ return v;
+}
+
+startTest({
+ desc: "Test Background Video Doesn't Suspend When If The Video Is Not In Tree.",
+ prefs: [
+ [ 'media.test.video-suspend', true ],
+ [ 'media.suspend-background-video.enabled', true ],
+ [ 'media.suspend-background-video.delay-ms', MIN_DELAY ]
+ ],
+ tests: gDecodeSuspendTests,
+ runTest: (test, token) => {
+ let v = createVideoNotAppendToDoc(test.name, token);
+ manager.started(token);
+
+ /* This test checks that suspend doesn't occur if a video element is not
+ append to tree. */
+ waitUntilPlaying(v)
+ .then(() => checkVideoDoesntSuspend(v))
+ .then(() => {
+ ok(true, 'Video ended before decode was suspended');
+ manager.finished(token); })
+ .catch((e) => {
+ ok(false, 'Test Failed: ' + e.toString());
+ manager.finished(token); });
+ }
+});
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_background_video_no_suspend_short_vid.html b/dom/media/test/test_background_video_no_suspend_short_vid.html
new file mode 100644
index 0000000000..d1716755fc
--- /dev/null
+++ b/dom/media/test/test_background_video_no_suspend_short_vid.html
@@ -0,0 +1,38 @@
+<!DOCTYPE html>
+<meta charset="utf-8">
+<title>Test Background Video Doesn't Suspend When Timeout Is Longer Than Video</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<script src="manifest.js"></script>
+<script src="background_video.js"></script>
+<link rel="stylesheet" href="/tests/SimpleTest/test.css"/>
+<script>
+"use strict";
+
+var manager = new MediaTestManager;
+
+startTest({
+ desc: "Test Background Video Doesn't Suspend When Timeout Is Longer Than Video.",
+ prefs: [
+ [ 'media.test.video-suspend', true ],
+ [ 'media.suspend-background-video.enabled', true ],
+ // Gizmo.mp4 is about 5.6s
+ [ 'media.suspend-background-video.delay-ms', 10000 ]
+ ],
+ tests: gDecodeSuspendTests,
+ runTest: (test, token) => {
+ let v = appendVideoToDoc(test.name, token);
+ manager.started(token);
+
+ /* This test checks that suspend doesn't occur when the delay is longer
+ than the duration of the video that's playing */
+ waitUntilPlaying(v)
+ .then(() => checkVideoDoesntSuspend(v))
+ .then(() => {
+ ok(true, 'Video ended before decode was suspended');
+ manager.finished(token); })
+ .catch((e) => {
+ ok(false, 'Test Failed: ' + e.toString());
+ manager.finished(token); });
+ }
+});
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_background_video_resume_after_end_show_last_frame.html b/dom/media/test/test_background_video_resume_after_end_show_last_frame.html
new file mode 100644
index 0000000000..68f7179e12
--- /dev/null
+++ b/dom/media/test/test_background_video_resume_after_end_show_last_frame.html
@@ -0,0 +1,141 @@
+<!DOCTYPE html>
+<meta charset="utf-8">
+<title>Test Background Video Suspends</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<script src="manifest.js"></script>
+<script src="background_video.js"></script>
+<link rel="stylesheet" href="/tests/SimpleTest/test.css"/>
+<script type="text/javascript">
+"use strict";
+
+var manager = new MediaTestManager;
+
+function testSameContent(video1, video2) {
+ if (video1.videoWidth != video2.videoWidth ||
+ video1.videoHeight != video2.videoHeight) {
+ ok(false, `${video1.token} video1 and video2 have different dimensions.`);
+ return;
+ }
+
+ let w = video1.videoWidth;
+ let h = video1.videoHeight;
+ let c1 = document.createElement('canvas');
+ let c2 = document.createElement('canvas');
+ c1.width = w;
+ c1.height = h;
+ c2.width = w;
+ c2.height = h;
+
+ let gfx1 = c1.getContext('2d');
+ let gfx2 = c2.getContext('2d');
+ if (!gfx1 || !gfx2) {
+ ok(false, "Unable to obtain context '2d' from canvas");
+ return;
+ }
+
+ gfx1.drawImage(video1, 0, 0, w, h);
+ gfx2.drawImage(video2, 0, 0, w, h);
+
+ // Get content out.
+ let contentWidth = 4;
+ let contentHeight = 4;
+ let imageData1 = gfx1.getImageData(0, 0, contentWidth, contentHeight);
+ let imageData2 = gfx2.getImageData(0, 0, contentWidth, contentHeight);
+ let pixels1 = imageData1.data;
+ let pixels2 = imageData2.data;
+
+ // Check that the content of two video are identical.
+ for (let i = 0; i < contentWidth*contentHeight; i++) {
+ let pixelCount = 4 * i;
+ if (pixels1[pixelCount+0] != pixels2[pixelCount+0] ||
+ pixels1[pixelCount+1] != pixels2[pixelCount+1] ||
+ pixels1[pixelCount+2] != pixels2[pixelCount+2] ||
+ pixels1[pixelCount+3] != pixels2[pixelCount+3]) {
+ ok(false, `${video1.token} video1 and video2 have different content.`);
+ return;
+ }
+ }
+
+ ok(true, `${video1.token} video1 and video2 have identical content.`);
+}
+
+function waitUntilSeekToLastFrame(video) {
+ Log(video.token, "Waiting for seeking to the last frame");
+ function callSeekToNextFrame() {
+ video.seekToNextFrame().then(
+ () => {
+ if (!video.seenEnded) {
+ callSeekToNextFrame();
+ }
+ },
+ () => {
+ // When seek reaches the end, the promise is resolved before 'ended'
+ // is fired. The resolver calls callSeekToNextFrame() to schedule
+ // another seek and then the 'ended' handler calls finish() to shut
+ // down the MediaDecoder which will reject the seek promise. So we don't
+ // raise an error in this case.
+ ok(video.seenEnded, "seekToNextFrame() failed.");
+ }
+ );
+ }
+
+ return new Promise(function(resolve, reject) {
+ video.seenEnded = false;
+ video.addEventListener("ended", () => {
+ video.seenEnded = true;
+ resolve();
+ }, true);
+ callSeekToNextFrame(video);
+ });
+}
+
+startTest({
+ desc: "Test resume an ended video shows the last frame.",
+ prefs: [
+ [ "media.test.video-suspend", true ],
+ [ "media.suspend-background-video.enabled", true ],
+ [ "media.suspend-background-video.delay-ms", 100 ],
+ [ "media.dormant-on-pause-timeout-ms", -1],
+ [ "media.decoder.skip-to-next-key-frame.enabled", false]
+ ],
+ tests: gDecodeSuspendTests,
+
+ runTest: async (test, token) => {
+ let v = appendVideoToDocWithoutLoad(token);
+ let vMuted = appendVideoToDocWithoutLoad(token+"-muted");
+ vMuted.muted = true;
+ let vReference = appendVideoToDocWithoutLoad(token+"-ref");
+ manager.started(token);
+
+ /*
+ * This test checks that, after a video element had finished its playback,
+ * resuming video decoder should seek to the last frame.
+ * This issue was found in bug 1358057.
+ */
+ await waitUntilVisible(v);
+ await Promise.all([
+ loadAndWaitUntilLoadedmetadata(v, test.name),
+ loadAndWaitUntilLoadedmetadata(vMuted, test.name),
+ loadAndWaitUntilLoadedmetadata(vReference, test.name, "auto"),
+ ]);
+
+ await waitUntilPlaying(v);
+ await waitUntilPlaying(vMuted);
+ await testVideoSuspendsWhenHidden(v);
+ await testVideoSuspendsWhenHidden(vMuted);
+ await Promise.all([
+ waitUntilEnded(v),
+ waitUntilEnded(vMuted),
+ waitUntilSeekToLastFrame(vReference),
+ ]);
+ await testVideoOnlySeekCompletedWhenShown(v);
+ await testVideoOnlySeekCompletedWhenShown(vMuted);
+ testSameContent(v, vReference);
+ testSameContent(vMuted, vReference);
+ removeNodeAndSource(v);
+ removeNodeAndSource(vMuted);
+ removeNodeAndSource(vReference);
+ manager.finished(token);
+ }
+});
+</script>
diff --git a/dom/media/test/test_background_video_resume_looping_video_without_audio.html b/dom/media/test/test_background_video_resume_looping_video_without_audio.html
new file mode 100644
index 0000000000..3ec64c6d79
--- /dev/null
+++ b/dom/media/test/test_background_video_resume_looping_video_without_audio.html
@@ -0,0 +1,81 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Resume suspended looping video which doesn't contain audio track</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script src="manifest.js"></script>
+ <script src="background_video.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+</head>
+<body>
+<script class="testbody" type="text/javascript">
+/**
+ * This test is used to ensure that the looping video (without audio track) which
+ * has been suspended can continute to playback correctly after we resume video
+ * decoding.
+ */
+async function startTest() {
+ const video = await createVisibleVideo();
+ await startVideo(video);
+ await suspendVideoDecoding(video);
+ await resumeVideoDecoding(video);
+ await checkIfVideoIsStillPlaying(video);
+ endTestAndClearVideo(video);
+}
+
+SimpleTest.waitForExplicitFinish();
+SpecialPowers.pushPrefEnv({ 'set': [
+ ["media.test.video-suspend", true],
+ ["media.suspend-background-video.enabled", true],
+ ["media.suspend-background-video.delay-ms", 0],
+ ]}, () => {
+ startTest();
+});
+
+/**
+ * The following are test helper functions.
+ */
+async function createVisibleVideo() {
+ let video = document.createElement("video");
+ video.src = "gizmo-noaudio.webm";
+ video.controls = true;
+ video.loop = true;
+ document.body.appendChild(video);
+ info(`ensure video becomes visible`);
+ await waitUntilVisible(video);
+ return video;
+}
+
+async function startVideo(video) {
+ info(`start playing video`);
+ const played = video && await video.play().then(() => true, () => false);
+ ok(played, "video has started playing");
+}
+
+async function suspendVideoDecoding(video) {
+ info(`suspend video decoding`);
+ video.setVisible(false);
+ await nextVideoSuspends(video);
+ info(`suspended video decoding`);
+}
+
+async function resumeVideoDecoding(video) {
+ info(`resume video decoding.`);
+ video.setVisible(true);
+ await nextVideoResumes(video);
+ info(`resumed video decoding`);
+}
+
+async function checkIfVideoIsStillPlaying(video) {
+ await once(video, "timeupdate");
+ ok(!video.paused, "video is still playing after resuming video decoding.")
+}
+
+function endTestAndClearVideo(video) {
+ removeNodeAndSource(video);
+ SimpleTest.finish();
+}
+
+</script>
+</body>
+</html>
diff --git a/dom/media/test/test_background_video_suspend.html b/dom/media/test/test_background_video_suspend.html
new file mode 100644
index 0000000000..f68740954b
--- /dev/null
+++ b/dom/media/test/test_background_video_suspend.html
@@ -0,0 +1,81 @@
+<!DOCTYPE html>
+<meta charset="utf-8">
+<title>Test Background Video Suspends</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<script src="manifest.js"></script>
+<script src="background_video.js"></script>
+<link rel="stylesheet" href="/tests/SimpleTest/test.css" />
+<script type="text/javascript">
+ "use strict";
+
+ var manager = new MediaTestManager;
+
+ var MIN_DELAY = 100;
+
+ function testDelay(v, start, min) {
+ let end = performance.now();
+ let delay = end - start;
+ ok(delay > min, `${v.token} suspended with a delay of ${delay} ms`);
+ }
+
+ async function runTest(test, token) {
+ let video = appendVideoToDocWithoutLoad(token);
+ manager.started(token);
+
+ let visible = waitUntilVisible(video);
+ let ended = nextVideoEnded(video);
+ let playing = nextVideoPlaying(video);
+ let resumes = nextVideoResumes(video);
+ let suspends = nextVideoSuspends(video);
+
+ Log(token, "Waiting until video becomes visible");
+ await visible;
+
+ Log(token, "Waiting for metadata loaded");
+ await loadAndWaitUntilLoadedmetadata(video, test.name);
+
+ Log(token, "Start playing");
+ video.play();
+
+ Log(token, "Waiting for video playing");
+ await playing;
+
+ let start = performance.now();
+
+ Log(token, "Set hidden");
+ video.setVisible(false);
+
+ Log(token, "Waiting for video suspend");
+ await suspends;
+
+ testDelay(video, start, MIN_DELAY);
+
+ Log(token, "Set visible");
+ video.setVisible(true);
+
+ Log(token, "Waiting for video resume");
+ await resumes;
+
+ Log(token, "Waiting for ended");
+ await ended;
+
+ ok(video.currentTime >= video.duration, 'current time approximates duration.');
+
+ removeNodeAndSource(video);
+ manager.finished(token);
+ }
+
+ startTest({
+ desc: 'Test Background Video Suspends',
+ prefs: [
+ ["media.test.video-suspend", true],
+ ["media.suspend-background-video.enabled", true],
+ // Use a short delay to ensure video decode suspend happens before end
+ // of video.
+ ["media.suspend-background-video.delay-ms", MIN_DELAY],
+ ["privacy.reduceTimerPrecision", false]
+ ],
+ tests: gDecodeSuspendTests,
+ runTest
+ });
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_background_video_suspend_ends.html b/dom/media/test/test_background_video_suspend_ends.html
new file mode 100644
index 0000000000..fea46afffe
--- /dev/null
+++ b/dom/media/test/test_background_video_suspend_ends.html
@@ -0,0 +1,55 @@
+<!DOCTYPE html>
+<meta charset="utf-8">
+<title>Test Background Suspended Video Fires 'ended' Event</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<script src="manifest.js"></script>
+<script src="background_video.js"></script>
+<link rel="stylesheet" href="/tests/SimpleTest/test.css" />
+<script type="text/javascript">
+ "use strict";
+
+ PARALLEL_TESTS = 1;
+
+ var manager = new MediaTestManager;
+
+ async function runTest(test, token) {
+ let video = appendVideoToDoc(test.name, token);
+ manager.started(token);
+
+ // This test checks that 'ended' event is received for videos with
+ // suspended video decoding. This is important for looping video logic
+ // handling in HTMLMediaElement.
+
+ let ended = nextVideoEnded(video);
+ let suspends = nextVideoSuspends(video);
+
+ Log(token, "Start playing");
+ video.play();
+
+ Log(token, "Set hidden");
+ video.setVisible(false);
+
+ Log(token, "Waiting for video suspend");
+ await suspends;
+
+ Log(token, "Waiting for ended");
+ await ended;
+
+ ok(video.currentTime >= video.duration, 'current time approximates duration.');
+
+ manager.finished(token);
+ }
+
+ startTest({
+ desc: "Test Background Suspended Video Fires 'ended' Event",
+ prefs: [
+ ["media.test.video-suspend", true],
+ ["media.suspend-background-video.enabled", true],
+ // User a short delay to ensure video decode suspend happens before end
+ // of video.
+ ["media.suspend-background-video.delay-ms", 1000]
+ ],
+ tests: gDecodeSuspendTests,
+ runTest
+ });
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_background_video_suspend_ready_state.html b/dom/media/test/test_background_video_suspend_ready_state.html
new file mode 100644
index 0000000000..e33d257e09
--- /dev/null
+++ b/dom/media/test/test_background_video_suspend_ready_state.html
@@ -0,0 +1,74 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+<title>Check element's ready state while suspending video decoding</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<script src="manifest.js"></script>
+<script src="background_video.js"></script>
+<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+<script type="application/javascript">
+
+/**
+ * This test is used to ensure that the media element's ready state won't be
+ * incorrectly changed to `HAVE_METADATA` or lower when seeking happens on a
+ * null decoder that is caused by suspending video decoding.
+ */
+add_task(async function setupTestingPref() {
+ await SpecialPowers.pushPrefEnv({
+ set: [
+ ["media.test.video-suspend", true],
+ ["media.suspend-background-video.enabled", true],
+ ["media.suspend-background-video.delay-ms", 0],
+ ],
+ });
+});
+
+add_task(async function testReadyStateWhenSuspendingVideoDecoding() {
+ const video = await createPlayingVideo();
+ assertVideoReadyStateIsSufficientForPlaying(video);
+ await suspendVideoDecoding(video);
+ await seekVideo(video);
+ assertVideoReadyStateIsSufficientForPlaying(video);
+});
+
+async function createPlayingVideo() {
+ let video = document.createElement("video");
+ video.src = "gizmo.mp4";
+ video.controls = true;
+ video.loop = true;
+ document.body.appendChild(video);
+ ok(await video.play().then(_=>true, _=>false), "video started playing");
+ return video;
+}
+
+function assertVideoReadyStateIsSufficientForPlaying(video) {
+ ok(video.readyState > HTMLMediaElement.HAVE_METADATA,
+ `Ready state ${video.readyState} is suffient for playing`);
+}
+
+async function suspendVideoDecoding(video) {
+ video.setVisible(false);
+ nextVideoSuspends(video);
+ info(`suspended video decoding`);
+ // Because the suspend event we received can only indicate MDSM finishes the
+ // setup of suspending decoding, but the actual setting the null decoder
+ // happens on MediaFormatReader where we are not able to observe the behavior
+ // via event. Instread of implementing something in MFR to propagate the event
+ // or changing the timing of current suspend event, which might increase the
+ // maintenance effect, Here choose to use a simple workaround to wait for a
+ // short while to ensure the decoding has been switched to the null decoder.
+ // eslint-disable-next-line mozilla/no-arbitrary-setTimeout
+ await new Promise(resolve => setTimeout(resolve, 2000));
+}
+
+async function seekVideo(video) {
+ video.currentTime = 0;
+ ok(await new Promise(r => video.onseeked = r).then(_=>true, _=>false),
+ `seeking completed`);
+}
+
+</script>
+</head>
+<body>
+</body>
+</html>
diff --git a/dom/media/test/test_background_video_tainted_by_capturestream.html b/dom/media/test/test_background_video_tainted_by_capturestream.html
new file mode 100644
index 0000000000..4ceba7fff9
--- /dev/null
+++ b/dom/media/test/test_background_video_tainted_by_capturestream.html
@@ -0,0 +1,46 @@
+<!DOCTYPE html>
+<meta charset="utf-8">
+<title>Test Background Video Is Tainted By captureStream</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<script src="manifest.js"></script>
+<script src="background_video.js"></script>
+<link rel="stylesheet" href="/tests/SimpleTest/test.css"/>
+<script type="text/javascript">
+"use strict";
+
+var manager = new MediaTestManager;
+
+function captureVideoAsStream(v) {
+ v.mozCaptureStream();
+}
+
+startTest({
+ desc: 'Test Background Video Is Tainted By captureStream',
+ prefs: [
+ [ "media.test.video-suspend", true ],
+ [ "media.suspend-background-video.enabled", true ],
+ [ "media.suspend-background-video.delay-ms", 1000 ]
+ ],
+ tests: gDecodeSuspendTests,
+ runTest: (test, token) => {
+ ok(true, `${test.name}`);
+ let v = appendVideoToDoc(test.name, token);
+ manager.started(token);
+
+ waitUntilPlaying(v)
+ .then(() => {
+ captureVideoAsStream(v);
+ ok(v.hasSuspendTaint(), "Video is tainted after captured");
+ return checkVideoDoesntSuspend(v);
+ })
+ .then(() => {
+ ok(true, 'Video ended before decode was suspended');
+ manager.finished(token);
+ })
+ .catch((e) => {
+ ok(false, 'Test failed: ' + e.toString());
+ manager.finished(token);
+ });
+ }
+});
+</script>
diff --git a/dom/media/test/test_background_video_tainted_by_createimagebitmap.html b/dom/media/test/test_background_video_tainted_by_createimagebitmap.html
new file mode 100644
index 0000000000..42bdd8e246
--- /dev/null
+++ b/dom/media/test/test_background_video_tainted_by_createimagebitmap.html
@@ -0,0 +1,42 @@
+<!DOCTYPE html>
+<meta charset="utf-8">
+<title>Test Background Video Is Tainted By createImageBitmap</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<script src="manifest.js"></script>
+<script src="background_video.js"></script>
+<link rel="stylesheet" href="/tests/SimpleTest/test.css"/>
+<script type="text/javascript">
+"use strict";
+
+var manager = new MediaTestManager;
+
+startTest({
+ desc: 'Test Background Video Is Tainted By createImageBitmap',
+ prefs: [
+ [ "media.test.video-suspend", true ],
+ [ "media.suspend-background-video.enabled", true ],
+ [ "media.suspend-background-video.delay-ms", 1000 ]
+ ],
+ tests: gDecodeSuspendTests,
+ runTest: (test, token) => {
+ ok(true, `${test.name}`);
+ let v = appendVideoToDoc(test.name, token);
+ manager.started(token);
+
+ waitUntilPlaying(v)
+ .then(() => createImageBitmap(v))
+ .then(() => {
+ ok(v.hasSuspendTaint(), "Video is tainted after drawing to canvas");
+ return checkVideoDoesntSuspend(v);
+ })
+ .then(() => {
+ ok(true, 'Video ended before decode was suspended');
+ manager.finished(token);
+ })
+ .catch((e) => {
+ ok(false, 'Test failed: ' + e.toString());
+ manager.finished(token);
+ });
+ }
+});
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_background_video_tainted_by_drawimage.html b/dom/media/test/test_background_video_tainted_by_drawimage.html
new file mode 100644
index 0000000000..d41cf53631
--- /dev/null
+++ b/dom/media/test/test_background_video_tainted_by_drawimage.html
@@ -0,0 +1,58 @@
+<!DOCTYPE html>
+<meta charset="utf-8">
+<title>Test Background Video Is Tainted By drawImage</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<script src="manifest.js"></script>
+<script src="background_video.js"></script>
+<link rel="stylesheet" href="/tests/SimpleTest/test.css"/>
+<script type="text/javascript">
+"use strict";
+
+var manager = new MediaTestManager;
+
+function drawVideoToCanvas(v) {
+ let w = v.width,
+ h = v.height,
+ c = document.createElement('canvas');
+ c.width = w;
+ c.height = h;
+ document.body.appendChild(c);
+
+ let gfx = c.getContext('2d');
+ if (!gfx) {
+ throw Error("Unable to obtain context '2d' from canvas");
+ }
+
+ gfx.drawImage(v, 0, 0, w, h);
+}
+
+startTest({
+ desc: 'Test Background Video Is Tainted By drawImage',
+ prefs: [
+ [ "media.test.video-suspend", true ],
+ [ "media.suspend-background-video.enabled", true ],
+ [ "media.suspend-background-video.delay-ms", 1000 ]
+ ],
+ tests: gDecodeSuspendTests,
+ runTest: (test, token) => {
+ ok(true, `${test.name}`);
+ let v = appendVideoToDoc(test.name, token);
+ manager.started(token);
+
+ waitUntilPlaying(v)
+ .then(() => {
+ drawVideoToCanvas(v);
+ ok(v.hasSuspendTaint(), "Video is tainted after drawing to canvas");
+ return checkVideoDoesntSuspend(v);
+ })
+ .then(() => {
+ ok(true, 'Video ended before decode was suspended');
+ manager.finished(token);
+ })
+ .catch((e) => {
+ ok(false, 'Test failed: ' + e.toString());
+ manager.finished(token);
+ });
+ }
+});
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_buffered.html b/dom/media/test/test_buffered.html
new file mode 100644
index 0000000000..86d8eec28a
--- /dev/null
+++ b/dom/media/test/test_buffered.html
@@ -0,0 +1,117 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=462957
+-->
+<head>
+ <title>Test for Bug 462957</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=462957">Mozilla Bug 462957</a>
+
+<p id="display"></p>
+<div id="content" style="display: none">
+
+</div>
+<pre id="test">
+<script type="application/javascript">
+
+// Test for Bug 462957; HTMLMediaElement.buffered.
+
+var manager = new MediaTestManager;
+
+function testBuffered(e) {
+ var v = e.target;
+
+ // The whole media should be buffered...
+ var b = v.buffered;
+ is(b.length, 1, v._name + ": Should be buffered in one range");
+ is(b.start(0), 0, v._name + ": First range start should be media start");
+ ok(Math.abs(b.end(0) - v.duration) < 0.1, v._name + ": First range end should be media end");
+
+ // Ensure INDEX_SIZE_ERR is thrown when we access outside the range
+ var caught = false;
+ try {
+ b.start(-1);
+ } catch (ex) {
+ caught = ex.name == "IndexSizeError" && ex.code == DOMException.INDEX_SIZE_ERR;
+ }
+ is(caught, true, v._name + ": Should throw INDEX_SIZE_ERR on under start bounds range");
+
+ caught = false;
+ try {
+ b.end(-1);
+ } catch (ex) {
+ caught = ex.name == "IndexSizeError" && ex.code == DOMException.INDEX_SIZE_ERR;
+ }
+ is(caught, true, v._name + ": Should throw INDEX_SIZE_ERR on under end bounds range");
+
+ caught = false;
+ try {
+ b.start(b.length);
+ } catch (ex) {
+ caught = ex.name == "IndexSizeError" && ex.code == DOMException.INDEX_SIZE_ERR;
+ }
+ is(caught, true, v._name + ": Should throw INDEX_SIZE_ERR on over start bounds range");
+
+ caught = false;
+ try {
+ b.end(b.length);
+ } catch (ex) {
+ caught = ex.name == "IndexSizeError" && ex.code == DOMException.INDEX_SIZE_ERR;
+ }
+ is(caught, true, v._name + ": Should throw INDEX_SIZE_ERR on over end bounds range");
+
+ removeNodeAndSource(v);
+ manager.finished(v._token);
+}
+
+function fetch(url, fetched_callback) {
+ var xhr = new XMLHttpRequest();
+ xhr.open("GET", url, true);
+ xhr.responseType = "blob";
+
+ var loaded = function (event) {
+ if (xhr.status == 200 || xhr.status == 206) {
+ ok(true, `${url}: Fetch succeeded, status=${xhr.status}`);
+ // Request fulfilled. Note sometimes we get 206... Presumably because either
+ // httpd.js or Necko cached the result.
+ fetched_callback(window.URL.createObjectURL(xhr.response));
+ } else {
+ ok(false, `${url}: Fetch failed, headers=${xhr.getAllResponseHeaders()}`);
+ }
+ };
+
+ xhr.addEventListener("load", loaded);
+ xhr.send();
+}
+
+function startTest(test, token) {
+ // Fetch the media resource using XHR so we can be sure the entire
+ // resource is loaded before we test buffered ranges. This ensures
+ // we have deterministic behaviour.
+ var onfetched = function(uri) {
+ var v = document.createElement('video');
+ v._token = token;
+ v.src = uri;
+ v._name = test.name;
+ v._test = test;
+ v.addEventListener("loadeddata", testBuffered, {once: true});
+ document.body.appendChild(v);
+ };
+
+ manager.started(token);
+ fetch(test.name, onfetched);
+}
+
+// Note: No need to set media test prefs, since we're using XHR to fetch
+// media data.
+manager.runTests(gSeekTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_bug1113600.html b/dom/media/test/test_bug1113600.html
new file mode 100644
index 0000000000..37a9cb0adb
--- /dev/null
+++ b/dom/media/test/test_bug1113600.html
@@ -0,0 +1,50 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test that a video element captured to a stream mid-playback can be played to the end</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+PARALLEL_TESTS = 1;
+SimpleTest.requestCompleteLog();
+var manager = new MediaTestManager;
+
+function startTest(test, token) {
+ var v = document.createElement('video');
+ v.style = "background-color:#aca;";
+ v.width = 160;
+ v.height = 120;
+
+ manager.started(token);
+
+ v.src = test.name;
+
+ v.ontimeupdate = function() {
+ if (v.currentTime < test.duration / 4) {
+ // Allow some time to pass before starting the capture.
+ return;
+ }
+ v.ontimeupdate = null;
+ v.mozCaptureStreamUntilEnded();
+ info(test.name + " capture started at " + v.currentTime + ". Duration=" + test.duration);
+ };
+
+ v.onended = function() {
+ ok(true, test.name + " ended");
+ removeNodeAndSource(v);
+ manager.finished(token);
+ };
+
+ document.body.appendChild(v);
+ v.play();
+}
+
+manager.runTests(gSmallTests, startTest);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_bug1120222.html b/dom/media/test/test_bug1120222.html
new file mode 100644
index 0000000000..69027bc2e3
--- /dev/null
+++ b/dom/media/test/test_bug1120222.html
@@ -0,0 +1,42 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=1120222
+-->
+<head>
+ <title>Test for Bug 1120222</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var manager = new MediaTestManager;
+
+let count = 0;
+
+function startTest(test, token) {
+ manager.started(token);
+ var elemType = /^audio/.test(test.type) ? "audio" : "video";
+ var v = document.createElement(elemType);
+ v.token = token;
+ v.src = test.name;
+
+ v.addEventListener("play", (e) => {
+ document.title = `Test Title ${count++}`;
+ ok(true, "changed title while media was playing")
+ removeNodeAndSource(e.target);
+ manager.finished(e.target.token);
+ });
+
+ document.body.appendChild(v);
+ v.play();
+}
+
+manager.runTests(gSmallTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_bug1242338.html b/dom/media/test/test_bug1242338.html
new file mode 100644
index 0000000000..7b72153a6e
--- /dev/null
+++ b/dom/media/test/test_bug1242338.html
@@ -0,0 +1,66 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test Bug 1242338</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function startTest(test, token) {
+ var video = document.createElement('video');
+ video.preload = "metadata";
+ video.token = token;
+
+ var handler = {
+ "ontimeout": function() {
+ Log(token, "timed out");
+ }
+ };
+ manager.started(token, handler);
+
+ video.src = test.name;
+ video.name = test.name;
+
+ function finish() {
+ video.finished = true;
+ video.removeEventListener("loadedmetadata", onLoadedmetadata);
+ video.removeEventListener("ended", onEnded);
+ removeNodeAndSource(video);
+ manager.finished(video.token);
+ }
+
+ function onLoadedmetadata() {
+ // seek to the media's duration
+ var duration = video.duration;
+ console.log("onloadedmetadata(), duration = " + duration);
+ video.currentTime = duration;
+ }
+
+ function onEnded() {
+ ok(video.ended, test.name + " checking playback has ended");
+ ok(!video.finished, test.name + " shouldn't be finished");
+ ok(!video.seenEnded, test.name + " shouldn't be ended");
+ video.seenEnded = true;
+
+ ok(true, "Seeking to the duration triggers ended event");
+ finish();
+ }
+
+ video.addEventListener("loadedmetadata", onLoadedmetadata);
+ video.addEventListener("ended", onEnded);
+
+ document.body.appendChild(video);
+}
+
+manager.runTests(gSeekTests, startTest);
+
+</script>
+</pre>
+</body>
+</html> \ No newline at end of file
diff --git a/dom/media/test/test_bug1248229.html b/dom/media/test/test_bug1248229.html
new file mode 100644
index 0000000000..3165795622
--- /dev/null
+++ b/dom/media/test/test_bug1248229.html
@@ -0,0 +1,35 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test garbage collection of captured stream (bug 1248229)</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body onload="doTest()">
+<video id="v" src="black100x100-aspect3to2.ogv"></video>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+SimpleTest.waitForExplicitFinish();
+
+function doTest() {
+ /* global v */
+ window.oak = v.mozCaptureStreamUntilEnded();
+ v.mozCaptureStreamUntilEnded();
+ v.play();
+
+ v.onended = function() {
+ info("Got ended.");
+ v.onended = null;
+ SpecialPowers.exactGC(function() {
+ info("GC completed.");
+ v.play();
+ SimpleTest.finish();
+ });
+ }
+}
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_bug1431810_opus_downmix_to_mono.html b/dom/media/test/test_bug1431810_opus_downmix_to_mono.html
new file mode 100644
index 0000000000..647ddf0489
--- /dev/null
+++ b/dom/media/test/test_bug1431810_opus_downmix_to_mono.html
@@ -0,0 +1,139 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: disable phase inversion in opus decoder</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<audio preload=none id="a" controls></audio>
+<audio preload=none id="b" controls></audio>
+<script class="testbody" type="text/javascript">
+/*
+ This test makes use of an (stereo) opus file with phase inversion of 180 degrees (right = -left => right + left = 0).
+ Firstly, the phase inversion is verified on a normal stereo playback.
+ Secondly, mono playback is forced which results in the phase inversion being disabled (Bug 1431810).
+*/
+SimpleTest.waitForExplicitFinish();
+
+/* global a, b */
+
+function areChannelsInverted(b1, b2) {
+ for (var i = 0; i < b1.length; i++) {
+ if (Math.abs(b1[i] + b2[i]) > 9e-2) {
+ return false;
+ }
+ }
+ return true;
+}
+
+function areChannelsEqual(b1, b2) {
+ for (var i = 0; i < b1.length; i++) {
+ if (Math.abs(b1[i] - b2[i]) > 9e-3) {
+ return false;
+ }
+ }
+ return true;
+}
+
+function isSilent(b) {
+ for (var i = 0; i < b.length; i++) {
+ if (b[i] != 0.0) {
+ return false;
+ }
+ }
+ return true;
+}
+
+function mediaElementWithPhaseInversion(audioContext, mediaElement, success) {
+ let audio_source = audioContext.createMediaElementSource(mediaElement);
+ let script_processor = audioContext.createScriptProcessor();
+ audio_source.connect(script_processor);
+
+ mediaElement.onplay = () => {
+ script_processor.onaudioprocess = (e) => {
+ let right = e.inputBuffer.getChannelData(0);
+ let left = e.inputBuffer.getChannelData(1);
+
+ // This is leading or trailing silence
+ // produced by ScriptProcessor.
+ if (isSilent(right) && isSilent(left)) {
+ return;
+ }
+
+ ok(areChannelsInverted(right, left), "Channels must be inverted");
+ }
+ }
+
+ mediaElement.onended = () => {
+ ok(true, "End of file.");
+ mediaElement.onended = null;
+ script_processor.onaudioprocess = null;
+ success();
+ }
+
+ mediaElement.src = "test-stereo-phase-inversion-180.opus";
+ // Normal playback channels will by inverted
+ mediaElement.play();
+}
+
+function mediaElementWithPhaseInversionDisabled(audioContext, mediaElement, success) {
+ let audio_source = audioContext.createMediaElementSource(mediaElement);
+ let script_processor = audioContext.createScriptProcessor();
+ audio_source.connect(script_processor);
+
+ mediaElement.onplay = () => {
+ script_processor.onaudioprocess = (e) => {
+ let right = e.inputBuffer.getChannelData(0);
+ let left = e.inputBuffer.getChannelData(1);
+
+ // This is leading or trailing silence
+ // produced by ScriptProcessor.
+ if (isSilent(right) && isSilent(left)) {
+ return;
+ }
+
+ ok(!areChannelsInverted(right, left), "Channels must not be inverted");
+ ok(areChannelsEqual(right, left), "Channels must be equal");
+ }
+ }
+
+ mediaElement.onended = () => {
+ ok(true, "End of file.");
+ mediaElement.onended = null;
+ script_processor.onaudioprocess = null;
+ success();
+ }
+
+ mediaElement.src = "test-stereo-phase-inversion-180.opus";
+
+ // Downmix to mono will force to disable opus phase inversion
+ SpecialPowers.pushPrefEnv({"set": [["accessibility.monoaudio.enable", true]]})
+ .then(() => {
+ mediaElement.play();
+ });
+}
+
+let ac = new AudioContext();
+
+function testPhaseInversion(mediaElement) {
+ return new Promise((accept, reject) => {
+ mediaElementWithPhaseInversion(ac, a, accept);
+ });
+}
+
+function testPhaseInversionDisabled(mediaElement) {
+ return new Promise((accept, reject) => {
+ mediaElementWithPhaseInversionDisabled(ac, b, accept);
+ });
+}
+
+// Start testing
+testPhaseInversion(a)
+.then( () => testPhaseInversionDisabled(b) )
+.then( () => SimpleTest.finish() )
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_bug1512958.html b/dom/media/test/test_bug1512958.html
new file mode 100644
index 0000000000..2513515542
--- /dev/null
+++ b/dom/media/test/test_bug1512958.html
@@ -0,0 +1,74 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test that pausing and resuming a captured media element with audio doesn't stall</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<audio id="a"></audio>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+function dumpEvent({target, type}) {
+ info(`${target.name} GOT EVENT ${type} currentTime=${target.currentTime} ` +
+ `paused=${target.paused} ended=${target.ended} ` +
+ `readyState=${target.readyState}`);
+}
+
+function wait(ms) {
+ return new Promise(resolve => setTimeout(resolve, ms));
+}
+
+const a = document.getElementById('a');
+
+const events = ["timeupdate", "seeking", "seeked", "ended", "playing", "pause"];
+for (let ev of events) {
+ a.addEventListener(ev, dumpEvent);
+}
+
+(async () => {
+ try {
+ SimpleTest.waitForExplicitFinish();
+ SimpleTest.requestFlakyTimeout("Timeouts for shortcutting test-timeout");
+
+ const test = getPlayableAudio(gTrackTests.filter(t => t.duration > 2));
+ if (!test) {
+ todo(false, "No playable audio");
+ return;
+ }
+
+ // Start playing and capture
+ a.src = test.name;
+ a.name = test.name;
+ const ac = new AudioContext();
+ ac.createMediaElementSource(a);
+ a.play();
+ do {
+ await new Promise(r => a.ontimeupdate = r);
+ } while(a.currentTime == 0)
+
+ // Pause to trigger recreating tracks in DecodedStream
+ a.pause();
+ await new Promise(r => a.onpause = r);
+
+ // Resuming should now work. Bug 1512958 would cause a stall because the
+ // original track wasn't ended and we'd block on it.
+ // See https://bugzilla.mozilla.org/show_bug.cgi?id=1512958#c5
+ a.play();
+ await new Promise(r => a.onplaying = r);
+ a.currentTime = test.duration - 1;
+ await Promise.race([
+ new Promise(res => a.onended = res),
+ wait(30000).then(() => Promise.reject(new Error("Timeout"))),
+ ]);
+ } catch(e) {
+ ok(false, e);
+ } finally {
+ SimpleTest.finish();
+ }
+})();
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_bug1553262.html b/dom/media/test/test_bug1553262.html
new file mode 100644
index 0000000000..19b937f7fe
--- /dev/null
+++ b/dom/media/test/test_bug1553262.html
@@ -0,0 +1,31 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+<meta charset="utf-8">
+<title>Bug 1553262</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<link rel="stylesheet" href="/tests/SimpleTest/test.css"/>
+</head>
+<body>
+<script>
+ const canvas = document.createElement('canvas')
+ const context = canvas.getContext('2d')
+
+ const video = document.createElement('video')
+ const source = new AudioContext().createMediaElementSource(video)
+ const stream = canvas.captureStream()
+
+ const xhr = new XMLHttpRequest()
+
+ context.rect(256, -32768, 16, 16)
+ video.srcObject = stream
+ for (let i = 0; i < 16; i++) {
+ xhr.open('P', '', false)
+ xhr.send()
+ }
+
+ video.srcObject = stream
+ ok(true, "success")
+</script>
+</body>
+</html>
diff --git a/dom/media/test/test_bug448534.html b/dom/media/test/test_bug448534.html
new file mode 100644
index 0000000000..ed1135d85a
--- /dev/null
+++ b/dom/media/test/test_bug448534.html
@@ -0,0 +1,71 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=448534
+-->
+
+<head>
+ <title>Test for Bug 448534</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=448535">Mozilla Bug 448534</a>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function loaded(event) {
+ var v = event.target;
+ info(v.token + ": event=" + event.type);
+ if (v._finished)
+ return;
+ v.play();
+}
+
+function started(event) {
+ var v = event.target;
+ info(v.token + ": event=" + event.type);
+ // For a short file, it could reach the end before 'play' received. We will
+ // skip the test for 'paused' would be true when ended.
+ if (v._finished || v.ended)
+ return;
+ ok(!v.paused, v.token + ": Video should not be paused while playing");
+ v.remove();
+ v._played = true;
+}
+
+function stopped(event) {
+ var v = event.target;
+ info(v.token + ": event=" + event.type);
+ if (v._finished)
+ return;
+ v._finished = true;
+ ok(v.paused, v.token + ": Video should be paused after removing from the Document");
+ removeNodeAndSource(v);
+ manager.finished(v.token);
+}
+
+
+function startTest(test, token) {
+ var v = document.createElement('video');
+ v.preload = "metadata";
+ v.token = token;
+ manager.started(token);
+ v.src = test.name;
+ v._played = false;
+ v._finished = false;
+ v.addEventListener("loadedmetadata", loaded);
+ v.addEventListener("play", started);
+ v.addEventListener("pause", stopped);
+ document.body.appendChild(v);
+}
+
+manager.runTests(gSmallTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_bug463162.xhtml b/dom/media/test/test_bug463162.xhtml
new file mode 100644
index 0000000000..a84b3488d1
--- /dev/null
+++ b/dom/media/test/test_bug463162.xhtml
@@ -0,0 +1,78 @@
+<html xmlns="http://www.w3.org/1999/xhtml"
+ xmlns:html="http://www.w3.org/1999/xhtml"
+ xmlns:svg="http://www.w3.org/2000/svg">
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=463162
+-->
+<head>
+ <title>Test for Bug 463162</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script src="/tests/SimpleTest/EventUtils.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=463162">Mozilla Bug 463162</a>
+
+<script class="testbody" type="text/javascript">
+<![CDATA[
+
+var gExpectedResult = {
+ 'a1' : 'error',
+ 'a2' : 'loaded',
+ 'a3' : 'loaded',
+ 'a4' : 'error',
+};
+
+var gResultCount = 0;
+
+function onError(event, id) {
+ is('error', gExpectedResult[id], 'unexpected error loading ' + id);
+ gResultCount++;
+ dump('error('+id+') expected ' + gExpectedResult[id] + ' gResultCount=' + gResultCount + '\n');
+ if (gResultCount == 4)
+ SimpleTest.finish();
+}
+
+function onMetaData(id) {
+ is('loaded', gExpectedResult[id], 'unexpected loadedmetadata loading ' + id);
+ gResultCount++;
+ dump('onMetaData('+id+') expected ' + gExpectedResult[id] + ' gResultCount=' + gResultCount + '\n');
+ if (gResultCount == 4)
+ SimpleTest.finish();
+}
+
+]]>
+</script>
+
+<video id="a1" preload="metadata" onloadedmetadata="onMetaData('a1');"><sauce/><source type="bad" src="404" onerror="onError(event, 'a1');"/></video>
+<video id="a2" preload="metadata" onloadedmetadata="onMetaData('a2');"><source onerror="onError(event, 'a2');"/></video>
+<video id="a3" preload="metadata" onloadedmetadata="onMetaData('a3');"><html:source onerror="onError(event, 'a3');"/></video>
+<video id="a4" preload="metadata" onloadedmetadata="onMetaData('a4');"><svg:source/><source onerror="onError(event, 'a4');" type="bad" src="404"/></video>
+
+<script class="testbody" type="text/javascript">
+<![CDATA[
+
+function setSource(id, res) {
+ var v = document.getElementById(id);
+ v.firstChild.src = res.name;
+ v.firstChild.type = res.type;
+}
+
+var t = getPlayableVideo(gSmallTests);
+
+setSource('a1', t);
+setSource('a2', t);
+setSource('a3', t);
+setSource('a4', t);
+
+SimpleTest.waitForExplicitFinish();
+
+]]>
+</script>
+
+<pre id="test">
+
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_bug465498.html b/dom/media/test/test_bug465498.html
new file mode 100644
index 0000000000..157a972e77
--- /dev/null
+++ b/dom/media/test/test_bug465498.html
@@ -0,0 +1,83 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: Bug 465498 - Seeking after playback ended</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=465498">Mozilla Bug 465498</a>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function startTest(e) {
+ var v = e.target;
+ info(v._name + " loadedmetadata");
+ e.target.play();
+}
+
+function playbackEnded(e) {
+ var v = e.target;
+ info(v._name + " ended");
+ if (v._finished)
+ return;
+ ok(v.currentTime >= v.duration - 0.1 && v.currentTime <= v.duration + 0.1,
+ "Checking currentTime at end: " + v.currentTime + " for " + v._name);
+ ok(v.ended, "Checking playback has ended for " + v._name);
+ v.pause();
+ v.currentTime = 0;
+ ok(!v.ended, "Checking ended is no longer true for " + v._name);
+ v._seeked = true;
+}
+
+function seekEnded(e) {
+ var v = e.target;
+ info(v._name + " seeked");
+ if (v._finished)
+ return;
+ ok(v.currentTime == 0, "Checking currentTime after seek: " +
+ v.currentTime + " for " + v._name);
+ ok(!v.ended, "Checking ended is false for " + v._name);
+ v._finished = true;
+ removeNodeAndSource(v);
+ manager.finished(v.token);
+}
+
+function seeking(e) {
+ var v = e.target;
+ info(v._name + " seeking");
+}
+
+function initTest(test, token) {
+ var type = getMajorMimeType(test.type);
+ var v = document.createElement(type);
+ if (!v.canPlayType(test.type))
+ return;
+ v.preload = "metadata";
+ v.token = token;
+ manager.started(token);
+ v._name = test.name;
+
+ var s = document.createElement("source");
+ s.type = test.type;
+ s.src = test.name;
+ v.appendChild(s);
+
+ v._seeked = false;
+ v._finished = false;
+ v.addEventListener("loadedmetadata", startTest);
+ v.addEventListener("ended", playbackEnded);
+ v.addEventListener("seeked", seekEnded);
+ v.addEventListener("seeking", seeking);
+ document.body.appendChild(v);
+}
+
+manager.runTests(gSmallTests, initTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_bug495145.html b/dom/media/test/test_bug495145.html
new file mode 100644
index 0000000000..3686e10270
--- /dev/null
+++ b/dom/media/test/test_bug495145.html
@@ -0,0 +1,95 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=495145
+-->
+
+<head>
+ <title>Bug 495145 - pausing while ended shouldn't cause problems</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=495145">Mozilla Bug 495145</a>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function start(e) {
+ e.target.play();
+}
+
+function ended1(e) {
+ var v = e.target;
+ if (v._finished)
+ return;
+
+ ++v._endCount;
+ if (v._endCount == 2) {
+ ok(true, "Playing after pause while ended works for " + v._name);
+ v._finished = true;
+ v.removeEventListener("loadedmetadata", start);
+ v.removeEventListener("ended", ended1);
+ removeNodeAndSource(v);
+ manager.finished(v.token);
+ return;
+ }
+
+ v.pause();
+ v.play();
+}
+
+function ended2(e) {
+ var v = e.target;
+ if (v._finished)
+ return;
+
+ v.pause();
+ v.currentTime = 0;
+}
+
+function seeked2(e) {
+ var v = e.target;
+ if (v._finished)
+ return;
+
+ ok(v.paused, "Paused after seek after pause while ended for " + v._name);
+ v._finished = true;
+ v.removeEventListener("loadedmetadata", start);
+ v.removeEventListener("ended", ended2);
+ v.removeEventListener("seeked", seeked2);
+ removeNodeAndSource(v);
+ manager.finished(v.token);
+}
+
+function createVideo(test, x, token) {
+ var v = document.createElement('video');
+ v.preload = "metadata";
+ v.token = token;
+ manager.started(token);
+ v.src = test.name;
+ v._name = test.name + "#" + x;
+ v._endCount = 0;
+ v._finished = false;
+ v.addEventListener("loadedmetadata", start);
+ v.addEventListener("ended", x == 1 ? ended1 : ended2);
+ if (x == 2)
+ v.addEventListener("seeked", seeked2);
+ document.body.appendChild(v);
+}
+
+function startTest(test, token) {
+ createVideo(test, 1, token + "a");
+ createVideo(test, 2, token + "b");
+}
+
+manager.runTests(gSmallTests, startTest);
+
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_bug495300.html b/dom/media/test/test_bug495300.html
new file mode 100644
index 0000000000..bf3b921402
--- /dev/null
+++ b/dom/media/test/test_bug495300.html
@@ -0,0 +1,63 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=495300
+-->
+
+<head>
+ <title>Bug 495300 - seeking to the end should behave as "ended"</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=495300">Mozilla Bug 495300</a>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function filename(uri) {
+ return uri.substr(uri.lastIndexOf("/")+1);
+}
+
+function mediaEnded(event) {
+ ok(true, "Got expected 'ended' event: " + filename(event.target.currentSrc));
+
+ if (event.target._expectedDuration)
+ ok(Math.abs(event.target.currentTime - event.target._expectedDuration) < 0.1,
+ "currentTime equals duration: " + filename(event.target.currentSrc));
+
+ event.target.removeEventListener("ended", mediaEnded);
+ manager.finished(event.target.token);
+ removeNodeAndSource(event.target);
+}
+
+function mediaLoadedmetadata(event) {
+ event.target.currentTime = event.target.duration;
+ event.target.removeEventListener("loadedmetadata", mediaLoadedmetadata);
+}
+
+function startTest(test, token) {
+ var elemType = /^audio/.test(test.type) ? "audio" : "video";
+ var v1 = document.createElement(elemType);
+ v1.preload = "auto";
+
+ v1.src = test.name;
+ if (test.duration) {
+ v1._expectedDuration = test.duration;
+ }
+ v1.addEventListener("loadedmetadata", mediaLoadedmetadata);
+ v1.addEventListener("ended", mediaEnded);
+ v1.load();
+
+ v1.token = token;
+ manager.started(token);
+}
+
+manager.runTests(gSeekTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_bug654550.html b/dom/media/test/test_bug654550.html
new file mode 100644
index 0000000000..f57afae2b4
--- /dev/null
+++ b/dom/media/test/test_bug654550.html
@@ -0,0 +1,84 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=654550
+-->
+
+<head>
+ <title>Test for Bug 654550</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=654550">Mozilla Bug 654550</a>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+ /* Test for Bug 654550 */
+
+ // Parallel test must be disabled for media.video_stats.enabled is a global setting
+ // to prevent the setting from changing unexpectedly in the middle of the test.
+ PARALLEL_TESTS = 1;
+ SimpleTest.waitForExplicitFinish();
+ var manager = new MediaTestManager;
+
+ function checkStats(v, aShouldBeEnabled) {
+ if (aShouldBeEnabled) {
+ ok(v.mozParsedFrames != 0,
+ "At least one value should be different from 0 if stats are enabled");
+ } else {
+ ok(!v.mozParsedFrames,
+ "mozParsedFrames should be 0 if stats are disabled");
+ ok(!v.mozDecodedFrames,
+ "mozDecodedFrames should be 0 if stats are disabled");
+ ok(!v.mozPresentedFrames,
+ "mozPresentedFrames should be 0 if stats are disabled");
+ ok(!v.mozPaintedFrames,
+ "mozPaintedFrames should be 0 if stats are disabled");
+ }
+
+ }
+
+ function ontimeupdate_statsEnabled(event) {
+ var v = event.target;
+ v.removeEventListener('timeupdate', ontimeupdate_statsEnabled);
+ checkStats(v, true);
+ SpecialPowers.popPrefEnv(
+ function() {
+ v.addEventListener("timeupdate", ontimeupdate_statsDisabled);
+ });
+ }
+
+ function ontimeupdate_statsDisabled(event) {
+ var v = event.target;
+ v.removeEventListener('timeupdate', ontimeupdate_statsDisabled);
+ checkStats(v, false);
+ removeNodeAndSource(v);
+ manager.finished(v.token);
+ }
+
+ function startTest(test, token) {
+ var v = document.createElement('video');
+ v.token = token;
+ v.src = test.name;
+ // playback may reach the end before pref is changed for the duration is short
+ // set 'loop' to true to keep playing so that we won't miss 'timeupdate' events
+ v.loop = true;
+ manager.started(token);
+ SpecialPowers.pushPrefEnv({"set": [["media.video_stats.enabled", true]]},
+ function() {
+ v.play();
+ v.addEventListener("timeupdate", ontimeupdate_statsEnabled);
+ });
+ }
+
+ SpecialPowers.pushPrefEnv({"set": [["media.video_stats.enabled", false]]},
+ function() {
+ manager.runTests(gVideoTests, startTest);
+ });
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_bug686942.html b/dom/media/test/test_bug686942.html
new file mode 100644
index 0000000000..66b48d69f0
--- /dev/null
+++ b/dom/media/test/test_bug686942.html
@@ -0,0 +1,68 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=686942
+-->
+
+<head>
+ <title>Test for Bug 448534</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=686942">Mozilla Bug 686942</a>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function onloaded(event) {
+ var v = event.target;
+ v.removeEventListener("loadedmetadata", onloaded);
+ v.currentTime = v.duration;
+
+}
+
+function checkNotPlaying(v) {
+ ok(v.currentTime == 0, "Should not be playing after seek to end and back to beginning");
+ v._finished = true;
+ manager.finished(v.token);
+ removeNodeAndSource(v);
+}
+
+function onseeked(event) {
+ var v = event.target;
+ v.removeEventListener("seeked", onseeked);
+ setTimeout(function() { checkNotPlaying(v); }, 500);
+}
+
+function onended(event) {
+ var v = event.target;
+ v.removeEventListener("ended", onended);
+ if (v._finished)
+ return;
+ v.addEventListener("seeked", onseeked);
+ v.currentTime = 0;
+}
+
+function startTest(test, token) {
+ var v = document.createElement('video');
+ v.preload = "auto";
+ v.token = token;
+ manager.started(token);
+ v.src = test.name;
+ v._played = false;
+ v._finished = false;
+ v.addEventListener("loadedmetadata", onloaded);
+ v.addEventListener("ended", onended);
+
+ document.body.appendChild(v);
+}
+
+manager.runTests(gSmallTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_bug726904.html b/dom/media/test/test_bug726904.html
new file mode 100644
index 0000000000..bf8e2536ca
--- /dev/null
+++ b/dom/media/test/test_bug726904.html
@@ -0,0 +1,56 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=726904
+-->
+
+<head>
+ <title>Media test: default video size</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body onload="bodyLoaded();">
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=726904">Mozilla Bug 726904</a>
+
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+SimpleTest.waitForExplicitFinish();
+
+var v1 = document.createElement("video"),
+ v2 = document.createElement("video"),
+ poster = "",
+ resource = getPlayableVideo(gSmallTests);
+
+function bodyLoaded(){
+ // Note: For DASH, width and height would vary once the video started playing, so
+ // the values would not correlate with those in manifest.js. Since this test has
+ // no playing, this should not affect the result.
+ is(v1.videoWidth, resource.width, "Intrinsic width should match video width");
+ is(v1.videoHeight, resource.height, "Intrinsic height should match video height");
+ is(v2.clientWidth, 400, "clientWidth should be 400");
+ is(v2.clientHeight, 400, "clientHeight should be 400");
+ SimpleTest.finish();
+}
+
+if (resource) {
+ v1.poster = v2.poster = poster;
+
+ v1.src = v2.src = "http://mochi.test:8888/tests/dom/media/test/" + resource.name;
+
+ v1.preload = "auto";
+ v2.preload = "none";
+
+ v1.muted = v2.muted = true;
+
+ document.body.appendChild(v1);
+ document.body.appendChild(v2);
+} else {
+ todo(false, "No types supported");
+}
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_bug874897.html b/dom/media/test/test_bug874897.html
new file mode 100644
index 0000000000..7801487fe3
--- /dev/null
+++ b/dom/media/test/test_bug874897.html
@@ -0,0 +1,68 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=874897
+-->
+
+<head>
+ <title>Test for Bug 874897</title>
+ <script type="text/javascript" src="/MochiKit/MochiKit.js"></script>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function loadeddata(e) {
+ var v = e.target;
+ ok(v.readyState >= v.HAVE_CURRENT_DATA,
+ "readyState must be >= HAVE_CURRENT_DATA for " + v._name);
+
+ var canvas = document.createElement("canvas");
+ canvas.width = 210;
+ canvas.height = 120;
+ document.body.appendChild(canvas);
+ var ctx = canvas.getContext("2d");
+ try {
+ ctx.drawImage(v, 0, 0, v.videoWidth, v.videoHeight, 0, 0, canvas.width, canvas.height);
+ ok(true, "Shouldn't throw exception while drawing to canvas from video for " + v._name);
+ } catch (ex) {
+ ok(false, "Shouldn't throw exception while drawing to canvas from video for " + v._name);
+ }
+
+ v._finished = true;
+ v.remove();
+ manager.finished(v.token);
+}
+
+function startTest(test, token) {
+ var type = getMajorMimeType(test.type);
+ if (type != "video")
+ return;
+
+ var v = document.createElement('video');
+ v.token = token;
+ manager.started(token);
+ v.src = test.name;
+ v._name = test.name;
+ v._finished = false;
+ v.autoplay = true;
+ v.style.display = "none";
+ v.addEventListener("loadeddata", loadeddata);
+ document.body.appendChild(v);
+}
+
+SimpleTest.waitForExplicitFinish();
+SpecialPowers.pushPrefEnv({"set": [["media.cache_size", 40000]]}, beginTest);
+function beginTest() {
+ manager.runTests(gAspectRatioTests, startTest);
+}
+
+</script>
+</pre>
+
+</body>
+</html>
diff --git a/dom/media/test/test_bug879717.html b/dom/media/test/test_bug879717.html
new file mode 100644
index 0000000000..c669773d3f
--- /dev/null
+++ b/dom/media/test/test_bug879717.html
@@ -0,0 +1,130 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test for bug 879717, check that a video element can be drawn into a canvas at various states of playback</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+var canvas = document.createElement('canvas');
+document.body.appendChild(canvas);
+
+var checkDrawImage = function(eventName, videoElement) {
+ var exception = null;
+ var exceptionName = "nothing";
+ try {
+ var ctx = canvas.getContext('2d');
+ ctx.drawImage(videoElement, 0, 0, canvas.width, canvas.height);
+ } catch (e) {
+ exception = e;
+ exceptionName = e.name;
+ }
+ ok(exception === null,
+ "drawImage shouldn't throw an exception on " + eventName +
+ " of " + videoElement.testName + ", got " + exceptionName);
+};
+
+var checkDrawImageEventHandler = function(ev) {
+ checkDrawImage(ev.type, ev.target);
+};
+var startTest = function(media, token) {
+ manager.started(token);
+
+ // File playback
+ var v1 = document.createElement("video");
+ v1.autoplay = true;
+
+ // Captured file playback
+ var v2 = document.createElement("video");
+
+ // Stream playback
+ var v3 = document.createElement("video");
+ v3.autoplay = true;
+
+ v1.gotLoadeddata = false;
+ v2.gotLoadeddata = false;
+ v3.gotLoadeddata = false;
+
+ v1.testName = "v1 (" + media.name + ")";
+ v2.testName = "v2 (Captured " + media.name + ")";
+ v3.testName = "v3 (Stream of " + media.name + ")";
+
+ checkDrawImage("beforeplay", v1);
+ checkDrawImage("beforeplay", v2);
+ checkDrawImage("beforeplay", v3);
+
+ v1.onloadedmetadata = checkDrawImageEventHandler;
+ v2.onloadedmetadata = checkDrawImageEventHandler;
+ v3.onloadedmetadata = checkDrawImageEventHandler;
+
+ v1.onplay = checkDrawImageEventHandler;
+ v2.onplay = checkDrawImageEventHandler;
+ v3.onplay = checkDrawImageEventHandler;
+
+ function onplaying(ev) {
+ if (!ev.target.gotPlaying) {
+ ev.target.gotPlaying = true;
+ checkDrawImageEventHandler(ev);
+ }
+ }
+ v1.onplaying = onplaying;
+ v2.onplaying = onplaying;
+ v3.onplaying = onplaying;
+
+ var onloadeddata = function(ev) {
+ ev.target.gotLoadeddata = true;
+ checkDrawImageEventHandler(ev);
+ };
+
+ v1.onloadeddata = onloadeddata;
+ v2.onloadeddata = onloadeddata;
+ v3.onloadeddata = onloadeddata;
+
+ var checkFinished = function() {
+ if (!v1.testFinished || !v2.testFinished || !v3.testFinished) {
+ return;
+ }
+
+ ok(v1.gotLoadeddata, v1.testName + " should have gotten the 'loadeddata' event callback");
+ ok(v2.gotLoadeddata, v2.testName + " should have gotten the 'loadeddata' event callback");
+ ok(v3.gotLoadeddata, v3.testName + " should have gotten the 'loadeddata' event callback");
+
+ manager.finished(token);
+ };
+
+ var onended = function(ev) {
+ checkDrawImageEventHandler(ev);
+ removeNodeAndSource(ev.target);
+ ev.target.testFinished = true;
+ checkFinished();
+ };
+
+ v1.onended = onended;
+ v2.onended = onended;
+ v3.onended = onended;
+
+ document.body.appendChild(v1);
+ document.body.appendChild(v2);
+ document.body.appendChild(v3);
+
+ v1.src = media.name;
+ v2.src = media.name;
+ v2.preload = 'metadata';
+
+ v2.addEventListener('loadedmetadata', function () {
+ v3.srcObject = v2.mozCaptureStreamUntilEnded();
+ v2.play();
+ });
+}
+
+manager.runTests(getPlayableVideos(gSmallTests), startTest);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_bug895305.html b/dom/media/test/test_bug895305.html
new file mode 100644
index 0000000000..56ed3c775c
--- /dev/null
+++ b/dom/media/test/test_bug895305.html
@@ -0,0 +1,42 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=895305
+https://bugzilla.mozilla.org/show_bug.cgi?id=905320
+-->
+<head>
+ <meta charset='utf-8'>
+ <title>Regression test for bug 895305 and 905320 - TextTrack* leaks</title>
+ <script type="text/javascript" src="/MochiKit/MochiKit.js"></script>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+</head>
+<body>
+<p id="display"></p>
+<div id="content" style="display: none">
+</div>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+SimpleTest.waitForExplicitFinish();
+
+var audio = document.createElement("audio");
+
+// Check leaking on TextTrackList objects.
+/* global ttl, ttc */
+window.ttl = audio.textTracks;
+ttl.addEventListener("click", function(){});
+
+// Check leaking on VTTCue objects.
+window.ttc = new VTTCue(3, 4, "Test.");
+ttc.addEventListener("click", function() {});
+
+// Check leaking on TextTrack objects.
+audio.addTextTrack("subtitles", "label", "en-CA");
+ttl[0].addEventListener("click", function() {});
+
+ok(true); // Need to have at least one assertion for Mochitest to be happy.
+SimpleTest.finish();
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_bug919265.html b/dom/media/test/test_bug919265.html
new file mode 100644
index 0000000000..40eff135e4
--- /dev/null
+++ b/dom/media/test/test_bug919265.html
@@ -0,0 +1,30 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=919265
+-->
+<head>
+ <meta charset='utf-8'>
+ <title>Regression test for bug 919265 - Leak on VTTCue::GetCueAsHTML()</title>
+ <script type="text/javascript" src="/MochiKit/MochiKit.js"></script>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+</head>
+<body>
+<p id="display"></p>
+<div id="content" style="display: none">
+</div>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+SimpleTest.waitForExplicitFinish();
+
+// We shouldn't leak upon shutdown.
+(new VTTCue(0, 0, "")).getCueAsHTML();
+
+// We need to assert something for Mochitest to be happy.
+ok(true);
+SimpleTest.finish();
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_can_play_type.html b/dom/media/test/test_can_play_type.html
new file mode 100644
index 0000000000..a28d137783
--- /dev/null
+++ b/dom/media/test/test_can_play_type.html
@@ -0,0 +1,40 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=469247
+-->
+<head>
+ <title>Test for Bug 469247</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=469247">Mozill
+a Bug 469247</a>
+<p id="display"></p>
+<div id="content" style="display: none">
+</div>
+
+<video id="v"></video>
+
+<pre id="test">
+<script type="application/javascript">
+
+var v = document.getElementById('v');
+
+function check(type, expected) {
+ is(v.canPlayType(type), expected, type);
+}
+
+// Invalid types
+check("foo/bar", "");
+check("", "");
+check("!!!", "");
+
+mediaTestCleanup();
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_can_play_type_mpeg.html b/dom/media/test/test_can_play_type_mpeg.html
new file mode 100644
index 0000000000..40a1225922
--- /dev/null
+++ b/dom/media/test/test_can_play_type_mpeg.html
@@ -0,0 +1,166 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=799315
+-->
+<head>
+ <title>Test for MP4 and MP3 support</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<p id="display"></p>
+<div id="content" style="display: none">
+</div>
+
+<video id="v"></video>
+
+<pre id="test">
+<script>
+
+function check_mp4(v, enabled) {
+ function check(type, expected) {
+ var ex = enabled ? expected : "";
+ is(v.canPlayType(type), ex, type + "='" + ex + "'");
+ }
+
+ check("video/mp4", "maybe");
+ check("video/x-m4v", "maybe");
+ check("audio/mp4", "maybe");
+ check("audio/x-m4a", "maybe");
+
+ // Not the MIME type that other browsers respond to, so we won't either.
+ check("audio/m4a", "");
+ check("video/m4v", "");
+
+ check("audio/aac", "maybe");
+ check("audio/aacp", "maybe");
+
+ // H.264 Constrained Baseline Profile Level 3.0, AAC-LC
+ check("video/mp4; codecs=\"avc1.42E01E, mp4a.40.2\"", "probably");
+
+ // H.264 Constrained Baseline Profile Level 3.0, mp3
+ check("video/mp4; codecs=\"avc1.42E01E, mp3\"", "probably");
+
+ check("video/mp4; codecs=\"avc1.42001E, mp4a.40.2\"", "probably");
+ check("video/mp4; codecs=\"avc1.58A01E, mp4a.40.2\"", "probably");
+
+ // H.264 Main Profile Level 3.0, AAC-LC
+ check("video/mp4; codecs=\"avc1.4D401E, mp4a.40.2\"", "probably");
+ // H.264 Main Profile Level 3.1, AAC-LC
+ check("video/mp4; codecs=\"avc1.4D401F, mp4a.40.2\"", "probably");
+ // H.264 Main Profile Level 4.0, AAC-LC
+ check("video/mp4; codecs=\"avc1.4D4028, mp4a.40.2\"", "probably");
+ // H.264 High Profile Level 3.0, AAC-LC
+ check("video/mp4; codecs=\"avc1.64001E, mp4a.40.2\"", "probably");
+ // H.264 High Profile Level 3.1, AAC-LC
+ check("video/mp4; codecs=\"avc1.64001F, mp4a.40.2\"", "probably");
+
+ check("video/mp4; codecs=\"avc1.42E01E\"", "probably");
+ check("video/mp4; codecs=\"avc1.42001E\"", "probably");
+ check("video/mp4; codecs=\"avc1.58A01E\"", "probably");
+ check("video/mp4; codecs=\"avc1.4D401E\"", "probably");
+ check("video/mp4; codecs=\"avc1.64001F\"", "probably");
+
+ // AAC-LC
+ check("audio/mp4; codecs=\"mp4a.40.2\"", "probably");
+ check("audio/mp4; codecs=mp4a.40.2", "probably");
+ check("audio/x-m4a; codecs=\"mp4a.40.2\"", "probably");
+ check("audio/x-m4a; codecs=mp4a.40.2", "probably");
+
+ check("audio/mp4; codecs=\"mp4a.40.2,\"", ""); // Invalid codecs string
+
+ // HE-AAC v1
+ check("audio/mp4; codecs=\"mp4a.40.5\"", "probably");
+ check("audio/mp4; codecs=mp4a.40.5", "probably");
+ check("audio/x-m4a; codecs=\"mp4a.40.5\"", "probably");
+ check("audio/x-m4a; codecs=mp4a.40.5", "probably");
+ // HE-AAC v2
+ check("audio/mp4; codecs=\"mp4a.40.29\"", "probably");
+
+ // Opus
+ check("audio/mp4; codecs=\"opus\"", "probably");
+ check("audio/mp4; codecs=opus", "probably");
+
+ // Flac.
+ var haveFlac = getPref("media.flac.enabled");
+ check("audio/mp4; codecs=\"flac\"", haveFlac ? "probably" : "");
+ check("audio/mp4; codecs=flac", haveFlac ? "probably" : "");
+
+ // VP9.
+ [ "video/mp4; codecs=vp9",
+ "video/mp4; codecs=\"vp9\"",
+ "video/mp4; codecs=\"vp9.0\""
+ ].forEach((codec) => {
+ // canPlayType should support VP9 in MP4...
+ check(codec, "probably");
+ if (!IsSupportedAndroid()) {
+ // VP9 codec is disabled on Android devices with no HW decoder. So skip it
+ // on Android for now.
+ ok(MediaSource.isTypeSupported(codec), "VP9 in MP4 should be supported in MSE");
+ }
+ });
+
+ var haveAV1 = getPref("media.av1.enabled");
+ check("video/mp4; codecs=\"av1\"", haveAV1 ? "probably" : "");
+}
+
+function check_mp3(v, enabled) {
+ function check(type, expected) {
+ var ex = enabled ? expected : "";
+ is(v.canPlayType(type), ex, type + "='" + ex + "'");
+ }
+
+ check("audio/mpeg", "maybe");
+ check("audio/mp3", "maybe");
+
+ check("audio/mpeg; codecs=\"mp3\"", "probably");
+ check("audio/mpeg; codecs=mp3", "probably");
+
+ check("audio/mp3; codecs=\"mp3\"", "probably");
+ check("audio/mp3; codecs=mp3", "probably");
+}
+
+function IsMacOS() {
+ return navigator.userAgent.includes("Mac OS");
+}
+
+function IsLinux() {
+ return navigator.userAgent.includes("Linux");
+}
+
+function getPref(name) {
+ return SpecialPowers.getBoolPref(name, false);
+}
+
+function IsSupportedAndroid() {
+ return getAndroidVersion() >= 14;
+}
+
+function IsJellyBeanOrLater() {
+ return getAndroidVersion() >= 16;
+}
+
+var haveMp4 =
+ getPref("media.wmf.enabled") ||
+ IsMacOS() ||
+ IsSupportedAndroid() ||
+ (IsLinux() && getPref("media.ffmpeg.enabled"));
+
+check_mp4(document.getElementById('v'), haveMp4);
+
+var haveMp3 =
+ getPref("media.wmf.enabled") ||
+ (IsLinux() && getPref("media.ffmpeg.enabled")) ||
+ (IsSupportedAndroid() &&
+ ((IsJellyBeanOrLater() && getPref("media.android-media-codec.enabled")))) ||
+ IsMacOS();
+
+check_mp3(document.getElementById('v'), haveMp3);
+
+mediaTestCleanup();
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_can_play_type_no_ogg.html b/dom/media/test/test_can_play_type_no_ogg.html
new file mode 100644
index 0000000000..2e63f191d2
--- /dev/null
+++ b/dom/media/test/test_can_play_type_no_ogg.html
@@ -0,0 +1,42 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=469247
+-->
+<head>
+ <title>Test for Bug 469247: Ogg backend disabled</title>
+ <script type="application/javascript" src="/MochiKit/MochiKit.js"></script>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=469247">Mozill
+a Bug 469247</a>
+<p id="display"></p>
+<div id="content" style="display: none">
+</div>
+
+<video id="v"></video>
+
+<pre id="test">
+<script src="can_play_type_ogg.js"></script>
+<script>
+
+SimpleTest.waitForExplicitFinish();
+
+function finish() {
+ mediaTestCleanup();
+ SimpleTest.finish();
+}
+
+SpecialPowers.pushPrefEnv({"set": [["media.ogg.enabled", false]]},
+ function() {
+ check_ogg(document.getElementById('v'), false, finish);
+ }
+);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_can_play_type_ogg.html b/dom/media/test/test_can_play_type_ogg.html
new file mode 100644
index 0000000000..3e44f8ba8c
--- /dev/null
+++ b/dom/media/test/test_can_play_type_ogg.html
@@ -0,0 +1,37 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=469247
+-->
+<head>
+ <title>Test for Bug 469247: Ogg backend</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=469247">Mozill
+a Bug 469247</a>
+<p id="display"></p>
+<div id="content" style="display: none">
+</div>
+
+<video id="v"></video>
+
+<pre id="test">
+<script src="can_play_type_ogg.js"></script>
+<script>
+
+SimpleTest.waitForExplicitFinish();
+
+function finish() {
+ mediaTestCleanup();
+ SimpleTest.finish();
+}
+
+check_ogg(document.getElementById('v'), true, finish);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_can_play_type_wave.html b/dom/media/test/test_can_play_type_wave.html
new file mode 100644
index 0000000000..e6d4e29d86
--- /dev/null
+++ b/dom/media/test/test_can_play_type_wave.html
@@ -0,0 +1,30 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=469247
+-->
+<head>
+ <title>Test for Bug 469247: WAVE backend</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=469247">Mozill
+a Bug 469247</a>
+<p id="display"></p>
+<div id="content" style="display: none">
+</div>
+
+<video id="v"></video>
+
+<pre id="test">
+<script src="can_play_type_wave.js"></script>
+<script>
+check_wave(document.getElementById('v'), true);
+
+mediaTestCleanup();
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_can_play_type_webm.html b/dom/media/test/test_can_play_type_webm.html
new file mode 100644
index 0000000000..6b0e2adfad
--- /dev/null
+++ b/dom/media/test/test_can_play_type_webm.html
@@ -0,0 +1,39 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=566245
+-->
+<head>
+ <title>Test for Bug 566245: WebM backend</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=566245">Mozill
+a Bug 566245</a>
+<p id="display"></p>
+<div id="content" style="display: none">
+</div>
+
+<video id="v"></video>
+
+<pre id="test">
+<script src="can_play_type_webm.js"></script>
+<script>
+ async function runTest() {
+ try {
+ await check_webm(document.getElementById('v'), true);
+ mediaTestCleanup();
+ } catch (e) {
+ info("Exception " + e.message);
+ ok(false, "Threw exception " + e.message);
+ }
+ SimpleTest.finish();
+ }
+ SimpleTest.waitForExplicitFinish();
+ addLoadEvent(runTest);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_capture_stream_av_sync.html b/dom/media/test/test_capture_stream_av_sync.html
new file mode 100644
index 0000000000..b5754c683f
--- /dev/null
+++ b/dom/media/test/test_capture_stream_av_sync.html
@@ -0,0 +1,276 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+<title>A/V sync test for stream capturing</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+<p>Following canvas will capture and show the video frame when the video becomes audible</p><br>
+<canvas id="canvas" width="640" height="480"></canvas>
+<script type="application/javascript">
+
+/**
+ * This test will capture stream before the video starts playing, and check if
+ * A/V sync will keep sync during playing.
+ */
+add_task(async function testAVSyncForStreamCapturing() {
+ createVideo();
+ captureStreamFromVideo();
+ await playMedia();
+ await testAVSync();
+ destroyVideo();
+});
+
+/**
+ * This test will check if A/V is still on sync after we switch the media sink
+ * from playback-based sink to mediatrack-based sink.
+ */
+add_task(async function testAVSyncWhenSwitchingMediaSink() {
+ createVideo();
+ await playMedia({resolveAfterReceivingTimeupdate : 5});
+ captureStreamFromVideo();
+ await testAVSync();
+ destroyVideo();
+});
+
+/**
+ * This test will check if A/V is still on sync after we change the playback
+ * rate on the captured stream.
+ */
+add_task(async function testAVSyncWhenChangingPlaybackRate() {
+ createVideo();
+ captureStreamFromVideo();
+ await playMedia();
+ const playbackRates = [0.25, 0.5, 1.0, 1.5, 2.0];
+ for (let rate of playbackRates) {
+ setPlaybackRate(rate);
+ // TODO : when playback rate set to 1.5+x, the A/V will become less stable
+ // in testing so we raise the fuzzy frames for that, but also increase the
+ // test times. As at that speed, precise A/V becomes trivial because we
+ // can't really tell the difference. But it would be good for us to
+ // investigate if we could make A/V sync work better at that extreme high
+ // rate.
+ if (rate >= 1.5) {
+ await testAVSync({ expectedAVSyncTestTimes : 4, fuzzyFrames : 10});
+ } else {
+ await testAVSync({ expectedAVSyncTestTimes : 2 });
+ }
+ }
+ destroyVideo();
+});
+
+/**
+ * Following are helper functions
+ */
+const DEBUG = false;
+function info_debug(msg) {
+ if (DEBUG) {
+ info(msg);
+ }
+}
+
+function createVideo() {
+ const video = document.createElement("video");
+ // This video is special for testing A/V sync, it only produce audible sound
+ // once per second, and when the sound comes out, you can check the position
+ // of the square to know if the A/V keeps sync.
+ video.src = "sync.webm";
+ video.loop = true;
+ video.controls = true;
+ video.width = 640;
+ video.height = 480;
+ video.id = "video";
+ document.body.appendChild(video);
+}
+
+function destroyVideo() {
+ const video = document.getElementById("video");
+ video.src = null;
+ video.remove();
+}
+
+async function playMedia({ resolveAfterReceivingTimeupdate } = {}) {
+ const video = document.getElementById("video");
+ ok(await video.play().then(_=>true,_=>false), "video started playing");
+ if (resolveAfterReceivingTimeupdate > 0) {
+ // Play it for a while to ensure the clock growing on the normal audio sink.
+ for (let idx = 0; idx < resolveAfterReceivingTimeupdate; idx++) {
+ await new Promise(r => video.ontimeupdate = r);
+ }
+ }
+}
+
+async function captureStreamFromVideo() {
+ const video = document.getElementById("video");
+ let ac = new AudioContext();
+ let analyser = ac.createAnalyser();
+ analyser.frequencyBuf = new Float32Array(analyser.frequencyBinCount);
+ analyser.smoothingTimeConstant = 0;
+ analyser.fftSize = 2048; // 1024 bins
+ let sourceNode = ac.createMediaElementSource(video);
+ sourceNode.connect(analyser);
+ analyser.connect(ac.destination);
+ video.analyser = analyser;
+}
+
+// This method will capture the stream from the video element, and check if A/V
+// keeps sync during capturing. `callback` parameter will be executed after
+// finishing capturing.
+// @param [optional] expectedAVSyncTestTimes
+// The amount of times that A/V sync test performs.
+// @param [optional] fuzzyFrames
+// This will fuzz the result from +0 (perfect sync) to -X to +X frames.
+async function testAVSync({ expectedAVSyncTestTimes = 5, fuzzyFrames = 5} = {}) {
+ return new Promise(r => {
+ const analyser = document.getElementById("video").analyser;
+ let testIdx = 0;
+ let hasDetectedAudibleFrame = false;
+ // As we only want to detect the audible frame at the first moment when
+ // sound becomes audible, so we always skip the first audible frame because
+ // it might not be the start, but the tail part (where audio is being
+ // decaying to silence) when we start detecting.
+ let hasSkippedFirstFrame = false;
+ analyser.notifyAnalysis = () => {
+ let {frequencyBuf} = analyser;
+ analyser.getFloatFrequencyData(frequencyBuf);
+ if (checkIfBufferIsSilent(frequencyBuf)) {
+ info_debug("no need to paint the silent frame");
+ hasDetectedAudibleFrame = false;
+ requestAnimationFrame(analyser.notifyAnalysis);
+ return;
+ }
+ if (hasDetectedAudibleFrame) {
+ info_debug("detected audible frame already");
+ requestAnimationFrame(analyser.notifyAnalysis);
+ return;
+ }
+ hasDetectedAudibleFrame = true;
+ if (!hasSkippedFirstFrame) {
+ info("skip the first audible frame");
+ hasSkippedFirstFrame = true;
+ requestAnimationFrame(analyser.notifyAnalysis);
+ return;
+ }
+ const video = document.getElementById("video");
+ info(`paint audible frame`);
+ const cvs = document.getElementById("canvas");
+ let context = cvs.getContext('2d');
+ context.drawImage(video, 0, 0, 640, 480);
+ if (checkIfAVIsOnSyncFuzzy(context, fuzzyFrames)) {
+ ok(true, `test ${testIdx++} times, a/v is in sync!`);
+ } else {
+ ok(false, `test ${testIdx++} times, a/v is out of sync!`);
+ }
+ if (testIdx == expectedAVSyncTestTimes) {
+ r();
+ return;
+ }
+ requestAnimationFrame(analyser.notifyAnalysis);
+ }
+ analyser.notifyAnalysis();
+ });
+}
+
+function checkIfBufferIsSilent(buffer) {
+ for (let data of buffer) {
+ // when sound is audible, its values are around -200 and the silence values
+ // are around -800.
+ if (data > -200) {
+ return false;
+ }
+ }
+ return true;
+}
+
+// This function will check the pixel data from the `context` to see if the
+// square appears in the right place. As we can't control the exact timing
+// of rendering video frames in the compositor, so the result would be fuzzy.
+function checkIfAVIsOnSyncFuzzy(context, fuzzyFrames) {
+ const squareLength = 48;
+ // Canvas is 640*480, so perfect sync is the left-top corner when the square
+ // shows up in the middle.
+ const perfectSync =
+ { x: 320 - squareLength/2.0 ,
+ y: 240 - squareLength/2.0 };
+ let isAVSyncFuzzy = false;
+ // Get the whole partial section of image and detect where the square is.
+ let imageData = context.getImageData(0, perfectSync.y, 640, squareLength);
+ for (let i = 0; i < imageData.data.length; i += 4) {
+ // If the pixel's color is red, then this position will be the left-top
+ // corner of the square.
+ if (isPixelColorRed(imageData.data[i], imageData.data[i+1],
+ imageData.data[i+2])) {
+ const pos = ImageIdxToRelativeCoordinate(imageData, i);
+ let diff = calculateFrameDiffenceInXAxis(pos.x, perfectSync.x);
+ info(`find the square in diff=${diff}`);
+ // Maybe we check A/V sync too early or too late, try to adjust the diff
+ // to guess the previous correct position where the square should be.
+ if (diff > fuzzyFrames) {
+ diff = adjustFrameDiffBasedOnMediaTime(diff);
+ const video = document.getElementById("video");
+ info(`adjusted diff to ${diff} (time=${video.currentTime})`);
+ }
+ if (diff <= fuzzyFrames) {
+ isAVSyncFuzzy = true;
+ }
+ context.putImageData(imageData, 0, 0);
+ break;
+ }
+ }
+ if (!isAVSyncFuzzy) {
+ const ctx = document.getElementById('canvas');
+ info(ctx.toDataURL());
+ }
+ return isAVSyncFuzzy;
+}
+
+// Input an imageData and its idx, then return a relative coordinate on the
+// range of given imageData.
+function ImageIdxToRelativeCoordinate(imageData, idx) {
+ const offset = idx / 4; // RGBA
+ return { x: offset % imageData.width, y: offset / imageData.width };
+}
+
+function calculateFrameDiffenceInXAxis(squareX, targetX) {
+ const offsetX = Math.abs(targetX - squareX);
+ const xSpeedPerFrame = 640 / 60; // video is 60fps
+ return offsetX / xSpeedPerFrame;
+}
+
+function isPixelColorRed(r, g, b) {
+ // As the rendering color would vary in the screen on different platforms, so
+ // we won't strict the R should be 255, just check if it's larger than a
+ // certain threshold.
+ return r > 200 && g < 10 && b < 10;
+}
+
+function setPlaybackRate(rate) {
+ const video = document.getElementById("video");
+ info(`change playback rate from ${video.playbackRate} to ${rate}`);
+ document.getElementById("video").playbackRate = rate;
+}
+
+function adjustFrameDiffBasedOnMediaTime(currentDiff) {
+ // The audio wave can be simply regarded as being composed by "start", "peak"
+ // and "tail". The "start" part is the sound gradually becoming louder and the
+ // "tail" is gradually becoming silent. We want to check the "peak" part which
+ // should happn on evert second regularly (1s, 2s, 3s ...) However, this check
+ // is triggered by `requestAnimationFrame()` and we can't guarantee that
+ // we're checking the peak part while the function is being called. Therefore,
+ // we have to do an adjustment by the video time, to know if we're checking
+ // the audio wave too early or too late in order to get a consistent result.
+ const video = document.getElementById("video");
+ const videoCurrentTimeFloatPortion = video.currentTime % 1;
+ const timeOffset =
+ videoCurrentTimeFloatPortion > 0.5 ?
+ 1 - videoCurrentTimeFloatPortion : // too early
+ videoCurrentTimeFloatPortion; // too late
+ const frameOffset = timeOffset / 0.016; // 60fps, 1 frame=0.016s
+ info(`timeOffset=${timeOffset}, frameOffset=${frameOffset}`);
+ return Math.abs(currentDiff - frameOffset);
+}
+
+</script>
+</head>
+<body>
+</body>
+</html>
diff --git a/dom/media/test/test_chaining.html b/dom/media/test/test_chaining.html
new file mode 100644
index 0000000000..6ebf94c88d
--- /dev/null
+++ b/dom/media/test/test_chaining.html
@@ -0,0 +1,92 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: chained ogg files.</title>
+ <meta charset='utf-8'>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var manager = new MediaTestManager;
+
+function finish_test(element) {
+ removeNodeAndSource(element);
+ manager.finished(element.token);
+}
+
+function onended(e) {
+ var t = e.target;
+ is(t._metadataCount, t._links, "We should have received "+ t._links +
+ " metadataloaded event. " + t.src);
+
+ // If we encounter a file that has links with a different numbers of channel,
+ // we stop the decoding at the end of the first link. Hence, we report normal
+ // |seekable| and |buffered| values.
+ if (t._links != 1) {
+ is(t.seekable.length, 0, "No seekable ranges should be reported." + t.src);
+ is(t.buffered.length, 0, "No buffered region should be reported." + t.src);
+ }
+
+ is(t.played.length, 1, "The played region should be continuous." + t.src);
+
+ if (t._links != 1) {
+ var oldCurrentTime = t.currentTime;
+ t.currentTime = 0.0;
+ is(t.currentTime, oldCurrentTime,
+ "Seeking should be disabled when playing chained media." + t.src);
+ }
+
+ finish_test(t);
+}
+
+function onmetadataloaded(e) {
+ var t = e.target;
+ if (! t._metadataCount) {
+ t._metadataCount = 0;
+ }
+
+ if (t._metadataCount > 1 && t._links === 1) {
+ ok(false, "We should receive only one \"loadedmetadata\" event for a chained file we don't support.")
+ }
+
+ // We should be able to assert equality here, but somehow it fails (rarely but
+ // still) on try. Instead, we give it a little slack and assert that the index
+ // increases monotonically.
+ ok(t.mozGetMetadata().index >= t._metadataCount || t._links === 1,
+ "The metadata index value should increase." + t.src);
+
+ ok(t.currentTime >= t._prevCurrentTime,
+ "The currenttime should be increased correctly in new chained part.");
+ t._prevCurrentTime = t.currentTime;
+
+ // The files have all a user comment name 'index' that increments at each link
+ // in the chained media.
+ t._metadataCount++;
+ if (!t.playing && !t.ended) {
+ t.play();
+ }
+}
+
+function startTest(test, token) {
+ var elemType = /^audio/.test(test.type) ? "audio" : "video";
+ var element = document.createElement(elemType);
+ document.body.appendChild(element);
+ manager.started(token);
+ element._links= test.links;
+ element.src = test.name;
+ element.token = token;
+ element.controls = true;
+ element.addEventListener("loadedmetadata", onmetadataloaded);
+ element.addEventListener("ended", onended);
+ element.preload = "metadata";
+ element._prevCurrentTime = 0;
+}
+
+manager.runTests(gChainingTests, startTest);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_cloneElementVisually_ended_video.html b/dom/media/test/test_cloneElementVisually_ended_video.html
new file mode 100644
index 0000000000..331dda2db9
--- /dev/null
+++ b/dom/media/test/test_cloneElementVisually_ended_video.html
@@ -0,0 +1,48 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <meta charset="utf-8">
+ <title>Test cloneElementVisually</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script type="application/javascript" src="https://example.com:443/tests/dom/media/test/cloneElementVisually_helpers.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css">
+</head>
+<body>
+<div id="content">
+ <h1>Original</h1>
+ <video id="original"></video>
+ <h1>Clone</h1>
+</div>
+<div id="results">
+ <h1>Results</h1>
+ <canvas id="left"></canvas>
+ <canvas id="right"></canvas>
+</div>
+
+<script type="application/javascript">
+
+/* import-globals-from cloneElementVisually_helpers.js */
+
+/**
+ * Test that when we start cloning a video that has already ended, the
+ * clone displays the last frame from the video.
+ */
+add_task(async () => {
+ await setup();
+
+ let originalVideo = document.getElementById("original");
+ let ended = waitForEventOnce(originalVideo, "ended");
+ await originalVideo.play();
+ await ended;
+
+ await withNewClone(originalVideo, async clone => {
+ await SpecialPowers.wrap(originalVideo).cloneElementVisually(clone);
+ ok(await assertVideosMatch(originalVideo, clone),
+ "Visual clone should display final frame.");
+ });
+});
+
+</script>
+
+</body>
+</html>
diff --git a/dom/media/test/test_cloneElementVisually_mediastream.html b/dom/media/test/test_cloneElementVisually_mediastream.html
new file mode 100644
index 0000000000..5bfcf7673f
--- /dev/null
+++ b/dom/media/test/test_cloneElementVisually_mediastream.html
@@ -0,0 +1,70 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <meta charset="utf-8">
+ <title>Test cloneElementVisually</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script type="application/javascript" src="https://example.com:443/tests/dom/media/test/cloneElementVisually_helpers.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css">
+</head>
+<body>
+<div id="content">
+ <h1>Original</h1>
+ <video id="original"></video>
+ <h1>MediaStream</h1>
+ <video id="streamTarget"></video>
+ <h1>Clone</h1>
+</div>
+<div id="results">
+ <h1>Results</h1>
+ <canvas id="left"></canvas>
+ <canvas id="right"></canvas>
+</div>
+
+<script type="application/javascript">
+
+/* import-globals-from cloneElementVisually_helpers.js */
+
+/**
+ * Test that we can clone a video that is playing a MediaStream.
+ */
+add_task(async () => {
+ await setup();
+
+ let originalVideo = document.getElementById("original");
+ let stream = originalVideo.mozCaptureStream();
+ let streamTarget = document.getElementById("streamTarget");
+ originalVideo.setAttribute("loop", true);
+ let playingPromise = waitForEventOnce(originalVideo, "playing");
+ await originalVideo.play();
+ await playingPromise;
+
+ streamTarget.srcObject = stream;
+ playingPromise = waitForEventOnce(streamTarget, "playing");
+ await streamTarget.play();
+ await playingPromise
+
+ await withNewClone(originalVideo, async clone => {
+ await SpecialPowers.wrap(streamTarget).cloneElementVisually(clone);
+
+ originalVideo.loop = false;
+ originalVideo.currentTime = originalVideo.duration - 0.1;
+ await waitForEventOnce(streamTarget, "ended");
+
+ ok(await assertVideosMatch(originalVideo, streamTarget),
+ "Should match Original video");
+ ok(await assertVideosMatch(streamTarget, clone),
+ "Should match MediaStream");
+ });
+
+ // Capturing a stream from a video "taints" it which prevents testing
+ // shutdown decoder behaviour. To avoid interfering with future tests,
+ // we replace the video.
+ let newVideo = originalVideo.cloneNode();
+ originalVideo.parentNode.replaceChild(newVideo, originalVideo);
+});
+
+</script>
+
+</body>
+</html>
diff --git a/dom/media/test/test_cloneElementVisually_mediastream_multitrack.html b/dom/media/test/test_cloneElementVisually_mediastream_multitrack.html
new file mode 100644
index 0000000000..04d1ed484c
--- /dev/null
+++ b/dom/media/test/test_cloneElementVisually_mediastream_multitrack.html
@@ -0,0 +1,88 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <meta charset="utf-8">
+ <title>Test cloneElementVisually</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script type="application/javascript" src="https://example.com:443/tests/dom/media/test/cloneElementVisually_helpers.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css">
+</head>
+<body>
+<div id="content">
+ <h1>Originals</h1>
+ <div id="originalContainer"></div>
+ <canvas id="canvas"></canvas>
+ <h1>MediaStream</h1>
+ <div id="streamTargetContainer"></div>
+ <h1>Clone</h1>
+</div>
+<div id="results">
+ <h1>Results</h1>
+ <canvas id="left"></canvas>
+ <canvas id="right"></canvas>
+</div>
+
+<script type="application/javascript">
+
+/* import-globals-from cloneElementVisually_helpers.js */
+
+/**
+ * Test that a clone of a video that is playing a MediaStream properly tracks
+ * the selected video track.
+ */
+add_task(async () => {
+ await SpecialPowers.pushPrefEnv({
+ set: [
+ ["media.track.enabled", true],
+ ],
+ });
+
+ let originalVideo = document.createElement("video");
+ originalVideo.id = "original";
+ document.getElementById("originalContainer").appendChild(originalVideo);
+
+ let streamTarget = document.createElement("video");
+ document.getElementById("streamTargetContainer").appendChild(streamTarget);
+
+ await setup();
+
+ let [track1] = originalVideo.mozCaptureStream().getTracks();
+ let playingPromise = waitForEventOnce(originalVideo, "playing");
+ await originalVideo.play();
+ await playingPromise;
+
+ let canvas = document.getElementById("canvas");
+ canvas.width = originalVideo.videoWidth / 2;
+ canvas.height = originalVideo.videoHeight / 2;
+ let ctx = canvas.getContext("2d");
+ let [track2] = canvas.captureStream().getTracks();
+ ctx.fillStyle = "green";
+ ctx.fillRect(0, 0, canvas.width, canvas.height);
+
+ streamTarget.srcObject = new MediaStream([track1, track2]);
+ playingPromise = waitForEventOnce(streamTarget, "playing");
+ await streamTarget.play();
+ await playingPromise;
+
+ await withNewClone(originalVideo, async clone => {
+ SpecialPowers.wrap(streamTarget).cloneElementVisually(clone);
+
+ let selectedTrackIdx = streamTarget.videoTracks.selectedIndex;
+ streamTarget.videoTracks[++selectedTrackIdx % 2].selected = true;
+ await waitForEventOnce(streamTarget, "resize");
+
+ ok(await assertVideosMatch(streamTarget, clone),
+ "Should match MediaStream");
+ });
+
+ // Capturing a stream from a video "taints" it which prevents testing
+ // shutdown decoder behaviour. To avoid interfering with future tests,
+ // we replace the video.
+ let newVideo = originalVideo.cloneNode();
+ originalVideo.parentNode.replaceChild(newVideo, originalVideo);
+});
+
+</script>
+
+</body>
+</html>
diff --git a/dom/media/test/test_cloneElementVisually_no_suspend.html b/dom/media/test/test_cloneElementVisually_no_suspend.html
new file mode 100644
index 0000000000..d576bf8ab3
--- /dev/null
+++ b/dom/media/test/test_cloneElementVisually_no_suspend.html
@@ -0,0 +1,90 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <meta charset="utf-8">
+ <title>Test cloneElementVisually</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script type="application/javascript" src="https://example.com:443/tests/dom/media/test/cloneElementVisually_helpers.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css">
+</head>
+<body>
+<div id="content">
+ <h1>Original</h1>
+ <video id="original"></video>
+ <h1>MediaStream</h1>
+ <video id="streamTarget"></video>
+ <h1>Clone</h1>
+</div>
+<div id="results">
+ <h1>Results</h1>
+ <canvas id="left"></canvas>
+ <canvas id="right"></canvas>
+</div>
+
+<script type="application/javascript">
+
+/* import-globals-from cloneElementVisually_helpers.js */
+
+/**
+ * Tests that cloning a video prevents the decoder from being suspended
+ * if the original video stops being visible.
+ */
+add_task(async () => {
+ await setup();
+
+ let originalVideo = document.getElementById("original");
+ await setVideoSrc(originalVideo, LONG_VIDEO);
+
+ await originalVideo.play();
+
+ // Ensure that hiding and pausing this video will cause us to
+ // try suspending it.
+ await ensureVideoSuspendable(originalVideo);
+
+ await withNewClone(originalVideo, async clone => {
+ SpecialPowers.wrap(originalVideo).cloneElementVisually(clone);
+
+ // Go back to the beginning of the video to give us enough time to
+ // fail to suspend the video when it's being cloned before the
+ // video ends.
+ originalVideo.removeAttribute("loop");
+ originalVideo.currentTime = 0;
+ await waitForEventOnce(originalVideo, "seeked");
+
+ let suspendTimerFired = false;
+
+ let listener = () => {
+ suspendTimerFired = true;
+ }
+ originalVideo.addEventListener("mozstartvideosuspendtimer", listener);
+
+ // Have to do this to access normally-preffed off binding methods for some
+ // reason.
+ // See bug 1544257.
+ SpecialPowers.wrap(originalVideo).setVisible(false);
+
+ await waitForEventOnce(originalVideo, "ended");
+
+ originalVideo.removeEventListener("mozstartvideosuspendtimer", listener);
+
+ ok(!suspendTimerFired,
+ "mozstartvideosuspendtimer should not have fired.");
+
+ // Have to do this to access normally-preffed off binding methods for some
+ // reason.
+ // See bug 1544257.
+ SpecialPowers.wrap(originalVideo).setVisible(true);
+ });
+
+ await originalVideo.play();
+
+ // With the clone gone, the original video should be able to suspend now.
+ await ensureVideoSuspendable(originalVideo);
+
+ await setVideoSrc(originalVideo, TEST_VIDEO_1);
+});
+
+</script>
+
+</body>
+</html>
diff --git a/dom/media/test/test_cloneElementVisually_paused.html b/dom/media/test/test_cloneElementVisually_paused.html
new file mode 100644
index 0000000000..1812becdd8
--- /dev/null
+++ b/dom/media/test/test_cloneElementVisually_paused.html
@@ -0,0 +1,45 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <meta charset="utf-8">
+ <title>Test cloneElementVisually</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script type="application/javascript" src="https://example.com:443/tests/dom/media/test/cloneElementVisually_helpers.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css">
+</head>
+<body>
+<div id="content">
+ <h1>Original</h1>
+ <video id="original"></video>
+ <h1>Clone</h1>
+</div>
+<div id="results">
+ <h1>Results</h1>
+ <canvas id="left"></canvas>
+ <canvas id="right"></canvas>
+</div>
+
+<script type="application/javascript">
+
+/* import-globals-from cloneElementVisually_helpers.js */
+
+/**
+ * Test that when we start cloning a paused video, the clone displays
+ * the first paused frame.
+ */
+add_task(async () => {
+ await setup();
+
+ let originalVideo = document.getElementById("original");
+ await withNewClone(originalVideo, async clone => {
+ await SpecialPowers.wrap(originalVideo).cloneElementVisually(clone);
+
+ ok(await assertVideosMatch(originalVideo, clone),
+ "Initial paused frame should match.");
+ });
+});
+
+</script>
+
+</body>
+</html>
diff --git a/dom/media/test/test_cloneElementVisually_poster.html b/dom/media/test/test_cloneElementVisually_poster.html
new file mode 100644
index 0000000000..40f60e4961
--- /dev/null
+++ b/dom/media/test/test_cloneElementVisually_poster.html
@@ -0,0 +1,53 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <meta charset="utf-8">
+ <title>Test cloneElementVisually with poster</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script type="application/javascript" src="https://example.com:443/tests/dom/media/test/cloneElementVisually_helpers.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css">
+</head>
+<body>
+<div id="content">
+ <h1>Original</h1>
+ <video id="original"></video>
+ <h1>Clone</h1>
+</div>
+<div id="results">
+ <h1>Results</h1>
+ <canvas id="left"></canvas>
+ <canvas id="right"></canvas>
+</div>
+
+<script type="application/javascript">
+
+/* import-globals-from cloneElementVisually_helpers.js */
+
+/**
+ * Test that when we clone a video with a poster, the poster does not prevent
+ * the cloned video from displaying properly (as in bug 1532692).
+ */
+add_task(async () => {
+ await setup();
+
+ let originalVideo = document.getElementById("original");
+ const POSTER_URL = "https://example.com:443/tests/dom/media/test/poster-test.jpg";
+ originalVideo.setAttribute("poster", POSTER_URL);
+
+ await withNewClone(originalVideo, async clone => {
+ await SpecialPowers.wrap(originalVideo).cloneElementVisually(clone);
+ originalVideo.loop = false;
+ originalVideo.currentTime = originalVideo.duration - 0.1;
+ let endedPromise = waitForEventOnce(originalVideo, "ended");
+ await originalVideo.play();
+ await endedPromise;
+
+ ok(await assertVideosMatch(originalVideo, clone),
+ "Video with a poster should clone properly.");
+ });
+});
+
+</script>
+
+</body>
+</html>
diff --git a/dom/media/test/test_cloneElementVisually_resource_change.html b/dom/media/test/test_cloneElementVisually_resource_change.html
new file mode 100644
index 0000000000..3a66906ea2
--- /dev/null
+++ b/dom/media/test/test_cloneElementVisually_resource_change.html
@@ -0,0 +1,67 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <meta charset="utf-8">
+ <title>Test cloneElementVisually</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script type="application/javascript" src="https://example.com:443/tests/dom/media/test/cloneElementVisually_helpers.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css">
+</head>
+<body>
+<div id="content">
+ <h1>Original</h1>
+ <video id="original"></video>
+ <h1>MediaStream</h1>
+ <video id="streamTarget"></video>
+ <h1>Clone</h1>
+</div>
+<div id="results">
+ <h1>Results</h1>
+ <canvas id="left"></canvas>
+ <canvas id="right"></canvas>
+</div>
+
+<script type="application/javascript">
+
+/* import-globals-from cloneElementVisually_helpers.js */
+
+/**
+ * Tests that cloning survives changes to the underlying video resource.
+ */
+add_task(async () => {
+ await setup();
+
+ let originalVideo = document.getElementById("original");
+ originalVideo.setAttribute("loop", true);
+ await originalVideo.play();
+
+ await withNewClone(originalVideo, async clone => {
+ SpecialPowers.wrap(originalVideo).cloneElementVisually(clone);
+
+ await waitForEventOnce(originalVideo, "timeupdate");
+
+ originalVideo.pause();
+ await waitForEventOnce(originalVideo, "pause");
+
+ ok(await assertVideosMatch(originalVideo, clone),
+ "Initial video should match.");
+
+ await setVideoSrc(originalVideo, TEST_VIDEO_2);
+
+ await originalVideo.play();
+ await waitForEventOnce(originalVideo, "timeupdate");
+
+ originalVideo.pause();
+ await waitForEventOnce(originalVideo, "pause");
+
+ ok(await assertVideosMatch(originalVideo, clone),
+ "New video should match.");
+ });
+
+ await setVideoSrc(originalVideo, TEST_VIDEO_1);
+});
+
+</script>
+
+</body>
+</html>
diff --git a/dom/media/test/test_clone_media_element.html b/dom/media/test/test_clone_media_element.html
new file mode 100644
index 0000000000..35e5cd69d0
--- /dev/null
+++ b/dom/media/test/test_clone_media_element.html
@@ -0,0 +1,54 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test: cloned media element should continue to play to the end even after the source of the original element is cleared</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+// tests must run in sequence otherwise concurrent running test will also
+// update media cache which will hide the fact media cache not updated
+// after changes in media cache streams.
+PARALLEL_TESTS = 1;
+
+function startTest(test, token) {
+ manager.started(token);
+ info("Trying to load " + token);
+ var v = document.createElement('video');
+ v.preload = "metadata";
+ v.token = token;
+ v.src = test.name;
+
+ v.onloadedmetadata = function(evt) {
+ info(evt.target.token + " metadata loaded.");
+ evt.target.onloadedmetadata = null;
+ var clone = evt.target.cloneNode(false);
+ clone.token = evt.target.token;
+ clone.play();
+
+ clone.onloadstart = function(event) {
+ info("cloned " + event.target.token + " start loading.");
+ event.target.onloadstart = null;
+ removeNodeAndSource(v);
+ }
+
+ clone.onended = function(event) {
+ ok(true, "cloned " + event.target.token + " ended.");
+ event.target.onended = null;
+ removeNodeAndSource(event.target);
+ manager.finished(event.target.token);
+ }
+ }
+}
+
+var manager = new MediaTestManager;
+manager.runTests(gSmallTests.concat(gPlayedTests), startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_closing_connections.html b/dom/media/test/test_closing_connections.html
new file mode 100644
index 0000000000..c5eb565447
--- /dev/null
+++ b/dom/media/test/test_closing_connections.html
@@ -0,0 +1,58 @@
+hg diff<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=479863
+-->
+<head>
+ <title>Test for Bug 479863 --- loading many connections</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="application/javascript" src="use_large_cache.js"></script>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=479863">Mozilla Bug 479863</a>
+<p id="display"></p>
+<div id="content" style="display: none">
+
+</div>
+
+<script type="application/javascript">
+window.onload = function() {
+ ok(true, "loaded metadata for all videos");
+ mediaTestCleanup();
+ SimpleTest.finish();
+}
+
+/* With normal per-domain connection limits and a naive implementation, we
+ won't ever be able to load all these videos because the first 15 (or whatever)
+ will each take up one HTTP connection (which will be suspended) and then
+ the others will be blocked by the per-domain HTTP connection limit. We
+ pass this test by closing the connection for non-buffered videos after
+ we've got the first frame.
+*/
+
+var resource = getPlayableVideo(gClosingConnectionsTest);
+
+SimpleTest.waitForExplicitFinish();
+function beginTest() {
+ if (resource != null) {
+ for (var i=0; i<20; ++i) {
+ var v = document.createElement("video");
+ v.src = resource.name;
+ v.preload = "metadata";
+ document.body.appendChild(v);
+ }
+ } else {
+ todo(false, "No types supported");
+ }
+}
+beginTest();
+</script>
+
+<pre id="test">
+<script type="application/javascript">
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_constants.html b/dom/media/test/test_constants.html
new file mode 100644
index 0000000000..1d4a8da250
--- /dev/null
+++ b/dom/media/test/test_constants.html
@@ -0,0 +1,228 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+ Adapted from:
+ http://simon.html5.org/test/html/dom/interfaces/htmlelement/htmlmediaelement/const-unsigned-short/001.htm
+-->
+<head>
+ <title>Media test: constants</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<video><source></video><audio><source></audio>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+is(HTMLElement.NETWORK_EMPTY, undefined);
+is(HTMLElement.NETWORK_IDLE, undefined);
+is(HTMLElement.NETWORK_LOADING, undefined);
+is(HTMLElement.NETWORK_NO_SOURCE, undefined);
+is(HTMLElement.HAVE_NOTHING, undefined);
+is(HTMLElement.HAVE_METADATA, undefined);
+is(HTMLElement.HAVE_CURRENT_DATA, undefined);
+is(HTMLElement.HAVE_FUTURE_DATA, undefined);
+is(HTMLElement.HAVE_ENOUGH_DATA, undefined);
+is(HTMLElement.MEDIA_ERR_ABORTED, undefined);
+is(HTMLElement.MEDIA_ERR_NETWORK, undefined);
+is(HTMLElement.MEDIA_ERR_DECODE, undefined);
+is(HTMLElement.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
+is(HTMLMediaElement.NETWORK_EMPTY, 0);
+is(HTMLMediaElement.NETWORK_IDLE, 1);
+is(HTMLMediaElement.NETWORK_LOADING, 2);
+is(HTMLMediaElement.NETWORK_NO_SOURCE, 3);
+is(HTMLMediaElement.HAVE_NOTHING, 0);
+is(HTMLMediaElement.HAVE_METADATA, 1);
+is(HTMLMediaElement.HAVE_CURRENT_DATA, 2);
+is(HTMLMediaElement.HAVE_FUTURE_DATA, 3);
+is(HTMLMediaElement.HAVE_ENOUGH_DATA, 4);
+is(HTMLMediaElement.MEDIA_ERR_ABORTED, undefined);
+is(HTMLMediaElement.MEDIA_ERR_NETWORK, undefined);
+is(HTMLMediaElement.MEDIA_ERR_DECODE, undefined);
+is(HTMLMediaElement.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
+is(HTMLVideoElement.NETWORK_EMPTY, 0);
+is(HTMLVideoElement.NETWORK_IDLE, 1);
+is(HTMLVideoElement.NETWORK_LOADING, 2);
+is(HTMLVideoElement.NETWORK_NO_SOURCE, 3);
+is(HTMLVideoElement.HAVE_NOTHING, 0);
+is(HTMLVideoElement.HAVE_METADATA, 1);
+is(HTMLVideoElement.HAVE_CURRENT_DATA, 2);
+is(HTMLVideoElement.HAVE_FUTURE_DATA, 3);
+is(HTMLVideoElement.HAVE_ENOUGH_DATA, 4);
+is(HTMLVideoElement.MEDIA_ERR_ABORTED, undefined);
+is(HTMLVideoElement.MEDIA_ERR_NETWORK, undefined);
+is(HTMLVideoElement.MEDIA_ERR_DECODE, undefined);
+is(HTMLVideoElement.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
+is(HTMLAudioElement.NETWORK_EMPTY, 0);
+is(HTMLAudioElement.NETWORK_IDLE, 1);
+is(HTMLAudioElement.NETWORK_LOADING, 2);
+is(HTMLAudioElement.NETWORK_NO_SOURCE, 3);
+is(HTMLAudioElement.HAVE_NOTHING, 0);
+is(HTMLAudioElement.HAVE_METADATA, 1);
+is(HTMLAudioElement.HAVE_CURRENT_DATA, 2);
+is(HTMLAudioElement.HAVE_FUTURE_DATA, 3);
+is(HTMLAudioElement.HAVE_ENOUGH_DATA, 4);
+is(HTMLAudioElement.MEDIA_ERR_ABORTED, undefined);
+is(HTMLAudioElement.MEDIA_ERR_NETWORK, undefined);
+is(HTMLAudioElement.MEDIA_ERR_DECODE, undefined);
+is(HTMLAudioElement.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
+is(HTMLSourceElement.NETWORK_EMPTY, undefined);
+is(HTMLSourceElement.NETWORK_IDLE, undefined);
+is(HTMLSourceElement.NETWORK_LOADING, undefined);
+is(HTMLSourceElement.NETWORK_NO_SOURCE, undefined);
+is(HTMLSourceElement.HAVE_NOTHING, undefined);
+is(HTMLSourceElement.HAVE_METADATA, undefined);
+is(HTMLSourceElement.HAVE_CURRENT_DATA, undefined);
+is(HTMLSourceElement.HAVE_FUTURE_DATA, undefined);
+is(HTMLSourceElement.HAVE_ENOUGH_DATA, undefined);
+is(HTMLSourceElement.MEDIA_ERR_ABORTED, undefined);
+is(HTMLSourceElement.MEDIA_ERR_NETWORK, undefined);
+is(HTMLSourceElement.MEDIA_ERR_DECODE, undefined);
+is(HTMLSourceElement.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
+is(MediaError.NETWORK_EMPTY, undefined);
+is(MediaError.NETWORK_IDLE, undefined);
+is(MediaError.NETWORK_LOADING, undefined);
+is(MediaError.NETWORK_NO_SOURCE, undefined);
+is(MediaError.HAVE_NOTHING, undefined);
+is(MediaError.HAVE_METADATA, undefined);
+is(MediaError.HAVE_CURRENT_DATA, undefined);
+is(MediaError.HAVE_FUTURE_DATA, undefined);
+is(MediaError.HAVE_ENOUGH_DATA, undefined);
+is(MediaError.MEDIA_ERR_ABORTED, 1);
+is(MediaError.MEDIA_ERR_NETWORK, 2);
+is(MediaError.MEDIA_ERR_DECODE, 3);
+is(MediaError.MEDIA_ERR_SRC_NOT_SUPPORTED, 4);
+is(document.body.NETWORK_EMPTY, undefined);
+is(document.body.NETWORK_IDLE, undefined);
+is(document.body.NETWORK_LOADING, undefined);
+is(document.body.NETWORK_NO_SOURCE, undefined);
+is(document.body.HAVE_NOTHING, undefined);
+is(document.body.HAVE_METADATA, undefined);
+is(document.body.HAVE_CURRENT_DATA, undefined);
+is(document.body.HAVE_FUTURE_DATA, undefined);
+is(document.body.HAVE_ENOUGH_DATA, undefined);
+is(document.body.MEDIA_ERR_ABORTED, undefined);
+is(document.body.MEDIA_ERR_NETWORK, undefined);
+is(document.body.MEDIA_ERR_DECODE, undefined);
+is(document.body.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
+is(document.getElementsByTagName("video")[0].NETWORK_EMPTY, 0);
+is(document.getElementsByTagName("video")[0].NETWORK_IDLE, 1);
+is(document.getElementsByTagName("video")[0].NETWORK_LOADING, 2);
+is(document.getElementsByTagName("video")[0].NETWORK_NO_SOURCE, 3);
+is(document.getElementsByTagName("video")[0].HAVE_NOTHING, 0);
+is(document.getElementsByTagName("video")[0].HAVE_METADATA, 1);
+is(document.getElementsByTagName("video")[0].HAVE_CURRENT_DATA, 2);
+is(document.getElementsByTagName("video")[0].HAVE_FUTURE_DATA, 3);
+is(document.getElementsByTagName("video")[0].HAVE_ENOUGH_DATA, 4);
+is(document.getElementsByTagName("video")[0].MEDIA_ERR_ABORTED, undefined);
+is(document.getElementsByTagName("video")[0].MEDIA_ERR_NETWORK, undefined);
+is(document.getElementsByTagName("video")[0].MEDIA_ERR_DECODE, undefined);
+is(document.getElementsByTagName("video")[0].MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
+is(document.getElementsByTagName("audio")[0].NETWORK_EMPTY, 0);
+is(document.getElementsByTagName("audio")[0].NETWORK_IDLE, 1);
+is(document.getElementsByTagName("audio")[0].NETWORK_LOADING, 2);
+is(document.getElementsByTagName("audio")[0].NETWORK_NO_SOURCE, 3);
+is(document.getElementsByTagName("audio")[0].HAVE_NOTHING, 0);
+is(document.getElementsByTagName("audio")[0].HAVE_METADATA, 1);
+is(document.getElementsByTagName("audio")[0].HAVE_CURRENT_DATA, 2);
+is(document.getElementsByTagName("audio")[0].HAVE_FUTURE_DATA, 3);
+is(document.getElementsByTagName("audio")[0].HAVE_ENOUGH_DATA, 4);
+is(document.getElementsByTagName("audio")[0].MEDIA_ERR_ABORTED, undefined);
+is(document.getElementsByTagName("audio")[0].MEDIA_ERR_NETWORK, undefined);
+is(document.getElementsByTagName("audio")[0].MEDIA_ERR_DECODE, undefined);
+is(document.getElementsByTagName("audio")[0].MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
+is(document.getElementsByTagName("source")[0].NETWORK_EMPTY, undefined);
+is(document.getElementsByTagName("source")[0].NETWORK_IDLE, undefined);
+is(document.getElementsByTagName("source")[0].NETWORK_LOADING, undefined);
+is(document.getElementsByTagName("source")[0].NETWORK_NO_SOURCE, undefined);
+is(document.getElementsByTagName("source")[0].HAVE_NOTHING, undefined);
+is(document.getElementsByTagName("source")[0].HAVE_METADATA, undefined);
+is(document.getElementsByTagName("source")[0].HAVE_CURRENT_DATA, undefined);
+is(document.getElementsByTagName("source")[0].HAVE_FUTURE_DATA, undefined);
+is(document.getElementsByTagName("source")[0].HAVE_ENOUGH_DATA, undefined);
+is(document.getElementsByTagName("source")[0].MEDIA_ERR_ABORTED, undefined);
+is(document.getElementsByTagName("source")[0].MEDIA_ERR_NETWORK, undefined);
+is(document.getElementsByTagName("source")[0].MEDIA_ERR_DECODE, undefined);
+is(document.getElementsByTagName("source")[0].MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
+is(HTMLElement.prototype.NETWORK_EMPTY, undefined);
+is(HTMLElement.prototype.NETWORK_IDLE, undefined);
+is(HTMLElement.prototype.NETWORK_LOADING, undefined);
+is(HTMLElement.prototype.NETWORK_NO_SOURCE, undefined);
+is(HTMLElement.prototype.HAVE_NOTHING, undefined);
+is(HTMLElement.prototype.HAVE_METADATA, undefined);
+is(HTMLElement.prototype.HAVE_CURRENT_DATA, undefined);
+is(HTMLElement.prototype.HAVE_FUTURE_DATA, undefined);
+is(HTMLElement.prototype.HAVE_ENOUGH_DATA, undefined);
+is(HTMLElement.prototype.MEDIA_ERR_ABORTED, undefined);
+is(HTMLElement.prototype.MEDIA_ERR_NETWORK, undefined);
+is(HTMLElement.prototype.MEDIA_ERR_DECODE, undefined);
+is(HTMLElement.prototype.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
+is(HTMLMediaElement.prototype.NETWORK_EMPTY, 0, "HTMLMediaElement.prototype.NETWORK_EMPTY");
+is(HTMLMediaElement.prototype.NETWORK_IDLE, 1, "HTMLMediaElement.prototype.NETWORK_IDLE");
+is(HTMLMediaElement.prototype.NETWORK_LOADING, 2, "HTMLMediaElement.prototype.NETWORK_LOADING");
+is(HTMLMediaElement.prototype.NETWORK_NO_SOURCE, 3, "HTMLMediaElement.prototype.NETWORK_NO_SOURCE");
+is(HTMLMediaElement.prototype.HAVE_NOTHING, 0, "HTMLMediaElement.prototype.HAVE_NOTHING");
+is(HTMLMediaElement.prototype.HAVE_METADATA, 1, "HTMLMediaElement.prototype.HAVE_METADATA");
+is(HTMLMediaElement.prototype.HAVE_CURRENT_DATA, 2, "HTMLMediaElement.prototype.HAVE_CURRENT_DATA");
+is(HTMLMediaElement.prototype.HAVE_FUTURE_DATA, 3, "HTMLMediaElement.prototype.HAVE_FUTURE_DATA");
+is(HTMLMediaElement.prototype.HAVE_ENOUGH_DATA, 4, "HTMLMediaElement.prototype.HAVE_ENOUGH_DATA");
+is(HTMLMediaElement.prototype.MEDIA_ERR_ABORTED, undefined, "HTMLMediaElement.prototype.MEDIA_ERR_ABORTED");
+is(HTMLMediaElement.prototype.MEDIA_ERR_NETWORK, undefined, "HTMLMediaElement.prototype.MEDIA_ERR_NETWORK");
+is(HTMLMediaElement.prototype.MEDIA_ERR_DECODE, undefined, "HTMLMediaElement.prototype.MEDIA_ERR_DECODE");
+is(HTMLMediaElement.prototype.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined, "HTMLMediaElement.prototype.MEDIA_ERR_SRC_NOT_SUPPORTED");
+is(HTMLVideoElement.prototype.NETWORK_EMPTY, 0);
+is(HTMLVideoElement.prototype.NETWORK_IDLE, 1);
+is(HTMLVideoElement.prototype.NETWORK_LOADING, 2);
+is(HTMLVideoElement.prototype.NETWORK_NO_SOURCE, 3);
+is(HTMLVideoElement.prototype.HAVE_NOTHING, 0);
+is(HTMLVideoElement.prototype.HAVE_METADATA, 1);
+is(HTMLVideoElement.prototype.HAVE_CURRENT_DATA, 2);
+is(HTMLVideoElement.prototype.HAVE_FUTURE_DATA, 3);
+is(HTMLVideoElement.prototype.HAVE_ENOUGH_DATA, 4);
+is(HTMLVideoElement.prototype.MEDIA_ERR_ABORTED, undefined);
+is(HTMLVideoElement.prototype.MEDIA_ERR_NETWORK, undefined);
+is(HTMLVideoElement.prototype.MEDIA_ERR_DECODE, undefined);
+is(HTMLVideoElement.prototype.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
+is(HTMLAudioElement.prototype.NETWORK_EMPTY, 0);
+is(HTMLAudioElement.prototype.NETWORK_IDLE, 1);
+is(HTMLAudioElement.prototype.NETWORK_LOADING, 2);
+is(HTMLAudioElement.prototype.NETWORK_NO_SOURCE, 3);
+is(HTMLAudioElement.prototype.HAVE_NOTHING, 0);
+is(HTMLAudioElement.prototype.HAVE_METADATA, 1);
+is(HTMLAudioElement.prototype.HAVE_CURRENT_DATA, 2);
+is(HTMLAudioElement.prototype.HAVE_FUTURE_DATA, 3);
+is(HTMLAudioElement.prototype.HAVE_ENOUGH_DATA, 4);
+is(HTMLAudioElement.prototype.MEDIA_ERR_ABORTED, undefined);
+is(HTMLAudioElement.prototype.MEDIA_ERR_NETWORK, undefined);
+is(HTMLAudioElement.prototype.MEDIA_ERR_DECODE, undefined);
+is(HTMLAudioElement.prototype.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
+is(HTMLSourceElement.prototype.NETWORK_EMPTY, undefined);
+is(HTMLSourceElement.prototype.NETWORK_IDLE, undefined);
+is(HTMLSourceElement.prototype.NETWORK_LOADING, undefined);
+is(HTMLSourceElement.prototype.NETWORK_NO_SOURCE, undefined);
+is(HTMLSourceElement.prototype.HAVE_NOTHING, undefined);
+is(HTMLSourceElement.prototype.HAVE_METADATA, undefined);
+is(HTMLSourceElement.prototype.HAVE_CURRENT_DATA, undefined);
+is(HTMLSourceElement.prototype.HAVE_FUTURE_DATA, undefined);
+is(HTMLSourceElement.prototype.HAVE_ENOUGH_DATA, undefined);
+is(HTMLSourceElement.prototype.MEDIA_ERR_ABORTED, undefined);
+is(HTMLSourceElement.prototype.MEDIA_ERR_NETWORK, undefined);
+is(HTMLSourceElement.prototype.MEDIA_ERR_DECODE, undefined);
+is(HTMLSourceElement.prototype.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
+is(MediaError.prototype.NETWORK_EMPTY, undefined);
+is(MediaError.prototype.NETWORK_IDLE, undefined);
+is(MediaError.prototype.NETWORK_LOADING, undefined);
+is(MediaError.prototype.NETWORK_NO_SOURCE, undefined);
+is(MediaError.prototype.HAVE_NOTHING, undefined);
+is(MediaError.prototype.HAVE_METADATA, undefined);
+is(MediaError.prototype.HAVE_CURRENT_DATA, undefined);
+is(MediaError.prototype.HAVE_FUTURE_DATA, undefined);
+is(MediaError.prototype.HAVE_ENOUGH_DATA, undefined);
+is(MediaError.prototype.MEDIA_ERR_ABORTED, 1);
+is(MediaError.prototype.MEDIA_ERR_NETWORK, 2);
+is(MediaError.prototype.MEDIA_ERR_DECODE, 3);
+is(MediaError.prototype.MEDIA_ERR_SRC_NOT_SUPPORTED, 4);
+ok(document.getElementsByTagName("video")[0].buffered instanceof TimeRanges, "video.buffered must be TimeRanges object");
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_controls.html b/dom/media/test/test_controls.html
new file mode 100644
index 0000000000..5c9015fdf2
--- /dev/null
+++ b/dom/media/test/test_controls.html
@@ -0,0 +1,33 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: controls</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<video id='v1'></video><audio id='a1'></audio>
+<video id='v2' controls></video><audio id='a2' controls></audio>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var v1 = document.getElementById('v1');
+var a1 = document.getElementById('a1');
+var v2 = document.getElementById('v2');
+var a2 = document.getElementById('a2');
+ok(!v1.controls, "v1.controls should be false by default");
+ok(!a1.controls, "v1.controls should be false by default");
+ok(v2.controls, "v2.controls should be true");
+ok(a2.controls, "v2.controls should be true");
+v2.controls=false;
+a2.controls=false;
+ok(!v2.controls, "v2.controls should be false");
+ok(!a2.controls, "a2.controls should be false");
+v2.controls=true;
+a2.controls=true;
+ok(v2.controls, "v2.controls should be true");
+ok(a2.controls, "a2.controls should be true");
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_cueless_webm_seek-1.html b/dom/media/test/test_cueless_webm_seek-1.html
new file mode 100644
index 0000000000..db58a89665
--- /dev/null
+++ b/dom/media/test/test_cueless_webm_seek-1.html
@@ -0,0 +1,136 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=657791
+-->
+<head>
+ <title>Test for Bug 657791</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=657791">Mozilla Bug 657791</a>
+
+<p id="display"></p>
+<div id="content" style="display: none">
+
+</div>
+<pre id="test">
+<script type="application/javascript">
+
+// Subset of seek tests for cueless WebMs. When random seeking (rather than just
+// in buffered ranges) is implemented for WebM, these tests can be removed and
+// the cueless WebM(s) references can be moved to the general test_seek test
+// array.
+// Test array is defined in manifest.js
+
+var manager = new MediaTestManager;
+
+// Exercise functionality as in test_seek-1
+function testWebM1(e) {
+ var v = e.target;
+ v.removeEventListener('loadeddata', testWebM1);
+
+ var startPassed = false;
+ var endPassed = false;
+ var seekFlagStart = false;
+ var seekFlagEnd = false;
+ var readonly = true;
+ var completed = false;
+
+ ok(v.buffered.length >= 1, "Should have a buffered range");
+ var halfBuffered = v.buffered.end(0) / 2;
+
+ function start() {
+ is(v.seekable.start(0), v.buffered.start(0), "Seekable start should be buffered start");
+ is(v.seekable.end(0), v.buffered.end(0), "Seekable end should be buffered end");
+ ok(!completed, "Should not be completed yet");
+ ok(!v.seeking, "seeking should default to false");
+ try {
+ v.seeking = true;
+ readonly = v.seeking === false;
+ }
+ catch(ex) {
+ readonly = "threw exception: " + ex;
+ }
+ is(readonly, true, "seeking should be readonly");
+
+ v.currentTime = halfBuffered;
+ seekFlagStart = v.seeking;
+ }
+
+ function seekStarted() {
+ ok(!completed, "should not be completed yet");
+ startPassed = true;
+ }
+
+ function seekEnded() {
+ ok(!completed, "should not be completed yet");
+ ok(Math.abs(v.currentTime - halfBuffered) < 0.1,
+ "Video currentTime should be around " + halfBuffered + ": " + v.currentTime + " (seeked)");
+ endPassed = true;
+ seekFlagEnd = v.seeking;
+ v.play();
+ }
+
+ function playbackEnded() {
+ ok(!completed, "should not be completed yet");
+
+ completed = true;
+ ok(startPassed, "seeking event");
+ ok(endPassed, "seeked event");
+ ok(seekFlagStart, "seeking flag on start should be true");
+ ok(!seekFlagEnd, "seeking flag on end should be false");
+ removeNodeAndSource(v);
+ manager.finished(v._token);
+ }
+
+ once(v, "ended", playbackEnded);
+ once(v, "seeking", seekStarted);
+ once(v, "seeked", seekEnded);
+
+ start();
+}
+
+// Fetch the media resource using XHR so we can be sure the entire
+// resource is loaded before we test buffered ranges. This ensures
+// we have deterministic behaviour.
+function fetch(url, fetched_callback) {
+ var xhr = new XMLHttpRequest();
+ xhr.open("GET", url, true);
+ xhr.responseType = "blob";
+
+ var loaded = function (event) {
+ if (xhr.status == 200 || xhr.status == 206) {
+ // Request fulfilled. Note sometimes we get 206... Presumably because either
+ // httpd.js or Necko cached the result.
+ fetched_callback(window.URL.createObjectURL(xhr.response));
+ } else {
+ ok(false, "Fetch failed headers=" + xhr.getAllResponseHeaders());
+ }
+ };
+
+ xhr.addEventListener("load", loaded);
+ xhr.send();
+}
+
+function startTest(test, token) {
+ var onfetched = function(uri) {
+ var v = document.createElement('video');
+ v._token = token;
+ v.src = uri;
+ v.addEventListener("loadeddata", testWebM1);
+ document.body.appendChild(v);
+ }
+ manager.started(token);
+ fetch(test.name, onfetched);
+}
+
+SimpleTest.waitForExplicitFinish();
+manager.runTests(gCuelessWebMTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_cueless_webm_seek-2.html b/dom/media/test/test_cueless_webm_seek-2.html
new file mode 100644
index 0000000000..720cc18399
--- /dev/null
+++ b/dom/media/test/test_cueless_webm_seek-2.html
@@ -0,0 +1,126 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=657791
+-->
+<head>
+ <title>Test for Bug 657791</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=657791">Mozilla Bug 657791</a>
+
+<p id="display"></p>
+<div id="content" style="display: none">
+
+</div>
+<pre id="test">
+<script type="application/javascript">
+
+// Subset of seek tests for cueless WebMs. When random seeking (rather than just
+// in buffered ranges) is implemented for WebM, these tests can be removed and
+// the cueless WebM(s) references can be moved to the general test_seek test
+// array.
+// Test array is defined in manifest.js
+
+var manager = new MediaTestManager;
+
+// Exercise functionality as in test_seek-2
+function testWebM2(e) {
+ var v = e.target;
+ v.removeEventListener('loadeddata', testWebM2);
+
+ var startPassed = false;
+ var endPassed = false;
+ var completed = false;
+
+ ok(v.buffered.length >= 1, "Should have a buffered range");
+ var halfBuffered = v.buffered.end(0) / 2;
+
+ function start() {
+ if (completed)
+ return;
+
+ is(v.seekable.start(0), v.buffered.start(0), "Seekable start should be buffered start");
+ is(v.seekable.end(0), v.buffered.end(0), "Seekable end should be buffered end");
+ v.currentTime=halfBuffered;
+ v.play();
+ }
+
+ function seekStarted() {
+ if (completed)
+ return;
+
+ startPassed = true;
+ }
+
+ function seekEnded() {
+ if (completed)
+ return;
+
+ endPassed = true;
+ }
+
+ function playbackEnded() {
+ if (completed)
+ return;
+
+ completed = true;
+ ok(startPassed, "send seeking event");
+ ok(endPassed, "send seeked event");
+ ok(v.ended, "Checking playback has ended");
+ ok(Math.abs(v.currentTime - v.duration) <= 0.1, "Checking currentTime at end: " + v.currentTime);
+ removeNodeAndSource(v);
+ manager.finished(v._token);
+ }
+
+ v.addEventListener("ended", playbackEnded);
+ v.addEventListener("seeking", seekStarted);
+ v.addEventListener("seeked", seekEnded);
+
+ start();
+}
+
+// Fetch the media resource using XHR so we can be sure the entire
+// resource is loaded before we test buffered ranges. This ensures
+// we have deterministic behaviour.
+function fetch(url, fetched_callback) {
+ var xhr = new XMLHttpRequest();
+ xhr.open("GET", url, true);
+ xhr.responseType = "blob";
+
+ var loaded = function (event) {
+ if (xhr.status == 200 || xhr.status == 206) {
+ // Request fulfilled. Note sometimes we get 206... Presumably because either
+ // httpd.js or Necko cached the result.
+ fetched_callback(window.URL.createObjectURL(xhr.response));
+ } else {
+ ok(false, "Fetch failed headers=" + xhr.getAllResponseHeaders());
+ }
+ };
+
+ xhr.addEventListener("load", loaded);
+ xhr.send();
+}
+
+function startTest(test, token) {
+ var onfetched = function(uri) {
+ var v = document.createElement('video');
+ v._token = token;
+ v.src = uri;
+ v.addEventListener("loadeddata", testWebM2);
+ document.body.appendChild(v);
+ }
+ manager.started(token);
+ fetch(test.name, onfetched);
+}
+
+SimpleTest.waitForExplicitFinish();
+manager.runTests(gCuelessWebMTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_cueless_webm_seek-3.html b/dom/media/test/test_cueless_webm_seek-3.html
new file mode 100644
index 0000000000..d6e3e50d7d
--- /dev/null
+++ b/dom/media/test/test_cueless_webm_seek-3.html
@@ -0,0 +1,120 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=657791
+-->
+<head>
+ <title>Test for Bug 657791</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=657791">Mozilla Bug 657791</a>
+
+<p id="display"></p>
+<div id="content" style="display: none">
+
+</div>
+<pre id="test">
+<script type="application/javascript">
+
+// Subset of seek tests for cueless WebMs. When random seeking (rather than just
+// in buffered ranges) is implemented for WebM, these tests can be removed and
+// the cueless WebM(s) references can be moved to the general test_seek test
+// array.
+// Test array is defined in manifest.js
+
+var manager = new MediaTestManager;
+
+// Exercise functionality as in test_seek-3
+function testWebM3(e) {
+ var v = e.target;
+ v.removeEventListener('loadeddata', testWebM3);
+
+ var completed = false;
+ var gotTimeupdate = false;
+
+ ok(v.buffered.length >= 1, "Should have a buffered range");
+ var halfBuffered = v.buffered.end(0) / 2;
+
+ function start() {
+ if (completed)
+ return;
+
+ is(v.seekable.start(0), v.buffered.start(0), "Seekable start should be buffered start");
+ is(v.seekable.end(0), v.buffered.end(0), "Seekable end should be buffered end");
+ v.currentTime=halfBuffered;
+ }
+
+ function timeupdate() {
+ gotTimeupdate = true;
+ v.removeEventListener("timeupdate", timeupdate);
+ }
+
+ function seekStarted() {
+ if (completed)
+ return;
+
+ v.addEventListener("timeupdate", timeupdate);
+ }
+
+ function seekEnded() {
+ if (completed)
+ return;
+
+ var t = v.currentTime;
+ ok(Math.abs(t - halfBuffered) <= 0.1, "Video currentTime should be around " + halfBuffered + ": " + t);
+ ok(gotTimeupdate, "Should have got timeupdate between seeking and seekended");
+ completed = true;
+ removeNodeAndSource(v);
+ manager.finished(v._token);
+ }
+
+ v.addEventListener("seeking", seekStarted);
+ v.addEventListener("seeked", seekEnded);
+
+ start()
+}
+
+// Fetch the media resource using XHR so we can be sure the entire
+// resource is loaded before we test buffered ranges. This ensures
+// we have deterministic behaviour.
+function fetch(url, fetched_callback) {
+ var xhr = new XMLHttpRequest();
+ xhr.open("GET", url, true);
+ xhr.responseType = "blob";
+
+ var loaded = function (event) {
+ if (xhr.status == 200 || xhr.status == 206) {
+ // Request fulfilled. Note sometimes we get 206... Presumably because either
+ // httpd.js or Necko cached the result.
+ fetched_callback(window.URL.createObjectURL(xhr.response));
+ } else {
+ ok(false, "Fetch failed headers=" + xhr.getAllResponseHeaders());
+ }
+ };
+
+ xhr.addEventListener("load", loaded);
+ xhr.send();
+}
+
+function startTest(test, token) {
+ var onfetched = function(uri) {
+ var v = document.createElement('video');
+ v._token = token;
+ v.src = uri;
+ v.addEventListener("loadeddata", testWebM3);
+ document.body.appendChild(v);
+ }
+ manager.started(token);
+ fetch(test.name, onfetched);
+}
+
+SimpleTest.waitForExplicitFinish();
+manager.runTests(gCuelessWebMTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_currentTime.html b/dom/media/test/test_currentTime.html
new file mode 100644
index 0000000000..b38c8c2c53
--- /dev/null
+++ b/dom/media/test/test_currentTime.html
@@ -0,0 +1,19 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: currentTime</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<video id='v1'></video><audio id='a1'></audio>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var v1 = document.getElementById('v1');
+var a1 = document.getElementById('a1');
+is(v1.currentTime, 0.0);
+is(a1.currentTime, 0.0);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_debug_data_helpers.html b/dom/media/test/test_debug_data_helpers.html
new file mode 100644
index 0000000000..f96aafc09b
--- /dev/null
+++ b/dom/media/test/test_debug_data_helpers.html
@@ -0,0 +1,74 @@
+<!DOCTYPE html>
+<html>
+<head>
+<title>Test the special debug APIs give expected data</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+<script type="text/javascript" src="manifest.js"></script>
+<script class="testbody" type="text/javascript">
+add_task(async function testMozRequestDebugInfo() {
+ let video = document.createElement("video");
+ video.src = "gizmo.mp4";
+ document.body.appendChild(video);
+ await video.play();
+ let debugData = await SpecialPowers.wrap(video).mozRequestDebugInfo();
+ // Verify various members are present and as expected.
+ ok(debugData, "Should get some debug data");
+ ok(debugData.decoder.hasAudio, "Should have audio");
+ ok(debugData.decoder.hasVideo, "Should have video");
+ is(
+ debugData.decoder.reader.videoWidth,
+ 560,
+ "Video should have expected width"
+ );
+ is(
+ debugData.decoder.reader.videoHeight,
+ 320,
+ "Video should have expected height"
+ );
+ ok(
+ debugData.decoder.stateMachine.mediaTime >= 0,
+ "Media time should be positive"
+ );
+ removeNodeAndSource(video);
+});
+
+add_task(async function testMozDebugReaderData() {
+ let video = document.createElement("video");
+ let mediaSource = new MediaSource();
+ video.src = URL.createObjectURL(mediaSource);
+ await once(mediaSource, "sourceopen");
+ const sourceBuffer = mediaSource.addSourceBuffer("video/webm");
+ let fetchResponse = await fetch("bipbop_short_vp8.webm");
+ sourceBuffer.appendBuffer(await fetchResponse.arrayBuffer());
+ await once(sourceBuffer, "updateend");
+ mediaSource.endOfStream();
+ await once(mediaSource, "sourceended");
+ document.body.appendChild(video);
+ await video.play();
+ let debugData = await SpecialPowers.wrap(mediaSource).mozDebugReaderData();
+ // Verify various members are present and as expected.
+ ok(debugData, "Should get some debug data");
+ is(debugData.reader.videoWidth, 400, "Video should have expected width");
+ is(debugData.reader.videoHeight, 300, "Video should have expected height");
+ ok(
+ debugData.demuxer.audioTrack.numSamples > 0,
+ "Audio track should have demuxed some samples"
+ );
+ ok(
+ debugData.demuxer.audioTrack.ranges.length,
+ "Audio track should have some buffered range"
+ );
+ ok(
+ debugData.demuxer.videoTrack.numSamples > 0,
+ "Video track should have demuxed some samples"
+ );
+ ok(
+ debugData.demuxer.videoTrack.ranges.length,
+ "Video track should have some buffered range"
+ );
+ removeNodeAndSource(video);
+});
+</script>
+</head>
+</html>
diff --git a/dom/media/test/test_decode_error.html b/dom/media/test/test_decode_error.html
new file mode 100644
index 0000000000..27b83cd3f9
--- /dev/null
+++ b/dom/media/test/test_decode_error.html
@@ -0,0 +1,66 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: unknown/invalid formats raise decode error</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var manager = new MediaTestManager;
+
+function startTest(test, token) {
+ var ok = function (condition, name) {
+ SimpleTest.ok(condition, test.name + ": " + name);
+ }
+ var is = function (a, b, name) {
+ SimpleTest.is(a, b, test.name + ": " + name);
+ }
+
+ var v = document.createElement("video");
+ manager.started(token);
+ v.addEventListener("error", function (event) {
+ var el = event.currentTarget;
+ is(event.type, "error", "Expected event of type 'error'");
+ ok(el.error, "Element 'error' attr expected to have a value");
+ ok(el.error instanceof MediaError, "Element 'error' attr expected to be MediaError");
+ if (v.readyState == v.HAVE_NOTHING) {
+ is(el.error.code, MediaError.MEDIA_ERR_SRC_NOT_SUPPORTED, "Expected media not supported error");
+ } else {
+ is(el.error.code, MediaError.MEDIA_ERR_DECODE, "Expected a decode error");
+ }
+ ok(typeof el.error.message === 'string' || el.error.message instanceof String, "Element 'message' attr expected to be a string");
+ ok(el.error.message.length, "Element 'message' attr has content");
+ el._sawError = true;
+ manager.finished(token);
+ });
+
+ v.addEventListener("loadeddata", function () {
+ ok(false, "Unexpected loadeddata event");
+ manager.finished(token);
+ });
+
+ v.autoplay = true;
+ v.addEventListener("ended", function () {
+ ok(false, "Unexpected ended event");
+ manager.finished(token);
+ });
+
+ v.src = test.name; // implicitly starts a load.
+}
+
+SimpleTest.waitForExplicitFinish();
+SpecialPowers.pushPrefEnv({
+ "set": [
+ ["media.cache_size", 40000],
+ ]
+}, beginTest);
+function beginTest() {
+ manager.runTests(gDecodeErrorTests, startTest);
+}
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_decode_error_crossorigin.html b/dom/media/test/test_decode_error_crossorigin.html
new file mode 100644
index 0000000000..24c1430a5b
--- /dev/null
+++ b/dom/media/test/test_decode_error_crossorigin.html
@@ -0,0 +1,54 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Invalid formats raise decode errors with default messages for CORS cross-origin media</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+const manager = new MediaTestManager;
+let gotErrSrcNotSupported = false;
+let gotErrDecode = false;
+
+function startTest(test, token) {
+ const is = function(a, b, name) {
+ SimpleTest.is(a, b, `${test.name}: ${name}`);
+ };
+ const v = document.createElement("video");
+ manager.started(token);
+ v.addEventListener("error", event => {
+ if (v.readyState == v.HAVE_NOTHING) {
+ is(v.error.code, MediaError.MEDIA_ERR_SRC_NOT_SUPPORTED,
+ "Expected code for a load error");
+ is(v.error.message, "Failed to open media",
+ "Expected message for a load error");
+ gotErrSrcNotSupported = true;
+ } else {
+ is(v.error.code, MediaError.MEDIA_ERR_DECODE,
+ "Expected code for a decode error");
+ is(v.error.message, "Failed to decode media",
+ "Expected message for a decode error");
+ gotErrDecode = true;
+ }
+ manager.finished(token);
+ });
+
+ v.autoplay = true;
+
+ // CORS-cross-origin URL.
+ v.src = `http://example.com/tests/dom/media/test/${test.name}`;
+}
+
+gTestPrefs.push(["media.cache_size", 40000]);
+manager.onFinished = () => {
+ ok(gotErrSrcNotSupported, "At least one test led to src-not-supported");
+ ok(gotErrDecode, "At least one test led to a decode error");
+};
+manager.runTests(gErrorTests, startTest);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_decoder_disable.html b/dom/media/test/test_decoder_disable.html
new file mode 100644
index 0000000000..dd0d2cc51b
--- /dev/null
+++ b/dom/media/test/test_decoder_disable.html
@@ -0,0 +1,78 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=448600
+-->
+<head>
+ <title>Test for Bug 448600</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=448600">Mozilla Bug 448600</a>
+<p id="display"></p>
+
+
+<pre id="test">
+<script type="application/javascript">
+
+function filename(uri) {
+ return uri.substr(uri.lastIndexOf("/")+1);
+}
+
+function e(id) {
+ return document.getElementById(id);
+}
+
+var gLoadError = {};
+
+gLoadError.video1 = 0;
+gLoadError.video2 = 0;
+gLoadError.video3 = 0;
+
+var gErrorCount = 0;
+
+SimpleTest.waitForExplicitFinish();
+
+function finishTest() {
+ is(e('video1').currentSrc,
+ "",
+ 'video1 currentSrc should be empty when there\'s no playable typed source children');
+ is(filename(e('video2').currentSrc),
+ filename(e('video2').src),
+ 'video2 currentSrc should match src');
+ is(filename(e('video3').currentSrc),
+ filename(e('video3').src),
+ 'video3 currentSrc should match src');
+
+ is(gLoadError.video1, 2, "Expect one error per invalid source child on video1");
+ is(gLoadError.video2, 1, "Expect one error on video2");
+ is(gLoadError.video3, 1, "Expect one error on video3");
+
+ SimpleTest.finish();
+}
+
+function videoError(event, id) {
+ gLoadError[id]++;
+ gErrorCount++;
+ if (gErrorCount >= 4) {
+ finishTest();
+ }
+}
+
+</script>
+<!-- We make the resource URIs unique to ensure that they are (re)loaded with the new disable-decoder prefs. -->
+<div id="content">
+</div>
+<script>
+function makeVideos() {
+ document.getElementById('content').innerHTML = '<video id="video1" preload="metadata"><source type="video/ogg" src="320x240.ogv?decoder_disabled=1" onerror="videoError(event, \'video1\');"/><source type="audio/wave" src="r11025_u8_c1.wav?decoder_disabled=1" id=\'s2\' onerror="videoError(event, \'video1\');"/></video><video id="video2" preload="metadata" src="320x240.ogv?decoder_disabled=2" onerror="videoError(event, \'video2\');"></video><video id="video3" preload="metadata" src="r11025_u8_c1.wav?decoder_disabled=2" onerror="videoError(event, \'video3\');"></video>';
+}
+
+SpecialPowers.pushPrefEnv({"set": [["media.ogg.enabled", false], ["media.wave.enabled", false]]}, makeVideos);
+</script>
+
+</pre>
+
+</body>
+</html>
diff --git a/dom/media/test/test_defaultMuted.html b/dom/media/test/test_defaultMuted.html
new file mode 100644
index 0000000000..77bfa3d29a
--- /dev/null
+++ b/dom/media/test/test_defaultMuted.html
@@ -0,0 +1,54 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: defaultMuted</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="../../../dom/html/test/reflect.js"></script>
+</head>
+<body>
+ <a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=706731">Mozilla Bug 706731</a>
+ <p id="display"></p>
+ <div id="content" style="display: none"></div>
+ <video id='v1'></video><audio id='a1'></audio>
+ <video id='v2' muted></video><audio id='a2' muted></audio>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+ reflectBoolean({
+ element: document.createElement("video"),
+ attribute: { content: "muted", idl: "defaultMuted" },
+ });
+
+ reflectBoolean({
+ element: document.createElement("audio"),
+ attribute: { content: "muted", idl: "defaultMuted" },
+ });
+
+ var v1 = document.getElementById('v1');
+ var a1 = document.getElementById('a1');
+ var v2 = document.getElementById('v2');
+ var a2 = document.getElementById('a2');
+
+ // Check that muted state correspond to the default value.
+ is(v1.muted, false, "v1.muted should be false by default");
+ is(a1.muted, false, "a1.muted should be false by default");
+ is(v2.muted, true, "v2.muted should be true by default");
+ is(a2.muted, true, "a2.muted should be true by default");
+
+ // Changing defaultMuted value should not change current muted state.
+ v1.defaultMuted = true;
+ a1.defaultMuted = true;
+ v2.defaultMuted = false;
+ a2.defaultMuted = false;
+
+ is(v1.muted, false, "v1.muted should not have changed");
+ is(a1.muted, false, "a1.muted should not have changed");
+ is(v2.muted, true, "v2.muted should not have changed");
+ is(a2.muted, true, "a2.muted should not have changed");
+
+ mediaTestCleanup();
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_delay_load.html b/dom/media/test/test_delay_load.html
new file mode 100644
index 0000000000..05877aa911
--- /dev/null
+++ b/dom/media/test/test_delay_load.html
@@ -0,0 +1,108 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=479711
+-->
+<head>
+ <title>Test for Bug 479711</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+ <script>
+
+ var gRegisteredElements = [];
+ var testWindows = [];
+
+ function register(v) {
+ gRegisteredElements.push(v);
+ }
+
+ function loaded() {
+ info("onload fired!");
+
+ for (var i = 0; i < gRegisteredElements.length; ++i) {
+ var v = gRegisteredElements[i];
+ ok(v.readyState >= v.HAVE_CURRENT_DATA,
+ v._name + ":" + v.id + " is not ready before onload fired (" + v.readyState + ")");
+ }
+
+ for (i=0; i<testWindows.length; ++i) {
+ testWindows[i].close();
+ }
+
+ mediaTestCleanup();
+
+ SimpleTest.finish();
+ }
+
+ addLoadEvent(loaded);
+
+ </script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=479711">Mozilla Bug 479711</a>
+<p id="display"></p>
+<div id="content" style="display: none"></div>
+<pre id="test">
+<script type="application/javascript">
+
+/** Test for Bug 479711 **/
+
+function createVideo(name, type, id) {
+ var v = document.createElement("video");
+ v.preload = "metadata";
+ // Make sure each video is a unique resource
+ v.src = name + "?" + id;
+ v._name = name;
+ v.id = id;
+ register(v);
+ return v;
+}
+
+var test = getPlayableVideo(gSmallTests);
+
+// Straightforward add, causing a load.
+var v = createVideo(test.name, test.type, "1");
+document.body.appendChild(v);
+
+// Load, add, then remove.
+v = createVideo(test.name, test.type, "1");
+v.load();
+document.body.appendChild(v);
+v.remove();
+
+// Load and add.
+v = createVideo(test.name, test.type, "2");
+v.load();
+document.body.appendChild(v);
+
+// Load outside of doc.
+v = createVideo(test.name, test.type, "3");
+v.load();
+
+// Open a new window for the following test. We open it here instead of in
+// the event handler to ensure that our document load event doesn't fire while
+// window.open is spinning the event loop.
+var w = window.open("", "testWindow", "width=400,height=400");
+testWindows.push(w);
+
+v = createVideo(test.name, test.type, "4");
+v.onloadstart = function(e) {
+ // Using a new window to do this is a bit annoying, but if we use an iframe here,
+ // delaying of the iframe's load event might interfere with the firing of our load event
+ // in some confusing way. So it's simpler just to use another window.
+ w.document.body.appendChild(v);
+};
+v.load(); // load started while in this document, this doc's load will block until
+ // the video's finished loading (in the other document).
+
+if (gRegisteredElements.length) {
+ SimpleTest.waitForExplicitFinish();
+} else {
+ todo(false, "No types supported");
+}
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_duration_after_error.html b/dom/media/test/test_duration_after_error.html
new file mode 100644
index 0000000000..ad6bbe414f
--- /dev/null
+++ b/dom/media/test/test_duration_after_error.html
@@ -0,0 +1,54 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test playback of media files that should have errors</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function checkDuration(name, e, test) {
+ if (test.duration) {
+ ok(Math.abs(e.duration - test.duration) < 0.1,
+ name + " duration (" + e.duration + ") should be around " + test.duration);
+ } else {
+ ok(false, "Test doesn't include the duration field!")
+ }
+}
+
+function startTest(test, token) {
+ manager.started(token);
+
+ let v = document.createElement('video');
+ v._loadedMetadata = false;
+ let name = test.name;
+
+ v.onloadedmetadata = function() {
+ v.onloadedmetadata = null;
+ v._loadedMetadata = true;
+ ok(v._loadedMetadata , name + " has loaded metadata.");
+ }
+
+ v.onerror = function() {
+ v.onerror = null;
+ ok(v._loadedMetadata , name + " should load metadata before getting error.");
+ checkDuration(name, v, test);
+ manager.finished(token);
+ }
+
+ v.src = name;
+ document.body.appendChild(v);
+ v.play();
+}
+
+manager.runTests(gDurationTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_autoplay.html b/dom/media/test/test_eme_autoplay.html
new file mode 100644
index 0000000000..c8947f63e0
--- /dev/null
+++ b/dom/media/test/test_eme_autoplay.html
@@ -0,0 +1,115 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test Encrypted Media Extensions</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="https://example.com:443/tests/dom/media/test/eme.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+/* import-globals-from eme.js */
+var manager = new MediaTestManager;
+
+var EMEmanifest = [
+ {
+ name:"bipbop 10s",
+ tracks: [
+ {
+ name:"video",
+ type:"video/mp4; codecs=\"avc1.4d4015\"",
+ fragments:[ "bipbop-cenc-video-10s.mp4",
+ ]
+ }
+ ],
+ keys: {
+ "7e571d037e571d037e571d037e571d11" : "7e5733337e5733337e5733337e573311",
+ },
+ sessionType:"temporary",
+ sessionCount:1,
+ duration:10.01
+ },
+];
+
+function startTest(test, token)
+{
+ manager.started(token);
+
+ let v = document.createElement("video");
+ v.controls = true;
+ v.autoplay = true;
+
+ document.body.appendChild(v);
+
+ var eventCounts = { play: 0, playing: 0};
+ function ForbiddenEvents(e) {
+ var video = e.target;
+ ok(video.readyState >= video.HAVE_FUTURE_DATA, "Must not have received event too early");
+ is(eventCounts[e.type], 0, "event should have only be fired once");
+ eventCounts[e.type]++;
+ }
+ // Log events for debugging.
+ var events = ["suspend", "play", "canplay", "canplaythrough", "loadstart", "loadedmetadata",
+ "loadeddata", "playing", "ended", "error", "stalled", "emptied", "abort",
+ "waiting", "pause", "durationchange", "seeking", "seeked"];
+ function logEvent(e) {
+ info("got " + e.type + " event");
+ }
+ events.forEach(function(e) {
+ v.addEventListener(e, logEvent);
+ });
+ v.addEventListener("play", ForbiddenEvents);
+ v.addEventListener("playing", ForbiddenEvents);
+
+ var gotWaitingForKey = 0;
+
+ let waitForKey = new EMEPromise;
+ v.addEventListener("waitingforkey", function() {
+ gotWaitingForKey += 1;
+ waitForKey.resolve();
+ });
+
+ v.addEventListener("loadedmetadata", function() {
+ ok(SpecialPowers.do_lookupGetter(v, "isEncrypted").apply(v),
+ TimeStamp(token) + " isEncrypted should be true");
+ is(v.isEncrypted, undefined, "isEncrypted should not be accessible from content");
+ });
+
+ let finish = new EMEPromise;
+ v.addEventListener("playing", function() {
+ ok(true, TimeStamp(token) + " got playing event");
+ // We expect only one waitingForKey as we delay until all sessions are ready.
+ // I.e. one waitingForKey should be fired, after which, we process all sessions,
+ // so it should not be possible to be blocked by a key after that point.
+ ok(gotWaitingForKey == 1, "Expected number 1 wait, got: " + gotWaitingForKey);
+
+ finish.resolve();
+ });
+
+ Promise.all([
+ LoadInitData(v, test, token),
+ CreateAndSetMediaKeys(v, test, token),
+ LoadTest(test, v, token, false /* do not call endOfStream */),
+ waitForKey.promise])
+ .then(values => {
+ let initData = values[0];
+ return ProcessInitData(v, test, token, initData);
+ })
+ .then(sessions => {
+ Log(token, "Updated all sessions, loading complete");
+ finish.promise.then(() => CloseSessions(v, sessions));
+ return finish.promise;
+ })
+ .catch(reason => ok(false, reason))
+ .then(() => manager.finished(token));
+}
+
+SimpleTest.waitForExplicitFinish();
+manager.runTests(EMEmanifest, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_canvas_blocked.html b/dom/media/test/test_eme_canvas_blocked.html
new file mode 100644
index 0000000000..22986493c4
--- /dev/null
+++ b/dom/media/test/test_eme_canvas_blocked.html
@@ -0,0 +1,58 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test Encrypted Media Extensions</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="eme.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var manager = new MediaTestManager();
+
+function startTest(test, token) {
+ manager.started(token);
+
+ let v = document.createElement("video");
+ v.preload = "auto"; // Required due to "canplay" not firing for MSE unless we do this.
+
+ var p1 = new EMEPromise();
+ v.addEventListener("loadeddata", function(ev) {
+ var video = ev.target;
+ var canvas = document.createElement("canvas");
+ canvas.width = video.videoWidth;
+ canvas.height = video.videoHeight;
+ document.body.appendChild(canvas);
+ var ctx = canvas.getContext("2d");
+ ctx.drawImage(video, 0, 0);
+ if (canvas.width || canvas.height) {
+ let pixels = ctx.getImageData(0, 0, canvas.width, canvas.height).data;
+ for (let byte of pixels) {
+ if (byte != 0) {
+ ok(false, "Should not draw EME video to canvas");
+ // We don't need a log for each pixel, break after 1st failure.
+ break;
+ }
+ }
+ }
+ p1.resolve();
+ });
+
+ let p2 = SetupEME(v, test, token);
+
+ Promise.all([p1.promise, p2])
+ .catch(reason => ok(false, reason))
+ .then(() => {
+ CleanUpMedia(v);
+ manager.finished(token);
+ });
+}
+
+SimpleTest.waitForExplicitFinish();
+manager.runTests(gEMETests, startTest);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_createMediaKeys_iframes.html b/dom/media/test/test_eme_createMediaKeys_iframes.html
new file mode 100644
index 0000000000..6fe3dc9809
--- /dev/null
+++ b/dom/media/test/test_eme_createMediaKeys_iframes.html
@@ -0,0 +1,192 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test creation of MediaKeys in iframes</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="eme.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody">
+// Helper functions.
+
+// We take navigator explicitly as an argument to avoid ambiguity in fetching
+// it. This is to avoid issues with the following
+// ```
+// iframe.contentWindow.createMediaKeys = createMediaKeys;
+// await iframe.contentWindow.createMediaKeys();
+// ```
+// If we don't pass a navigator, and just use `navigator` in the function, this
+// ends up being equivalent to
+// ```
+// iframe.contentWindow.createMediaKeys = createMediaKeys;
+// await iframe.contentWindow.createMediaKeys(window.navigator);
+// ```
+// i.e. the function will use the navigator from the global window for the top
+// browsing context, not the iframe's. This would result in the tests not
+// correctly testing within the iframe.
+async function createMediaKeys(aNavigator) {
+ const clearKeyOptions = [
+ {
+ initDataTypes: ["webm"],
+ videoCapabilities: [{ contentType: 'video/webm; codecs="vp9"' }],
+ },
+ ];
+
+ let access = await aNavigator.requestMediaKeySystemAccess(
+ "org.w3.clearkey",
+ clearKeyOptions
+ );
+
+ return access.createMediaKeys();
+}
+// End helper functions.
+
+// These tests check that the following work using different iframe combinations
+// - navigator.requestMediaKeySystem(...) successfully grants access.
+// - the resulting MediaKeySystemAccess object's createMediaKeys() creates
+// MediaKeys as expected.
+
+// Same origin iframe, using src attribute, wait for onload.
+add_task(async () => {
+ info(
+ "Starting same origin iframe, using src attribute, wait for onload test"
+ );
+ let iframe = document.createElement("iframe");
+ let iframeLoadPromise = new Promise(r => {
+ iframe.onload = r;
+ });
+ iframe.src = "file_eme_createMediaKeys.html";
+ document.body.appendChild(iframe);
+ await iframeLoadPromise;
+ info("iframe loaded");
+
+ // Setup our handler for when the iframe messages to tell us if it
+ // created MediaKeys or not.
+ let iframeMessagePromise = new Promise(r => {
+ window.onmessage = message => {
+ is(
+ message.data,
+ "successCreatingMediaKeys",
+ "iframe should have posted us a message saying keys were successfully created"
+ );
+ r();
+ };
+ });
+ // Post a message to the iframe to ask it to try and create media keys.
+ iframe.contentWindow.postMessage("", "*");
+ // Wait until we've got a message back from our iframe.
+ await iframeMessagePromise;
+});
+
+// Same origin iframe, call via JS, wait for onload.
+add_task(async () => {
+ info("Starting same origin iframe, call via JS, wait for onload test");
+ let iframe = document.createElement("iframe");
+ let iframeLoadPromise = new Promise(r => {
+ iframe.onload = r;
+ });
+ iframe.src = ""; // No src iframes are same origin.
+ document.body.appendChild(iframe);
+ await iframeLoadPromise;
+ info("iframe loaded");
+
+ try {
+ iframe.contentWindow.createMediaKeys = createMediaKeys;
+ let mediaKeys = await iframe.contentWindow.createMediaKeys(
+ iframe.contentWindow.navigator
+ );
+ ok(mediaKeys, "Should get media keys");
+ } catch (e) {
+ ok(
+ false,
+ `Should not get any errors while trying to get media keys, got ${e}`
+ );
+ }
+});
+
+// Same origin iframe, call via JS, *do not* wait for onload.
+//
+// Note, sites shouldn't do this, because
+// https://bugzilla.mozilla.org/show_bug.cgi?id=543435
+// means not waiting for onload results in weird behavior, however
+// https://bugzilla.mozilla.org/show_bug.cgi?id=1675360
+// shows sites doing this in the wild because historically this worked in
+// Firefox.
+//
+// Breaking this test case isn't necessarily against any specifications
+// I'm (bryce) aware of, but it will probably break site compat, so be really
+// sure you want to before doing so.
+add_task(async () => {
+ info(
+ "Starting same origin iframe, call via JS, *do not* wait for onload test"
+ );
+ let iframe = document.createElement("iframe");
+ let iframeLoadPromise = new Promise(r => {
+ iframe.onload = r;
+ });
+ iframe.src = ""; // No src iframes are same origin.
+ document.body.appendChild(iframe);
+ info("iframe appended (we're not waiting for load)");
+
+ try {
+ iframe.contentWindow.createMediaKeys = createMediaKeys;
+ let mediaKeys = await iframe.contentWindow.createMediaKeys(
+ iframe.contentWindow.navigator
+ );
+ ok(mediaKeys, "Should get media keys");
+
+ // We await the load to see if they keys persist through the load.
+ // This could fail if gecko internally associates the keys with the
+ // about:blank page that is replaced by the load.
+ await iframeLoadPromise;
+ ok(mediaKeys, "Media keys should still exist after the load");
+ } catch (e) {
+ ok(
+ false,
+ `Should not get any errors while trying to get media keys, got ${e}`
+ );
+ }
+});
+
+// Different origin iframe, using src attribute, wait for onload
+add_task(async () => {
+ info(
+ "Starting different origin iframe, using src attribute, wait for onload test"
+ );
+ let iframe = document.createElement("iframe");
+ let iframeLoadPromise = new Promise(r => {
+ iframe.onload = r;
+ });
+ // Make our iframe cross origin (see build/pgo/server-locations.txt for more
+ // info the url used).
+ iframe.src =
+ "https://w3c-test.org:443/tests/dom/media/test/file_eme_createMediaKeys.html";
+ iframe.allow = "encrypted-media";
+ document.body.appendChild(iframe);
+ await iframeLoadPromise;
+ info("iframe loaded");
+
+ // Setup our handler for when the iframe messages to tell us if it
+ // created MediaKeys or not.
+ let iframeMessagePromise = new Promise(r => {
+ window.onmessage = message => {
+ is(
+ message.data,
+ "successCreatingMediaKeys",
+ "iframe should have posted us a message saying keys were successfully created"
+ );
+ r();
+ };
+ });
+ // Post a message to the iframe to ask it to try and create media keys.
+ iframe.contentWindow.postMessage("", "*");
+ // Wait until we've got a message back from our iframe.
+ await iframeMessagePromise;
+});
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_detach_media_keys.html b/dom/media/test/test_eme_detach_media_keys.html
new file mode 100644
index 0000000000..69b812032a
--- /dev/null
+++ b/dom/media/test/test_eme_detach_media_keys.html
@@ -0,0 +1,63 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test Encrypted Media Extensions</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="eme.js"></script>
+</head>
+<body>
+<pre id="test">
+<video id="v" controls></video>
+<script class="testbody" type="text/javascript">
+
+SimpleTest.waitForExplicitFinish();
+
+function createAndSet() {
+ return new Promise(function(resolve, reject) {
+ var m;
+ navigator.requestMediaKeySystemAccess(CLEARKEY_KEYSYSTEM, gCencMediaKeySystemConfig)
+ .then(function (access) {
+ return access.createMediaKeys();
+ })
+ .then(function (mediaKeys) {
+ m = mediaKeys;
+ return document.getElementById("v").setMediaKeys(mediaKeys);
+ })
+ .then(function() {
+ resolve(m);
+ });
+ }
+)}
+
+var m1,m2;
+
+// Test that if we create and set two MediaKeys on one video element,
+// that if the first MediaKeys we set on the media elemnt is still usable
+// after the second MediaKeys has been set on the media element.
+
+createAndSet().then((m) => {
+ m1 = m; // Stash MediaKeys.
+ return createAndSet();
+})
+.then((m) => {
+ m2 = m;
+ is(document.getElementById("v").mediaKeys, m2, "Should have set MediaKeys on media element");
+ ok(document.getElementById("v").mediaKeys != m1, "First MediaKeys should no longer be set on media element");
+ var s = m1.createSession("temporary");
+ return s.generateRequest("webm", StringToArrayBuffer(atob('YAYeAX5Hfod+V9ANHtANHg==')));
+})
+.then(() => {
+ ok(true, "Was able to generateRequest using second CDM");
+ SimpleTest.finish();
+}, () => {
+ ok(false, "Was *NOT* able to generateRequest using second CDM");
+ SimpleTest.finish();
+});
+
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_detach_reattach_same_mediakeys_during_playback.html b/dom/media/test/test_eme_detach_reattach_same_mediakeys_during_playback.html
new file mode 100644
index 0000000000..e47fae5891
--- /dev/null
+++ b/dom/media/test/test_eme_detach_reattach_same_mediakeys_during_playback.html
@@ -0,0 +1,141 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test Encrypted Media Extensions</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="https://example.com:443/tests/dom/media/test/eme.js"></script>
+</head>
+<body>
+<pre id="test">
+<video id="v" controls></video>
+<script class="testbody" type="text/javascript">
+/* import-globals-from eme.js */
+var manager = new MediaTestManager;
+
+var EMEmanifest = [
+ {
+ name:"bipbop 10s",
+ tracks: [
+ {
+ name:"video",
+ type:"video/mp4; codecs=\"avc1.4d4015\"",
+ fragments:[ "bipbop-cenc-video-10s.mp4",
+ ]
+ }
+ ],
+ keys: {
+ "7e571d037e571d037e571d037e571d11" : "7e5733337e5733337e5733337e573311",
+ },
+ sessionType:"temporary",
+ sessionCount:1,
+ duration:10.01
+ },
+];
+
+function sleep(time) {
+ return new Promise((resolve) => setTimeout(resolve, time));
+}
+
+// To check if playback can be blocked and resumed correctly after
+// detaching original mediakeys and reattach it back.
+function startTest(test, token)
+{
+ manager.started(token);
+
+ var mk_ori;
+ let finish = new EMEPromise;
+
+ let v = document.getElementById("v");
+ let sessions = [];
+ function onSessionCreated(session) {
+ sessions.push(session);
+ }
+
+ function closeSessions() {
+ let p = new EMEPromise;
+ Promise.all(sessions.map(s => s.close()))
+ .then(p.resolve, p.reject);
+ return p.promise;
+ }
+
+ function setMediaKeysToElement(mk, solve, reject) {
+ v.setMediaKeys(mk).then(solve, reject);
+ }
+
+ function ReattachOriMediaKeys() {
+ function onOriMediaKeysSetOK() {
+ ok(true, TimeStamp(token) + " (ENCRYPTED) Set original MediaKeys back OK!");
+ }
+ function onOriMediaKeysSetFailed() {
+ ok(false, " Failed to set original mediakeys back.");
+ }
+
+ function onCanPlayAgain(ev) {
+ Promise.all([closeSessions()])
+ .then(() => {
+ ok(true, " (ENCRYPTED) Playback can be resumed.");
+ manager.finished(token);
+ }, () => {
+ ok(false, TimeStamp(token) + " Sessions are closed incorrectly.");
+ manager.finished(token);
+ });
+ }
+
+ once(v, "canplay", onCanPlayAgain);
+ setMediaKeysToElement(mk_ori, onOriMediaKeysSetOK, onOriMediaKeysSetFailed)
+ }
+
+ function triggerSeek() {
+ v.currentTime = v.duration / 2;
+ }
+
+ function onCanPlay(ev) {
+ function onSetMediaKeysToNullOK() {
+ ok(true, TimeStamp(token) + " Set MediaKeys to null. OK!");
+
+ triggerSeek();
+
+ SimpleTest.requestFlakyTimeout("To reattach mediakeys back again in 5s.");
+ sleep(5000).then(ReattachOriMediaKeys);
+ }
+ function onSetMediaKeysToNullFailed() {
+ ok(false, TimeStamp(token) + " Set MediaKeys to null. FAILED!");
+ }
+
+ SimpleTest.requestFlakyTimeout("To detach mediakeys after receiving 'canplay' event in 2s");
+ sleep(2000).then(() => {
+ setMediaKeysToElement(null, onSetMediaKeysToNullOK, onSetMediaKeysToNullFailed);
+ });
+ }
+
+ once(v, "canplay", onCanPlay);
+
+ var p1 = LoadInitData(v, test, token);
+ var p2 = CreateAndSetMediaKeys(v, test, token);
+ var p3 = LoadTest(test, v, token);
+ Promise.all([p1, p2, p3])
+ .then(values => {
+ let initData = values[0];
+ // stash the mediakeys
+ mk_ori = v.mediaKeys;
+ initData.map(ev => {
+ let session = v.mediaKeys.createSession();
+ onSessionCreated(session);
+ MakeRequest(test, token, ev, session);
+ });
+ })
+ .then(() => {
+ return finish.promise;
+ })
+ .catch(reason => ok(false, reason))
+}
+
+SimpleTest.waitForExplicitFinish();
+manager.runTests(EMEmanifest, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_getstatusforpolicy.html b/dom/media/test/test_eme_getstatusforpolicy.html
new file mode 100644
index 0000000000..2031d04df5
--- /dev/null
+++ b/dom/media/test/test_eme_getstatusforpolicy.html
@@ -0,0 +1,93 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test getStatusForPolicy on ClearKey CDM</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="eme.js"></script>
+</head>
+<body>
+<pre id="test">
+<video id="v" controls></video>
+<script class="testbody" type="text/javascript">
+
+add_task(async function setupTestingPrefs() {
+ await SpecialPowers.pushPrefEnv({
+ set: [
+ ["media.eme.hdcp-policy-check.enabled", true],
+ ],
+ });
+});
+
+// We expect that ClearKey has HDCP 2.0 compliant.
+const expectedResults = [
+ {
+ minHdcpVersion : "1.0",
+ expectedResult : "usable"
+ },
+ {
+ minHdcpVersion : "1.1",
+ expectedResult : "usable"
+ },
+ {
+ minHdcpVersion : "1.2",
+ expectedResult : "usable"
+ },
+ {
+ minHdcpVersion : "1.3",
+ expectedResult : "usable"
+ },
+ {
+ minHdcpVersion : "1.4",
+ expectedResult : "usable"
+ },
+ {
+ minHdcpVersion : "2.0",
+ expectedResult : "usable"
+ },
+ {
+ minHdcpVersion : "2.1",
+ expectedResult : "usable"
+ },
+ {
+ minHdcpVersion : "2.2",
+ expectedResult : "output-restricted"
+ },
+ {
+ minHdcpVersion : "2.3",
+ expectedResult : "output-restricted"
+ },
+];
+
+add_task(async function testGetStatusForPolicy() {
+ for (let result of expectedResults) {
+ let mediaKey = await createMediaKeysAndSet();
+ let video = document.getElementById("v");
+ is(video.mediaKeys, mediaKey,
+ "Should have set MediaKeys on media element");
+ let keyStatus = await
+ video.mediaKeys.getStatusForPolicy({minHdcpVersion : result.minHdcpVersion})
+ .catch(e => ok(false, "getStatusForPolicy failed!"));
+ info(`getStatusForPolicy for HDCP ${result.minHdcpVersion} : ${keyStatus}`);
+ is(keyStatus, result.expectedResult,
+ `Expected ${result.expectedResult}, got ${keyStatus}`);
+ }
+});
+
+// Helper function
+function createMediaKeysAndSet() {
+ return navigator.requestMediaKeySystemAccess(CLEARKEY_KEYSYSTEM, gCencMediaKeySystemConfig)
+ .then(function (access) {
+ return access.createMediaKeys();
+ })
+ .then(function (mediaKeys) {
+ document.getElementById("v").setMediaKeys(mediaKeys);
+ return mediaKeys;
+ });
+}
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_initDataTypes.html b/dom/media/test/test_eme_initDataTypes.html
new file mode 100644
index 0000000000..587e6fc161
--- /dev/null
+++ b/dom/media/test/test_eme_initDataTypes.html
@@ -0,0 +1,130 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test Encrypted Media Extensions</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="eme.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var tests = [
+ {
+ name: "One keyId",
+ initDataType: 'keyids',
+ initData: '{"kids":["LwVHf8JLtPrv2GUXFW2v_A"]}',
+ expectedRequest: '{"kids":["LwVHf8JLtPrv2GUXFW2v_A"],"type":"temporary"}',
+ sessionType: 'temporary',
+ expectPass: true,
+ },
+ {
+ name: "Two keyIds",
+ initDataType: 'keyids',
+ initData: '{"kids":["LwVHf8JLtPrv2GUXFW2v_A", "0DdtU9od-Bh5L3xbv0Xf_A"]}',
+ expectedRequest: '{"kids":["LwVHf8JLtPrv2GUXFW2v_A","0DdtU9od-Bh5L3xbv0Xf_A"],"type":"temporary"}',
+ sessionType: 'temporary',
+ expectPass: true,
+ },
+ {
+ name: "Two keyIds, temporary session",
+ initDataType: 'keyids',
+ initData: '{"type":"temporary", "kids":["LwVHf8JLtPrv2GUXFW2v_A", "0DdtU9od-Bh5L3xbv0Xf_A"]}',
+ expectedRequest: '{"kids":["LwVHf8JLtPrv2GUXFW2v_A","0DdtU9od-Bh5L3xbv0Xf_A"],"type":"temporary"}',
+ sessionType: 'temporary',
+ expectPass: true,
+ },
+ {
+ name: "Two keyIds, persistent session, type before kids",
+ initDataType: 'keyids',
+ initData: '{"type":"persistent-license", "kids":["LwVHf8JLtPrv2GUXFW2v_A", "0DdtU9od-Bh5L3xbv0Xf_A"]}',
+ expectedRequest: '{"kids":["LwVHf8JLtPrv2GUXFW2v_A","0DdtU9od-Bh5L3xbv0Xf_A"],"type":"persistent-license"}',
+ sessionType: 'persistent-license',
+ expectPass: false,
+ },
+ {
+ name: "Invalid keyId",
+ initDataType: 'keyids',
+ initData: '{"kids":["0"]}',
+ sessionType: 'temporary',
+ expectPass: false,
+ },
+ {
+ name: "Empty keyId",
+ initDataType: 'keyids',
+ initData: '{"kids":[""]}',
+ sessionType: 'temporary',
+ expectPass: false,
+ },
+ {
+ name: "Invalid initData",
+ initDataType: 'keyids',
+ initData: 'invalid initData',
+ sessionType: 'temporary',
+ expectPass: false,
+ },
+ {
+ name: "'webm' initDataType",
+ initDataType: 'webm',
+ initData: 'YAYeAX5Hfod+V9ANHtANHg==',
+ expectedRequest: '{"kids":["YAYeAX5Hfod-V9ANHtANHg"],"type":"temporary"}',
+ sessionType: 'temporary',
+ expectPass: true,
+ },
+ {
+ name: "'webm' initDataType with non 16 byte keyid",
+ initDataType: 'webm',
+ initData: 'YAYeAX5Hfod',
+ expectedRequest: '{\"kids\":[\"YAYeAX5Hfoc\"],\"type\":\"temporary\"}',
+ sessionType: 'temporary',
+ expectPass: true,
+ },
+];
+
+function PrepareInitData(initDataType, initData)
+{
+ if (initDataType == "keyids") {
+ return new TextEncoder().encode(initData);
+ } else if (initDataType == "webm") {
+ return StringToArrayBuffer(atob(initData));
+ }
+}
+
+function Test(test) {
+ return new Promise(function(resolve, reject) {
+ var configs = [{
+ initDataTypes: [test.initDataType],
+ videoCapabilities: [{contentType: 'video/mp4' }],
+ }];
+ navigator.requestMediaKeySystemAccess('org.w3.clearkey', configs)
+ .then((access) => access.createMediaKeys())
+ .then((mediaKeys) => {
+ var session = mediaKeys.createSession(test.sessionType);
+ session.addEventListener("message", function(event) {
+ is(event.messageType, "license-request", "'" + test.name + "' MediaKeyMessage type should be license-request.");
+ var text = new TextDecoder().decode(event.message);
+ is(text, test.expectedRequest, "'" + test.name + "' got expected response.");
+ is(text == test.expectedRequest, test.expectPass,
+ "'" + test.name + "' expected to " + (test.expectPass ? "pass" : "fail"));
+ resolve();
+ });
+ var initData = PrepareInitData(test.initDataType, test.initData);
+ return session.generateRequest(test.initDataType, initData);
+ }
+ ).catch((x) => {
+ ok(!test.expectPass, "'" + test.name + "' expected to fail.");
+ resolve();
+ });
+ });
+}
+
+SimpleTest.waitForExplicitFinish();
+Promise.all(tests.map(Test)).then(function() {
+ SimpleTest.finish();
+});
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_mfcdm_generate_request.html b/dom/media/test/test_eme_mfcdm_generate_request.html
new file mode 100644
index 0000000000..57be325e36
--- /dev/null
+++ b/dom/media/test/test_eme_mfcdm_generate_request.html
@@ -0,0 +1,93 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media Engine only test : test EME API for supported key systems</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<script class="testbody" type="text/javascript">
+
+/**
+ * This test is aiming to test whether we can use EME API to generate a request
+ * for all supported key systems. However, we use our mock CDM (MF ClearKey) for
+ * external key systems like Widevine and PlayReady in order to avoid any
+ * possible interfering, such as returning incorrect support types (bug 1851914)
+ * and we can also test hardward DRM key system without being limited by
+ * machine capability.
+ *
+ * As this test doesn't involve any communication with the license server, that
+ * is why we can use our mock CDM (no decryption needed).
+ */
+
+add_task(async function setupTestingPrefs() {
+ await SpecialPowers.pushPrefEnv({
+ set: [
+ ["media.wmf.media-engine.enabled", 2],
+ ["media.eme.playready.enabled", true],
+ ["media.eme.widevine.experiment.enabled", true],
+ // This is used to trigger Widevine CDM installation check
+ ["media.gmp-widevinecdm-l1.enabled", true],
+ ["media.eme.wmf.clearkey.enabled", true],
+ // Use the mock clearkey CDM to create cdm for all other key systems.
+ ["media.eme.wmf.use-mock-cdm-for-external-cdms", true],
+ // Our mock CDM doesn't implement 'IsTypeSupportedEx()', only 'IsTypeSupported()'
+ ["media.eme.playready.istypesupportedex", false],
+ ],
+ });
+});
+
+const kKeySystems = [
+ "com.microsoft.playready.recommendation",
+ "com.microsoft.playready.recommendation.3000",
+ "com.microsoft.playready.recommendation.3000.clearlead",
+ "com.widevine.alpha.experiment",
+ "com.widevine.alpha.experiment2",
+ "org.w3.clearkey",
+];
+
+add_task(async function testKeySystemRequestForMFCDMs() {
+ for (let keySystem of kKeySystems) {
+ await testKeySystemRequest(keySystem);
+ info(`done testing ${keySystem}!`);
+ }
+});
+
+async function testKeySystemRequest(keySystem) {
+ const fakeKID = '{"kids":["a1234567890"]}';
+ // Mock CDM only supports temporary type.
+ const sessionType = 'temporary';
+ const configs = [{
+ initDataTypes: ['keyids'],
+ videoCapabilities: [{ contentType: `video/mp4;codecs="avc1.640028"` }],
+ sessionTypes : [sessionType],
+ }];
+ info(`requestMediaKeySystemAccess for ${keySystem}`);
+ let access = await navigator.requestMediaKeySystemAccess(keySystem, configs)
+ .catch(e => ok(false, `failed to create key system access`));
+
+ info('creating media key');
+ let mediaKeys = await access.createMediaKeys()
+ .catch(e => ok(false, `failed to create media key`));;
+
+ info(`creating a temporary media key session`);
+ let session = mediaKeys.createSession(sessionType);
+ let messagePromise = new Promise(r => {
+ session.addEventListener('message', event => {
+ is(event.messageType, 'license-request',
+ 'MediaKeyMessage type should be license-request');
+ ok(true, `created request and received message event`);
+ session.close().then(() => { r(); });
+ });
+ });
+ await session.generateRequest(
+ 'keyids',
+ new TextEncoder().encode(fakeKID))
+ .catch(e => ok(false, `failed to generate request`));
+ await messagePromise;
+}
+
+</script>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_mfcdm_getstatusforpolicy.html b/dom/media/test/test_eme_mfcdm_getstatusforpolicy.html
new file mode 100644
index 0000000000..7e22be25d9
--- /dev/null
+++ b/dom/media/test/test_eme_mfcdm_getstatusforpolicy.html
@@ -0,0 +1,115 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test getStatusForPolicy on MFCDM</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="eme.js"></script>
+</head>
+<body>
+<pre id="test">
+<video id="v" controls></video>
+<script class="testbody" type="text/javascript">
+
+add_task(async function setupTestingPrefs() {
+ await SpecialPowers.pushPrefEnv({
+ set: [
+ ["media.eme.hdcp-policy-check.enabled", true],
+ ["media.wmf.media-engine.enabled", 2],
+ ["media.eme.playready.enabled", true],
+ ["media.eme.widevine.experiment.enabled", true],
+ // This is used to trigger Widevine CDM installation check
+ ["media.gmp-widevinecdm-l1.enabled", true],
+ ["media.eme.wmf.clearkey.enabled", true],
+ // Use the mock clearkey CDM to create cdm for all other key systems.
+ ["media.eme.wmf.use-mock-cdm-for-external-cdms", true],
+ // Our mock CDM doesn't implement 'IsTypeSupportedEx()', only 'IsTypeSupported()'
+ ["media.eme.playready.istypesupportedex", false],
+ ],
+ });
+});
+
+// We expect that all key system have HDCP 2.1 compliant, which is pre-defined
+// in `WMFClearKeyCDMFactory::IsTypeSupported`.
+const expectedResults = [
+ {
+ minHdcpVersion : "1.0",
+ expectedResult : "usable"
+ },
+ {
+ minHdcpVersion : "1.1",
+ expectedResult : "usable"
+ },
+ {
+ minHdcpVersion : "1.2",
+ expectedResult : "usable"
+ },
+ {
+ minHdcpVersion : "1.3",
+ expectedResult : "usable"
+ },
+ {
+ minHdcpVersion : "1.4",
+ expectedResult : "usable"
+ },
+ {
+ minHdcpVersion : "2.0",
+ expectedResult : "usable"
+ },
+ {
+ minHdcpVersion : "2.1",
+ expectedResult : "usable"
+ },
+ {
+ minHdcpVersion : "2.2",
+ expectedResult : "output-restricted"
+ },
+ {
+ minHdcpVersion : "2.3",
+ expectedResult : "output-restricted"
+ },
+];
+
+const kKeySystems = [
+ "com.microsoft.playready.recommendation",
+ "com.microsoft.playready.recommendation.3000",
+ "com.microsoft.playready.recommendation.3000.clearlead",
+ "com.widevine.alpha.experiment",
+ "com.widevine.alpha.experiment2",
+ "org.w3.clearkey",
+];
+
+add_task(async function testGetStatusForPolicy() {
+ for (let keySystem of kKeySystems) {
+ for (let result of expectedResults) {
+ let mediaKey = await createMediaKeysAndSet(keySystem);
+ let video = document.getElementById("v");
+ is(video.mediaKeys, mediaKey,
+ "Should have set MediaKeys on media element");
+ let keyStatus = await
+ video.mediaKeys.getStatusForPolicy({minHdcpVersion : result.minHdcpVersion})
+ .catch(e => ok(false, "getStatusForPolicy failed!"));
+ info(`getStatusForPolicy for HDCP ${result.minHdcpVersion} : ${keyStatus}`);
+ is(keyStatus, result.expectedResult,
+ `Expected ${result.expectedResult}, got ${keyStatus}`);
+ }
+ }
+});
+
+// Helper function
+function createMediaKeysAndSet(keySystem) {
+ return navigator.requestMediaKeySystemAccess(keySystem, gCencMediaKeySystemConfig)
+ .then(function (access) {
+ return access.createMediaKeys();
+ })
+ .then(function (mediaKeys) {
+ document.getElementById("v").setMediaKeys(mediaKeys);
+ return mediaKeys;
+ });
+}
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_missing_pssh.html b/dom/media/test/test_eme_missing_pssh.html
new file mode 100644
index 0000000000..29f77d021a
--- /dev/null
+++ b/dom/media/test/test_eme_missing_pssh.html
@@ -0,0 +1,92 @@
+<!DOCTYPE HTML>
+<html>
+ <head>
+ <title>Test Encrypted Media Extensions</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="eme.js"></script>
+ </head>
+ <body>
+ <audio controls id="audio"></audio>
+ <pre id="test">
+ <script class="testbody" type="text/javascript">
+
+ // Tests that a fragmented MP4 file without a PSSH, but with valid encrypted
+ // tracks with valid TENC boxes, is able to load with EME.
+ // We setup MSE before starting up EME, so that we exercise the "waiting for
+ // cdm" step in the MediaDecoderStateMachine.
+
+ SimpleTest.waitForExplicitFinish();
+
+ var pssh = [
+ 0x00, 0x00, 0x00, 0x00,
+ 0x70, 0x73, 0x73, 0x68, // BMFF box header (76 bytes, 'pssh')
+ 0x01, 0x00, 0x00, 0x00, // Full box header (version = 1, flags = 0)
+ 0x10, 0x77, 0xef, 0xec, 0xc0, 0xb2, 0x4d, 0x02, // SystemID
+ 0xac, 0xe3, 0x3c, 0x1e, 0x52, 0xe2, 0xfb, 0x4b,
+ 0x00, 0x00, 0x00, 0x01, // KID_count (1)
+ 0x2f, 0xef, 0x8a, 0xd8, 0x12, 0xdf, 0x42, 0x97,
+ 0x83, 0xe9, 0xbf, 0x6e, 0x5e, 0x49, 0x3e, 0x53,
+ 0x00, 0x00, 0x00, 0x00 // Size of Data (0)
+ ];
+
+ var audio = document.getElementById("audio");
+
+ function LoadEME() {
+ var options = [{
+ initDataType: 'cenc',
+ audioType: 'audio/mp4; codecs="mp4a.40.2"',
+ }];
+ navigator.requestMediaKeySystemAccess("org.w3.clearkey", options)
+ .then((keySystemAccess) => {
+ return keySystemAccess.createMediaKeys();
+ }, bail("Failed to request key system access."))
+
+ .then((mediaKeys) => {
+ audio.setMediaKeys(mediaKeys);
+ var session = mediaKeys.createSession();
+ once(session, "message", (message) => {
+ is(message.messageType, 'license-request', "Expected a license-request");
+ var license = new TextEncoder().encode(JSON.stringify({
+ 'keys': [{
+ 'kty':'oct',
+ 'kid':'L--K2BLfQpeD6b9uXkk-Uw',
+ 'k':HexToBase64('7f412f0575f44f718259beef56ec7771')
+ }],
+ 'type': 'temporary'
+ }));
+ session.update(license);
+ });
+ session.generateRequest('cenc', new Uint8Array(pssh));
+ });
+ }
+
+ function DownloadMedia(url, type, mediaSource) {
+ return new Promise(function(resolve, reject) {
+ var sourceBuffer = mediaSource.addSourceBuffer(type);
+ fetchWithXHR(url, (response) => {
+ once(sourceBuffer, "updateend", resolve);
+ sourceBuffer.appendBuffer(new Uint8Array(response));
+ });
+ });
+ }
+
+ function LoadMSE() {
+ var ms = new MediaSource();
+ audio.src = URL.createObjectURL(ms);
+
+ once(ms, "sourceopen", ()=>{
+ DownloadMedia('short-audio-fragmented-cenc-without-pssh.mp4', 'audio/mp4; codecs="mp4a.40.2"', ms)
+ .then(() => { ms.endOfStream(); LoadEME();});
+ });
+
+ audio.addEventListener("loadeddata", SimpleTest.finish);
+ }
+
+ LoadMSE();
+
+ </script>
+ </pre>
+ </body>
+</html>
diff --git a/dom/media/test/test_eme_non_mse_fails.html b/dom/media/test/test_eme_non_mse_fails.html
new file mode 100644
index 0000000000..6ff17d59ff
--- /dev/null
+++ b/dom/media/test/test_eme_non_mse_fails.html
@@ -0,0 +1,95 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Bug 1131392 - Test that EME does not work for non-MSE media</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="eme.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+/* import-globals-from eme.js */
+var manager = new MediaTestManager;
+
+function DoSetMediaKeys(v, test, token)
+{
+ var options = [{
+ initDataTypes: ["cenc"],
+ audioCapabilities: [{contentType: test.audioType}],
+ videoCapabilities: [{contentType: test.videoType}],
+ }];
+
+ return navigator.requestMediaKeySystemAccess(CLEARKEY_KEYSYSTEM, options)
+
+ .then(function(keySystemAccess) {
+ return keySystemAccess.createMediaKeys();
+ })
+
+ .catch(function() {
+ ok(false, token + " was not expecting failure (yet)");
+ })
+
+ .then(function(mediaKeys) {
+ return v.setMediaKeys(mediaKeys);
+ });
+}
+
+function TestSetMediaKeys(test, token)
+{
+ manager.started(token);
+
+ var v = document.createElement("video");
+
+ v.addEventListener("encrypted", function() {
+ ok(false, token + " should not fire encrypted event");
+ });
+
+ var loadedMetadata = false;
+ v.addEventListener("loadedmetadata", function() {
+ loadedMetadata = true;
+ });
+
+ v.addEventListener("error", function() {
+ ok(true, token + " expected error event");
+ ok(loadedMetadata, token + " expected loadedmetadata to have fired");
+ manager.finished(token);
+ });
+
+ v.src = test.name;
+}
+
+function TestSetSrc(test, token)
+{
+ manager.started(token);
+
+ var v = document.createElement("video");
+ v.addEventListener("error", function(err) {
+ ok(true, token + " got error setting src on video element, as expected");
+ manager.finished(token);
+ });
+
+ DoSetMediaKeys(v, test, token)
+
+ .then(function() {
+ v.src = test.name;
+ })
+
+ .catch(function() {
+ ok(false, token + " got error setting media keys");
+ });
+}
+
+function startTest(test, token)
+{
+ TestSetMediaKeys(test, token + "_setMediaKeys");
+ TestSetSrc(test, token + "_setSrc");
+}
+
+SimpleTest.waitForExplicitFinish();
+manager.runTests(gEMENonMSEFailTests, startTest);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_playback.html b/dom/media/test/test_eme_playback.html
new file mode 100644
index 0000000000..bcfa058e34
--- /dev/null
+++ b/dom/media/test/test_eme_playback.html
@@ -0,0 +1,188 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test Encrypted Media Extensions</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="https://example.com:443/tests/dom/media/test/eme.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+/* import-globals-from eme.js */
+var manager = new MediaTestManager;
+
+function ArrayBuffersEqual(a, b) {
+ if (a.byteLength != b.byteLength) {
+ return false;
+ }
+ var ua = new Uint8Array(a);
+ var ub = new Uint8Array(b);
+ for (var i = 0; i < ua.length; i++) {
+ if (ua[i] != ub[i]) {
+ return false;
+ }
+ }
+ return true;
+}
+
+function KeysChangeFunc(session, keys, token) {
+ session.keyIdsReceived = [];
+ for (var keyid in keys) {
+ Log(token, "Set " + keyid + " to false in session[" + session.sessionId + "].keyIdsReceived");
+ session.keyIdsReceived[keyid] = false;
+ }
+ return function(ev) {
+ var s = ev.target;
+ s.gotKeysChanged = true;
+
+ var keyList = [];
+ var valueList = [];
+ var map = s.keyStatuses;
+
+ // Test that accessing keys not known to the CDM has expected behaviour.
+ var absentKey = new Uint8Array([0xeb, 0xdd, 0x62, 0xf1, 0x68, 0x14, 0xd2, 0x7b,
+ 0x68, 0xef, 0x12, 0x2a, 0xfc, 0xe4, 0xae, 0x3c]);
+ is(map.has(absentKey), false, "Shouldn't have a key that's not in the media");
+ is(map.get(absentKey), undefined, "Unknown keys should undefined status");
+
+ // Verify known keys have expected status.
+ for (let [key, val] of map.entries()) {
+ is(key.constructor, ArrayBuffer, "keyId should be ArrayBuffer");
+ ok(map.has(key), "MediaKeyStatusMap.has() should work.");
+ is(map.get(key), val, "MediaKeyStatusMap.get() should work.");
+ keyList.push(key);
+ valueList.push(val);
+ is(val, "usable", token + ": key status should be usable");
+ var kid = Base64ToHex(window.btoa(ArrayBufferToString(key)));
+ ok(kid in s.keyIdsReceived, TimeStamp(token) + " session[" + s.sessionId + "].keyIdsReceived contained " + kid + " as expected.");
+ s.keyIdsReceived[kid] = true;
+ }
+
+ var index = 0;
+ for (var keyId of map.keys()) {
+ ok(ArrayBuffersEqual(keyId, keyList[index]), "MediaKeyStatusMap.keys() should correspond to entries");
+ index++;
+ }
+ index = 0;
+ for (let val of map.values()) {
+ is(val, valueList[index], "MediaKeyStatusMap.values() should correspond to entries");
+ index++;
+ }
+ }
+}
+
+function startTest(test, token)
+{
+ manager.started(token);
+
+ var sessions = [];
+
+ function onSessionCreated(session) {
+ sessions.push(session);
+ session.addEventListener("keystatuseschange", KeysChangeFunc(session, test.keys, token));
+
+ session.numKeystatuseschangeEvents = 0;
+ session.numOnkeystatuseschangeEvents = 0;
+
+ session.addEventListener("keystatuseschange", function() {
+ session.numKeystatuseschangeEvents += 1;
+ });
+ session.onkeystatuseschange = function() {
+ session.numOnkeystatuseschangeEvents += 1;
+ };
+
+ session.numMessageEvents = 0;
+ session.numOnMessageEvents = 0;
+ session.addEventListener("message", function() {
+ session.numMessageEvents += 1;
+ });
+ session.onmessage = function() {
+ session.numOnMessageEvents += 1;
+ };
+ }
+
+ let v = document.createElement("video");
+ document.body.appendChild(v);
+
+ var gotEncrypted = 0;
+ let finish = new EMEPromise;
+
+ v.addEventListener("encrypted", function(ev) {
+ gotEncrypted += 1;
+ });
+
+ v.addEventListener("loadedmetadata", function() {
+ ok(SpecialPowers.do_lookupGetter(v, "isEncrypted").apply(v),
+ TimeStamp(token) + " isEncrypted should be true");
+ is(v.isEncrypted, undefined, "isEncrypted should not be accessible from content");
+ });
+
+ v.addEventListener("ended", function(ev) {
+ ok(true, TimeStamp(token) + " got ended event");
+
+ is(gotEncrypted, test.sessionCount,
+ TimeStamp(token) + " encrypted events expected: " + test.sessionCount
+ + ", actual: " + gotEncrypted);
+
+ ok(Math.abs(test.duration - v.duration) < 0.1,
+ TimeStamp(token) + " Duration of video should be corrrect");
+ ok(Math.abs(test.duration - v.currentTime) < 0.1,
+ TimeStamp(token) + " Current time should be same as duration");
+
+ // Verify all sessions had all keys went sent to the CDM usable, and thus
+ // that we received keystatuseschange event(s).
+ is(sessions.length, test.sessionCount, TimeStamp(token) + " should have "
+ + test.sessionCount
+ + " session" + (test.sessionCount === 1 ? "" : "s"));
+ var keyIdsReceived = [];
+ for (var keyid in test.keys) { keyIdsReceived[keyid] = false; }
+ for (var i = 0; i < sessions.length; i++) {
+ var session = sessions[i];
+ ok(session.gotKeysChanged,
+ TimeStamp(token) + " session[" + session.sessionId
+ + "] should have received at least one keychange event");
+ for (let kid in session.keyIdsReceived) {
+ Log(token, "session[" + session.sessionId + "] key " + kid + " = " + (session.keyIdsReceived[kid] ? "true" : "false"));
+ if (session.keyIdsReceived[kid]) { keyIdsReceived[kid] = true; }
+ }
+ ok(session.numKeystatuseschangeEvents > 0, TimeStamp(token) + " should get key status changes");
+ is(session.numKeystatuseschangeEvents, session.numOnkeystatuseschangeEvents,
+ TimeStamp(token) + " should have as many keystatuseschange event listener calls as event handler calls.");
+
+ ok(session.numMessageEvents > 0, TimeStamp(token) + " should get message events");
+ is(session.numMessageEvents, session.numOnMessageEvents,
+ TimeStamp(token) + " should have as many message event listener calls as event handler calls.");
+ }
+ for (let kid in keyIdsReceived) {
+ ok(keyIdsReceived[kid], TimeStamp(token) + " key with id " + kid + " was usable as expected");
+ }
+
+ CloseSessions(v, sessions).then(finish.resolve, finish.reject);
+ });
+
+ Promise.all([
+ LoadInitData(v, test, token),
+ CreateAndSetMediaKeys(v, test, token),
+ LoadTest(test, v, token)])
+ .then(values => {
+ v.play();
+ let initData = values[0];
+ initData.map(ev => {
+ let session = v.mediaKeys.createSession();
+ onSessionCreated(session);
+ MakeRequest(test, token, ev, session);
+ });
+ return finish.promise;
+ })
+ .catch(reason => ok(false, reason))
+ .then(() => manager.finished(token));
+}
+
+SimpleTest.waitForExplicitFinish();
+manager.runTests(gEMETests, startTest);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_protection_query.html b/dom/media/test/test_eme_protection_query.html
new file mode 100644
index 0000000000..8bf97d8100
--- /dev/null
+++ b/dom/media/test/test_eme_protection_query.html
@@ -0,0 +1,250 @@
+<!DOCTYPE html>
+<html>
+<head>
+ <title>Test Encrypted Media Extensions - Protection Query</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="eme.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+// Tests in this file check that output protection queries are performed and
+// handled correctly. This is done by using a special clear key key system that
+// emits key status to track protection status.
+
+// Special key system used for these tests.
+const kClearKeyWithProtectionQuery =
+ "org.mozilla.clearkey_with_protection_query";
+const kTestFile = "bipbop-cenc-video-10s.mp4";
+const kTestMimeType = 'video/mp4; codecs="avc1.4d4015"';
+const kTestKeyId = "7e571d037e571d037e571d037e571d11"; // Hex representation
+const kTestKey = "7e5733337e5733337e5733337e573311";
+
+// This is the special key-id used by the mozilla clearkey CDM to signal
+// protection query status. As hex it is "6f75747075742d70726f74656374696f6e",
+// the hex translates to ascii "output-protection".
+const kProtectionQueryKeyIdString = "output-protection";
+
+// Options for requestMediaKeySystemAccess
+const kKeySystemOptions = [
+ {
+ initDataTypes: ["cenc"],
+ videoCapabilities: [{ contentType: kTestMimeType }],
+ },
+];
+
+// Helper to setup EME on `video`.
+// @param video the HTMLMediaElement to configure EME on.
+// @returns a media key session for the video. Callers can use this to
+// configure the `onkeystatuseschange` event handler for each test. Callers
+// *should not* configure other aspects of the session as this helper already
+// does so.
+async function setupEme(video) {
+ // Start setting up EME.
+ let access = await navigator.requestMediaKeySystemAccess(
+ kClearKeyWithProtectionQuery,
+ kKeySystemOptions
+ );
+ let mediaKeys = await access.createMediaKeys();
+ await video.setMediaKeys(mediaKeys);
+
+ let session = video.mediaKeys.createSession();
+
+ video.onencrypted = async encryptedEvent => {
+ session.onmessage = messageEvent => {
+ // Handle license messages. Hard code the license because we always test
+ // with the same file and we know what the license should be.
+ const license = {
+ keys: [
+ {
+ kty: "oct",
+ kid: HexToBase64(kTestKeyId),
+ k: HexToBase64(kTestKey),
+ },
+ ],
+ type: "temporary",
+ };
+
+ const encodedLicense = new TextEncoder().encode(JSON.stringify(license));
+
+ session.update(encodedLicense);
+ };
+
+ session.generateRequest(
+ encryptedEvent.initDataType,
+ encryptedEvent.initData
+ );
+ };
+
+ return session;
+}
+
+// Helper to setup MSE media on `video`.
+// @param video the HTMLMediaElement to configure MSE on.
+async function setupMse(video) {
+ const mediaSource = new MediaSource();
+ video.src = URL.createObjectURL(mediaSource);
+ await once(mediaSource, "sourceopen");
+ const sourceBuffer = mediaSource.addSourceBuffer("video/mp4");
+ let fetchResponse = await fetch(kTestFile);
+ sourceBuffer.appendBuffer(await fetchResponse.arrayBuffer());
+ await once(sourceBuffer, "updateend");
+ mediaSource.endOfStream();
+ await once(mediaSource, "sourceended");
+}
+
+// Helper to create a video element and append it to the page.
+function createAndAppendVideo() {
+ const video = document.createElement("video");
+ video.id = "video";
+ // Loop in case tests run slowly, we want video to keep playing until we
+ // get expected events.
+ video.loop = true;
+ document.body.appendChild(video);
+ return video;
+}
+
+// Helper to remove a video from the page.
+function removeVideo() {
+ let video = document.getElementById("video");
+ CleanUpMedia(video);
+}
+
+// Helper to get the status for the kProtectionQueryKeyIdString key id. A
+// session can (and will) have other keys with their own status, but we want
+// to check this special key to find the protection query status.
+function getKeyStatusForProtectionKeyId(session) {
+ for (let [keyId, status] of session.keyStatuses) {
+ if (ArrayBufferToString(keyId) == kProtectionQueryKeyIdString) {
+ return status;
+ }
+ }
+ return null;
+}
+
+async function getDisplayMedia() {
+ SpecialPowers.wrap(document).notifyUserGestureActivation();
+ return navigator.mediaDevices.getDisplayMedia();
+}
+
+// Tests playing encrypted media, starting a screen capture during playback,
+// then stopping the capture while playback continues.
+async function testProtectionQueryWithCaptureDuringVideo() {
+ let video = createAndAppendVideo();
+
+ // Configure the video and start it playing. KeyId should be usable (not restricted).
+ let session = await setupEme(video);
+ let keyStatusChangedPromise1 = new Promise(
+ resolve =>
+ (session.onkeystatuseschange = () => {
+ // We may get status changes prior to kProtectionQueryKeyIdString changing,
+ // ensure we wait for the first kProtectionQueryKeyIdString change.
+ if (getKeyStatusForProtectionKeyId(session)) {
+ resolve();
+ }
+ })
+ );
+ await setupMse(video);
+ await Promise.all([video.play(), keyStatusChangedPromise1]);
+ is(
+ getKeyStatusForProtectionKeyId(session),
+ "usable",
+ "Should be usable as capture hasn't started"
+ );
+
+ let keyStatusChangedPromise2 = new Promise(
+ resolve => (session.onkeystatuseschange = resolve)
+ );
+ let [displayMediaStream] = await Promise.all([
+ // Start a screen capture, this should restrict output.
+ getDisplayMedia(),
+ keyStatusChangedPromise2,
+ ]);
+ is(
+ getKeyStatusForProtectionKeyId(session),
+ "output-restricted",
+ "Should be output-restricted as capture is happening"
+ );
+
+ // Stop the screen capture, output should be usable again.
+ let keyStatusChangedPromise3 = new Promise(
+ resolve => (session.onkeystatuseschange = resolve)
+ );
+ displayMediaStream.getTracks().forEach(track => track.stop());
+ displayMediaStream = null;
+ await keyStatusChangedPromise3;
+ is(
+ getKeyStatusForProtectionKeyId(session),
+ "usable",
+ "Should be usable as capture has stopped"
+ );
+
+ removeVideo();
+}
+
+// Tests starting a screen capture, then starting encrypted playback, then
+// stopping the screen capture while encrypted playback continues.
+async function testProtectionQueryWithCaptureStartingBeforeVideo() {
+ // Start capture before setting up video.
+ let displayMediaStream = await getDisplayMedia();
+
+ let video = createAndAppendVideo();
+
+ // Configure the video and start it playing. KeyId should be restricted already.
+ let session = await setupEme(video);
+ let keyStatusChangedPromise1 = new Promise(
+ resolve =>
+ (session.onkeystatuseschange = () => {
+ // We may get status changes prior to kProtectionQueryKeyIdString changing,
+ // ensure we wait for the first kProtectionQueryKeyIdString change. In
+ // rare cases the first protection status can be "usable" due to racing
+ // between playback and the machinery that detects WebRTC capture. To
+ // avoid this, wait for the first 'output-restricted' notification,
+ // which will either be the first event, or will quickly follow 'usable'.
+ if (getKeyStatusForProtectionKeyId(session) == "output-restricted") {
+ resolve();
+ }
+ })
+ );
+ await setupMse(video);
+ await Promise.all([video.play(), keyStatusChangedPromise1]);
+ is(
+ getKeyStatusForProtectionKeyId(session),
+ "output-restricted",
+ "Should be restricted as capture is happening"
+ );
+
+ // Stop the screen capture, output should be usable again.
+ let keyStatusChangedPromise2 = new Promise(
+ resolve => (session.onkeystatuseschange = resolve)
+ );
+ displayMediaStream.getTracks().forEach(track => track.stop());
+ displayMediaStream = null;
+ await keyStatusChangedPromise2;
+ is(
+ getKeyStatusForProtectionKeyId(session),
+ "usable",
+ "Should be usable as capture has stopped"
+ );
+
+ removeVideo();
+}
+
+add_task(async function setupEnvironment() {
+ await SpecialPowers.pushPrefEnv({
+ set: [
+ // Need test key systems for test key system.
+ ["media.clearkey.test-key-systems.enabled", true],
+ // Need relaxed navigator permissions for getDisplayMedia.
+ ["media.navigator.permission.disabled", true],
+ ],
+ });
+});
+add_task(testProtectionQueryWithCaptureDuringVideo);
+add_task(testProtectionQueryWithCaptureStartingBeforeVideo);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_pssh_in_moof.html b/dom/media/test/test_eme_pssh_in_moof.html
new file mode 100644
index 0000000000..d1965be844
--- /dev/null
+++ b/dom/media/test/test_eme_pssh_in_moof.html
@@ -0,0 +1,141 @@
+<!DOCTYPE HTML>
+<html>
+
+ <head>
+ <title>Test Encrypted Media Extensions</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="eme.js"></script>
+ </head>
+
+ <body>
+ <script class="testbody" type="text/javascript">
+ let manager = new MediaTestManager;
+
+ let psshTests = [
+ { name: "bear-640x360-cenc-key-rotation",
+ tracks : [
+ {
+ name: "video",
+ type: "video/mp4; codecs=\"avc1.64000d\"",
+ fragments: ["bear-640x360-v_frag-cenc-key_rotation.mp4"],
+ initDatas: 6
+ },
+ {
+ name: "audio",
+ type: "audio/mp4; codecs=\"mp4a.40.2\"",
+ fragments: ["bear-640x360-a_frag-cenc-key_rotation.mp4"],
+ initDatas: 7
+ }
+ ],
+ keys : {
+ "30313233343536373839303132333435" :
+ "ebdd62f16814d27b68ef122afce4ae3c"
+ }
+ },
+ { name: "bipbop-clearkey-keyrotation-clear-lead",
+ tracks : [
+ {
+ name: "video",
+ type: "video/mp4; codecs=\"avc1.4d4015\"",
+ fragments: ["bipbop-clearkey-keyrotation-clear-lead-video.mp4"],
+ initDatas: 3
+ },
+ {
+ name: "audio",
+ type: "audio/mp4; codecs=\"mp4a.40.2\"",
+ fragments: ["bipbop-clearkey-keyrotation-clear-lead-audio.mp4"],
+ initDatas: 3
+ }
+ ],
+ keys : {
+ "00112233445566778899aabbccddeeff" :
+ "00112233445566778899aabbccddeeff",
+ "112233445566778899aabbccddeeff00" :
+ "112233445566778899aabbccddeeff00"
+ }
+ }
+ ];
+
+ // Specialized create media keys function, since the one in eme.js relies
+ // on listening for encrypted events, and we want to manage those
+ // ourselves within this test.
+ async function createAndSetMediaKeys(video, test, token) {
+ function streamType(type) {
+ var x = test.tracks.find(o => o.name == type);
+ return x ? x.type : undefined;
+ }
+
+ var configuration = { initDataTypes: ["cenc"] };
+ if (streamType("video")) {
+ configuration.videoCapabilities = [{contentType: streamType("video")}];
+ }
+ if (streamType("audio")) {
+ configuration.audioCapabilities = [{contentType: streamType("audio")}];
+ }
+ let keySystemAccess = await navigator.requestMediaKeySystemAccess(
+ "org.w3.clearkey", [configuration]);
+ let mediaKeys = await keySystemAccess.createMediaKeys();
+ return video.setMediaKeys(mediaKeys);
+ }
+
+ function startTest(test, token) {
+ manager.started(token);
+ let video = document.createElement("video");
+ document.body.appendChild(video);
+
+ let encryptedEventCount = 0;
+
+ let initDatas = new Map();
+
+ function handleEncrypted(event) {
+ // We get one 'encrypted' event for every run of contiguous PSSH boxes
+ // in each stream. Note that some of the PSSH boxes contained in the
+ // "bear" streams aren't in the Common Open PSSH box format, so our
+ // ClearKey CDM will reject those license requests. Some of the init
+ // data is also repeated.
+ encryptedEventCount++;
+
+ let hexStr = StringToHex(new TextDecoder().decode(event.initData));
+ if (initDatas.has(hexStr)) {
+ // Already have a session for this.
+ return;
+ }
+
+ let initData = new Uint8Array(event.initData);
+ is(event.initDataType, "cenc", "'encrypted' event should have 'cenc' " +
+ "initDataType");
+ Log(token, "encrypted event => len=" + initData.length + " " + hexStr);
+ let session = event.target.mediaKeys.createSession();
+ initDatas.set(hexStr, session);
+ session.addEventListener("message", e => {
+ e.target.update(
+ GenerateClearKeyLicense(e.message, test.keys));
+ });
+
+ session.generateRequest(event.initDataType, event.initData);
+ }
+
+ video.addEventListener("encrypted", handleEncrypted);
+
+ video.addEventListener("ended", () => {
+ let expectedEncryptedEvents =
+ test.tracks.reduce((sum, track) => sum += track.initDatas, 0);
+ is(encryptedEventCount, expectedEncryptedEvents,
+ "Should get one 'encrypted' event per run of contiguous PSSH " +
+ "boxes in media.");
+ manager.finished(token);
+ });
+
+ video.autoplay = true;
+
+ createAndSetMediaKeys(video, test, token)
+ .then(() => LoadTest(test, video, token))
+ }
+
+ manager.runTests(psshTests, startTest);
+
+ </script>
+ </body>
+</html>
diff --git a/dom/media/test/test_eme_requestKeySystemAccess.html b/dom/media/test/test_eme_requestKeySystemAccess.html
new file mode 100644
index 0000000000..b044fe8c84
--- /dev/null
+++ b/dom/media/test/test_eme_requestKeySystemAccess.html
@@ -0,0 +1,477 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test Encrypted Media Extensions</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="eme.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+const SUPPORTED_LABEL = "pass label";
+
+function ValidateConfig(name, expected, observed) {
+ info("ValidateConfig " + name);
+ info("expected cfg=" + JSON.stringify(expected));
+ info("observed cfg=" + JSON.stringify(observed));
+
+ is(observed.label, expected.label, name + " label should match");
+ if (expected.initDataTypes) {
+ ok(expected.initDataTypes.every((element, index, array) => observed.initDataTypes.includes(element)), name + " initDataTypes should match.");
+ }
+
+ if (expected.audioCapabilities) {
+ ok(expected.audioCapabilities.length == 1, "Test function can only handle one capability.");
+ ok(observed.audioCapabilities.length == 1, "Test function can only handle one capability.");
+ is(observed.audioCapabilities[0].contentType, expected.audioCapabilities[0].contentType, name + " audioCapabilities should match.");
+ }
+ if (typeof expected.videoCapabilities !== 'undefined') {
+ info("expected.videoCapabilities=" + expected.videoCapabilities);
+ dump("expected.videoCapabilities=" + expected.videoCapabilities + "\n");
+ ok(expected.videoCapabilities.length == 1, "Test function can only handle one capability.");
+ ok(observed.videoCapabilities.length == 1, "Test function can only handle one capability.");
+ is(observed.videoCapabilities[0].contentType, expected.videoCapabilities[0].contentType, name + " videoCapabilities should match.");
+ }
+ if (expected.sessionTypes) {
+ is(expected.sessionTypes.length, observed.sessionTypes.length, "Should have expected number of sessionTypes");
+ for (var i = 0; i < expected.sessionTypes.length; i++) {
+ is(expected[i], observed[i], "Session type " + i + " should match");
+ }
+ }
+}
+
+function Test(test) {
+ var name = "'" + test.name + "'";
+ return new Promise(function(resolve, reject) {
+ var p;
+ if (test.options) {
+ var keySystem = (test.keySystem !== undefined) ? test.keySystem : CLEARKEY_KEYSYSTEM;
+ p = navigator.requestMediaKeySystemAccess(keySystem, test.options);
+ } else {
+ p = navigator.requestMediaKeySystemAccess(keySystem);
+ }
+ p.then(
+ function(keySystemAccess) {
+ ok(test.shouldPass, name + " passed and was expected to " + (test.shouldPass ? "pass" : "fail"));
+ is(keySystemAccess.keySystem, CLEARKEY_KEYSYSTEM, "CDM keySystem should be in MediaKeySystemAccess.keySystem");
+ ValidateConfig(name, test.expectedConfig, keySystemAccess.getConfiguration());
+ resolve();
+ },
+ function(ex) {
+ if (test.shouldPass) {
+ info(name + " failed: " + ex);
+ }
+ ok(!test.shouldPass, name + " failed and was expected to " + (test.shouldPass ? "pass" : "fail"));
+ resolve();
+ });
+ });
+}
+
+var tests = [
+ {
+ name: 'Empty keySystem string',
+ keySystem: '',
+ options: [
+ {
+ initDataTypes: ['cenc'],
+ videoCapabilities: [{contentType: 'video/mp4'}],
+ }
+ ],
+ shouldPass: false,
+ },
+ {
+ name: 'Empty options specified',
+ options: [ ],
+ shouldPass: false,
+ },
+ {
+ name: 'Undefined options',
+ shouldPass: false,
+ },
+ {
+ name: 'Basic MP4 cenc',
+ options: [
+ {
+ label: SUPPORTED_LABEL,
+ initDataTypes: ['cenc'],
+ audioCapabilities: [{contentType: 'audio/mp4'}],
+ videoCapabilities: [{contentType: 'video/mp4'}],
+ }
+ ],
+ expectedConfig: {
+ label: SUPPORTED_LABEL,
+ initDataTypes: ['cenc'],
+ audioCapabilities: [{contentType: 'audio/mp4'}],
+ videoCapabilities: [{contentType: 'video/mp4'}],
+ },
+ shouldPass: true,
+ },
+ {
+ name: 'Invalid keysystem failure',
+ keySystem: 'bogusKeySystem',
+ options: [
+ {
+ initDataTypes: ['cenc'],
+ videoCapabilities: [{contentType: 'video/mp4'}],
+ }
+ ],
+ shouldPass: false,
+ },
+ {
+ name: 'Invalid initDataType',
+ options: [
+ {
+ initDataTypes: ['bogus'],
+ audioCapabilities: [{contentType: 'audio/mp4'}],
+ }
+ ],
+ shouldPass: false,
+ },
+ {
+ name: 'Valid initDataType after invalid',
+ options: [
+ {
+ label: SUPPORTED_LABEL,
+ initDataTypes: ['bogus', 'invalid', 'cenc'],
+ audioCapabilities: [{contentType: 'audio/mp4'}],
+ }
+ ],
+ expectedConfig: {
+ label: SUPPORTED_LABEL,
+ initDataTypes: ['cenc'],
+ audioCapabilities: [{contentType: 'audio/mp4'}],
+ },
+ shouldPass: true,
+ },
+ {
+ name: 'Invalid videoType',
+ options: [
+ {
+ initDataTypes: ['cenc'],
+ videoCapabilities: [{contentType: 'video/bogus'}],
+ }
+ ],
+ shouldPass: false,
+ },
+ {
+ name: 'Invalid distinctiveIdentifier fails',
+ options: [
+ {
+ initDataTypes: ['cenc'],
+ videoCapabilities: [{contentType: 'video/mp4'}],
+ distinctiveIdentifier: 'bogus',
+ persistentState: 'bogus',
+ }
+ ],
+ shouldPass: false,
+ },
+ {
+ name: 'distinctiveIdentifier is prohibited for ClearKey',
+ options: [
+ {
+ initDataTypes: ['cenc'],
+ videoCapabilities: [{contentType: 'video/mp4'}],
+ distinctiveIdentifier: 'required',
+ }
+ ],
+ shouldPass: false,
+ },
+ {
+ name: 'Invalid persistentState fails',
+ options: [
+ {
+ initDataTypes: ['cenc'],
+ videoCapabilities: [{contentType: 'video/mp4'}],
+ persistentState: 'bogus',
+ }
+ ],
+ shouldPass: false,
+ },
+ {
+ name: 'Invalid robustness unsupported',
+ options: [
+ {
+ initDataTypes: ['cenc'],
+ videoCapabilities: [{contentType: 'video/mp4', robustness: 'very much so'}],
+ }
+ ],
+ shouldPass: false,
+ },
+ {
+ name: 'Unexpected config entry should be ignored',
+ options: [
+ {
+ label: SUPPORTED_LABEL,
+ initDataTypes: ['cenc'],
+ videoCapabilities: [{contentType: 'video/mp4'}],
+ unexpectedEntry: 'this should be ignored',
+ }
+ ],
+ expectedConfig: {
+ label: SUPPORTED_LABEL,
+ initDataTypes: ['cenc'],
+ videoCapabilities: [{contentType: 'video/mp4'}],
+ },
+ shouldPass: true,
+ },
+ {
+ name: 'Invalid option followed by valid',
+ options: [
+ {
+ label: "this config should not be supported",
+ initDataTypes: ['bogus'],
+ },
+ {
+ label: SUPPORTED_LABEL,
+ initDataTypes: ['cenc'],
+ videoCapabilities: [{contentType: 'video/mp4'}],
+ }
+ ],
+ expectedConfig: {
+ label: SUPPORTED_LABEL,
+ initDataTypes: ['cenc'],
+ videoCapabilities: [{contentType: 'video/mp4'}],
+ },
+ shouldPass: true,
+ },
+ {
+ name: 'Persistent-license should not be supported by ClearKey',
+ options: [
+ {
+ initDataTypes: ['cenc'],
+ videoCapabilities: [{contentType: 'video/mp4'}],
+ sessionTypes: ['persistent-license'],
+ persistentState: 'optional',
+ }
+ ],
+ shouldPass: false,
+ },
+ {
+ name: 'Persistent-usage-record should not be supported by ClearKey',
+ options: [
+ {
+ initDataTypes: ['cenc'],
+ videoCapabilities: [{contentType: 'video/mp4'}],
+ sessionTypes: ['persistent-usage-record'],
+ persistentState: 'optional',
+ }
+ ],
+ shouldPass: false,
+ },
+ {
+ name: 'MP4 audio container',
+ options: [
+ {
+ label: SUPPORTED_LABEL,
+ initDataTypes: ['cenc'],
+ audioCapabilities: [{contentType: 'audio/mp4'}],
+ }
+ ],
+ expectedConfig: {
+ label: SUPPORTED_LABEL,
+ initDataTypes: ['cenc'],
+ audioCapabilities: [{contentType: 'audio/mp4'}],
+ },
+ shouldPass: true,
+ },
+ {
+ name: 'MP4 audio container with AAC-LC',
+ options: [
+ {
+ label: SUPPORTED_LABEL,
+ initDataTypes: ['cenc'],
+ audioCapabilities: [{contentType: 'audio/mp4; codecs="mp4a.40.2"'}],
+ }
+ ],
+ expectedConfig: {
+ label: SUPPORTED_LABEL,
+ initDataTypes: ['cenc'],
+ audioCapabilities: [{contentType: 'audio/mp4; codecs="mp4a.40.2"'}],
+ },
+ shouldPass: true,
+ },
+ {
+ name: 'MP4 audio container with invalid codecs',
+ options: [
+ {
+ initDataTypes: ['cenc'],
+ audioCapabilities: [{contentType: 'audio/mp4; codecs="bogus"'}],
+ }
+ ],
+ shouldPass: false,
+ },
+ {
+ name: 'MP4 audio container with mp3 is unsupported',
+ options: [
+ {
+ initDataTypes: ['cenc'],
+ audioCapabilities: [{contentType: 'audio/mp4; codecs="mp3"'}],
+ }
+ ],
+ shouldPass: false,
+ },
+ {
+ name: 'MP4 video container type with an mp3 codec is unsupported',
+ options: [
+ {
+ initDataTypes: ['cenc'],
+ videoCapabilities: [{contentType: 'video/mp4; codecs="mp3"'}],
+ }
+ ],
+ shouldPass: false,
+ },
+ {
+ name: 'MP4 audio container type with a video codec is unsupported',
+ options: [
+ {
+ initDataTypes: ['cenc'],
+ audioCapabilities: [{contentType: 'audio/mp4; codecs="avc1.42E01E"'}],
+ }
+ ],
+ shouldPass: false,
+ },
+ {
+ name: 'MP4 video container with constrained baseline h.264',
+ options: [
+ {
+ label: SUPPORTED_LABEL,
+ initDataTypes: ['cenc'],
+ videoCapabilities: [{contentType: 'video/mp4; codecs="avc1.42E01E"'}],
+ }
+ ],
+ expectedConfig: {
+ label: SUPPORTED_LABEL,
+ initDataTypes: ['cenc'],
+ videoCapabilities: [{contentType: 'video/mp4; codecs="avc1.42E01E"'}],
+ },
+ shouldPass: true,
+ },
+ {
+ name: 'MP4 video container with invalid codecs',
+ options: [
+ {
+ initDataTypes: ['cenc'],
+ videoCapabilities: [{contentType: 'video/mp4; codecs="bogus"'}],
+ }
+ ],
+ shouldPass: false,
+ },
+ {
+ name: 'MP4 video container with both audio and video codec type in videoType',
+ options: [
+ {
+ initDataTypes: ['cenc'],
+ videoCapabilities: [{contentType: 'video/mp4; codecs="avc1.42E01E,mp4a.40.2"'}],
+ }
+ ],
+ shouldPass: false,
+ },
+ {
+ name: 'MP4 audio and video type both specified',
+ options: [
+ {
+ label: SUPPORTED_LABEL,
+ initDataTypes: ['cenc'],
+ videoCapabilities: [{contentType: 'video/mp4; codecs="avc1.42E01E"'}],
+ audioCapabilities: [{contentType: 'audio/mp4; codecs="mp4a.40.2"'}],
+ }
+ ],
+ expectedConfig: {
+ label: SUPPORTED_LABEL,
+ initDataTypes: ['cenc'],
+ videoCapabilities: [{contentType: 'video/mp4; codecs="avc1.42E01E"'}],
+ audioCapabilities: [{contentType: 'audio/mp4; codecs="mp4a.40.2"'}],
+ },
+ shouldPass: true,
+ },
+ {
+ name: 'Basic WebM video',
+ options: [
+ {
+ label: SUPPORTED_LABEL,
+ initDataTypes: ['webm'],
+ videoCapabilities: [{contentType: 'video/webm'}],
+ }
+ ],
+ expectedConfig: {
+ label: SUPPORTED_LABEL,
+ initDataTypes: ['webm'],
+ videoCapabilities: [{contentType: 'video/webm'}],
+ },
+ shouldPass: true,
+ },
+ {
+ name: 'Basic WebM audio',
+ options: [
+ {
+ label: SUPPORTED_LABEL,
+ initDataTypes: ['webm'],
+ audioCapabilities: [{contentType: 'audio/webm'}],
+ }
+ ],
+ expectedConfig: {
+ label: SUPPORTED_LABEL,
+ initDataTypes: ['webm'],
+ audioCapabilities: [{contentType: 'audio/webm'}],
+ },
+ shouldPass: true,
+ },
+ {
+ name: 'Webm with Vorbis audio and VP8 video.',
+ options: [
+ {
+ label: SUPPORTED_LABEL,
+ initDataTypes: ['webm'],
+ videoCapabilities: [{contentType: 'video/webm;codecs="vp8"'}],
+ audioCapabilities: [{contentType: 'audio/webm;codecs="vorbis"'}],
+ }
+ ],
+ expectedConfig: {
+ label: SUPPORTED_LABEL,
+ initDataTypes: ['webm'],
+ videoCapabilities: [{contentType: 'video/webm;codecs="vp8"'}],
+ audioCapabilities: [{contentType: 'audio/webm;codecs="vorbis"'}],
+ },
+ shouldPass: true,
+ },
+ {
+ name: 'Webm with Vorbis audio and VP9 video.',
+ options: [
+ {
+ label: SUPPORTED_LABEL,
+ initDataTypes: ['webm'],
+ videoCapabilities: [{contentType: 'video/webm;codecs="vp9"'}],
+ audioCapabilities: [{contentType: 'audio/webm;codecs="vorbis"'}],
+ }
+ ],
+ expectedConfig: {
+ label: SUPPORTED_LABEL,
+ initDataTypes: ['webm'],
+ videoCapabilities: [{contentType: 'video/webm;codecs="vp9"'}],
+ audioCapabilities: [{contentType: 'audio/webm;codecs="vorbis"'}],
+ },
+ shouldPass: true,
+ },
+ {
+ name: 'Webm with bogus video.',
+ options: [
+ {
+ initDataTypes: ['webm'],
+ videoCapabilities: [{contentType: 'video/webm;codecs="bogus"'}],
+ }
+ ],
+ shouldPass: false,
+ },
+];
+
+SimpleTest.waitForExplicitFinish();
+Promise.all(tests.map(Test)).then(function() {
+ SimpleTest.finish();
+});
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_requestMediaKeySystemAccess_with_app_approval.html b/dom/media/test/test_eme_requestMediaKeySystemAccess_with_app_approval.html
new file mode 100644
index 0000000000..ceba459dae
--- /dev/null
+++ b/dom/media/test/test_eme_requestMediaKeySystemAccess_with_app_approval.html
@@ -0,0 +1,202 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test Encrypted Media Extensions access can be gated by application</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="eme.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+// These test cases should be used to make a request to
+// requestMediaKeySystemAccess and have the following members:
+// name: a name describing the test.
+// askAppApproval: used to set prefs such so that Gecko will ask for app
+// approval for EME if true, or not if false.
+// appApproves: used to set prefs to simulate app approval of permission
+// request, true if the app approves the request, false if not.
+// expectedKeySystemAccess: true if we expect to be granted key system access,
+// false if not.
+const testCases = [
+ {
+ name: "Don't check for app approval",
+ askAppApproval: false,
+ expectedKeySystemAccess: true,
+ },
+ {
+ name: "Check for app approval and app denies request",
+ askAppApproval: true,
+ appApproves: false,
+ expectedKeySystemAccess: false,
+ },
+ {
+ name: "Check for app approval and app allows request",
+ askAppApproval: true,
+ appApproves: true,
+ expectedKeySystemAccess: true,
+ },
+];
+
+// Options for requestMediaKeySystemAccess that are expected to work.
+const options = [{
+ initDataTypes: ['webm'],
+ audioCapabilities: [
+ { contentType: 'audio/webm; codecs="opus"' },
+ ],
+ videoCapabilities: [
+ { contentType: 'video/webm; codecs="vp8"' }
+ ]
+}];
+
+async function setTestPrefs({askAppApproval, appApproves}) {
+ if (!askAppApproval) {
+ // Test doesn't want app approval, set pref so we don't ask and unset prefs
+ // used to determine permission response as we don't need them.
+ await SpecialPowers.pushPrefEnv({
+ set: [["media.eme.require-app-approval", false]],
+ clear: [
+ ["media.eme.require-app-approval.prompt.testing"],
+ ["media.eme.require-app-approval.prompt.testing.allow"],
+ ]
+ });
+ return;
+ }
+
+ // Test wants app approval, and will approve deny requests per appApproces
+ // value, set prefs accordingly.
+ await SpecialPowers.pushPrefEnv({
+ set: [
+ ["media.eme.require-app-approval", true],
+ ["media.eme.require-app-approval.prompt.testing", true],
+ ["media.eme.require-app-approval.prompt.testing.allow", appApproves],
+ ],
+ });
+}
+
+// Run a test case that makes a single requestMediaKeySystemAccess call. The
+// outcome of such a test run should depend on the test case's setting of
+// preferences controlling the eme app approval.
+async function testSingleRequest(testCase) {
+ await setTestPrefs(testCase);
+
+ try {
+ await navigator.requestMediaKeySystemAccess(CLEARKEY_KEYSYSTEM, options);
+ ok(testCase.expectedKeySystemAccess,
+ `testSingleRequest ${testCase.name}: allowed media key system access.`);
+ } catch(e) {
+ is(e.name,
+ "NotSupportedError",
+ "Should get NotSupportedError when request is blocked.");
+ is(e.message,
+ "The application embedding this user agent has blocked MediaKeySystemAccess",
+ "Should get blocked error message.");
+ ok(!testCase.expectedKeySystemAccess,
+ `testSingleRequest ${testCase.name}: denied media key system access.`);
+ }
+}
+
+// Run a test case that, but using invalid arguments for
+// requestMediaKeySystemAccess. Because we expect the args to be checked
+// before requesting app approval, this test ensures that we always fail when
+// using bad args, regardless of the app approval prefs set.
+async function testRequestWithInvalidArgs(testCase) {
+ const badOptions = [{
+ initDataTypes: ['badType'],
+ audioCapabilities: [
+ { contentType: 'audio/webm; codecs="notACodec"' },
+ ],
+ videoCapabilities: [
+ { contentType: 'video/webm; codecs="notACodec"' }
+ ]
+ }];
+
+ await setTestPrefs(testCase);
+
+ // Check that calls with a bad key system fail.
+ try {
+ await navigator.requestMediaKeySystemAccess("BadKeySystem", options);
+ ok(false,
+ `testRequestWithInvalidArgs ${testCase.name}: should not get access when using bad key system.`);
+ } catch(e) {
+ is(e.name,
+ "NotSupportedError",
+ "Should get NotSupportedError using bad key system.");
+ is(e.message,
+ "Key system is unsupported",
+ "Should get not supported key system error message.");
+ }
+
+ // Check that calls with the bad options fail.
+ try {
+ await navigator.requestMediaKeySystemAccess(CLEARKEY_KEYSYSTEM, badOptions);
+ ok(false,
+ `testRequestWithInvalidArgs ${testCase.name}: should not get access when using bad options.`);
+ } catch(e) {
+ is(e.name,
+ "NotSupportedError",
+ "Should get NotSupportedError using bad options.");
+ is(e.message,
+ "Key system configuration is not supported",
+ "Should get not supported config error message.");
+ }
+}
+
+// Run a test case and make multiple requests with the same case. Check that
+// all requests are resolved with the expected outcome.
+async function testMultipleRequests(testCase) {
+ // Number of requests to concurrently make.
+ const NUM_REQUESTS = 5;
+
+ await setTestPrefs(testCase);
+
+ let resolveHandler = () => {
+ ok(testCase.expectedKeySystemAccess,
+ `testMultipleRequests ${testCase.name}: allowed media key system access.`);
+ }
+
+ let rejectHandler = e => {
+ is(e.name,
+ "NotSupportedError",
+ "Should get NotSupportedError when request is blocked.");
+ is(e.message,
+ "The application embedding this user agent has blocked MediaKeySystemAccess",
+ "Should get blocked error message.");
+ ok(!testCase.expectedKeySystemAccess,
+ `testMultipleRequests ${testCase.name}: denied media key system access.`);
+ }
+
+ let accessPromises = [];
+ for(let i = 0; i < NUM_REQUESTS; i++) {
+ // Request access then chain to our resolve and reject handlers. The
+ // handlers assert test state then resolve the promise chain. Ensuring the
+ // chain is always resolved allows us to correctly await all outstanding
+ // requests -- otherwise rejects short circuit the Promise.all call below.
+ let accessPromise = navigator.requestMediaKeySystemAccess(CLEARKEY_KEYSYSTEM, options)
+ .then(resolveHandler, rejectHandler);
+ accessPromises.push(accessPromise);
+ }
+ // Wait for all promises to be resolved. If not, we'll time out. Because
+ // our reject handler chains back into a resolved promise, this should wait
+ // for all requests to be serviced, even when requestMediaKeySystemAccess's
+ // promise is rejected.
+ await Promise.all(accessPromises);
+}
+
+// The tests rely on prefs being set, so run them in sequence. If we run in
+// parallel the tests break each other by overriding prefs.
+for (const testCase of testCases) {
+ add_task(() => testSingleRequest(testCase));
+}
+for (const testCase of testCases) {
+ add_task(() => testRequestWithInvalidArgs(testCase));
+}
+for (const testCase of testCases) {
+ add_task(() => testMultipleRequests(testCase));
+}
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_request_notifications.html b/dom/media/test/test_eme_request_notifications.html
new file mode 100644
index 0000000000..6c44f892a6
--- /dev/null
+++ b/dom/media/test/test_eme_request_notifications.html
@@ -0,0 +1,82 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test Encrypted Media Extensions</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="eme.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var manager = new MediaTestManager;
+
+function SetPrefs(prefs) {
+ return SpecialPowers.pushPrefEnv({"set": prefs});
+}
+
+function observe() {
+ return new Promise(function(resolve, reject) {
+ var observer = function(subject, topic, data) {
+ SpecialPowers.Services.obs.removeObserver(observer, "mediakeys-request");
+ resolve(JSON.parse(data).status);
+ };
+ SpecialPowers.Services.obs.addObserver(observer, "mediakeys-request");
+ });
+}
+
+function Test(test) {
+ var p = test.prefs ? SetPrefs(test.prefs) : Promise.resolve();
+ var name = "'" + test.keySystem + "'";
+
+ var res = observe().then((status) => {
+ is(status, test.expectedStatus, name + " expected status");
+ });
+
+ p.then(() => navigator.requestMediaKeySystemAccess(test.keySystem, gCencMediaKeySystemConfig))
+ .then((keySystemAccess) => keySystemAccess.createMediaKeys());
+
+ return res;
+}
+
+var tests = [
+ {
+ keySystem: CLEARKEY_KEYSYSTEM,
+ expectedStatus: 'cdm-created',
+ prefs: [["media.eme.enabled", false]]
+ },
+ {
+ keySystem: "com.widevine.alpha",
+ expectedStatus: 'api-disabled',
+ prefs: [["media.eme.enabled", false]]
+ },
+ {
+ keySystem: "com.widevine.alpha",
+ expectedStatus: 'cdm-disabled',
+ prefs: [["media.eme.enabled", true], ["media.gmp-widevinecdm.enabled", false]]
+ },
+ {
+ keySystem: "com.widevine.alpha",
+ expectedStatus: 'cdm-not-installed',
+ prefs: [["media.eme.enabled", true], ["media.gmp-widevinecdm.enabled", true]]
+ },
+ {
+ keySystem: CLEARKEY_KEYSYSTEM,
+ expectedStatus: 'cdm-created',
+ prefs: [["media.eme.enabled", true]]
+ }
+];
+
+SimpleTest.waitForExplicitFinish();
+tests
+ .reduce(function(p, c, i, array) {
+ return p.then(function() {
+ return Test(c);
+ });
+ }, Promise.resolve())
+ .then(SimpleTest.finish);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_sample_groups_playback.html b/dom/media/test/test_eme_sample_groups_playback.html
new file mode 100644
index 0000000000..cef1e26b33
--- /dev/null
+++ b/dom/media/test/test_eme_sample_groups_playback.html
@@ -0,0 +1,130 @@
+<!DOCTYPE HTML>
+<html>
+
+<head>
+ <title>Test Encrypted Media Extensions</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="eme.js"></script>
+</head>
+
+<body>
+ <video controls id="video"></video>
+ <pre id="test">
+ <script class="testbody" type="text/javascript">
+
+ // Tests that files with a default key and a seperate sample keyids in the
+ // sgpd box play correctly (if the keyid from the sgpd box is not parsed
+ // or assigned to the sample we will wait indefinitely for the default
+ // key).
+
+ SimpleTest.waitForExplicitFinish();
+
+ // Test files for samples encrypted with different media keys.
+ var gEMESampleGoupTests = [
+ {
+ name:"video with 4 keys in sgpd (sbgp in traf sgpd in stbl)",
+ track: {
+ name:"video",
+ type:"video/mp4; codecs=\"avc1.64000d\"",
+ fragments:[ "sample-encrypted-sgpdstbl-sbgptraf.mp4"
+ ]
+ },
+ keys: {
+ // "keyid" : "key"
+ "279926496a7f5d25da69f2b3b2799a7f": "5544694d47473326622665665a396b36",
+ "597669572e55547e656b56586e2f6f68": "7959493a764556786527517849756635",
+ "205b2b293a342f3d3268293e6f6f4e29": "3a4f3674376d6c48675a273464447b40",
+ "32783e367c2e4d4d6b46467b3e6b5478": "3e213f6d45584f51713d534f4b417855",
+ },
+ sessionType:"temporary",
+ sessionCount:1,
+ duration:2,
+ },
+ ],
+ test = gEMESampleGoupTests[0];
+
+ var video = document.getElementById("video");
+ video.addEventListener("encrypted", () => {
+ Log(test.name, "Recieved encrypted event");
+ });
+
+ video.addEventListener("waitingforkey", () => {
+ Log(test.name, "waitingforkey");
+ ok(false, test.name + " Video is waitingforkey, indicating that the samples are not being assigned the correct id from the sgpd box!");
+ SimpleTest.finish();
+ });
+
+ function LoadEME() {
+ var options = [{
+ initDataType: "cenc",
+ videoType: test.track.type,
+ }];
+
+ return navigator.requestMediaKeySystemAccess("org.w3.clearkey", options)
+ .then((keySystemAccess) => {
+ return keySystemAccess.createMediaKeys();
+ }, bail("Failed to request key system access."))
+
+ .then((mediaKeys) => {
+ video.setMediaKeys(mediaKeys);
+
+ var session = mediaKeys.createSession();
+ once(session, "message", (ev) => {
+ is(ev.messageType, "license-request", "Expected a license-request");
+ session.update(GenerateClearKeyLicense(ev.message, test.keys));
+ });
+
+ var json = JSON.stringify({
+ "kids":Object.keys(test.keys).map(HexToBase64)
+ });
+ var request = new TextEncoder().encode(json);
+ session.generateRequest("keyids", request)
+ .then(e => {
+ Log(test.name, "Request license success");
+ }, reason => {
+ Log("Request license failed! " + reason);
+ });
+ });
+ }
+
+ function DownloadMedia(url, type, mediaSource) {
+ return new Promise((resolve, reject) => {
+ var sourceBuffer = mediaSource.addSourceBuffer(type);
+ fetchWithXHR(url, (response) => {
+ once(sourceBuffer, "updateend", resolve);
+ sourceBuffer.appendBuffer(new Uint8Array(response));
+ });
+ });
+ }
+
+ function LoadMSE() {
+ // Only set the source of the video and download the tracks after we
+ // have set the license keys, so we don't hit the waitingforkey event
+ // unless samples are being incorrectly assigned the default key
+ // (and we can safely fail).
+ LoadEME()
+ .then(() => {
+ var ms = new MediaSource();
+ video.src = URL.createObjectURL(ms);
+
+ once(ms, "sourceopen", () => {
+ Promise.all(test.track.fragments.map(fragment => DownloadMedia(fragment, test.track.type, ms)))
+ .then(() => {
+ ms.endOfStream();
+ video.play();
+ });
+ });
+
+ once(video, "ended", SimpleTest.finish);
+ });
+ }
+
+ LoadMSE();
+
+ </script>
+ </pre>
+</body>
+
+</html> \ No newline at end of file
diff --git a/dom/media/test/test_eme_session_callable_value.html b/dom/media/test/test_eme_session_callable_value.html
new file mode 100644
index 0000000000..3139c36157
--- /dev/null
+++ b/dom/media/test/test_eme_session_callable_value.html
@@ -0,0 +1,34 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test Encrypted Media Extensions</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="eme.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+function Test() {
+ navigator.requestMediaKeySystemAccess(CLEARKEY_KEYSYSTEM, gCencMediaKeySystemConfig)
+ .then(access => access.createMediaKeys())
+ .then(mediaKeys => {
+ var initData = (new TextEncoder()).encode( 'this is an invalid license, and that is ok');
+ var s = mediaKeys.createSession("temporary");
+ s.generateRequest("cenc", initData); // ignore result.
+ // "update()" call should fail, because MediaKeySession is "not callable"
+ // yet, since CDM won't have had a chance to set the sessionId on MediaKeySession.
+ return s.update(initData);
+ })
+ .then(()=>{ok(false, "An exception should be thrown; MediaKeySession should be not callable."); SimpleTest.finish();},
+ ()=>{ok(true, "We expect this to fail; MediaKeySession should be not callable."); SimpleTest.finish();});
+}
+
+SimpleTest.waitForExplicitFinish();
+Test();
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_setMediaKeys_before_attach_MediaSource.html b/dom/media/test/test_eme_setMediaKeys_before_attach_MediaSource.html
new file mode 100644
index 0000000000..1493cfdcb8
--- /dev/null
+++ b/dom/media/test/test_eme_setMediaKeys_before_attach_MediaSource.html
@@ -0,0 +1,37 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test Encrypted Media Extensions</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="eme.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+function beginTest() {
+ var video = document.createElement("video");
+
+ navigator.requestMediaKeySystemAccess(CLEARKEY_KEYSYSTEM, gCencMediaKeySystemConfig)
+ .then(function(keySystemAccess) {
+ return keySystemAccess.createMediaKeys();
+ })
+ .then(mediaKeys => {
+ return video.setMediaKeys(mediaKeys);
+ })
+ .then(() => {
+ var ms = new MediaSource();
+ ms.addEventListener("sourceopen", ()=>{ok(true, "MediaSource should open"); SimpleTest.finish();});
+ video.addEventListener("error", ()=>{ok(false, "Shouldn't error."); SimpleTest.finish();});
+ video.src = URL.createObjectURL(ms);
+ });
+}
+
+SimpleTest.waitForExplicitFinish();
+beginTest();
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_special_key_system.html b/dom/media/test/test_eme_special_key_system.html
new file mode 100644
index 0000000000..0addc06ebc
--- /dev/null
+++ b/dom/media/test/test_eme_special_key_system.html
@@ -0,0 +1,63 @@
+<!DOCTYPE html>
+<html>
+<head>
+ <title>Test Encrypted Media Extensions - Special key system</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+function requestProtectionQueryKeySystemAccess() {
+ const kKeySystemOptions = [
+ {
+ initDataTypes: ["cenc"],
+ videoCapabilities: [{ contentType: 'video/mp4; codecs="avc1.4d4015"' }],
+ },
+ ];
+ const kClearKeyWithProtectionQuery =
+ "org.mozilla.clearkey_with_protection_query";
+ return navigator.requestMediaKeySystemAccess(
+ kClearKeyWithProtectionQuery,
+ kKeySystemOptions
+ );
+}
+// Tests that org.mozilla.clearkey_with_protection_query cannot be accessed from JS if not preffed on.
+add_task(async function protectionQueryKeySystemShouldBeHidden() {
+ try {
+ // Should throw since special key systems are not enabled.
+ await requestProtectionQueryKeySystemAccess();
+ ok(
+ false,
+ "Test should have thrown by default because media.clearkey.test-key-systems.enabled should not be set"
+ );
+ } catch (e) {
+ is(e.name, "NotSupportedError", "Should get not supported error");
+ is(
+ e.message,
+ "Key system is unsupported",
+ "Should get message that key system is unsupported"
+ );
+ }
+});
+
+// Tests that org.mozilla.clearkey_with_protection_query can be accessed from JS if preffed on.
+add_task(async function protectionQueryKeySystemShouldBeExposed() {
+ await SpecialPowers.pushPrefEnv({
+ set: [["media.clearkey.test-key-systems.enabled", true]],
+ });
+ try {
+ // Should not throw since special key systems are enabled.
+ let access = await requestProtectionQueryKeySystemAccess();
+ ok(
+ access,
+ "Should get access to protection query key system when preffed on"
+ );
+ } catch (e) {
+ ok(false, "Test should not have thrown and key system should be available");
+ }
+});
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_stream_capture_blocked_case1.html b/dom/media/test/test_eme_stream_capture_blocked_case1.html
new file mode 100644
index 0000000000..0fc8d28364
--- /dev/null
+++ b/dom/media/test/test_eme_stream_capture_blocked_case1.html
@@ -0,0 +1,59 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test Encrypted Media Extensions</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="eme.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var manager = new MediaTestManager;
+
+function startTest(test, token)
+{
+ // Case 1. setting MediaKeys on an element captured by MediaElementSource should fail.
+ var case1token = token + "_case1";
+ let v1 = document.createElement("video");
+
+ function setMediaKeys() {
+ let p = new EMEPromise;
+ CreateMediaKeys(v1, test, case1token)
+ .then(mediaKeys => {
+ v1.setMediaKeys(mediaKeys)
+ .then(() => {
+ p.reject(`${case1token} setMediaKeys shouldn't succeed.`);
+ }, () => {
+ ok(true, TimeStamp(case1token) + " setMediaKeys failed as expected.");
+ p.resolve();
+ })
+ }, p.reject);
+ return p.promise;
+ }
+
+ var context = new AudioContext();
+ context.createMediaElementSource(v1);
+ v1.addEventListener("loadeddata", function(ev) {
+ ok(false, TimeStamp(case1token) + " should never reach loadeddata, as setMediaKeys should fail");
+ });
+
+ manager.started(case1token);
+
+ Promise.all([
+ LoadTest(test, v1, case1token),
+ setMediaKeys()])
+ .catch(reason => ok(false, reason))
+ .then(() => {
+ CleanUpMedia(v1);
+ manager.finished(case1token);
+ });
+}
+
+SimpleTest.waitForExplicitFinish();
+manager.runTests(gEMETests, startTest);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_stream_capture_blocked_case2.html b/dom/media/test/test_eme_stream_capture_blocked_case2.html
new file mode 100644
index 0000000000..b60538caf0
--- /dev/null
+++ b/dom/media/test/test_eme_stream_capture_blocked_case2.html
@@ -0,0 +1,52 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test Encrypted Media Extensions</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="eme.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var manager = new MediaTestManager;
+
+function startTest(test, token)
+{
+ // Case 2. creating a MediaElementSource on a media element should always succeed
+ // (no matter whether it's restricted content or not), and
+ var p1 = new EMEPromise;
+ var case2token = token + "_case2";
+ let v2 = document.createElement("video");
+
+ v2.addEventListener("loadeddata", function(ev) {
+ ok(true, case2token + " should reach loadeddata");
+ var threw = false;
+ try {
+ var context = new AudioContext();
+ context.createMediaElementSource(v2);
+ } catch (e) {
+ threw = true;
+ }
+ ok(!threw, "Should always work when creating a MediaElementSource.");
+ p1.resolve();
+ });
+
+ manager.started(case2token);
+ let p2 = SetupEME(v2, test, case2token);
+
+ Promise.all([p1.promise, p2])
+ .catch(reason => ok(false, reason))
+ .then(() => {
+ CleanUpMedia(v2);
+ manager.finished(case2token);
+ });
+}
+
+SimpleTest.waitForExplicitFinish();
+manager.runTests(gEMETests, startTest);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_stream_capture_blocked_case3.html b/dom/media/test/test_eme_stream_capture_blocked_case3.html
new file mode 100644
index 0000000000..e25a900956
--- /dev/null
+++ b/dom/media/test/test_eme_stream_capture_blocked_case3.html
@@ -0,0 +1,50 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test Encrypted Media Extensions</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="eme.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var manager = new MediaTestManager;
+
+function startTest(test, token)
+{
+ // Case 3. capturing a media element with mozCaptureStream that has a MediaKeys should fail.
+ var p1 = new EMEPromise;
+ var case3token = token + "_case3";
+ let v3 = document.createElement("video");
+
+ v3.addEventListener("loadeddata", function(ev) {
+ ok(true, TimeStamp(case3token) + " should reach loadeddata");
+ var threw = false;
+ try {
+ v3.mozCaptureStreamUntilEnded();
+ } catch (e) {
+ threw = true;
+ }
+ ok(threw, TimeStamp(case3token) + " Should throw an error calling mozCaptureStreamUntilEnded an EME video.");
+ p1.resolve();
+ });
+
+ manager.started(case3token);
+ let p2 = SetupEME(v3, test, case3token);
+
+ Promise.all([p1.promise, p2])
+ .catch(reason => ok(false, reason))
+ .then(() => {
+ CleanUpMedia(v3);
+ manager.finished(case3token);
+ });
+}
+
+SimpleTest.waitForExplicitFinish();
+manager.runTests(gEMETests, startTest);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_unsetMediaKeys_then_capture.html b/dom/media/test/test_eme_unsetMediaKeys_then_capture.html
new file mode 100644
index 0000000000..3ecdc79dbf
--- /dev/null
+++ b/dom/media/test/test_eme_unsetMediaKeys_then_capture.html
@@ -0,0 +1,108 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test Encrypted Media Extensions</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="https://example.com:443/tests/dom/media/test/eme.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+/* import-globals-from eme.js */
+var manager = new MediaTestManager;
+
+// Test that if we can capture a video frame while playing clear content after
+// removing the MediaKeys object which was used for a previous encrypted content
+// playback on the same video element
+function startTest(test, token)
+{
+ manager.started(token);
+ var sessions = [];
+ function onSessionCreated(session) {
+ sessions.push(session);
+ }
+
+ function closeSessions() {
+ let p = new EMEPromise;
+ Promise.all(sessions.map(s => s.close()))
+ .then(p.resolve, p.reject);
+ return p.promise;
+ }
+
+ let v = document.createElement("video");
+ document.body.appendChild(v);
+
+ let finish = new EMEPromise;
+
+ function onVideoEnded(ev) {
+ ok(true, TimeStamp(token) + " (ENCRYPTED) content playback ended.");
+
+ function playClearVideo() {
+ var p1 = once(v, 'loadeddata', (e) => {
+ ok(true, TimeStamp(token) + " Receiving event 'loadeddata' for (CLEAR) content.");
+ let canvasElem = document.createElement('canvas');
+ document.body.appendChild(canvasElem);
+ let ctx2d = canvasElem.getContext('2d');
+
+ var gotTypeError = false;
+ try {
+ ctx2d.drawImage(v, 0, 0);
+ } catch (ex) {
+ if (ex instanceof TypeError) {
+ gotTypeError = true;
+ }
+ }
+ ok(!gotTypeError, TimeStamp(token) + " Canvas2D context drawImage succeed.")
+ });
+ v.src = 'bipbop_225w_175kbps.mp4';
+ v.play();
+ Promise.all([p1]).then(() => {
+ manager.finished(token);
+ }, () => {
+ ok(false, TimeStamp(token) + " Something wrong.");
+ manager.finished(token);
+ });
+ }
+
+ closeSessions()
+ .then(() => {
+ v.setMediaKeys(null)
+ .then(() => {
+ ok(true, TimeStamp(token) + " Setting MediaKeys to null.");
+ playClearVideo();
+ }, () => {
+ ok(false, TimeStamp(token) + " Setting MediaKeys to null.");
+ });;
+ });
+ }
+
+ once(v, "ended", onVideoEnded);
+
+ // Create a MediaKeys object and set to HTMLMediaElement then start the playback.
+ Promise.all([
+ LoadInitData(v, test, token),
+ CreateAndSetMediaKeys(v, test, token),
+ LoadTest(test, v, token)])
+ .then(values => {
+ let initData = values[0];
+ v.play();
+ initData.map(ev => {
+ let session = v.mediaKeys.createSession();
+ onSessionCreated(session);
+ MakeRequest(test, token, ev, session);
+ });
+ })
+ .then(() => {
+ return finish.promise;
+ })
+ .catch(reason => ok(false, reason))
+}
+
+SimpleTest.waitForExplicitFinish();
+manager.runTests(gEMETests, startTest);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_waitingforkey.html b/dom/media/test/test_eme_waitingforkey.html
new file mode 100644
index 0000000000..c12a3b228f
--- /dev/null
+++ b/dom/media/test/test_eme_waitingforkey.html
@@ -0,0 +1,78 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test Encrypted Media Extensions</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="https://example.com:443/tests/dom/media/test/eme.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+/* import-globals-from eme.js */
+var manager = new MediaTestManager;
+
+function startTest(test, token)
+{
+ manager.started(token);
+
+ let v = document.createElement("video");
+ document.body.appendChild(v);
+
+ var gotWaitingForKey = 0;
+ var gotOnwaitingforkey = 0;
+
+ let waitForKey = new EMEPromise;
+ v.addEventListener("waitingforkey", function() {
+ gotWaitingForKey += 1;
+ waitForKey.resolve();
+ });
+
+ v.onwaitingforkey = function() {
+ gotOnwaitingforkey += 1;
+ };
+
+ v.addEventListener("loadedmetadata", function() {
+ ok(SpecialPowers.do_lookupGetter(v, "isEncrypted").apply(v),
+ TimeStamp(token) + " isEncrypted should be true");
+ is(v.isEncrypted, undefined, "isEncrypted should not be accessible from content");
+ });
+
+ let finish = new EMEPromise;
+ v.addEventListener("ended", function() {
+ ok(true, TimeStamp(token) + " got ended event");
+ // We expect only one waitingForKey as we delay until all sessions are ready.
+ // I.e. one waitingForKey should be fired, after which, we process all sessions,
+ // so it should not be possible to be blocked by a key after that point.
+ ok(gotWaitingForKey == 1, "Expected number 1 wait, got: " + gotWaitingForKey);
+ ok(gotOnwaitingforkey == gotWaitingForKey, "Should have as many event listener calls as event handler calls, got: " + gotOnwaitingforkey);
+
+ finish.resolve();
+ });
+
+ Promise.all([
+ LoadInitData(v, test, token),
+ CreateAndSetMediaKeys(v, test, token),
+ LoadTest(test, v, token),
+ waitForKey.promise])
+ .then(values => {
+ let initData = values[0];
+ return ProcessInitData(v, test, token, initData);
+ })
+ .then(sessions => {
+ Log(token, "Updated all sessions, loading complete -> Play");
+ v.play();
+ finish.promise.then(() => CloseSessions(v, sessions));
+ return finish.promise;
+ })
+ .catch(reason => ok(false, reason))
+ .then(() => manager.finished(token));
+}
+
+SimpleTest.waitForExplicitFinish();
+manager.runTests(gEMETests, startTest);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_wideinve_l1_installation.html b/dom/media/test/test_eme_wideinve_l1_installation.html
new file mode 100644
index 0000000000..fa6a0c350b
--- /dev/null
+++ b/dom/media/test/test_eme_wideinve_l1_installation.html
@@ -0,0 +1,86 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test Widevine L1 installation</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="eme.js"></script>
+</head>
+<body>
+<script class="testbody" type="text/javascript">
+
+// This test mostly follows the patten of test_eme_request_notifications.html.
+
+var manager = new MediaTestManager;
+
+function SetPrefs(prefs) {
+ return SpecialPowers.pushPrefEnv({"set": prefs});
+}
+
+function observe() {
+ return new Promise(function(resolve, reject) {
+ var observer = function(subject, topic, data) {
+ SpecialPowers.Services.obs.removeObserver(observer, "mediakeys-request");
+ resolve(JSON.parse(data).status);
+ };
+ SpecialPowers.Services.obs.addObserver(observer, "mediakeys-request");
+ });
+}
+
+function Test(test) {
+ var p = test.prefs ? SetPrefs(test.prefs) : Promise.resolve();
+ var name = "'" + test.keySystem + "'";
+
+ var res = observe().then((status) => {
+ is(status, test.expectedStatus, name + " expected status");
+ });
+
+ p.then(() => navigator.requestMediaKeySystemAccess(test.keySystem, gCencMediaKeySystemConfig))
+ .then((keySystemAccess) => keySystemAccess.createMediaKeys());
+
+ return res;
+}
+
+SimpleTest.waitForExplicitFinish();
+
+const tests = [
+ // GMP L1 pref is disabled
+ {
+ keySystem: "com.widevine.alpha.experiment",
+ expectedStatus: 'cdm-disabled',
+ prefs: [
+ ["media.eme.enabled", true],
+ ["media.wmf.media-engine.enabled", 2],
+ ["media.eme.widevine.experiment.enabled", true],
+ ["media.gmp-widevinecdm-l1.enabled", false],
+ ]
+ },
+ {
+ keySystem: "com.widevine.alpha.experiment2",
+ expectedStatus: 'cdm-disabled',
+ },
+ // Enable GMP L1 pref
+ {
+ keySystem: "com.widevine.alpha.experiment",
+ expectedStatus: 'cdm-not-installed',
+ prefs: [
+ ["media.gmp-widevinecdm-l1.enabled", true],
+ ]
+ },
+ {
+ keySystem: "com.widevine.alpha.experiment2",
+ expectedStatus: 'cdm-not-installed',
+ },
+];
+
+tests.reduce(function(p, c, i, array) {
+ return p.then(function() {
+ return Test(c);
+ });
+}, Promise.resolve())
+.then(SimpleTest.finish);
+
+</script>
+</body>
+</html>
diff --git a/dom/media/test/test_eme_wv_privacy.html b/dom/media/test/test_eme_wv_privacy.html
new file mode 100644
index 0000000000..a050d2a528
--- /dev/null
+++ b/dom/media/test/test_eme_wv_privacy.html
@@ -0,0 +1,53 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test generating encrypted request in Widevine privacy mode</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="eme.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+function Test() {
+ return new Promise(function(resolve, reject) {
+ var configs = [{
+ initDataTypes: ['keyids'],
+ videoCapabilities: [{contentType: 'video/mp4' }],
+ }];
+ navigator.requestMediaKeySystemAccess('com.widevine.alpha', configs)
+ .then((access) => access.createMediaKeys(),
+ (e) => { throw e; })
+ .then((mediaKeys) => {
+ var session = mediaKeys.createSession('temporary');
+ session.addEventListener('message', event => {
+ is(event.messageType, 'license-request',
+ 'MediaKeyMessage type should be license-request.');
+ const request = new Uint8Array(event.message);
+ info('request generated[' + request.length + ']: [' + request + ']');
+ ok(request.length == 2 && request[0] == 0x08 && request[1] == 0x04,
+ 'Widevine license request should be [0x08, 0x04] in privacy mode');
+ session.close().then(() => { resolve(); });
+ });
+ return session.generateRequest('keyids',
+ new TextEncoder().encode('{"kids":["MDAwMDAwMDAwMDAwMDAwMQ=="]}'));
+ },
+ (e) => { throw e; })
+ .catch((e) => {
+ is(e.name, 'NotSupportedError',
+ 'Should only be rejected when "not supported"');
+ resolve();
+ });
+ });
+}
+
+SimpleTest.waitForExplicitFinish();
+Test().then(function() {
+ SimpleTest.finish();
+});
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_empty_resource.html b/dom/media/test/test_empty_resource.html
new file mode 100644
index 0000000000..8f65bb503d
--- /dev/null
+++ b/dom/media/test/test_empty_resource.html
@@ -0,0 +1,58 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=1094549
+-->
+<head>
+ <title>Test for Bug 1094549</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=1094549">Mozilla Bug 1094549</a>
+
+<p id="display"></p>
+<div id="content" style="display: none">
+
+</div>
+<pre id="test">
+<script type="application/javascript">
+
+// Shorter timeout for this test should finish soon.
+SimpleTest.requestLongerTimeout(0.3);
+
+function finish(v) {
+ isnot(v.error, null, "should've got an error event");
+ SimpleTest.finish();
+}
+
+function onload() {
+ info("iframe loaded");
+ var v = SpecialPowers.wrap(document.body.getElementsByTagName("iframe")[0])
+ .contentDocument.body.getElementsByTagName("video")[0];
+
+ // Got 'error' as expected, finish the test.
+ if (v.error) {
+ finish(v);
+ return;
+ }
+
+ // Otherwise, wait for it.
+ v.onerror = function() {
+ finish(v);
+ }
+}
+
+SimpleTest.waitForExplicitFinish();
+var f = document.createElement("iframe");
+// Assign a resource file with zero length and expect the error event from
+// the video element since decoding metadata will fail.
+f.src = "data:video/webm,";
+f.addEventListener("load", onload);
+document.body.appendChild(f);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_error_in_video_document.html b/dom/media/test/test_error_in_video_document.html
new file mode 100644
index 0000000000..e376ea95e3
--- /dev/null
+++ b/dom/media/test/test_error_in_video_document.html
@@ -0,0 +1,59 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=604067
+-->
+<head>
+ <title>Test for Bug 604067</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=604067">Mozilla Bug 604067</a>
+
+<p id="display"></p>
+<div id="content" style="display: none">
+
+</div>
+<pre id="test">
+<script type="application/javascript">
+
+/** Test for Bug 604067 **/
+
+function documentVideo() {
+ return document.body.getElementsByTagName("iframe")[0]
+ .contentDocument.body.getElementsByTagName("video")[0];
+}
+
+function check() {
+ var v = documentVideo();
+
+ // Debug info for Bug 608634
+ ok(true, "iframe src=" + document.body.getElementsByTagName("iframe")[0].src);
+ is(v.readyState, v.HAVE_NOTHING, "Ready state");
+
+ isnot(v.error, null, "Error object");
+ is(v.networkState, v.NETWORK_NO_SOURCE, "Network state");
+ is(v.error.code, MediaError.MEDIA_ERR_SRC_NOT_SUPPORTED, "Expected media not supported error");
+ SimpleTest.finish();
+}
+
+// Find an error test that we'd think we should be able to play (if it
+// wasn't already known to fail).
+var t = getPlayableVideo(gErrorTests);
+if (!t) {
+ todo(false, "No types supported");
+} else {
+ SimpleTest.waitForExplicitFinish();
+
+ var f = document.createElement("iframe");
+ f.src = t.name;
+ f.addEventListener("load", check);
+ document.body.appendChild(f);
+}
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_error_on_404.html b/dom/media/test/test_error_on_404.html
new file mode 100644
index 0000000000..8f82b8a07e
--- /dev/null
+++ b/dom/media/test/test_error_on_404.html
@@ -0,0 +1,84 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=476731
+-->
+<head>
+ <title>Test for Bug 476731</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=476731">Mozilla Bug </a>
+<p id="display"></p>
+<div id="content" style="display: none">
+</div>
+<pre id="test">
+<script type="application/javascript">
+/** Test for Bug 476731 **/
+
+var videos = [];
+
+function FinishedLoads() {
+ if (!videos.length)
+ return false;
+ for (var i=0; i<videos.length; ++i) {
+ if (videos[i]._loadedData) {
+ // A video loadeddata, it should have 404'd. Signal the end of the test
+ // so the error will be reported.
+ return true;
+ }
+ if (!videos[i]._loadError) {
+ return false;
+ }
+ }
+ return true;
+}
+
+function loadError(evt) {
+ var v = evt.target;
+ is(v._loadError, false, "Shouldn't receive multiple error events for " + v.src);
+ v._loadError = true;
+ is(v._loadStart, true, "Receive loadstart for " + v.src);
+ is(v._loadedData, false, "Shouldn't receive loadeddata for " + v.src);
+ if (FinishedLoads(videos))
+ SimpleTest.finish();
+}
+
+function loadedData(evt) {
+ evt.target._loadedData = true;
+}
+
+function loadStart(evt) {
+ evt.target._loadStart = true;
+}
+
+// Create all video objects.
+for (var i=0; i<g404Tests.length; ++i) {
+ var v = document.createElement("video");
+ v.preload = "metadata";
+ var test = g404Tests[i];
+ if (!v.canPlayType(test.type)) {
+ continue;
+ }
+ v.src = test.name;
+ v._loadedData = false;
+ v._loadStart = false;
+ v._loadError = false;
+ v.addEventListener("error", loadError);
+ v.addEventListener("loadstart", loadStart);
+ v.addEventListener("loadeddata", loadedData);
+ document.body.appendChild(v); // Will start load.
+ videos.push(v);
+}
+
+if (!videos.length) {
+ todo(false, "No types supported");
+} else {
+ SimpleTest.waitForExplicitFinish();
+}
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_fastSeek-forwards.html b/dom/media/test/test_fastSeek-forwards.html
new file mode 100644
index 0000000000..1e50021b13
--- /dev/null
+++ b/dom/media/test/test_fastSeek-forwards.html
@@ -0,0 +1,77 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=1022913
+-->
+<head>
+ <meta charset="utf-8">
+ <title>Test for Bug 1022913</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=1022913">Mozilla Bug 1022913</a>
+<p id="display"></p>
+<div id="content" style="display: none">
+
+</div>
+<pre id="test">
+</pre>
+ <script type="application/javascript">
+
+ // Test that if we're doing a fastSeek() forwards that we don't end up
+ // seeking backwards. This can happen if the keyframe before the seek
+ // target is before the current playback position. We'd prefer to seek to
+ // the keyframe after the seek target in this case, but we don't implement
+ // this yet (bug 1026330).
+ var manager = new MediaTestManager;
+
+ var onSecondSeekComplete = function(event) {
+ var v = event.target;
+ v.removeEventListener("seeked", onSecondSeekComplete);
+ ok(v.currentTime >= v.firstSeekTarget, v.name + " seek never go backwards. time=" + v.currentTime + " firstSeekTarget=" + v.firstSeekTarget + " secondSeekTarget=" + v.secondSeekTarget);
+ manager.finished(v.token);
+ removeNodeAndSource(v);
+ };
+
+ var onFirstSeekComplete = function(event) {
+ var v = event.target;
+ v.removeEventListener("seeked", onFirstSeekComplete);
+ // Seek to 75% of the way between the start and the first keyframe
+ // using fastSeek. We then test that the currentTime doesn't drop back
+ // to the previous keyframe, currentTime should go forwards.
+ v.addEventListener("seeked", onSecondSeekComplete);
+ v.secondSeekTarget = v.keyframes[1] * 0.75;
+ v.fastSeek(v.secondSeekTarget);
+ }
+
+ var onLoadedMetadata = function(event) {
+ // Seek to the mid-point between the start and the first keyframe.
+ var v = event.target;
+ v.removeEventListener("loadedmetadata", onLoadedMetadata);
+ v.addEventListener("seeked", onFirstSeekComplete);
+ v.firstSeekTarget = v.keyframes[1] * 0.5;
+ v.currentTime = v.firstSeekTarget;
+ }
+
+ function startTest(test, token) {
+ manager.started(token);
+ let v = document.createElement("video");
+ v.src = test.name;
+ v.name = test.name;
+ v.preload = "metadata";
+ v.token = token;
+ v.target = 0;
+ v.keyframes = test.keyframes;
+ v.keyframeIndex = 0;
+ ok(v.keyframes.length >= 2, v.name + " - video should have at least two sync points");
+ v.addEventListener("loadedmetadata", onLoadedMetadata);
+ document.body.appendChild(v);
+ }
+
+ manager.runTests(gFastSeekTests, startTest);
+
+ </script>
+</body>
+</html>
diff --git a/dom/media/test/test_fastSeek.html b/dom/media/test/test_fastSeek.html
new file mode 100644
index 0000000000..2245bcc9bf
--- /dev/null
+++ b/dom/media/test/test_fastSeek.html
@@ -0,0 +1,88 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=778077
+-->
+<head>
+ <meta charset="utf-8">
+ <title>Test for Bug 778077</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=778077">Mozilla Bug 778077</a>
+<p id="display"></p>
+<div id="content" style="display: none">
+
+</div>
+<pre id="test">
+</pre>
+ <script type="application/javascript">
+
+ /** Test for Bug 778077 - HTMLMediaElement.fastSeek() **/
+ // Iterate through a list of keyframe timestamps, and seek to
+ // halfway between the keyframe and the keyframe after it.
+ var manager = new MediaTestManager;
+
+ function doSeek(v) {
+ // fastSeek to half way between this keyframe and the next, or if this is the last
+ // keyframe seek to halfway between this keyframe and the end of media.
+ var nextKeyFrame = (v.keyframeIndex + 1) < v.keyframes.length ? v.keyframes[v.keyframeIndex + 1] : v.duration;
+ v.target = (v.keyframes[v.keyframeIndex] + nextKeyFrame) / 2;
+ v.fastSeek(v.target);
+ ok(Math.abs(v.currentTime - v.target) < 0.01,
+ v.name + " seekTo=" + v.target + " currentTime (" + v.currentTime + ") should be close to seek target initially");
+ }
+
+ function onloadedmetadata(event) {
+ var v = event.target;
+ doSeek(v);
+ }
+
+ function onseeked(event) {
+ var v = event.target;
+ var keyframe = v.keyframes[v.keyframeIndex];
+
+ // Check that the current time ended up roughly after the keyframe.
+ // We must be a bit fuzzy here, as the decoder backend may actually
+ // seek to the audio sample prior to the keyframe.
+ ok(v.currentTime >= keyframe - 0.05,
+ v.name + " seekTo=" + v.target + " currentTime (" + v.currentTime +
+ ") should be end up roughly after keyframe (" + keyframe + ") after fastSeek");
+
+ ok(v.currentTime <= v.target,
+ v.name + " seekTo=" + v.target + " currentTime (" + v.currentTime +
+ ") should be end up less than target after fastSeek");
+
+ v.keyframeIndex++
+ if (v.keyframeIndex == v.keyframes.length) {
+ v.src = "";
+ v.remove();
+ manager.finished(v.token);
+ } else {
+ doSeek(v);
+ }
+ }
+
+ function startTest(test, token) {
+ manager.started(token);
+ let v = document.createElement("video");
+ v.src = test.name;
+ v.name = test.name;
+ v.preload = "metadata";
+ v.token = token;
+ v.target = 0;
+ v.keyframes = test.keyframes;
+ v.keyframeIndex = 0;
+ ok(v.keyframes.length, v.name + " - video should have at least one sync point");
+ v.addEventListener("loadedmetadata", onloadedmetadata);
+ v.addEventListener("seeked", onseeked);
+ document.body.appendChild(v);
+ }
+
+ manager.runTests(gFastSeekTests, startTest);
+
+ </script>
+</body>
+</html>
diff --git a/dom/media/test/test_fragment_noplay.html b/dom/media/test/test_fragment_noplay.html
new file mode 100644
index 0000000000..6a1119f342
--- /dev/null
+++ b/dom/media/test/test_fragment_noplay.html
@@ -0,0 +1,127 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: fragment tests</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="fragment_noplay.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+// Fragment parameters to try
+var gFragmentParams = [
+ // W3C Media fragment tests
+ // http://www.w3.org/2008/WebVideo/Fragments/TC/ua-test-cases
+ { fragment: "#t=banana", start: null, end: null }, // TC0027-UA
+ { fragment: "#t=3,banana", start: null, end: null }, // TC0028-UA
+ { fragment: "#t=banana,7", start: null, end: null }, // TC0029-UA
+ { fragment: "#t='3'", start: null, end: null }, // TC0030-UA
+ { fragment: "#t=3-7", start: null, end: null }, // TC0031-UA
+ { fragment: "#t=3:7", start: null, end: null }, // TC0032-UA
+ { fragment: "#t=3,7,9", start: null, end: null }, // TC0033-UA
+ { fragment: "#t%3D3", start: null, end: null }, // TC0034-UA
+ { fragment: "#%74=3", start: 3, end: null }, // TC0035-UA
+ { fragment: "#t=%33", start: 3, end: null }, // TC0036-UA
+ { fragment: "#t=3%2C7", start: 3, end: 7 }, // TC0037-UA
+ { fragment: "#t=%6Ept:3", start: 3, end: null }, // TC0038-UA
+ { fragment: "#t=npt%3A3", start: 3, end: null }, // TC0039-UA
+ { fragment: "#t=-1,3", start: null, end: null }, // TC0044-UA
+ { fragment: "#t=3&", start: 3, end: null }, // TC0051-UA
+ { fragment: "#u=12&t=3", start: 3, end: null }, // TC0052-UA
+ { fragment: "#t=foo:7&t=npt:3", start: 3, end: null }, // TC0053-UA
+ { fragment: "#&&=&=tom&jerry=&t=3&t=meow:0#", start: 3, end: null }, // TC0054-UA
+ { fragment: "#t=7&t=3", start: 3, end: null }, // TC0055-UA
+ { fragment: "#T=3,7", start: null, end: null }, // TC0058-UA
+ { fragment: "#t=", start: null, end: null }, // TC0061-UA
+ { fragment: "#t=.", start: null, end: null }, // TC0062-UA
+ { fragment: "#t=.0", start: null, end: null }, // TC0063-UA
+ { fragment: "#t=0s", start: null, end: null }, // TC0064-UA
+ { fragment: "#t=,0s", start: null, end: null }, // TC0065-UA
+ { fragment: "#t=0s,0s", start: null, end: null }, // TC0066-UA
+ { fragment: "#t=00:00:00s", start: null, end: null }, // TC0067-UA
+ { fragment: "#t=s", start: null, end: null }, // TC0068-UA
+ { fragment: "#t=npt:", start: null, end: null }, // TC0069-UA
+ { fragment: "#t=1e-1:", start: null, end: null }, // TC0070-UA
+ { fragment: "#t=00:00:01.1e-1", start: null, end: null }, // TC0071-UA
+ { fragment: "#t=3.", start: 3, end: null }, // TC0072-UA
+ { fragment: "#t=0:0:0", start: null, end: null }, // TC0073-UA
+ { fragment: "#t=0:00:60", start: null, end: null }, // TC0074-UA
+ { fragment: "#t=0:01:60", start: null, end: null }, // TC0075-UA
+ { fragment: "#t=0:60:00", start: null, end: null }, // TC0076-UA
+ { fragment: "#t=0:000:000", start: null, end: null }, // TC0077-UA
+ { fragment: "#t=00:00:03,00:00:07", start: 3, end: 7 }, // TC0078-UA
+ { fragment: "#t=3,00:00:07", start: 3, end: 7 }, // TC0079-UA
+ { fragment: "#t=00:00.", start: null, end: null }, // TC0080-UA
+ { fragment: "#t=0:00:00.", start: null, end: null }, // TC0081-UA
+ { fragment: "#t=0:00:10e-1", start: null, end: null }, // TC0082-UA
+ { fragment: "#t=0:00:60.000", start: null, end: null }, // TC0083-UA
+ { fragment: "#t=0:60:00.000", start: null, end: null }, // TC0084-UA
+ { fragment: "#t=3,7&t=foo", start: 3, end: 7 }, // TC0085-UA
+ { fragment: "#foo&t=3,7", start: 3, end: 7 }, // TC0086-UA
+ { fragment: "#t=3,7&foo", start: 3, end: 7 }, // TC0087-UA
+ { fragment: "#t=3,7&&", start: 3, end: 7 }, // TC0088-UA
+ { fragment: "#&t=3,7", start: 3, end: 7 }, // TC0089-UA
+ { fragment: "#&&t=3,7", start: 3, end: 7 }, // TC0090-UA
+ { fragment: "#&t=3,7&", start: 3, end: 7 }, // TC0091-UA
+ { fragment: "#t%3d10", start: null, end: null }, // TC0092-UA
+ { fragment: "#t=10%26", start: null, end: null }, // TC0093-UA
+ { fragment: "#&t=3,7,", start: null, end: null } // TC0094-UA
+];
+
+function createTestArray() {
+ var tests = [];
+ var tmpVid = document.createElement("video");
+
+ for (var testNum=0; testNum<gFragmentTests.length; testNum++) {
+ var test = gFragmentTests[testNum];
+ if (!tmpVid.canPlayType(test.type)) {
+ continue;
+ }
+
+ for (var fragNum=0; fragNum<gFragmentParams.length; fragNum++) {
+ var p = gFragmentParams[fragNum];
+ var t = {};
+ t.name = test.name + p.fragment;
+ t.type = test.type;
+ t.duration = test.duration;
+ t.start = p.start;
+ t.end = p.end;
+ tests.push(t);
+ }
+ }
+ return tests;
+}
+
+function startTest(test, token) {
+ var video = document.createElement('video');
+ manager.started(token);
+ video.preload = "metadata";
+ video.src = test.name;
+ video.token = token;
+ video.controls = true;
+ document.body.appendChild(video);
+ var name = test.name + " fragment test";
+ var localIs = function(n) { return function(a, b, msg) {
+ is(a, b, n + ": " + msg);
+ }}(name);
+ var localOk = function(n) { return function(a, msg) {
+ ok(a, n + ": " + msg);
+ }}(name);
+ var localFinish = function(v, m) { return function() {
+ removeNodeAndSource(v);
+ m.finished(v.token);
+ }}(video, manager);
+ window.test_fragment_noplay(video, test.start, test.end, localIs, localOk, localFinish);
+}
+
+manager.runTests(createTestArray(), startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_fragment_play.html b/dom/media/test/test_fragment_play.html
new file mode 100644
index 0000000000..eab422bab1
--- /dev/null
+++ b/dom/media/test/test_fragment_play.html
@@ -0,0 +1,91 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: seek tests</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="fragment_play.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+PARALLEL_TESTS = 1;
+var manager = new MediaTestManager;
+
+// Fragment parameters to try. These tests
+// try playing the video. Tests for other fragment
+// formats are in test_fragment_noplay.html.
+var gFragmentParams = [
+ { fragment: "", start: null, end: null },
+ { fragment: "#t=,", start: null, end: null },
+ { fragment: "#t=3,3", start: null, end: null },
+ { fragment: "#t=7,3", start: null, end: null },
+ { fragment: "#t=7,15", start: 7, end: null },
+ { fragment: "#t=15,20", start: 9.287982, end: null },
+ { fragment: "#t=5", start: 5, end: null },
+ { fragment: "#t=5.5", start: 5.5, end: null },
+ { fragment: "#t=5,", start: null, end: null },
+ { fragment: "#t=,5", start: 0, end: 5, todo: "See bugs 682141 and 720248" },
+ { fragment: "#t=2.5,5.5", start: 2.5, end: 5.5, todo: "See bugs 682141 and 720248" },
+ { fragment: "#t=1,2.5", start: 1, end: 2.5, todo: "See bugs 682141 and 720248" },
+ { fragment: "#t=,15", start: 0, end: null }
+];
+
+function createTestArray() {
+ var tests = [];
+ var tmpVid = document.createElement("video");
+
+ for (var testNum=0; testNum<gFragmentTests.length; testNum++) {
+ var test = gFragmentTests[testNum];
+ if (!tmpVid.canPlayType(test.type)) {
+ continue;
+ }
+
+ for (var fragNum=0; fragNum<gFragmentParams.length; fragNum++) {
+ var p = gFragmentParams[fragNum];
+ var t = {};
+ t.name = test.name + p.fragment;
+ t.type = test.type;
+ t.duration = test.duration;
+ t.start = p.start;
+ t.end = p.end;
+ t.todo = p.todo;
+ tests.push(t);
+ }
+ }
+ return tests;
+}
+
+function startTest(test, token) {
+ if (test.todo) {
+ todo(false, test.todo);
+ return;
+ }
+ var video = document.createElement('video');
+ manager.started(token);
+ video.preload = "metadata";
+ video.src = test.name;
+ video.token = token;
+ video.controls = true;
+ document.body.appendChild(video);
+ var name = test.name + " fragment test";
+ var localIs = function(n) { return function(a, b, msg) {
+ is(a, b, n + ": " + msg);
+ }}(name);
+ var localOk = function(n) { return function(a, msg) {
+ ok(a, n + ": " + msg);
+ }}(name);
+ var localFinish = function(v, m) { return function() {
+ removeNodeAndSource(v);
+ m.finished(v.token);
+ }}(video, manager);
+ window.test_fragment_play(video, test.start, test.end, localIs, localOk, localFinish);
+}
+
+manager.runTests(createTestArray(), startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_hevc_playback.html b/dom/media/test/test_hevc_playback.html
new file mode 100644
index 0000000000..6327008102
--- /dev/null
+++ b/dom/media/test/test_hevc_playback.html
@@ -0,0 +1,45 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+<title>Test HEVC video playback</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+<script type="application/javascript">
+
+/**
+ * This test is used to check whether non-MSE HEVC video can be played on the
+ * platform. HEVC can only be played if we have platform decoder supports
+ * hardware decoding. Currently we can only do that on Windows.
+ */
+add_task(async function testHEVCPlayback() {
+ let video = document.createElement('video');
+ info(`try to play HEVC video`);
+ video.src = "hevc_white_frame.mp4";
+ video.controls = true;
+ document.body.appendChild(video);
+ ok(await video.play().then(_=>true, _=>false), "video started playing");
+ ok(await new Promise(r => video.onended = r).then(_=>true, _=>false),
+ "video played to end");
+
+});
+
+
+// This test video contains an inband SPS that is different from the SPS in the
+// extradata, so we would recreate a new decoder.
+add_task(async function testHEVCInbandConfigChange() {
+ let video = document.createElement('video');
+ info(`try to play HEVC video`);
+ video.src = "hevc_white_red_frames.mp4";
+ video.playbackRate = 2.0; // speed up the test.
+ video.controls = true;
+ document.body.appendChild(video);
+ ok(await video.play().then(_=>true, _=>false), "video started playing");
+ ok(await new Promise(r => video.onended = r).then(_=>true, _=>false),
+ "video played to end");
+});
+
+</script>
+</head>
+<body>
+</body>
+</html>
diff --git a/dom/media/test/test_hevc_support.html b/dom/media/test/test_hevc_support.html
new file mode 100644
index 0000000000..36fe44a500
--- /dev/null
+++ b/dom/media/test/test_hevc_support.html
@@ -0,0 +1,47 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+<title>Test HEVC video support</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+<script type="application/javascript">
+
+/**
+ * This test is used to check we can get correct HEVC support information from
+ * `canPlayType` and mediaCapabilities API on different platforms.
+ */
+add_task(async function testHEVCSupport() {
+ const os = SpecialPowers.Services.appinfo.OS;
+
+ const contentType = 'video/mp4; codecs="hev1.1.6.L93.B0"';
+ const canPlay = document.createElement('video').canPlayType(contentType);
+ if (os == "WINNT") {
+ ok(canPlay != "", `HEVC is supported`);
+ } else {
+ ok(canPlay == "", "HEVC is not supported on current platform");
+ }
+
+ const videoConfiguration = {
+ type: "file",
+ video: {
+ contentType,
+ width: 1920,
+ height: 1080,
+ bitrate: 1000000,
+ framerate: 30,
+ },
+ };
+ const capability = await navigator.mediaCapabilities.decodingInfo(videoConfiguration);
+ if (os == "WINNT") {
+ ok(capability.supported, `HEVC is supported`);
+ ok(capability.powerEfficient, `HEVC is powerEfficient`);
+ } else {
+ ok(!capability.supported, "HEVC is not supported on current platform");
+ }
+});
+
+</script>
+</head>
+<body>
+</body>
+</html>
diff --git a/dom/media/test/test_hls_player_independency.html b/dom/media/test/test_hls_player_independency.html
new file mode 100644
index 0000000000..cea5c140ed
--- /dev/null
+++ b/dom/media/test/test_hls_player_independency.html
@@ -0,0 +1,53 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test playback of 2 HLS video at the same page </title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+ <div id='player1'>
+ <video id='player4x3' controls autoplay>
+ </video>
+ </div>
+ <p> 4x3 basic stream<span>
+ <span>
+ <div height = 10>
+ <span>
+ <div id='player2'>
+ <video id='player16x9' controls autoplay>
+ </video>
+ </div>
+ <p> 16x9 basic stream<span>
+
+<script class="testbody" type="text/javascript">
+
+function startTest() {
+ var v4x3 = document.getElementById('player4x3');
+ var v16x9 = document.getElementById('player16x9');
+
+ var p1 = once(v4x3, 'ended', function onended(e) {
+ is(v4x3.videoWidth, 400, "4x3 content, the width should be 400.");
+ is(v4x3.videoHeight, 300, "4x3 content, the height should be 300.");
+ });
+
+ var p2 = once(v16x9, 'ended', function onended(e) {
+ is(v16x9.videoWidth, 416, "16x9 content, the width should be 416.");
+ is(v16x9.videoHeight, 234, "16x9 content, the height should be 234.");
+ });
+
+ v4x3.src = serverUrl + "/bipbop_4x3_single.m3u8";
+ v16x9.src = serverUrl + "/bipbop_16x9_single.m3u8";
+ Promise.all([p1, p2]).then(() => {
+ SimpleTest.finish();
+ });
+}
+
+startTest();
+SimpleTest.waitForExplicitFinish();
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_hw_video_decoding.html b/dom/media/test/test_hw_video_decoding.html
new file mode 100644
index 0000000000..f93ab9b0a8
--- /dev/null
+++ b/dom/media/test/test_hw_video_decoding.html
@@ -0,0 +1,119 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+<title>Test video hardware decoding</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+<script type="application/javascript">
+
+/**
+ * This test is used to check hardware video decoding on different platforms.
+ */
+const PLATFORMS = {
+ // TODO : add Linux and Android, media-gpu doesn't run on them yet
+ WINNT: {
+ process: "GPU",
+ tests: [
+ {
+ file: "gizmo.mp4",
+ decoder: "wmf H264 codec hardware video decoder",
+ },
+ // TODO : enable VP9 for Windows. The current machine on CI doesn't
+ // support VP9 HW. See bug 1861755.
+ // {
+ // file: "gizmo.webm",
+ // decoder: "wmf VP9 codec hardware video decoder",
+ // },
+ // TODO : add AV1, see bug 1861501.
+ ],
+ },
+ Darwin: {
+ process: "RDD",
+ tests: [
+ {
+ file: "gizmo.mp4",
+ decoder: "apple hardware VT decoder",
+ },
+ // TODO : enable VP9 for MacOS. The current machine on CI doesn't support
+ // VP9 HW. See bug 1861741.
+ // {
+ // file: "gizmo.webm",
+ // decoder: "apple hardware VT decoder",
+ // },
+ ],
+ },
+};
+
+add_task(async function testHardwareVideoDecoding() {
+ const platformName = SpecialPowers.Services.appinfo.OS;
+ const platformTest = PLATFORMS[platformName];
+ for (const test of platformTest.tests) {
+ info(
+ `Testing ${test.file} on ${platformName} : expect ${test.decoder} in ${platformTest.process}`
+ );
+ await createAndPlayVideo(test.file);
+ await assertRunningProcessAndDecoderName({
+ expectedProcess: platformTest.process,
+ expectedDecoder: test.decoder,
+ });
+ await waitVideoPlayToEnd();
+ }
+});
+
+// Following are helper functions
+async function createAndPlayVideo(fileUrl) {
+ const video = document.createElement("video");
+ document.body.appendChild(video);
+ video.src = fileUrl;
+ ok(
+ await video.play().then(
+ () => true,
+ () => false
+ ),
+ "video started playing"
+ );
+ // Waiting for timeupdate to ensure the video decoder has been created
+ // correctly.
+ await new Promise(r => (video.ontimeupdate = r));
+}
+
+async function assertRunningProcessAndDecoderName(
+ { expectedProcess, expectedDecoder } = {}
+) {
+ const video = document.querySelector("video");
+ ok(!video.paused, "checking a playing video that should be hw decoding");
+
+ const debugInfo = await SpecialPowers.wrap(video).mozRequestDebugInfo();
+ const videoDecoderName = debugInfo.decoder.reader.videoDecoderName;
+
+ const isExpectedDecoder =
+ videoDecoderName.indexOf(`${expectedDecoder}`) == 0;
+ ok(
+ isExpectedDecoder,
+ `Playback running by decoder '${videoDecoderName}', expected '${expectedDecoder}'`
+ );
+
+ const isExpectedProcess =
+ videoDecoderName.indexOf(`(${expectedProcess} remote)`) > 0;
+ ok(
+ isExpectedProcess,
+ `Playback running in process '${videoDecoderName}', expected '${expectedProcess}'`
+ );
+}
+
+async function waitVideoPlayToEnd() {
+ const video = document.querySelector("video");
+ if (video.ended) {
+ ok(true, "video reached to end!");
+ } else {
+ await new Promise(r => (video.onended = r));
+ }
+ // Remove current video so that we can check next video later.
+ document.body.removeChild(video);
+}
+
+</script>
+</head>
+<body>
+</body>
+</html>
diff --git a/dom/media/test/test_imagecapture.html b/dom/media/test/test_imagecapture.html
new file mode 100644
index 0000000000..d61dd358bb
--- /dev/null
+++ b/dom/media/test/test_imagecapture.html
@@ -0,0 +1,128 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=1041393
+-->
+<head>
+ <meta charset="utf-8">
+ <title>ImageCapture tests</title>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script type="text/javascript" src="gUM_support.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=1041393">ImageCapture tests</a>
+<script type="application/javascript">
+
+// Check if the callback returns even no JS reference on it.
+async function gcTest(track) {
+ const repeat = 100;
+ const promises = [];
+ for (let i = 0; i < repeat; ++i) {
+ const imageCapture = new ImageCapture(track);
+ promises.push(new Promise((resolve, reject) => {
+ imageCapture.onphoto = resolve;
+ imageCapture.onerror = () =>
+ reject(new Error(`takePhoto gcTest failure for capture ${i}`));
+ }));
+ imageCapture.takePhoto();
+ }
+ info("Call gc ");
+ SpecialPowers.gc();
+ await Promise.all(promises);
+}
+
+// Continue calling takePhoto() in rapid succession.
+async function rapidTest(track) {
+ const repeat = 100;
+ const imageCapture = new ImageCapture(track);
+ // "error" can fire synchronously in `takePhoto`
+ const errorPromise = new Promise(r => imageCapture.onerror = r);
+ for (let i = 0; i < repeat; ++i) {
+ imageCapture.takePhoto();
+ }
+ for (let i = 0; i < repeat; ++i) {
+ await Promise.race([
+ new Promise(r => imageCapture.onphoto = r),
+ errorPromise.then(() => Promise.reject(new Error("Capture failed"))),
+ ]);
+ }
+}
+
+// Check if the blob is decodable.
+async function blobTest(track) {
+ const imageCapture = new ImageCapture(track);
+ // "error" can fire synchronously in `takePhoto`
+ const errorPromise = new Promise(r => imageCapture.onerror = r);
+ imageCapture.takePhoto();
+ const blob = await Promise.race([
+ new Promise(r => imageCapture.onphoto = r),
+ errorPromise.then(() => Promise.reject(new Error("Capture failed"))),
+ ]);
+
+ const img = new Image();
+ img.src = URL.createObjectURL(blob.data);
+ await new Promise((resolve, reject) => {
+ img.onload = resolve;
+ img.onerror = () => reject(new Error("Decode failed"));
+ });
+}
+
+// It should return an error event after disabling video track.
+async function trackTest(track) {
+ const imageCapture = new ImageCapture(track);
+ track.enabled = false;
+ try {
+ const errorPromise = new Promise(r => imageCapture.onerror = r);
+ imageCapture.takePhoto();
+ const error = await Promise.race([
+ errorPromise,
+ new Promise(r => imageCapture.onphoto = r).then(
+ () => Promise.reject(new Error("Disabled video track should fail"))),
+ ]);
+ is(error.imageCaptureError.code, error.imageCaptureError.PHOTO_ERROR,
+ "Expected error code")
+ } finally {
+ track.enabled = true;
+ }
+}
+
+async function init() {
+ // Use loopback camera if available, otherwise fake.
+ // MediaTrackGraph will be the backend of ImageCapture.
+ await setupGetUserMediaTestPrefs();
+ let stream = await navigator.mediaDevices.getUserMedia({video: true});
+ return stream.getVideoTracks()[0];
+}
+
+async function start() {
+ try {
+ const track = await init();
+ info("ImageCapture track disable test.");
+ await trackTest(track);
+ info("ImageCapture blob test.");
+ await blobTest(track);
+ info("ImageCapture rapid takePhoto() test.");
+ await rapidTest(track);
+ info("ImageCapture multiple instances test.");
+ await gcTest(track);
+ } catch (e) {
+ ok(false, "Unexpected error during test: " + e);
+ } finally {
+ SimpleTest.finish();
+ }
+}
+
+SimpleTest.requestCompleteLog();
+SimpleTest.waitForExplicitFinish();
+
+SpecialPowers.pushPrefEnv({
+ "set": [
+ ["dom.imagecapture.enabled", true],
+ ["media.navigator.permission.disabled", true],
+ ],
+}, start);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_info_leak.html b/dom/media/test/test_info_leak.html
new file mode 100644
index 0000000000..e09b63b031
--- /dev/null
+++ b/dom/media/test/test_info_leak.html
@@ -0,0 +1,174 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=478957
+-->
+<head>
+ <title>Test for Bug 478957</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=478957">Mozilla Bug 478957</a>
+<p id="display"></p>
+<div id="content" style="display: none">
+
+</div>
+
+<div id="log" style="font-size: small;"></div>
+
+<pre id="test">
+<script type="application/javascript">
+
+/** Test for Bug 478957 **/
+
+// Tests whether we leak events and state change info when loading stuff from local files from a webserver.
+
+var manager = new MediaTestManager;
+
+var gEventTypes = [ 'loadstart', 'progress', 'suspend', 'abort', 'error', 'emptied', 'stalled', 'play', 'pause', 'loadedmetadata', 'loadeddata', 'waiting', 'playing', 'canplay', 'canplaythrough', 'seeking', 'seeked', 'timeupdate', 'ended', 'ratechange', 'durationchange', 'volumechange' ];
+
+var gExpectedEvents = ['loadstart', 'error'];
+
+function createTestArray() {
+ var tests = [];
+ var tmpVid = document.createElement("video");
+
+ return makeInfoLeakTests().then(infoLeakTests => {
+ for (var testNum=0; testNum < infoLeakTests.length; testNum++) {
+ var test = infoLeakTests[testNum];
+ if (!tmpVid.canPlayType(test.type)) {
+ continue;
+ }
+
+ var t = {};
+ t.name = test.src;
+ t.type = test.type;
+
+ tests.push(t);
+ }
+ return tests;
+ })
+}
+
+function log(msg) {
+ info(msg);
+ var l = document.getElementById('log');
+ l.innerHTML += msg + "<br>";
+}
+
+function finish(v) {
+ log("finish: " + v.name);
+ clearInterval(v.checkStateInterval);
+
+ for (var i=0; i<gEventTypes.length; i++) {
+ v.removeEventListener(gEventTypes[i], listener);
+ }
+ removeNodeAndSource(v);
+
+ manager.finished(v.token);
+ v = null;
+}
+
+function listener(evt) {
+ var v = evt.target;
+ log(filename(v.name) + ': got ' + evt.type);
+
+ // On slow machines like B2G emulator, progress timer could time out before
+ // receiving any HTTP notification. We will ignore the 'stalled' event to
+ // pass the tests.
+ if (evt.type == 'stalled') {
+ return;
+ }
+
+ ok(v.eventNum < gExpectedEvents.length, filename(v.name) + " Too many events received");
+ var expected = (v.eventNum < gExpectedEvents.length) ? gExpectedEvents[v.eventNum] : "NoEvent";
+ is(evt.type, expected, filename(v.name) + " Events received in wrong order");
+ v.eventNum++;
+ if (v.eventNum == gExpectedEvents.length) {
+ // In one second, move onto the next test. This give a chance for any
+ // other events to come in. Note: we don't expect any events to come
+ // in, unless we've leaked some info, and 1 second should be enough time
+ // for the leak to show up.
+ setTimeout(function() {finish(v);}, 1000);
+ }
+}
+
+function createMedia(type, src, token) {
+ var tag = getMajorMimeType(type);
+ var v = document.createElement(tag);
+ for (var i=0; i<gEventTypes.length; i++) {
+ v.addEventListener(gEventTypes[i], listener);
+ }
+ v.preload = "metadata";
+ v.src = src;
+ v.name = src;
+ document.body.appendChild(v);
+ v.eventNum = 0;
+ v.token = token;
+ setTimeout(
+ function() {
+ v.checkStateInterval = setInterval(function(){checkState(v);},1);
+ }, 0);
+}
+
+// Define our own ok() and is() functions. The mochitest ones take ages constructing the log
+// of all the passes, so only report failures.
+function test_ok(b, msg) {
+ if (!b) {
+ log("FAILED test_ok: " + msg);
+ ok(b, msg);
+ }
+}
+
+function test_is(a, b, msg) {
+ if (a != b) {
+ log("FAILED test_is: " + msg);
+ is(a,b,msg);
+ }
+}
+
+function filename(uri) {
+ return uri.substr(uri.lastIndexOf("/")+1);
+}
+
+function checkState(v) {
+ test_ok(v.networkState <= HTMLMediaElement.NETWORK_LOADING ||
+ v.networkState == HTMLMediaElement.NETWORK_NO_SOURCE,
+ "NetworkState of " + v.networkState + " was leaked.");
+ test_ok(v.readyState == HTMLMediaElement.HAVE_NOTHING,
+ "Ready state of " + v.readyState + " was leaked");
+ test_is(v.seeking, false, "Seeking leaked");
+ test_is(v.currentTime, 0, "Leaked currentTime");
+ test_ok(isNaN(v.duration), "Leaked duration");
+ test_is(v.paused, true, "Paused leaked");
+ test_is(v.ended, false, "Ended leaked");
+ test_is(v.autoplay, false, "Autoplay leaked");
+ test_is(v.controls, false, "Controls leaked");
+ test_is(v.muted, false, "muted leaked");
+ test_ok(v.error==null || v.error.code==MediaError.MEDIA_ERR_SRC_NOT_SUPPORTED,
+ "Error code should not exist or be SRC_NOT_SUPPORTED. v.error=" +
+ (v.error ? v.error.code : "null"));
+ test_ok(filename(v.currentSrc) == filename(v.name) ||
+ v.networkState == HTMLMediaElement.NETWORK_NO_SOURCE,
+ "currentSrc should match candidate uri, if we've got a valid source");
+}
+
+
+function startTest(test, token) {
+ manager.started(token);
+ log("Testing: " + test.type + " @ " + test.name);
+ createMedia(test.type, test.name, token);
+}
+
+SimpleTest.waitForExplicitFinish();
+createTestArray().then(testArray => {
+ manager.runTests(testArray, startTest);
+});
+
+</script>
+</pre>
+
+</body>
+</html>
diff --git a/dom/media/test/test_invalid_reject.html b/dom/media/test/test_invalid_reject.html
new file mode 100644
index 0000000000..583847fe12
--- /dev/null
+++ b/dom/media/test/test_invalid_reject.html
@@ -0,0 +1,58 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <meta charset="UTF-8" />
+ <title>Test rejection of invalid files</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function startTest(test, token) {
+ var v = document.createElement('video');
+ manager.started(token);
+
+ // Set up event handlers. Seeing any of these is a failure.
+ function badEvent(type) { return function(e) {
+ ok(false, test.name + " should not fire '" + type + "' event");
+ }};
+ var events = [
+ 'loadeddata', 'load',
+ 'canplay', 'canplaythrough',
+ 'playing'
+ ];
+ events.forEach( function(e) {
+ v.addEventListener(e, badEvent(e));
+ });
+
+ // Seeing a decoder error is a success.
+ v.addEventListener("error", function onerror(e) {
+ if (v.readyState == v.HAVE_NOTHING) {
+ is(v.error.code, v.error.MEDIA_ERR_SRC_NOT_SUPPORTED,
+ "decoder should reject " + test.name);
+ } else {
+ is(v.error.code, v.error.MEDIA_ERR_DECODE,
+ "decoder should reject " + test.name);
+ }
+ v.removeEventListener('error', onerror);
+ manager.finished(token);
+ });
+
+ // Now try to load and play the file, which should result in the
+ // error event handler above being called, terminating the test.
+ document.body.appendChild(v);
+ v.src = test.name;
+ v.play();
+}
+
+manager.runTests(gInvalidTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_invalid_reject_play.html b/dom/media/test/test_invalid_reject_play.html
new file mode 100644
index 0000000000..3e658f94b8
--- /dev/null
+++ b/dom/media/test/test_invalid_reject_play.html
@@ -0,0 +1,44 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <meta charset="UTF-8" />
+ <title>Test rejection of invalid files during playback</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function startTest(test, token) {
+ var v = document.createElement('video');
+ manager.started(token);
+
+ // Seeing a decoder error is a success.
+ v.addEventListener("error", function onerror(e) {
+ is(v.error.code, v.error.MEDIA_ERR_DECODE,
+ "decoder should reject " + test.name);
+ v.removeEventListener("error", onerror);
+ manager.finished(token);
+ });
+
+ v.addEventListener("ended", function onended(e) {
+ ok(false, "decoder should have rejected file before playback ended");
+ v.removeEventListener("ended", onended);
+ manager.finished(token);
+ });
+
+ document.body.appendChild(v);
+ v.src = test.name;
+ v.play();
+}
+
+manager.runTests(gInvalidPlayTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_invalid_seek.html b/dom/media/test/test_invalid_seek.html
new file mode 100644
index 0000000000..8aa514b977
--- /dev/null
+++ b/dom/media/test/test_invalid_seek.html
@@ -0,0 +1,32 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: invalide seek test</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<video id='v'></video>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+// http://www.whatwg.org/specs/web-apps/current-work/#dom-media-seek
+// If the media element's readyState is HAVE_NOTHING, then the user agent
+// must raise an InvalidStateError exception.
+var v = document.getElementById('v');
+var passed = false;
+
+ok(v.readyState == HTMLMediaElement.HAVE_NOTHING,
+ "Invalid ready state in media element");
+
+try {
+ v.seek(1);
+}
+catch(e) {
+ passed = true;
+}
+
+ok(passed, "Video did not raise error during invalid seek");
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_load.html b/dom/media/test/test_load.html
new file mode 100644
index 0000000000..de8fd63948
--- /dev/null
+++ b/dom/media/test/test_load.html
@@ -0,0 +1,217 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=479859
+-->
+<head>
+ <title>Test for Bug 479859</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="application/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=479859">Mozilla Bug 479859</a>
+<p id="display"></p>
+<div id="content" style="display: none">
+
+</div>
+<pre id="test">
+<script type="text/javascript">
+
+function log(msg) {
+ //document.getElementById('log').innerHTML += "<p>" + msg + "</p>";
+}
+
+// We don't track: progress, canplay, canplaythrough and stalled events,
+// as these can be delivered out of order, and/or multiple times.
+var gEventTypes = [ 'loadstart', 'abort', 'error', 'emptied', 'play',
+ 'pause', 'loadedmetadata', 'loadeddata', 'waiting', 'playing', 'seeking',
+ 'seeked', 'timeupdate', 'ended', 'ratechange', 'durationchange', 'volumechange' ];
+
+var gEventNum = 0;
+var gTestNum = 0;
+var gTestFileNum = 0;
+var gExpectedEvents = null;
+var gTest = null;
+var gTestName = "?";
+
+function listener(evt) {
+ log('event ' + evt.type);
+ ok(gEventNum < gExpectedEvents.length, gTestName+" - corrent number of events received");
+ var expected = (gEventNum < gExpectedEvents.length) ? gExpectedEvents[gEventNum] : "NoEvent";
+ is(evt.type, expected, gTestName+" - events received in order");
+ gEventNum++;
+ if (gEventNum == gExpectedEvents.length) {
+ setTimeout(nextTest, 0);
+ }
+}
+
+function source_error(evt) {
+ log('event source_error');
+ ok(evt.type == "error", "Should only get error events here");
+ ok(gEventNum < gExpectedEvents.length, gTestName+" - corrent number of events received");
+ var expected = (gEventNum < gExpectedEvents.length) ? gExpectedEvents[gEventNum] : "NoEvent";
+ is("source_error", expected, gTestName+" - events received in order");
+ gEventNum++;
+ if (gEventNum == gExpectedEvents.length) {
+ setTimeout(nextTest, 0);
+ }
+}
+
+var gMedia = null;
+
+function createMedia(tag) {
+ gMedia = document.createElement(tag);
+ gMedia.preload = "metadata";
+ for (var i=0; i<gEventTypes.length; i++) {
+ gMedia.addEventListener(gEventTypes[i], listener);
+ }
+}
+
+function addSource(src, type) {
+ var s = document.createElement("source");
+ s.addEventListener("error", source_error);
+ s.src = src;
+ s.type = type;
+ gMedia.appendChild(s);
+ return s;
+}
+
+function prependSource(src, type) {
+ var s = document.createElement("source");
+ s.addEventListener("error", source_error);
+ s.src = src;
+ s.type = type;
+ gMedia.insertBefore(s, gMedia.firstChild);
+ return s;
+}
+
+var gTests = [
+ {
+ // Test 0: adding video to doc, then setting src should load implicitly.
+ create(src, type) {
+ document.body.appendChild(gMedia);
+ gMedia.src = src;
+ },
+ expectedEvents: ['loadstart', 'durationchange', 'loadedmetadata', 'loadeddata']
+ }, {
+ // Test 1: adding video to doc, then adding source.
+ create(src, type) {
+ document.body.appendChild(gMedia);
+ addSource(src, type);
+ },
+ expectedEvents: ['loadstart', 'durationchange', 'loadedmetadata', 'loadeddata']
+ },{
+ // Test 2: video with multiple source, the first of which are bad, we should load the last,
+ // and receive error events for failed loads on the source children.
+ create(src, type) {
+ document.body.appendChild(gMedia);
+ addSource("404a", type);
+ addSource("404b", type);
+ addSource(src, type);
+ },
+ expectedEvents: ['loadstart', 'source_error', 'source_error', 'durationchange', 'loadedmetadata', 'loadeddata']
+ }, {
+ // Test 3: video with bad src, good <source>, ensure that <source> aren't used.
+ create(src, type) {
+ gMedia.src = "404a";
+ addSource(src, type);
+ document.body.appendChild(gMedia);
+ },
+ expectedEvents: ['loadstart', 'error']
+ }, {
+ // Test 4: video with only bad source, loading, then adding a good source
+ // - should resume load.
+ create(src, type) {
+ addSource("404a", type);
+ var s2 = addSource("404b", type);
+ s2.addEventListener("error",
+ function(e) {
+ // Should awaken waiting load, causing successful load.
+ addSource(src, type);
+ });
+ document.body.appendChild(gMedia);
+ },
+ expectedEvents: ['loadstart', 'source_error', 'source_error', 'durationchange', 'loadedmetadata', 'loadeddata']
+ }, {
+ // Test 5: video with only 1 bad source, let it fail to load, then prepend
+ // a good <source> to the video, it shouldn't be selected, because the
+ // "pointer" should be after the last child - the bad source.
+ prepended: false,
+ create(src, type) {
+ var prepended = false;
+ addSource("404a", type);
+ var s2 = addSource("404b", type);
+ s2.addEventListener("error",
+ function(e) {
+ // Should awaken waiting load, causing successful load.
+ if (!prepended) {
+ prependSource(src, type);
+ prepended = true;
+ }
+ });
+ document.body.appendChild(gMedia);
+ },
+ expectedEvents: ['loadstart', 'source_error', 'source_error']
+ }, {
+ // Test 6: (Bug 1165203) preload="none" then followed by an explicit
+ // call to load() should load metadata
+ create(src, type) {
+ gMedia.preload = "none";
+ gMedia.src = src;
+ document.body.appendChild(gMedia);
+ addSource(src, type);
+ gMedia.load();
+ },
+ expectedEvents: ['emptied', 'loadstart', 'durationchange', 'loadedmetadata', 'loadeddata']
+ }
+];
+
+function nextTest() {
+ if (gMedia) {
+ for (var i=0; i<gEventTypes.length; i++) {
+ gMedia.removeEventListener(gEventTypes[i], listener);
+ }
+ removeNodeAndSource(gMedia);
+ gMedia = null;
+ }
+ gEventNum = 0;
+
+ if (gTestNum == gTests.length) {
+ gTestNum = 0;
+ ++gTestFileNum;
+ if (gTestFileNum == gSmallTests.length) {
+ SimpleTest.finish();
+ return;
+ }
+ }
+
+ var src = gSmallTests[gTestFileNum].name;
+ var type = gSmallTests[gTestFileNum].type;
+
+ var t = gTests[gTestNum];
+ gTestNum++;
+
+ createMedia(type.match(/^audio\//) ? "audio" : "video");
+ if (!gMedia.canPlayType(type)) {
+ // Unsupported type, skip to next test
+ nextTest();
+ return;
+ }
+
+ gTestName = "Test " + src + " " + (gTestNum - 1);
+ log("Starting " + gTestName);
+ gExpectedEvents = t.expectedEvents;
+
+ t.create(src, type);
+}
+
+addLoadEvent(nextTest);
+SimpleTest.waitForExplicitFinish();
+
+</script>
+</pre>
+
+<div id="log" style="font-size: small"></div>
+</body>
+</html>
diff --git a/dom/media/test/test_load_candidates.html b/dom/media/test/test_load_candidates.html
new file mode 100644
index 0000000000..2bc1eb531b
--- /dev/null
+++ b/dom/media/test/test_load_candidates.html
@@ -0,0 +1,84 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=465458
+-->
+<head>
+ <title>Test for Bug 465458</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=465458">Mozilla Bug 465458</a>
+<p id="display"></p>
+<div id="content" style="display: none">
+
+</div>
+<pre id="test">
+<script type="application/javascript">
+
+/** Test for Bug 465458 **/
+
+var manager = new MediaTestManager;
+
+function finish(evt) {
+ var v = evt.target;
+ is(v._error, 2, "Should have received 2 error events before loaded");
+ v._finished = true;
+ // remove error event handler, because this would otherwise
+ // cause a failure on Windows 7, see bug 1024535
+ v.onerror = null;
+ v.remove();
+ manager.finished(v.token);
+}
+
+function errorHandler(evt) {
+ evt.target.parentNode._error++;
+}
+
+var extension = {
+ "audio/wav" : "wav",
+ "audio/x-wav": "wav",
+ "video/ogg" : "ogv",
+ "audio/ogg" : "oga",
+ "video/webm" : "webm"
+};
+
+function startTest(test, token) {
+ var v = document.createElement('video');
+ v.preload = "auto";
+ v.onerror = function(){ok(false,"Error events on source children should not bubble");}
+ v.token = token;
+ manager.started(token);
+ v._error = 0;
+ v._finished = false;
+ v._name = test.name;
+
+ var s1 = document.createElement("source");
+ s1.type = test.type;
+ s1.src = "404." + extension[test.type];
+ s1.addEventListener("error", errorHandler);
+ v.appendChild(s1);
+
+ var s2 = document.createElement("source");
+ s2.type = test.type;
+ s2.src = "test_load_candidates.html"; // definitely an invalid media file, regardless of its actual mime type...
+ s2.addEventListener("error", errorHandler);
+ v.appendChild(s2);
+
+ var s3 = document.createElement("source");
+ s3.type = test.type;
+ s3.src = test.name;
+ v.appendChild(s3);
+
+ v.addEventListener("loadeddata", finish);
+ document.body.appendChild(v);
+}
+
+manager.runTests(gSmallTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_load_same_resource.html b/dom/media/test/test_load_same_resource.html
new file mode 100644
index 0000000000..f3e6992e8c
--- /dev/null
+++ b/dom/media/test/test_load_same_resource.html
@@ -0,0 +1,106 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test loading of the same resource in multiple elements</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+SimpleTest.requestCompleteLog();
+var manager = new MediaTestManager;
+
+function checkDuration(actual, expected, name) {
+ ok(Math.abs(actual - expected) < 0.1,
+ `${name} duration: ${actual} expected: ${expected}`);
+}
+
+function cloneLoaded(event) {
+ var e = event.target;
+ ok(true, `${e.token} loaded OK`);
+ checkDuration(e.duration, e._expectedDuration, e.token);
+ removeNodeAndSource(e);
+ manager.finished(e.token);
+}
+
+function tryClone(e) {
+ var clone = e.cloneNode(false);
+ clone.token = `${e.token}(cloned)`;
+ manager.started(clone.token);
+ manager.finished(e.token);
+
+ // Log events for debugging.
+ var events = ["suspend", "play", "canplay", "canplaythrough", "loadstart", "loadedmetadata",
+ "loadeddata", "playing", "ended", "error", "stalled", "emptied", "abort",
+ "waiting", "pause"];
+ function logEvent(evt) {
+ var event = evt.target;
+ info(`${event.token} got ${evt.type}`);
+ }
+ events.forEach(function(event) {
+ clone.addEventListener(event, logEvent);
+ });
+
+
+ checkDuration(e.duration, e._expectedDuration, e.token);
+ clone._expectedDuration = e._expectedDuration;
+
+ clone.addEventListener("loadeddata", cloneLoaded, {once: true});
+ clone.addEventListener("loadstart", function(evt) {
+ info(`${evt.target.token} starts loading.`);
+ // Since there is only one H264 decoder instance, we have to delete the
+ // decoder of the original element for the cloned element to load. However,
+ // we can't delete the decoder too early otherwise cloning decoder will
+ // fail to kick in. We wait for 'loadstart' event of the cloned element to
+ // know when the decoder is already cloned and we can delete the decoder of
+ // the original element.
+ removeNodeAndSource(e);
+ }, {once: true});
+}
+
+// This test checks that loading the same URI twice in different elements at the same time
+// uses the same resource without doing another network fetch. One of the gCloneTests
+// uses dynamic_resource.sjs to return one resource on the first fetch and a different resource
+// on the second fetch. These resources have different lengths, so if the cloned element
+// does a network fetch it will get a resource with the wrong length and we get a test
+// failure.
+
+async function initTest(test, token) {
+ var e = document.createElement("video");
+ e.preload = "auto";
+ e.src = test.name;
+ e._expectedDuration = test.duration;
+ ok(true, `Trying to load ${test.name}, duration=${test.duration}`);
+ e.token = token;
+ manager.started(token);
+
+ // Since 320x240.ogv is less than 32KB, we need to wait for the
+ // 'suspend' event to ensure the partial block is flushed to the cache
+ // otherwise the cloned resource will create a new channel when it
+ // has no data to read at position 0. The new channel will download
+ // a different file than the original resource and fail the duration
+ // test.
+ let p1 = once(e, "loadeddata");
+ let p2 = once(e, "suspend");
+ await p1;
+ await p2;
+ tryClone(e);
+}
+
+SimpleTest.waitForExplicitFinish();
+SpecialPowers.pushPrefEnv(
+ {"set": [
+ ["logging.MediaDecoder", "Debug"],
+ ["logging.HTMLMediaElement", "Debug"],
+ ["logging.MediaCache", "Debug"],
+ ]},
+ manager.runTests.bind(manager, gCloneTests, initTest));
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_load_source.html b/dom/media/test/test_load_source.html
new file mode 100644
index 0000000000..95a925b61f
--- /dev/null
+++ b/dom/media/test/test_load_source.html
@@ -0,0 +1,76 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=534571
+-->
+<head>
+ <title>Test for Bug 534571</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=534571">Mozilla Bug 534571</a>
+<p id="display"></p>
+<div id="content" style="display: none">
+
+</div>
+<pre id="test">
+<script type="application/javascript">
+
+/** Test for Bug 534571 **/
+
+// Test that when we load a video from a source child and then change the
+// source's src attribute and load again, that the subsequent loads work.
+
+var v = null;
+var s = null;
+
+function finish(event) {
+ ok(true, "Should have played both videos");
+ SimpleTest.finish();
+}
+
+var first = null;
+var second = null;
+
+function ended(event) {
+ s.type = second.type;
+ s.src = second.name;
+ v.removeEventListener("ended", ended);
+ v.addEventListener("ended", finish);
+ v.load();
+}
+
+// Find 2 videos we can play.
+v = document.createElement('video');
+for (var i=0; i<gPlayTests.length; i++) {
+ if (!v.canPlayType(gPlayTests[i].type))
+ continue;
+ if (!first) {
+ first = gPlayTests[i];
+ } else if (!second) {
+ second = gPlayTests[i];
+ break;
+ }
+}
+
+if (first && second) {
+ s = document.createElement('source');
+ s.type = first.type;
+ s.src = first.name;
+ v.appendChild(s);
+ v.autoplay = true;
+ v.addEventListener("ended", ended);
+ document.body.appendChild(v);
+ SimpleTest.waitForExplicitFinish();
+} else {
+ todo(false, "Need at least two media of supported types for this test!");
+}
+
+
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_load_source_empty_type.html b/dom/media/test/test_load_source_empty_type.html
new file mode 100644
index 0000000000..9ceee9af46
--- /dev/null
+++ b/dom/media/test/test_load_source_empty_type.html
@@ -0,0 +1,36 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Load resource from source element with empty type</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script src="manifest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+</head>
+<body>
+<video id="type"><source src="gizmo.mp4" type></video>
+<video id="type-and-equal"><source src="gizmo.mp4" type=></video>
+<video id="type-and-equal-quotation-marks"><source src="gizmo.mp4" type=""></video>
+<script class="testbody" type="text/javascript">
+/**
+ * This test is used to ensure that media element can start loading when its
+ * child source element with empty type property. We would test following forms,
+ * `type`, `type=` and `type=""`.
+ */
+SimpleTest.waitForExplicitFinish();
+
+const videos = document.getElementsByTagName("video");
+const videoNum = videos.length;
+let loadedElementsNum = 0;
+
+for (let video of videos) {
+ video.addEventListener("loadeddata",
+ () => {
+ ok(true, `loaded element '${video.id}'`);
+ if (++loadedElementsNum == videoNum) {
+ SimpleTest.finish();
+ }
+ }, { once: true});
+}
+</script>
+</body>
+</html>
diff --git a/dom/media/test/test_loop.html b/dom/media/test/test_loop.html
new file mode 100644
index 0000000000..943059edad
--- /dev/null
+++ b/dom/media/test/test_loop.html
@@ -0,0 +1,57 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test looping support</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function startTest(test, token) {
+ manager.started(token);
+ var v = document.createElement('video');
+ v.token = token;
+ v.src = test.name;
+ v.name = test.name;
+ v.playCount = 0;
+ v.seekingCount = 0;
+ v.seekedCount = 0;
+ v.loop = true;
+
+ v.addEventListener("play", function (e) {
+ e.target.playCount += 1;
+ ok(e.target.playCount == 1, "Should get exactly one play event.");
+ });
+
+ v.addEventListener("seeking", function (e) {
+ e.target.seekingCount += 1;
+ });
+
+ v.addEventListener("seeked", function (e) {
+ e.target.seekedCount += 1;
+ if (e.target.seekedCount == 2) {
+ ok(e.target.seekingCount == 2, "Expect matched pairs of seeking/seeked events.");
+ e.target.loop = false;
+ }
+ });
+
+ v.addEventListener("ended", function (e) {
+ ok(!e.target.loop, "Shouldn't get ended event while looping.");
+ removeNodeAndSource(e.target);
+ manager.finished(e.target.token);
+ });
+
+ document.body.appendChild(v);
+ v.play();
+}
+
+manager.runTests(gSmallTests, startTest);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_looping_eventsOrder.html b/dom/media/test/test_looping_eventsOrder.html
new file mode 100644
index 0000000000..7d070de72f
--- /dev/null
+++ b/dom/media/test/test_looping_eventsOrder.html
@@ -0,0 +1,52 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Looping events order</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<script type="text/javascript">
+/**
+ * This test is used to ensure the events order when media is looping back to
+ * the start position. We should see events in following order.
+ * 'seeking' -> 'timeupdate' -> 'seeked'.
+ */
+SimpleTest.waitForExplicitFinish();
+
+var tests = [
+ { name:"small-shot.ogg", type:"audio/ogg" },
+ { name:"seek-short.webm", type:"video/webm" }
+];
+
+async function testTimeupdateChanged({name, type}) {
+ info(`- start testPlay for name=${name} -`);
+ const element = document.createElement(getMajorMimeType(type));
+ element.src = name;
+ element.loop = true;
+
+ await once(element, "canplay");
+ ok(await element.play().then(() => true, () => false), `start playing ${name}`);
+
+ let gotTimeUpdated = false;
+ await once(element, "seeking");
+ element.addEventListener("timeupdate", function() {
+ gotTimeUpdated = true;
+ }, {once: true});
+ await once(element, "seeked");
+ ok(gotTimeUpdated, "Got timeupdate between seeking and seeked.");
+
+ removeNodeAndSource(element);
+}
+
+(async function startTest() {
+ for (let test of tests) {
+ await testTimeupdateChanged(test);
+ }
+ SimpleTest.finish();
+})();
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_media_selection.html b/dom/media/test/test_media_selection.html
new file mode 100644
index 0000000000..42f5a9bd43
--- /dev/null
+++ b/dom/media/test/test_media_selection.html
@@ -0,0 +1,142 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: media selection</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="application/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function maketest(attach_media, name, type, check_metadata) {
+ return function (token) {
+ var e = document.createElement('video');
+ e.preload = "metadata";
+ token = name + "-" + token;
+ manager.started(token);
+ var errorRun = false;
+ if (check_metadata) {
+ e.addEventListener('loadedmetadata', function () {
+ ok(e.readyState >= HTMLMediaElement.HAVE_METADATA,
+ 'test ' + token + ' readyState ' + e.readyState + ' expected >= ' + HTMLMediaElement.HAVE_METADATA);
+ is(e.currentSrc.substring(e.currentSrc.length - name.length), name, 'test ' + token);
+ // The load can go idle due to cache size limits
+ ok(e.networkState >= HTMLMediaElement.NETWORK_IDLE,
+ 'test ' + token + ' networkState = ' + e.networkState + ' expected >= ' + HTMLMediaElement.NETWORK_IDLE);
+ check_metadata(e);
+ removeNodeAndSource(e);
+ manager.finished(token);
+ });
+ } else {
+ e.addEventListener('error', function onerror(event) {
+ is(errorRun, false, "error handler should run once only!");
+ errorRun = true;
+ is(e.readyState, HTMLMediaElement.HAVE_NOTHING,
+ 'test ' + token + ' readyState should be HAVE_NOTHING when load fails.');
+ e.removeEventListener('error', onerror, true);
+ removeNodeAndSource(e);
+ manager.finished(token);
+ }, true);
+ }
+ attach_media(e, name, type);
+ }
+}
+
+function set_src(element, name, type) {
+ element.src = name;
+ document.body.appendChild(element);
+}
+
+function add_source(element, name, type) {
+ do_add_source(element, name, type);
+ document.body.appendChild(element);
+}
+
+function do_add_source(element, name, type) {
+ var source = document.createElement('source');
+ if (type) {
+ source.type = type;
+ }
+ source.src = name;
+ element.appendChild(source);
+}
+
+function add_sources_last(element, name, type) {
+ do_add_source(element, name, 'unsupported/type');
+ do_add_source(element, name, type);
+ document.body.appendChild(element);
+}
+
+function add_sources_first(element, name, type) {
+ do_add_source(element, name, type);
+ do_add_source(element, name, 'unsupported/type');
+ document.body.appendChild(element);
+}
+
+function late_add_sources_last(element, name, type) {
+ document.body.appendChild(element);
+ do_add_source(element, name, 'unsupported/type');
+ do_add_source(element, name, type);
+}
+
+function late_add_sources_first(element, name, type) {
+ document.body.appendChild(element);
+ do_add_source(element, name, type);
+ do_add_source(element, name, 'unsupported/type');
+}
+
+var nextTest = 0;
+var subtests = [
+ maketest(add_source, 'unknown.raw', 'bogus/type', null)
+];
+
+var tmpVid = document.createElement('video');
+
+for (var i = 0; i < gSmallTests.length; ++i) {
+ var test = gSmallTests[i];
+ var src = test.name;
+ var type = test.type;
+
+ if (!tmpVid.canPlayType(type))
+ continue;
+
+ // The following nested function hack is to ensure that 'test' is correctly
+ // captured in the closure and we don't end up getting the value 'test'
+ // had in the last iteration of the loop. I blame Brendan.
+ var check = function(t) { return function (e) {
+ checkMetadata(t.name, e, t);
+ }}(test);
+
+ var otherType = type.match(/^video\//) ? "audio/x-wav" : "video/ogg";
+ subtests.push(maketest(set_src, src, null, check),
+ maketest(add_source, src, null, check),
+ maketest(add_source, src, type, check),
+ maketest(add_sources_last, src, null, check),
+ maketest(add_sources_first, src, type, check),
+
+ // type hint matches a decoder, actual type matches different decoder
+ maketest(add_source, src, otherType, check),
+ maketest(add_source, 'unknown.raw', type, null),
+
+ // should not start loading, type excludes it from media candiate list
+ maketest(add_source, src, 'bogus/type', null),
+
+ // element doesn't notice source children attached later, needs bug 462455 fixed
+ maketest(late_add_sources_last, src, type, check),
+ maketest(late_add_sources_first, src, type, check));
+}
+
+function startTest(t, token) {
+ t(token);
+}
+
+manager.runTests(subtests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_media_sniffer.html b/dom/media/test/test_media_sniffer.html
new file mode 100644
index 0000000000..68e2bba8af
--- /dev/null
+++ b/dom/media/test/test_media_sniffer.html
@@ -0,0 +1,67 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: sniffing</title>
+ <meta charset='utf-8'>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var manager = new MediaTestManager;
+
+function finish_test(element) {
+ element.removeEventListener("error", onerror);
+ removeNodeAndSource(element);
+ manager.finished(element.token);
+}
+
+function onmetadataloaded(e) {
+ var t = e.target;
+ ++t.srcIndex;
+ ok(true, "The media loads when loaded via " + t.src);
+ if (t.srcIndex < t.srcList.length) {
+ t.src = t.srcList[t.srcIndex];
+ } else {
+ finish_test(t);
+ }
+}
+
+function onerror(e) {
+ var t = e.target;
+ t.removeEventListener('error', onerror);
+ ok(false, "The media could not be loaded." + t.src + "\n");
+ finish_test(t);
+}
+
+function startTest(test, token) {
+ var elemType = /^audio/.test(test.type) ? "audio" : "video";
+ var element = document.createElement(elemType);
+ // This .sjs file serve the media file without Content-Type header, or with a
+ // specific Content-Type header.
+ var baseSrc = 'contentType.sjs?file=' + test.name;
+ element.srcList = [
+ baseSrc + "&nomime",
+ baseSrc + "&type=application/octet-stream",
+ baseSrc + "&type=audio/wav",
+ baseSrc + "&type=text/html",
+ baseSrc + "&type=absolute_nonsense"
+ ];
+ element.srcIndex = 0;
+ element.src = element.srcList[element.srcIndex];
+ element.token = token;
+ element.controls = true;
+ element.preload = "metadata";
+ document.body.appendChild(element);
+ manager.started(token);
+ element.addEventListener("loadedmetadata", onmetadataloaded);
+ element.addEventListener("error", onerror);
+}
+
+manager.runTests(gSnifferTests, startTest);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediacapabilities_resistfingerprinting.html b/dom/media/test/test_mediacapabilities_resistfingerprinting.html
new file mode 100644
index 0000000000..1f07d1c707
--- /dev/null
+++ b/dom/media/test/test_mediacapabilities_resistfingerprinting.html
@@ -0,0 +1,69 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <meta charset="utf-8">
+ <title>Test for Bug 1369309</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script type="application/javascript" src="manifest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=1461454">Mozilla Bug 1461454</a>
+<a target="_blank" href="https://gitlab.torproject.org/tpo/applications/tor-browser/-/issues/13543">Tor Issue 13543</a>
+
+<script type="application/javascript">
+async function testWhetherSpoofed(resistFingerprinting) {
+
+ var unsupportedVideoConfiguration = {
+ contentType: 'video/bogus',
+ width: 800,
+ height: 600,
+ bitrate: 3000,
+ framerate: 24,
+ };
+ var supportedVideoConfiguration = {
+ contentType: 'video/webm; codecs="vp09.00.10.08"',
+ width: 800,
+ height: 600,
+ bitrate: 3000,
+ framerate: 24,
+ };
+
+ await SpecialPowers.pushPrefEnv({
+ "set": [
+ ["privacy.resistFingerprinting", resistFingerprinting]
+ ],
+ });
+
+ var result;
+
+ result = await navigator.mediaCapabilities.decodingInfo({
+ type: 'file',
+ video: unsupportedVideoConfiguration
+ });
+ is(result.supported, false, "video/bogus should be unsupported.");
+ is(result.smooth, false, "smooth is false when unsupported.");
+ is(result.powerEfficient, false, "powerEfficient is false when unsupported.");
+
+ result = await navigator.mediaCapabilities.decodingInfo({
+ type: 'file',
+ video: supportedVideoConfiguration
+ });
+ is(result.supported, true, "'video/webm; codecs=\"vp09.00.10.08\"' should be supported.");
+ if (resistFingerprinting) {
+ is(result.smooth, true, "smooth should be spoofed to true in RFP mode.");
+ is(result.powerEfficient, false, "powerEfficient should be spoofed to false in RFP mode.");
+ }
+}
+
+async function start() {
+ await testWhetherSpoofed(true);
+ await testWhetherSpoofed(false);
+ SimpleTest.finish();
+}
+
+SimpleTest.waitForExplicitFinish();
+start();
+</script>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_avoid_recursion.html b/dom/media/test/test_mediarecorder_avoid_recursion.html
new file mode 100644
index 0000000000..0c733b9fc2
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_avoid_recursion.html
@@ -0,0 +1,61 @@
+<html>
+<head>
+ <title>MediaRecorder infinite recursion with requestData() calls in "dataavailable" event</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="gUM_support.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=897776">Mozill
+a Bug 897776</a>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+async function startTest() {
+ try {
+ await setupGetUserMediaTestPrefs();
+ let stream = await navigator.mediaDevices.getUserMedia({audio: true});
+ let mediaRecorder = new MediaRecorder(stream);
+ let count = 0;
+ mediaRecorder.onerror = function () {
+ ok(false, 'Unexpected onerror callback fired');
+ SimpleTest.finish();
+ };
+ mediaRecorder.onwarning = function () {
+ ok(false, 'Unexpected onwarning callback fired');
+ };
+ mediaRecorder.ondataavailable = function (e) {
+ ++count;
+ info("got ondataavailable data size = " + e.data.size);
+ // no more requestData() to prevent busy main thread from starving
+ // the encoding thread
+ if (count == 30) {
+ info("track.stop");
+ stream.getTracks()[0].stop();
+ } else if (count < 30 && mediaRecorder.state == 'recording') {
+ info("requestData again");
+ mediaRecorder.requestData();
+ }
+ };
+ mediaRecorder.onstop = function () {
+ ok(true, "requestData within ondataavailable successfully avoided infinite recursion");
+ SimpleTest.finish();
+ };
+ mediaRecorder.start();
+ info("mediaRecorder start");
+ mediaRecorder.requestData();
+ info("mediaRecorder requestData");
+ } catch (e) {
+ ok(false, 'Unexpected error fired with: ' + e);
+ SimpleTest.finish();
+ }
+}
+
+SimpleTest.waitForExplicitFinish();
+startTest();
+
+</script>
+</pre>
+</body>
+</html>
+
diff --git a/dom/media/test/test_mediarecorder_bitrate.html b/dom/media/test/test_mediarecorder_bitrate.html
new file mode 100644
index 0000000000..693b27e3bf
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_bitrate.html
@@ -0,0 +1,127 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test MediaRecorder Bitrate</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var manager = new MediaTestManager;
+var results = [];
+
+/**
+ * Starts a test on every media recorder file included to check that
+ * the bitrate control works
+ */
+function startTest(test, token) {
+ manager.started(token);
+ runTest(test, token, 1000000);
+ runTest(test, token, 100000);
+}
+
+function runTest(test, token, bitrate) {
+ var element = document.createElement('video');
+
+ element.token = token;
+
+ element.src = test.name;
+ element.preload = "metadata";
+ element.onloadedmetadata = function () {
+ info("loadedmetadata");
+ const stream = element.mozCaptureStreamUntilEnded();
+ element.onloadedmetadata = null;
+ element.play();
+
+ const mediaRecorder = new MediaRecorder(stream, {videoBitsPerSecond: bitrate});
+ mediaRecorder.start();
+ is(mediaRecorder.state, 'recording', 'Media recorder should be recording');
+ is(mediaRecorder.stream, stream,
+ 'Media recorder stream = element stream at the start of recording');
+
+ var onStopFired = false;
+ var onDataAvailableFired = false;
+ var encoded_size = 0;
+
+ mediaRecorder.onerror = function () {
+ ok(false, 'Unexpected onerror callback fired');
+ };
+
+ mediaRecorder.onwarning = function () {
+ ok(false, 'Unexpected onwarning callback fired');
+ };
+
+ // This handler verifies that only a single onstop event handler is fired.
+ mediaRecorder.onstop = function () {
+ if (onStopFired) {
+ ok(false, 'onstop unexpectedly fired more than once');
+ } else {
+ onStopFired = true;
+
+ // ondataavailable should always fire before onstop
+ if (onDataAvailableFired) {
+ ok(true, 'onstop fired after ondataavailable');
+ info("test " + test.name + " encoded@" + bitrate + "=" + encoded_size);
+ if (results[test.name]) {
+ var big, small, temp;
+ big = {};
+ big.bitrate = bitrate;
+ big.size = encoded_size;
+ small = results[test.name];
+ // Don't assume the order that these will finish in
+ if (results[test.name].bitrate > bitrate) {
+ temp = big;
+ big = small;
+ small = temp;
+ }
+ // Ensure there is a big enough difference in the encoded
+ // sizes
+ ok(small.size*1.25 < big.size,
+ test.name + ' encoded@' + big.bitrate + '=' + big.size +
+ ' > encoded@' + small.bitrate + '=' + small.size);
+ manager.finished(token);
+ } else {
+ results[test.name] = {};
+ results[test.name].bitrate = bitrate;
+ results[test.name].size = encoded_size;
+ }
+ } else {
+ ok(false, 'onstop fired without an ondataavailable event first');
+ }
+ }
+ };
+
+ // This handler verifies that only a single ondataavailable event handler
+ // is fired with the blob generated having greater than zero size
+ // and a correct mime type.
+ mediaRecorder.ondataavailable = function (evt) {
+ if (onDataAvailableFired) {
+ ok(false, 'ondataavailable unexpectedly fired more than once');
+ } else {
+ onDataAvailableFired = true;
+
+ ok(evt instanceof BlobEvent,
+ 'Events fired from ondataavailable should be BlobEvent');
+ is(evt.type, 'dataavailable',
+ 'Event type should dataavailable');
+ ok(evt.data.size > 0,
+ 'Blob data received should be greater than zero');
+ encoded_size = evt.data.size;
+
+ // onstop should not have fired before ondataavailable
+ if (onStopFired) {
+ ok(false, 'ondataavailable unexpectedly fired later than onstop');
+ manager.finished(token);
+ }
+ }
+ };
+ };
+}
+
+manager.runTests(gMediaRecorderVideoTests, startTest);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_creation.html b/dom/media/test/test_mediarecorder_creation.html
new file mode 100644
index 0000000000..cd23cb7a96
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_creation.html
@@ -0,0 +1,45 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test MediaRecorder Creation</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var manager = new MediaTestManager;
+
+/**
+ * Starts a test on every media recorder file included to check that
+ * a media recorder object created with a stream derived from a media
+ * element with that file produces the correct starting attribute values.
+ */
+function startTest(test, token) {
+ var element = document.createElement('audio');
+
+ element.token = token;
+ manager.started(token);
+
+ element.src = test.name;
+ element.test = test;
+ element.stream = element.mozCaptureStreamUntilEnded();
+
+ var mediaRecorder = new MediaRecorder(element.stream);
+
+ is(mediaRecorder.stream, element.stream,
+ 'Stream should be provided stream on creation');
+ is(mediaRecorder.mimeType, '',
+ 'mimeType should be an empty string on creation');
+ is(mediaRecorder.state, 'inactive',
+ 'state should be inactive on creation');
+
+ manager.finished(token);
+}
+
+manager.runTests(gMediaRecorderTests, startTest);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_creation_fail.html b/dom/media/test/test_mediarecorder_creation_fail.html
new file mode 100644
index 0000000000..2630f069b0
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_creation_fail.html
@@ -0,0 +1,61 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test MediaRecorder Record with media.ogg.enabled = false</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+function testThrows(stream, options, errorName, message, testName) {
+ try {
+ new MediaRecorder(stream, options);
+ } catch(e) {
+ is(e.name, errorName, testName);
+ is(e.message, message, testName);
+ return;
+ }
+ ok(!errorName, testName);
+}
+(async () => {
+ SimpleTest.waitForExplicitFinish();
+ await SpecialPowers.pushPrefEnv({set: [
+ ["media.ogg.enabled", false],
+ ["media.encoder.webm.enabled", false],
+ ]});
+ const stream = new AudioContext().createMediaStreamDestination().stream;
+ testThrows(
+ stream, {mimeType: "audio/ogg"}, "NotSupportedError",
+ "MediaRecorder constructor: audio/ogg indicates an unsupported container",
+ "Creating an ogg recorder without ogg support throws");
+ testThrows(
+ stream, {mimeType: "audio/webm"}, "NotSupportedError",
+ "MediaRecorder constructor: audio/webm indicates an unsupported container",
+ "Creating a webm recorder without webm support throws");
+ testThrows(
+ stream, {mimeType: "video/webm"}, "NotSupportedError",
+ "MediaRecorder constructor: video/webm indicates an unsupported container",
+ "Creating a webm recorder without webm support throws");
+ testThrows(
+ stream, {mimeType: "apa/bepa"}, "NotSupportedError",
+ "MediaRecorder constructor: apa/bepa is not a valid media type",
+ "Creating a recorder for a bogus mime type throws");
+ testThrows(
+ stream, {}, null, null,
+ "Creating a default recorder never throws, even without container support");
+ const recorder = new MediaRecorder(stream);
+ try {
+ recorder.start();
+ ok(false, "Starting a recorder without container support should throw");
+ } catch(e) {
+ is(e.name, "NotSupportedError",
+ "Starting a recorder without container support throws");
+ }
+ SimpleTest.finish();
+})();
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_fires_start_event_once_when_erroring.html b/dom/media/test/test_mediarecorder_fires_start_event_once_when_erroring.html
new file mode 100644
index 0000000000..537e1dbb47
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_fires_start_event_once_when_erroring.html
@@ -0,0 +1,45 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Bug 1395022 - MediaRecorder fires start event when erroring.</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=1395022">Mozilla Bug 1395022</a>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+function startTest() {
+ let audioContext = new AudioContext();
+ let destination1 = audioContext.createMediaStreamDestination();
+ let oscilator = audioContext.createOscillator();
+ oscilator.connect(destination1);
+ oscilator.start();
+
+ let destination2 = audioContext.createMediaStreamDestination();
+
+ let rec = new MediaRecorder(destination1.stream);
+
+ let numStartEvents = 0;
+
+ rec.onstart = () => {
+ numStartEvents += 1;
+ is(numStartEvents, 1, "One start event should be fired by the recorder");
+ // Trigger an error in the recorder
+ destination1.stream.addTrack(destination2.stream.getTracks()[0]);
+ };
+
+ rec.onerror = () => {
+ is(numStartEvents, 1, "One start event should have been fired by the recorder");
+ SimpleTest.finish();
+ };
+
+ rec.start();
+}
+
+SimpleTest.waitForExplicitFinish();
+startTest();
+
+</script>
+</head>
+</html> \ No newline at end of file
diff --git a/dom/media/test/test_mediarecorder_multipletracks.html b/dom/media/test/test_mediarecorder_multipletracks.html
new file mode 100644
index 0000000000..40fbfb92ce
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_multipletracks.html
@@ -0,0 +1,68 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test that MediaRecorder fires an error event when started with >1 track of a kind</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script src="/tests/dom/media/webrtc/tests/mochitests/head.js"></script>
+</head>
+<body>
+<script class="testbody">
+"use strict";
+
+(async () => {
+ SimpleTest.waitForExplicitFinish();
+ SimpleTest.requestFlakyTimeout("Failure timeouts");
+
+ try {
+ const stream = new MediaStream;
+ const recorder = new MediaRecorder(stream);
+
+ {
+ info("Testing with two audio tracks");
+ const ac = new AudioContext;
+ const track = ac.createMediaStreamDestination().stream.getTracks()[0];
+ stream.addTrack(track);
+ stream.addTrack(track.clone());
+ mustThrowWith("Starting with multiple audio tracks",
+ "NotSupportedError",
+ () => recorder.start());
+ await Promise.all([
+ haveNoEvent(recorder, "start"),
+ haveNoEvent(recorder, "error"),
+ haveNoEvent(recorder, "stop"),
+ ]);
+ for (let t of stream.getTracks()) {
+ stream.removeTrack(t);
+ t.stop();
+ }
+ }
+
+ {
+ info("Testing with two video tracks");
+ const canvas = document.createElement("canvas");
+ canvas.getContext("2d");
+ const track = canvas.captureStream().getTracks()[0];
+ stream.addTrack(track);
+ stream.addTrack(track.clone());
+ mustThrowWith("Starting with multiple video tracks",
+ "NotSupportedError",
+ () => recorder.start());
+ await Promise.all([
+ haveNoEvent(recorder, "start"),
+ haveNoEvent(recorder, "error"),
+ haveNoEvent(recorder, "stop"),
+ ]);
+ for (let t of stream.getTracks()) {
+ stream.removeTrack(t);
+ t.stop();
+ }
+ }
+ } catch(e) {
+ ok(false, `${e}\n${e.stack}`);
+ } finally {
+ SimpleTest.finish();
+ }
+})();
+</script>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_onerror_pause.html b/dom/media/test/test_mediarecorder_onerror_pause.html
new file mode 100644
index 0000000000..55ee5fb535
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_onerror_pause.html
@@ -0,0 +1,107 @@
+<html>
+<head>
+ <title>Bug 957439 - Media Recording - Assertion fail at Pause if unsupported input stream.</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="gUM_support.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=957439">Mozilla Bug 957439</a>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+function unexpected(e) {
+ ok(false, `Got unexpected ${e.type} event`);
+}
+
+
+async function startTest() {
+ // also do general checks on mimetype support for audio-only
+ ok(MediaRecorder.isTypeSupported("audio/ogg"),
+ 'Should support audio/ogg');
+ ok(MediaRecorder.isTypeSupported('audio/ogg; codecs=opus'),
+ 'Should support audio/ogg+opus');
+ ok(!MediaRecorder.isTypeSupported('audio/ogg; codecs=foobar'),
+ 'Should not support audio/ogg + unknown_codec');
+ ok(MediaRecorder.isTypeSupported("video/webm"),
+ 'Should support video/webm');
+ ok(!MediaRecorder.isTypeSupported("video/mp4"),
+ 'Should not support video/mp4');
+
+ try {
+ await setupGetUserMediaTestPrefs();
+ const expectedMimeType = 'video/webm; codecs="vp8, opus"';
+ const stream = await navigator.mediaDevices.getUserMedia({audio: true, video: true});
+ const [audioTrack] = stream.getAudioTracks();
+
+ // Expected event sequence should be:
+ // 1. start
+ // 2. error (from removed track)
+ // 3. dataavailable
+ // 4. stop
+ const mediaRecorder = new MediaRecorder(stream);
+ is(mediaRecorder.stream, stream, 'Stream should be provided on creation');
+
+ mediaRecorder.onstart = unexpected;
+ mediaRecorder.onerror = unexpected;
+ mediaRecorder.ondataavailable = unexpected;
+ mediaRecorder.onstop = unexpected;
+
+ mediaRecorder.start();
+ is(mediaRecorder.state, 'recording', 'state should be recording');
+ is(mediaRecorder.mimeType, '', 'mimetype should be unset');
+
+ await new Promise(r => mediaRecorder.onstart = r);
+ mediaRecorder.onstart = unexpected;
+ ok(true, 'start event fired');
+ is(mediaRecorder.mimeType, expectedMimeType, 'mimetype should be set');
+
+ // Trigger an error
+ stream.removeTrack(audioTrack);
+
+ const err = await new Promise(r => mediaRecorder.onerror = r);
+ mediaRecorder.onerror = unexpected;
+ ok(true, 'error event fired');
+ is(err.error.name, 'InvalidModificationError',
+ 'Error name should be InvalidModificationError.');
+ ok(err.error.stack.includes('test_mediarecorder_onerror_pause.html'),
+ 'Events fired from onerror should include an error with a stack trace indicating ' +
+ 'an error in this test');
+ is(mediaRecorder.mimeType, '', 'mimetype should be unset');
+ is(mediaRecorder.state, 'inactive', 'state is inactive');
+
+ try {
+ mediaRecorder.pause();
+ ok(false, 'pause should fire an exception if called on an inactive recorder');
+ } catch(e) {
+ ok(e instanceof DOMException, 'pause should fire an exception ' +
+ 'if called on an inactive recorder');
+ is(e.name, 'InvalidStateError', 'Exception name should be InvalidStateError');
+ }
+
+ const evt = await new Promise(r => mediaRecorder.ondataavailable = r);
+ mediaRecorder.ondataavailable = unexpected;
+ ok(true, 'dataavailable event fired');
+ isnot(evt.data.size, 0, 'data size should not be zero');
+ ok(evt instanceof BlobEvent,
+ 'Events fired from ondataavailable should be BlobEvent');
+ is(evt.data.type, expectedMimeType, 'blob mimeType is set');
+
+ await new Promise(r => mediaRecorder.onstop = r);
+ mediaRecorder.onstop = unexpected;
+ ok(true, 'onstop event fired');
+ is(mediaRecorder.state, 'inactive', 'state should be inactive');
+ } catch (err) {
+ ok(false, `Unexpected error fired with: ${err}`);
+ } finally {
+ SimpleTest.finish();
+ }
+}
+
+SimpleTest.waitForExplicitFinish();
+window.onload = startTest;
+
+</script>
+</head>
+</html>
diff --git a/dom/media/test/test_mediarecorder_pause_resume_video.html b/dom/media/test/test_mediarecorder_pause_resume_video.html
new file mode 100644
index 0000000000..a9b3d6a034
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_pause_resume_video.html
@@ -0,0 +1,130 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test MediaRecorder Recording doesn't record during pause</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script src="/tests/dom/canvas/test/captureStream_common.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<pre id="test">
+<div id="content">
+ <canvas id="video-src-canvas"></canvas>
+ <video id="recorded-video"></video>
+</div>
+<script class="testbody" type="text/javascript">
+
+function startTest() {
+ // Setup canvas and take a stream from it
+ let canvas = document.getElementById("video-src-canvas");
+
+ let canvas_size = 100;
+ let new_canvas_size = 50;
+
+ canvas.width = canvas.height = canvas_size;
+
+ let helper = new CaptureStreamTestHelper2D(100, 100);
+ helper.drawColor(canvas, helper.red);
+
+ let canvasStream = canvas.captureStream();
+ // Canvas set up
+
+ // Check values for events
+ let numDataAvailabledRaised = 0;
+ // Recorded data that will be playback.
+ let blob;
+
+ let mediaRecorder = new MediaRecorder(canvasStream);
+ is(mediaRecorder.stream, canvasStream,
+ "Media recorder stream = canvas stream at the start of recording");
+
+ mediaRecorder.onwarning = () => ok(false, "warning unexpectedly fired");
+
+ mediaRecorder.onerror = () => ok(false, "Recording failed");
+
+ mediaRecorder.ondataavailable = ev => {
+ info("Got 'dataavailable' event");
+ ++numDataAvailabledRaised;
+ // Save recorded data for playback
+ blob = ev.data;
+ };
+
+ mediaRecorder.onstart = () => {
+ info("Got 'start' event");
+ // We just want one frame encoded before we pause
+ mediaRecorder.pause();
+ // We may rewrite this once we settle Bug 1363915, could listen for pause event instead
+ is(mediaRecorder.state, 'paused', 'Media recorder should be paused');
+
+ // Wait a while, then while paused draw blue at another size, then again
+ // green at the original size and resume.
+ let numberOfPaintsSincePause = 0;
+ window.requestAnimationFrame(function draw() {
+ numberOfPaintsSincePause++;
+ if (numberOfPaintsSincePause == 8) {
+ canvas.width = canvas.height = new_canvas_size;
+ helper.drawColor(canvas, helper.blue);
+ } else if (numberOfPaintsSincePause == 60) {
+ canvas.width = canvas.height = canvas_size;
+ helper.drawColor(canvas, helper.green);
+ } else if (numberOfPaintsSincePause == 68) {
+ // Waited 8 draws since changing canvas to green, should be safe to resume
+ mediaRecorder.resume();
+ } else if (numberOfPaintsSincePause > 120) {
+ mediaRecorder.stop();
+ return; // Early return, we don't want to request any more animation frames
+ }
+ window.requestAnimationFrame(draw);
+ });
+ };
+
+ mediaRecorder.onstop = () => {
+ info("Got 'stop' event");
+ is(mediaRecorder.state, 'inactive', 'Media recorder should be incative after stop');
+ is(numDataAvailabledRaised, 1, "Expected 1 dataavailable event");
+
+ ok(blob, "Should have gotten a data blob");
+ let video = document.getElementById("recorded-video");
+ video.id = "recorded-video";
+ video.src = URL.createObjectURL(blob);
+ // Setup a check to make sure we don't play back any blue
+ let checkVideoHasNoBlue = () => {
+ if(helper.isPixel(helper.getPixel(video), helper.blue, 128)) {
+ ok(false, "Video should have no blue frames");
+ // Remove handler so we don't spam the log
+ video.ontimeupdate = null;
+ }
+ };
+ video.ontimeupdate = checkVideoHasNoBlue;
+ video.onerror = () => {
+ ok(false, "Should be able to play the recording. Got error. code=" + video.error.code);
+ SimpleTest.finish();
+ };
+ video.onended = () => {
+ ok(helper.isPixel(helper.getPixel(video), helper.green, 128), "Last frame should be green");
+ SimpleTest.finish();
+ };
+ // The video will resize once it loads its metadata, only listen for resizes after that
+ video.onloadedmetadata = () => {
+ ok(video.videoWidth === canvas_size && video.videoHeight === canvas_size,
+ "video element should be same size as canvas once metadata is loaded");
+ // We shouldn't have any resize events once the video is loaded
+ video.onresize = () => {
+ ok(false, "Should not have any resize events!");
+ };
+ };
+
+ video.play();
+ };
+
+ mediaRecorder.start();
+ is(mediaRecorder.state, "recording", "Media recorder should be recording");
+}
+
+SimpleTest.waitForExplicitFinish();
+startTest();
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_playback_can_repeat.html b/dom/media/test/test_mediarecorder_playback_can_repeat.html
new file mode 100644
index 0000000000..4cf2735088
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_playback_can_repeat.html
@@ -0,0 +1,87 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test MediaRecorder Recording creates videos that can playback more than once</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script src="/tests/dom/canvas/test/captureStream_common.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<pre id="test">
+<div id="content">
+ <canvas id="video-src-canvas"></canvas>
+ <video id="recorded-video" loop></video>
+</div>
+<script class="testbody" type="text/javascript">
+
+(async function() {
+ try {
+ SimpleTest.waitForExplicitFinish();
+ await startTest();
+ } catch(e) {
+ ok(false, `Got error msg '${e}'`);
+ } finally {
+ SimpleTest.finish();
+ }
+})();
+
+async function startTest() {
+ let canvas = document.getElementById("video-src-canvas");
+
+ let canvas_size = 100;
+ canvas.width = canvas.height = canvas_size;
+ let helper = new CaptureStreamTestHelper2D(canvas_size, canvas_size);
+ helper.drawColor(canvas, helper.red);
+
+ let stream = canvas.captureStream();
+ let mediaRecorder = new MediaRecorder(stream);
+ is(mediaRecorder.stream, stream,
+ "Media recorder stream = canvas stream at the beginning of recording");
+ new Promise(r => mediaRecorder.onerror = err => r(err)).then(
+ err => Promise.reject(`MediaRecorder: error unexpectedly fired. Code ${err.name}`));
+
+ mediaRecorder.start();
+ await new Promise(r => mediaRecorder.onstart = r);
+ info("Media recorder started");
+ // Feed some more data into the recorder so the output has a non trivial duration.
+ // This avoids the case where we have only 1 frame in the output, which breaks
+ // looping (this is expected as a WebM with 1 video frame has no duration).
+ let counter = 0;
+ let draw = () => {
+ counter++;
+ helper.drawColor(canvas, helper.blue);
+ if(counter > 2) {
+ mediaRecorder.stop();
+ return;
+ }
+ requestAnimationFrame(draw);
+ };
+ requestAnimationFrame(draw);
+
+ // When recorder is stopped get recorded data.
+ const data = await new Promise(r => mediaRecorder.ondataavailable = ev => r(ev));
+ info(`Media recorder get availiable data`);
+ const blob = data.data;
+
+ await new Promise(r => mediaRecorder.onstop = r);
+ info("Media recorder stopped");
+ ok(blob, "Should have gotten a data blob");
+ const video = document.getElementById("recorded-video");
+ new Promise(r => video.onerror = err => r(err)).then(
+ err => Promise.reject(`Video: error unexpectedly fired. Code ${err.code}`));
+ video.src = URL.createObjectURL(blob);
+ video.play();
+ await new Promise(r => video.onplaying = r);
+ for (let idx = 0; idx < 2; idx++) {
+ await new Promise(r => video.onseeking = r);
+ ok(true, `waiting until video finishes seeking`);
+ await new Promise(r => video.onseeked = r);
+ ok(true, "video finished seeked");
+ }
+ video.pause();
+}
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_principals.html b/dom/media/test/test_mediarecorder_principals.html
new file mode 100644
index 0000000000..00dcaa5a5b
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_principals.html
@@ -0,0 +1,132 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=489415
+-->
+<head>
+ <title>Test for MediaRecorder Reaction to Principal Change</title>
+ <script type="application/javascript" src="/MochiKit/MochiKit.js"></script>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<div>
+ <a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=1018299">Test for MediaRecorder Principal Handling</a>
+</div>
+
+<pre id="test">
+<script type="text/javascript">
+SimpleTest.waitForExplicitFinish();
+
+let throwOutside = e => setTimeout(() => { throw e; });
+
+// Loading data from a resource that changes origins while streaming should
+// be detected by the media cache and result in a null principal so that the
+// MediaRecorder usages below fail.
+
+// This test relies on midflight-redirect.sjs returning the the first quarter of
+// the resource as a byte range response, and then hanging up, and when Firefox
+// requests the remainder midflight-redirect.sjs serves a redirect to another origin.
+
+async function testPrincipals(resource) {
+ if (!resource) {
+ todo(false, "No types supported");
+ return;
+ }
+ await testPrincipals1(resource);
+ await testPrincipals2(resource);
+}
+
+function makeVideo() {
+ let video = document.createElement("video");
+ video.preload = "metadata";
+ video.controls = true;
+ document.body.appendChild(video);
+ return video;
+}
+
+// First test: Load file from same-origin first, then get redirected to
+// another origin before attempting to record stream.
+async function testPrincipals1(resource) {
+ let video = makeVideo();
+ video.src =
+ "http://mochi.test:8888/tests/dom/media/test/midflight-redirect.sjs" +
+ "?resource=" + resource.name + "&type=" + resource.type;
+
+ let errorBarrier = once(video, "error");
+ // Wait for the video to load to metadata. We can then start capturing.
+ // Must also handle the download bursting and hitting the error before we
+ // reach loadedmetadata. Normally we reach loadedmetadata first, but
+ // rarely we hit the redirect first.
+ await Promise.race([once(video, "loadedmetadata"), errorBarrier]);
+
+ let rec = new MediaRecorder(video.mozCaptureStreamUntilEnded());
+ video.play();
+
+ // Wait until we hit a playback error. This means our download has hit the redirect.
+ await errorBarrier;
+
+ // Try to record, it should be blocked with a security error.
+ try {
+ rec.start();
+ ok(false, "mediaRecorder.start() must throw SecurityError, but didn't throw at all");
+ } catch (ex) {
+ is(ex.name, "SecurityError", "mediaRecorder.start() must throw SecurityError");
+ }
+ removeNodeAndSource(video);
+}
+
+// Second test: Load file from same-origin first, but record ASAP, before
+// getting redirected to another origin.
+async function testPrincipals2(resource) {
+ let video = makeVideo();
+ video.src =
+ "http://mochi.test:8888/tests/dom/media/test/midflight-redirect.sjs" +
+ "?resource=" + resource.name + "&type=" + resource.type;
+
+ // Wait for the video to load to metadata. We can then start capturing.
+ // Must also handle the download bursting and hitting the error before we
+ // reach loadedmetadata. Normally we reach loadedmetadata first, but
+ // rarely we hit the redirect first.
+ await Promise.race([once(video, "loadedmetadata"), once(video, "error")]);
+
+ let ended = false;
+ once(video, "ended", () => ended = true);
+
+ // Start capturing. It should work.
+ let rec;
+ let errorBarrier;
+ try {
+ rec = new MediaRecorder(video.mozCaptureStreamUntilEnded());
+ errorBarrier = nextEvent(rec, "error");
+ rec.start();
+ ok(true, "mediaRecorder.start() should not throw here, and didn't");
+ } catch (ex) {
+ ok(false, "mediaRecorder.start() unexpectedly threw " + ex.name + " (" + ex.message + ")");
+ }
+
+ // Play the video, this should result in a SecurityError on the recorder.
+ let hasStopped = once(rec, "stop");
+ video.play();
+ let error = (await errorBarrier).error;
+ is(error.name, "SecurityError", "mediaRecorder.onerror must fire SecurityError");
+ ok(error.stack.includes('test_mediarecorder_principals.html'),
+ 'Events fired from onerror should include an error with a stack trace indicating ' +
+ 'an error in this test');
+ is(ended, false, "Playback should not have reached end");
+ await hasStopped;
+ is(ended, false, "Playback should definitely not have reached end");
+
+ removeNodeAndSource(video);
+}
+
+testPrincipals({ name:"pixel_aspect_ratio.mp4", type:"video/mp4", duration:28 })
+ .catch(e => throwOutside(e))
+ .then(() => SimpleTest.finish());
+
+</script>
+</pre>
+
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_record_4ch_audiocontext.html b/dom/media/test/test_mediarecorder_record_4ch_audiocontext.html
new file mode 100644
index 0000000000..99f57f181e
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_record_4ch_audiocontext.html
@@ -0,0 +1,76 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test MediaRecorder Record AudioContext with four channels</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+
+<script class="testbody" type="text/javascript">
+
+function startTest() {
+ var context = new AudioContext();
+ var buffer = context.createBuffer(4, 80920, context.sampleRate);
+ for (var i = 0; i < 80920; ++i) {
+ for(var j = 0; j < 4; ++j) {
+ buffer.getChannelData(j)[i] = Math.sin(1000 * 2 * Math.PI * i / context.sampleRate);
+ }
+ }
+
+ var source = context.createBufferSource();
+ source.buffer = buffer;
+ source.loop = true;
+ var dest = context.createMediaStreamDestination();
+ var stopTriggered = false;
+ var onstopTriggered = false;
+ dest.channelCount = 4;
+ var expectedMimeType = 'audio/ogg; codecs=opus';
+ source.channelCountMode = 'explicit';
+ source.connect(dest);
+ var elem = document.createElement('audio');
+ elem.srcObject = dest.stream;
+ source.start(0);
+ elem.play();
+ let mMediaRecorder = new MediaRecorder(dest.stream);
+ mMediaRecorder.onwarning = function() {
+ ok(false, 'onwarning unexpectedly fired');
+ };
+
+ mMediaRecorder.onerror = function() {
+ ok(false, 'onerror unexpectedly fired');
+ };
+
+ mMediaRecorder.onstop = function() {
+ ok(true, 'onstop fired successfully');
+ is(mMediaRecorder.state, 'inactive', 'check recording status is inactive');
+ onstopTriggered = true;
+ SimpleTest.finish();
+ };
+ mMediaRecorder.ondataavailable = function (e) {
+ ok(e.data.size > 0, 'check blob has data');
+ is(e.data.type, expectedMimeType, 'blob should have expected mimetype');
+ if (!stopTriggered) {
+ is(mMediaRecorder.mimeType, expectedMimeType, 'recorder should have expected mimetype');
+ mMediaRecorder.stop();
+ is(mMediaRecorder.mimeType, '', 'recorder should have reset its mimetype');
+ stopTriggered = true;
+ } else if (onstopTriggered) {
+ ok(false, 'ondataavailable should come before onstop event');
+ }
+ };
+ try {
+ mMediaRecorder.start(1000);
+ is('recording', mMediaRecorder.state, "check record state recording");
+ } catch (e) {
+ ok(false, 'Can t record audio context');
+ }
+}
+
+startTest();
+SimpleTest.waitForExplicitFinish();
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_record_addtracked_stream.html b/dom/media/test/test_mediarecorder_record_addtracked_stream.html
new file mode 100644
index 0000000000..639531f7f2
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_record_addtracked_stream.html
@@ -0,0 +1,182 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test MediaRecorder recording a constructed MediaStream</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script src="/tests/dom/canvas/test/captureStream_common.js"></script>
+ <script src="/tests/dom/media/webrtc/tests/mochitests/head.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<pre id="test">
+<div id="content">
+</div>
+<script>
+SimpleTest.waitForExplicitFinish();
+runTestWhenReady(async () => {
+ const canvas = document.createElement("canvas");
+ canvas.width = canvas.height = 100;
+ document.getElementById("content").appendChild(canvas);
+
+ const helper = new CaptureStreamTestHelper2D(100, 100);
+ helper.drawColor(canvas, helper.red);
+
+ const audioCtx = new AudioContext();
+ const osc = audioCtx.createOscillator();
+ osc.frequency.value = 1000;
+ osc.start();
+ const dest = audioCtx.createMediaStreamDestination();
+ osc.connect(dest);
+
+ const stream = dest.stream;
+
+ // Timeouts are experienced intermittently in Linux due to no sound in the
+ // destination. As a workaround wait for the source sound to arrive.
+ const sourceAnalyser = new AudioStreamAnalyser(audioCtx, stream);
+ const sourceAudioReady = sourceAnalyser.waitForAnalysisSuccess(array => {
+ const freq = osc.frequency.value;
+ const lowerFreq = freq / 2;
+ const upperFreq = freq + 1000;
+ const lowerAmp = array[sourceAnalyser.binIndexForFrequency(lowerFreq)];
+ const freqAmp = array[sourceAnalyser.binIndexForFrequency(freq)];
+ const upperAmp = array[sourceAnalyser.binIndexForFrequency(upperFreq)];
+ info("Analysing source audio. "
+ + lowerFreq + ": " + lowerAmp + ", "
+ + freq + ": " + freqAmp + ", "
+ + upperFreq + ": " + upperAmp);
+ return lowerAmp < 50 && freqAmp > 200 && upperAmp < 50;
+ });
+ await sourceAudioReady;
+ info("Source Audio content ok");
+
+ canvas.captureStream(0).getVideoTracks().forEach(t => stream.addTrack(t));
+
+ const blobs = [];
+
+ let mediaRecorder = new MediaRecorder(stream);
+ is(mediaRecorder.stream, stream,
+ "Media recorder stream = constructed stream at the start of recording");
+
+
+ mediaRecorder.ondataavailable = evt => {
+ info("ondataavailable fired");
+
+ is(mediaRecorder.state, "inactive", "Blob received after stopping");
+ is(blobs.length, 0, "This is the first and only blob");
+ ok(evt instanceof BlobEvent,
+ "Events fired from ondataavailable should be BlobEvent");
+ is(evt.type, "dataavailable",
+ "Event type should dataavailable");
+ ok(evt.data.size >= 0,
+ "Blob data size received is greater than or equal to zero");
+
+ blobs.push(evt.data);
+ };
+
+ const stopped = haveEvent(mediaRecorder, "stop", wait(5000, new Error("Timeout")));
+ const stoppedNoErrors = Promise.all([
+ stopped,
+ haveNoEvent(mediaRecorder, "warning", stopped),
+ haveNoEvent(mediaRecorder, "error", stopped)
+ ]);
+
+ mediaRecorder.start();
+ is(mediaRecorder.state, "recording", "Media recorder should be recording");
+
+ await haveEvent(mediaRecorder, "start", wait(5000, new Error("Timeout")));
+ info("onstart fired");
+
+ // The recording can be too short to cause any checks with
+ // waitForAnalysisSuccess(). Waiting a bit here solves this.
+ await wait(500);
+
+ is(mediaRecorder.state, "recording",
+ "Media recorder is recording before being stopped");
+ mediaRecorder.stop();
+ is(mediaRecorder.state, "inactive",
+ "Media recorder is inactive after being stopped");
+ is(mediaRecorder.stream, stream,
+ "Media recorder stream = constructed stream post recording");
+
+ await stoppedNoErrors;
+ info("Got 'stop' event");
+
+ ok(blobs.length == 1, "Should have gotten one data blob");
+
+ // Clean up recording sources
+ osc.stop();
+ stream.getTracks().forEach(t => t.stop());
+
+ // Sanity check the recording
+ const video = document.createElement("video");
+ document.getElementById("content").appendChild(video);
+ video.id = "recorded-video";
+
+ const blob = new Blob(blobs);
+ ok(blob.size > 0, "Recorded blob should contain data");
+
+ video.src = URL.createObjectURL(blob);
+ video.preload = "metadata";
+
+ info("Waiting for metadata to be preloaded");
+
+ await haveEvent(video, "loadedmetadata", wait(5000, new Error("Timeout")));
+ info("Playback of recording loaded metadata");
+
+ const recordingStream = video.mozCaptureStream();
+ is(recordingStream.getVideoTracks().length, 1,
+ "Recording should have one video track");
+ is(recordingStream.getAudioTracks().length, 1,
+ "Recording should have one audio track");
+
+ const ended = haveEvent(video, "ended", wait(5000, new Error("Timeout")));
+ const endedNoError = Promise.all([
+ ended,
+ haveNoEvent(video, "error", ended),
+ ]);
+
+ const analyser = new AudioStreamAnalyser(audioCtx, recordingStream);
+ const audioReady = analyser.waitForAnalysisSuccess(array => {
+ const freq = osc.frequency.value;
+ const lowerFreq = freq / 2;
+ const upperFreq = freq + 1000;
+ const lowerAmp = array[analyser.binIndexForFrequency(lowerFreq)];
+ const freqAmp = array[analyser.binIndexForFrequency(freq)];
+ const upperAmp = array[analyser.binIndexForFrequency(upperFreq)];
+ info("Analysing audio. "
+ + lowerFreq + ": " + lowerAmp + ", "
+ + freq + ": " + freqAmp + ", "
+ + upperFreq + ": " + upperAmp);
+ return lowerAmp < 50 && freqAmp > 200 && upperAmp < 50;
+ }, endedNoError.then(() => new Error("Audio check failed")));
+
+ const videoReady = helper.pixelMustBecome(
+ video, helper.red, {
+ threshold: 128,
+ infoString: "Should become red",
+ cancelPromise: endedNoError.then(() => new Error("Video check failed")),
+ });
+
+ video.play();
+
+ try {
+ await endedNoError;
+ } finally {
+ analyser.disconnect();
+ let url = video.src;
+ video.src = "";
+ URL.revokeObjectURL(url);
+ }
+
+ info("Playback of recording ended without error");
+
+ await audioReady;
+ info("Audio content ok");
+
+ await videoReady;
+ info("Video content ok");
+});
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_record_audiocontext.html b/dom/media/test/test_mediarecorder_record_audiocontext.html
new file mode 100644
index 0000000000..686faaeb6b
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_record_audiocontext.html
@@ -0,0 +1,65 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test MediaRecorder Record AudioContext</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+
+<script class="testbody" type="text/javascript">
+
+function startTest() {
+ var context = new AudioContext();
+ var buffer = context.createBuffer(1, 80920, context.sampleRate);
+ for (var i = 0; i < 80920; ++i) {
+ buffer.getChannelData(0)[i] = Math.sin(1000 * 2 * Math.PI * i / context.sampleRate);
+ }
+
+ var source = context.createBufferSource();
+ source.buffer = buffer;
+ source.loop = true;
+
+ var dest = context.createMediaStreamDestination();
+ source.connect(dest);
+ var elem = document.createElement('audio');
+ elem.srcObject = dest.stream;
+ source.start(0);
+ elem.play();
+ let mMediaRecorder = new MediaRecorder(dest.stream);
+ mMediaRecorder.onwarning = function() {
+ ok(false, 'onwarning unexpectedly fired');
+ };
+
+ mMediaRecorder.onerror = function() {
+ ok(false, 'onerror unexpectedly fired');
+ };
+
+ mMediaRecorder.onstop = function() {
+ ok(true, 'onstop fired successfully');
+ is(mMediaRecorder.state, 'inactive', 'check recording status is inactive');
+ SimpleTest.finish();
+ };
+ mMediaRecorder.ondataavailable = function (e) {
+ if (mMediaRecorder.state == 'recording') {
+ is(mMediaRecorder.mimeType, 'audio/ogg; codecs=opus', 'Expected MediaRecorder mimetype');
+ is(e.data.type, 'audio/ogg; codecs=opus', 'Expected Blob mimetype');
+ ok(e.data.size > 0, 'check blob has data');
+ mMediaRecorder.stop();
+ }
+ };
+ try {
+ mMediaRecorder.start(1000);
+ is('recording', mMediaRecorder.state, "check record state recording");
+ } catch (e) {
+ ok(false, 'Can t record audio context');
+ }
+}
+
+startTest();
+SimpleTest.waitForExplicitFinish();
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_record_audiocontext_mlk.html b/dom/media/test/test_mediarecorder_record_audiocontext_mlk.html
new file mode 100644
index 0000000000..4128702aef
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_record_audiocontext_mlk.html
@@ -0,0 +1,24 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>capture for possible memory leak when record AudioContext</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=973765">Mozill
+a Bug 973765</a>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+ // This test case want to capture the memory leak if exit the browser after running those script.
+ var ac = new window.AudioContext();
+ var destStream = ac.createMediaStreamDestination().stream;
+ var recorder = new MediaRecorder(destStream);
+ recorder.start(1000);
+ is(recorder.state, 'recording', 'Media recorder should be recording');
+ is(recorder.stream, destStream,
+ 'Media recorder stream = element stream at the start of recording');
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_record_audionode.html b/dom/media/test/test_mediarecorder_record_audionode.html
new file mode 100644
index 0000000000..8a57437b81
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_record_audionode.html
@@ -0,0 +1,135 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test MediaRecorder Record AudioContext Node</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=968109">Mozilla Bug 968109</a>
+
+<script class="testbody" type="text/javascript">
+
+SimpleTest.waitForExplicitFinish();
+
+function setUpSource(contextType, nodeType) {
+ // Use contextType to choose offline or real-time context.
+ const context = contextType == "offline"?
+ new OfflineAudioContext(2 , 80920, 44100) : new AudioContext();
+ const buffer = context.createBuffer(2, 80920, context.sampleRate);
+ for (let i = 0; i < 80920; ++i) {
+ buffer.getChannelData(0)[i] = Math.sin(1000 * 2 * Math.PI * i / context.sampleRate);
+ buffer.getChannelData(1)[i] = Math.sin(1000 * 2 * Math.PI * i / context.sampleRate);
+ }
+
+ const source = context.createBufferSource();
+ source.buffer = buffer;
+ source.loop = true;
+
+ source.start(0);
+
+ // nodeType decides which node in graph should be the source of recording.
+ let node;
+ if (nodeType == "source") {
+ node = source;
+ } else if (nodeType == "splitter") {
+ const splitter = context.createChannelSplitter();
+ source.connect(splitter);
+ node = splitter;
+ } else if (nodeType == "destination") {
+ source.connect(context.destination);
+ node = context.destination;
+ }
+ // Explicitly start offline context.
+ if (contextType == "offline") {
+ context.startRendering();
+ }
+
+ return node;
+}
+
+async function testRecord(source, mimeType) {
+ const isOffline = source.context instanceof OfflineAudioContext;
+ const recorder = new MediaRecorder(source, 0, {mimeType});
+ is(recorder.mimeType, mimeType, "Mime type is set");
+ const extendedMimeType = `${mimeType || "audio/ogg"}; codecs=opus`;
+
+ recorder.onwarning = () => ok(false, "should not fire onwarning");
+ recorder.onerror = () => ok(false, "should not fire onerror");
+ if (!isOffline) {
+ recorder.onstop = () => ok(false, "should not fire stop yet");
+ }
+
+ recorder.start(1000);
+ is("recording", recorder.state, "state should become recording after calling start()");
+ is(recorder.mimeType, mimeType, "Mime type is not changed by start()");
+
+ await new Promise(r => recorder.onstart = r);
+ is(recorder.mimeType, extendedMimeType, "Mime type is fully defined");
+
+ const chunks = [];
+ let {data} = await new Promise(r => recorder.ondataavailable = r);
+ if (!isOffline) {
+ is(recorder.state, "recording", "Expected to still be recording");
+ }
+ is(data.type, extendedMimeType, "Blob has fully defined mimetype");
+ isnot(data.size, 0, "should get data and its length should be > 0");
+ chunks.push(data);
+
+ if (isOffline) {
+ await new Promise(r => recorder.onstop = r);
+ is(recorder.state, "inactive", "Offline context should end by itself");
+ } else {
+ is(recorder.state, "recording", "Expected to still be recording");
+ recorder.stop();
+ ({data} = await new Promise(r => recorder.ondataavailable = r));
+ is(recorder.state, "inactive", "Expected to be inactive after last blob");
+ isnot(data.size, 0, "should get data and its length should be > 0");
+ chunks.push(data);
+
+ await new Promise(r => recorder.onstop = r);
+ is(recorder.state, "inactive", "state should remain inactive after stop event");
+ }
+ return new Blob(chunks, {type: chunks[0].type});
+}
+
+addLoadEvent(async () => {
+ const src = setUpSource();
+ let didThrow = false;
+ try {
+ new MediaRecorder(src);
+ } catch (e) {
+ didThrow = true;
+ }
+ ok(didThrow, "MediaRecorder(AudioNode) should be hidden behind a pref");
+
+ await SpecialPowers.pushPrefEnv({set: [
+ ["media.recorder.audio_node.enabled", true],
+ ]});
+
+ // Test with various context and source node types.
+ for (const mimeType of [
+ "audio/ogg",
+ "audio/webm",
+ "video/webm",
+ "",
+ ]) {
+ for (const {context, node} of [
+ {context: "", node: "source"},
+ {context: "", node: "splitter"},
+ {context: "offline", node: "destination"},
+ ]) {
+ info(`Testing recording ${context || "regular"} context and ${node} ` +
+ `node with mimeType '${mimeType}'`);
+ await testRecord(setUpSource(context, node), mimeType);
+ }
+ }
+
+ SimpleTest.finish();
+});
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_record_canvas_captureStream.html b/dom/media/test/test_mediarecorder_record_canvas_captureStream.html
new file mode 100644
index 0000000000..0b6cd6dbb5
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_record_canvas_captureStream.html
@@ -0,0 +1,75 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test MediaRecorder Recording canvas stream</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script src="/tests/dom/canvas/test/captureStream_common.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<pre id="test">
+<div id="content">
+</div>
+<script class="testbody" type="text/javascript">
+
+function startTest() {
+ var canvas = document.createElement("canvas");
+ canvas.width = canvas.height = 100;
+ document.getElementById("content").appendChild(canvas);
+
+ var helper = new CaptureStreamTestHelper2D(100, 100);
+ helper.drawColor(canvas, helper.red);
+
+ var stream = canvas.captureStream(0);
+
+ var blob;
+
+ let mediaRecorder = new MediaRecorder(stream);
+ is(mediaRecorder.stream, stream,
+ "Media recorder stream = canvas stream at the start of recording");
+
+ mediaRecorder.onwarning = () => ok(false, "warning unexpectedly fired");
+
+ mediaRecorder.onerror = () => ok(false, "Recording failed");
+
+ mediaRecorder.ondataavailable = ev => {
+ is(blob, undefined, "Should only get one dataavailable event");
+ blob = ev.data;
+ };
+
+ mediaRecorder.onstart = () => {
+ info("Got 'start' event");
+ // We just want one frame encoded, to see that the recorder produces something readable.
+ mediaRecorder.stop();
+ };
+
+ mediaRecorder.onstop = () => {
+ info("Got 'stop' event");
+ ok(blob, "Should have gotten a data blob");
+
+ var video = document.createElement("video");
+ video.id = "recorded-video";
+ video.src = URL.createObjectURL(blob);
+ video.play();
+ video.onerror = err => {
+ ok(false, "Should be able to play the recording. Got error. code=" + video.error.code);
+ SimpleTest.finish();
+ };
+ document.getElementById("content").appendChild(video);
+ helper.pixelMustBecome(video, helper.red, {
+ threshold: 128,
+ infoString: "Should become red"
+ }).then(SimpleTest.finish);
+ };
+
+ mediaRecorder.start();
+ is(mediaRecorder.state, "recording", "Media recorder should be recording");
+}
+
+SimpleTest.waitForExplicitFinish();
+startTest();
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_record_changing_video_resolution.html b/dom/media/test/test_mediarecorder_record_changing_video_resolution.html
new file mode 100644
index 0000000000..d6354ee5a1
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_record_changing_video_resolution.html
@@ -0,0 +1,174 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test MediaRecorder Recording canvas stream that dynamically changes resolution</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<pre id="test">
+<div id="content">
+</div>
+<script class="testbody" type="text/javascript">
+
+function startTest() {
+ let canvas = document.createElement("canvas");
+ const resolution_change = [
+ {width: 100, height: 100, color: "red"},
+ {width: 150, height: 150, color: "blue"},
+ {width: 100, height: 100, color: "red"},
+ ];
+ canvas.width = resolution_change[0].width;
+ canvas.height = resolution_change[0].height;
+
+ let ctx = canvas.getContext("2d");
+ ctx.fillStyle = resolution_change[0].color;
+ ctx.fillRect(0, 0, canvas.width, canvas.height);
+
+ // The recorded stream coming from canvas.
+ let stream = canvas.captureStream();
+
+ // Check values for events
+ let numDataAvailabledRaised = 0;
+ let numResizeRaised = 0;
+ // Recorded data that will be playback.
+ let blob;
+
+ // Media recorder for VP8 and canvas stream.
+ let mediaRecorder = new MediaRecorder(stream);
+ is(mediaRecorder.stream, stream,
+ "Media recorder stream = canvas stream at the start of recording");
+
+ // Not expected events.
+ mediaRecorder.onwarning = () => ok(false, "MediaRecorder: onwarning unexpectedly fired");
+ mediaRecorder.onerror = err => {
+ ok(false, "MediaRecorder: onerror unexpectedly fired. Code " + err.name);
+ SimpleTest.finish();
+ };
+
+ // When recorder is stopped get recorded data.
+ mediaRecorder.ondataavailable = ev => {
+ info("Got 'dataavailable' event");
+ ++numDataAvailabledRaised;
+ is(blob, undefined, "Should only get one dataavailable event");
+ // Save recorded data for playback
+ blob = ev.data;
+ };
+
+ mediaRecorder.onstart = () => {
+ info('onstart fired successfully');
+ };
+
+ mediaRecorder.onstop = () => {
+ info("Got 'stop' event");
+ is(numDataAvailabledRaised, 1, "Should have gotten 1 dataavailable event");
+ // Playback stream and verify resolution changes.
+ ok(blob, "Should have gotten a data blob");
+
+ let video = document.createElement("video");
+ video.id = "recorded-video";
+ video.src = URL.createObjectURL(blob);
+ video.preload = "metadata";
+
+ video.onerror = err => {
+ ok(false, "Should be able to play the recording. Got error. code=" + video.error.code);
+ SimpleTest.finish();
+ };
+
+ // Check that the encoded frames have the correct sizes.
+ video.onresize = function() {
+ if (numResizeRaised < resolution_change.length) {
+ is(video.videoWidth, resolution_change[numResizeRaised].width,
+ "onresize width should be as expected");
+ is(video.videoHeight, resolution_change[numResizeRaised].height,
+ "onresize height should be as expected");
+ } else {
+ ok(false, "Got more resize events than expected");
+ }
+ ++numResizeRaised;
+ };
+
+ video.onloadedmetadata = function() {
+ info("loadedmetadata");
+ seekThroughFrames();
+ };
+
+ video.onended = function() {
+ is(numResizeRaised, resolution_change.length, "Expected number of resize events");
+ SimpleTest.finish();
+ // This shouldn't be needed, however video.ended may not be set after
+ // seeking to the final frame. This can result in seekToNextFrame being
+ // called again by seekThroughFrames and onended being invoked again,
+ // resulting in multiple finish() calls.
+ // FIXME: https://bugzilla.mozilla.org/show_bug.cgi?id=1386489
+ video.onended = null;
+ };
+
+ document.getElementById("content").appendChild(video);
+
+ function seekThroughFrames() {
+ info("Seeking to next frame");
+ video.seekToNextFrame()
+ .then(() => {
+ info("Seeking to next frame finished. width=" + video.videoWidth
+ + ", height=" + video.videoHeight);
+
+ if (video.ended) {
+ return;
+ }
+
+ // After seeking finished we queue the next seek task on the event
+ // loop so it gets in the same queue as the "resize" events.
+ setTimeout(seekThroughFrames, 0);
+ })
+ .catch(error => {
+ ok(false, "seekToNextFrame rejected: " + error);
+ });
+ }
+ };
+
+ // Start here by stream recorder.
+ mediaRecorder.start();
+ is(mediaRecorder.state, "recording", "Media recorder should be recording");
+ requestAnimationFrame(draw);
+
+ // Change resolution in every frame
+ // Stop recorder on last frame
+ let countFrames = 0;
+ let previous_time = performance.now();
+ function draw(timestamp) {
+ if (timestamp - previous_time < 100) {
+ requestAnimationFrame(draw);
+ return;
+ }
+ previous_time = timestamp;
+
+ if (countFrames == resolution_change.length) {
+ mediaRecorder.stop();
+ return;
+ }
+
+ canvas.width = resolution_change[countFrames].width;
+ canvas.height = resolution_change[countFrames].height;
+ ctx.fillStyle = resolution_change[countFrames].color;
+ // Resize and draw canvas
+ ctx.fillRect(0, 0, canvas.width, canvas.height);
+ // Register draw to be called on next rendering
+ requestAnimationFrame(draw);
+ countFrames++;
+ }
+}
+
+SimpleTest.waitForExplicitFinish();
+SpecialPowers.pushPrefEnv(
+ {
+ "set": [
+ ["media.seekToNextFrame.enabled", true ],
+ ["media.video-queue.send-to-compositor-size", 1]
+ ]
+ }, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_record_downsize_resolution.html b/dom/media/test/test_mediarecorder_record_downsize_resolution.html
new file mode 100644
index 0000000000..f9422a3897
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_record_downsize_resolution.html
@@ -0,0 +1,148 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test MediaRecorder Recording canvas dynamically changes to greater resolution</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script src="/tests/dom/canvas/test/captureStream_common.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<pre id="test">
+<div id="content">
+</div>
+<script class="testbody" type="text/javascript">
+
+function startTest() {
+ var canvas = document.createElement("canvas");
+ var canvas_size = 100;
+ var new_canvas_size = 50;
+ canvas.width = canvas.height = canvas_size;
+
+ var helper = new CaptureStreamTestHelper2D(canvas_size, canvas_size);
+ helper.drawColor(canvas, helper.red);
+
+ // The recorded stream coming from canvas.
+ var stream = canvas.captureStream();
+
+ // Check values for events
+ var numDataAvailabledRaised = 0;
+ var numResizeRaised = 0;
+ // Recorded data that will be playback.
+ var blob;
+
+ // Media recorder for VP8 and canvas stream.
+ var mediaRecorder = new MediaRecorder(stream);
+ is(mediaRecorder.stream, stream,
+ "Media recorder stream = canvas stream at the beginning of recording");
+
+ // Not expected events.
+ mediaRecorder.onwarning = () => ok(false, "MediaRecorder: onwarning unexpectedly fired");
+ mediaRecorder.onerror = err => {
+ ok(false, "MediaRecorder: onerror unexpectedly fired. Code " + err.name);
+ SimpleTest.finish();
+ }
+
+ // When recorder is stopped get recorded data.
+ mediaRecorder.ondataavailable = ev => {
+ info("Got 'dataavailable' event");
+ ++numDataAvailabledRaised;
+ is(blob, undefined, "On dataavailable event blob is undefined");
+ // Save recorded data for playback
+ blob = ev.data;
+ };
+
+ mediaRecorder.onstart = () => {
+ info('onstart fired successfully');
+ };
+
+ mediaRecorder.onstop = () => {
+ info("Got 'stop' event");
+ is(numDataAvailabledRaised, 1, "Expected 1 dataavailable event");
+
+ // Playback stream and verify resolution changes.
+ ok(blob, "Should have gotten a data blob");
+ var video = document.createElement("video");
+ video.id = "recorded-video";
+ video.src = URL.createObjectURL(blob);
+ video.onerror = err => {
+ ok(false, "Should be able to play the recording. Got error. code=" + video.error.code);
+ SimpleTest.finish();
+ };
+
+ // Check here that resize is correct in the playback stream.
+ video.onresize = function() {
+ ++numResizeRaised;
+ if (numResizeRaised == 1) {
+ is(this.videoWidth, canvas_size, "1st resize event original width");
+ is(this.videoHeight, canvas_size, "1st resize event original height ");
+ } else if (numResizeRaised == 2) {
+ is(this.videoWidth, new_canvas_size, "2nd resize event new width");
+ is(this.videoHeight, new_canvas_size, "2nd resize event new height");
+ } else {
+ ok(false, "Only 2 numResizeRaised events are expected");
+ }
+ };
+
+ video.onended = () => {
+ is(numResizeRaised, 2, "Expected 2 resize event");
+ };
+ document.getElementById("content").appendChild(video);
+ video.play();
+
+ // Check last color
+ helper.pixelMustBecome(video, helper.red, {
+ threshold: 128,
+ infoString: "Should become red",
+ }).then(() => {
+ video.onresize = {};
+ video.onended = {};
+ SimpleTest.finish();
+ });
+ };
+
+ // Start here by stream recorder.
+ mediaRecorder.start();
+ is(mediaRecorder.state, "recording", "Media recorder started");
+ requestAnimationFrame(draw);
+
+ // Change resolution every 100ms
+ var countFrames=0;
+ var previous_time = performance.now();
+ function draw(timestamp) {
+ if (timestamp - previous_time < 100) {
+ requestAnimationFrame(draw);
+ return;
+ }
+ previous_time = timestamp;
+
+ var size = 0;
+ var color = "";
+ if (countFrames < 1) {
+ // Initial size
+ size = canvas_size;
+ color = helper.blue;
+ } else if (countFrames < 2) {
+ // upsize
+ size = new_canvas_size;
+ color = helper.red;
+ } else {
+ // Stop recoredr on last frame
+ mediaRecorder.stop();
+ return;
+ }
+ // Resize and draw canvas
+ canvas.width = canvas.height = size;
+ helper.drawColor(canvas, color);
+ // Register next draw on every call
+ requestAnimationFrame(draw);
+ countFrames++;
+ }
+}
+
+SimpleTest.waitForExplicitFinish();
+startTest();
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_record_getdata_afterstart.html b/dom/media/test/test_mediarecorder_record_getdata_afterstart.html
new file mode 100644
index 0000000000..3b181ed8db
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_record_getdata_afterstart.html
@@ -0,0 +1,81 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Bug 951008 Test MediaRecorder Record has start event</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function startTest(test, token) {
+
+ var element = document.createElement('audio');
+ var hasonstart = false;
+ var hasondataavailable = false;
+ var mMediaRecorder;
+
+ element.token = token;
+ manager.started(token);
+ element.src = test.name;
+ element.test = test;
+ element.stream = element.mozCaptureStream();
+
+ mMediaRecorder = new MediaRecorder(element.stream);
+ is(mMediaRecorder.mimeType, '', 'Expected MediaRecorder mimetype');
+ mMediaRecorder.onwarning = function() {
+ ok(false, 'onwarning unexpectedly fired');
+ };
+
+ mMediaRecorder.onerror = function() {
+ ok(false, 'onerror unexpectedly fired');
+ };
+
+ mMediaRecorder.onstart = function() {
+ info('onstart fired successfully');
+ hasonstart = true;
+ is(mMediaRecorder.mimeType, 'audio/ogg; codecs=opus',
+ "MediaRecorder mimetype as expected");
+ mMediaRecorder.requestData();
+ };
+
+ mMediaRecorder.onstop = function() {
+ info('onstop fired successfully');
+ ok(hasondataavailable, "should have ondataavailable before onstop");
+ is(mMediaRecorder.state, 'inactive', 'check recording status is inactive');
+ SimpleTest.finish();
+ };
+
+ mMediaRecorder.ondataavailable = function (e) {
+ info('ondataavailable fired successfully');
+ if (mMediaRecorder.state == 'recording') {
+ hasondataavailable = true;
+ ok(hasonstart, "should have had start event first");
+ is(e.data.type, mMediaRecorder.mimeType,
+ "blob's mimeType matches the recorder's");
+ mMediaRecorder.stop();
+ }
+ };
+
+ // Start recording once metadata are parsed.
+ element.onloadedmetadata = function() {
+ element.oncanplaythrough = null;
+ mMediaRecorder.start(0);
+ is(mMediaRecorder.state, 'recording', 'Media recorder should be recording');
+ is(mMediaRecorder.stream, element.stream,
+ 'Media recorder stream = element stream at the start of recording');
+ };
+
+ element.play();
+}
+
+manager.runTests(gMediaRecorderTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_record_gum_video_timeslice.html b/dom/media/test/test_mediarecorder_record_gum_video_timeslice.html
new file mode 100644
index 0000000000..961a9644b2
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_record_gum_video_timeslice.html
@@ -0,0 +1,94 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test MediaRecorder Record gUM video with Timeslice</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="gUM_support.js"></script>
+</head>
+<body>
+<pre id="test">
+<div id="content" style="display: none">
+</div>
+<script class="testbody" type="text/javascript">
+
+async function startTest() {
+ try {
+ await setupGetUserMediaTestPrefs();
+ let stream = await navigator.mediaDevices.getUserMedia({audio: true, video: true});
+ let dataAvailableCount = 0;
+ let onDataAvailableFirst = false;
+ const expectedMimeType = 'video/webm; codecs="vp8, opus"';
+
+ let mediaRecorder = new MediaRecorder(stream);
+ is(mediaRecorder.stream, stream,
+ 'Media recorder stream = element stream at the start of recording');
+ mediaRecorder.onwarning = function() {
+ ok(false, 'onwarning unexpectedly fired');
+ };
+
+ mediaRecorder.onerror = function() {
+ ok(false, 'onerror unexpectedly fired');
+ };
+
+ mediaRecorder.onstop = function() {
+ ok(false, 'Unexpected onstop callback fired');
+ };
+
+ mediaRecorder.onstart = function() {
+ is(mediaRecorder.mimeType, expectedMimeType, 'Expected mime type');
+ };
+
+ mediaRecorder.ondataavailable = function (evt) {
+ info('ondataavailable fired');
+ dataAvailableCount++;
+
+ ok(evt instanceof BlobEvent,
+ 'Events fired from ondataavailable should be BlobEvent');
+ is(evt.type, 'dataavailable',
+ 'Event type should dataavailable');
+ ok(evt.data.size >= 0,
+ 'Blob data size ' + evt.data.size + ' received is greater than or equal to zero');
+ is(evt.data.type, expectedMimeType, 'Expected blob mime type');
+
+ // We'll stop recording upon the 1st blob being received
+ if (dataAvailableCount === 1) {
+ mediaRecorder.onstop = function (event) {
+ info('onstop fired');
+
+ if (!onDataAvailableFirst) {
+ ok(false, 'onstop unexpectedly fired before ondataavailable');
+ }
+
+ ok(true, 'onstop fired successfully');
+ is(mediaRecorder.state, 'inactive',
+ 'check recording status is inactive');
+ SimpleTest.finish();
+ };
+
+ mediaRecorder.stop();
+ is(mediaRecorder.state, 'inactive',
+ 'Media recorder is inactive after being stopped');
+
+ } else if (dataAvailableCount === 2) {
+ // Ensure we've received at least two ondataavailable events before
+ // onstop
+ onDataAvailableFirst = true;
+ }
+ };
+
+ mediaRecorder.start(250);
+ is(mediaRecorder.state, 'recording', 'Media recorder should be recording');
+ is(mediaRecorder.mimeType, '', 'Expected mime type');
+ } catch (err) {
+ ok(false, 'Unexpected error fired with: ' + err);
+ SimpleTest.finish();
+ }
+}
+
+SimpleTest.waitForExplicitFinish();
+startTest();
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_record_gum_video_timeslice_mixed.html b/dom/media/test/test_mediarecorder_record_gum_video_timeslice_mixed.html
new file mode 100644
index 0000000000..5e9e4d3023
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_record_gum_video_timeslice_mixed.html
@@ -0,0 +1,100 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test MediaRecorder Record gUM video with Timeslice, and playback of mixed memory and file blobs</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="gUM_support.js"></script>
+</head>
+<body>
+<pre id="test">
+<script type="text/javascript">
+function unexpected({type}) {
+ ok(false, `${type} unexpectedly fired`);
+}
+
+(async () => {
+ SimpleTest.waitForExplicitFinish();
+ let blobUrl = null;
+ let stream = null;
+ try {
+ // This is the memory limit per blob. If a blob is larger than this,
+ // MediaRecorder will put it in a file. For this test we need to get at
+ // least one blob under, and one blob over the limit.
+ const memoryLimit = 50000;
+ await SpecialPowers.pushPrefEnv({set: [
+ ["media.recorder.max_memory", memoryLimit],
+ ]});
+ // We always use fake devices since the loopback ones don't make enough
+ // pixels change per frame to make the encoded frames large enough.
+ await pushGetUserMediaTestPrefs({fakeAudio: true, fakeVideo: true});
+ stream = await navigator.mediaDevices.getUserMedia(
+ {audio: true, video: true});
+ const blobs = [];
+
+ let mediaRecorder = new MediaRecorder(stream);
+ is(mediaRecorder.stream, stream,
+ "Media recorder stream = element stream at the start of recording");
+ mediaRecorder.start();
+ mediaRecorder.addEventListener("warning", unexpected);
+ mediaRecorder.addEventListener("error", unexpected);
+ mediaRecorder.addEventListener("stop", unexpected);
+ await new Promise(r => mediaRecorder.onstart = r);
+
+ for (let hasMemory = false; !hasMemory;) {
+ mediaRecorder.requestData();
+ const {data} = await new Promise(r => mediaRecorder.ondataavailable = r);
+ blobs.push(data);
+ ok(data.size < memoryLimit, "Blob should be small enough at start");
+ hasMemory = data.size > 0 && data.size < memoryLimit;
+ info(`Blob is ${data.size} bytes.${hasMemory ? " In memory." : ""}`);
+ }
+ info("Got a memory blob");
+
+ SimpleTest.requestFlakyTimeout("Wait for file blob");
+ for (let hasFile = false, waitTimeMs = 500; !hasFile; waitTimeMs *= 4) {
+ info(`Waiting ${waitTimeMs/1000} seconds for file blob`);
+ await new Promise(r => setTimeout(r, waitTimeMs));
+ mediaRecorder.requestData();
+ const {data} = await new Promise(r => mediaRecorder.ondataavailable = r);
+ blobs.push(data);
+ hasFile = data.size > memoryLimit;
+ info(`Blob is ${data.size} bytes. In ${hasFile ? "file" : "memory"}.`);
+ }
+ info("Got a file blob");
+
+ mediaRecorder.stop();
+ const {data} = await new Promise(r => mediaRecorder.ondataavailable = r);
+ blobs.push(data);
+ mediaRecorder.removeEventListener("stop", unexpected);
+ await new Promise(r => mediaRecorder.onstop = r);
+
+ const video = document.createElement("video");
+ const blob = new Blob(blobs);
+ blobUrl = URL.createObjectURL(blob);
+ video.src = blobUrl;
+ info(`Starting playback. Blob-size=${blob.size}`);
+ video.play();
+
+ await Promise.race([
+ new Promise(res => video.onended = res),
+ new Promise((res, rej) => video.onerror = () => rej(video.error.message)),
+ ]);
+ } catch (e) {
+ ok(false, e);
+ } finally {
+ if (stream) {
+ for (const t of stream.getTracks()) {
+ t.stop();
+ }
+ }
+ if (blobUrl) {
+ URL.revokeObjectURL(blobUrl);
+ }
+ SimpleTest.finish();
+ }
+})();
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_record_immediate_stop.html b/dom/media/test/test_mediarecorder_record_immediate_stop.html
new file mode 100644
index 0000000000..8ed7c321a0
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_record_immediate_stop.html
@@ -0,0 +1,115 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test MediaRecorder Immediate Stop</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var manager = new MediaTestManager;
+
+/**
+ * Stops the media recorder immediately after starting the recorder. This test
+ * verifies whether the media recorder can handle this scenario nicely. The
+ * return blob size should be greater than zero, but its duration would be zero
+ * length when play.
+ */
+function startTest(test, token) {
+ var element = document.createElement('audio');
+ var expectedMimeType = test.type;
+
+ element.token = token;
+ manager.started(token);
+
+ element.src = test.name;
+ element.test = test;
+ element.stream = element.mozCaptureStreamUntilEnded();
+
+ var mediaRecorder =
+ new MediaRecorder(element.stream, {mimeType: expectedMimeType});
+ var onStopFired = false;
+ var onDataAvailableFired = false;
+
+ mediaRecorder.onerror = function () {
+ ok(false, 'Unexpected onerror callback fired');
+ };
+
+ mediaRecorder.onwarning = function () {
+ ok(false, 'Unexpected onwarning callback fired');
+ };
+
+ // This handler verifies that only a single onstop event handler is fired.
+ mediaRecorder.onstop = function () {
+ if (onStopFired) {
+ ok(false, 'onstop unexpectedly fired more than once');
+ } else {
+ onStopFired = true;
+
+ // ondataavailable should always fire before onstop
+ if (onDataAvailableFired) {
+ manager.finished(token);
+ } else {
+ ok(false, 'onstop fired without an ondataavailable event first');
+ }
+ }
+ };
+
+ // This handler verifies that only a single ondataavailable event handler
+ // is fired with the blob generated having greater than zero size
+ // and a correct mime type.
+ mediaRecorder.ondataavailable = function (evt) {
+ if (onDataAvailableFired) {
+ ok(false, 'ondataavailable unexpectedly fired more than once');
+ } else {
+ onDataAvailableFired = true;
+
+ ok(evt instanceof BlobEvent,
+ 'Events fired from ondataavailable should be BlobEvent');
+ is(evt.type, 'dataavailable',
+ 'Event type should dataavailable');
+
+ // The initialization of encoder can be cancelled.
+ // On some platforms, the stop method may run after media stream track
+ // available, so the blob can contain the header data.
+ is(evt.data.type, expectedMimeType,
+ 'Blob data received and should have mime type');
+ is(mediaRecorder.mimeType, expectedMimeType,
+ 'Media Recorder mime type in ondataavailable = ' + expectedMimeType);
+ ok(evt.data.size >= 0, 'Blob size can not be negative');
+
+ // onstop should not have fired before ondataavailable
+ if (onStopFired) {
+ ok(false, 'ondataavailable unexpectedly fired later than onstop');
+ manager.finished(token);
+ }
+ }
+ };
+
+ // This handler completes a start and stop of recording and verifies
+ // respective media recorder state.
+ element.onloadedmetadata = function () {
+ element.onloadedmetadata = null;
+ element.play();
+ mediaRecorder.start();
+ is(mediaRecorder.state, 'recording', 'Media recorder should be recording');
+ is(mediaRecorder.stream, element.stream,
+ 'Media recorder stream = element stream at the start of recording');
+
+ mediaRecorder.stop();
+ is(mediaRecorder.state, 'inactive',
+ 'Media recorder is inactive after being stopped');
+ is(mediaRecorder.stream, element.stream,
+ 'Media recorder stream = element stream post recording');
+ };
+
+ element.preload = "metadata";
+}
+
+manager.runTests(gMediaRecorderTests, startTest);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_record_no_timeslice.html b/dom/media/test/test_mediarecorder_record_no_timeslice.html
new file mode 100644
index 0000000000..6ed148a108
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_record_no_timeslice.html
@@ -0,0 +1,106 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test MediaRecorder Record No Timeslice</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var manager = new MediaTestManager;
+
+/**
+ * Starts a test on every media recorder file included to check that a
+ * stream derived from the file can be recorded with no time slice provided.
+ */
+function startTest(test, token) {
+ var element = document.createElement('audio');
+ var expectedMimeType = test.type;
+
+ element.token = token;
+ manager.started(token);
+
+ element.src = test.name;
+ element.test = test;
+ element.stream = element.mozCaptureStreamUntilEnded();
+
+ var mediaRecorder =
+ new MediaRecorder(element.stream, {mimeType: expectedMimeType});
+ var onStopFired = false;
+ var onDataAvailableFired = false;
+
+ mediaRecorder.onerror = function () {
+ ok(false, 'Unexpected onerror callback fired');
+ };
+
+ mediaRecorder.onwarning = function () {
+ ok(false, 'Unexpected onwarning callback fired');
+ };
+
+ // This handler verifies that only a single onstop event handler is fired.
+ mediaRecorder.onstop = function () {
+ if (onStopFired) {
+ ok(false, 'onstop unexpectedly fired more than once');
+ } else {
+ onStopFired = true;
+
+ // ondataavailable should always fire before onstop
+ if (onDataAvailableFired) {
+ ok(true, 'onstop fired after ondataavailable');
+ manager.finished(token);
+ } else {
+ ok(false, 'onstop fired without an ondataavailable event first');
+ }
+ }
+ };
+
+ // This handler verifies that only a single ondataavailable event handler
+ // is fired with the blob generated having greater than zero size
+ // and a correct mime type.
+ mediaRecorder.ondataavailable = function (evt) {
+ if (onDataAvailableFired) {
+ ok(false, 'ondataavailable unexpectedly fired more than once');
+ } else {
+ onDataAvailableFired = true;
+
+ ok(evt instanceof BlobEvent,
+ 'Events fired from ondataavailable should be BlobEvent');
+ is(evt.type, 'dataavailable',
+ 'Event type should dataavailable');
+ ok(evt.data.size > 0,
+ 'Blob data received should be greater than zero');
+ is(evt.data.type, expectedMimeType,
+ 'Blob data received should have type = ' + expectedMimeType);
+
+ is(mediaRecorder.mimeType, expectedMimeType,
+ 'Mime type in ondataavailable = ' + expectedMimeType);
+
+ // onstop should not have fired before ondataavailable
+ if (onStopFired) {
+ ok(false, 'ondataavailable unexpectedly fired later than onstop');
+ manager.finished(token);
+ }
+ }
+ };
+
+ element.preload = "metadata";
+
+ element.onloadedmetadata = function () {
+ element.onloadedmetadata = null;
+ mediaRecorder.start();
+ is(mediaRecorder.state, 'recording',
+ 'Media recorder should be recording');
+ is(mediaRecorder.stream, element.stream,
+ 'Media recorder stream = element stream at the start of recording');
+
+ element.play();
+ }
+}
+
+manager.runTests(gMediaRecorderTests, startTest);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_record_session.html b/dom/media/test/test_mediarecorder_record_session.html
new file mode 100644
index 0000000000..88795d82d0
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_record_session.html
@@ -0,0 +1,75 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=909670
+-->
+<head>
+ <meta charset="utf-8">
+ <title>Test for Media Recoder recording session</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var manager = new MediaTestManager;
+
+function startTest(test, token) {
+ var element = document.createElement('audio');
+
+ element.token = token;
+ manager.started(token);
+
+ element.src = test.name;
+ element.test = test;
+ element.stream = element.mozCaptureStream();
+
+ var mStopCount = 0;
+ // Start and stop recording session three times continuously.
+ var mExpectStopCount = 3;
+ var mediaRecorder = new MediaRecorder(element.stream);
+
+ // Stop callback.
+ // Suppose to receive mExpectStopCount
+ mediaRecorder.onstop = function stopCallback() {
+ mStopCount++;
+
+ info("MediaRecorder.onstop callback: (" + mStopCount + ")");
+
+ if (mExpectStopCount === mStopCount)
+ {
+ manager.finished(token);
+ }
+ }
+
+ // data avaliable.
+ mediaRecorder.ondataavailable = function(evt) {}
+
+ mediaRecorder.onerror = function(err) {
+ ok(false, 'Unexpected error fired with:' + err);
+ }
+
+ mediaRecorder.onwarning = function() {
+ ok(false, 'Unexpected warning fired');
+ }
+
+ element.preload = "metadata";
+
+ element.onloadedmetadata = function () {
+ element.onloadedmetadata = null;
+ element.play();
+ for (var i = 0; i < mExpectStopCount; i++) {
+ mediaRecorder.start(1000);
+ mediaRecorder.stop();
+ }
+ }
+
+}
+
+manager.runTests(gMediaRecorderTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_record_startstopstart.html b/dom/media/test/test_mediarecorder_record_startstopstart.html
new file mode 100644
index 0000000000..b4cc62c709
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_record_startstopstart.html
@@ -0,0 +1,75 @@
+ <!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test MediaRecorder crash on sequence start stop start method</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<pre id="test">
+<div id="content" style="display: none">
+</div>
+<script class="testbody" type="text/javascript">
+
+function startTest() {
+ var ac = new window.AudioContext();
+ var dest = ac.createMediaStreamDestination();
+ var recorder = new MediaRecorder(dest.stream);
+ var stopCount = 0;
+ var dataavailable = 0;
+ var expectedMimeType = 'audio/ogg; codecs=opus';
+ recorder.onstop = function (e) {
+ info('onstop fired');
+ is(recorder.stream, dest.stream,
+ 'Media recorder stream = element stream post recording');
+ stopCount++;
+ if (stopCount == 2) {
+ if (dataavailable >= 2) {
+ SimpleTest.finish();
+ } else {
+ ok(false, 'Should have at least two dataavailable events');
+ }
+ }
+ }
+ recorder.ondataavailable = function (evt) {
+ info('ondataavailable fired');
+ ok(evt instanceof BlobEvent,
+ 'Events fired from ondataavailable should be BlobEvent');
+ is(evt.type, 'dataavailable',
+ 'Event type should dataavailable');
+ // If script runs slower, it may generate header data in blob from encoder
+ if (evt.data.size > 0) {
+ info('blob size = ' + evt.data.size);
+ is(evt.data.type, expectedMimeType,
+ 'Blob data received should have type = ' + expectedMimeType);
+ }
+ dataavailable++;
+ }
+ recorder.onerror = function (e) {
+ ok(false, 'it should execute normally without exception');
+ }
+ recorder.onwarning = function() {
+ ok(false, 'onwarning unexpectedly fired');
+ };
+
+ recorder.start(2000);
+ is(recorder.state, 'recording', 'Media recorder should be recording');
+ recorder.stop();
+ is(recorder.state, 'inactive', 'check recording status is inactive');
+ recorder.start(10000); // This bug would crash on this line without this fix.
+ is(recorder.state, 'recording', 'check recording status is recording');
+ // Simulate delay stop, only delay stop no no stop can trigger crash.
+ setTimeout(function() {
+ recorder.stop();
+ is(recorder.state, 'inactive','check recording status is recording');
+ }, 1000);
+}
+
+SimpleTest.requestFlakyTimeout("untriaged");
+startTest();
+SimpleTest.waitForExplicitFinish();
+</script>
+</pre>
+</body>
+</html>
+
diff --git a/dom/media/test/test_mediarecorder_record_timeslice.html b/dom/media/test/test_mediarecorder_record_timeslice.html
new file mode 100644
index 0000000000..3e547e77b4
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_record_timeslice.html
@@ -0,0 +1,105 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test MediaRecorder Record Timeslice</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var manager = new MediaTestManager;
+
+/**
+ * Starts a test on every media recorder file included to check that a stream
+ * derived from the file can be recorded with a timeslice provided
+ */
+function startTest(test, token) {
+ var element = document.createElement('audio');
+ var expectedMimeType = test.type;
+
+ element.token = token;
+ manager.started(token);
+
+ element.src = test.name;
+ element.test = test;
+ element.preload = "auto";
+
+ // Set up MediaRecorder once loadedmetadata fires and tracks are available.
+ element.onloadedmetadata = function() {
+ element.onloadedmetadata = null;
+
+ const stream = element.mozCaptureStream();
+ const mediaRecorder =
+ new MediaRecorder(stream, {mimeType: expectedMimeType});
+
+ mediaRecorder.onerror = function () {
+ ok(false, 'Unexpected onerror callback fired');
+ };
+
+ mediaRecorder.onwarning = function () {
+ ok(false, 'Unexpected onwarning callback fired');
+ };
+
+ mediaRecorder.onstop = function () {
+ ok(false, 'Unexpected onstop callback fired');
+ };
+
+ var dataAvailableCount = 0;
+ var onDataAvailableFirst = false;
+
+ // This handler fires every 250ms to generate a blob.
+ mediaRecorder.ondataavailable = function (evt) {
+ info('ondataavailable fired');
+ dataAvailableCount++;
+
+ ok(evt instanceof BlobEvent,
+ 'Events fired from ondataavailable should be BlobEvent');
+ is(evt.type, 'dataavailable',
+ 'Event type should dataavailable');
+ ok(evt.data.size >= 0,
+ 'Blob data size received is greater than or equal to zero');
+
+ is(evt.data.type, expectedMimeType,
+ 'Blob data received should have type = ' + expectedMimeType);
+ is(mediaRecorder.mimeType, expectedMimeType,
+ 'Mime type in ondataavailable = ' + mediaRecorder.mimeType);
+
+ // We'll stop recording upon the 1st blob being received
+ if (dataAvailableCount === 1) {
+ mediaRecorder.onstop = function (event) {
+ info('onstop fired');
+
+ if (!onDataAvailableFirst) {
+ ok(false, 'onstop unexpectedly fired before ondataavailable');
+ }
+ element.pause();
+ manager.finished(token);
+ };
+
+ mediaRecorder.stop();
+ is(mediaRecorder.state, 'inactive',
+ 'Media recorder is inactive after being stopped');
+ is(mediaRecorder.stream, stream,
+ 'Media recorder stream = element stream post recording');
+
+ } else if (dataAvailableCount === 2) {
+ // Ensure we've received at least two ondataavailable events before onstop
+ onDataAvailableFirst = true;
+ }
+ };
+
+ mediaRecorder.start(1);
+ element.play();
+ is(mediaRecorder.state, 'recording', 'Media recorder should be recording');
+ is(mediaRecorder.stream, stream,
+ 'Media recorder stream = element stream at the start of recording');
+ };
+}
+
+manager.runTests(gMediaRecorderTests, startTest);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_record_upsize_resolution.html b/dom/media/test/test_mediarecorder_record_upsize_resolution.html
new file mode 100644
index 0000000000..d02fd08e44
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_record_upsize_resolution.html
@@ -0,0 +1,148 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test MediaRecorder Recording canvas dynamically changes to greater resolution</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script src="/tests/dom/canvas/test/captureStream_common.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<pre id="test">
+<div id="content">
+</div>
+<script class="testbody" type="text/javascript">
+
+function startTest() {
+ var canvas = document.createElement("canvas");
+ var canvas_size = 100;
+ var new_canvas_size = 150;
+ canvas.width = canvas.height = canvas_size;
+
+ var helper = new CaptureStreamTestHelper2D(canvas_size, canvas_size);
+ helper.drawColor(canvas, helper.red);
+
+ // The recorded stream coming from canvas.
+ var stream = canvas.captureStream();
+
+ // Check values for events
+ var numDataAvailabledRaised = 0;
+ var numResizeRaised = 0;
+ // Recorded data that will be playback.
+ var blob;
+
+ // Media recorder for VP8 and canvas stream.
+ var mediaRecorder = new MediaRecorder(stream);
+ is(mediaRecorder.stream, stream,
+ "Media recorder stream = canvas stream at the beginning of recording");
+
+ // Not expected events.
+ mediaRecorder.onwarning = () => ok(false, "MediaRecorder: onwarning unexpectedly fired");
+ mediaRecorder.onerror = err => {
+ ok(false, "MediaRecorder: onerror unexpectedly fired. Code " + err.name);
+ SimpleTest.finish();
+ }
+
+ // When recorder is stopped get recorded data.
+ mediaRecorder.ondataavailable = ev => {
+ info("Got 'dataavailable' event");
+ ++numDataAvailabledRaised;
+ is(blob, undefined, "On dataavailable event blob is undefined");
+ // Save recorded data for playback
+ blob = ev.data;
+ };
+
+ mediaRecorder.onstart = () => {
+ info('onstart fired successfully');
+ };
+
+ mediaRecorder.onstop = () => {
+ info("Got 'stop' event");
+ is(numDataAvailabledRaised, 1, "Expected 1 dataavailable event");
+
+ // Playback stream and verify resolution changes.
+ ok(blob, "Should have gotten a data blob");
+ var video = document.createElement("video");
+ video.id = "recorded-video";
+ video.src = URL.createObjectURL(blob);
+ video.onerror = err => {
+ ok(false, "Should be able to play the recording. Got error. code=" + video.error.code);
+ SimpleTest.finish();
+ };
+
+ // Check here that resize is correct in the playback stream.
+ video.onresize = function() {
+ ++numResizeRaised;
+ if (numResizeRaised == 1) {
+ is(this.videoWidth, canvas_size, "1st resize event original width");
+ is(this.videoHeight, canvas_size, "1st resize event original height ");
+ } else if (numResizeRaised == 2) {
+ is(this.videoWidth, new_canvas_size, "2nd resize event new width");
+ is(this.videoHeight, new_canvas_size, "2nd resize event new height");
+ } else {
+ ok(false, "Only 2 numResizeRaised events are expected");
+ }
+ };
+
+ video.onended = () => {
+ is(numResizeRaised, 2, "Expected 2 resize event");
+ };
+ document.getElementById("content").appendChild(video);
+ video.play();
+
+ // Check last color
+ helper.pixelMustBecome(video, helper.red, {
+ threshold: 128,
+ infoString: "Should become red",
+ }).then(() => {
+ video.onresize = {};
+ video.onended = {};
+ SimpleTest.finish();
+ });
+ };
+
+ // Start here by stream recorder.
+ mediaRecorder.start();
+ is(mediaRecorder.state, "recording", "Media recorder started");
+ requestAnimationFrame(draw);
+
+ // Change resolution every 100 ms
+ var countFrames=0;
+ var previous_time = performance.now();
+ function draw(timestamp) {
+ if (timestamp - previous_time < 100) {
+ requestAnimationFrame(draw);
+ return;
+ }
+ previous_time = timestamp;
+
+ var size = 0;
+ var color = "";
+ if (countFrames < 1) {
+ // Initial size
+ size = canvas_size;
+ color = helper.blue;
+ } else if (countFrames < 2) {
+ // upsize
+ size = new_canvas_size;
+ color = helper.red;
+ }else {
+ // Stop recoredr on last frame
+ mediaRecorder.stop();
+ return;
+ }
+ // Resize and draw canvas
+ canvas.width = canvas.height = size;
+ helper.drawColor(canvas, color);
+ // Register next draw on every call
+ requestAnimationFrame(draw);
+ countFrames++;
+ }
+}
+
+SimpleTest.waitForExplicitFinish();
+startTest();
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_reload_crash.html b/dom/media/test/test_mediarecorder_reload_crash.html
new file mode 100644
index 0000000000..f6d008261f
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_reload_crash.html
@@ -0,0 +1,29 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test that reloading media recorder object</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=894348">Mozill
+a Bug 894348</a>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+ for (let i = 0; i< 5; i++) {
+ /* eslint-disable no-undef */
+ try { o0 = document.createElement('audio') } catch(e) { }
+ try { o0.src = "sound.ogg" } catch(e) { }
+ try { (document.body || document.documentElement).appendChild(o0) } catch(e) { }
+ try { o1 = o0.mozCaptureStreamUntilEnded(); } catch(e) { }
+ try { o2 = new MediaRecorder(o1) } catch(e) { }
+ try { o2.start(0) } catch(e) { }
+ /* eslint-enable no-undef */
+ SpecialPowers.gc();
+ }
+ ok(true, "pass the crash test");
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_state_event_order.html b/dom/media/test/test_mediarecorder_state_event_order.html
new file mode 100644
index 0000000000..feba055f4d
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_state_event_order.html
@@ -0,0 +1,83 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test MediaRecorder fires an event after changing state</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var manager = new MediaTestManager;
+
+/**
+ * The flow of test is start()=>pause()=>resume()=>stop(). In each steps,
+ * this test verifies whether each MediaRecorder methods properly change
+ * its state before firing an event. Checking the state is done in the
+ * corresponding event handlers.
+ */
+function startTest(test, token) {
+ var element = document.createElement('audio');
+
+ element.token = token;
+ manager.started(token);
+
+ element.src = test.name;
+ element.test = test;
+ element.stream = element.mozCaptureStream();
+
+ var mediaRecorder = new MediaRecorder(element.stream);
+
+ mediaRecorder.onwarning = function() {
+ ok(false, 'onwarning unexpectedly fired');
+ };
+
+ mediaRecorder.onerror = function() {
+ ok(false, 'onerror unexpectedly fired');
+ };
+
+ mediaRecorder.onstart = function() {
+ info('onstart fired successfully');
+ is(mediaRecorder.state, 'recording',
+ 'Media Recorder changes state to recording before firing a start event');
+ mediaRecorder.pause();
+ };
+
+ mediaRecorder.onpause = function() {
+ info('onpause fired successfully');
+ is(mediaRecorder.state, 'paused',
+ 'Media Recorder changes state to paused before firing a pause event');
+ mediaRecorder.resume();
+ };
+
+ mediaRecorder.onresume = function() {
+ info('onresume fired successfully');
+ is(mediaRecorder.state, 'recording',
+ 'Media Recorder changes state to recording before firing a resume event');
+ mediaRecorder.stop();
+ };
+
+ mediaRecorder.onstop = function() {
+ info('onstop fired successfully');
+ is(mediaRecorder.state, 'inactive',
+ 'Media Recorder changes state to inactive before firing a stop event');
+ SimpleTest.finish();
+ };
+
+ // Call start() once metadata are parsed.
+ element.onloadedmetadata = function() {
+ element.play();
+ mediaRecorder.start();
+ is(mediaRecorder.state, 'recording', 'Media recorder should be recording');
+ is(mediaRecorder.stream, element.stream,
+ 'Media recorder stream = element stream at the start of recording');
+ };
+}
+
+manager.runTests(gMediaRecorderTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_state_transition.html b/dom/media/test/test_mediarecorder_state_transition.html
new file mode 100644
index 0000000000..5d6483b7de
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_state_transition.html
@@ -0,0 +1,280 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test MediaRecorder State Transition</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var manager = new MediaTestManager;
+
+// List of operation tests for media recorder objects to verify if running
+// these operations should result in an exception or not
+var operationTests = [
+ {
+ operations: ['stop'],
+ isValid: true
+ },
+ {
+ operations: ['requestData'],
+ isValid: false
+ },
+ {
+ operations: ['pause'],
+ isValid: false
+ },
+ {
+ operations: ['resume'],
+ isValid: false
+ },
+ {
+ operations: ['start'],
+ isValid: true
+ },
+ {
+ operations: ['start'],
+ isValid: true,
+ timeSlice: 200
+ },
+ {
+ operations: ['start', 'pause'],
+ isValid: true
+ },
+ {
+ operations: ['start', 'pause'],
+ isValid: true,
+ timeSlice: 200
+ },
+ {
+ operations: ['start', 'start'],
+ isValid: false
+ },
+ {
+ operations: ['start', 'resume'],
+ isValid: true
+ },
+ {
+ operations: ['pause', 'start'],
+ isValid: false
+ },
+ {
+ operations: ['resume', 'start'],
+ isValid: false
+ },
+ {
+ operations: ['requestData', 'start'],
+ isValid: false
+ },
+ {
+ operations: ['stop', 'start'],
+ isValid: true
+ },
+ {
+ operations: ['start', 'stop'],
+ isValid: true
+ },
+ {
+ operations: ['start', 'stop'],
+ isValid: true,
+ timeSlice: 200
+ },
+ {
+ operations: ['start', 'requestData'],
+ isValid: true
+ },
+ {
+ operations: ['start', 'requestData'],
+ isValid: true,
+ timeSlice: 200
+ },
+ {
+ operations: ['start', 'pause', 'stop'],
+ isValid: true
+ },
+ {
+ operations: ['start', 'pause', 'start'],
+ isValid: false
+ },
+ {
+ operations: ['start', 'pause', 'pause'],
+ isValid: true
+ },
+ {
+ operations: ['start', 'pause', 'requestData'],
+ isValid: true
+ },
+ {
+ operations: ['start', 'pause', 'resume'],
+ isValid: true
+ },
+ {
+ operations: ['start', 'pause', 'resume'],
+ isValid: true,
+ timeSlice: 200
+ },
+ {
+ operations: ['start', 'resume', 'resume'],
+ isValid: true
+ },
+ {
+ operations: ['start', 'resume', 'resume'],
+ isValid: true,
+ timeSlice: 200
+ },
+ {
+ operations: ['start', 'resume', 'stop'],
+ isValid: true
+ },
+ {
+ operations: ['start', 'resume', 'stop'],
+ isValid: true,
+ timeSlice: 200
+ },
+ {
+ operations: ['start', 'stop', 'start'],
+ isValid: true
+ },
+ {
+ operations: ['start', 'stop', 'start'],
+ isValid: true,
+ timeSlice: 200
+ },
+ {
+ operations: ['start', 'stop', 'pause'],
+ isValid: false
+ },
+ {
+ operations: ['start', 'stop', 'resume'],
+ isValid: false
+ },
+ {
+ operations: ['start', 'stop', 'requestData'],
+ isValid: false
+ },
+ {
+ operations: ['start', 'stop', 'stop'],
+ isValid: true
+ },
+ {
+ operations: ['start', 'pause', 'resume', 'resume'],
+ isValid: true
+ },
+ {
+ operations: ['start', 'pause', 'resume', 'resume'],
+ isValid: true,
+ timeSlice: 200
+ },
+ {
+ operations: ['start', 'pause', 'pause', 'resume'],
+ isValid: true
+ },
+ {
+ operations: ['start', 'pause', 'pause', 'resume'],
+ isValid: true,
+ timeSlice: 200
+ },
+ {
+ operations: ['start', 'stop', 'start', 'stop'],
+ isValid: true
+ },
+ {
+ operations: ['start', 'stop', 'start', 'stop'],
+ isValid: true,
+ timeSlice: 200
+ },
+ {
+ operations: ['start', 'stop', 'start', 'pause'],
+ isValid: true
+ },
+ {
+ operations: ['start', 'stop', 'start', 'pause'],
+ isValid: true,
+ timeSlice: 200
+ },
+ {
+ operations: ['start', 'stop', 'start', 'resume'],
+ isValid: true
+ },
+ {
+ operations: ['start', 'stop', 'start', 'resume'],
+ isValid: true,
+ timeSlice: 200
+ },
+ {
+ operations: ['start', 'stop', 'start', 'requestData'],
+ isValid: true
+ },
+ {
+ operations: ['start', 'stop', 'start', 'requestData'],
+ isValid: true,
+ timeSlice: 200
+ },
+];
+
+/**
+ * Runs through each available state transition test by running all
+ * available operations on a media recorder object. Then, we report
+ * back if the test was expected through an exception or not.
+ *
+ * @param {MediaStream} testStream the media stream used for media recorder
+ * operation tests
+ */
+function runStateTransitionTests(testStream) {
+ for (const operationTest of operationTests) {
+ var mediaRecorder = new MediaRecorder(testStream);
+ var operationsString = operationTest.operations.toString();
+
+ try {
+ for (const operation of operationTest.operations) {
+ if (operationTest.timeSlice && operation === 'start') {
+ operationsString += ' with timeslice ' + operationTest.timeSlice;
+ mediaRecorder[operation](operationTest.timeSlice);
+ } else {
+ mediaRecorder[operation]();
+ }
+ }
+
+ ok(operationTest.isValid, `${operationsString} should succeed`);
+ } catch (err) {
+ if (operationTest.isValid) {
+ ok(false, `${operationsString} failed unexpectedly with ${err.name}`);
+ } else {
+ is(err.name, "InvalidStateError",
+ `${operationsString} expected to fail with InvalidStateError`);
+ }
+ }
+ }
+}
+
+/**
+ * Starts a test on every media recorder file included to check that various
+ * state transition flows that can happen in the media recorder object throw
+ * exceptions when they are expected to and vice versa.
+ */
+function startTest(test, token) {
+ var element = document.createElement('audio');
+
+ element.token = token;
+ manager.started(token);
+
+ element.src = test.name;
+ element.test = test;
+ element.stream = element.mozCaptureStream();
+
+ element.oncanplaythrough = function () {
+ element.oncanplaythrough = null;
+ runStateTransitionTests(element.stream);
+ manager.finished(token);
+ };
+
+ element.play();
+}
+
+manager.runTests(gMediaRecorderTests, startTest);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediarecorder_webm_support.html b/dom/media/test/test_mediarecorder_webm_support.html
new file mode 100644
index 0000000000..6b115ee33a
--- /dev/null
+++ b/dom/media/test/test_mediarecorder_webm_support.html
@@ -0,0 +1,56 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media Recording - test WebM MIME support</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+ok(MediaRecorder.isTypeSupported('audio/webm'),
+ 'Should support audio/webm');
+ok(MediaRecorder.isTypeSupported('AUDIO/WEBM'),
+ 'Should support audio/webm, upper case');
+ok(MediaRecorder.isTypeSupported('AuDiO/wEbM'),
+ 'Should support audio/webm, mixed case');
+
+ok(MediaRecorder.isTypeSupported('audio/webm;codecs=opus'),
+ 'Should support audio/webm;codecs=opus');
+ok(MediaRecorder.isTypeSupported('AUDIO/WEBM;CODECS=opus'),
+ 'Should support audio/webm;codecs=opus, upper case');
+ok(MediaRecorder.isTypeSupported('AuDiO/wEbM;cOdEcS=opus'),
+ 'Should support audio/webm;codecs=opus, mixed case');
+
+ok(MediaRecorder.isTypeSupported('video/webm'),
+ 'Should support video/webm');
+ok(MediaRecorder.isTypeSupported('VIDEO/WEBM'),
+ 'Should support video/webm, upper case');
+ok(MediaRecorder.isTypeSupported('vIdEo/WeBm'),
+ 'Should support video/webm, mixed case');
+
+ok(MediaRecorder.isTypeSupported('video/webm; codecs="vp8"'),
+ 'Should support video/webm; codecs="vp8"');
+ok(MediaRecorder.isTypeSupported('VIDEO/WEBM; CODECS="vp8"'),
+ 'Should support video/webm; codecs="vp8", upper case');
+ok(MediaRecorder.isTypeSupported('vIdEo/WeBm; CoDeCs="vp8"'),
+ 'Should support video/webm; codecs="vp8", mixed case');
+
+ok(MediaRecorder.isTypeSupported('video/webm; codecs="vp8.0"'),
+ 'Should support video/webm; codecs="vp8.0"');
+ok(MediaRecorder.isTypeSupported('VIDEO/WEBM; CODECS="vp8.0"'),
+ 'Should support video/webm; codecs="vp8.0", upper case');
+ok(MediaRecorder.isTypeSupported('vIdEo/WeBm; CoDeCs="vp8.0"'),
+ 'Should support video/webm; codecs="vp8.0", mixed case');
+
+ok(!MediaRecorder.isTypeSupported('video/webm; codecs="vp8, vorbis"'),
+ 'Should not support video/webm + vp8/vorbis');
+ok(!MediaRecorder.isTypeSupported('video/webm; codecs="vp9, vorbis"'),
+ 'Should not support video/webm + vp9/vorbis');
+ok(MediaRecorder.isTypeSupported('video/webm; codecs="vp8, opus"'),
+ 'Should support video/webm + vp8/opus');
+ok(!MediaRecorder.isTypeSupported('video/webm; codecs="vp9, opus"'),
+ 'Should not support video/webm + vp9/opus');
+</script>
+</head>
+</html>
diff --git a/dom/media/test/test_mediastream_as_eventarget.html b/dom/media/test/test_mediastream_as_eventarget.html
new file mode 100644
index 0000000000..f09c277583
--- /dev/null
+++ b/dom/media/test/test_mediastream_as_eventarget.html
@@ -0,0 +1,33 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <meta charset="utf-8">
+ <title>Test MediaStream as EventTarget</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" href="/tests/SimpleTest/test.css"/>
+ <script>
+ var ms = new MediaStream();
+ var listenerForFooWasCalled = false;
+ ms.addEventListener("foo", () => { listenerForFooWasCalled = true; });
+ var listenerForAddTrackWasCalled = false;
+ ms.addEventListener("addtrack", () => { listenerForAddTrackWasCalled = true; });
+ var handlerForAddTrackWasCalled = false;
+ ms.onaddtrack = () => { handlerForAddTrackWasCalled = true; };
+
+ ms.dispatchEvent(new Event("foo"));
+ ms.dispatchEvent(new Event("addtrack"));
+
+ ok(listenerForFooWasCalled,
+ "Should have called the event listener for 'foo'");
+ ok(listenerForAddTrackWasCalled,
+ "Should have called the event listener for 'addtrack'");
+ ok(handlerForAddTrackWasCalled,
+ "Should have called the event handler for 'addtrack'")
+ </script>
+</head>
+<body>
+<p id="display"></p>
+<div id="content" style="display: none"></div>
+<pre id="test"></pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediatrack_consuming_mediaresource.html b/dom/media/test/test_mediatrack_consuming_mediaresource.html
new file mode 100644
index 0000000000..515df5c053
--- /dev/null
+++ b/dom/media/test/test_mediatrack_consuming_mediaresource.html
@@ -0,0 +1,198 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test track interfaces when consuming media resources</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+const manager = new MediaTestManager;
+
+function startTest(test, token) {
+ const elemType = getMajorMimeType(test.type);
+ const element = document.createElement(elemType);
+
+ let audioOnchange = 0;
+ let audioOnaddtrack = 0;
+ let audioOnremovetrack = 0;
+ let videoOnchange = 0;
+ let videoOnaddtrack = 0;
+ let videoOnremovetrack = 0;
+ let isPlaying = false;
+
+ isnot(element.audioTracks, undefined,
+ 'HTMLMediaElement::AudioTracks() property should be available.');
+ isnot(element.videoTracks, undefined,
+ 'HTMLMediaElement::VideoTracks() property should be available.');
+
+ element.audioTracks.onaddtrack = function(e) {
+ audioOnaddtrack++;
+ }
+
+ element.audioTracks.onremovetrack = function(e) {
+ audioOnremovetrack++;
+ }
+
+ element.audioTracks.onchange = function(e) {
+ audioOnchange++;
+ }
+
+ element.videoTracks.onaddtrack = function(e) {
+ videoOnaddtrack++;
+ }
+
+ element.videoTracks.onremovetrack = function(e) {
+ videoOnremovetrack++;
+ }
+
+ element.videoTracks.onchange = function(e) {
+ videoOnchange++;
+ }
+
+ function checkTrackNotRemoved() {
+ is(audioOnremovetrack, 0, 'Should have no calls of onremovetrack on audioTracks.');
+ is(videoOnremovetrack, 0, 'Should have no calls of onremovetrack on videoTracks.');
+ if (isPlaying) {
+ is(element.audioTracks.length, test.hasAudio ? 1 : 0,
+ 'Expected length of audioTracks.');
+ is(element.videoTracks.length, test.hasVideo ? 1 : 0,
+ 'Expected length of videoTracks.');
+ }
+ }
+
+ function checkTrackRemoved() {
+ is(element.audioTracks.length, 0, 'The length of audioTracks should be 0.');
+ is(element.videoTracks.length, 0, 'The length of videoTracks should be 0.');
+ if (isPlaying) {
+ is(audioOnremovetrack, test.hasAudio ? 1 : 0,
+ 'Expected calls of onremovetrack on audioTracks.');
+ is(videoOnremovetrack, test.hasVideo ? 1 : 0,
+ 'Expected calls of onremovetrack on videoTracks.');
+ }
+ }
+
+ function onended() {
+ ok(true, 'Event ended is expected to be fired on element.');
+ checkTrackNotRemoved();
+ element.onended = null;
+ element.onplaying = null;
+ element.onpause = null;
+ element.src = "";
+ is(element.audioTracks.length, 0, 'audioTracks have been forgotten');
+ is(element.videoTracks.length, 0, 'videoTracks have been forgotten');
+ is(audioOnremovetrack, 0, 'No audio removetrack events yet');
+ is(videoOnremovetrack, 0, 'No video removetrack events yet');
+ setTimeout(() => {
+ checkTrackRemoved();
+ manager.finished(element.token);
+ }, 100);
+ }
+
+ function checkTrackAdded() {
+ isPlaying = true;
+ if (test.hasAudio) {
+ is(audioOnaddtrack, 1, 'Calls of onaddtrack on audioTracks should be 1.');
+ is(element.audioTracks.length, 1, 'The length of audioTracks should be 1.');
+ ok(element.audioTracks[0].enabled, 'Audio track should be enabled as default.');
+ }
+ if (test.hasVideo) {
+ is(videoOnaddtrack, 1, 'Calls of onaddtrack on videoTracks should be 1.');
+ is(element.videoTracks.length, 1, 'The length of videoTracks should be 1.');
+ is(element.videoTracks.selectedIndex, 0,
+ 'The first video track is set selected as default.');
+ }
+ }
+
+ function setTrackEnabled(enabled) {
+ if (test.hasAudio) {
+ element.audioTracks[0].enabled = enabled;
+ }
+ if (test.hasVideo) {
+ element.videoTracks[0].selected = enabled;
+ }
+ }
+
+ function checkTrackChanged(calls, enabled) {
+ if (test.hasAudio) {
+ is(audioOnchange, calls, 'Calls of onchange on audioTracks should be '+calls);
+ is(element.audioTracks[0].enabled, enabled,
+ 'Enabled value of the audio track should be ' +enabled);
+ }
+ if (test.hasVideo) {
+ is(videoOnchange, calls, 'Calls of onchange on videoTracks should be '+calls);
+ is(element.videoTracks[0].selected, enabled,
+ 'Selected value of the video track should be ' +enabled);
+ var index = enabled ? 0 : -1;
+ is(element.videoTracks.selectedIndex, index,
+ 'SelectedIndex of video tracks should be ' +index);
+ }
+ }
+
+ function onpause() {
+ element.onpause = null;
+ if (element.ended) {
+ return;
+ }
+ if (steps == 1) {
+ setTrackEnabled(false);
+ element.onplaying = onplaying;
+ element.play();
+ steps++;
+ } else if (steps == 2) {
+ setTrackEnabled(true);
+ element.onplaying = onplaying;
+ element.play();
+ steps++;
+ }
+ }
+
+ function onplaying() {
+ element.onplaying = null;
+ if (element.ended) {
+ return;
+ }
+ if (steps == 1) {
+ element.onpause = onpause;
+ element.pause();
+ checkTrackAdded();
+ } else if (steps == 2) {
+ element.onpause = onpause;
+ element.pause();
+ checkTrackChanged(1, false);
+ } else if (steps == 3) {
+ checkTrackChanged(2, true);
+ }
+ }
+
+ var steps = 0;
+
+ element.token = token;
+ manager.started(token);
+
+ element.src = test.name;
+ element.test = test;
+ element.onplaying = onplaying;
+ element.onended = onended;
+ element.play();
+ steps++;
+}
+
+SimpleTest.waitForExplicitFinish();
+SpecialPowers.pushPrefEnv(
+ {
+ "set": [
+ ["media.track.enabled", true]
+ ]
+ },
+ function() {
+ manager.runTests(gTrackTests, startTest);
+ });
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediatrack_consuming_mediastream.html b/dom/media/test/test_mediatrack_consuming_mediastream.html
new file mode 100644
index 0000000000..b930ca4fdc
--- /dev/null
+++ b/dom/media/test/test_mediatrack_consuming_mediastream.html
@@ -0,0 +1,146 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test track interfaces when consuming a MediaStream from gUM</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="gUM_support.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+async function startTest() {
+ let steps = 0;
+ let audioOnchange = 0;
+ let audioOnaddtrack = 0;
+ let videoOnchange = 0;
+ let videoOnaddtrack = 0;
+ let isPlaying = false;
+ let element = document.createElement("video");
+ let stream;
+ try {
+ await setupGetUserMediaTestPrefs();
+ stream = await navigator.mediaDevices.getUserMedia({audio: true, video: true});
+ } catch (err) {
+ ok(false, 'Unexpected error fired with: ' + err);
+ SimpleTest.finish();
+ return;
+ }
+
+ element.audioTracks.onaddtrack = function(e) {
+ audioOnaddtrack++;
+ };
+
+ element.audioTracks.onchange = function(e) {
+ audioOnchange++;
+ };
+
+ element.videoTracks.onaddtrack = function(e) {
+ videoOnaddtrack++;
+ };
+
+ element.videoTracks.onchange = function(e) {
+ videoOnchange++;
+ };
+
+ function checkTrackRemoved() {
+ if (isPlaying) {
+ is(element.audioTracks.length, 0, 'The length of audioTracks should be 0.');
+ is(element.videoTracks.length, 0, 'The length of videoTracks should be 0.');
+ }
+ }
+
+ element.onended = function() {
+ ok(true, 'Event ended is expected to be fired on element.');
+ checkTrackRemoved();
+ element.onended = null;
+ element.onplaying = null;
+ element.onpause = null;
+ SimpleTest.finish();
+ }
+
+ function checkTrackAdded() {
+ isPlaying = true;
+ is(audioOnaddtrack, 1, 'Calls of onaddtrack on audioTracks should be 1.');
+ is(element.audioTracks.length, 1, 'The length of audioTracks should be 1.');
+ ok(element.audioTracks[0].enabled, 'Audio track should be enabled as default.');
+ is(videoOnaddtrack, 1, 'Calls of onaddtrack on videoTracks should be 1.');
+ is(element.videoTracks.length, 1, 'The length of videoTracks should be 1.');
+ is(element.videoTracks.selectedIndex, 0,
+ 'The first video track is set selected as default.');
+ }
+
+ function setTrackEnabled(enabled) {
+ element.audioTracks[0].enabled = enabled;
+ element.videoTracks[0].selected = enabled;
+ }
+
+ function checkTrackChanged(calls, enabled) {
+ is(audioOnchange, calls, 'Calls of onchange on audioTracks should be '+calls);
+ is(element.audioTracks[0].enabled, enabled,
+ 'Enabled value of the audio track should be ' +enabled);
+ is(videoOnchange, calls, 'Calls of onchange on videoTracks should be '+calls);
+ is(element.videoTracks[0].selected, enabled,
+ 'Selected value of the video track should be ' +enabled);
+ var index = enabled ? 0 : -1;
+ is(element.videoTracks.selectedIndex, index,
+ 'SelectedIndex of video tracks should be ' +index);
+ }
+
+ function onpause() {
+ element.onpause = null;
+ if (element.ended) {
+ return;
+ }
+ if (steps == 1) {
+ setTrackEnabled(false);
+ element.onplaying = onplaying;
+ element.play();
+ steps++;
+ } else if (steps == 2) {
+ setTrackEnabled(true);
+ element.onplaying = onplaying;
+ element.play();
+ steps++;
+ }
+ }
+
+ function onplaying() {
+ element.onplaying = null;
+ if (element.ended) {
+ return;
+ }
+ if (steps == 1) {
+ element.onpause = onpause;
+ element.pause();
+ checkTrackAdded();
+ } else if (steps == 2) {
+ element.onpause = onpause;
+ element.pause();
+ checkTrackChanged(1, false);
+ } else if (steps == 3) {
+ checkTrackChanged(2, true);
+ stream.getTracks().forEach(t => t.stop());
+ }
+ }
+
+ element.onplaying = onplaying;
+ element.srcObject = stream;
+
+ steps++;
+ await element.play();
+}
+
+SimpleTest.waitForExplicitFinish();
+SpecialPowers.pushPrefEnv(
+ {
+ "set": [
+ ["media.track.enabled", true]
+ ]
+ }, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediatrack_events.html b/dom/media/test/test_mediatrack_events.html
new file mode 100644
index 0000000000..5eae94f804
--- /dev/null
+++ b/dom/media/test/test_mediatrack_events.html
@@ -0,0 +1,135 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test events of media track interfaces</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="gUM_support.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+async function startTest() {
+ let steps = 0;
+ let element = document.createElement("video");
+ let stream;
+ try {
+ await setupGetUserMediaTestPrefs();
+ stream = await navigator.mediaDevices.getUserMedia({audio: true, video: true});
+ } catch (err) {
+ ok(false, 'Unexpected error fired with: ' + err);
+ SimpleTest.finish();
+ return;
+ }
+
+ function verifyEvent(e, type) {
+ is(e.type, type, "Event type should be " + type);
+ ok(e.isTrusted, "Event should be trusted.");
+ ok(!e.bubbles, "Event shouldn't bubble.");
+ ok(!e.cancelable, "Event shouldn't be cancelable.");
+ }
+
+ element.audioTracks.onaddtrack = function(e) {
+ ok(e instanceof TrackEvent, "Event fired from onaddtrack should be a TrackEvent");
+ ok(true, 'onaddtrack is expected to be called from audioTracks.');
+ verifyEvent(e, "addtrack");
+ };
+
+ element.audioTracks.onremovetrack = function(e) {
+ ok(e instanceof TrackEvent, "Event fired from onremovetrack should be a TrackEvent");
+ ok(true, 'onremovetrack is expected to be called from audioTracks.');
+ verifyEvent(e, "removetrack");
+ };
+
+ element.audioTracks.onchange = function(e) {
+ ok(e instanceof window.Event, "Event fired from onchange should be a simple event.");
+ ok(true, 'onchange is expected to be called from audioTracks.');
+ verifyEvent(e, "change");
+ };
+
+ element.videoTracks.onaddtrack = function(e) {
+ ok(e instanceof TrackEvent, "Event fired from onaddtrack should be a TrackEvent");
+ ok(true, 'onaddtrack is expected to be called from videoTracks.');
+ verifyEvent(e, "addtrack");
+ };
+
+ element.videoTracks.onremovetrack = function(e) {
+ ok(e instanceof TrackEvent, "Event fired from onremovetrack should be a TrackEvent");
+ ok(true, 'onremovetrack is expected to be called from videoTracks.');
+ verifyEvent(e, "removetrack");
+ };
+
+ element.videoTracks.onchange = function(e) {
+ ok(e instanceof window.Event, "Event fired from onchange should be a simple event.");
+ ok(true, 'onchange is expected to be called from videoTracks.');
+ verifyEvent(e, "change");
+ };
+
+ element.onended = function() {
+ ok(true, 'Event ended is expected to be fired on element.');
+ element.onended = null;
+ element.onplaying = null;
+ element.onpause = null;
+ //This helps to prevent these events from firing after SimpleTest.finish()
+ //on B2G ICS Emulator, but not sure they have been run at all, then
+ element.audioTracks.onremovetrack = null;
+ element.audioTracks.onaddtrack = null;
+ element.audioTracks.onchange = null;
+ element.videoTracks.onremovetrack = null;
+ element.videoTracks.onaddtrack = null;
+ element.videoTracks.onchange = null;
+ SimpleTest.finish();
+ }
+
+ function onpause() {
+ element.onpause = null;
+ if (element.ended) {
+ return;
+ }
+ if (steps == 1) {
+ element.audioTracks[0].enabled = false;
+ element.videoTracks[0].selected = false;
+ element.onplaying = onplaying;
+ element.play();
+ steps++;
+ }
+ }
+
+ function onplaying() {
+ element.onplaying = null;
+ if (element.ended) {
+ return;
+ }
+ if (steps == 1) {
+ element.onpause = onpause;
+ element.pause();
+ } else if (steps == 2) {
+ stream.getTracks().forEach(t => t.stop());
+ }
+ }
+
+ element.onplaying = onplaying;
+ element.srcObject = stream;
+
+ isnot(element.audioTracks, undefined,
+ 'HTMLMediaElement::AudioTracks() property should be available.');
+ isnot(element.videoTracks, undefined,
+ 'HTMLMediaElement::VideoTracks() property should be available.');
+
+ steps++;
+ await element.play();
+}
+
+SimpleTest.waitForExplicitFinish();
+SpecialPowers.pushPrefEnv(
+ {
+ "set": [
+ ["media.track.enabled", true]
+ ]
+ }, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediatrack_parsing_ogg.html b/dom/media/test/test_mediatrack_parsing_ogg.html
new file mode 100644
index 0000000000..aabd40e2a3
--- /dev/null
+++ b/dom/media/test/test_mediatrack_parsing_ogg.html
@@ -0,0 +1,72 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test events of media track interfaces</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function localCheckMetadata(msg, e) {
+ ok(msg in gOggTrackInfoResults, "File: " + msg + " is in pre-parsed gOggTrackInfoResults list");
+ var r = gOggTrackInfoResults[msg];
+
+ var hasExpectedAudio = r && r.hasOwnProperty("audio_id");
+ var hasExpectedVideo = r && r.hasOwnProperty("video_id");
+
+ var hasParsedAudio = e.audioTracks.length >= 1;
+ var hasParsedVideo = e.videoTracks.length >= 1;
+
+ ok(!(hasExpectedAudio ^ hasParsedAudio), "Check availability of expected/parsed audio");
+ ok(!(hasExpectedVideo ^ hasParsedVideo), "Check availability of expected/parsed video");
+ if (hasParsedAudio) {
+ is(e.audioTracks.length, 1, "The length of audio track should be 1");
+ is(e.audioTracks[0].id, r.audio_id, "File: " + msg + ", Audio track id");
+ is(e.audioTracks[0].kind, r.audio_kind, "File: " + msg + ", Audio track kind");
+ is(e.audioTracks[0].language, r.audio_language, "File: " + msg + ", Audio track language");
+ is(e.audioTracks[0].label, r.audio_label, "File: " + msg + ", Audio track label");
+ }
+ if (hasParsedVideo) {
+ is(e.videoTracks.length, 1, "The length of video track should be 1");
+ is(e.videoTracks[0].id, r.video_id, "File: " + msg + ", Video track id");
+ is(e.videoTracks[0].kind, r.video_kind, "File: " + msg + ", Video track kind");
+ is(e.videoTracks[0].language, r.video_language, "File: " + msg + ", Video track language");
+ is(e.videoTracks[0].label, r.video_label, "File: " + msg + ", Video track label");
+ }
+}
+
+function startTest(test, token) {
+ var v = document.createElement('video');
+ v.preload = "metadata";
+ v.token = token;
+ manager.started(token);
+
+ v.src = test.name;
+ v.name = test.name;
+
+ v.onloadedmetadata = function(evt) {
+ localCheckMetadata(evt.target.name, evt.target);
+ evt.target.finished = true;
+ evt.target.onloadedmetadata = null;
+ removeNodeAndSource(evt.target);
+ manager.finished(evt.target.token);
+ };
+
+ document.body.appendChild(v);
+}
+
+SimpleTest.waitForExplicitFinish();
+SpecialPowers.pushPrefEnv({"set": [["media.track.enabled", true]]},
+ function() {
+ manager.runTests(gMultitrackInfoOggPlayList, startTest);
+ }
+);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mediatrack_replay_from_end.html b/dom/media/test/test_mediatrack_replay_from_end.html
new file mode 100644
index 0000000000..16b0cbeb97
--- /dev/null
+++ b/dom/media/test/test_mediatrack_replay_from_end.html
@@ -0,0 +1,160 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test media tracks if replay after playback has ended</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+const manager = new MediaTestManager;
+
+function startTest(test, token) {
+ // Scenario to test:
+ // 1. Audio tracks and video tracks should be added to the track list when
+ // metadata has loaded, and all tracks should remain even after we seek to
+ // the end.
+ // 2. No tracks should be added back to the list if we replay from the end,
+ // and no tracks should be removed from the list after we seek to the end.
+ // 3. After seek to the middle from end of playback, all tracks should remain
+ // in the list if we play from here, and no tracks should be removed from
+ // the list after we seek to the end.
+ // 4. Unsetting the media element's src attribute should remove all tracks.
+
+ const elemType = getMajorMimeType(test.type);
+ const element = document.createElement(elemType);
+
+ let audioOnaddtrack = 0;
+ let audioOnremovetrack = 0;
+ let videoOnaddtrack = 0;
+ let videoOnremovetrack = 0;
+ let isPlaying = false;
+ let steps = 0;
+
+ element.audioTracks.onaddtrack = function(e) {
+ audioOnaddtrack++;
+ }
+
+ element.audioTracks.onremovetrack = function(e) {
+ audioOnremovetrack++;
+ }
+
+ element.videoTracks.onaddtrack = function(e) {
+ videoOnaddtrack++;
+ }
+
+ element.videoTracks.onremovetrack = function(e) {
+ videoOnremovetrack++;
+ }
+
+ function testExpectedAddtrack(expectedCalls) {
+ if (test.hasAudio) {
+ is(audioOnaddtrack, expectedCalls,
+ 'Calls of onaddtrack on audioTracks should be '+expectedCalls+' times.');
+ }
+ if (test.hasVideo) {
+ is(videoOnaddtrack, expectedCalls,
+ 'Calls of onaddtrack on videoTracks should be '+expectedCalls+' times.');
+ }
+ }
+
+ function testExpectedRemovetrack(expectedCalls) {
+ if (test.hasAudio) {
+ is(audioOnremovetrack, expectedCalls,
+ 'Calls of onremovetrack on audioTracks should be '+expectedCalls+' times.');
+ }
+ if (test.hasVideo) {
+ is(videoOnremovetrack, expectedCalls,
+ 'Calls of onremovetrack on videoTracks should be '+expectedCalls+' times.');
+ }
+ }
+
+ function finishTesting() {
+ element.onpause = null;
+ element.onseeked = null;
+ element.onplaying = null;
+ element.onended = null;
+ manager.finished(element.token);
+ }
+
+ function onended() {
+ if (isPlaying) {
+ switch(steps) {
+ case 1:
+ testExpectedAddtrack(1);
+ testExpectedRemovetrack(0);
+ element.onplaying = onplaying;
+ element.play();
+ steps++;
+ break;
+ case 2:
+ testExpectedAddtrack(1);
+ testExpectedRemovetrack(0);
+ element.currentTime = element.duration * 0.5;
+ element.onplaying = onplaying;
+ element.play();
+ steps++;
+ break;
+ case 3:
+ testExpectedAddtrack(1);
+ testExpectedRemovetrack(0);
+ element.src = "";
+ setTimeout(() => {
+ testExpectedAddtrack(1);
+ testExpectedRemovetrack(1);
+ finishTesting();
+ }, 0);
+ break;
+ }
+ } else {
+ ok(true, 'Finish the test anyway if ended is fired before other events.');
+ finishTesting();
+ }
+ }
+
+ function seekToEnd() {
+ element.onpause = null;
+ element.currentTime = element.duration * 1.1;
+ }
+
+ function onseeked() {
+ element.onseeked = null;
+ element.onpause = seekToEnd;
+ element.pause();
+ }
+
+ function onplaying() {
+ isPlaying = true;
+ element.onplaying = null;
+ element.onseeked = onseeked;
+ }
+
+ element.token = token;
+ manager.started(token);
+
+ element.src = test.name;
+ element.test = test;
+ element.onplaying = onplaying;
+ element.onended = onended;
+ element.play();
+ steps++;
+}
+
+SimpleTest.waitForExplicitFinish();
+SpecialPowers.pushPrefEnv(
+ {
+ "set": [
+ ["media.track.enabled", true]
+ ]
+ },
+ function() {
+ manager.runTests(gTrackTests, startTest);
+ });
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_metadata.html b/dom/media/test/test_metadata.html
new file mode 100644
index 0000000000..2fc6ddffec
--- /dev/null
+++ b/dom/media/test/test_metadata.html
@@ -0,0 +1,81 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <meta charset="utf-8">
+ <title>Test returning metadata from media files with mozGetMetadata()</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<div id="output"></div>
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function startTest(test, token) {
+ var a = document.createElement('audio');
+ a.preload = "metadata";
+ a.token = token;
+ manager.started(token);
+
+ a.src = test.name;
+ a.name = test.name;
+
+ // Tags should not be available immediately.
+ var exception_fired = false;
+ try {
+ a.mozGetMetadata();
+ } catch (e) {
+ is(e.name, 'InvalidStateError',
+ "early mozGetMetadata() should throw InvalidStateError");
+ exception_fired = true;
+ }
+ ok(exception_fired,
+ "mozGetMetadata() should throw an exception before HAVE_METADATA");
+
+ // Wait until metadata has loaded.
+ a.addEventListener('loadedmetadata', function() {
+ // read decoded tags
+ let tags = a.mozGetMetadata();
+ ok(tags, "mozGetMetadata() should return a truthy value");
+ // Dump them out.
+ var d = document.getElementById('output');
+ var html = '<table>\n';
+ html += '<caption><p>Called getMozMetadata()'
+ html += ' on '+test.name+'</p></caption>\n';
+ html += '<tr><th>tag</th>';
+ html += '<th>decoded value</th><th>expected value</th></tr>\n';
+ for (let tag in tags) {
+ html += '<tr><td>'+tag+'</td>';
+ html += '<td>'+tags[tag]+'</td>';
+ html += '<td>'+test.tags[tag]+'</td>';
+ html += '</tr>\n';
+ }
+ if (!Object.keys(tags).length) {
+ html += '<tr><td colspan=3 align=center><em>no tags</em></td></tr>\n';
+ }
+ html += '</table>\n';
+ var div = document.createElement('div');
+ div.innerHTML = html;
+ d.appendChild(div);
+ // Verify decoded tag values.
+ for (let tag in tags) {
+ is(tags[tag], test.tags[tag], "Tag '"+tag+"' should match");
+ }
+ // Verify expected tag values
+ for (let tag in test.tags) {
+ is(tags[tag], test.tags[tag], "Tag '"+tag+"' should match");
+ }
+ removeNodeAndSource(a);
+ manager.finished(token);
+ });
+}
+
+manager.runTests(gMetadataTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_midflight_redirect_blocked.html b/dom/media/test/test_midflight_redirect_blocked.html
new file mode 100644
index 0000000000..ea85673b45
--- /dev/null
+++ b/dom/media/test/test_midflight_redirect_blocked.html
@@ -0,0 +1,87 @@
+<!DOCTYPE HTML>
+<html>
+ <head>
+ <title>Test mid-flight cross site redirects are blocked</title>
+ <script type="text/javascript" src="/MochiKit/MochiKit.js"></script>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ </head>
+ <body>
+ <pre id='test'>
+ <script class="testbody" type='application/javascript'>
+
+ function testIfLoadsToMetadata(test, useCors) {
+ return new Promise(function(resolve, reject) {
+ var elemType = getMajorMimeType(test.type);
+ var element = document.createElement(elemType);
+
+ if (useCors) {
+ element.crossOrigin = "anonymous";
+ }
+
+ // Log events for debugging.
+ [
+ "suspend", "play", "canplay", "canplaythrough", "loadstart",
+ "loadedmetadata", "loadeddata", "playing", "ended", "error",
+ "stalled", "emptied", "abort", "waiting", "pause"
+ ].forEach((eventName) => {
+ element.addEventListener(eventName, (event)=> {
+ info(test.name + " " + event.type);
+ });
+ });
+
+ element.addEventListener("loadedmetadata", ()=>{
+ resolve(true);
+ removeNodeAndSource(element);
+ });
+
+ element.addEventListener("error", ()=>{
+ resolve(false);
+ removeNodeAndSource(element);
+ });
+
+ // Note: request redirect before the end of metadata, otherwise we won't
+ // error before metadata has loaded if mixed origins are allowed.
+ element.src = "midflight-redirect.sjs?resource=" + test.name
+ + (useCors ? "&cors" : "")
+ + "&type=" + test.type
+ + "&redirectAt=200";
+ element.preload = "metadata";
+ document.body.appendChild(element);
+ element.load()
+ });
+ }
+
+ let v = document.createElement("video");
+ const testCases = gSmallTests.filter(t => v.canPlayType(t.type));
+
+ async function testMediaLoad(message, {useCors}) {
+ for (let test of testCases) {
+ let loaded = await testIfLoadsToMetadata(test, useCors);
+ is(loaded, useCors, test.name + " " + message);
+ }
+ }
+
+ async function runTest() {
+ try {
+ SimpleTest.info("Test that all media do not play without CORS...");
+ await testMediaLoad("expected to be blocked", {useCors: false});
+
+ SimpleTest.info("Test that all media play if CORS used...");
+ await testMediaLoad("expected to play with CORS", {useCors: true});
+ } catch (e) {
+ info("Exception " + e.message);
+ ok(false, "Threw exception " + e.message);
+ }
+ SimpleTest.finish();
+ }
+
+ SimpleTest.waitForExplicitFinish();
+
+ addLoadEvent(runTest);
+
+ </script>
+ </pre>
+ </body>
+</html>
diff --git a/dom/media/test/test_mixed_principals.html b/dom/media/test/test_mixed_principals.html
new file mode 100644
index 0000000000..c1deaef697
--- /dev/null
+++ b/dom/media/test/test_mixed_principals.html
@@ -0,0 +1,94 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=489415
+-->
+<head>
+ <title>Test for Bug 489415</title>
+ <script type="application/javascript" src="/MochiKit/MochiKit.js"></script>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <style>
+ video {
+ width: 40%;
+ border: solid black 1px;
+ }
+ </style>
+</head>
+
+<body>
+ <a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=489415">Mozilla Bug 489415</a>
+ <p id="display"></p>
+ <pre id="test">
+ <script type="text/javascript">
+ SimpleTest.waitForExplicitFinish();
+
+ var pushPrefs = (...p) => SpecialPowers.pushPrefEnv({ set: p });
+ var count = 0;
+
+ function canReadBack(video) {
+ var c = document.createElement("canvas");
+ var ctx = c.getContext("2d");
+ ctx.drawImage(video, 0, 0);
+ try {
+ var data_url = c.toDataURL();
+ var empty_canvas = document.createElement("canvas");
+ if (empty_canvas.toDataURL() === data_url) {
+ info("Readback check returned same data URL as empty canvas.");
+ } else {
+ info("Readback check returned different data URL than empty canvas.");
+ }
+ return true;
+ } catch (ex) {
+ return false;
+ }
+ }
+
+ function runTest(origin, shouldReadBackOnLoad) {
+ return new Promise(function (resolve, reject) {
+ // Load will redirect mid-flight, which will be detected and should error,
+ // and we should no longer be able to readback.
+ var video = document.createElement("video");
+ video.preload = "metadata";
+ video.controls = true;
+ var url = "http://" + origin + "/tests/dom/media/test/midflight-redirect.sjs"
+ + "?resource=pixel_aspect_ratio.mp4&type=video/mp4";
+ SimpleTest.info("Loading from " + url);
+ video.src = url;
+ document.body.appendChild(video);
+
+ once(video, "loadeddata", () => {
+ is(canReadBack(video), shouldReadBackOnLoad, "Should be able to readback");
+ video.play();
+ });
+
+ once(video, "error", () => {
+ if(video.readyState == video.HAVE_METADATA) {
+ is(canReadBack(video), true, "Should be able to readback with readyState == HAVE_METADATA");
+ } else {
+ is(canReadBack(video), false, "Should not be able to readback with readyState != HAVE_METADATA");
+ }
+
+ removeNodeAndSource(video);
+ resolve();
+ });
+
+ once(video, "ended", () => {
+ ok(false, "Should not be able to playback to end, we should have errored!");
+ removeNodeAndSource(video);
+ resolve();
+ });
+
+ });
+ }
+
+ Promise.all([
+ runTest("mochi.test:8888", true),
+ runTest("example.org", false),
+ ]).then(SimpleTest.finish);
+
+ </script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mozHasAudio.html b/dom/media/test/test_mozHasAudio.html
new file mode 100644
index 0000000000..c25873c786
--- /dev/null
+++ b/dom/media/test/test_mozHasAudio.html
@@ -0,0 +1,42 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test playback of media files that should play OK</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function onloadedmetadata(e) {
+ var t = e.target;
+ is(t.mozHasAudio, t.hasAudio, "The element reports the wrong audio property." + t.token);
+ manager.finished(t.token);
+}
+
+function startTest(test, token) {
+ var elemType = /^audio/.test(test.type) ? "audio" : "video";
+ var element = document.createElement(elemType);
+ element.preload = "auto";
+
+ element.token = token;
+ manager.started(token);
+
+ element.src = test.name;
+ element.name = test.name;
+ element.hasAudio = elemType == "video" ? test.hasAudio : undefined;
+ element.addEventListener("loadedmetadata", onloadedmetadata);
+
+ element.load();
+}
+
+manager.runTests(gTrackTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_mp3_broadcast.html b/dom/media/test/test_mp3_broadcast.html
new file mode 100644
index 0000000000..7922d8b4b6
--- /dev/null
+++ b/dom/media/test/test_mp3_broadcast.html
@@ -0,0 +1,52 @@
+<!DOCTYPE HTML>
+<html>
+
+<head>
+ <title>Test playback of broadcast-like streams</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+
+<body>
+ <audio controls id=a style="width: 100%;"></audio>
+ <script type="module">
+ SimpleTest.waitForExplicitFinish();
+ const streams = [
+ // An mp3 bytestream consisting of a complete mp3 file with a XING
+ // header, that has duration information, but served without a
+ // `Content-Length`. It is followed by a second mp3 bytestream that
+ // also has a XING header. While some software are able to play the
+ // entire file, Web browser don't.
+ { src: "two-xing-header-no-content-length.mp3", duration: 1 },
+ // An mp3 bytestream consisting of a complete mp3 file with a XING
+ // header, that has duration information, but served without a
+ // `Content-Length` header. It is followed by a second mp3 bytestream that
+ // doesn't have a XING header.
+ // This scenario is typical in radio broadcast scenario, when the
+ // live-stream has a pre-recorded prelude. The reported duration,
+ // after "ended" has been received, is the duration of playback.
+ { src: "single-xing-header-no-content-length.mp3", duration: 11.050839},
+ ];
+ var audio = window.a;
+ // Prevent ESLint error about top-level await
+ (async function () {
+ for (let i of streams) {
+ audio.src = i.src;
+ audio.load();
+ audio.play();
+ audio.onerror = (e) => {
+ ok(false, `${i}: error: ${e.message}}`);
+ };
+ await once(audio, "ended");
+ ok(true, `${i}: playback through the end`);
+ is(audio.duration, i.duration, "Duration at end is correct");
+ is(audio.currentTime, i.duration, "Current time at end is correct");
+ }
+ SimpleTest.finish();
+ })()
+ </script>
+</body>
+
+</html>
+
diff --git a/dom/media/test/test_mp3_with_multiple_ID3v2.html b/dom/media/test/test_mp3_with_multiple_ID3v2.html
new file mode 100644
index 0000000000..1f2f946520
--- /dev/null
+++ b/dom/media/test/test_mp3_with_multiple_ID3v2.html
@@ -0,0 +1,30 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+<title>Play mp3 file with multiple ID3v2 tags</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+<script type="application/javascript">
+
+add_task(async function testPlayMP3WithMultipleID3Tags() {
+ info(`adjust cache size`);
+ await SpecialPowers.pushPrefEnv(
+ // The second ID3v2 header is huge (4622361 bytes) so the first audio samle
+ // in this file is in the position 4945370, so we have to extend the size of
+ // the cache.
+ {"set": [["media.cache_size", 5000000]]}
+ );
+
+ info(`create audio and wait its loading`);
+ let audio = document.createElement('audio');
+ audio.src = "multi_id3v2.mp3";
+ document.body.appendChild(audio);
+ await new Promise(r => audio.onloadeddata = r);
+ ok(true, `finishing loading data.`)
+});
+
+</script>
+</head>
+<body>
+</body>
+</html>
diff --git a/dom/media/test/test_multiple_mediastreamtracks.html b/dom/media/test/test_multiple_mediastreamtracks.html
new file mode 100644
index 0000000000..fb28d8652e
--- /dev/null
+++ b/dom/media/test/test_multiple_mediastreamtracks.html
@@ -0,0 +1,47 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test the ability of MediaStream with multiple MediaStreamTracks</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="gUM_support.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+async function startTest() {
+ try {
+ await setupGetUserMediaTestPrefs();
+ let orgStream = await navigator.mediaDevices.getUserMedia({audio: true, video: true});
+ let a = orgStream.getAudioTracks()[0];
+ let v = orgStream.getVideoTracks()[0];
+ let stream = new MediaStream([a, a, a, a, v, v, v].map(track => track.clone()));
+ let element = document.createElement("video");
+
+ element.onloadedmetadata = function() {
+ is(stream.getAudioTracks().length, 4, 'Length of audio tracks should be 4.');
+ is(stream.getVideoTracks().length, 3, 'Length of vudio tracks should be 3.');
+ SimpleTest.finish();
+ };
+
+ element.srcObject = stream;
+ element.play();
+ } catch (err) {
+ ok(false, 'Unexpected error fired with: ' + err);
+ SimpleTest.finish();
+ }
+}
+
+SimpleTest.waitForExplicitFinish();
+SpecialPowers.pushPrefEnv(
+ {
+ "set": [
+ ["media.track.enabled", true]
+ ]
+ }, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_networkState.html b/dom/media/test/test_networkState.html
new file mode 100644
index 0000000000..8001ca514c
--- /dev/null
+++ b/dom/media/test/test_networkState.html
@@ -0,0 +1,47 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: networkState</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body onunload="mediaTestCleanup();">
+<video id='v1'></video><audio id='a1'></audio>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+"use strict";
+var v1 = document.getElementById('v1');
+var a1 = document.getElementById('a1');
+var passed = "truthy";
+
+try {
+ v1.networkState = 0;
+} catch (e) {
+ passed = !passed;
+}
+try {
+ a1.networkState = 0;
+} catch (e) {
+ passed = !passed;
+}
+ok(passed === true,
+ "Setting networkState throws in strict mode (readonly attribute)");
+</script>
+
+<script class="testbody" type="text/javascript">
+var v1 = document.getElementById('v1');
+var a1 = document.getElementById('a1');
+var passed = false;
+
+var oldv1ns = v1.networkState, olda1ns = a1.networkState;
+try {
+ v1.networkState = 0;
+ a1.networkState = 0;
+ passed = v1.networkState === oldv1ns && a1.networkState === olda1ns;
+} catch (e) { }
+ok(passed, "Should not be able to modify networkState (readonly attribute)");
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_new_audio.html b/dom/media/test/test_new_audio.html
new file mode 100644
index 0000000000..e1f8964f73
--- /dev/null
+++ b/dom/media/test/test_new_audio.html
@@ -0,0 +1,48 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=528566
+-->
+<head>
+ <title>Test for Bug 528566</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=528566">Mozilla Bug 528566</a>
+<p id="display"></p>
+<div id="content" style="display: none">
+
+</div>
+<pre id="test">
+<script type="application/javascript">
+
+/** Test for Bug 528566 **/
+
+var manager = new MediaTestManager;
+
+var player = new Audio();
+
+function startTest(test, token) {
+ if (!player.canPlayType(test.type)) {
+ return;
+ }
+ manager.started(token);
+ var a = new Audio(test.name);
+ a.autoplay = true;
+ document.body.appendChild(a);
+ a.addEventListener("ended",
+ function(e){
+ ok(true, "[" + a.src + "]We should get to the end. Oh look we did.");
+ a.remove();
+ manager.finished(token);
+ });
+}
+
+manager.runTests(gAudioTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_no_load_event.html b/dom/media/test/test_no_load_event.html
new file mode 100644
index 0000000000..1e2717d983
--- /dev/null
+++ b/dom/media/test/test_no_load_event.html
@@ -0,0 +1,53 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=715469
+-->
+<head>
+ <meta charset="utf-8">
+ <title>Test for Bug 715469</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body onload="start();">
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=715469">Mozilla Bug 715469</a>
+<p id="display"></p>
+<div id="content" style="display: none">
+</div>
+<pre id="test">
+
+<script type="application/javascript">
+
+/** Test for Bug 715469 **/
+
+var gotLoadEvent = false;
+
+function start() {
+ var resource = getPlayableVideo(gSmallTests);
+ if (resource == null) {
+ todo(false, "No types supported");
+ } else {
+ SimpleTest.waitForExplicitFinish();
+ var v = document.createElement("video");
+ v.src = resource.name;
+
+ v.addEventListener("load", function() {
+ gotLoadEvent = true;
+ });
+
+ v.addEventListener("ended", finished);
+ document.body.appendChild(v);
+ v.play();
+ }
+}
+
+function finished() {
+ is(gotLoadEvent, false, "Should not receive a load on the video element");
+ SimpleTest.finish();
+}
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_not_reset_playbackRate_when_removing_nonloaded_media_from_document.html b/dom/media/test/test_not_reset_playbackRate_when_removing_nonloaded_media_from_document.html
new file mode 100644
index 0000000000..f1b8cc8b15
--- /dev/null
+++ b/dom/media/test/test_not_reset_playbackRate_when_removing_nonloaded_media_from_document.html
@@ -0,0 +1,46 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Do not reset playback rate when removing non-loaded media from a document</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script src="manifest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+</head>
+<body>
+<script class="testbody" type="text/javascript">
+/**
+ * When removing media from a document, it should only trigger internal pause,
+ * instead of the pause method, which would trigger loading process and reset
+ * the media's playback rate for non-loaded media.
+ */
+async function startTest() {
+ info(`create a media and append it to a document`);
+ const audio = document.createElement("audio");
+ document.body.appendChild(audio);
+
+ info(`change audio's playbackRate and remove it from a document`);
+ const expectedRate = 0.1;
+ audio.playbackRate = expectedRate;
+ await once(audio, "ratechange");
+ is(audio.playbackRate, expectedRate,
+ `${audio.playbackRate} is equal to ${expectedRate}`);
+ audio.remove();
+
+ info(`queue a macrotask to check if the playback rate is still unchanged`);
+ setTimeout(() => {
+ // If we unexpectedly reset the playback rate, it would happen in a
+ // microtask when removing media from a document [1] (Await a stable state),
+ // which would always be run before the macrotask.
+ // [1] https://html.spec.whatwg.org/#playing-the-media-resource:remove-an-element-from-a-document
+ is(audio.playbackRate, expectedRate,
+ `${audio.playbackRate} is equal to ${expectedRate}`);
+ SimpleTest.finish();
+ }, 0);
+}
+
+SimpleTest.waitForExplicitFinish();
+onload = startTest;
+
+</script>
+</body>
+</html>
diff --git a/dom/media/test/test_paused.html b/dom/media/test/test_paused.html
new file mode 100644
index 0000000000..17af4d3898
--- /dev/null
+++ b/dom/media/test/test_paused.html
@@ -0,0 +1,21 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: paused</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<video id='v1'></video><audio id='a1'></audio>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var v1 = document.getElementById('v1');
+var a1 = document.getElementById('a1');
+ok(v1.paused, "v1.paused must initially be true");
+ok(a1.paused, "a1.paused must initially be true");
+mediaTestCleanup();
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_paused_after_ended.html b/dom/media/test/test_paused_after_ended.html
new file mode 100644
index 0000000000..83f27a921f
--- /dev/null
+++ b/dom/media/test/test_paused_after_ended.html
@@ -0,0 +1,53 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: paused</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function ended(evt) {
+ var v = evt.target;
+ v.removeEventListener("ended", ended);
+ is(v.gotPause, true, "We should have received a \"pause\" event.")
+ is(v.paused, true, v._name + " must be paused after end");
+ manager.finished(v.token);
+ removeNodeAndSource(v);
+}
+
+function pause(evt) {
+ var v = evt.target;
+ v.removeEventListener("pause", pause);
+ v.gotPause = true;
+}
+
+function startTest(test, token) {
+ var v = document.createElement('video');
+ document.body.appendChild(v);
+ v.token = token;
+ manager.started(v.token);
+ v.src = test.name;
+ v._name = test.name;
+ v._finished = false;
+ v.load();
+ is(v.paused, true, v._name + " must be paused at start");
+
+ v.play();
+ is(v.paused, false, v._name + " must not be paused after play");
+
+ v.addEventListener("pause", pause);
+ v.addEventListener("ended", ended);
+}
+
+manager.runTests(gPausedAfterEndedTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_periodic_timeupdate.html b/dom/media/test/test_periodic_timeupdate.html
new file mode 100644
index 0000000000..5d40c7e38a
--- /dev/null
+++ b/dom/media/test/test_periodic_timeupdate.html
@@ -0,0 +1,100 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+<title>Periodic timeupdate test</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+<script type="application/javascript">
+
+/**
+ * To ensure that when dispatching periodic `timeupdate` event, we would only
+ * dispatch that once within 250ms according to the spec. This test would end
+ * after receiving certain number of timeupdate` event.
+ */
+const sNumPeriodicTimeupdatesToEndTestAfter = 5;
+const sTimeThreshold = 250;
+
+add_task(async function testPeriodicTimeupdateShouldOnlyBeDispatchedOnceWithin250Ms() {
+ const video = document.createElement('video');
+ video.src = "gizmo.mp4";
+ video.loop = true;
+ video._timeupdateCount = 0;
+ document.body.appendChild(video);
+ ok(await video.play().then(_=>true,_=>false), "video started playing");
+ const culprit = createCulpritToMakeMainThreadBusy();
+ await new Promise(r => {
+ function endTest() {
+ video.removeEventListener("timeupdate", checkTimeupdate);
+ culprit.shutdown();
+ r();
+ }
+ video.onseeking = () => {
+ info(`seeking starts (looping back to the head)`);
+ video._ignoreEvents = true;
+ }
+ video.onseeked = () => {
+ info(`seeking ends`);
+ video._ignoreEvents = false;
+ }
+ function checkTimeupdate(event) {
+ // When reaching to the end, video would perform a seek to the start
+ // position where one mandatory `timeupdate` would be dispatched.
+ if (video._ignoreEvents) {
+ info(`ignore non-periodic timeupdate because that is allowed to be dispatched within ${sTimeThreshold}ms`);
+ return;
+ }
+
+ const now = performance.now();
+ if (video._prevTime === undefined) {
+ info(`recevied the first 'timeupdate'`);
+ video._prevTime = now;
+ return;
+ }
+
+ const timeDiff = now - video._prevTime;
+ if (timeDiff < sTimeThreshold) {
+ ok(false, `Time diff ${timeDiff} is smaller than ${sTimeThreshold}ms!`);
+ endTest();
+ return;
+ }
+
+ ok(true, `Time diff ${timeDiff} since last time received 'timeupdate'`);
+ video._prevTime = now;
+ info(`check timeupdate ${++video._timeupdateCount} time`);
+ if (video._timeupdateCount == sNumPeriodicTimeupdatesToEndTestAfter) {
+ endTest();
+ }
+ };
+ video.addEventListener("timeupdate", checkTimeupdate);
+ });
+});
+
+window.onmessage = _ => blockMainThreadForMilliseconds(1);
+
+/**
+ * Following are helper functions
+ */
+function blockMainThreadForMilliseconds(ms) {
+ const lastTime = performance.now();
+ while (lastTime + ms > performance.now());
+}
+
+function createCulpritToMakeMainThreadBusy() {
+ let culprit = {};
+ culprit._id = setInterval(_ => {
+ blockMainThreadForMilliseconds(1000);
+ }, 0);
+ culprit.shutdown = _ => {
+ clearInterval(culprit._id);
+ }
+ for (let i = 0; i < 5000; ++i) {
+ window.postMessage("foo", "*");
+ }
+ return culprit;
+}
+
+</script>
+</head>
+<body>
+</body>
+</html>
diff --git a/dom/media/test/test_play_events.html b/dom/media/test/test_play_events.html
new file mode 100644
index 0000000000..44b819b488
--- /dev/null
+++ b/dom/media/test/test_play_events.html
@@ -0,0 +1,61 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: play() method</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+
+<script>
+
+var manager = new MediaTestManager;
+
+var tokens = {
+ 0: ["play"],
+ "play": ["canplay"],
+ "canplay": ["playing"],
+ "playing": ["canplay", "canplaythrough"],
+ "canplaythrough": ["canplay", "canplaythrough"]
+};
+
+function gotPlayEvent(event) {
+ var v = event.target;
+ ok(tokens[v._state].includes(event.type),
+ "Check expected event got " + event.type + " at " + v._state + " for " + v.src +
+ " tokens["+v._state+"]=" + tokens[v._state] +
+ " tokens["+v._state+"].indexOf(event.type)=" + tokens[v._state].indexOf(event.type));
+ v._state = event.type;
+}
+
+function ended(event) {
+ var v = event.target;
+ removeNodeAndSource(v);
+ manager.finished(v.token);
+}
+
+function initTest(test, token) {
+ var v = document.createElement('video');
+ v.token = token;
+ manager.started(token);
+ v._state = 0;
+
+ ["play", "canplay", "playing", "canplaythrough"].forEach(function (e) {
+ v.addEventListener(e, gotPlayEvent);
+ });
+
+ v.addEventListener("ended", ended);
+
+ v.src = test.name;
+ document.body.appendChild(v); // Causes load.
+ v.play();
+}
+
+manager.runTests(gSmallTests, initTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_play_events_2.html b/dom/media/test/test_play_events_2.html
new file mode 100644
index 0000000000..022bb5373a
--- /dev/null
+++ b/dom/media/test/test_play_events_2.html
@@ -0,0 +1,60 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: play() method via DOM 0 handlers</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+
+<script>
+var manager = new MediaTestManager;
+
+var tokens = {
+ 0: ["play"],
+ "play": ["canplay"],
+ "canplay": ["playing"],
+ "playing": ["canplay", "canplaythrough"],
+ "canplaythrough": ["canplay", "canplaythrough"]
+};
+
+function gotPlayEvent(event) {
+ var v = event.target;
+ ok(tokens[v._state].includes(event.type),
+ "Check expected event got " + event.type + " at " + v._state + " for " + v.src);
+ v._state = event.type;
+}
+
+function ended(event) {
+ var v = event.target;
+ v._finished = true;
+ removeNodeAndSource(v);
+ manager.finished(v.token);
+}
+
+function startTest(test, token) {
+ var v = document.createElement('video');
+ v.token = token;
+ manager.started(token);
+ v._state = 0;
+ v._finished = false;
+
+ ["play", "canplay", "playing", "canplaythrough"].forEach(function (e) {
+ v["on" + e] = gotPlayEvent;
+ });
+
+ v.onended = ended;
+
+ v.src = test.name;
+ document.body.appendChild(v); // Causes load.
+ v.play();
+}
+
+manager.runTests(gSmallTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_play_promise_1.html b/dom/media/test/test_play_promise_1.html
new file mode 100644
index 0000000000..ce4e287f34
--- /dev/null
+++ b/dom/media/test/test_play_promise_1.html
@@ -0,0 +1,42 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: promise-based play() method</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+
+<script>
+// Name: playBeforeCanPlay
+// Case: invoke play() on an element that doesn't have enough data
+// Expected result: resolve the promise
+
+let manager = new MediaTestManager;
+
+function initTest(test, token) {
+ manager.started(token);
+
+ let element = document.createElement(getMajorMimeType(test.type));
+ element.src = test.name;
+ ok(element.readyState == HTMLMediaElement.HAVE_NOTHING);
+
+ element.play().then(
+ (result) => {
+ if (result == undefined) {
+ ok(true, `${token} is resolved with ${result}.`);
+ } else {
+ ok(false, `${token} is resolved with ${result}.`);
+ }
+ },
+ (error) => {
+ ok(false, `${token} is rejected with ${error.name}.`);
+ }
+ ).then( () => { manager.finished(token); } );
+}
+
+manager.runTests(gSmallTests, initTest);
+
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_play_promise_10.html b/dom/media/test/test_play_promise_10.html
new file mode 100644
index 0000000000..1c0096986c
--- /dev/null
+++ b/dom/media/test/test_play_promise_10.html
@@ -0,0 +1,40 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: promise-based play() method</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+
+<script>
+// Name: loadRejectsPendingPromises
+// Case: invoke load() on an element with pending promises.
+// Expected result: reject all the pending promises with AbortError DOM exception.
+
+let manager = new MediaTestManager;
+
+function initTest(test, token) {
+ manager.started(token);
+
+ let element = document.createElement(getMajorMimeType(test.type));
+ element.play().then(
+ (result) => {
+ ok(false, `${token} is resolved with ${result}.`);
+ },
+ (error) => {
+ if (error.name == "AbortError") {
+ ok(true, `${token} is rejected with ${error.name}.`);
+ } else {
+ ok(false, `${token} is rejected with ${error.name}.`);
+ }
+ }
+ ).then( () => { manager.finished(token); } );
+ element.load();
+}
+
+manager.runTests(gSmallTests, initTest);
+
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_play_promise_11.html b/dom/media/test/test_play_promise_11.html
new file mode 100644
index 0000000000..a59baf5657
--- /dev/null
+++ b/dom/media/test/test_play_promise_11.html
@@ -0,0 +1,40 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: promise-based play() method</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+
+<script>
+// Name: newSrcRejectPendingPromises
+// Case: change src of an element with pending promises.
+// Expected result: reject all the pending promises.
+
+let manager = new MediaTestManager;
+
+function initTest(test, token) {
+ manager.started(token);
+
+ let element = document.createElement(getMajorMimeType(test.type));
+ element.play().then(
+ (result) => {
+ ok(false, `${token} is resolved with ${result}.`);
+ },
+ (error) => {
+ if (error.name == "AbortError") {
+ ok(true, `${token} is rejected with ${error.name}.`);
+ } else {
+ ok(false, `${token} is rejected with ${error.name}.`);
+ }
+ }
+ ).then( () => { manager.finished(token); } );
+ element.src = test.name;
+}
+
+manager.runTests(gSmallTests, initTest);
+
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_play_promise_12.html b/dom/media/test/test_play_promise_12.html
new file mode 100644
index 0000000000..50972885eb
--- /dev/null
+++ b/dom/media/test/test_play_promise_12.html
@@ -0,0 +1,45 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: promise-based play() method</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+
+<script>
+// Name: pausePlayAfterPlaybackStarted
+// Case: invoke pause() and then play() on an element that is already playing.
+// Expected result: resolve the promise with undefined.
+
+let manager = new MediaTestManager;
+
+function initTest(test, token) {
+ manager.started(token);
+
+ let element = document.createElement(getMajorMimeType(test.type));
+ element.preload = "auto";
+ element.src = test.name;
+ element.play();
+ once(element, "playing").then(() => {
+ element.pause();
+ element.play().then(
+ (result) => {
+ if (result == undefined) {
+ ok(true, `${token} is resolved with ${result}.`);
+ } else {
+ ok(false, `${token} is resolved with ${result}.`);
+ }
+ },
+ (error) => {
+ ok(false, `${token} is rejected with ${error.name}.`);
+ }
+ ).then( () => { manager.finished(token); } );
+ });
+}
+
+manager.runTests(gSmallTests, initTest);
+
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_play_promise_13.html b/dom/media/test/test_play_promise_13.html
new file mode 100644
index 0000000000..aacab88895
--- /dev/null
+++ b/dom/media/test/test_play_promise_13.html
@@ -0,0 +1,49 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: promise-based play() method</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+
+<script>
+// Name: loadAlgorithmDoesNotCancelTasks
+// Case: re-invoke the load() on an element which had dispatched a task to resolve a promise.
+// Expected result: the already dispatched promise should still be resolved with undefined.
+
+let manager = new MediaTestManager;
+
+function initTest(test, token) {
+ manager.started(token);
+
+ let element = document.createElement(getMajorMimeType(test.type));
+ element.preload = "auto";
+ element.src = test.name;
+
+ // We must wait for "canplay" event; otherwise, invoke play() will lead to a
+ // pending promise and will then be rejected by the following load().
+ once(element, "canplay").then(() => {
+ // The play() promise will be queued to be resolved immediately, which means
+ // the promise is not in the pending list.
+ element.play().then(
+ (result) => {
+ if (result == undefined) {
+ ok(true, `${token} is resolved with ${result}.`);
+ } else {
+ ok(false, `${token} is resolved with ${result}.`);
+ }
+ },
+ (error) => {
+ ok(false, `${token} is rejected with ${error.name}.`);
+ }
+ ).then( () => { manager.finished(token); } );
+ element.src = test.name; // Re-invoke the load algorithm again.
+ });
+}
+
+manager.runTests(gSmallTests, initTest);
+
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_play_promise_14.html b/dom/media/test/test_play_promise_14.html
new file mode 100644
index 0000000000..066138ecad
--- /dev/null
+++ b/dom/media/test/test_play_promise_14.html
@@ -0,0 +1,56 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: promise-based play() method</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+
+<script>
+// Name: loadAlgorithmKeepPromisesPendingWhenNotPausing
+// Case: step1: create an element with its paused member to be fause and networkState to be NETWORK_EMPTY.
+// stpe2: invoke load() on the element and the load() leaves the promise pending.
+// Expected result: the pending promise should finally be resolved.
+
+let manager = new MediaTestManager;
+
+function initTest(test, token) {
+ manager.started(token);
+
+ let element = document.createElement(getMajorMimeType(test.type));
+ // Invoke play() -> (1) the promise will be left pending.
+ // (2) invoke load() -> (1) set the networkState to be NETWORK_NO_SOURCE.
+ // (2) queue a task to run resouce selection algorithm.
+ element.play().then(
+ (result) => {
+ if (result == undefined) {
+ ok(true, `${token} is resolved with ${result}.`);
+ } else {
+ ok(false, `${token} is resolved with ${result}.`);
+ }
+ },
+ (error) => {
+ ok(false, `${token} is rejected with ${error.name}.`);
+ }
+ ).then( () => { manager.finished(token); } );
+
+ once(element, "play").then(() => {
+ // The resouce selection algorithm has been done.
+ // -> set the networkState to be NETWORK_EMPTY because there is no valid resource to load.
+ ok(element.networkState == HTMLMediaElement.NETWORK_EMPTY);
+
+ // Invoke load() again and since the networkState is NETWORK_EMPTY, the load() does not reject the pending promise.
+ // The load() will queue a task to run resouce selection algorithm which will change the readyState and finally resolve the pending promise.
+ element.src = test.name;
+
+ // Since the networkState is NETWORK_EMPTY, the load() does not set paused to be true.
+ ok(!element.paused, `loadAlgorithmKeepPromisesPendingWhenNotPausing(${token}).paused should be false.`);
+ });
+}
+
+manager.runTests(gSmallTests, initTest);
+
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_play_promise_15.html b/dom/media/test/test_play_promise_15.html
new file mode 100644
index 0000000000..b8bce48651
--- /dev/null
+++ b/dom/media/test/test_play_promise_15.html
@@ -0,0 +1,51 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: promise-based play() method</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+
+<script>
+// Name: loadAlgorithmRejectPromisesWhenPausing
+// Case: step1: create an element with its paused member to be fause and networkState to be NETWORK_NO_SOURCE.
+// stpe2: invoke load() on the element and the load() rejects the pending promise.
+// Expected result: reject the pending promise with AbortError DOM exception.
+
+let manager = new MediaTestManager;
+
+function initTest(test, token) {
+ manager.started(token);
+
+ let element = document.createElement(getMajorMimeType(test.type));
+ // Invoke play() -> (1) the promise will be left pending.
+ // (2) invoke load() -> (1) set the networkState to be NETWORK_NO_SOURCE.
+ // (2) queue a task to run resouce selection algorithm.
+ element.play().then(
+ (result) => {
+ ok(false, `${token} is resolved with ${result}.`);
+ },
+ (error) => {
+ if (error.name == "AbortError") {
+ ok(true, `${token} is rejected with ${error.name}.`);
+ } else {
+ ok(false, `${token} is rejected with ${error.name}.`);
+ }
+ }
+ ).then( () => { manager.finished(token); } );
+
+ ok(element.networkState == HTMLMediaElement.NETWORK_NO_SOURCE);
+
+ // Invoke load() again and since the networkState is NETWORK_NO_SOURCE, the load() rejects the pending promise.
+ element.src = test.name;
+
+ // Since the networkState is not NETWORK_EMPTY, the load() sets paused to be true.
+ ok(element.paused, `loadAlgorithmRejectPromisesWhenPausing(${token}).paused should be true.`);
+}
+
+manager.runTests(gSmallTests, initTest);
+
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_play_promise_16.html b/dom/media/test/test_play_promise_16.html
new file mode 100644
index 0000000000..a643e7fff8
--- /dev/null
+++ b/dom/media/test/test_play_promise_16.html
@@ -0,0 +1,47 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: promise-based play() method</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+
+<script>
+// Name: loadAlgorithmResolveOrdering
+// Case: invoke load() on an element should resolve pending promises in order.
+// Expected result: the pending promises are resolved in order.
+
+let manager = new MediaTestManager;
+
+function initTest(test, token) {
+ manager.started(token);
+
+ let element = document.createElement(getMajorMimeType(test.type));
+ element.preload = "auto";
+ element.src = test.name;
+ once(element, "canplay").then(() => {
+ let firstPromiseResolved = false;
+
+ // play
+ element.play().then(
+ () => { firstPromiseResolved = true; },
+ () => { ok(false, `loadAlgorithmResolveOrdering(${token}) should not be rejected.`); }
+ );
+
+ // play again
+ element.play().then(
+ () => { ok(firstPromiseResolved, `loadAlgorithmResolveOrdering(${token}), the first play should already be resolved.`); },
+ () => { ok(false, `loadAlgorithmResolveOrdering(${token}) should not be rejected.`); }
+ ).then( () => { manager.finished(token); } );
+
+ // triger load
+ element.src = test.name;
+ });
+}
+
+manager.runTests(gSmallTests, initTest);
+
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_play_promise_17.html b/dom/media/test/test_play_promise_17.html
new file mode 100644
index 0000000000..5e5eb71fd5
--- /dev/null
+++ b/dom/media/test/test_play_promise_17.html
@@ -0,0 +1,43 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: promise-based play() method</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+
+<script>
+// Name: loadAlgorithmRejectOrdering
+// Case: invoke load() on an element should reject pending promises in order.
+// Expected result: the pending promises are rejected in order.
+
+let manager = new MediaTestManager;
+
+function initTest(test, token) {
+ manager.started(token);
+
+ let element = document.createElement(getMajorMimeType(test.type));
+ let firstPromiseRejected = false;
+
+ // play
+ element.play().then(
+ () => { ok(false, `loadAlgorithmRejectOrdering(${token}) should not be resolved.`); },
+ () => { firstPromiseRejected = true; }
+ );
+
+ // play again
+ element.play().then(
+ () => { ok(false, `loadAlgorithmRejectOrdering(${token}) should not be resolved.`); },
+ () => { ok(firstPromiseRejected, `loadAlgorithmRejectOrdering(${token}), the first play should already be rejected.`); }
+ ).then( () => { manager.finished(token); } );
+
+ // triger load
+ element.src = test.name;
+}
+
+manager.runTests(gSmallTests, initTest);
+
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_play_promise_18.html b/dom/media/test/test_play_promise_18.html
new file mode 100644
index 0000000000..e0c0837fdf
--- /dev/null
+++ b/dom/media/test/test_play_promise_18.html
@@ -0,0 +1,46 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: promise-based play() method</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+
+<script>
+// Name: 'playing' event should come before promise resolving
+// Case: invoke play() on an element which has valis source
+// Expected result: receive a 'playing' event before the promise is resolved
+
+let manager = new MediaTestManager;
+
+function initTest(test, token) {
+ manager.started(token);
+
+ let element = document.createElement(getMajorMimeType(test.type));
+ element.src = test.name;
+ element.receivedPlayingEvent = false;
+
+ element.addEventListener("playing", () => {
+ element.receivedPlayingEvent = true;
+ });
+
+ element.play().then(
+ (result) => {
+ if (result == undefined && element.receivedPlayingEvent) {
+ ok(true, `${token} is resolved with ${result}.`);
+ } else {
+ ok(false, `${token} is resolved with ${result} and receivedPlayingEvent = ${element.receivedPlayingEvent}`);
+ }
+ },
+ (error) => {
+ ok(false, `${token} is rejected with ${error.name}.`);
+ }
+ ).then( () => { manager.finished(token); } );
+}
+
+manager.runTests(gSmallTests, initTest);
+
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_play_promise_2.html b/dom/media/test/test_play_promise_2.html
new file mode 100644
index 0000000000..c4befdbe91
--- /dev/null
+++ b/dom/media/test/test_play_promise_2.html
@@ -0,0 +1,43 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: promise-based play() method</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+
+<script>
+// Name: playWhenCanPlay
+// Case: invoke play() on an element that has enough data
+// Expected result: resolve the promise
+
+let manager = new MediaTestManager;
+
+function initTest(test, token) {
+ manager.started(token);
+
+ let element = document.createElement(getMajorMimeType(test.type));
+ element.preload = "auto";
+ element.src = test.name;
+ once(element, "canplay").then(() => {
+ element.play().then(
+ (result) => {
+ if (result == undefined) {
+ ok(true, `${token} is resolved with ${result}.`);
+ } else {
+ ok(false, `${token} is resolved with ${result}.`);
+ }
+ },
+ (error) => {
+ ok(false, `${token} is rejected with ${error.name}.`);
+ }
+ ).then( () => { manager.finished(token); } );
+ });
+}
+
+manager.runTests(gSmallTests, initTest);
+
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_play_promise_3.html b/dom/media/test/test_play_promise_3.html
new file mode 100644
index 0000000000..8ffacdae03
--- /dev/null
+++ b/dom/media/test/test_play_promise_3.html
@@ -0,0 +1,47 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: promise-based play() method</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+
+<script>
+// Name: playAfterPlaybackStarted
+// Case: invoke play() on an element that is already playing
+// Expected result: resolve the promise
+
+let manager = new MediaTestManager;
+
+function initTest(test, token) {
+ manager.started(token);
+
+ let element = document.createElement(getMajorMimeType(test.type));
+ element.preload = "auto";
+ element.src = test.name;
+ once(element, "playing").then(() => {
+ ok(element.readyState >= HTMLMediaElement.HAVE_FUTURE_DATA, `playAfterPlaybackStarted(${token})`);
+ ok(!element.paused, `playAfterPlaybackStarted(${token})`);
+ element.play().then(
+ (result) => {
+ if (result == undefined) {
+ ok(true, `${token} is resolved with ${result}.`);
+ } else {
+ ok(false, `${token} is resolved with ${result}.`);
+ }
+ },
+ (error) => {
+ ok(false, `${token} is rejected with ${error.name}.`);
+ }
+ ).then( () => { manager.finished(token); } );
+ });
+
+ element.play();
+}
+
+manager.runTests(gSmallTests, initTest);
+
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_play_promise_4.html b/dom/media/test/test_play_promise_4.html
new file mode 100644
index 0000000000..dfcd96e0b9
--- /dev/null
+++ b/dom/media/test/test_play_promise_4.html
@@ -0,0 +1,41 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: promise-based play() method</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="play_promise.js"></script>
+</head>
+<body>
+<pre id="test">
+
+<script>
+// Name: playNotSupportedContent
+// Case: invoke play() on an element with an unsupported content
+// Expected result: reject the promise with NotSupportedError DOM exception
+
+let manager = new MediaTestManager;
+
+function initTest(test, token) {
+ manager.started(token);
+
+ let element = document.createElement(getMajorMimeType(test.type));
+ element.src = getNotSupportedFile(test.name);
+ element.play().then(
+ (result) => {
+ ok(false, `${token} is resolved with ${result}.`);
+ },
+ (error) => {
+ if (error.name == "NotSupportedError") {
+ ok(true, `${token} is rejected with ${error.name}.`);
+ } else {
+ ok(false, `${token} is rejected with ${error.name}.`);
+ }
+ }
+ ).then( () => { manager.finished(token); } );
+}
+
+manager.runTests(gSmallTests, initTest);
+
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_play_promise_5.html b/dom/media/test/test_play_promise_5.html
new file mode 100644
index 0000000000..a4c3eeb936
--- /dev/null
+++ b/dom/media/test/test_play_promise_5.html
@@ -0,0 +1,44 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: promise-based play() method</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="play_promise.js"></script>
+</head>
+<body>
+<pre id="test">
+
+<script>
+// Name: playWithErrorAlreadySet
+// Case: invoke play() on an element with MEDIA_ERR_SRC_NOT_SUPPORTED has been set
+// Expected result: reject the promise with NotSupportedError DOM exception
+
+let manager = new MediaTestManager;
+
+function initTest(test, token) {
+ manager.started(token);
+
+ let element = document.createElement(getMajorMimeType(test.type));
+ element.src = getNotSupportedFile(test.name);
+ once(element, "error").then(() => {
+ ok(element.error.code == MediaError.MEDIA_ERR_SRC_NOT_SUPPORTED);
+ element.play().then(
+ (result) => {
+ ok(false, `${token} is resolved with ${result}.`);
+ },
+ (error) => {
+ if (error.name == "NotSupportedError") {
+ ok(true, `${token} is rejected with ${error.name}.`);
+ } else {
+ ok(false, `${token} is rejected with ${error.name}.`);
+ }
+ }
+ ).then( () => { manager.finished(token); } );
+ });
+}
+
+manager.runTests(gSmallTests, initTest);
+
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_play_promise_6.html b/dom/media/test/test_play_promise_6.html
new file mode 100644
index 0000000000..dcbea3f108
--- /dev/null
+++ b/dom/media/test/test_play_promise_6.html
@@ -0,0 +1,45 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: promise-based play() method</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="play_promise.js"></script>
+</head>
+<body>
+<pre id="test">
+
+<script>
+// Name: playSwitchToValidSrcAfterError
+// Case: invoke play() on an element which had its source changed (to a valid source) after suffering from an error
+// Expected result: resolve the promise with undefined
+
+let manager = new MediaTestManager;
+
+function initTest(test, token) {
+ manager.started(token);
+
+ let element = document.createElement(getMajorMimeType(test.type));
+ element.src = getNotSupportedFile(test.name);
+ once(element, "error").then(() => {
+ ok(element.error.code == MediaError.MEDIA_ERR_SRC_NOT_SUPPORTED);
+ element.src = test.name;
+ element.play().then(
+ (result) => {
+ if (result == undefined) {
+ ok(true, `${token} is resolved with ${result}.`);
+ } else {
+ ok(false, `${token} is resolved with ${result}.`);
+ }
+ },
+ (error) => {
+ ok(false, `${token} is rejected with ${error.name}.`);
+ }
+ ).then( () => { manager.finished(token); } );
+ });
+}
+
+manager.runTests(gSmallTests, initTest);
+
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_play_promise_7.html b/dom/media/test/test_play_promise_7.html
new file mode 100644
index 0000000000..1abbe42f33
--- /dev/null
+++ b/dom/media/test/test_play_promise_7.html
@@ -0,0 +1,47 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: promise-based play() method</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="play_promise.js"></script>
+</head>
+<body>
+<pre id="test">
+
+<script>
+// Name: playSwitchToInvalidSrcAfterError
+// Case: invoke play() on an element which had its source changed (to a invalid source) after suffering from an error
+// Expected result: reject the promise with NotSupportedError DOM exception
+
+let manager = new MediaTestManager;
+
+function initTest(test, token) {
+ manager.started(token);
+
+ let element = document.createElement(getMajorMimeType(test.type));
+ element.src = getNotSupportedFile(test.name);
+ once(element, "error").then(() => {
+ ok(element.error.code == MediaError.MEDIA_ERR_SRC_NOT_SUPPORTED);
+ element.src = getNotSupportedFile(test.name);
+ ok(element.error == null);
+ ok(element.readyState == HTMLMediaElement.HAVE_NOTHING);
+ element.play().then(
+ (result) => {
+ ok(false, `${token} is resolved with ${result}.`);
+ },
+ (error) => {
+ if (error.name == "NotSupportedError") {
+ ok(true, `${token} is rejected with ${error.name}.`);
+ } else {
+ ok(false, `${token} is rejected with ${error.name}.`);
+ }
+ }
+ ).then( () => { manager.finished(token); } );
+ });
+}
+
+manager.runTests(gSmallTests, initTest);
+
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_play_promise_8.html b/dom/media/test/test_play_promise_8.html
new file mode 100644
index 0000000000..d61e12c9fe
--- /dev/null
+++ b/dom/media/test/test_play_promise_8.html
@@ -0,0 +1,47 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: promise-based play() method</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+
+<script>
+// Name: playAndPauseWhenCanplay
+// Case: invlke play() and then pause() on an element that already has enough data to play
+// Expected result: resolve the promise
+// Note: the pause() doesn't cancel the play() because it was alredy been queued to be resolved.
+
+let manager = new MediaTestManager;
+
+function initTest(test, token) {
+ manager.started(token);
+
+ let element = document.createElement(getMajorMimeType(test.type));
+ element.preload = "auto";
+ element.src = test.name;
+ once(element, "canplay").then(() => {
+ element.play().then(
+ (result) => {
+ if (result == undefined) {
+ ok(true, `${token} is resolved with ${result}.`);
+ } else {
+ ok(false, `${token} is resolved with ${result}.`);
+ }
+ },
+ (error) => {
+ ok(false, `${token} is rejected with ${error.name}.`);
+ }
+ ).then( () => { manager.finished(token); } );
+ ok(!element.paused, `playAndPauseWhenCanplay(${token}) element should not be paused.`);
+ element.pause();
+ ok(element.paused, `playAndPauseWhenCanplay(${token}) element should be paused.`);
+ });
+}
+
+manager.runTests(gSmallTests, initTest);
+
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_play_promise_9.html b/dom/media/test/test_play_promise_9.html
new file mode 100644
index 0000000000..c3a3856515
--- /dev/null
+++ b/dom/media/test/test_play_promise_9.html
@@ -0,0 +1,44 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: promise-based play() method</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+
+<script>
+// Name: playAndPauseBeforeCanPlay
+// Case: invlke play() and then pause() on an element that deoen't have enough data to play.
+// Expected result: reject the promise with AbortError DOM exception.
+// Note: the pause() cancels the play() because the promise is still pending.
+
+let manager = new MediaTestManager;
+
+function initTest(test, token) {
+ manager.started(token);
+
+ let element = document.createElement(getMajorMimeType(test.type));
+ ok(element.readyState == HTMLMediaElement.HAVE_NOTHING);
+ element.play().then(
+ (result) => {
+ ok(false, `${token} is resolved with ${result}.`);
+ },
+ (error) => {
+ if (error.name == "AbortError") {
+ ok(true, `${token} is rejected with ${error.name}.`);
+ } else {
+ ok(false, `${token} is rejected with ${error.name}.`);
+ }
+ }
+ ).then( () => { manager.finished(token); } );
+ ok(!element.paused, `playAndPauseBeforeCanPlay(${token}) element should not be paused.`);
+ element.pause();
+ ok(element.paused, `playAndPauseBeforeCanPlay(${token}) element should be paused.`);
+}
+
+manager.runTests(gSmallTests, initTest);
+
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_play_twice.html b/dom/media/test/test_play_twice.html
new file mode 100644
index 0000000000..e94f28a031
--- /dev/null
+++ b/dom/media/test/test_play_twice.html
@@ -0,0 +1,95 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test playback of media files that should play OK</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function startTest(test, token) {
+ var video = document.createElement('video');
+ video.token = token;
+ manager.started(token);
+ video.src = test.name;
+ video.name = test.name;
+ video.playingCount = 0;
+ video._playedOnce = false;
+
+ var check = function(t, v) { return function() {
+ checkMetadata(t.name, v, test);
+ }}(test, video);
+
+ var noLoad = function(t, v) { return function() {
+ ok(false, t.name + " should not fire 'load' event");
+ }}(test, video);
+
+ function finish(v) {
+ removeNodeAndSource(v);
+ manager.finished(v.token);
+ }
+
+ function mayFinish(v) {
+ if (v.seenEnded && v.seenSuspend) {
+ finish(v);
+ }
+ }
+
+ var checkEnded = function(t, v) { return function() {
+ if (t.duration) {
+ ok(Math.abs(v.currentTime - t.duration) < 0.1,
+ t.name + " current time at end: " + v.currentTime);
+ }
+
+ is(v.readyState, v.HAVE_CURRENT_DATA, t.name + " checking readyState");
+ ok(v.ended, t.name + " checking playback has ended");
+ ok(v.playingCount > 0, "Expect at least one playing event");
+ v.playingCount = 0;
+
+ if (v._playedOnce) {
+ v.seenEnded = true;
+ mayFinish(v);
+ } else {
+ v._playedOnce = true;
+ v.play();
+ }
+ }}(test, video);
+
+ var checkSuspended = function(t, v) { return function() {
+ if (v.seenSuspend) {
+ return;
+ }
+
+ v.seenSuspend = true;
+ ok(true, t.name + " got suspend");
+ mayFinish(v);
+ }}(test, video);
+
+ var checkPlaying = function(t, v) { return function() {
+ is(t.name, v.name, "Should be testing file we think we're testing...");
+ v.playingCount++;
+ }}(test, video);
+
+ video.addEventListener("load", noLoad);
+ video.addEventListener("loadedmetadata", check);
+ video.addEventListener("playing", checkPlaying);
+
+ // We should get "ended" and "suspend" events for every resource
+ video.addEventListener("ended", checkEnded);
+ video.addEventListener("suspend", checkSuspended);
+
+ document.body.appendChild(video);
+ video.play();
+}
+
+manager.runTests(gReplayTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_playback.html b/dom/media/test/test_playback.html
new file mode 100644
index 0000000000..5e28861e93
--- /dev/null
+++ b/dom/media/test/test_playback.html
@@ -0,0 +1,108 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test playback of media files that should play OK</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function startTest(test, token) {
+ var video = document.createElement('video');
+ video.preload = "metadata";
+ video.token = token;
+ video.prevTime = 0;
+ video.seenEnded = false;
+ video.seenSuspend = false;
+
+ var handler = {
+ "ontimeout": function() {
+ Log(token, "timed out: ended=" + video.seenEnded + ", suspend=" + video.seenSuspend);
+ }
+ };
+ manager.started(token, handler);
+
+ video.src = test.name;
+ video.name = test.name;
+
+ var check = function(t, v) { return function() {
+ is(t.name, v.name, t.name + ": Name should match #1");
+ checkMetadata(t.name, v, t);
+ }}(test, video);
+
+ var noLoad = function(t, v) { return function() {
+ ok(false, t.name + " should not fire 'load' event");
+ }}(test, video);
+
+ var noError = function(t, v) { return function() {
+ ok(false, t.name + " should not fire 'error' event " + v.error.message);
+ }}(test, video);
+
+ var finish = function() {
+ video.finished = true;
+ video.removeEventListener("timeupdate", timeUpdate);
+ removeNodeAndSource(video);
+ manager.finished(video.token);
+ }
+
+ // We should get "ended" and "suspend" events to finish the test.
+ var mayFinish = function() {
+ if (video.seenEnded && video.seenSuspend) {
+ finish();
+ }
+ }
+
+ var checkEnded = function(t, v) { return function() {
+ is(t.name, v.name, t.name + ": Name should match #2");
+ checkMetadata(t.name, v, test);
+ is(v.readyState, v.HAVE_CURRENT_DATA, t.name + " checking readyState");
+ ok(v.ended, t.name + " checking playback has ended");
+ ok(!v.finished, t.name + " shouldn't be finished");
+ ok(!v.seenEnded, t.name + " shouldn't be ended");
+
+ v.seenEnded = true;
+ mayFinish();
+ }}(test, video);
+
+ var checkSuspended = function(t, v) { return function() {
+ if (v.seenSuspend) {
+ return;
+ }
+ is(t.name, v.name, t.name + ": Name should match #3");
+
+ v.seenSuspend = true;
+ mayFinish();
+ }}(test, video);
+
+ var timeUpdate = function(t, v) { return function() {
+ if (v.prevTime > v.currentTime) {
+ ok(false, t.name + " time should run forwards: p=" +
+ v.prevTime + " c=" + v.currentTime);
+ }
+ v.prevTime = v.currentTime;
+ }}(test, video);
+
+ video.addEventListener("load", noLoad);
+ video.addEventListener("error", noError);
+ video.addEventListener("loadedmetadata", check);
+ video.addEventListener("timeupdate", timeUpdate);
+
+ // We should get "ended" and "suspend" events for every resource
+ video.addEventListener("ended", checkEnded);
+ video.addEventListener("suspend", checkSuspended);
+
+ document.body.appendChild(video);
+ video.play();
+}
+
+manager.runTests(gPlayTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_playback_and_bfcache.html b/dom/media/test/test_playback_and_bfcache.html
new file mode 100644
index 0000000000..3b1236c530
--- /dev/null
+++ b/dom/media/test/test_playback_and_bfcache.html
@@ -0,0 +1,72 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <meta charset="utf-8">
+ <title>Test media playback and bfcache</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" href="/tests/SimpleTest/test.css"/>
+ <script>
+ SimpleTest.requestFlakyTimeout("Need some timer to wait for the audio to play");
+ SimpleTest.waitForExplicitFinish();
+ var duration = 0;
+
+ // The test opens a page and another page with a media element is loaded.
+ // The media element plays an audio file and starts again and sends
+ // statistics about it and then history.back() is called. The test waits
+ // for 1s + duration of the audio file and goes forward. The audio playback
+ // shouldn't have progressed while the page was in the bfcache.
+ function test() {
+ let bc1 = new BroadcastChannel("bc1");
+ let pageshow1Count = 0;
+ bc1.onmessage = function(e) {
+ if (e.data == "pageshow") {
+ ++pageshow1Count;
+ info("Page 1 pageshow " + pageshow1Count);
+ if (pageshow1Count == 1) {
+ bc1.postMessage("loadNext");
+ } else if (pageshow1Count == 2) {
+ setTimeout(function() {
+ bc1.postMessage("forward");
+ bc1.close();
+ }, (Math.round(duration) + 1) * 1000);
+ }
+ }
+ };
+
+ let bc2 = new BroadcastChannel("bc2");
+ let pageshow2Count = 0;
+ let statisticsCount = 0;
+ bc2.onmessage = function(e) {
+ if (e.data.event == "pageshow") {
+ ++pageshow2Count;
+ info("Page 2 pageshow " + pageshow2Count);
+ if (pageshow2Count == 2) {
+ ok(e.data.persisted, "Should have persisted the page.");
+ bc2.postMessage("statistics");
+ }
+ } else {
+ ++statisticsCount;
+ if (statisticsCount == 1) {
+ duration = e.data.duration;
+ bc2.postMessage("back");
+ } else {
+ is(statisticsCount, 2, "Should got two play events.");
+ ok(e.data.currentTime < e.data.duration,
+ "Should have stopped the playback while the page was in bfcache." +
+ "currentTime: " + e.data.currentTime + " duration: " + e.data.duration);
+ bc2.close();
+ SimpleTest.finish();
+ }
+ }
+ };
+
+ window.open("file_playback_and_bfcache.html", "", "noopener");
+ }
+ </script>
+</head>
+<body onload="test()">
+<p id="display"></p>
+<div id="content" style="display: none"></div>
+<pre id="test"></pre>
+</body>
+</html>
diff --git a/dom/media/test/test_playback_errors.html b/dom/media/test/test_playback_errors.html
new file mode 100644
index 0000000000..7b3f046099
--- /dev/null
+++ b/dom/media/test/test_playback_errors.html
@@ -0,0 +1,48 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test playback of media files that should have errors</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function startTest(test, token) {
+ var video = document.createElement('video');
+ manager.started(token);
+ video._errorCount = 0;
+ video._ignore = false;
+ function endedTest(v) {
+ if (v._ignore)
+ return;
+ v._ignore = true;
+ v.remove();
+ manager.finished(token);
+ }
+ var checkError = function(t, v) { return function(evt) {
+ v._errorCount++;
+ is(v._errorCount, 1, t.name + " only one error fired");
+ endedTest(v);
+ }}(test, video);
+ var checkEnded = function(t, v) { return function() {
+ ok(false, t.name + " successfully played");
+ endedTest(v);
+ }}(test, video);
+ video.addEventListener("error", checkError);
+ video.addEventListener("ended", checkEnded);
+ video.src = test.name;
+ document.body.appendChild(video);
+ video.play();
+}
+
+manager.runTests(gErrorTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_playback_hls.html b/dom/media/test/test_playback_hls.html
new file mode 100644
index 0000000000..7d5c777fc1
--- /dev/null
+++ b/dom/media/test/test_playback_hls.html
@@ -0,0 +1,91 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test playback of HLS with simple m3u8 that should play OK</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function startTest(test, token) {
+ var video = document.createElement('video');
+ video.preload = "metadata";
+ video.token = token;
+ video.prevTime = 0;
+ video.seenEnded = false;
+
+ var handler = {
+ "ontimeout": function() {
+ Log(token, "timed out: ended=" + video.seenEnded);
+ }
+ };
+ manager.started(token, handler);
+
+ video.src = test.name;
+ video.name = test.name;
+
+ var check = function(t, v) { return function() {
+ is(t.name, v.name, t.name + ": Name should match #1");
+ checkMetadata(t.name, v, t);
+ }}(test, video);
+
+ var noLoad = function(t, v) { return function() {
+ ok(false, t.name + " should not fire 'load' event");
+ }}(test, video);
+
+ var finish = function() {
+ video.finished = true;
+ video.removeEventListener("timeupdate", timeUpdate);
+ removeNodeAndSource(video);
+ manager.finished(video.token);
+ }
+
+ // We should get "ended" events to finish the test.
+ var mayFinish = function() {
+ if (video.seenEnded) {
+ finish();
+ }
+ }
+
+ var checkEnded = function(t, v) { return function() {
+ is(t.name, v.name, t.name + ": Name should match #2");
+ checkMetadata(t.name, v, test);
+ is(v.readyState, v.HAVE_CURRENT_DATA, t.name + " checking readyState");
+ ok(v.ended, t.name + " checking playback has ended");
+ ok(!v.finished, t.name + " shouldn't be finished");
+ ok(!v.seenEnded, t.name + " shouldn't be ended");
+
+ v.seenEnded = true;
+ mayFinish();
+ }}(test, video);
+
+ var timeUpdate = function(t, v) { return function() {
+ if (v.prevTime > v.currentTime) {
+ ok(false, t.name + " time should run forwards: p=" +
+ v.prevTime + " c=" + v.currentTime);
+ }
+ v.prevTime = v.currentTime;
+ }}(test, video);
+
+ video.addEventListener("load", noLoad);
+ video.addEventListener("loadedmetadata", check);
+ video.addEventListener("timeupdate", timeUpdate);
+
+ // We should get "ended" events for the hls resource
+ video.addEventListener("ended", checkEnded);
+
+ document.body.appendChild(video);
+ video.play();
+}
+
+manager.runTests(gHLSTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_playback_rate.html b/dom/media/test/test_playback_rate.html
new file mode 100644
index 0000000000..b44ebe27ee
--- /dev/null
+++ b/dom/media/test/test_playback_rate.html
@@ -0,0 +1,175 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test for the playbackRate property </title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type='application/javascript'>
+
+let manager = new MediaTestManager;
+
+function rangeCheck(lhs, rhs, threshold) {
+ var diff = Math.abs(lhs - rhs);
+ if (diff < threshold) {
+ return true;
+ }
+ return false;
+}
+
+function checkPlaybackRate(wallclock, media, expected, threshold) {
+ if (rangeCheck(media / wallclock, expected, threshold)) {
+ return true;
+ }
+ return false;
+}
+
+// Those value are expected to match those at the top of HTMLMediaElement.cpp.
+let VERY_SLOW_RATE = 1 / 32,
+ SLOW_RATE = 1 / 16,
+ FAST_RATE = 16,
+ VERY_FAST_RATE = 20,
+ NULL_RATE = 0.0;
+
+function ontimeupdate(e) {
+ var t = e.target;
+ // Skip short files for SoundTouch doesn't work well on small number of samples.
+ if (t.gotEnded || t.duration < 2) {
+ return;
+ }
+ t.testedForSlowdown = true;
+ if (t.currentTime > t.duration / 2) {
+ t.oldCurrentTime = t.currentTime;
+ t.timestamp = Date.now();
+ var delta = t.oldCurrentTime,
+ delta_wallclock = (t.timestamp - t.startTimestamp - t.bufferingTime) / 1000;
+
+ t.preservesPitch = false;
+ is(t.preservesPitch, false, t.name + ": If we disable the pitch preservation, it should appear as such.");
+
+ t.bufferingTime = 0;
+
+ is(t.playbackRate, SLOW_RATE, t.name + ": The playback rate shoud be "+SLOW_RATE+".");
+ ok(checkPlaybackRate(delta_wallclock, delta, SLOW_RATE, 0.25), t.name + ": We are effectively slowing down playback. (" + delta_wallclock + ", " + delta + ")");
+ t.removeEventListener("timeupdate", ontimeupdate);
+ t.addEventListener("pause", onpaused);
+ t.playbackRate = NULL_RATE;
+ t.oldCurrentTime = t.currentTime;
+ setTimeout(function() {
+ afterNullPlaybackRate(e);
+ }, 100);
+ }
+}
+
+function onpaused(e) {
+ var t = e.target;
+ t.pausedReceived = true;
+}
+
+function afterNullPlaybackRate(e) {
+ var t = e.target;
+
+ // skip if we have received 'ended' event or 'ended' event is pending.
+ if (t.gotEnded || t.ended) {
+ return;
+ }
+
+ t.testedForNull = true;
+
+ ok(t.currentTime == t.oldCurrentTime, t.name + ": Current time should not change when playbackRate is null (" + t.currentTime + " " + t.oldCurrentTime + ").");
+ ok(!t.paused, t.name + ": The element should not be in paused state.");
+ t.removeEventListener("paused", onpaused);
+ is(t.pausedReceived, undefined, t.name + ": Paused event should not have been received.");
+ t.timestamp = Date.now();
+ t.oldCurrentTime = t.currentTime;
+ t.playbackRate = VERY_FAST_RATE;
+ is(t.playbackRate, VERY_FAST_RATE, t.name + ": Playback rate should be clamped to " + VERY_FAST_RATE + ".");
+}
+
+function onended(e) {
+ var t = e.target;
+ t.gotEnded = true;
+
+ t.bufferingTime = 0;
+ // If we got "ended" too early, skip these tests.
+ if (t.testedForSlowdown && t.testedForNull) {
+ is(t.playbackRate, FAST_RATE, t.name + ": The playback rate should still be "+FAST_RATE+".");
+ ok(!t.muted, t.name + ": The audio should be muted when playing at high speed, but should not appear as such.");
+ is(t.currentTime, t.duration, t.name + ": Current time should be equal to the duration (not change by playback rate).");
+ }
+ finish_test(t);
+}
+
+function onratechange(e) {
+ if (!e.target.ratechangecount) {
+ e.target.ratechangecount = 0;
+ }
+ e.target.ratechangecount++;
+}
+
+function finish_test(element) {
+ removeNodeAndSource(element);
+ manager.finished(element.token);
+}
+
+// These two functions handle the case when the playback pauses for buffering. It
+// adjusts the timestamps to be accurate. Despite the fact that the web servers
+// is supposed to be on the same machine, buffering pauses can occur (rarely,
+// but still).
+function onplaying(e) {
+ var t = e.target;
+ if (t.bufferingTimestamp != undefined) {
+ t.bufferingTime += (Date.now() - t.bufferingTimestamp);
+ t.bufferingTimestamp = undefined;
+ }
+}
+
+function onwaiting(e) {
+ var t = e.target;
+ t.bufferingTimestamp = Date.now();
+}
+
+function onvolumechange(e) {
+ ok(false, e.target.name + ": We should not receive a volumechange event when changing the playback rate.");
+}
+
+function startTest(test, token) {
+ let elemType = /^audio/.test(test.type) ? "audio" : "video";
+ let element = document.createElement(elemType);
+ element.src = test.name;
+ element.name = test.name;
+ element.preload = "metadata";
+ element.token = token;
+ element.controls = true;
+ element.bufferingTime = 0;
+ document.body.appendChild(element);
+ element.addEventListener("ratechange", onratechange);
+ element.addEventListener("timeupdate", ontimeupdate);
+ element.addEventListener("ended", onended);
+ element.addEventListener("waiting", onwaiting);
+ element.addEventListener("playing", onplaying);
+ element.addEventListener("volumechange", onvolumechange);
+ manager.started(token);
+ element.startTimestamp = Date.now();
+ is(element.preservesPitch, true, test.name + ": Pitch preservation should be enabled by default.");
+ element.addEventListener("loadedmetadata", function() {
+ is(element.playbackRate, 1.0, test.name + ": playbackRate should be initially 1.0");
+ is(element.defaultPlaybackRate, 1.0, test.name + ": defaultPlaybackRate should be initially 1.0");
+ element.playbackRate = VERY_SLOW_RATE;
+ is(element.playbackRate, VERY_SLOW_RATE, test.name + ": PlaybackRate should be " + VERY_SLOW_RATE + ".");
+ element.play();
+ element.playbackRate = SLOW_RATE;
+ });
+}
+
+manager.runTests(gPlayedTests, startTest);
+
+</script>
+</pre>
+<div id="elements">
+</div>
+</body>
+</html>
diff --git a/dom/media/test/test_playback_rate_playpause.html b/dom/media/test/test_playback_rate_playpause.html
new file mode 100644
index 0000000000..fb57e50942
--- /dev/null
+++ b/dom/media/test/test_playback_rate_playpause.html
@@ -0,0 +1,66 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test that the playbackRate property is not reset when resuming the playback</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type='application/javascript'>
+
+let manager = new MediaTestManager;
+
+function ontimeupdate(e) {
+ var t = e.target;
+ if (t.currentTime != 0.0) {
+ info(t.token + " t.currentTime=" + t.currentTime +"\n");
+ t.removeEventListener("timeupdate", ontimeupdate);
+ t.pause();
+ is(t.playbackRate, 0.5, "PlaybackRate should not have changed after pause.");
+ } else {
+ info(t.token + " t.currentTime == 0.0.\n");
+ }
+}
+
+function onpaused(e) {
+ var t = e.target;
+ dump(t.token + " onpaused.\n");
+ t.play();
+ is(t.playbackRate, 0.5, "PlaybackRate should not have changed after resuming playback.");
+ finish_test(t);
+}
+
+function finish_test(element) {
+ dump(element.token + " finish_test.\n");
+ removeNodeAndSource(element);
+ manager.finished(element.token);
+}
+
+function startTest(test, token) {
+ let elemType = /^audio/.test(test.type) ? "audio" : "video";
+ let element = document.createElement(elemType);
+ element.src = test.name;
+ element.token = token;
+ element.controls = true;
+ element.playbackRate = 0.5;
+ element.preload = "metadata";
+ element.addEventListener("timeupdate", ontimeupdate);
+ element.addEventListener("pause", onpaused);
+ element.addEventListener("loadedmetadata", function() {
+ dump(element.token + " loadedmetadata\n");
+ element.play();
+ });
+ document.body.appendChild(element);
+ manager.started(token);
+}
+
+manager.runTests(gPlayedTests, startTest);
+
+</script>
+</pre>
+<div id="elements">
+</div>
+</body>
+</html>
diff --git a/dom/media/test/test_playback_reactivate.html b/dom/media/test/test_playback_reactivate.html
new file mode 100644
index 0000000000..dec5330ce5
--- /dev/null
+++ b/dom/media/test/test_playback_reactivate.html
@@ -0,0 +1,77 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test playback with dormant of media files that should play OK</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+/* This testcase wants to test a video element's playback is not break
+ by dormant.
+ When the metadata is loaded, we remove the video element to trigger dormant.
+ Then set a timer to append the video element back and play it.
+ Test pass if the video plays to the end.
+*/
+
+var manager = new MediaTestManager;
+
+function startTest(test, token) {
+ var video = document.createElement('video');
+ video.preload = "metadata";
+ video.token = token;
+
+ var handler = {
+ "ontimeout": function() {
+ Log(token, "timed out: ended=" + video.seenEnded + ", suspend=" + video.seenSuspend);
+ }
+ };
+ manager.started(token, handler);
+
+ video.src = test.name;
+ video.name = test.name;
+
+ var check = function(t, v) { return function() {
+ is(t.name, v.name, t.name + ": Name should match #1");
+ Log(v.token, "removeChild: " + v.name);
+ document.body.removeChild(v);
+ var appendAndPlayElement = function() {
+ Log(v.token, "appendChild: " + v.name);
+ document.body.appendChild(v);
+ Log(v.token, "Element play: " + v.name);
+ v.play();
+ }
+ setTimeout(appendAndPlayElement, 2000);
+ }}(test, video);
+
+ var finish = function() {
+ video.finished = true;
+ removeNodeAndSource(video);
+ manager.finished(video.token);
+ }
+
+ var checkEnded = function(t, v) { return function() {
+ is(t.name, v.name, t.name + ": Name should match #2");
+ checkMetadata(t.name, v, t);
+ is(v.readyState, v.HAVE_CURRENT_DATA, t.name + " checking readyState");
+ ok(v.ended, t.name + " checking playback has ended");
+
+ finish();
+ }}(test, video);
+
+
+ video.addEventListener("loadedmetadata", check);
+ video.addEventListener("ended", checkEnded);
+
+ document.body.appendChild(video);
+}
+
+manager.runTests(gSmallTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_played.html b/dom/media/test/test_played.html
new file mode 100644
index 0000000000..f6c05f7070
--- /dev/null
+++ b/dom/media/test/test_played.html
@@ -0,0 +1,288 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+<title>Test played member for media elements</title>
+<script type="text/javascript" src="/MochiKit/MochiKit.js"></script>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+<script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id='test'>
+<script class="testbody" type='application/javascript'>
+
+let manager = new MediaTestManager;
+
+function finish_test(element) {
+ removeNodeAndSource(element);
+ manager.finished(element.token);
+}
+
+// Check that a file has been played in its entirety.
+function check_full_file_played(element) {
+ element.addEventListener('ended', (function(e) {
+ let interval_count = e.target.played.length;
+ is(interval_count, 1, element.token + ": played.length must be 1");
+ is(element.played.start(0), 0, element.token + ": start time shall be 0");
+ is(element.played.end(0), e.target.duration, element.token + ": end time shall be duration");
+ finish_test(e.target);
+ }));
+}
+
+var tests = [
+// Without playing, check that player.played.length == 0.
+{
+ setup(element) {
+ element.addEventListener("loadedmetadata", function() {
+ is(element.played.length, 0, element.token + ": initial played.length equals zero");
+ finish_test(element);
+ });
+ },
+ name: "test1"
+},
+// Play the file, test the range we have.
+{
+ setup(element) {
+ check_full_file_played(element);
+ element.play();
+ },
+ name: "test2"
+},
+
+// Play the second half of the file, pause, play
+// an check we have only one range.
+{
+ setup (element) {
+ element.onended = function (e) {
+ var t = e.target;
+ t.onended = null;
+ check_full_file_played(t);
+ t.pause();
+ t.currentTime = 0;
+ t.play();
+ };
+ element.addEventListener("loadedmetadata", function() {
+ element.currentTime = element.duration / 2;
+ element.play();
+ });
+ },
+ name: "test3"
+},
+
+// Play the first half of the file, seek back, while
+// continuing to play. We shall have only one range.
+{
+ setup (element) {
+ let onTimeUpdate = function() {
+ if (element.currentTime > element.duration / 2) {
+ info(element.token + ": currentTime=" + element.currentTime + ", duration=" + element.duration);
+ element.removeEventListener("timeupdate", onTimeUpdate);
+ element.pause();
+ var oldEndRange = element.played.end(0);
+ element.currentTime = element.duration / 4;
+ is(element.played.end(0), oldEndRange,
+ element.token + ": When seeking back, |played| should not be changed");
+ element.play();
+ }
+ }
+ element.addEventListener("timeupdate", onTimeUpdate);
+ check_full_file_played(element);
+ element.play();
+ },
+ name: "test4"
+},
+
+// Play and seek to have two ranges, and check that, as well a
+// boundaries.
+{
+ setup (element) {
+ let seekTarget = 0;
+ let onTimeUpdate = function() {
+ if (element.currentTime > element.duration / 2) {
+ info(element.token + ": currentTime=" + element.currentTime + ", duration=" + element.duration);
+ element.removeEventListener("timeupdate", onTimeUpdate);
+ element.pause();
+ // Remember seek target for later comparison since duration may change
+ // during playback.
+ seekTarget = element.currentTime = element.duration / 10;
+ element.currentTime = seekTarget;
+ element.play();
+ }
+ }
+
+ element.addEventListener("loadedmetadata", function() {
+ element.addEventListener("timeupdate", onTimeUpdate);
+ });
+
+
+ element.addEventListener("ended", (function() {
+ if(element.played.length > 1) {
+ is(element.played.length, 2, element.token + ": element.played.length == 2");
+ is(element.played.start(1), seekTarget, element.token + ": we should have seeked forward by one tenth of the duration");
+ is(element.played.end(1), element.duration, element.token + ": end of second range shall be the total duration");
+ }
+ is(element.played.start(0), 0, element.token + ": start of first range shall be 0");
+ finish_test(element);
+ }));
+
+ element.play();
+ },
+ name: "test5"
+},
+
+// Play to create two ranges, in the reverse order. check that they are sorted.
+{
+ setup (element) {
+ function end() {
+ element.pause();
+ let p = element.played;
+ ok(p.length >= 1, element.token + ": There should be at least one range=" + p.length);
+ is(p.start(0), seekTarget, element.token + ": Start of first range should be the sixth of the duration");
+ ok(p.end(p.length - 1) > 5 * element.duration / 6, element.token + ": End of last range should be greater that five times the sixth of the duration");
+ finish_test(element);
+ }
+
+ let seekTarget = 0;
+ function pauseseekrestart() {
+ element.pause();
+ // Remember seek target for later comparison since duration may change
+ // during playback.
+ seekTarget = element.duration / 6;
+ element.currentTime = seekTarget;
+ element.play();
+ }
+
+ function onTimeUpdate_pauseseekrestart() {
+ if (element.currentTime > 5 * element.duration / 6) {
+ element.removeEventListener("timeupdate", onTimeUpdate_pauseseekrestart);
+ pauseseekrestart();
+ element.addEventListener("timeupdate", onTimeUpdate_end);
+ }
+ }
+
+ function onTimeUpdate_end() {
+ if (element.currentTime > 3 * element.duration / 6) {
+ element.removeEventListener("timeupdate", onTimeUpdate_end);
+ end();
+ }
+ }
+
+ element.addEventListener("timeupdate", onTimeUpdate_pauseseekrestart);
+
+ element.addEventListener('loadedmetadata', function() {
+ element.currentTime = 4 * element.duration / 6;
+ element.play();
+ });
+ },
+ name: "test6"
+},
+// Seek repeatedly without playing. No range should appear.
+{
+ setup(element) {
+ let index = 1;
+
+ element.addEventListener('seeked', function() {
+ index++;
+ element.currentTime = index * element.duration / 5;
+ is(element.played.length, 0, element.token + ": played.length should be 0");
+ if (index == 5) {
+ finish_test(element);
+ }
+ });
+
+ element.addEventListener('loadedmetadata', function() {
+ element.currentTime = element.duration / 5;
+ });
+ },
+ name: "test7"
+},
+// Play for a bit, then seek to the end: two ranges when "ended" is received.
+// Load another resource (invalid): no range
+// Load another resource (valid): still no range
+{
+ setup(element) {
+ let index = 1;
+ let seeked = false;
+
+ element.addEventListener('timeupdate', function() {
+ index++;
+ if (index == 2) {
+ is(element.played.length, 1);
+ element.currentTime = element.currentTime + element.duration / 100;
+ }
+ if (seeked) {
+ is(element.played.length, 2, "After seeking, played.length should be 2");
+ }
+ });
+ element.addEventListener('seeked', function() {
+ seeked = true;
+ });
+ element.addEventListener('ended', function() {
+ // For very short media file, timing is unreliable, it can be that the
+ // file has finished playing, before the seek, and so there is only one
+ // range in `played`.
+ if (seeked) {
+ is(element.played.length, 2, element.token + "After ended, played.length should be 2");
+ } else {
+ is(element.played.length, 1, element.token + "After ended, a short media file played.length should be 1 if it has ended before the seek");
+ }
+ let src_bck = element.src;
+ element.src = "bogus.duh";
+ is(element.played.length, 0, "After subsequent invalid load, played.length should be 0");
+ element.src = src_bck;
+ is(element.played.length, 0, "After subsequent valid load, played.length should be 0");
+ finish_test(element);
+ });
+ element.addEventListener('loadedmetadata', function() {
+ element.play();
+ });
+ },
+ name: "test8"
+}
+];
+
+function createTestArray() {
+ var A = [];
+ for (var i=0; i<tests.length; i++) {
+ for (var k=0; k<gPlayedTests.length; k++) {
+ var t = {};
+ t.setup = tests[i].setup;
+ t.name = tests[i].name + "-" + gPlayedTests[k].name;
+ t.type = gPlayedTests[k].type;
+ t.src = gPlayedTests[k].name;
+ A.push(t);
+ }
+ }
+ return A;
+}
+
+function startTest(test, token) {
+ var elemType = getMajorMimeType(test.type);
+ var element = document.createElement(elemType);
+ element.src = test.src;
+ element.token = token;
+ element.preload = "metadata";
+ test.setup(element);
+ manager.started(token);
+
+ // Log events for debugging.
+ var events = ["suspend", "play", "canplay", "canplaythrough", "loadstart", "loadedmetadata",
+ "loadeddata", "playing", "ended", "error", "stalled", "emptied", "abort",
+ "waiting", "pause"];
+ function logEvent(e) {
+ var v = e.target;
+ info(v.token + ": got " + e.type);
+ }
+ events.forEach(function(e) {
+ element.addEventListener(e, logEvent);
+ });
+
+}
+
+
+manager.runTests(createTestArray(), startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_preload_actions.html b/dom/media/test/test_preload_actions.html
new file mode 100644
index 0000000000..d027147c71
--- /dev/null
+++ b/dom/media/test/test_preload_actions.html
@@ -0,0 +1,581 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=548523
+-->
+<head>
+ <title>Test for Bug 548523</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=548523">Mozilla Bug 548523</a>
+<p id="display"></p>
+<div id="content" style="display: none">
+
+</div>
+<!-- <button onClick="SimpleTest.finish();">Finish</button> -->
+<div>Tests complete: <span id="log" style="font-size: small;"></span></div>
+<pre id="test">
+<script type="application/javascript">
+
+/** Test for Bug 548523 **/
+
+SimpleTest.requestCompleteLog();
+var manager = new MediaTestManager;
+
+manager.onFinished = function() {
+ is(gotLoadEvent, true, "Should not have delayed the load event indefinitely");
+};
+
+var test = getPlayableVideo(gSeekTests);
+var baseName = test.name;
+var gTest = test;
+var bogusSrc = "bogus.duh";
+var bogusType = "video/bogus";
+var gotLoadEvent = false;
+var finished = false;
+
+addLoadEvent(function() {gotLoadEvent=true;});
+
+function log(m) {
+ var l = document.getElementById("log");
+ l.innerHTML += m;
+}
+
+function maybeFinish(v, n) {
+ if (v._finished) {
+ return;
+ }
+ v._finished = true;
+ log(n + ",");
+ removeNodeAndSource(v);
+ manager.finished(v.token);
+}
+
+function filename(uri) {
+ return uri.substr(uri.lastIndexOf("/")+1);
+}
+
+// Every test must have a setup(v) function, and must call maybeFinish() when test is complete.
+var tests = [
+ {
+ // 1. Add preload:none video with src to document. Load should halt at NETWORK_IDLE and HAVE_NOTHING,
+ // after receiving a suspend event. Should not receive loaded events until after we call load().
+ // Note the suspend event is explictly sent by our "stop the load" code, but other tests can't rely
+ // on it for the preload:metadata case, as there can be multiple suspend events when loading metadata.
+ suspend(e) {
+ var v = e.target;
+ is(v._gotLoadStart, true, "(1) Must get loadstart.");
+ is(v._gotLoadedMetaData, false, "(1) Must not get loadedmetadata.");
+ is(v.readyState, v.HAVE_NOTHING, "(1) ReadyState must be HAVE_NOTHING");
+ // bug 962949
+ // is(v.networkState, v.NETWORK_IDLE, "(1) NetworkState must be NETWORK_IDLE");
+ maybeFinish(v, 1);
+ },
+
+ setup(v) {
+ v._gotLoadStart = false;
+ v._gotLoadedMetaData = false;
+ v.preload = "none";
+ v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;});
+ v.addEventListener("loadstart", function(e){v._gotLoadStart = true;});
+ v.addEventListener("suspend", this.suspend);
+ v.src = test.name;
+ document.body.appendChild(v); // Causes implicit load, which will be halted due to preload:none.
+ },
+
+ name: "test1",
+ },
+ {
+ // 2. Add preload:metadata video with src to document. Should halt with NETWORK_IDLE, HAVE_CURRENT_DATA
+ // after suspend event and after loadedmetadata.
+ loadeddata(e) {
+ var v = e.target;
+ is(v._gotLoadStart, true, "(2) Must get loadstart.");
+ is(v._gotLoadedMetaData, true, "(2) Must get loadedmetadata.");
+ ok(v.readyState >= v.HAVE_CURRENT_DATA, "(2) ReadyState must be >= HAVE_CURRENT_DATA");
+ // bug 962949
+ // is(v.networkState, v.NETWORK_IDLE, "(2) NetworkState must be NETWORK_IDLE");
+ maybeFinish(v, 2);
+ },
+
+ setup(v) {
+ v._gotLoadStart = false;
+ v._gotLoadedMetaData = false;
+ v.preload = "metadata";
+ v.addEventListener("loadstart", function(e){v._gotLoadStart = true;});
+ v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;});
+ v.addEventListener("loadeddata", this.loadeddata);
+ v.src = test.name;
+ document.body.appendChild(v); // Causes implicit load, which will be halted after
+ // metadata due to preload:metadata.
+ },
+
+ name: "test2",
+ },
+ {
+ // 3. Add preload:auto to document. Should receive canplaythrough eventually.
+ canplaythrough(e) {
+ var v = e.target;
+ is(v._gotLoadStart, true, "(3) Must get loadstart.");
+ is(v._gotLoadedMetaData, true, "(3) Must get loadedmetadata.");
+ maybeFinish(v, 3);
+ },
+
+ setup(v) {
+ v._gotLoadStart = false;
+ v._gotLoadedMetaData = false;
+ v.preload = "auto";
+ v.addEventListener("loadstart", function(e){v._gotLoadStart = true;});
+ v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;});
+ v.addEventListener("canplaythrough", this.canplaythrough);
+ v.src = test.name; // Causes implicit load.
+ document.body.appendChild(v);
+ },
+
+ name: "test3",
+ },
+ {
+ // 4. Add preload:none video to document. Call play(), should load then play through.
+ suspend(e) {
+ var v = e.target;
+ if (v._gotSuspend) {
+ return; // We can receive multiple suspend events, like the one after download completes.
+ }
+ v._gotSuspend = true;
+ is(v._gotLoadStart, true, "(4) Must get loadstart.");
+ is(v._gotLoadedMetaData, false, "(4) Must not get loadedmetadata.");
+ is(v.readyState, v.HAVE_NOTHING, "(4) ReadyState must be HAVE_NOTHING");
+ // bug 962949
+ // is(v.networkState, v.NETWORK_IDLE, "(4) NetworkState must be NETWORK_IDLE");
+ v.play(); // Should load and play through.
+ },
+
+ ended(e) {
+ ok(true, "(4) Got playback ended");
+ maybeFinish(e.target, 4);
+ },
+
+ setup(v) {
+ v._gotLoadStart = false;
+ v._gotLoadedMetaData = false;
+ v._gotSuspend = false;
+ v.preload = "none";
+ v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;});
+ v.addEventListener("loadstart", function(e){v._gotLoadStart = true;});
+ v.addEventListener("suspend", this.suspend);
+ v.addEventListener("ended", this.ended);
+ v.src = test.name;
+ document.body.appendChild(v);
+ },
+
+ name: "test4",
+ },
+ {
+ // 5. preload:none video without resource, add to document, will implicitly start a
+ // preload:none load. Add a src, it shouldn't load.
+ suspend(e) {
+ var v = e.target;
+ is(v._gotLoadStart, true, "(5) Must get loadstart.");
+ is(v._gotLoadedMetaData, false, "(5) Must not get loadedmetadata.");
+ is(v.readyState, v.HAVE_NOTHING, "(5) ReadyState must be HAVE_NOTHING");
+ // bug 962949
+ // is(v.networkState, v.NETWORK_IDLE, "(5) NetworkState must be NETWORK_IDLE");
+ maybeFinish(v, 5);
+ },
+
+ setup(v) {
+ v._gotLoadStart = false;
+ v._gotLoadedMetaData = false;
+ v.preload = "none";
+ v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;});
+ v.addEventListener("loadstart", function(e){v._gotLoadStart = true;});
+ v.addEventListener("suspend", this.suspend);
+ document.body.appendChild(v); // Causes implicit load, which will be halted due to no resource.
+ v.src = test.name; // Load should start, and halt at preload:none.
+ },
+
+ name: "test5",
+ },
+ {
+ // 6. preload:none video without resource, add to document, will implicitly start a
+ // preload:none load. Add a source, it shouldn't load.
+ suspend(e) {
+ var v = e.target;
+ is(v._gotLoadStart, true, "(6) Must get loadstart.");
+ is(v._gotLoadedMetaData, false, "(6) Must not get loadedmetadata.");
+ is(v.readyState, v.HAVE_NOTHING, "(6) ReadyState must be HAVE_NOTHING");
+ // bug 962949
+ // is(v.networkState, v.NETWORK_IDLE, "(6) NetworkState must be NETWORK_IDLE");
+ maybeFinish(v, 6);
+ },
+
+ setup(v) {
+ v._gotLoadStart = false;
+ v._gotLoadedMetaData = false;
+ v.preload = "none";
+ v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;});
+ v.addEventListener("loadstart", function(e){v._gotLoadStart = true;});
+ v.addEventListener("suspend", this.suspend);
+ document.body.appendChild(v); // Causes implicit load, which will be halted due to no resource.
+ var s = document.createElement("source");
+ s.src = test.name;
+ s.type = test.type;
+ v.appendChild(s); // Load should start, and halt at preload:none.
+ },
+
+ name: "test6",
+ },
+ {
+ // 7. create a preload:none document with multiple sources, the first of which is invalid.
+ // Add to document, then play. It should load and play through the second source.
+ suspend(e) {
+ var v = e.target;
+ if (v._gotSuspend)
+ return; // We can receive multiple suspend events, like the one after download completes.
+ v._gotSuspend = true;
+ is(v._gotLoadStart, true, "(7) Must get loadstart.");
+ is(v._gotLoadedMetaData, false, "(7) Must not get loadedmetadata.");
+ is(v.readyState, v.HAVE_NOTHING, "(7) ReadyState must be HAVE_NOTHING");
+ // bug 962949
+ // is(v.networkState, v.NETWORK_IDLE, "(7) NetworkState must be NETWORK_IDLE");
+ v.play(); // Should load and play through.
+ },
+
+ ended(e) {
+ ok(true, "(7) Got playback ended");
+ var v = e.target;
+ is(v._gotErrorEvent, true, "(7) Should get error event from first source load failure");
+ maybeFinish(v, 7);
+ },
+
+ setup(v) {
+ v._gotLoadStart = false;
+ v._gotLoadedMetaData = false;
+ v.preload = "none";
+ v._gotErrorEvent = false;
+ v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;});
+ v.addEventListener("loadstart", function(e){v._gotLoadStart = true;});
+ v.addEventListener("suspend", this.suspend);
+ v.addEventListener("ended", this.ended);
+ var s1 = document.createElement("source");
+ s1.src = "not-a-real-file.404"
+ s1.type = test.type;
+ s1.addEventListener("error", function(e){v._gotErrorEvent = true;});
+ v.appendChild(s1);
+ var s2 = document.createElement("source");
+ s2.src = test.name;
+ s2.type = test.type;
+ v.appendChild(s2);
+ document.body.appendChild(v); // Causes implicit load, which will be halt at preload:none on the second resource.
+ },
+
+ name: "test7",
+ },
+ {
+ // 8. Change preload value from none to metadata should cause metadata to be loaded.
+ loadeddata(e) {
+ var v = e.target;
+ is(v._gotLoadedMetaData, true, "(8) Must get loadedmetadata.");
+ ok(v.readyState >= v.HAVE_CURRENT_DATA, "(8) ReadyState must be >= HAVE_CURRENT_DATA on suspend.");
+ // bug 962949
+ // is(v.networkState, v.NETWORK_IDLE, "(8) NetworkState must be NETWORK_IDLE when load is halted");
+ maybeFinish(v, 8);
+ },
+
+ setup(v) {
+ v._gotLoadedMetaData = false;
+ v.preload = "none";
+ v.addEventListener("loadstart", function(e){v.preload = "metadata";});
+ v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;});
+ v.addEventListener("loadeddata", this.loadeddata);
+ v.src = test.name; // Causes implicit load.
+ document.body.appendChild(v);
+ },
+
+ name: "test8",
+ },
+ /*{
+ // 9. Change preload value from metadata to auto should cause entire media to be loaded.
+ // For some reason we don't always receive the canplaythrough event, particuarly on this test.
+ // We've disabled this test until bug 568402 is fixed.
+ canplaythrough:
+ function(e) {
+ var v = e.target;
+ is(v._gotLoadStart, true, "(9) Must get loadstart.");
+ is(v._gotLoadedMetaData, true, "(9) Must get loadedmetadata.");
+ maybeFinish(v, 9);
+ },
+
+ setup:
+ function(v) {
+ v._gotLoadStart = false;
+ v._gotLoadedMetaData = false;
+ v.preload = "metadata";
+ v.addEventListener("loadstart", function(e){v._gotLoadStart = true;}, false);
+ v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;}, false);
+ v.addEventListener("loadeddata", function(){v.preload = "auto"}, false);
+ v.addEventListener("canplaythrough", this.canplaythrough, false);
+ v.src = test.name; // Causes implicit load.
+ document.body.appendChild(v);
+ },
+ },*/
+ {
+ // 10. Change preload value from none to auto should cause entire media to be loaded.
+ canplaythrough(e) {
+ var v = e.target;
+ is(v._gotLoadedMetaData, true, "(10) Must get loadedmetadata.");
+ maybeFinish(v, 10);
+ },
+
+ setup(v) {
+ v._gotLoadedMetaData = false;
+ v.preload = "none";
+ v.addEventListener("loadstart", function(e){v.preload = "auto";});
+ v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;});
+ v.addEventListener("canplaythrough", this.canplaythrough);
+ v.src = test.name; // Causes implicit load.
+ document.body.appendChild(v);
+ },
+
+ name: "test10",
+ },
+ {
+ // 11. Change preload value from none to metadata should cause metadata to load.
+ loadeddata(e) {
+ var v = e.target;
+ is(v._gotLoadedMetaData, true, "(11) Must get loadedmetadata.");
+ ok(v.readyState >= v.HAVE_CURRENT_DATA, "(11) ReadyState must be >= HAVE_CURRENT_DATA.");
+ // bug 962949
+ // is(v.networkState, v.NETWORK_IDLE, "(11) NetworkState must be NETWORK_IDLE.");
+ maybeFinish(v, 11);
+ },
+
+ setup(v) {
+ v._gotLoadedMetaData = false;
+ v.preload = "none";
+ v.addEventListener("loadstart", function(e){v.preload = "metadata";});
+ v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;});
+ v.addEventListener("loadeddata", this.loadeddata);
+ v.src = test.name; // Causes implicit load.
+ document.body.appendChild(v);
+ },
+
+ name: "test11",
+ },
+ {
+ // 13. Change preload value from auto to none after specifying a src
+ // should load according to preload none, no buffering should have taken place
+ suspend(e) {
+ var v = e.target;
+ is(v._gotLoadStart, true, "(13) Must get loadstart.");
+ is(v._gotLoadedMetaData, false, "(13) Must not get loadedmetadata.");
+ is(v.readyState, v.HAVE_NOTHING, "(13) ReadyState must be HAVE_NOTHING");
+ // bug 962949
+ // is(v.networkState, v.NETWORK_IDLE, "(13) NetworkState must be NETWORK_IDLE");
+ maybeFinish(v, 13);
+ },
+
+ setup(v) {
+ v._gotLoadStart = false;
+ v._gotLoadedMetaData = false;
+ v.preload = "auto";
+ v.src = test.name;
+ v.preload = "none";
+ v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;});
+ v.addEventListener("loadstart", function(e){v._gotLoadStart = true;});
+ v.addEventListener("suspend", this.suspend);
+ document.body.appendChild(v); // Causes implicit load, should load according to preload none
+ document.createElement("source");
+ },
+
+ name: "test13",
+ },
+ {
+ // 14. Add preload:metadata video with src to document. Play(), should play through.
+ loadeddata(e) {
+ var v = e.target;
+ is(v._gotLoadStart, true, "(14) Must get loadstart.");
+ is(v._gotLoadedMetaData, true, "(14) Must get loadedmetadata.");
+ ok(v.readyState >= v.HAVE_CURRENT_DATA, "(14) ReadyState must be >= HAVE_CURRENT_DATA");
+ // bug 962949
+ // is(v.networkState, v.NETWORK_IDLE, "(14) NetworkState must be NETWORK_IDLE");
+ v.play();
+ },
+
+ ended(e) {
+ ok(true, "(14) Got playback ended");
+ var v = e.target;
+ maybeFinish(v, 14);
+ },
+
+ setup(v) {
+ v._gotLoadStart = false;
+ v._gotLoadedMetaData = false;
+ v.preload = "metadata";
+ v.addEventListener("loadstart", function(e){v._gotLoadStart = true;});
+ v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;});
+ v.addEventListener("ended", this.ended);
+ v.addEventListener("loadeddata", this.loadeddata);
+ v.src = test.name;
+ document.body.appendChild(v); // Causes implicit load, which will be halted after
+ // metadata due to preload:metadata.
+ },
+
+ name: "test14",
+ },
+ {
+ // 15. Autoplay should override preload:none.
+ ended(e) {
+ ok(true, "(15) Got playback ended.");
+ var v = e.target;
+ maybeFinish(v, 15);
+ },
+
+ setup(v) {
+ v._gotLoadStart = false;
+ v._gotLoadedMetaData = false;
+ v.preload = "none";
+ v.autoplay = true;
+ v.addEventListener("loadstart", function(e){v._gotLoadStart = true;});
+ v.addEventListener("loadedmetadata", function(e){v._gotLoadedMetaData = true;});
+ v.addEventListener("ended", this.ended);
+ v.src = test.name; // Causes implicit load.
+ document.body.appendChild(v);
+
+ // Log events for debugging.
+ var events = ["suspend", "play", "canplay", "canplaythrough", "loadstart", "loadedmetadata",
+ "loadeddata", "playing", "ended", "error", "stalled", "emptied", "abort",
+ "waiting", "pause"];
+ function logEvent(e) {
+ info(e.target.token + ": got " + e.type);
+ }
+ events.forEach(function(e) {
+ v.addEventListener(e, logEvent);
+ });
+ },
+
+ name: "test15",
+ },
+ {
+ // 16. Autoplay should override preload:metadata.
+ ended(e) {
+ ok(true, "(16) Got playback ended.");
+ var v = e.target;
+ maybeFinish(v, 16);
+ },
+
+ setup(v) {
+ v.preload = "metadata";
+ v.autoplay = true;
+ v.addEventListener("ended", this.ended);
+ v.src = test.name; // Causes implicit load.
+ document.body.appendChild(v);
+ },
+
+ name: "test16",
+ },
+ {
+ // 17. On a preload:none video, adding autoplay should disable preload none, i.e. don't break autoplay!
+ ended(e) {
+ ok(true, "(17) Got playback ended.");
+ var v = e.target;
+ maybeFinish(v, 17);
+ },
+
+ setup(v) {
+ v.addEventListener("ended", this.ended);
+ v.preload = "none";
+ document.body.appendChild(v); // Causes implicit load, which will be halted due to preload:none.
+ v.autoplay = true;
+ v.src = test.name;
+ },
+
+ name: "test17",
+ },
+ {
+ // 18. On a preload='none' video, call play() before load algorithms's sync
+ // has run, the play() call should override preload='none'.
+ ended(e) {
+ ok(true, "(18) Got playback ended.");
+ var v = e.target;
+ maybeFinish(v, 18);
+ },
+
+ setup(v) {
+ v.addEventListener("ended", this.ended);
+ v.preload = "none";
+ v.src = test.name; // Schedules async section to continue load algorithm.
+ document.body.appendChild(v);
+ v.play(); // Should cause preload:none to be overridden.
+ },
+
+ name: "test18",
+ },
+ {
+ // 19. Set preload='auto' on first video source then switching preload='none' and swapping the video source to another.
+ // The second video should not start playing as it's preload state has been changed to 'none' from 'auto'
+ setup(v) {
+ v.preload = "auto";
+ v.src = test.name;
+ // add a listener for when the video has loaded, so we know preload auto has worked
+ v.onloadedmetadata = function() {
+ is(v.preload, "auto", "(19) preload is initially auto");
+ // set preload state to none and switch video sources
+ v.preload="none";
+ v.src = test.name + "?asdf";
+
+ v.onloadedmetadata = function() {
+ ok(false, "(19) 'loadedmetadata' shouldn't fire when preload is none");
+ }
+
+ var ontimeout = function() {
+ v.removeEventListener("suspend", onsuspend);
+ ok(false, "(19) 'suspend' should've fired");
+ maybeFinish(v, 19);
+ }
+ var cancel = setTimeout(ontimeout, 10000);
+
+ var onsuspend = function() {
+ v.removeEventListener("suspend", onsuspend);
+ clearTimeout(cancel);
+ is(v.readyState, 0, "(19) no buffering has taken place");
+ maybeFinish(v, 19);
+ }
+ v.addEventListener("suspend", onsuspend);
+ }
+ document.body.appendChild(v);
+ },
+
+ name: "test19",
+ }
+];
+
+var iterationCount = 0;
+function startTest(t, token) {
+ if (t == tests[0]) {
+ ++iterationCount;
+ info("iterationCount=" + iterationCount);
+ }
+ if (iterationCount == 2) {
+ // Do this series of tests on logically different resources
+ t.name = baseName + "?" + Math.floor(Math.random()*100000);
+ }
+ var v = document.createElement("video");
+ v.token = token;
+ t.setup(v);
+ manager.started(token);
+}
+
+var twiceTests = tests.concat(tests);
+SimpleTest.waitForExplicitFinish();
+SpecialPowers.pushPrefEnv({"set": [["media.cache_size", 40000]]}, beginTest);
+function beginTest() {
+ manager.runTests(twiceTests, startTest);
+}
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_preload_attribute.html b/dom/media/test/test_preload_attribute.html
new file mode 100644
index 0000000000..1e415035c5
--- /dev/null
+++ b/dom/media/test/test_preload_attribute.html
@@ -0,0 +1,44 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=479863
+-->
+<head>
+ <title>Test for Bug 479863</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=479863">Mozilla Bug 479863</a>
+<p id="display"></p>
+<div id="content" style="display: none">
+
+</div>
+
+<video id='v1'></video><audio id='a1'></audio>
+<video id='v2' preload="auto"></video><audio id='a2' preload="auto"></audio>
+
+<pre id="test">
+<script type="application/javascript">
+var v1 = document.getElementById('v1');
+var a1 = document.getElementById('a1');
+var v2 = document.getElementById('v2');
+var a2 = document.getElementById('a2');
+is(v1.getAttribute("preload"), null, "video preload via getAttribute should be null by default");
+is(a1.getAttribute("preload"), null, "video preload via getAttribute should be null by default");
+is(v1.preload, "", "v1.preload should be empty by default");
+is(a1.preload, "", "a1.preload should be empty by default");
+is(v2.preload, "auto", "v2.preload should be auto");
+is(a2.preload, "auto", "a2.preload should be auto");
+
+v1.preload = "auto";
+a1.preload = "auto";
+is(v1.preload, "auto", "video.preload should be auto");
+is(a1.preload, "auto", "audio.preload should be auto");
+is(v1.getAttribute("preload"), "auto", "video preload attribute should be auto");
+is(a1.getAttribute("preload"), "auto", "video preload attribute should be auto");
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_preload_suspend.html b/dom/media/test/test_preload_suspend.html
new file mode 100644
index 0000000000..7f1146360f
--- /dev/null
+++ b/dom/media/test/test_preload_suspend.html
@@ -0,0 +1,112 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test for Bug 479863</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var manager = new MediaTestManager;
+
+function checkSuspendCount(evt) {
+ var v = evt.target;
+ ++v.suspendCount;
+ is(v.networkState, v.NETWORK_IDLE, v.name + " got suspended, count=" + v.suspendCount);
+ if (v.suspendCount == v.expectedSuspendCount) {
+ removeNodeAndSource(v);
+ manager.finished(v.name);
+ }
+ if (v.suspendCount > v.expectedSuspendCount) {
+ ok(false, v.name + " got too many suspend events");
+ }
+}
+
+var tests = [
+ {
+ name: 'v1',
+ preload: 'none',
+ expectedSuspendCount: 2,
+ onsuspend(evt) {
+ checkSuspendCount(evt);
+ if (evt.target.suspendCount == 1) {
+ evt.target.preload = 'auto';
+ }
+ }
+ },
+ {
+ name: 'v2',
+ preload: 'auto',
+ expectedSuspendCount: 1,
+ onsuspend: checkSuspendCount
+ },
+ {
+ name: 'v3',
+ preload: 'none',
+ autoplay: true,
+ expectedSuspendCount: 1,
+ onsuspend: checkSuspendCount
+ },
+ {
+ name: 'v4',
+ preload: 'none',
+ expectedSuspendCount: 2,
+ onsuspend(evt) {
+ checkSuspendCount(evt);
+ if (evt.target.suspendCount == 1) {
+ evt.target.play();
+ }
+ }
+ },
+ // disable v5 since media element doesn't support 'load' event anymore.
+ /*{
+ name: 'v5',
+ preload: 'none',
+ expectedSuspendCount: 2,
+ onsuspend: function(evt) {
+ checkSuspendCount(evt);
+ if (evt.target.suspendCount == 1) {
+ evt.target.currentTime = 0.1;
+ }
+ }
+ },*/
+ {
+ name: 'v6',
+ preload: 'none',
+ expectedSuspendCount: 2,
+ onsuspend(evt) {
+ checkSuspendCount(evt);
+ if (evt.target.suspendCount == 1) {
+ evt.target.autoplay = true;
+ }
+ }
+ }
+];
+
+function startTest(test, token) {
+ var v = document.createElement("video");
+ v.name = test.name;
+ var key = Math.random();
+ v.src = "seek.ogv?key=" + key + "&id=" + v.name;
+ v.preload = test.preload;
+ v.suspendCount = 0;
+ v.expectedSuspendCount = test.expectedSuspendCount;
+ if (test.autoplay) {
+ v.autoplay = true;
+ }
+ v.onsuspend = test.onsuspend;
+ document.body.appendChild(v);
+ manager.started(v.name);
+}
+
+SimpleTest.waitForExplicitFinish();
+SpecialPowers.pushPrefEnv({"set": [["media.cache_size", 40000]]}, function() {
+ manager.runTests(tests, startTest);
+});
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_preserve_playbackrate_after_ui_play.html b/dom/media/test/test_preserve_playbackrate_after_ui_play.html
new file mode 100644
index 0000000000..98bdd66675
--- /dev/null
+++ b/dom/media/test/test_preserve_playbackrate_after_ui_play.html
@@ -0,0 +1,60 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title> Bug 1013933 - preserve playbackRate after clicking play button </title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script src="/tests/SimpleTest/EventUtils.js"></script>
+ <script type="application/javascript" src="browserElementTestHelpers.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+</head>
+<body>
+
+<div id="content">
+ <video width="320" height="240" id="video" controls mozNoDynamicControls preload="auto"></video>
+</div>
+
+<script type="text/javascript">
+/*
+ * Positions of the UI elements, relative to the upper-left corner of the
+ * <video> box.
+ */
+const videoHeight = 240;
+const playButtonWidth = 28;
+const playButtonHeight = 28;
+const playButtonCenterX = 0 + Math.round(playButtonWidth / 2);
+const playButtonCenterY = videoHeight - Math.round(playButtonHeight / 2);
+
+var expectedPlaybackRate = 0.5
+
+function runTest() {
+ var video = document.getElementById("video");
+ video.src = "audio.wav";
+ video.loop = true;
+ video.playbackRate = expectedPlaybackRate;
+
+ video.oncanplaythrough = function() {
+ video.oncanplaythrough = null;
+ is(video.paused, true, "video is not playing yet.");
+ is(video.playbackRate, expectedPlaybackRate,
+ "playbackRate is correct before clicking play button.");
+
+ // Click the play button
+ synthesizeMouse(video, playButtonCenterX, playButtonCenterY, { });
+ };
+
+ video.onplay = function() {
+ video.onplay = null;
+ is(video.paused, false, "video starts playing.");
+ is(video.playbackRate, expectedPlaybackRate,
+ "playbackRate is correct after clicking play button.");
+ video.pause();
+ SimpleTest.finish();
+ };
+}
+
+window.addEventListener("load", runTest);
+SimpleTest.waitForExplicitFinish();
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_progress.html b/dom/media/test/test_progress.html
new file mode 100644
index 0000000000..b958b2b335
--- /dev/null
+++ b/dom/media/test/test_progress.html
@@ -0,0 +1,52 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: progress events</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function do_progress(e) {
+ var v = e.target;
+ ok(!v._finished, "Check no progress events after completed for " + v._name);
+}
+
+function do_ended(e) {
+ var v = e.target;
+ ok(!v._finished, "Only one ended event for " + v._name);
+ v._finished = true;
+ v.removeEventListener("ended", do_ended);
+ v.removeEventListener("progress", do_progress);
+ removeNodeAndSource(v);
+ manager.finished(v.token);
+}
+
+function startTest(test, token) {
+ var type = /^video/.test(test.type) ? "video" : "audio";
+ var v = document.createElement(type);
+ v.token = token;
+ manager.started(token);
+ v.src = test.name;
+ v.autoplay = true;
+ v._name = test.name;
+ v._finished = false;
+ v.addEventListener("ended", do_ended);
+ v.addEventListener("progress", do_progress);
+ document.body.appendChild(v);
+}
+
+SimpleTest.waitForExplicitFinish();
+SpecialPowers.pushPrefEnv({"set": [["media.cache_size", 40000]]}, beginTest);
+function beginTest() {
+ manager.runTests(gProgressTests, startTest);
+}
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_reactivate.html b/dom/media/test/test_reactivate.html
new file mode 100644
index 0000000000..43675f7771
--- /dev/null
+++ b/dom/media/test/test_reactivate.html
@@ -0,0 +1,64 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test reactivation of a media element from a dead document</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+
+<iframe id="frame" src="reactivate_helper.html"></iframe>
+
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var elements;
+
+function playElement(e) {
+ // All elements played out, finish the test case.
+ if (!e) {
+ SimpleTest.finish();
+ return;
+ }
+
+ e.play();
+ info("Element play: " + e._name);
+ var reviveElement = function() {
+ document.body.appendChild(e);
+ e.onended = function() {
+ info("Element ended: " + e._name);
+ removeNodeAndSource(e);
+ // Play next element.
+ playElement(elements.pop());
+ }
+ }
+ setTimeout(reviveElement, 2000);
+}
+
+function loadedAll(elementList) {
+ elements = elementList;
+
+ // Log events for debugging.
+ var events = ["suspend", "play", "canplay", "canplaythrough", "loadstart", "loadedmetadata",
+ "loadeddata", "playing", "ended", "error", "stalled", "emptied", "abort",
+ "waiting", "pause"];
+ function logEvent(e) {
+ info(e.target._name + ": got " + e.type);
+ }
+ elementList.forEach(function(element) {
+ events.forEach(function(evt) {
+ element.addEventListener(evt, logEvent);
+ });
+ });
+
+ // Blow away the subframe
+ document.body.removeChild(document.getElementById("frame"));
+
+ // Play elements one by one to avoid hitting bug 847903 on MacOSX.
+ playElement(elements.pop());
+}
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_readyState.html b/dom/media/test/test_readyState.html
new file mode 100644
index 0000000000..17c363c503
--- /dev/null
+++ b/dom/media/test/test_readyState.html
@@ -0,0 +1,51 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: readyState</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<video id='v1'></video><audio id='a1'></audio>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+"use strict";
+var v1 = document.getElementById('v1');
+var a1 = document.getElementById('a1');
+var passed = "truthy";
+
+is(v1.readyState, 0);
+is(a1.readyState, 0);
+
+try {
+ v1.readyState = 0;
+} catch (e) {
+ passed = !passed;
+}
+try {
+ a1.readyState = 0;
+} catch (e) {
+ passed = !passed;
+}
+ok(passed === true,
+ "Setting readyState throws in strict mode (readonly attribute)");
+</script>
+
+<script class="testbody" type="text/javascript">
+var v1 = document.getElementById('v1');
+var a1 = document.getElementById('a1');
+var passed = false;
+
+is(v1.readyState, 0);
+is(a1.readyState, 0);
+
+try {
+ v1.readyState = 1;
+ a1.readyState = 1;
+ passed = v1.readyState === 0 && a1.readyState === 0;
+} catch(e) { }
+ok(passed, "Should not be able to set readyState (readonly attribute)");
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_referer.html b/dom/media/test/test_referer.html
new file mode 100644
index 0000000000..4ce78d5fb3
--- /dev/null
+++ b/dom/media/test/test_referer.html
@@ -0,0 +1,88 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=584480
+-->
+<head>
+ <title>Test for Bug 584480</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=584480">Mozilla Bug </a>
+<p id="display"></p>
+<div id="content" style="display: none">
+
+</div>
+<pre id="test">
+<script type="application/javascript">
+var media = [];
+
+function checkComplete() {
+ for (var i=0; i<media.length; ++i) {
+ if (!media[i]._complete) {
+ return;
+ }
+ }
+
+ SimpleTest.finish();
+}
+
+function removeNode(v) {
+ v.removeEventListener("error", loadError);
+ v.removeEventListener("loadedmetadata", loadedMetadata);
+ v.remove();
+ v.src = "";
+}
+
+function loadError(evt) {
+ // If no referer is sent then the sjs returns an error
+ ok(false, "check referer is sent with media request");
+ evt.target._complete = true;
+ checkComplete();
+ removeNode(evt.target);
+}
+
+function loadedMetadata(evt) {
+ // If a referer is sent then the sjs returns a valid media
+ ok(true, "check referer is sent with media request");
+ evt.target._complete = true;
+ checkComplete();
+ removeNode(evt.target);
+}
+
+// Create all media objects.
+for (var i=0; i<gSmallTests.length; ++i) {
+ var test = gSmallTests[i];
+ var type;
+ if (/^video/.test(test.type)) {
+ type = "video"
+ } else {
+ type = "audio";
+ }
+ var v = document.createElement(type);
+ if (!v.canPlayType(test.type)) {
+ continue;
+ }
+ // ensure metadata is loaded for default preload is none on b2g
+ v.preload = "metadata";
+ v.autoplay = "true";
+ v._complete = false;
+ v.addEventListener("error", loadError);
+ v.addEventListener("loadedmetadata", loadedMetadata);
+ v.src = 'referer.sjs?name=' + test.name + '&type=' + test.type;
+ document.body.appendChild(v); // Will start load.
+ media.push(v);
+}
+
+if (!media.length) {
+ todo(false, "No types supported");
+} else {
+ SimpleTest.waitForExplicitFinish();
+}
+</script>
+</pre>
+
+</body>
+</html>
diff --git a/dom/media/test/test_replay_metadata.html b/dom/media/test/test_replay_metadata.html
new file mode 100644
index 0000000000..dab022885f
--- /dev/null
+++ b/dom/media/test/test_replay_metadata.html
@@ -0,0 +1,120 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=467972
+-->
+<head>
+ <title>Test for Bug 467972</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=467972">Mozilla Bug 467972</a>
+
+<p id="display"></p>
+<div id="content" style="display: none">
+
+</div>
+<pre id="test">
+<script type="application/javascript">
+
+// Test for Bug 467972. Tests that when we play to end, seek to 0, and play again, that we don't
+// send/receive multiple loadeddata and loadedmetadata events.
+
+var manager = new MediaTestManager;
+
+function seekStarted(evt) {
+ var v = evt.target;
+ v._gotSeekStarted = true;
+}
+
+function seekEnded(evt) {
+ var v = evt.target;
+ v._gotSeekEnded = true;
+ v.play();
+}
+
+function loadedData(evt) {
+ var v = evt.target;
+ v._loadedDataCount++;
+ ok(v._loadedDataCount <= 1, "No more than 1 onloadeddata event for " + v._name);
+}
+
+function loadedMetaData(evt) {
+ var v = evt.target;
+ v._loadedMetaDataCount++;
+ ok(v._loadedMetaDataCount <= 1, "No more than 1 onloadedmetadata event for " + v._name);
+ checkMetadata(v._name, v, v._test);
+ v.play();
+}
+
+function playing(evt) {
+ evt.target._playingCount++;
+}
+
+function removeNodeAndListener(n) {
+ n.removeEventListener("loadedmetadata", loadedMetaData);
+ n.removeEventListener("ended", playbackEnded);
+ n.removeEventListener("playing", playing);
+ n.removeEventListener("loadeddata", loadedData);
+ n.removeEventListener("seeking", seekStarted);
+ n.removeEventListener("seeked", seekEnded);
+ removeNodeAndSource(n);
+}
+
+function playbackEnded(evt) {
+ var v = evt.target;
+ v._endCount++;
+ ok(v.currentTime >= v.duration-0.1 && v.currentTime <= v.duration + 0.1,
+ "CurrentTime (" + v.currentTime + ") should be around " + v.duration
+ + " for " + v._name);
+ if (!v._playedOnce) {
+ v.currentTime = 0;
+ ok(v.seeking, "seeking should be true for " + v._name);
+ ok(!v.ended, "ended shouldn't be true once seeking has begun for " + v._name);
+ v._playedOnce = true;
+ } else {
+ ok(v._gotSeekEnded, "Should have received seekended for " + v._name);
+ ok(v._gotSeekStarted, "Should have received seekstarted for " + v._name);
+ is(v._loadedDataCount, 1, "Should have 1 onloadeddata event for " + v._name);
+ is(v._loadedMetaDataCount, 1, "Should have 1 onloadedmetadata event for " + v._name);
+ is(v._endCount, 2, "Should have received two ended events for " + v._name);
+ ok(v._playingCount > 0, "Should have at least one playing event for " + v._name);
+ v._finished = true;
+ removeNodeAndListener(v);
+ manager.finished(v.token);
+ }
+}
+
+function startTest(test, token) {
+ var v = document.createElement('video');
+ v.preload = "auto";
+ v.token = token;
+ manager.started(token);
+ v.src = test.name;
+ v._name = test.name;
+ v._playedOnce = false;
+ v._gotSeekEnded = false;
+ v._gotSeekStarted = false;
+ v._loadedDataCount = 0;
+ v._loadedMetaDataCount = 0;
+ v._playingCount = 0;
+ v._endCount = 0;
+ v._test = test;
+ v._finished = false;
+ v.addEventListener("loadedmetadata", loadedMetaData);
+ v.addEventListener("ended", playbackEnded);
+ v.addEventListener("playing", playing);
+ v.addEventListener("loadeddata", loadedData);
+ v.addEventListener("seeking", seekStarted);
+ v.addEventListener("seeked", seekEnded);
+ document.body.appendChild(v);
+}
+
+manager.runTests(gSmallTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_reset_events_async.html b/dom/media/test/test_reset_events_async.html
new file mode 100644
index 0000000000..482ec55986
--- /dev/null
+++ b/dom/media/test/test_reset_events_async.html
@@ -0,0 +1,58 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=975270
+-->
+<head>
+ <meta charset="utf-8">
+ <title>Test for Bug </title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="application/javascript" src="manifest.js"></script>
+ <script type="application/javascript">
+
+ /** Test for Bug 975270 **/
+ // Test that 'emptied' and 'abort' events are fired asynchronously when re-starting
+ // media load.
+ SimpleTest.waitForExplicitFinish();
+
+ var a = document.createElement("audio");
+ a._abort = 0;
+ a._emptied = 0;
+ a.preload = "metadata"; // On B2G we default to preload:none.
+
+ is(a.networkState, HTMLMediaElement.NETWORK_EMPTY, "Shouldn't be loading");
+
+ a.addEventListener("abort", function(e) { a._abort++; });
+ a.addEventListener("emptied", function(e) { a._emptied++; });
+ a.addEventListener("loadedmetadata",
+ function(e) {
+ is(a._abort, 0, "Should not have received 'abort' before 'loadedmetadata");
+ is(a._emptied, 0, "Should not have received 'emptied' before 'loadedmetadata");
+
+ a.addEventListener("loadstart",
+ function() {
+ is(a._abort, 1, "Should have received 'abort' before 'loadstart");
+ is(a._emptied, 1, "Should have received 'emptied' before 'loadstart");
+ SimpleTest.finish();
+ });
+
+ a.src = "";
+ is(a._abort, 0, "Should not have received 'abort' during setting a.src=''");
+ is(a._emptied, 0, "Should not have received 'emptied' during setting a.src=''");
+ });
+
+ a.src = getPlayableAudio(gSmallTests).name;
+
+ </script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=">Mozilla Bug </a>
+<p id="display"></p>
+<div id="content" style="display: none">
+
+</div>
+<pre id="test">
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_reset_src.html b/dom/media/test/test_reset_src.html
new file mode 100644
index 0000000000..a30f9cc3ef
--- /dev/null
+++ b/dom/media/test/test_reset_src.html
@@ -0,0 +1,98 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=804875
+-->
+
+<head>
+ <title>Test for bug 804875</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script type="text/javascript" src="manifest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<a target="_blank"
+href="https://bugzilla.mozilla.org/show_bug.cgi?id=804875">Mozilla Bug 804875</a>
+
+<canvas></canvas>
+
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function finish(v) {
+ removeNodeAndSource(v);
+ manager.finished(v.token);
+}
+
+function onLoadedData_Audio(e) {
+ var t = e.target;
+ is(t.videoHeight, 0, t.name + ": videoHeight should be zero when there is no video.");
+ is(t.videoWidth, 0, t.name + ": videoWidth should be zero when there is no video.");
+ is(t.mozPaintedFrames, 0, t.name + ": mozPaintedFrames should be zero when there is no video.");
+ is(t.mozFrameDelay, 0, t.name + ": mozFrameDelay should be zero when there is no video.");
+ var c = document.getElementsByTagName("canvas")[0].getContext("2d");
+ var threw = false;
+ try {
+ c.drawImage(t, 0, 0, t.videoHeight, t.videoWidth);
+ } catch (ex) {
+ threw = true;
+ }
+ ok(!threw, t.name + ": Even though we don't succeed to draw a video frame on the canvas, we shouldn't throw.");
+ finish(t);
+}
+
+function onTimeUpdate_Video(e) {
+ var t = e.target;
+ if (t.currentTime < t.duration / 4) {
+ return;
+ }
+ t.removeEventListener("timeupdate", onTimeUpdate_Video);
+ // There's no guarantee that a video frame composite notification reaches
+ // us before timeupdate fires.
+ ok(t.mozPaintedFrames >= 0, t.name + ": mozPaintedFrames should be positive or zero, is " + t.mozPaintedFrames + ".");
+ ok(t.mozFrameDelay >= 0, t.name + ": mozFrameDelay should be positive or zero, is " + t.mozFrameDelay + ".");
+
+ if (t._firstTime) {
+ // eslint-disable-next-line no-self-assign
+ t.src = t.src;
+ t._firstTime = false;
+ } else {
+ var source = getPlayableAudio(gPlayTests);
+ if (!source) {
+ todo("No audio file available.")
+ finish(t);
+ } else {
+ t.removeEventListener("loadeddata", onLoadedData_Video);
+ t.addEventListener("loadeddata", onLoadedData_Audio);
+ t.src = source.name;
+ }
+ }
+}
+
+function onLoadedData_Video(e) {
+ var t = e.target;
+ isnot(t.videoHeight, 0, t.name + ": We should have a videoHeight.");
+ isnot(t.videoWidth, 0, t.name + ": We should have a videoWidth.");
+ t.addEventListener("timeupdate", onTimeUpdate_Video);
+ t.play();
+}
+
+function startTest(test, token) {
+ var v = document.createElement('video');
+ document.body.appendChild(v);
+ v._firstTime = true;
+ v.addEventListener("loadeddata", onLoadedData_Video);
+ v.src = test.name;
+ v.token = token;
+ v.name = test.name;
+ v.play();
+ manager.started(token);
+}
+
+manager.runTests(getPlayableVideos(gSmallTests.concat(gSeekTests)), startTest);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_resolution_change.html b/dom/media/test/test_resolution_change.html
new file mode 100644
index 0000000000..403f9b505d
--- /dev/null
+++ b/dom/media/test/test_resolution_change.html
@@ -0,0 +1,52 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test playback of files with resolution changes</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function loadedData(e) {
+ var v = e.target;
+ v.addEventListener("resize", resize);
+ v.play();
+}
+
+function resize(e) {
+ var v = e.target;
+ v.seenResolutionChange = true;
+}
+
+function ended(e) {
+ var v = e.target;
+ ok(v.seenResolutionChange, v.token + ": A resolution change should have ocurred by the end of playback");
+ removeNodeAndSource(v);
+ manager.finished(v.token);
+}
+
+function startTest(test, token) {
+ var v = document.createElement('video');
+ v.preload = "metadata";
+ v.token = token;
+ v.src = test.name;
+ v.seenResolutionChange = false;
+
+ v.addEventListener("loadeddata", loadedData)
+ v.addEventListener("ended", ended);
+
+ manager.started(token);
+ document.body.appendChild(v);
+}
+
+manager.runTests(gResolutionChangeTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_resume.html b/dom/media/test/test_resume.html
new file mode 100644
index 0000000000..0e22894be1
--- /dev/null
+++ b/dom/media/test/test_resume.html
@@ -0,0 +1,47 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: Test resume of server-dropped connections</title>
+ <script type="text/javascript" src="/MochiKit/MochiKit.js"></script>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<audio preload="auto" id="a"></audio>
+<iframe id="f"></iframe>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var key = Math.round(Math.random()*1000000000);
+var a = document.getElementById("a");
+var f = document.getElementById("f");
+
+function didEnd() {
+ ok(a.currentTime > 2.26, "Reached correct end time (got " + a.currentTime + ", expected > 2.26");
+ SimpleTest.finish();
+}
+
+function didSendCancel() {
+ a.addEventListener("ended", didEnd);
+ a.play();
+}
+
+function didSuspend() {
+ a.removeEventListener("suspend", didSuspend);
+
+ // Cache must have filled up, or something. Tell the Web server to drop
+ // our connection.
+ f.addEventListener("load", didSendCancel);
+ f.src = "cancellable_request.sjs?cancelkey=" + key;
+}
+
+if (!a.canPlayType("audio/wave")) {
+ todo(false, "Test requires support for audio/wave");
+} else {
+ a.addEventListener("suspend", didSuspend);
+ a.src = "cancellable_request.sjs?key=" + key;
+ SimpleTest.waitForExplicitFinish();
+}
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_seamless_looping.html b/dom/media/test/test_seamless_looping.html
new file mode 100644
index 0000000000..94d020c834
--- /dev/null
+++ b/dom/media/test/test_seamless_looping.html
@@ -0,0 +1,199 @@
+<!DOCTYPE html>
+<html>
+<head>
+ <title>Test for seamless loop of HTMLAudioElements</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<canvas id="canvas" width="300" height="300"></canvas>
+<script type="application/javascript">
+/**
+ * This test is used to ensure every time we loop audio, the audio can loop
+ * seamlessly which means there won't have any silenece or noise between the
+ * end and the start.
+ */
+
+SimpleTest.waitForExplicitFinish();
+
+// Set DEBUG to true to add a canvas with a little drawing of what is going
+// on, and actually outputs the audio to the speakers.
+var DEBUG = true;
+var LOOPING_COUNT = 0;
+var MAX_LOOPING_COUNT = 10;
+// Test files are at 44100Hz, files are one second long, and contain therefore
+// 100 periods
+var TONE_FREQUENCY = 441;
+
+(async function testSeamlesslooping() {
+ let wavFileURL = {
+ name: URL.createObjectURL(new Blob([createSrcBuffer()], { type: 'audio/wav'
+ })),
+ type: "audio/wav"
+ };
+
+ let testURLs = gSeamlessLoopingTests.splice(0)
+ testURLs.push(wavFileURL);
+ for (let testFile of testURLs) {
+ LOOPING_COUNT = 0;
+ info(`- create looping audio element ${testFile.name}`);
+ let audio = createAudioElement(testFile.name);
+
+ info(`- start audio and analyze audio wave data to ensure looping audio without any silence or noise -`);
+ await playAudioAndStartAnalyzingWaveData(audio);
+
+ info(`- test seamless looping multiples times -`);
+ for (LOOPING_COUNT = 0; LOOPING_COUNT < MAX_LOOPING_COUNT; LOOPING_COUNT++) {
+ await once(audio, "seeked");
+ info(`- the round ${LOOPING_COUNT} of the seamless looping succeeds -`);
+ }
+ window.audio.remove();
+ window.ac.close();
+ }
+
+ info(`- end of seamless looping test -`);
+ SimpleTest.finish();
+})();
+
+/**
+ * Test utility functions
+ */
+function createSrcBuffer() {
+ // Generate the sine in floats, then convert, for simplicity.
+ let channels = 1;
+ let sampleRate = 44100;
+ let buffer = new Float32Array(sampleRate * channels);
+ let phase = 0;
+ const TAU = 2 * Math.PI;
+ for (let i = 0; i < buffer.length; i++) {
+ // Adjust the gain a little so we're sure it's not going to clip. This is
+ // important because we're converting to 16bit integer right after, and
+ // clipping will clearly introduce a discontinuity that will be
+ // mischaracterized as a looping click.
+ buffer[i] = Math.sin(phase) * 0.99;
+ phase += TAU * TONE_FREQUENCY / 44100;
+ if (phase > 2 * TAU) {
+ phase -= TAU;
+ }
+ }
+
+ // Make a RIFF header, it's 23 bytes
+ let buf = new Int16Array(buffer.length + 23);
+ buf[0] = 0x4952;
+ buf[1] = 0x4646;
+ buf[2] = (2 * buffer.length + 15) & 0x0000ffff;
+ buf[3] = ((2 * buffer.length + 15) & 0xffff0000) >> 16;
+ buf[4] = 0x4157;
+ buf[5] = 0x4556;
+ buf[6] = 0x6d66;
+ buf[7] = 0x2074;
+ buf[8] = 0x0012;
+ buf[9] = 0x0000;
+ buf[10] = 0x0001;
+ buf[11] = 1;
+ buf[12] = 44100 & 0x0000ffff;
+ buf[13] = (44100 & 0xffff0000) >> 16;
+ buf[14] = (2 * channels * sampleRate) & 0x0000ffff;
+ buf[15] = ((2 * channels * sampleRate) & 0xffff0000) >> 16;
+ buf[16] = 0x0004;
+ buf[17] = 0x0010;
+ buf[18] = 0x0000;
+ buf[19] = 0x6164;
+ buf[20] = 0x6174;
+ buf[21] = (2 * buffer.length) & 0x0000ffff;
+ buf[22] = ((2 * buffer.length) & 0xffff0000) >> 16;
+
+ // convert to int16 and copy.
+ for (let i = 0; i < buffer.length; i++) {
+ buf[i + 23] = Math.round(buffer[i] * (1 << 15));
+ }
+ return buf;
+}
+
+function createAudioElement(url) {
+ /* global audio */
+ window.audio = document.createElement("audio");
+ audio.src = url;
+ audio.controls = true;
+ audio.loop = true;
+ document.body.appendChild(audio);
+ return audio;
+}
+
+async function playAudioAndStartAnalyzingWaveData(audio) {
+ createAudioWaveAnalyser(audio);
+ ok(await once(audio, "canplay").then(() => true, () => false),
+ `audio can start playing.`)
+ ok(await audio.play().then(() => true, () => false),
+ `audio started playing successfully.`);
+}
+
+function createAudioWaveAnalyser(source) {
+ /* global ac, analyser */
+ window.ac = new AudioContext();
+ window.analyser = ac.createAnalyser();
+ analyser.frequencyBuf = new Float32Array(analyser.frequencyBinCount);
+ analyser.smoothingTimeConstant = 0;
+ analyser.fftSize = 2048; // 1024 bins
+
+ let sourceNode = ac.createMediaElementSource(source);
+ sourceNode.connect(analyser);
+
+ if (DEBUG) {
+ analyser.connect(ac.destination);
+ analyser.timeDomainBuf = new Float32Array(analyser.frequencyBinCount);
+ let cvs = document.querySelector("canvas");
+ analyser.c = cvs.getContext("2d");
+ analyser.w = cvs.width;
+ analyser.h = cvs.height;
+ }
+
+ analyser.notifyAnalysis = () => {
+ if (LOOPING_COUNT >= MAX_LOOPING_COUNT) {
+ return;
+ }
+ let {frequencyBuf} = analyser;
+ analyser.getFloatFrequencyData(frequencyBuf);
+ // Let things stabilize at the beginning. See bug 1441509.
+ if (LOOPING_COUNT > 1) {
+ analyser.doAnalysis(frequencyBuf, ac.sampleRate);
+ }
+
+ if (DEBUG) {
+ let {c, w, h, timeDomainBuf} = analyser;
+ c.clearRect(0, 0, w, h);
+ analyser.getFloatTimeDomainData(timeDomainBuf);
+ for (let i = 0; i < frequencyBuf.length; i++) {
+ c.fillRect(i, h, 1, -frequencyBuf[i] + analyser.minDecibels);
+ }
+
+ for (let i = 0; i < timeDomainBuf.length; i++) {
+ c.fillRect(i, h / 2, 1, -timeDomainBuf[i] * h / 2);
+ }
+ }
+
+ requestAnimationFrame(analyser.notifyAnalysis);
+ }
+
+ analyser.doAnalysis = (buf, ctxSampleRate) => {
+ // The size of an FFT is twice the number of bins in its output.
+ let fftSize = 2 * buf.length;
+ // first find a peak where we expect one.
+ let binIndexTone = 1 + Math.round(TONE_FREQUENCY * fftSize / ctxSampleRate);
+ ok(buf[binIndexTone] > -35,
+ `Could not find a peak: ${buf[binIndexTone]} db at ${TONE_FREQUENCY}Hz
+ (${source.src})`);
+
+ // check that the energy some octaves higher is very low.
+ let binIndexOutsidePeak = 1 + Math.round(TONE_FREQUENCY * 4 * buf.length / ctxSampleRate);
+ ok(buf[binIndexOutsidePeak] < -84,
+ `Found unexpected high frequency content: ${buf[binIndexOutsidePeak]}db
+ at ${TONE_FREQUENCY * 4}Hz (${source.src})`);
+ }
+
+ analyser.notifyAnalysis();
+}
+</script>
+</body>
+</html>
diff --git a/dom/media/test/test_seamless_looping_cancel_looping_future_frames.html b/dom/media/test/test_seamless_looping_cancel_looping_future_frames.html
new file mode 100644
index 0000000000..28a56af7ff
--- /dev/null
+++ b/dom/media/test/test_seamless_looping_cancel_looping_future_frames.html
@@ -0,0 +1,61 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+<title>Seamless looping test cancel looping - no future frame</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+<script type="text/javascript" src="manifest.js"></script>
+</head>
+<script type="application/javascript">
+
+/**
+ * This test aims to check whether the future frames, which are the frames
+ * beyond EOS and will be played in the next round of looping, would be discared
+ * correctly if we cancel looping in the first looping round.
+ */
+add_task(async function testSeamlessLoopingVideoCancelLoopingDiscardFutureFrames() {
+ info(`create video and play it`);
+ let video = document.createElement('video');
+ video.loop = true;
+ video.src = "gizmo.mp4";
+ document.body.appendChild(video);
+ // speed up the test.
+ video.playbackRate = 2;
+ await video.play();
+
+ info(`wait until the video reaches to near end`);
+ let checkPoint1, checkPoint2;
+ await new Promise(r => {
+ video.ontimeupdate = _ => {
+ // When media almost reaches to the end, the audio track should already
+ // reach to the end before and started storing some future frames for next
+ // round of looping.
+ if (video.currentTime > video.duration - 1.5) {
+ info(`cancel looping`);
+ video.loop = false;
+ video.ontimeupdate = null;
+ checkPoint1 = new Date();
+ r();
+ }
+ }
+ });
+
+ // Because we've canceled looping, playback should reach to the end soon
+ // (less than 0.5 seconds, but we use 1 second as a threshold to avoid
+ // intermittent failure) But if we didn't discard future frames correctly,
+ // that would cause that the video plays more audio frames and requires more
+ // time to reach the end. Note, in the error case I encountered, the playback
+ // didn't discard future frames and kept decoding since then, which causes
+ // playing extra a whole audio track again.
+ info(`wait until video reaches to the end within correct remaining time`);
+ await once(video, "ended");
+ checkPoint2 = new Date();
+ const diffInSeconds = Math.abs(checkPoint1 - checkPoint2) / 1000;
+ ok(diffInSeconds < 2,
+ `finished playing within ${diffInSeconds} without incorrectly playing any future frame`);
+});
+
+</script>
+<body>
+</body>
+</html>
diff --git a/dom/media/test/test_seamless_looping_duration.html b/dom/media/test/test_seamless_looping_duration.html
new file mode 100644
index 0000000000..4a64a81759
--- /dev/null
+++ b/dom/media/test/test_seamless_looping_duration.html
@@ -0,0 +1,63 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+<title>Seamless looping test duration</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+<script type="text/javascript" src="manifest.js"></script>
+</head>
+<script type="application/javascript">
+
+/**
+ * This test aims to check that the media duration shouldn't be changed during
+ * seamless looping.
+ */
+add_task(async function testSeamlessLoopingDuration() {
+ info(`create video and play it`);
+ let video = document.createElement('video');
+ video.loop = true;
+ video.src = "gizmo-short.mp4";
+ document.body.appendChild(video);
+ await video.play();
+
+ video.ondurationchange =
+ _ => ok(false, "shouldn't change duration during looping!");
+
+ info(`test seamless looping multiples times`);
+ let MAX_LOOPING_COUNT = 10;
+ for (let count = 0; count < MAX_LOOPING_COUNT; count++) {
+ await once(video, "seeking");
+ await once(video, "seeked");
+ ok(true, `the round ${count} of the seamless looping succeeds`);
+ }
+});
+
+// This one tests the situation where both tracks reached EOS before entering
+// looping state.
+add_task(async function testSeamlessLoopingDuration2() {
+ info(`create video and play it to the end`);
+ let video = document.createElement('video');
+ video.src = "gizmo-short.mp4";
+ document.body.appendChild(video);
+ await video.play();
+ await once(video, "ended");
+
+ info(`play video again`);
+ video.ondurationchange =
+ _ => ok(false, "shouldn't change duration during looping!");
+ video.loop = true;
+ await video.play();
+
+ info(`test seamless looping multiples times`);
+ let MAX_LOOPING_COUNT = 10;
+ for (let count = 0; count < MAX_LOOPING_COUNT; count++) {
+ await once(video, "seeking");
+ await once(video, "seeked");
+ ok(true, `the round ${count} of the seamless looping succeeds`);
+ }
+});
+
+</script>
+<body>
+</body>
+</html>
diff --git a/dom/media/test/test_seamless_looping_media_element_state.html b/dom/media/test/test_seamless_looping_media_element_state.html
new file mode 100644
index 0000000000..c94de04912
--- /dev/null
+++ b/dom/media/test/test_seamless_looping_media_element_state.html
@@ -0,0 +1,51 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+<title>Seamless looping media element state</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+<script type="text/javascript" src="manifest.js"></script>
+</head>
+<script type="application/javascript">
+
+/**
+ * This test aims to check if the media element would change its ready state
+ * under `HAVE_CURRENT_DATA` when the seamless looping is performed. Because
+ * during seamless looping, we should always have current frame.
+ */
+add_task(async function testSeamlessLoopingMediaElementState() {
+ await SpecialPowers.pushPrefEnv({
+ set: [
+ ["media.seamless-looping-video", true],
+ ],
+ });
+
+ info(`create video`);
+ let video = document.createElement('video');
+ video.loop = true;
+ video.src = "gizmo-short.mp4";
+ document.body.appendChild(video);
+ await video.play();
+
+ info(`test seamless looping multiples times`);
+ let MAX_LOOPING_COUNT = 10;
+ for (let count = 0; count < MAX_LOOPING_COUNT; count++) {
+ await once(video, "seeking");
+ // If the video looping is not seamless, when playback reaches to the end,
+ // MDSM would trigger a seek in order to get the new frame from the start
+ // position. That would notify the media element that the status of the next
+ // frame is not available now due to seeking (NEXT_FRAME_UNAVAILABLE_SEEKING)
+ // and causes the media element dispatching `waiting` event.
+ video.onwaiting = () => {
+ ok(false, "should not get waiting event");
+ }
+ await once(video, "seeked");
+ video.onwaiting = null;
+ ok(true, `the round ${count} of the seamless looping succeeds`);
+ }
+});
+
+</script>
+<body>
+</body>
+</html>
diff --git a/dom/media/test/test_seamless_looping_not_keep_painting_old_video_frames.html b/dom/media/test/test_seamless_looping_not_keep_painting_old_video_frames.html
new file mode 100644
index 0000000000..6f0bc45f55
--- /dev/null
+++ b/dom/media/test/test_seamless_looping_not_keep_painting_old_video_frames.html
@@ -0,0 +1,57 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+<title>Seamless looping test : do not keep painting old video frames</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+<script type="text/javascript" src="manifest.js"></script>
+</head>
+<script type="application/javascript">
+
+/**
+ * This test is used to ensure that after seeking to EOS directly, video can
+ * display its frame sucessfully, not showing the incorrect frame which should
+ * be discarded during seeking. The test video contains three seconds white
+ * frames [0s-3s] and one second black frames [3s-4s], so after seeking to
+ * EOS and start looping video again, video should display white frames, instead
+ * of black frames.
+ */
+add_task(async function testSeamlessLoopingDoNotKeepPaintingOldVideoFrame() {
+ info(`create video and play it`);
+ const WIDTH = 10, HEIGHT = 10;
+ let video = document.createElement('video');
+ video.loop = true;
+ video.src = "white-3s-black-1s.webm";
+ video.width = WIDTH;
+ video.height = HEIGHT;
+ document.body.appendChild(video);
+ await video.play();
+
+ info(`seek to EOS (${video.duration}s) and waiting for looping`);
+ video.currentTime = video.duration;
+ await once(video, "seeked");
+
+ // We can't control the exact timing when the compositor would render next
+ // frame so we wait for the next timeupdate to ensure the media has been
+ // moved forward to the place where video should show white frames.
+ info(`wait for timeupdate then check the painted frame`);
+ await once(video, "timeupdate");
+ const canvas = document.createElement('canvas');
+ const context = canvas.getContext('2d');
+ canvas.width = WIDTH;
+ canvas.height = HEIGHT;
+ context.drawImage(video, 0, 0, canvas.width, canvas.height);
+ const pixelData = context.getImageData(0, 0, WIDTH, HEIGHT).data;
+ for (let idx = 0; idx < WIDTH * HEIGHT; idx++) {
+ let pixelCount = 4 * idx; // RGBA
+ // White frame's RGB value should be [255,255,255]
+ is(pixelData[pixelCount + 0], 255, `R should be 255`);
+ is(pixelData[pixelCount + 1], 255, `G should be 255`);
+ is(pixelData[pixelCount + 2], 255, `B should be 255`);
+ }
+});
+
+</script>
+<body>
+</body>
+</html>
diff --git a/dom/media/test/test_seamless_looping_resume_video_decoding.html b/dom/media/test/test_seamless_looping_resume_video_decoding.html
new file mode 100644
index 0000000000..14f7cdd331
--- /dev/null
+++ b/dom/media/test/test_seamless_looping_resume_video_decoding.html
@@ -0,0 +1,64 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+<title>Seamless looping test with resuming video decoding</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+<script type="text/javascript" src="manifest.js"></script>
+<script src="background_video.js"></script>
+</head>
+<script type="application/javascript">
+
+/**
+ * This test aims to check if seamless looping can work well with the mechanism
+ * of resuming video decoding. When resuming decoding, MDSM will leave the
+ * looping state and go to video only seek state. We want to ensure that the
+ * timestamp of new video data that state requests will also be adjusted in
+ * order to keep video playback ongoing without video being frozen because
+ * frames get discarded.
+ */
+add_task(async function testSeamlessLoopingResumeVideoDecoding() {
+ await SpecialPowers.pushPrefEnv({
+ set: [
+ ["media.test.video-suspend", true],
+ ["media.suspend-background-video.enabled", true],
+ ["media.suspend-background-video.delay-ms", 0],
+ ],
+ });
+
+ info(`create video and play it`);
+ let video = document.createElement('video');
+ video.loop = true;
+ video.src = "gizmo.mp4";
+ document.body.appendChild(video);
+ // speed up the test.
+ video.playbackRate = 2;
+ await video.play();
+
+ info(`test seamless looping once`);
+ await once(video, "seeked");
+ ok(true, `loop back happened`);
+
+ info(`suspend video decoding`);
+ video.setVisible(false);
+ await nextVideoSuspends(video);
+ info(`suspended video decoding`);
+
+ info(`resume video decoding (enter video-only seek state)`);
+ video.setVisible(true);
+ await testVideoOnlySeekCompletedWhenShown(video);
+ info(`resumed video decoding and finished video-only seeking`);
+
+ const lastPaintedFramesAmount = video.mozPaintedFrames;
+ info(`end test after looping one more time`);
+ await once(video, "seeked");
+
+ const currentPaintedFrameAmount = video.mozPaintedFrames;
+ ok(lastPaintedFramesAmount < currentPaintedFrameAmount,
+ `painted frames keeps growing from ${lastPaintedFramesAmount} to ${currentPaintedFrameAmount}`);
+});
+
+</script>
+<body>
+</body>
+</html>
diff --git a/dom/media/test/test_seamless_looping_seek_current_time.html b/dom/media/test/test_seamless_looping_seek_current_time.html
new file mode 100644
index 0000000000..888e5437c4
--- /dev/null
+++ b/dom/media/test/test_seamless_looping_seek_current_time.html
@@ -0,0 +1,62 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+<title>Seamless looping current time after seek</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+<script type="text/javascript" src="manifest.js"></script>
+</head>
+<script type="application/javascript">
+
+/**
+ * This test is used to ensure that the playback position won't be reset to ZERO
+ * incorrectly when performing seek on a looping media.
+ */
+add_task(async function testSeekVideoOnlyPlayback() {
+ info(`create and play a media which contains only video track`);
+ let video = createLoopingMedia("video", "gizmo-noaudio.webm");
+ await video.play();
+ await assertSeekingForwardShouldIncreaseCurrentTime(video);
+ removeNodeAndSource(video);
+});
+
+add_task(async function testSeekAudioOnlyPlayback() {
+ info(`create and play a media which contains only audio track`);
+ let audio = createLoopingMedia("audio", "flac-s24.flac");
+ await audio.play();
+ await assertSeekingForwardShouldIncreaseCurrentTime(audio);
+ removeNodeAndSource(audio);
+});
+
+add_task(async function testSeekBothTracksPlayback() {
+ info(`create and play a media which contains both tracks`);
+ let video = createLoopingMedia("video", "gizmo.mp4");
+ await video.play();
+ await assertSeekingForwardShouldIncreaseCurrentTime(video);
+ removeNodeAndSource(video);
+});
+
+// Following are helper functions
+function createLoopingMedia(type, src) {
+ let media = document.createElement(type);
+ media.loop = true;
+ media.src = src;
+ document.body.appendChild(media);
+ return media;
+}
+
+async function assertSeekingForwardShouldIncreaseCurrentTime(media) {
+ const currentTimeBeforeSeek = media.currentTime;
+ const target = media.duration / 2;
+ media.currentTime = target;
+ await once(media, "seeked");
+ const currentTimeAfterSeek = media.currentTime;
+ ok(currentTimeAfterSeek > currentTimeBeforeSeek,
+ `media current time should keep going forward (target=${target}, ` +
+ `currentTime=${currentTimeBeforeSeek} -> ${currentTimeAfterSeek})`);
+}
+
+</script>
+<body>
+</body>
+</html>
diff --git a/dom/media/test/test_seamless_looping_shorter_audio_than_video.html b/dom/media/test/test_seamless_looping_shorter_audio_than_video.html
new file mode 100644
index 0000000000..6e1105563e
--- /dev/null
+++ b/dom/media/test/test_seamless_looping_shorter_audio_than_video.html
@@ -0,0 +1,37 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+<title>Seamless looping : shorter audio track than video track</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+<script type="text/javascript" src="manifest.js"></script>
+</head>
+<script type="application/javascript">
+
+/**
+ * This test is used to check whether we can play and loop a file in which the
+ * audio track is shorter than its video track. The file we used in the test has
+ * audio track (duration 3,001,000us), and video track (duration 3,047,000us)
+ */
+add_task(async function testSeamlessLoopingShorterAudioTrack() {
+ info(`create video and play it`);
+ let video = document.createElement('video');
+ video.loop = true;
+ video.controls = true;
+ video.src = "shorter_audio_than_video_3s.webm";
+ document.body.appendChild(video);
+ await video.play();
+
+ info(`test seamless looping multiples times`);
+ let MAX_LOOPING_COUNT = 1;
+ for (let count = 0; count < MAX_LOOPING_COUNT; count++) {
+ await once(video, "seeking");
+ await once(video, "seeked");
+ ok(true, `the round ${count} of the seamless looping succeeds`);
+ }
+});
+
+</script>
+<body>
+</body>
+</html>
diff --git a/dom/media/test/test_seamless_looping_video.html b/dom/media/test/test_seamless_looping_video.html
new file mode 100644
index 0000000000..3bf99b0a62
--- /dev/null
+++ b/dom/media/test/test_seamless_looping_video.html
@@ -0,0 +1,73 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+<title>Seamless looping video canvas test</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+<script type="text/javascript" src="manifest.js"></script>
+</head>
+<canvas id="canvas"></canvas>
+<video id="v"></video>
+<script type="application/javascript">
+
+/**
+ * This test aims to check if the video is seamless looping by capturing the
+ * image when loop happens. We use a video contains only white frames, so the
+ * captured frame should always be a white frame. If looping is not seamless,
+ * the captured frame would become a black frame.
+ */
+const WIDTH = 10, HEIGHT = 10;
+
+add_task(async function testSeamlessLoopingVideoCanvas() {
+ await SpecialPowers.pushPrefEnv({
+ set: [
+ ["media.seamless-looping-video", true],
+ ],
+ });
+
+ info(`load video which only contains white frames`);
+ let video = document.getElementById("v");
+ video.loop = true;
+ video.src = "white-short.webm";
+ video.width = WIDTH;
+ video.height = HEIGHT;
+ await video.play();
+
+ info(`setup canvas`);
+ const cvs = document.getElementById("canvas");
+ cvs.width = WIDTH;
+ cvs.height = HEIGHT;
+
+ info(`test seamless looping multiples times`);
+ let MAX_LOOPING_COUNT = 10;
+ for (let count = 0; count < MAX_LOOPING_COUNT; count++) {
+ await once(video, "seeking");
+ assertPaintedFrameIsWhiteFrame();
+ await once(video, "seeked");
+ ok(true, `the round ${count} of the seamless looping succeeds`);
+ }
+});
+
+function assertPaintedFrameIsWhiteFrame() {
+ info(`catpure image from video`);
+ const cvs = document.getElementById("canvas");
+ let context = cvs.getContext('2d');
+ if (!context) {
+ ok(false, "can't get 2d context");
+ }
+
+ context.drawImage(document.getElementById("v"), 0, 0, WIDTH, HEIGHT);
+ let imageData = context.getImageData(0, 0, WIDTH, HEIGHT);
+ for (let idx = 0; idx < WIDTH * HEIGHT; idx++) {
+ let pixelCount = 4 * idx; // RGBA
+ let data = imageData.data;
+ // White frame's RGB value should be [255,255,255]
+ is(data[pixelCount + 0], 255, `R should be 255`);
+ is(data[pixelCount + 1], 255, `G should be 255`);
+ is(data[pixelCount + 2], 255, `B should be 255`);
+ }
+}
+</script>
+<body>
+</body>
+</html>
diff --git a/dom/media/test/test_seek-1.html b/dom/media/test/test_seek-1.html
new file mode 100644
index 0000000000..8690a475b6
--- /dev/null
+++ b/dom/media/test/test_seek-1.html
@@ -0,0 +1,84 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: seek tests</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="seek_support.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+// The data being used in these tests is specified in manifest.js.
+// The functions to build the test array and to run a specific test are in
+// seek_support.js.
+
+const SEEK_TEST_NUMBER = 1;
+
+function test_seek1(v, seekTime, is, ok, finish) {
+
+var startPassed = false;
+var endPassed = false;
+var seekFlagStart = false;
+var seekFlagEnd = false;
+var readonly = true;
+var completed = false;
+
+function startTest() {
+ ok(!completed, "Should not be completed yet");
+ ok(!v.seeking, "seeking should default to false");
+ try {
+ v.seeking = true;
+ readonly = v.seeking === false;
+ }
+ catch(e) {
+ readonly = "threw exception: " + e;
+ }
+ is(readonly, true, "seeking should be readonly");
+
+ v.currentTime = seekTime;
+ seekFlagStart = v.seeking;
+}
+
+function seekStarted() {
+ ok(!completed, "should not be completed yet");
+ ok(Math.abs(v.currentTime - seekTime) < 0.1,
+ "Video currentTime should be around " + seekTime + ": " + v.currentTime + " (seeking)");
+ startPassed = true;
+}
+
+function seekEnded() {
+ ok(!completed, "shuld not be completed yet");
+ ok(Math.abs(v.currentTime - seekTime) < 0.1,
+ "Video currentTime should be around " + seekTime + ": " + v.currentTime + " (seeked)");
+ endPassed = true;
+ seekFlagEnd = v.seeking;
+ v.play();
+}
+
+function playbackEnded() {
+ ok(!completed, "should not be completed yet");
+
+ completed = true;
+ ok(startPassed, "seeking event");
+ ok(endPassed, "seeked event");
+ ok(seekFlagStart, "seeking flag on start should be true");
+ ok(!seekFlagEnd, "seeking flag on end should be false");
+ finish();
+}
+
+once(v, "ended", playbackEnded);
+once(v, "loadedmetadata", startTest);
+once(v, "seeking", seekStarted);
+once(v, "seeked", seekEnded);
+
+}
+
+manager.runTests(createTestArray(), startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_seek-10.html b/dom/media/test/test_seek-10.html
new file mode 100644
index 0000000000..6c02384ed4
--- /dev/null
+++ b/dom/media/test/test_seek-10.html
@@ -0,0 +1,56 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: seek tests</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="seek_support.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+// The data being used in these tests is specified in manifest.js.
+// The functions to build the test array and to run a specific test are in
+// seek_support.js.
+
+const SEEK_TEST_NUMBER = 10;
+
+function test_seek10(v, seekTime, is, ok, finish) {
+
+// Test bug 523335 - ensure that if we close a stream while seeking, we
+// don't hang during shutdown. This test won't "fail" per se if it's regressed,
+// it will instead start to cause random hangs in the mochitest harness on
+// shutdown.
+
+function startTest() {
+ // Must be duration*0.9 rather than seekTime, else we don't hit that problem.
+ // This is probably due to the seek bisection finishing too quickly, before
+ // we can close the stream.
+ v.currentTime = v.duration * 0.9;
+}
+
+function done(evt) {
+ ok(true, "We don't acutally test anything...");
+ finish();
+}
+
+function seeking() {
+ ok(v.currentTime >= seekTime - 0.1, "Video currentTime should be around " + seekTime + ": " + v.currentTime);
+ v.onerror = done;
+ v.src = "not a valid video file.";
+ v.load(); // Cause the existing stream to close.
+}
+
+v.addEventListener("loadeddata", startTest);
+v.addEventListener("seeking", seeking);
+
+}
+
+manager.runTests(createTestArray(), startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_seek-11.html b/dom/media/test/test_seek-11.html
new file mode 100644
index 0000000000..2207f684d7
--- /dev/null
+++ b/dom/media/test/test_seek-11.html
@@ -0,0 +1,76 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: seek tests</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="seek_support.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+// The data being used in these tests is specified in manifest.js.
+// The functions to build the test array and to run a specific test are in
+// seek_support.js.
+
+PARALLEL_TESTS = 1;
+const SEEK_TEST_NUMBER = 11;
+
+function test_seek11(v, seekTime, is, ok, finish) {
+
+// Test for bug 476973, multiple seeks to the same position shouldn't cause problems.
+
+var seekedNonZero = false;
+var completed = false;
+var target = 0;
+
+function startTest() {
+ if (completed)
+ return;
+ target = v.duration / 2;
+ v.currentTime = target;
+ v.currentTime = target;
+ v._seekTarget = target;
+}
+
+function startSeeking() {
+ ok(v.currentTime >= v._seekTarget - 0.1,
+ "Video currentTime should be around " + v._seekTarget + ": " + v.currentTime);
+ if (!seekedNonZero) {
+ v.currentTime = target;
+ v._seekTarget = target;
+ seekedNonZero = true;
+ }
+}
+
+function seekEnded() {
+ if (completed)
+ return;
+
+ if (v.currentTime > 0) {
+ ok(v.currentTime > target - 0.1 && v.currentTime < target + 0.1,
+ "Seek to wrong destination " + v.currentTime);
+ v.currentTime = 0.0;
+ v._seekTarget = 0.0;
+ } else {
+ ok(seekedNonZero, "Successfully seeked to nonzero");
+ ok(true, "Seek back to zero was successful");
+ completed = true;
+ finish();
+ }
+}
+
+v.addEventListener("loadedmetadata", startTest);
+v.addEventListener("seeking", startSeeking);
+v.addEventListener("seeked", seekEnded);
+
+}
+
+manager.runTests(createTestArray(), startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_seek-12.html b/dom/media/test/test_seek-12.html
new file mode 100644
index 0000000000..28decabadc
--- /dev/null
+++ b/dom/media/test/test_seek-12.html
@@ -0,0 +1,59 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: seek tests</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="seek_support.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+// The data being used in these tests is specified in manifest.js.
+// The functions to build the test array and to run a specific test are in
+// seek_support.js.
+
+const SEEK_TEST_NUMBER = 12;
+
+function test_seek12(v, seekTime, is, ok, finish) {
+var completed = false;
+
+function startTest() {
+ if (completed)
+ return;
+ ok(!v.seeking, "seeking should default to false");
+ v.currentTime = seekTime;
+ is(v.currentTime, seekTime, "currentTime must report seek target immediately");
+ is(v.seeking, true, "seeking flag on start should be true");
+}
+
+function seekStarted() {
+ if (completed)
+ return;
+ //is(v.currentTime, seekTime, "seeking: currentTime must be seekTime");
+ ok(Math.abs(v.currentTime - seekTime) < 0.01, "seeking: currentTime must be seekTime");
+}
+
+function seekEnded() {
+ if (completed)
+ return;
+ completed = true;
+ //is(v.currentTime, seekTime, "seeked: currentTime must be seekTime");
+ ok(Math.abs(v.currentTime - seekTime) < 0.01, "seeked: currentTime must be seekTime");
+ is(v.seeking, false, "seeking flag on end should be false");
+ finish();
+}
+
+v.addEventListener("loadedmetadata", startTest);
+v.addEventListener("seeking", seekStarted);
+v.addEventListener("seeked", seekEnded);
+}
+
+manager.runTests(createTestArray(), startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_seek-13.html b/dom/media/test/test_seek-13.html
new file mode 100644
index 0000000000..81eda9b658
--- /dev/null
+++ b/dom/media/test/test_seek-13.html
@@ -0,0 +1,72 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: seek tests</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="seek_support.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+// The data being used in these tests is specified in manifest.js.
+// The functions to build the test array and to run a specific test are in
+// seek_support.js.
+
+const SEEK_TEST_NUMBER = 13;
+
+function test_seek13(v, seekTime, is, ok, finish) {
+var completed = false;
+
+function startTest() {
+ if (completed)
+ return;
+ ok(!v.seeking, "seeking should default to false");
+ v.currentTime = v.duration;
+ is(v.currentTime, v.duration, "currentTime must report seek target immediately");
+ is(v.seeking, true, "seeking flag on start should be true");
+}
+
+function seekStarted() {
+ if (completed)
+ return;
+ //is(v.currentTime, v.duration, "seeking: currentTime must be duration");
+ ok(Math.abs(v.currentTime - v.duration) < 0.01,
+ "seeking: currentTime (" + v.currentTime + ") must be duration (" + v.duration + ")");
+}
+
+function seekEnded() {
+ if (completed)
+ return;
+ //is(v.currentTime, v.duration, "seeked: currentTime must be duration");
+ ok(Math.abs(v.currentTime - v.duration) < 0.01,
+ "seeked: currentTime (" + v.currentTime + ") must be duration (" + v.duration + ")");
+ is(v.seeking, false, "seeking flag on end should be false");
+}
+
+function playbackEnded() {
+ if (completed)
+ return;
+ completed = true;
+ //is(v.currentTime, v.duration, "ended: currentTime must be duration");
+ ok(Math.abs(v.currentTime - v.duration) < 0.01,
+ "ended: currentTime (" + v.currentTime + ") must be duration (" + v.duration + ")");
+ is(v.seeking, false, "seeking flag on end should be false");
+ is(v.ended, true, "ended must be true");
+ finish();
+}
+
+v.addEventListener("loadedmetadata", startTest);
+v.addEventListener("seeking", seekStarted);
+v.addEventListener("seeked", seekEnded);
+v.addEventListener("ended", playbackEnded);
+}
+
+manager.runTests(createTestArray(), startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_seek-14.html b/dom/media/test/test_seek-14.html
new file mode 100644
index 0000000000..7e19fe3bd3
--- /dev/null
+++ b/dom/media/test/test_seek-14.html
@@ -0,0 +1,43 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: seek tests</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="seek_support.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+const SEEK_TEST_NUMBER = 14;
+
+function test_seek14(v, seekTime, is, ok, finish) {
+ var completed = false;
+
+ function startTest() {
+ v.play();
+ v.currentTime = v.duration;
+ }
+
+ function playbackEnded() {
+ if (completed) {
+ ok(false, "'ended' should only fire once.");
+ return;
+ }
+ completed = true;
+ // Finish the test after 700ms. We should receive only one 'ended' event.
+ setTimeout(finish, 700);
+ }
+
+ v.addEventListener("loadedmetadata", startTest);
+ v.addEventListener("ended", playbackEnded);
+}
+
+manager.runTests(createTestArray(), startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_seek-2.html b/dom/media/test/test_seek-2.html
new file mode 100644
index 0000000000..7b666df961
--- /dev/null
+++ b/dom/media/test/test_seek-2.html
@@ -0,0 +1,76 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: seek tests</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="seek_support.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+// The data being used in these tests is specified in manifest.js.
+// The functions to build the test array and to run a specific test are in
+// seek_support.js.
+
+PARALLEL_TESTS = 1;
+const SEEK_TEST_NUMBER = 2;
+
+function test_seek2(v, seekTime, is, ok, finish) {
+
+// Test seeking works if current time is set before video is
+// playing.
+var startPassed = false;
+var endPassed = false;
+var completed = false;
+
+function startTest() {
+ if (completed)
+ return;
+
+ v.currentTime=seekTime;
+ v.play();
+}
+
+function seekStarted() {
+ if (completed)
+ return;
+
+ ok(v.currentTime >= seekTime - 0.1, "Video currentTime should be around " + seekTime + ": " + v.currentTime);
+ startPassed = true;
+}
+
+function seekEnded() {
+ if (completed)
+ return;
+
+ endPassed = true;
+}
+
+function playbackEnded() {
+ if (completed)
+ return;
+
+ completed = true;
+ ok(startPassed, "send seeking event");
+ ok(endPassed, "send seeked event");
+ ok(v.ended, "Checking playback has ended");
+ ok(Math.abs(v.currentTime - v.duration) <= 0.1, "Checking currentTime at end: " + v.currentTime);
+ finish();
+}
+
+v.addEventListener("ended", playbackEnded);
+v.addEventListener("loadedmetadata", startTest);
+v.addEventListener("seeking", seekStarted);
+v.addEventListener("seeked", seekEnded);
+
+}
+
+manager.runTests(createTestArray(), startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_seek-3.html b/dom/media/test/test_seek-3.html
new file mode 100644
index 0000000000..c030f03d20
--- /dev/null
+++ b/dom/media/test/test_seek-3.html
@@ -0,0 +1,68 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: seek tests</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="seek_support.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+// The data being used in these tests is specified in manifest.js.
+// The functions to build the test array and to run a specific test are in
+// seek_support.js.
+
+const SEEK_TEST_NUMBER = 3;
+
+function test_seek3(v, seekTime, is, ok, finish) {
+
+// Test seeking works if current time is set but video is not played.
+var completed = false;
+var gotTimeupdate = false;
+
+function startTest() {
+ if (completed)
+ return;
+
+ v.currentTime=seekTime;
+}
+
+function timeupdate() {
+ gotTimeupdate = true;
+ v.removeEventListener("timeupdate", timeupdate);
+}
+
+function seekStarted() {
+ if (completed)
+ return;
+
+ ok(v.currentTime >= seekTime - 0.1, "Video currentTime should be around " + seekTime + ": " + v.currentTime);
+ v.addEventListener("timeupdate", timeupdate);
+}
+
+function seekEnded() {
+ if (completed)
+ return;
+
+ var t = v.currentTime;
+ ok(Math.abs(t - seekTime) <= 0.1, "Video currentTime should be around " + seekTime + ": " + t);
+ ok(gotTimeupdate, "Should have got timeupdate between seeking and seekended");
+ completed = true;
+ finish();
+}
+
+v.addEventListener("loadedmetadata", startTest);
+v.addEventListener("seeking", seekStarted);
+v.addEventListener("seeked", seekEnded);
+
+}
+
+manager.runTests(createTestArray(), startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_seek-4.html b/dom/media/test/test_seek-4.html
new file mode 100644
index 0000000000..4e5c1fee59
--- /dev/null
+++ b/dom/media/test/test_seek-4.html
@@ -0,0 +1,70 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: seek tests</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="seek_support.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+// The data being used in these tests is specified in manifest.js.
+// The functions to build the test array and to run a specific test are in
+// seek_support.js.
+
+const SEEK_TEST_NUMBER = 4;
+
+function test_seek4(v, seekTime, is, ok, finish) {
+
+// Test for a seek, followed by another seek before the first is complete.
+var seekCount = 0;
+var completed = false;
+
+function startTest() {
+ if (completed)
+ return;
+
+ v.currentTime=seekTime;
+ v._seekTarget=seekTime;
+}
+
+function seekStarted() {
+ if (completed)
+ return;
+
+ seekCount += 1;
+
+ ok(v.currentTime >= v._seekTarget - 0.1,
+ "Video currentTime should be around " + v._seekTarget + ": " + v.currentTime);
+ if (seekCount == 1) {
+ v.currentTime=seekTime/2;
+ v._seekTarget=seekTime/2;
+ }
+}
+
+function seekEnded() {
+ if (completed)
+ return;
+
+ if (seekCount == 2) {
+ ok(Math.abs(v.currentTime - seekTime/2) <= 0.1, "seek on target: " + v.currentTime);
+ completed = true;
+ finish();
+ }
+}
+
+v.addEventListener("loadedmetadata", startTest);
+v.addEventListener("seeking", seekStarted);
+v.addEventListener("seeked", seekEnded);
+
+}
+
+manager.runTests(createTestArray(), startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_seek-5.html b/dom/media/test/test_seek-5.html
new file mode 100644
index 0000000000..a86477f77c
--- /dev/null
+++ b/dom/media/test/test_seek-5.html
@@ -0,0 +1,69 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: seek tests</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="seek_support.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+// The data being used in these tests is specified in manifest.js.
+// The functions to build the test array and to run a specific test are in
+// seek_support.js.
+
+const SEEK_TEST_NUMBER = 5;
+
+function test_seek5(v, seekTime, is, ok, finish) {
+
+// Test for a seek, followed by a play before the seek completes, ensure we play at the end of the seek.
+var startPassed = false;
+var endPassed = false;
+var completed = false;
+
+function startTest() {
+ if (completed)
+ return;
+
+ v.currentTime=seekTime;
+}
+
+function seekStarted() {
+ if (completed)
+ return;
+ ok(v.currentTime >= seekTime - 0.1, "Video currentTime should be around " + seekTime + ": " + v.currentTime);
+ startPassed = true;
+ v.play();
+}
+
+function seekEnded() {
+ if (completed)
+ return;
+ endPassed = true;
+}
+
+function playbackEnded() {
+ if (completed)
+ return;
+ ok(startPassed, "Got seeking event");
+ ok(endPassed, "Got seeked event");
+ completed = true;
+ finish();
+}
+
+v.addEventListener("ended", playbackEnded);
+v.addEventListener("loadedmetadata", startTest);
+v.addEventListener("seeking", seekStarted);
+v.addEventListener("seeked", seekEnded);
+
+}
+
+manager.runTests(createTestArray(), startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_seek-6.html b/dom/media/test/test_seek-6.html
new file mode 100644
index 0000000000..32554c4c2f
--- /dev/null
+++ b/dom/media/test/test_seek-6.html
@@ -0,0 +1,64 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: seek tests</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="seek_support.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+// The data being used in these tests is specified in manifest.js.
+// The functions to build the test array and to run a specific test are in
+// seek_support.js.
+
+const SEEK_TEST_NUMBER = 6;
+
+function test_seek6(v, seekTime, is, ok, finish) {
+
+// Test for bug identified by Chris Pearce in comment 40 on
+// bug 449159.
+var seekCount = 0;
+var completed = false;
+var interval;
+
+function poll() {
+ v.currentTime;
+}
+
+function startTest() {
+ if (completed)
+ return;
+ interval = setInterval(poll, 10);
+ v.currentTime = Math.random() * v.duration;
+}
+
+function seekEnded() {
+ if (completed)
+ return;
+
+ seekCount++;
+ ok(true, "Seek " + seekCount);
+ if (seekCount == 3) {
+ clearInterval(interval);
+ completed = true;
+ finish();
+ } else {
+ v.currentTime = Math.random() * v.duration;
+ }
+}
+
+v.addEventListener("loadedmetadata", startTest);
+v.addEventListener("seeked", seekEnded);
+
+}
+
+manager.runTests(createTestArray(), startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_seek-7.html b/dom/media/test/test_seek-7.html
new file mode 100644
index 0000000000..96139d6f83
--- /dev/null
+++ b/dom/media/test/test_seek-7.html
@@ -0,0 +1,59 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: seek tests</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="seek_support.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+// The data being used in these tests is specified in manifest.js.
+// The functions to build the test array and to run a specific test are in
+// seek_support.js.
+
+const SEEK_TEST_NUMBER = 7;
+
+function test_seek7(v, seekTime, is, ok, finish) {
+
+// If a NaN is passed to currentTime, make sure this is caught
+// otherwise an infinite loop in the Ogg backend occurs.
+var completed = false;
+var thrown1 = false;
+var thrown3 = false;
+
+function startTest() {
+ if (completed)
+ return;
+
+ try {
+ v.currentTime = NaN;
+ } catch(e) {
+ thrown1 = true;
+ }
+
+ try {
+ v.currentTime = Math.random;
+ } catch(e) {
+ thrown3 = true;
+ }
+
+ completed = true;
+ ok(thrown1, "Setting currentTime to invalid value of NaN");
+ ok(thrown3, "Setting currentTime to invalid value of a function");
+ finish();
+}
+
+v.addEventListener("loadedmetadata", startTest);
+
+}
+
+manager.runTests(createTestArray(), startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_seek-8.html b/dom/media/test/test_seek-8.html
new file mode 100644
index 0000000000..2f6e390eef
--- /dev/null
+++ b/dom/media/test/test_seek-8.html
@@ -0,0 +1,42 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: seek tests</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="seek_support.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+// The data being used in these tests is specified in manifest.js.
+// The functions to build the test array and to run a specific test are in
+// seek_support.js.
+
+const SEEK_TEST_NUMBER = 8;
+
+function test_seek8(v, seekTime, is, ok, finish) {
+
+function startTest() {
+ v.currentTime = 1000;
+}
+
+function seekEnded() {
+ ok(Math.abs(v.currentTime - v.duration) < 0.2,
+ "currentTime " + v.currentTime + " close to " + v.duration);
+ finish();
+}
+
+v.addEventListener("loadedmetadata", startTest);
+v.addEventListener("seeked", seekEnded);
+
+}
+
+manager.runTests(createTestArray(), startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_seek-9.html b/dom/media/test/test_seek-9.html
new file mode 100644
index 0000000000..c03f6a75e3
--- /dev/null
+++ b/dom/media/test/test_seek-9.html
@@ -0,0 +1,41 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: seek tests</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="seek_support.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+// The data being used in these tests is specified in manifest.js.
+// The functions to build the test array and to run a specific test are in
+// seek_support.js.
+
+const SEEK_TEST_NUMBER = 9;
+
+function test_seek9(v, seekTime, is, ok, finish) {
+
+function startTest() {
+ v.currentTime = -1000;
+}
+
+function seekEnded() {
+ is(v.currentTime, 0, "currentTime clamped to 0");
+ finish();
+}
+
+v.addEventListener("loadedmetadata", startTest);
+v.addEventListener("seeked", seekEnded);
+
+}
+
+manager.runTests(createTestArray(), startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_seekLies.html b/dom/media/test/test_seekLies.html
new file mode 100644
index 0000000000..f3141eaabd
--- /dev/null
+++ b/dom/media/test/test_seekLies.html
@@ -0,0 +1,28 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: server lies about range requests</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script type="text/javascript" src="manifest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body onunload="mediaTestCleanup();">
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+function on_metadataloaded() {
+ var v = document.getElementById('v');
+ var d = Math.round(v.duration*1000);
+ ok(d == 4000, "Checking duration: " + d);
+ SimpleTest.finish();
+}
+
+SimpleTest.waitForExplicitFinish();
+</script>
+</pre>
+<video id='v'
+ preload="metadata"
+ src='seekLies.sjs'
+ onloadedmetadata='on_metadataloaded();'></video>
+</body>
+</html>
diff --git a/dom/media/test/test_seekToNextFrame.html b/dom/media/test/test_seekToNextFrame.html
new file mode 100644
index 0000000000..755d06e622
--- /dev/null
+++ b/dom/media/test/test_seekToNextFrame.html
@@ -0,0 +1,95 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test seekToNextFrame of media files that should play OK</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function startTest(test, token) {
+ var video = document.createElement('video');
+ video.preload = "metadata";
+ video.token = token;
+ video.seenSeeking = false;
+ video.seenEnded = false;
+
+ var handler = {
+ "ontimeout": function() {
+ Log(token, "timed out: ended=" + video.seenEnded);
+ }
+ };
+ manager.started(token, handler);
+
+ video.src = test.name;
+ video.name = test.name;
+
+ function callSeekToNextFrame() {
+ video.seekToNextFrame().then(
+ () => {
+ if (!video.seenSeeking) {
+ ok(false, video.token + ": Should have already received seeking event.");
+ }
+ video.seenSeeking = false;
+ if (!video.ended) {
+ callSeekToNextFrame();
+ }
+ },
+ () => {
+ ok(false, video.token + ": seekToNextFrame() failed.");
+ }
+ );
+ }
+
+ var onLoadedmetadata = function(t, v) { return function() {
+ callSeekToNextFrame();
+ }}(test, video);
+
+ var finish = function() {
+ video.finished = true;
+ video.removeEventListener("loadedmetadata", onLoadedmetadata);
+ video.removeEventListener("seeking", onSeeking);
+ removeNodeAndSource(video);
+ manager.finished(video.token);
+ }
+
+ var onEnded = function(t, v) { return function() {
+ v.seenEnded = true;
+ finish();
+ }}(test, video);
+
+ var onSeeking = function(t, v) { return function() {
+ if (v.seenSeeking) {
+ ok(false, v.token + ": Should yet receive seeking event.");
+ }
+ v.seenSeeking = true;
+ }}(test, video);
+
+ video.addEventListener("loadedmetadata", onLoadedmetadata);
+ video.addEventListener("seeking", onSeeking);
+ video.addEventListener("ended", onEnded);
+
+ document.body.appendChild(video);
+}
+
+SimpleTest.waitForExplicitFinish();
+SpecialPowers.pushPrefEnv(
+ {
+ "set": [
+ ["media.seekToNextFrame.enabled", true ],
+ ["media.dormant-on-pause-timeout-ms", -1]
+ ]
+ },
+ function() {
+ manager.runTests(gSeekToNextFrameTests, startTest);
+ });
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_seek_duration.html b/dom/media/test/test_seek_duration.html
new file mode 100644
index 0000000000..0240f31f94
--- /dev/null
+++ b/dom/media/test/test_seek_duration.html
@@ -0,0 +1,62 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: seek tests</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="seek_support.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+/**
+ * This test is used to make sure video's duration won't be changed when it
+ * reachs to the end after seeking to position where the time is very close to
+ * video's end time.
+ */
+
+SimpleTest.waitForExplicitFinish();
+
+async function test(name, videoProperties)
+{
+ const video = document.createElement('video');
+ video.src = "bunny.webm";
+ Object.assign(video, videoProperties);
+ document.body.appendChild(video);
+
+ const loadedMetadata = once(video, "loadedmetadata");
+ const canplay = once(video, "canplay");
+ const end = once(video, "ended");
+
+ info(`- wait for ${name} video loading metadata -`);
+ await loadedMetadata;
+ const originalDuration = video.duration;
+
+ info(`- seek ${name} video to the position which is close to end time -`);
+ // video's duration is 2.1 and the last key frame is in 2.0, we want to seek
+ // to that keyframe.
+ video.currentTime = originalDuration - 0.1;
+
+ info(`- play ${name} video until it ends -`);
+ await canplay;
+ await video.play();
+ await end;
+
+ is(video.duration, originalDuration, `${name} duration shouldn't change`);
+ removeNodeAndSource(video);
+}
+
+(async function startTest() {
+ await Promise.all([
+ test("unmuted", {}),
+ test("muted", {muted: true}),
+ ]);
+ SimpleTest.finish();
+})();
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_seek_negative.html b/dom/media/test/test_seek_negative.html
new file mode 100644
index 0000000000..98f0b9c910
--- /dev/null
+++ b/dom/media/test/test_seek_negative.html
@@ -0,0 +1,77 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: seeking to a negative time with readyState HAVE_NOTHING</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function startTest(test, token) {
+ var type = getMajorMimeType(test.type);
+ var v = document.createElement(type);
+ v.token = token;
+ manager.started(token);
+
+ // Seek to negative start time.
+ v.currentTime = -123;
+ is(v.readyState, v.HAVE_NOTHING, "readyState is HAVE_NOTHING");
+ ok(!v.seeking, "can't be seeking prior src defined");
+ is(v.currentTime, -123, "currentTime is original seek time");
+
+ v.src = test.name;
+
+ // Initialize running variables.
+ v._name = test.name;
+ v._seekStarted = false;
+ v._seekCompleted = false;
+ v._metadata = false;
+
+ var events = [ "suspend", "play", "canplay", "canplaythrough", "loadstart",
+ "loadedmetadata", "loadeddata", "playing", "ended", "error",
+ "stalled", "emptied", "abort", "waiting", "pause" ];
+ function logEvent(e) {
+ var video = e.target;
+ Log(e.target.token, "got " + e.type + " with currentTime = " + video.currentTime);
+ }
+ events.forEach(function(e) {
+ v.addEventListener(e, logEvent);
+ });
+
+ once(v, "seeking", function() {
+ v._seekStarted = true;
+ ok(v.currentTime >= 0, "currentTime should be positive");
+ });
+ once(v, "seeked", function() {
+ v._seekCompleted = true;
+ ok(v.currentTime >= 0, "currentTime should be positive");
+ });
+ once(v, "loadedmetadata", function() {
+ v._metadata = true;
+ ok(v.seeking, "element is seeking once readyState is HAVE_METADATA");
+ ok(v.currentTime >= 0, "currentTime should be positive");
+ });
+ once(v, "ended", function() {
+ ok(v._seekStarted, "seek should have started");
+ ok(v._seekCompleted, "seek should have completed");
+ ok(v._metadata, "loadedmetadata fired");
+ ok(v.currentTime >= 0, "currentTime should be positive");
+ removeNodeAndSource(v);
+ manager.finished(v.token);
+ });
+
+ v.play();
+}
+
+
+manager.runTests(gSmallTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_seek_nosrc.html b/dom/media/test/test_seek_nosrc.html
new file mode 100644
index 0000000000..56461007bd
--- /dev/null
+++ b/dom/media/test/test_seek_nosrc.html
@@ -0,0 +1,58 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: seek tests</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+SimpleTest.waitForExplicitFinish();
+
+var SEEK_TIME = 3.5;
+var seekStarted = false;
+var seekCompleted = false;
+var metadata = false;
+
+var v = document.createElement('video');
+document.body.appendChild(v);
+SimpleTest.registerCleanupFunction(function () {
+ v.remove();
+});
+
+try {
+ v.currentTime = SEEK_TIME;
+} catch (e) {
+ ok(false, "should not fire '" + e + "' event");
+}
+is(v.readyState, v.HAVE_NOTHING, "readyState is HAVE_NOTHING");
+ok(!v.seeking, "can't be seeking prior src defined");
+is(v.currentTime, SEEK_TIME, "currentTime is default playback start position");
+once(v, "seeking", function() {
+ seekStarted = true;
+});
+once(v, "seeked", function() {
+ seekCompleted = true;
+});
+once(v, "loadedmetadata", function() {
+ metadata = true;
+ ok(v.seeking, "element is seeking once readyState is HAVE_METADATA");
+});
+once(v, "ended", function() {
+ ok(seekStarted, "seek should have started");
+ ok(seekCompleted, "seek should have completed");
+ ok(metadata, "loadedmetadata fired");
+ ok(v.currentTime >= SEEK_TIME, "currentTime should be after seek time");
+ SimpleTest.finish();
+});
+
+v.src = "seek.webm";
+v.play();
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_seek_out_of_range.html b/dom/media/test/test_seek_out_of_range.html
new file mode 100644
index 0000000000..a477ebdd19
--- /dev/null
+++ b/dom/media/test/test_seek_out_of_range.html
@@ -0,0 +1,48 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: seeking off the end of a file</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+// Test if the ended event works correctly.
+
+async function initTest(test, token) {
+ var type = getMajorMimeType(test.type);
+ var v = document.createElement(type);
+ v.preload = "auto";
+ v.token = token;
+ manager.started(token);
+ v.src = test.name;
+ v._name = test.name;
+ document.body.appendChild(v);
+
+ await once(v, "loadedmetadata");
+ info(`${v._name}: seeking to the end of the media.`);
+ v.currentTime = 3.0 * v.duration;
+ // Wait for 'seeked' and 'ended' to be fired.
+ await Promise.all([once(v, "seeked"), once(v, "ended")]);
+ // Check currentTime is near the end of the media.
+ ok(Math.abs(v.duration - v.currentTime) < 0.1,
+ "Should be at end of media for " + v._name + " t=" + v.currentTime + " d=" + v.duration);
+ // Call play() to start playback from the beginning.
+ v.play();
+ await once(v, "ended");
+ ok(v.ended, "Checking ended set after seeking to EOF and playing for " + v._name);
+ removeNodeAndSource(v);
+ manager.finished(v.token);
+}
+
+manager.runTests(gSmallTests, initTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_seek_promise_bug1344357.html b/dom/media/test/test_seek_promise_bug1344357.html
new file mode 100644
index 0000000000..6c441bc1cf
--- /dev/null
+++ b/dom/media/test/test_seek_promise_bug1344357.html
@@ -0,0 +1,35 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: bug 1344357</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+
+<script>
+
+// This test always succeeds at runtime but should not cause any leaks at the end of mochitests.
+
+let manager = new MediaTestManager;
+
+function initTest(test, token) {
+ manager.started(token);
+
+ var win = window.open();
+ var video = win.document.createElement("video");
+ video.autoplay = true;
+ video.src = "http://example.com/tests/dom/media/test/" + test.name;
+ win.document.body.appendChild(video);
+ video.currentTime = test.duration / 2;
+ video.addEventListener("seeking", () => {
+ win.close();
+ manager.finished(token);
+ }, true);
+}
+
+manager.runTests(gSmallTests, initTest);
+
+</script> \ No newline at end of file
diff --git a/dom/media/test/test_seekable1.html b/dom/media/test/test_seekable1.html
new file mode 100644
index 0000000000..45c84b35df
--- /dev/null
+++ b/dom/media/test/test_seekable1.html
@@ -0,0 +1,66 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+<title>Test seekable member for media elements</title>
+<script type="text/javascript" src="/MochiKit/MochiKit.js"></script>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+<script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id='test'>
+<script class="testbody" type='application/javascript'>
+
+let manager = new MediaTestManager;
+
+function finish_test(element) {
+ if (element.parentNode)
+ element.remove();
+ element.src="";
+ manager.finished(element.token);
+}
+
+var tests = [
+// Test using a finite media stream, and a server supporting range requests
+{
+setup(element) {
+ is(element.seekable.length, 0, "seekable.length should be initialy 0.");
+ element.addEventListener("loadedmetadata", function() {
+ is(element.seekable.length, 1, "seekable.length should be 1 for a server supporting range requests.");
+
+ is(element.seekable.start(0), 0.0, "The start of the first range should be the initialTime.");
+ is(element.seekable.end(0), element.duration, "The end of the first range should be the duration.")
+ finish_test(element);
+ });
+ }
+}
+];
+
+function createTestArray() {
+ var A = [];
+ for (var k=0; k < gProgressTests.length; k++) {
+ var t = {};
+ t.setup = tests[0].setup;
+ t.name = gProgressTests[k].name;
+ t.type = gProgressTests[k].type;
+ A.push(t);
+ }
+ return A;
+}
+
+function startTest(test, token) {
+ var elemType = getMajorMimeType(test.type);
+ var element = document.createElement(elemType);
+ element.preload = "auto";
+ element.src = test.name;
+ element.token = token;
+ test.setup(element);
+ manager.started(token);
+}
+
+manager.runTests(createTestArray(), startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_setSinkId_after_loop.html b/dom/media/test/test_setSinkId_after_loop.html
new file mode 100644
index 0000000000..6224daa417
--- /dev/null
+++ b/dom/media/test/test_setSinkId_after_loop.html
@@ -0,0 +1,126 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+<title>Test setSinkId() after looping</title>
+<script src="/tests/SimpleTest/SimpleTest.js"></script>
+<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+</head>
+<script type="application/javascript">
+"use strict";
+
+/**
+This test captures loopback device audio to test that a media element produces
+audio output soon after a setSinkId() call after looping. In bug 1838756,
+incorrect timekeeping caused silence in the output as long as the duration of
+loops completed.
+ **/
+add_task(async () => {
+ const audio = new Audio();
+ audio.src = "sin-441-1s-44100.flac";
+ audio.loop = true;
+ audio.controls = true;
+ // Mute initially so that delayed audio from before setSinkId does not
+ // trigger a false pass.
+ audio.muted = true;
+ document.body.appendChild(audio);
+ const canplaythroughPromise = new Promise(r => audio.oncanplaythrough = r);
+ // Let the audio loop at least twice, which would result in 2 extra seconds
+ // of silence with bug 1838756.
+ let count = 0;
+ const loopedPromise = new Promise(r => audio.onseeked = () => {
+ if (++count == 2) {
+ r();
+ }
+ });
+
+ await SpecialPowers.pushPrefEnv({set: [
+ // skip gUM permission prompt
+ ["media.navigator.permission.disabled", true],
+ // enumerateDevices() without focus
+ ["media.devices.unfocused.enabled", true],
+ ]});
+ let loopbackInputLabel =
+ SpecialPowers.getCharPref("media.audio_loopback_dev", "");
+ if (!navigator.userAgent.includes("Linux")) {
+ todo_isnot(loopbackInputLabel, "", "audio_loopback_dev");
+ return;
+ }
+ isnot(loopbackInputLabel, "",
+ "audio_loopback_dev. Try --use-test-media-devices.");
+ const loopbackStream =
+ await navigator.mediaDevices.getUserMedia({audio: true});
+ is(loopbackStream.getTracks()[0].label, loopbackInputLabel,
+ "loopback track label");
+
+ await canplaythroughPromise;
+ let loopbackNode;
+ try {
+ audio.play();
+ await new Promise(r => audio.onplaying = r);
+ const ac = new AudioContext;
+ loopbackNode = ac.createMediaStreamSource(loopbackStream);
+ const processor1 = ac.createScriptProcessor(4096, 1, 0);
+ loopbackNode.connect(processor1);
+
+ // Check that the loopback stream contains silence now.
+ const {inputBuffer} = await new Promise(r => processor1.onaudioprocess = r);
+ loopbackNode.disconnect();
+ is(inputBuffer.getChannelData(0).find(value => value != 0.0), undefined,
+ "should have silence");
+
+ // Find output device
+ const devices = await navigator.mediaDevices.enumerateDevices();
+ let loopbackOutputLabel =
+ SpecialPowers.getCharPref("media.cubeb.output_device", "");
+ const outputDeviceInfo = devices.find(
+ ({kind, label}) => kind == "audiooutput" && label == loopbackOutputLabel
+ );
+ ok(outputDeviceInfo, `found ${loopbackOutputLabel}`);
+
+ await loopedPromise;
+ await audio.setSinkId(outputDeviceInfo.deviceId);
+ audio.muted = false;
+ // Use a new ScriptProcessor so that the first audioprocess event provides
+ // the rendering thread time after unmuting.
+ const processor2 = ac.createScriptProcessor(4096, 1, 0);
+ loopbackNode.connect(processor2);
+
+ let seenAudioCount = 0;
+ let lastSample, firstTime;
+ while (seenAudioCount < ac.sampleRate) {
+ const event = await new Promise(r => processor2.onaudioprocess = r);
+ let samples = event.inputBuffer.getChannelData(0);
+ for (const sample of samples) {
+ if (!seenAudioCount) {
+ if (!firstTime) {
+ firstTime = event.playbackTime;
+ }
+ if (sample != 0.0) {
+ seenAudioCount = 1;
+ } else {
+ const delay = event.playbackTime - firstTime;
+ if (delay > 1.0) {
+ ok(false, `${delay} seconds passed without receiving audio`);
+ return;
+ }
+ }
+ } else if (sample != 0.0) {
+ ++seenAudioCount;
+ } else if (lastSample == 0.0) {
+ ok(false, "should not have two consecutive zero sample in audio");
+ return;
+ }
+ lastSample = sample;
+ }
+ }
+ ok(true, "have one continuous second of audio");
+ } finally {
+ if (loopbackNode) {
+ loopbackNode.disconnect();
+ }
+ audio.pause();
+ loopbackStream.getTracks()[0].stop();
+ }
+});
+</script>
+</html>
diff --git a/dom/media/test/test_source.html b/dom/media/test/test_source.html
new file mode 100644
index 0000000000..944e95b1df
--- /dev/null
+++ b/dom/media/test/test_source.html
@@ -0,0 +1,91 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: append source child</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<video id="v1"></video>
+<audio id="a1"></audio>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var v1 = document.getElementById("v1");
+var a1 = document.getElementById("a1");
+v1.preload = "auto";
+a1.preload = "auto";
+
+is(v1.src, "", "src should be null");
+is(a1.src, "", "src should be null");
+is(v1.currentSrc, "", "currentSrc should be null");
+is(a1.currentSrc, "", "currentSrc should be null");
+is(v1.childNodes.length, 0, "should have no children");
+is(a1.childNodes.length, 0, "should have no children");
+
+function newSource(filter) {
+ var candidates = gSmallTests.filter(function(x){return filter.test(x.type);});
+ if (candidates.length) {
+ var e = document.createElement("source");
+ e.type = candidates[0].type;
+ e.src = candidates[0].name;
+ return e;
+ }
+ return null
+
+}
+
+var audioLoaded = false;
+var videoLoaded = false;
+
+function loaded(e) {
+ var media = e.target;
+ ok(media.networkState > 0, "networkState should be > 0");
+ is(media.childNodes.length, 1, "should have 1 child");
+ var sourceFile = media.currentSrc.substring(media.currentSrc.lastIndexOf('/')+1);
+ var resource = media.firstChild.src.substring(media.firstChild.src.lastIndexOf('/')+1);
+ is(sourceFile, resource, "loaded wrong resource!");
+ if (media == a1)
+ audioLoaded = true;
+ else if (media == v1)
+ videoLoaded = true;
+ if (audioLoaded && videoLoaded) {
+ SimpleTest.finish();
+ }
+}
+
+v1.addEventListener('loadeddata', loaded);
+a1.addEventListener('loadeddata', loaded);
+
+var videoSource = newSource(/^video/);
+if (videoSource) {
+ v1.appendChild(videoSource);
+ v1.load();
+} else {
+ // No video backends? Don't test anything.
+ videoLoaded = true;
+}
+
+var audioSource = newSource(/^audio/);
+if (audioSource) {
+ a1.appendChild(audioSource);
+ a1.load();
+} else {
+ audioLoaded = true;
+}
+
+if (!audioLoaded && !videoLoaded) {
+ SimpleTest.waitForExplicitFinish();
+} else {
+ if (audioLoaded) {
+ todo(false, "No audio types supported");
+ }
+ if (videoLoaded) {
+ todo(false, "No video types supported");
+ }
+}
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_source_null.html b/dom/media/test/test_source_null.html
new file mode 100644
index 0000000000..485edafd2d
--- /dev/null
+++ b/dom/media/test/test_source_null.html
@@ -0,0 +1,33 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=752087
+-->
+<head>
+ <title>Test for Bug 752087</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=752087">Mozilla Bug 752087</a>
+<p id="display"></p>
+<div id="content" style="display: none">
+
+</div>
+<pre id="test">
+</pre>
+
+<video id="v">
+<script>
+var v = document.getElementById('v');
+v.src = null; // crashes on NULL access if not handled
+
+ok(true, "setting video.src to null didn't crash!");
+var srcPath = v.src.split('/');
+ok(srcPath.length >= 3, "src should be within dom/media/test");
+is(srcPath[srcPath.length - 1], "null", "Setting src to null is handled like 'null' string");
+</script>
+</video>
+</body>
+</html>
diff --git a/dom/media/test/test_source_write.html b/dom/media/test/test_source_write.html
new file mode 100644
index 0000000000..29fffebd10
--- /dev/null
+++ b/dom/media/test/test_source_write.html
@@ -0,0 +1,40 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=462455
+-->
+<head>
+ <title>Test for Bug 462455</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=462455">Mozilla Bug 462455</a>
+<p id="display"></p>
+<div id="content" style="display: none">
+
+</div>
+<pre id="test">
+</pre>
+
+<video id="v">
+<script>
+var loadStarted = false;
+document.write('Pause parsing!');
+var v = document.getElementById('v');
+
+var resource = getPlayableVideo(gSmallTests);
+if (resource != null) {
+ var s = document.createElement("source");
+ s.src = resource.name;
+ s.type = resource.type;
+ v.appendChild(s);
+}
+
+ok(v.networkState == HTMLMediaElement.NETWORK_NO_SOURCE,
+ "We shouldn't start network load until the current task returns.");
+</script>
+</video>
+</body>
+</html>
diff --git a/dom/media/test/test_standalone.html b/dom/media/test/test_standalone.html
new file mode 100644
index 0000000000..620878a394
--- /dev/null
+++ b/dom/media/test/test_standalone.html
@@ -0,0 +1,61 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: standalone video documents</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body onload="doTest()">
+
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var iframes = [];
+
+for (let i=0; i<gSmallTests.length; ++i) {
+ var test = gSmallTests[i];
+
+ // We can't play WAV files in stand alone documents, so just don't
+ // run the test on non-video content types.
+ var tag = getMajorMimeType(test.type);
+ if (tag != "video" || !document.createElement("video").canPlayType(test.type))
+ continue;
+
+ let f = document.createElement("iframe");
+ f.src = test.name;
+ f._test = test;
+ f.id = "frame" + i;
+ iframes.push(f);
+ document.body.appendChild(f);
+}
+
+
+function filename(uri) {
+ return uri.substr(uri.lastIndexOf("/")+1);
+}
+
+function doTest()
+{
+ for (let i=0; i<iframes.length; ++i) {
+ let f = document.getElementById(iframes[i].id);
+ var v = f.contentDocument.body.firstChild;
+ is(v.tagName.toLowerCase(), "video", "Is video element");
+ var src = filename(v.currentSrc);
+ is(src, iframes[i]._test.name, "Name ("+src+") should match ("+iframes[i]._test.name+")");
+ is(v.controls, true, "Controls set (" + src + ")");
+ is(v.autoplay, true, "Autoplay set (" + src + ")");
+ }
+ SimpleTest.finish();
+}
+
+if (!iframes.length) {
+ todo(false, "No types supported");
+} else {
+ SimpleTest.waitForExplicitFinish();
+}
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_streams_capture_origin.html b/dom/media/test/test_streams_capture_origin.html
new file mode 100644
index 0000000000..13d5589c0d
--- /dev/null
+++ b/dom/media/test/test_streams_capture_origin.html
@@ -0,0 +1,90 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test for Bug 1189506</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=1189506">Mozilla Bug 1189506</a>
+<p id="display"></p>
+<video id="vin"></video>
+<video id="vout"></video>
+<video id="vout_cors" crossorigin></video>
+<canvas id="cin" width="40" height="30"></canvas>
+<canvas id="cout" width="40" height="30"></canvas>
+<canvas id="cout_cors" width="40" height="30"></canvas>
+<div id="content" style="display: none">
+</div>
+<pre id="test">
+<script type="application/javascript">
+/* global vin, vout, vout_cors, cin, cout, cout_cors */
+
+/** Test for Bug 1189506 **/
+
+SimpleTest.waitForExplicitFinish();
+
+async function start() {
+ const resource = getPlayableVideo(gSmallTests).name;
+ vin.src = "http://example.org:8000/tests/dom/media/test/" + resource;
+ vin.preload = "metadata";
+
+ await new Promise(r => vin.onloadedmetadata = r);
+ vout.srcObject = vin.mozCaptureStreamUntilEnded();
+ vout_cors.srcObject = vin.mozCaptureStreamUntilEnded();
+ vin.play();
+ vout.play();
+ vout_cors.play();
+
+ await new Promise(r => vout.onended = r);
+ is(vin.ended, vout.ended, "Source media element ends first");
+
+ const ctxin = SpecialPowers.wrap(cin.getContext("2d"));
+ ctxin.drawImage(vin, 0, 0);
+
+ {
+ info("Testing that the last frame is rendered");
+ const powerCtx = SpecialPowers.wrap(cout.getContext("2d"));
+ powerCtx.drawImage(vout, 0, 0);
+ const datain = ctxin.getImageData(0, 0, cin.width, cin.height);
+ const dataout = powerCtx.getImageData(0, 0, cout.width, cout.height);
+ for (let i = 0; i < datain.data.length; i += 4) {
+ const pixelin = datain.data.slice(i, i + 4).join(',');
+ const pixelout = dataout.data.slice(i, i + 4).join(',');
+ if (pixelin != pixelout) {
+ is(pixelout, pixelin, `Pixel #${i/4} is rendered as expected`);
+ break;
+ }
+ }
+ is(datain.data.length / 4, cin.width * cin.height,
+ "Checked expected number of pixels");
+ }
+
+ {
+ info("Testing that the principal is set");
+ const ctx = cout.getContext("2d");
+ ctx.drawImage(vout, 0, 0);
+ SimpleTest.doesThrow(() => ctx.getImageData(0, 0, cout.width, cout.height),
+ "SecurityError");
+ }
+
+ {
+ info("Testing that the crossorigin attribute is ignored for MediaStreams");
+ const ctx = cout_cors.getContext("2d");
+ ctx.drawImage(vout_cors, 0, 0);
+ SimpleTest.doesThrow(
+ () => ctx.getImageData(0, 0, cout_cors.width, cout_cors.height),
+ "SecurityError");
+ }
+}
+
+(async () => {
+ try { await start(); }
+ catch(e) { ok(false, `Rejected with ${e}`); }
+ finally { SimpleTest.finish(); }
+})();
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_streams_element_capture.html b/dom/media/test/test_streams_element_capture.html
new file mode 100644
index 0000000000..098136dba7
--- /dev/null
+++ b/dom/media/test/test_streams_element_capture.html
@@ -0,0 +1,125 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test that a MediaStream captured from one element plays back in another</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+const manager = new MediaTestManager();
+
+function checkDrawImage(vout, msg) {
+ const canvas = document.createElement("canvas");
+ const ctx = canvas.getContext("2d");
+ ctx.drawImage(vout, 0, 0);
+ const imgData = ctx.getImageData(0, 0, 1, 1);
+ is(imgData.data[3], 255, msg);
+}
+
+function maybeTodoGreaterThanOrEqual(a, b, msg) {
+ const isExpected = a >= b;
+ (isExpected ? ok : todo)(
+ isExpected,
+ `Got ${a}, expected at least ${b}; ${msg}`
+ );
+}
+
+async function startTest(test, token) {
+ manager.started(token);
+ const v = document.createElement('video');
+ const vout = document.createElement('video');
+
+ v.token = token;
+ v.id = "MediaDecoder";
+ vout.id = "MediaStream";
+
+ document.body.appendChild(vout);
+
+ // Log events for debugging.
+ const events = ["suspend", "play", "canplay", "canplaythrough", "loadstart", "loadedmetadata",
+ "loadeddata", "playing", "ended", "error", "stalled", "emptied", "abort",
+ "waiting", "pause"];
+ function logEvent(e) {
+ Log(token, `${e.target.id} got ${e.type}`);
+ }
+ for (const e of events) {
+ v.addEventListener(e, logEvent);
+ vout.addEventListener(e, logEvent);
+ };
+
+ v.src = test.name;
+ v.preload = 'metadata';
+ await new Promise(r => v.onloadedmetadata = r);
+
+ const stream = v.mozCaptureStreamUntilEnded();
+ vout.srcObject = stream;
+ is(vout.srcObject, stream,
+ `${token} set output element .srcObject correctly`);
+
+ isnot(stream.getTracks().length, 0, `${token} results in some tracks`);
+ const hadVideoTracks = stream.getVideoTracks().length >= 1;
+ // Wait for the resource fetch algorithm to have run, so that the media
+ // element is hooked up to the MediaStream and ready to go. If we don't do
+ // this, we're not guaranteed to render the very first video frame, which
+ // can make this test fail the drawImage test when a video resource only
+ // contains one frame.
+ await new Promise(r => vout.onloadstart = r);
+ v.play();
+ vout.play();
+
+ await Promise.race([
+ Promise.all([
+ new Promise(r => vout.onended = r),
+ new Promise(r => v.onended = r),
+ ]),
+ new Promise((res, rej) => vout.onerror = () => rej(new Error(vout.error.message))),
+ new Promise((res, rej) => v.onerror = () => rej(new Error(v.error.message))),
+ ]);
+
+ let duration = test.duration;
+ if (typeof(test.contentDuration) == "number") {
+ duration = test.contentDuration;
+ }
+ if (duration) {
+ // Unexpected results are tracked in bug 1839502
+ maybeTodoGreaterThanOrEqual(vout.currentTime, duration,
+ `${token} current time at end`);
+ }
+ is(vout.readyState, vout.HAVE_CURRENT_DATA,
+ `${token} checking readyState`);
+ ok(vout.ended, `${token} checking playback has ended`);
+ if (hadVideoTracks) {
+ ok(test.type.match(/^video/), `${token} is a video resource`);
+ checkDrawImage(vout, `${token} checking video frame pixel has been drawn`);
+ }
+ vout.remove();
+ removeNodeAndSource(v);
+}
+
+(async () => {
+ SimpleTest.requestCompleteLog();
+ SimpleTest.waitForExplicitFinish();
+ await SpecialPowers.pushPrefEnv(
+ { "set": [
+ ["privacy.reduceTimerPrecision", false],
+ ]});
+ let tests = gPlayTests;
+ // Filter out bug1377278.webm due to bug 1541401.
+ tests = tests.filter(t => !t.name.includes("1377278"));
+
+ manager.runTests(tests, async (test, token) => {
+ try {
+ await startTest(test, token);
+ } catch(e) {
+ ok(false, `Caught exception for ${token}: ${e}`);
+ }
+ manager.finished(token);
+ });
+})();
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_streams_element_capture_mediatrack.html b/dom/media/test/test_streams_element_capture_mediatrack.html
new file mode 100644
index 0000000000..6b4332d8af
--- /dev/null
+++ b/dom/media/test/test_streams_element_capture_mediatrack.html
@@ -0,0 +1,100 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test that a media element captureStream works when disabling MediaTracks</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+const manager = new MediaTestManager();
+
+async function startTest(test, token) {
+ manager.started(token);
+ const v = document.createElement('video');
+
+ document.body.appendChild(v);
+ v.token = token;
+ v.id = "MediaDecoder";
+
+ // Log events for debugging.
+ const events = ["suspend", "play", "canplay", "canplaythrough", "loadstart", "loadedmetadata",
+ "loadeddata", "playing", "ended", "error", "stalled", "emptied", "abort",
+ "waiting", "pause"];
+ function logEvent(e) {
+ Log(token, `${token}: ${e.target.id} got ${e.type}`);
+ }
+ for (const e of events) {
+ v.addEventListener(e, logEvent);
+ };
+
+ v.src = test.name;
+ v.preload = 'metadata';
+ await new Promise(r => v.onloadedmetadata = r);
+
+ const stream = v.mozCaptureStream();
+
+ is(stream.getAudioTracks().length, Math.min(1, v.audioTracks.length),
+ `${token}: Expected number of audio tracks`);
+ is(stream.getVideoTracks().length, Math.min(1, v.videoTracks.length),
+ `${token}: Expected number of video tracks`);
+
+ if (v.audioTracks.length) {
+ v.audioTracks[0].enabled = false;
+ const track = stream.getAudioTracks()[0];
+ await new Promise(r => track.onended = r);
+ is(track.readyState, "ended", `${token}: Audio track has ended on removal`);
+ await new Promise(r => stream.onremovetrack = r);
+ is(stream.getAudioTracks().length, 0,
+ `${token}: Audio track was removed on removetrack`);
+ }
+
+ if (v.videoTracks.length) {
+ v.videoTracks[0].selected = false;
+ const track = stream.getVideoTracks()[0];
+ await new Promise(r => track.onended = r);
+ is(track.readyState, "ended", `${token}: Video track has ended on removal`);
+ await new Promise(r => stream.onremovetrack = r);
+ is(stream.getVideoTracks().length, 0,
+ `${token}: Video track was removed on removetrack`);
+ }
+
+ stream.onaddtrack = () => ok(false, "Unexpected addtrack event");
+
+ v.play();
+
+ await Promise.race([
+ new Promise(r => v.ontimeupdate = r),
+ new Promise((res, rej) => v.onerror = () => rej(new Error(v.error.message))),
+ ]);
+
+ is(stream.getTracks().length, 0, `${token}: no tracks appeared during playback`);
+ removeNodeAndSource(v);
+}
+
+(async () => {
+ SimpleTest.requestCompleteLog();
+ SimpleTest.waitForExplicitFinish();
+ await SpecialPowers.pushPrefEnv(
+ { "set": [
+ ["media.track.enabled", true],
+ ]});
+ let tests = gPlayTests;
+ // Filter out bug1377278.webm due to bug 1541401.
+ tests = tests.filter(t => !t.name.includes("1377278"));
+
+ manager.runTests(tests, async (test, token) => {
+ try {
+ await startTest(test, token);
+ } catch(e) {
+ ok(false, `Caught exception for ${token}: ${e}`);
+ }
+ manager.finished(token);
+ });
+})();
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_streams_element_capture_playback.html b/dom/media/test/test_streams_element_capture_playback.html
new file mode 100644
index 0000000000..ce083069ef
--- /dev/null
+++ b/dom/media/test/test_streams_element_capture_playback.html
@@ -0,0 +1,47 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test that capturing a stream doesn't stop the underlying element from firing events</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<audio id="a"></audio>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+SimpleTest.waitForExplicitFinish();
+
+var a = document.getElementById('a');
+var validTimeUpdate = false;
+
+function startTest() {
+ a.src = "big.wav";
+ var context = new AudioContext();
+ var node = context.createMediaElementSource(a);
+ node.connect(context.destination);
+ a.addEventListener("timeupdate", function() {
+ if (a.currentTime > 0.0 && a.currentTime < 5.0 && !validTimeUpdate) {
+ validTimeUpdate = true;
+ ok(true, "Received reasonable currentTime in a timeupdate");
+ SimpleTest.finish();
+ }
+ });
+ a.addEventListener("ended", function() {
+ if (!validTimeUpdate) {
+ ok(false, "Received reasonable currentTime in a timeupdate");
+ SimpleTest.finish();
+ }
+ });
+ a.play();
+}
+
+if (a.canPlayType("audio/wave")) {
+ startTest();
+} else {
+ todo(false, "No playable audio");
+}
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_streams_element_capture_reset.html b/dom/media/test/test_streams_element_capture_reset.html
new file mode 100644
index 0000000000..625b0fe23f
--- /dev/null
+++ b/dom/media/test/test_streams_element_capture_reset.html
@@ -0,0 +1,174 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test that reloading, pausing and seeking in a media element that's being captured behaves as expected</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script src="manifest.js"></script>
+</head>
+<body>
+<video id="v"></video>
+<video id="vout"></video>
+<video id="vout_untilended"></video>
+<pre id="test">
+<script>
+const v = document.getElementById('v');
+const vout = document.getElementById('vout');
+const vout_untilended = document.getElementById('vout_untilended');
+
+function dumpEvent(event) {
+ const video = event.target;
+ info(
+ `${video.name}:${video.id} GOT EVENT ${event.type} ` +
+ `currentTime=${video.currentTime} paused=${video.paused} ` +
+ `ended=${video.ended} readyState=${video.readyState}`
+ );
+}
+
+function unexpected(event) {
+ ok(false, `${event.type} event received on ${event.target.id} unexpectedly`);
+};
+
+const events = ["timeupdate", "seeking", "seeked", "ended", "playing", "pause"];
+for (const e of events) {
+ v.addEventListener(e, dumpEvent);
+ vout.addEventListener(e, dumpEvent);
+ vout_untilended.addEventListener(e, dumpEvent);
+}
+
+function isWithinEps(a, b, msg) {
+ ok(Math.abs(a - b) < 0.01,
+ "Got " + a + ", expected " + b + "; " + msg);
+}
+
+function isGreaterThanOrEqualEps(a, b, msg) {
+ ok(a >= b - 0.01,
+ "Got " + a + ", expected at least " + b + "; " + msg);
+}
+
+async function startTest(test) {
+ const seekTime = test.duration/2;
+ const contentDuration = test.contentDuration ?? test.duration;
+
+ v.src = test.name;
+ v.name = test.name;
+ vout.name = test.name;
+ vout_untilended.name = test.name;
+ v.preload = "metadata";
+ await new Promise(r => v.onloadedmetadata = r);
+
+ vout.srcObject = v.mozCaptureStream();
+ vout.play();
+
+ vout_untilended.srcObject = v.mozCaptureStreamUntilEnded();
+ vout_untilended.play();
+
+ for (const track of [
+ ...vout.srcObject.getTracks(),
+ ...vout_untilended.srcObject.getTracks(),
+ ]) {
+ ok(track.muted, `${track.kind} track ${track.id} should be muted`);
+ }
+
+ v.play();
+
+ await Promise.all([
+ ...vout.srcObject.getTracks(),
+ ...vout_untilended.srcObject.getTracks()
+ ].map(t => new Promise(r => t.onunmute = r)));
+
+ await new Promise(r => v.onended = r);
+ isGreaterThanOrEqualEps(v.currentTime, test.duration,
+ "checking v.currentTime at first 'ended' event");
+
+ await Promise.all([
+ new Promise(r => vout.onended = r),
+ new Promise(r => vout_untilended.onended = r),
+ ]);
+
+ isGreaterThanOrEqualEps(vout.currentTime, contentDuration,
+ "checking vout.currentTime at first 'ended' event");
+ ok(vout.ended, "checking vout has actually ended");
+ ok(vout_untilended.ended, "checking vout_untilended has actually ended");
+
+ vout_untilended.srcObject.onaddtrack = unexpected;
+ vout_untilended.onplaying = unexpected;
+ vout_untilended.onended = unexpected;
+
+ const voutPreSeekCurrentTime = vout.currentTime;
+ v.currentTime = seekTime;
+ await new Promise(r => v.onseeked = r);
+
+ is(v.currentTime, seekTime, "Finished seeking");
+ is(vout.currentTime, voutPreSeekCurrentTime,
+ "checking vout.currentTime has not changed after seeking");
+
+ v.play();
+ vout.play();
+
+ await new Promise(r => v.onended = r);
+ isGreaterThanOrEqualEps(v.currentTime, test.duration,
+ "checking v.currentTime at second 'ended' event");
+
+ await new Promise(r => vout.onended = r);
+ isGreaterThanOrEqualEps(vout.currentTime,
+ (test.duration - seekTime) + contentDuration,
+ "checking vout.currentTime after seeking and playing through again");
+
+ v.src = test.name + "?1";
+ vout.play();
+ await v.play();
+
+ isnot(vout.srcObject.getTracks().length, 0, "There are some output tracks");
+
+ vout.onended = unexpected;
+ vout.srcObject.onremovetrack = unexpected;
+
+ v.pause();
+ await Promise.all(
+ vout.srcObject.getTracks().map(t => new Promise(r => t.onmute = r))
+ );
+
+ for (const track of vout.srcObject.getTracks()) {
+ track.onunmute = unexpected;
+ }
+
+ v.currentTime = 0;
+ await new Promise(r => v.onseeked = r);
+
+ v.play();
+ await Promise.all(
+ vout.srcObject.getTracks().map(t => new Promise(r => t.onunmute = r))
+ );
+
+ vout.srcObject.onremovetrack = null;
+
+ await new Promise(r => v.onended = r);
+ isGreaterThanOrEqualEps(v.currentTime, test.duration,
+ "checking v.currentTime at third 'ended' event");
+
+ await new Promise(r => vout.onended = r);
+ isGreaterThanOrEqualEps(vout.currentTime,
+ (test.duration - seekTime) + contentDuration*2,
+ "checking vout.currentTime after seeking, playing through and reloading");
+}
+
+(async () => {
+ SimpleTest.waitForExplicitFinish();
+ try {
+ const testVideo = getPlayableVideo(gSmallTests);
+ if (testVideo) {
+ await startTest(testVideo);
+ } else {
+ todo(false, "No playable video");
+ }
+ } catch(e) {
+ ok(false, `Error: ${e}`);
+ } finally {
+ SimpleTest.finish();
+ }
+})();
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_streams_element_capture_twice.html b/dom/media/test/test_streams_element_capture_twice.html
new file mode 100644
index 0000000000..0e30be1801
--- /dev/null
+++ b/dom/media/test/test_streams_element_capture_twice.html
@@ -0,0 +1,79 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test that capturing a media element, then reloading and capturing again, works</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<video id="v"></video>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+SimpleTest.waitForExplicitFinish();
+
+const v = document.getElementById('v');
+
+function dumpEvent(event) {
+ const video = event.target;
+ info(video.name + " GOT EVENT " + event.type +
+ " currentTime=" + video.currentTime +
+ " paused=" + video.paused +
+ " ended=" + video.ended +
+ " readyState=" + video.readyState);
+}
+
+const events = ["timeupdate", "seeking", "seeked", "ended", "playing", "pause"];
+for (let i = 0; i < events.length; ++i) {
+ v.addEventListener(events[i], dumpEvent);
+}
+
+async function startTest(src) {
+ v.preload = "metadata";
+ v.src = src;
+ await new Promise(r => v.onloadedmetadata = r);
+ const s1 = v.mozCaptureStream();
+ const tracks = s1.getTracks();
+ is(tracks.length, 2, "Expected total tracks, s1, capture 1");
+ is(s1.getAudioTracks().length, 1, "Expected audio tracks, s1, capture 1");
+ is(s1.getVideoTracks().length, 1, "Expected video tracks, s1, capture 1");
+ is(s1.getAudioTracks()[0].readyState, "live", "Live audio, s1, capture 1");
+ is(s1.getVideoTracks()[0].readyState, "live", "Live video, s1, capture 1");
+
+ v.src = null;
+ for (let i = 0; i < tracks.length; ++i) {
+ await Promise.race(tracks.map(t => new Promise(r => t.onended = r)));
+ await new Promise(r => s1.onremovetrack = r);
+ }
+ is(s1.getTracks().length, 0, "Expected total tracks, s1, metadata 2");
+
+ v.src = src;
+ await new Promise(r => v.onloadedmetadata = r);
+ is(s1.getTracks().length, 2, "Expected total tracks, s1, metadata 2");
+ is(s1.getAudioTracks().length, 1, "Expected audio tracks, s1, metadata 2");
+ is(s1.getVideoTracks().length, 1, "Expected video tracks, s1, metadata 2");
+ is(s1.getAudioTracks()[0].readyState, "live", "Live audio, s1, metadata 2");
+ is(s1.getVideoTracks()[0].readyState, "live", "Live video, s1, metadata 2");
+
+ const s2 = v.mozCaptureStream();
+ is(s1.getTracks().length, 2, "Expected total tracks remains, s1, capture 2");
+ is(s2.getTracks().length, 2, "Expected total tracks, s2, capture 2");
+ is(s2.getAudioTracks().length, 1, "Expected audio tracks, s2, capture 2");
+ is(s2.getVideoTracks().length, 1, "Expected video tracks, s2, capture 2");
+ is(s2.getAudioTracks()[0].readyState, "live", "Live audio, s2, capture 2");
+ is(s2.getVideoTracks()[0].readyState, "live", "Live video, s2, capture 2");
+}
+
+(async function() {
+ try {
+ await startTest("short-video.ogv");
+ } catch(e) {
+ ok(false, `Caught error: ${e}${e.stack ? '\n' + e.stack : ''}`);
+ } finally {
+ SimpleTest.finish();
+ }
+})();
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_streams_firstframe.html b/dom/media/test/test_streams_firstframe.html
new file mode 100644
index 0000000000..ea1dbd35c1
--- /dev/null
+++ b/dom/media/test/test_streams_firstframe.html
@@ -0,0 +1,67 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test that a non-autoplaying, non-playing element with a MediaStream source triggers canplay and shows a first frame</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script src="manifest.js"></script>
+ <script src="/tests/dom/canvas/test/captureStream_common.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+async function runTest() {
+ const canvas = document.createElement("canvas");
+ canvas.getContext("2d");
+ const helper = new CaptureStreamTestHelper2D(100, 100);
+
+ const video = document.createElement("video");
+ document.body.appendChild(video);
+
+ video.srcObject = canvas.captureStream();
+ helper.drawColor(canvas, helper.red);
+
+ await Promise.race([
+ new Promise(r => video.oncanplay = r),
+ new Promise(r => setTimeout(r, 30000))
+ .then(() => Promise.reject(new Error("Canplay timeout"))),
+ ]);
+
+ ok(true, "Got \"canplay\"");
+ is(video.readyState, video.HAVE_ENOUGH_DATA, "Expected readyState");
+ ok(helper.isPixel(helper.getPixel(video), helper.red),
+ "First frame is rendered before playing");
+
+ helper.drawColor(canvas, helper.green);
+ await helper.pixelMustNotBecome(video, helper.green, {
+ time: 1000,
+ infoString: "Rendered first frame doesn't change on new frame from source"
+ });
+ ok(helper.isPixel(helper.getPixel(video), helper.red),
+ "First frame is still rendered");
+
+ video.play();
+ helper.drawColor(canvas, helper.blue);
+ await helper.pixelMustBecome(video, helper.blue, {
+ infoString: "New frame gets rendered when playing"
+ });
+
+ video.srcObject.getTracks().forEach(t => t.stop());
+}
+
+(async function() {
+ SimpleTest.waitForExplicitFinish();
+ SimpleTest.requestFlakyTimeout("Explicit timeout reasons");
+ try {
+ await runTest();
+ } catch(e) {
+ ok(false, e);
+ }
+ SimpleTest.finish();
+})();
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_streams_gc.html b/dom/media/test/test_streams_gc.html
new file mode 100644
index 0000000000..d2ba23042b
--- /dev/null
+++ b/dom/media/test/test_streams_gc.html
@@ -0,0 +1,44 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test garbage collection of captured stream (bug 806754)</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body onload="doTest()">
+<audio id="a" preload="metadata"></audio>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+SimpleTest.waitForExplicitFinish();
+
+var a = document.getElementById('a');
+a.src = getPlayableAudio(gSmallTests).name;
+
+function forceGC() {
+ SpecialPowers.gc();
+ SpecialPowers.forceGC();
+ SpecialPowers.forceCC();
+}
+
+function doTest() {
+ a.mozCaptureStreamUntilEnded();
+
+ a.addEventListener("seeked", function() {
+ a.play();
+
+ a.addEventListener("play", function() {
+ a.addEventListener("ended", function() {
+ ok(true, "GC completed OK");
+ SimpleTest.finish();
+ });
+ });
+ });
+
+ a.currentTime = a.duration;
+ setTimeout(forceGC, 0);
+}
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_streams_individual_pause.html b/dom/media/test/test_streams_individual_pause.html
new file mode 100644
index 0000000000..c49b563d76
--- /dev/null
+++ b/dom/media/test/test_streams_individual_pause.html
@@ -0,0 +1,77 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test for bug 1073406. Pausing a video element should not pause another playing the same stream.</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="text/javascript" src="gUM_support.js"></script>
+</head>
+<body>
+<video id="video1" autoplay></video>
+<video id="video2" autoplay></video>
+<script class="testbody" type="text/javascript">
+function getVideoImagePixelData(v) {
+ const canvas = document.createElement("canvas");
+ const ctx = canvas.getContext("2d");
+ ctx.drawImage(v, 0, 0);
+ const imgData = ctx.getImageData(canvas.width/2, canvas.height/2, 1, 1).data;
+ return "r" + imgData[0] +
+ "g" + imgData[1] +
+ "b" + imgData[2] +
+ "a" + imgData[3];
+}
+
+async function startTest() {
+ // This test expects fake devices so that the video color will change
+ // over time, explicitly request fakes.
+ await pushGetUserMediaTestPrefs({fakeAudio: true, fakeVideo: true});
+ const stream = await navigator.mediaDevices.getUserMedia({video: true});
+ const video1 = document.getElementById('video1');
+ const video2 = document.getElementById('video2');
+
+ video1.srcObject = stream;
+ video2.srcObject = stream;
+
+ await new Promise(r => video1.onplaying = r);
+ video1.pause();
+ await new Promise(r => video1.onpause = r);
+
+ const v1PausedImageData = getVideoImagePixelData(video1);
+ const v2PausedImageData = getVideoImagePixelData(video2);
+
+ while (getVideoImagePixelData(video2) == v2PausedImageData) {
+ info("video2 has not progressed. Waiting.");
+ await new Promise(r => video2.ontimeupdate = r);
+ }
+
+ // Wait for a while to be sure video1 would have gotten a frame
+ // if it is playing.
+ for (let i = 3; i != 0; i--) {
+ await new Promise(r => video2.ontimeupdate = r);
+ }
+ info("video2 progressed OK");
+
+ isnot(video1.currentTime, video2.currentTime,
+ "v1 and v2 should not be at the same currentTime");
+ is(getVideoImagePixelData(video1), v1PausedImageData,
+ "video1 video frame should not have updated since video1 paused");
+
+ for (const t of stream.getTracks()) {
+ t.stop();
+ }
+}
+
+SimpleTest.waitForExplicitFinish();
+(async function() {
+ try {
+ await startTest();
+ } catch(error) {
+ ok(false, "getUserMedia should not fail, got " + error.name);
+ } finally {
+ SimpleTest.finish();
+ }
+})();
+</script>
+</body>
+</html>
diff --git a/dom/media/test/test_streams_srcObject.html b/dom/media/test/test_streams_srcObject.html
new file mode 100644
index 0000000000..8ea5797d14
--- /dev/null
+++ b/dom/media/test/test_streams_srcObject.html
@@ -0,0 +1,60 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test interactions of src and srcObject</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body onload="doTests()">
+<audio id="a1"></audio>
+<audio id="a2"></audio>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+SimpleTest.waitForExplicitFinish();
+
+var doTest = srcObject => new Promise(resolve => {
+ var a = document.getElementById('a1');
+ a.src = getPlayableAudio(gSmallTests).name;
+ var b = new Audio();
+
+ var newSrc = a.src + "?2";
+ b.src = newSrc;
+ is(b[srcObject], null, "Initial srcObject is null");
+ var stream = a.mozCaptureStream();
+ b[srcObject] = stream;
+ is(b[srcObject], stream, "Stream set correctly");
+ try {
+ b[srcObject] = "invalid";
+ ok(false, "Setting srcObject to an invalid value should throw.");
+ } catch (e) {
+ ok(e instanceof TypeError, "Exception should be a TypeError");
+ }
+ is(b[srcObject], stream, "Stream not set to invalid value");
+ is(b.src, newSrc, "src attribute not affected by setting srcObject");
+ var step = 0;
+ b.addEventListener("loadedmetadata", function() {
+ if (step == 0) {
+ is(b.currentSrc, "", "currentSrc set to empty string while playing srcObject");
+ b[srcObject] = null;
+ is(b[srcObject], null, "Stream set to null");
+ // The resource selection algorithm will run again and choose b.src
+ } else if (step == 1) {
+ is(b.currentSrc, b.src, "currentSrc set to src now that srcObject is null");
+ resolve();
+ }
+ ++step;
+ });
+ a.play();
+ b.play();
+});
+
+var doTests = () => doTest("srcObject")
+ .catch(e => ok(false, "Unexpected error: " + e))
+ .then(() => SimpleTest.finish())
+ .catch(e => ok(false, "Coding error: " + e));
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_streams_tracks.html b/dom/media/test/test_streams_tracks.html
new file mode 100644
index 0000000000..845bc36ca3
--- /dev/null
+++ b/dom/media/test/test_streams_tracks.html
@@ -0,0 +1,66 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test MediaStreamTrack interfaces</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+const manager = new MediaTestManager;
+
+function testTracks(tracks, hasTrack, kind, src) {
+ is(tracks.length, hasTrack ? 1 : 0, `Correct ${kind} track count for ${src}`);
+ for (const track of tracks) {
+ is(track.readyState, "live", `Track ${track.id} should still be live`);
+ is(track.kind, kind, `Correct track kind for track ${track.id} of ${src}`);
+ ok(/\{........-....-....-....-............\}/.test(track.id),
+ `id ${track.id} for ${track.kind} track of ${src} has correct form`);
+ }
+}
+
+async function startTest(test, token) {
+ try {
+ info(`Starting test of ${test.name}`);
+ const element = document.createElement("video");
+
+ element.token = token;
+ manager.started(token);
+
+ element.src = test.name;
+ element.test = test;
+ const stream = element.mozCaptureStreamUntilEnded();
+
+ element.play();
+
+ await new Promise(r => element.onloadedmetadata = r);
+
+ testTracks(stream.getAudioTracks(), test.hasAudio, "audio", test.name);
+ testTracks(stream.getVideoTracks(), test.hasVideo, "video", test.name);
+ const tracks = stream.getTracks();
+
+ await new Promise(r => element.onended = r);
+
+ for (let i = 0; i < tracks.length; ++i) {
+ await Promise.race(
+ tracks.map(t => new Promise(r => t.onended = r))
+ );
+ await new Promise(r => stream.onremovetrack = r);
+ }
+
+ testTracks(stream.getAudioTracks(), false, "audio", test.name);
+ testTracks(stream.getVideoTracks(), false, "video", test.name);
+ } catch(e) {
+ ok(false, `Caught error: ${e}`);
+ } finally {
+ manager.finished(token);
+ }
+}
+
+manager.runTests(gTrackTests, startTest);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_suspend_media_by_inactive_docshell.html b/dom/media/test/test_suspend_media_by_inactive_docshell.html
new file mode 100644
index 0000000000..7af2b63a3a
--- /dev/null
+++ b/dom/media/test/test_suspend_media_by_inactive_docshell.html
@@ -0,0 +1,72 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test suspending media by inactive docShell</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script src="manifest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+</head>
+<body>
+<video id="testVideo" src="gizmo.mp4" loop></video>
+<script class="testbody" type="text/javascript">
+/**
+ * When calling `browser.suspendMediaWhenInactive`, it can set the docShell's
+ * corresponding flag that is used to suspend media when the docShell is
+ * inactive. This test is used to check if we can suspend/resume the media
+ * correctly when changing docShell's active state.
+ */
+async function startTest() {
+ const video = document.getElementById("testVideo");
+
+ info(`start video`);
+ await video.play();
+
+ info(`set docShell inactive which would suspend media`);
+ await setDocShellActive(false);
+
+ info(`set docShell active which would resume media`);
+ await setDocShellActive(true);
+
+ SimpleTest.finish();
+}
+
+SimpleTest.waitForExplicitFinish();
+SpecialPowers.pushPrefEnv(
+ {"set": [["media.testing-only-events", true]]}, startTest);
+
+/**
+ * The following are test helper functions.
+ */
+function mediaSuspendedStateShouldEqualTo(expected) {
+ const video = document.getElementById("testVideo");
+ const result = SpecialPowers.wrap(video).isSuspendedByInactiveDocOrDocShell;
+ is(result, expected, `media's suspended state is correct`);
+}
+
+function setDocShellActive(isActive) {
+ const win = SpecialPowers.wrap(window);
+ const docShell = win.docShell;
+ let p = new Promise(r => {
+ docShell.chromeEventHandler.addEventListener("MozMediaSuspendChanged",
+ () => {
+ mediaSuspendedStateShouldEqualTo(!isActive);
+ r();
+ }, {once : true}
+ );
+ });
+
+ SpecialPowers.spawnChrome([isActive], active => {
+ // This flag is used to prevent media from playing when docShell is
+ // inactive. After updating `docshell.isActive`, it would suspend/resume
+ // media and we // wait suspending/resuming finishing by listening to
+ // `MozMediaSuspendChanged`.
+ this.browsingContext.top.suspendMediaWhenInactive = true;
+ this.browsingContext.top.isActive = active;
+ });
+
+ return p;
+}
+
+</script>
+</body>
+</html>
diff --git a/dom/media/test/test_temporary_file_blob_video_plays.html b/dom/media/test/test_temporary_file_blob_video_plays.html
new file mode 100644
index 0000000000..87f6b3c4e6
--- /dev/null
+++ b/dom/media/test/test_temporary_file_blob_video_plays.html
@@ -0,0 +1,75 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test MediaRecorder Recording canvas stream</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script src="/tests/dom/canvas/test/captureStream_common.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<pre id="test">
+<div id="content">
+</div>
+<script class="testbody" type="text/javascript">
+
+function startTest() {
+ var canvas = document.createElement("canvas");
+ canvas.width = canvas.height = 100;
+ document.getElementById("content").appendChild(canvas);
+
+ var helper = new CaptureStreamTestHelper2D(100, 100);
+ helper.drawColor(canvas, helper.red);
+
+ var stream = canvas.captureStream(0);
+
+ var blob;
+
+ let mediaRecorder = new MediaRecorder(stream);
+ is(mediaRecorder.stream, stream,
+ "Media recorder stream = canvas stream at the start of recording");
+
+ mediaRecorder.onwarning = () => ok(false, "warning unexpectedly fired");
+
+ mediaRecorder.onerror = () => ok(false, "Recording failed");
+
+ mediaRecorder.ondataavailable = ev => {
+ is(blob, undefined, "Should only get one dataavailable event");
+ blob = ev.data;
+ };
+
+ mediaRecorder.onstart = () => {
+ info("Got 'start' event");
+ // We just want one frame encoded, to see that the recorder produces something readable.
+ mediaRecorder.stop();
+ };
+
+ mediaRecorder.onstop = () => {
+ info("Got 'stop' event");
+ ok(blob, "Should have gotten a data blob");
+
+ var video = document.createElement("video");
+ video.id = "recorded-video";
+ video.src = URL.createObjectURL(blob);
+ video.play();
+ video.onerror = err => {
+ ok(false, "Should be able to play the recording. Got error. code=" + video.error.code);
+ SimpleTest.finish();
+ };
+ document.getElementById("content").appendChild(video);
+ helper.pixelMustBecome(video, helper.red, {
+ threshold: 128,
+ infoString: "Should become red",
+ }).then(SimpleTest.finish);
+ };
+
+ mediaRecorder.start();
+ is(mediaRecorder.state, "recording", "Media recorder should be recording");
+}
+
+SimpleTest.waitForExplicitFinish();
+SpecialPowers.pushPrefEnv({set:[["media.recorder.max_memory", 1]]}, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_timeupdate_small_files.html b/dom/media/test/test_timeupdate_small_files.html
new file mode 100644
index 0000000000..fca35e5b71
--- /dev/null
+++ b/dom/media/test/test_timeupdate_small_files.html
@@ -0,0 +1,88 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=495319
+-->
+
+<head>
+ <title>Bug 495319 - playing back small audio files should fire timeupdate</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=495319">Mozilla Bug 495319</a>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function ended(e) {
+ var v = e.target;
+ ++v.counter.ended;
+ is(v.counter.ended, 1, v._name + " should see ended only once");
+ ok(v.counter.timeupdate > 0, v._name + " should see at least one timeupdate: " + v.currentTime);
+
+ // Rest event counters for we don't allow events after ended.
+ eventsToLog.forEach(function(event) {
+ v.counter[event] = 0;
+ });
+
+ // Finish the test after 500ms. We shouldn't receive any timeupdate events
+ // after the ended event, so this gives time for any pending timeupdate events
+ // to fire so we can ensure we don't regress behaviour.
+ setTimeout(
+ function() {
+ // Remove the event listeners before removing the video from the document.
+ // We should receive a timeupdate and pause event when we remove the element
+ // from the document (as the element is specified to behave as if pause() was
+ // invoked when it's removed from a document), and we don't want those
+ // confusing the test results.
+ v.removeEventListener("ended", ended);
+ eventsToLog.forEach(function(event) {
+ v.removeEventListener(event, logEvent);
+ });
+ removeNodeAndSource(v);
+ manager.finished(v.token);
+ },
+ 500);
+}
+
+var eventsToLog = ["play", "canplay", "canplaythrough", "loadstart", "loadedmetadata",
+ "loadeddata", "playing", "timeupdate", "error", "stalled", "emptied", "abort",
+ "waiting", "pause"];
+
+function logEvent(event) {
+ var v = event.target;
+ ++v.counter[event.type];
+ if (v.counter.ended > 0) {
+ is(v.counter[event.type], 0, v._name + " got unexpected " + event.type + " after ended");
+ }
+}
+
+function startTest(test, token) {
+ var type = getMajorMimeType(test.type);
+ var v = document.createElement(type);
+ v.token = token;
+ manager.started(token);
+ v.src = test.name;
+ v._name = test.name;
+
+ // Keep how many events received for each event type.
+ v.counter = {};
+ eventsToLog.forEach(function(e) {
+ v.addEventListener(e, logEvent);
+ v.counter[e] = 0;
+ });
+ v.addEventListener("ended", ended);
+ v.counter.ended = 0;
+ document.body.appendChild(v);
+ v.play();
+}
+
+manager.runTests(gSmallTests, startTest);
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_unseekable.html b/dom/media/test/test_unseekable.html
new file mode 100644
index 0000000000..52c7eccc50
--- /dev/null
+++ b/dom/media/test/test_unseekable.html
@@ -0,0 +1,101 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: unseekable</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+/*
+
+Test that unseekable media can't be seeked. We load a media that shouldn't
+be seekable, and play through once. While playing through we repeatedly try
+to seek and check that nothing happens when we do. We also verify that the
+seekable ranges are empty.
+
+*/
+
+var manager = new MediaTestManager;
+
+var onseeking = function(event) {
+ var v = event.target;
+ v.actuallySeeked = true;
+};
+
+var onseeked = function(event) {
+ var v = event.target;
+ v.actuallySeeked = true;
+};
+
+var ontimeupdate = function(event) {
+ var v = event.target;
+
+ // Check that when we seek nothing happens.
+ var t = v.currentTime;
+ v.currentTime = v.currentTime /= 2;
+ ok(Math.abs(t - v.currentTime) < 0.01, "Current time shouldn't change when seeking in unseekable media: " + v.name);
+
+ // Check that the seekable ranges are empty.
+ is(v.seekable.length, 0, "Should have no seekable ranges in unseekable media: " + v.name);
+};
+
+var onended = function(event) {
+ var v = event.target;
+
+ // Remove the event listeners so that they can't run if there are any pending
+ // events.
+ v.removeEventListener("seeking", onseeking);
+ v.removeEventListener("seeked", onseeked);
+ v.removeEventListener("timeupdate", ontimeupdate);
+ v.removeEventListener("ended", onended);
+
+ v.src = "";
+ if (v.parentNode) {
+ v.remove();
+ }
+
+ // Verify that none of the seeks we did in timeupdate actually seeked.
+ ok(!v.actuallySeeked, "Should not be able to seek in unseekable media: " + v.name);
+
+ manager.finished(v.token);
+}
+
+function startTest(test, token) {
+ var v = document.createElement('video');
+ manager.started(token);
+ v.name = test.name;
+ v.src = test.name;
+ v.token = token;
+ v.autoplay = "true";
+
+ v.actuallySeeked = false;
+
+ v.addEventListener("seeking", onseeking);
+ v.addEventListener("seeked", onseeked);
+ v.addEventListener("timeupdate", ontimeupdate);
+ v.addEventListener("ended", onended);
+
+ document.body.appendChild(v);
+}
+
+function canPlay(candidates) {
+ var v = document.createElement("video");
+ var resources = candidates.filter(function(x){return v.canPlayType(x.type);});
+ return (resources.length);
+}
+
+if (canPlay(gUnseekableTests)) {
+ manager.runTests(gUnseekableTests, startTest);
+} else {
+ todo(false, "No files of supported format to test");
+}
+
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_videoDocumentTitle.html b/dom/media/test/test_videoDocumentTitle.html
new file mode 100644
index 0000000000..dd52dba26c
--- /dev/null
+++ b/dom/media/test/test_videoDocumentTitle.html
@@ -0,0 +1,57 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=463830
+-->
+<head>
+ <title>Test for Bug 463830</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=463830">Mozilla Bug 463830</a>
+<p id="display"></p>
+<iframe id="i"></iframe>
+<div id="content" style="display: none">
+
+</div>
+<pre id="test">
+<script type="application/javascript">
+
+/** Test for Bug 463830 **/
+
+var gTests = [
+ { file: "320x240.ogv", title: "320x240.ogv" },
+ { file: "bug461281.ogg", title: "bug461281.ogg" },
+];
+
+var gTestNum = 0;
+
+addLoadEvent(runTest);
+
+var title;
+var i = document.getElementById("i");
+
+function runTest() {
+ if (gTestNum == gTests.length) {
+ SimpleTest.finish();
+ return;
+ }
+ if (gTestNum == 0) {
+ i.addEventListener("load", function() {
+ is(i.contentDocument.title, title, "Doc title incorrect");
+ setTimeout(runTest, 0);
+ });
+ }
+
+ title = gTests[gTestNum].title;
+ i.src = gTests[gTestNum].file;
+ gTestNum++;
+}
+
+SimpleTest.waitForExplicitFinish();
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_videoPlaybackQuality_totalFrames.html b/dom/media/test/test_videoPlaybackQuality_totalFrames.html
new file mode 100644
index 0000000000..1b69a3b64f
--- /dev/null
+++ b/dom/media/test/test_videoPlaybackQuality_totalFrames.html
@@ -0,0 +1,46 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Count the tatol frames of a video</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var manager = new MediaTestManager;
+
+var startTest = function(test, token) {
+ manager.started(token);
+ var v = document.createElement('video');
+ v.token = token;
+ document.body.appendChild(v);
+ v.src = test.name;
+
+ function ended(event) {
+ var video = event.target;
+ is(video.getVideoPlaybackQuality().totalVideoFrames, test.totalFrameCount,test.name+ " totalFrames should match!");
+ removeNodeAndSource(video);
+ manager.finished(video.token);
+ }
+ v.addEventListener("ended", ended);
+ v.play();
+};
+
+SimpleTest.waitForExplicitFinish();
+SpecialPowers.pushPrefEnv(
+ {
+ "set": [
+ ["media.decoder.skip-to-next-key-frame.enabled", false],
+ ["media.av1.use-dav1d", true]
+ ]
+ },
+ function() {
+ manager.runTests(getPlayableVideos(gFrameCountTests), startTest);
+ });
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_video_dimensions.html b/dom/media/test/test_video_dimensions.html
new file mode 100644
index 0000000000..4d9c2a185f
--- /dev/null
+++ b/dom/media/test/test_video_dimensions.html
@@ -0,0 +1,96 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test that a video element has set video dimensions on loadedmetadata</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+var manager = new MediaTestManager;
+
+var startTest = function(test, token) {
+ manager.started(token);
+ var v1 = document.createElement('video');
+ var v2 = document.createElement('video');
+ var vout = document.createElement('video');
+
+ // Avoid a race for hardware resources between v1 and v2 on platforms with
+ // a hardware decoder, like B2G.
+ v1.preload = 'none';
+ v2.preload = 'none';
+
+ var numVideoElementsFinished = 0;
+
+ var ondurationchange = function(ev) {
+ var v = ev.target;
+ info(v.testName + " got durationchange");
+ v.durationchange = true;
+ };
+ var onresize = function(ev) {
+ var v = ev.target;
+ info(v.testName + " got resize");
+ ok(!v.resize, v.testName + " should only fire resize once for same size");
+ v.resize = true;
+ ok(v.durationchange, v.testName +
+ " durationchange event should have been emitted before resize");
+ is(v.videoWidth, test.width, v.testName + " width should be set on resize");
+ is(v.videoHeight, test.height, v.testName + " height should be set on resize");
+ };
+ var onloadedmetadata = function(ev) {
+ var v = ev.target;
+ info(v.testName + " got loadedmetadata");
+ ok(!v.loadedmetadata, v.testName + " should only fire loadedmetadata once");
+ v.loadedmetadata = true;
+ ok(v.resize, v.testName +
+ " resize event should have been emitted before loadedmetadata");
+
+ numVideoElementsFinished += 1;
+ if (v === v1) {
+ removeNodeAndSource(v1);
+ v2.load();
+ }
+
+ if (v === v2) {
+ vout.srcObject = v2.mozCaptureStreamUntilEnded();
+ v2.play();
+ vout.play();
+ }
+
+ if (numVideoElementsFinished === 3) {
+ removeNodeAndSource(v2);
+ removeNodeAndSource(vout);
+ manager.finished(token);
+ }
+ };
+ var setupElement = function(v, id) {
+ v.durationchange = false;
+ v.ondurationchange = ondurationchange;
+ v.resize = false;
+ v.onresize = onresize;
+ v.loadedmetadata = false;
+ v.onloadedmetadata = onloadedmetadata;
+ document.body.appendChild(v);
+ };
+
+ v1.testName = test.name;
+ v2.testName = test.name + " (Captured)";
+ vout.testName = test.name + " (Stream)";
+
+ v1.src = test.name;
+ v2.src = test.name;
+
+ setupElement(v1, "v1");
+ setupElement(v2, "v2");
+ setupElement(vout, "vout");
+
+ v1.play();
+};
+
+manager.runTests(getPlayableVideos(gSmallTests), startTest);
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_video_gzip_encoding.html b/dom/media/test/test_video_gzip_encoding.html
new file mode 100644
index 0000000000..355a245713
--- /dev/null
+++ b/dom/media/test/test_video_gzip_encoding.html
@@ -0,0 +1,25 @@
+<!DOCTYPE HTML>
+<html>
+ <head>
+ <title>Bug 1370177 gzipped mp4 with Content-Length</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+ </head>
+ <body>
+ <!--
+ Tests that an MP4 file served over a "Content-Encoding: gzip"
+ HTTP channel with a "Content-Length" header set to the length
+ of the compressed file works.
+ -->
+ <video id='v' src="http://mochi.test:8888/tests/dom/media/test/gzipped_mp4.sjs" controls autoplay></video>
+ <script>
+ SimpleTest.waitForExplicitFinish();
+ var v = document.getElementById('v');
+ v.addEventListener("ended", ()=>{
+ SimpleTest.finish();
+ mediaTestCleanup();
+ });
+ </script>
+ </body>
+</html>
diff --git a/dom/media/test/test_video_in_audio_element.html b/dom/media/test/test_video_in_audio_element.html
new file mode 100644
index 0000000000..a53adf9414
--- /dev/null
+++ b/dom/media/test/test_video_in_audio_element.html
@@ -0,0 +1,68 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=1060896
+-->
+<head>
+ <meta charset="utf-8">
+ <title>Test for Bug 1060896</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+ <script type="application/javascript">
+
+ /**
+ * Test for Bug 1060896; tests that loading a video inside an audio element works.
+ **/
+
+ var manager = new MediaTestManager;
+
+ function error(event) {
+ var a = event.target;
+ ok(!a.mozHasAudio, "Media must've had no active tracks to play");
+ a.removeEventListener("error", error);
+ a.removeEventListener("ended", ended);
+ removeNodeAndSource(a);
+ manager.finished(a.token);
+ }
+
+ function ended(event) {
+ var a = event.target;
+ a.removeEventListener("error", error);
+ a.removeEventListener("ended", ended);
+ removeNodeAndSource(a);
+ manager.finished(a.token);
+ }
+
+ function initTest(test, token) {
+ var a = document.createElement('audio');
+ a.token = token;
+ manager.started(token);
+ a.autoplay = true;
+
+ a.addEventListener("error", error);
+ a.addEventListener("ended", ended);
+
+ a.src = test.name;
+ }
+
+ var videos = getPlayableVideos(gSmallTests);
+ // Bug 1216012, skip the test on emulator-kk.
+ if (getAndroidVersion() == 19) {
+ todo(false, "Test disabled on emulator-kk.");
+ } else {
+ manager.runTests(videos, initTest);
+ }
+
+ </script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=1060896">Mozilla Bug 1060896</a>
+<p id="display"></p>
+<div id="content" style="display: none">
+
+</div>
+<pre id="test">
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_video_low_power_telemetry.html b/dom/media/test/test_video_low_power_telemetry.html
new file mode 100644
index 0000000000..be609f7ceb
--- /dev/null
+++ b/dom/media/test/test_video_low_power_telemetry.html
@@ -0,0 +1,205 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=1737682
+
+This is a test of the macOS video low power telemetry, defined as GFX_MACOS_VIDEO_LOW_POWER
+enums in Histograms.json. The test will load some video media files, play them for some
+number of frames, then check that appropriate telemetry has been recorded. Since the
+telemetry fires based on the number of frames painted, the test is structured around
+the `media.video_stats.enabled` pref, which allows us to monitor the number of frames
+that have been shown. The telemetry fires every 600 frames. To make sure we have enough
+painted frames to trigger telemetry, we run the media for many frames and then check
+that the expected telemetry value has been recorded at least once.
+
+The tests are run via the MediaTestManager, which loads and plays the video sequentially.
+Since we want to test different telemetry outcomes, we have some special setup and teardown
+steps we run before each video. We keep track of which test we are running with a simple
+1-based index, `testIndex`. So our test structure is:
+
+1) Set some initial preferences that need to be set for the whole test series.
+2) Start MediaTestManager with a set of media to play.
+3) When a video arrives, increment testIndex and call preTest.
+4) Run the video for PAINTED_FRAMES frames, then call postTest, which checks telemetry
+ and cleans up.
+5) Tell the MediaTestManager we've finished with that video, which triggers the next.
+-->
+
+<head>
+ <title>Test of macOS video low power telemetry</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=1737682">Mozilla Bug 1737682</a>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+ // Parallel test must be disabled for media.video_stats.enabled is a global setting
+ // to prevent the setting from changing unexpectedly in the middle of the test.
+ PARALLEL_TESTS = 1;
+ SimpleTest.waitForExplicitFinish();
+
+ // How many frames do we run? Must be more than the value set by the
+ // gfx.core-animation.low-power-telemetry-frames pref, because that's how
+ // many frames must be shown before telemetry is emitted. We present for
+ // longer than that to ensure that the telemetry is emitted at least once.
+ // This also has to be enough frames to overcome the animated "<video> is
+ // now fullscreen." popup that appears over the video. While that modal
+ // is visible and animating away, it will register as an overlay and
+ // prevent low power mode. As an estimate, we need at least 200 frames to
+ // ensure that animation has finished and there has been time to fire the
+ // telemetry at least once after it has finished.
+ const TELEMETRY_FRAMES = SpecialPowers.getIntPref("gfx.core-animation.low-power-telemetry-frames");
+ const PAINTED_FRAMES = Math.max(Math.floor(TELEMETRY_FRAMES * 1.1), 200 + TELEMETRY_FRAMES);
+ info(`Running each video for ${PAINTED_FRAMES} frames.`);
+
+ // Taken from TelemetryHistogramEnums.h
+ const GFX_MACOS_VIDEO_LOW_POWER_LowPower = 1;
+ const GFX_MACOS_VIDEO_LOW_POWER_FailWindowed = 3;
+
+ var manager = new MediaTestManager;
+
+ // Define some state variables that we'll use to manage multiple setup,
+ // check, and teardown steps using the same preTest and postTest functions.
+ var testIndex = 0;
+ var testsComplete = false;
+
+ async function retrieveSnapshotAndClearTelemetry() {
+ return SpecialPowers.spawnChrome([], () => {
+ let hist = Services.telemetry.getHistogramById("GFX_MACOS_VIDEO_LOW_POWER");
+ let snap = hist.snapshot();
+ hist.clear();
+ return snap;
+ });
+ }
+
+ function checkSnapshot(snap, category) {
+ if (!snap) {
+ return;
+ }
+
+ // For debugging purposes, build a string of the snapshot values hashmap.
+ let valuesString = '';
+ let keys = Object.keys(snap.values);
+ keys.forEach(k => {
+ valuesString += `[${k}] = ${snap.values[k]}, `;
+ });
+ info(`Test ${testIndex} telemetry values are ${valuesString}`);
+
+ // Since we've run the media for more frames than the telemetry needs to
+ // fire, we should have at least 1 of the expected value.
+ let val = snap.values[category];
+ if (!val) {
+ val = 0;
+ }
+ ok(val > 0, `Test ${testIndex} enum ${category} should have at least 1 occurrence; found ${val}.`);
+ }
+
+ async function doPreTest(v) {
+ if (testsComplete) {
+ manager.finished(v.token);
+ return;
+ }
+
+ switch (testIndex) {
+ case 1: {
+ // Test 1 (FailWindowed): No special setup.
+ break;
+ }
+
+ case 2: {
+ // Test 2 (LowPower): Enter fullscreen.
+
+ // Wait one frame to ensure the old video is no longer in the layer tree, which
+ // causes problems with the telemetry.
+ await new Promise(resolve => requestAnimationFrame(resolve));
+
+ info("Attempting to enter fullscreen.");
+ ok(document.fullscreenEnabled, "Document should permit fullscreen-ing of elements.");
+ await SpecialPowers.wrap(v).requestFullscreen();
+ ok(document.fullscreenElement, "Document should have one element in fullscreen.");
+ break;
+ }
+ }
+ }
+
+ async function doPostTest(v) {
+ info(`Test ${testIndex} attempting to retrieve telemetry.`);
+ let snap = await retrieveSnapshotAndClearTelemetry();
+ ok(snap, `Test ${testIndex} should have telemetry.`);
+
+ switch (testIndex) {
+ case 1: {
+ // Test 1 (FailWindowed): Just check.
+ checkSnapshot(snap, GFX_MACOS_VIDEO_LOW_POWER_FailWindowed);
+ break;
+ }
+
+ case 2: {
+ // Test 2 (LowPower): Check, then exit fullscreen.
+ checkSnapshot(snap, GFX_MACOS_VIDEO_LOW_POWER_LowPower);
+
+ info("Attempting to exit fullscreen.");
+ await SpecialPowers.wrap(document).exitFullscreen();
+ ok(!document.fullscreenElement, "Document should be out of fullscreen.");
+
+ // This is the last test.
+ testsComplete = true;
+ break;
+ }
+ }
+ }
+
+ function ontimeupdate(event) {
+ let v = event.target;
+ // Count painted frames to see when we should hit some telemetry threshholds.
+ if (v.mozPaintedFrames >= PAINTED_FRAMES) {
+ v.pause();
+ v.removeEventListener("timeupdate", ontimeupdate);
+
+ doPostTest(v).then(() => {
+ let token = v.token;
+ removeNodeAndSource(v);
+ manager.finished(token);
+ });
+ }
+ }
+
+ function startTest(test, token) {
+ manager.started(token);
+
+ testIndex++;
+ info(`Starting test ${testIndex} video ${test.name}.`);
+
+ let v = document.createElement('video');
+ v.addEventListener("timeupdate", ontimeupdate);
+ v.token = token;
+ v.src = test.name;
+ v.loop = true;
+ document.body.appendChild(v);
+
+ doPreTest(v).then(() => {
+ info(`Playing test ${testIndex}.`);
+ v.play();
+ });
+ }
+
+ SpecialPowers.pushPrefEnv({"set": [
+ ["media.video_stats.enabled", true],
+ ["gfx.core-animation.specialize-video", true],
+ ]},
+ async function() {
+ // Clear out existing telemetry in case previous tests were displaying
+ // video.
+ info("Clearing initial telemetry.");
+ await retrieveSnapshotAndClearTelemetry();
+
+ ok(gVideoLowPowerTests.length >= 2, "Should have enough videos to test.");
+ manager.runTests(gVideoLowPowerTests, startTest);
+ });
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_video_stats_resistfingerprinting.html b/dom/media/test/test_video_stats_resistfingerprinting.html
new file mode 100644
index 0000000000..2bc239b367
--- /dev/null
+++ b/dom/media/test/test_video_stats_resistfingerprinting.html
@@ -0,0 +1,90 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+Mozilla Bug:
+https://bugzilla.mozilla.org/show_bug.cgi?id=1369309
+Tor Ticket:
+https://trac.torproject.org/projects/tor/ticket/15757
+-->
+<head>
+ <meta charset="utf-8">
+ <title>Test for Bug 1369309</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <script type="application/javascript" src="manifest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
+</head>
+<body>
+<a target="_blank" href="https://bugzilla.mozilla.org/show_bug.cgi?id=682299">Mozilla Bug 1369309</a>
+<a target="_blank" href="https://trac.torproject.org/projects/tor/ticket/15757">Tor Ticket 15757</a>
+
+<!-- The main testing script -->
+<script class="testbody" type="text/javascript">
+ var manager = new MediaTestManager;
+ const SPOOFED_FRAMES_PER_SECOND = 30;
+ const SPOOFED_DROPPED_RATIO = 0.05;
+ const MS_PER_TIME_ATOM = 100; // Not the default anymore, but what we test here.
+ // Push the setting of 'privacy.resistFingerprinting' into gTestPrefs, which
+ // will be set during MediaTestManager.runTests().
+ gTestPrefs.push(
+ ["privacy.resistFingerprinting", true],
+ ["privacy.resistFingerprinting.reduceTimerPrecision.microseconds", MS_PER_TIME_ATOM * 1000],
+ ["privacy.resistFingerprinting.reduceTimerPrecision.jitter", false],
+ // We use 240p as the target resolution since 480p is greater than every video
+ // source in our test suite, so we need to use 240p here for allowing us to
+ // test dropped rate here.
+ ["privacy.resistFingerprinting.target_video_res", 240]
+ );
+ var testCases = [
+ { name:"320x240.ogv", type:"video/ogg", width:320, height:240, duration:0.266, drop: false },
+ { name:"seek.webm", type:"video/webm", width:320, height:240, duration:3.966, drop: false },
+ { name:"gizmo.mp4", type:"video/mp4", width:560, height:320, duration:5.56, drop: true }
+ ];
+
+ function checkStats(v, shouldDrop) {
+ // Rounding the current time to 100ms.
+ const secondsPerAtom = MS_PER_TIME_ATOM / 1000;
+ const currentTimeAtoms = Math.floor(v.currentTime / secondsPerAtom);
+ const currentTime = currentTimeAtoms * secondsPerAtom;
+ let dropRate = 0;
+
+ if (shouldDrop) {
+ dropRate = SPOOFED_DROPPED_RATIO;
+ }
+
+ is(v.mozParsedFrames, parseInt(currentTime * SPOOFED_FRAMES_PER_SECOND, 10),
+ "mozParsedFrames should be spoofed if fingerprinting resistance is enabled");
+ is(v.mozDecodedFrames, parseInt(currentTime * SPOOFED_FRAMES_PER_SECOND, 10),
+ "mozDecodedFrames should be spoofed if fingerprinting resistance is enabled");
+ is(v.mozPresentedFrames,
+ parseInt(currentTime * SPOOFED_FRAMES_PER_SECOND * (1 - dropRate), 10),
+ "mozPresentedFrames should be spoofed if fingerprinting resistance is enabled");
+ is(v.mozPaintedFrames,
+ parseInt(currentTime * SPOOFED_FRAMES_PER_SECOND * (1 - dropRate), 10),
+ "mozPaintedFrames should be spoofed if fingerprinting resistance is enabled");
+ is(v.mozFrameDelay, 0.0,
+ "mozFrameDelay should be 0.0 if fingerprinting resistance is enabled");
+ let playbackQuality = v.getVideoPlaybackQuality();
+ is(playbackQuality.totalVideoFrames, parseInt(currentTime * SPOOFED_FRAMES_PER_SECOND, 10),
+ "VideoPlaybackQuality.totalVideoFrames should be spoofed if fingerprinting resistance is enabled");
+ is(playbackQuality.droppedVideoFrames, parseInt(currentTime * SPOOFED_FRAMES_PER_SECOND * dropRate, 10),
+ "VideoPlaybackQuality.droppedVideoFrames should be spoofed if fingerprinting resistance is enabled");
+ }
+
+ function startTest(test, token) {
+ let v = document.createElement("video");
+ v.token = token;
+ v.src = test.name;
+ manager.started(token);
+ once(v, "ended", () => {
+ checkStats(v, test.drop);
+ removeNodeAndSource(v);
+ manager.finished(v.token);
+ });
+ v.play();
+ }
+
+ manager.runTests(testCases, startTest);
+
+</script>
+</body>
+</html>
diff --git a/dom/media/test/test_video_to_canvas.html b/dom/media/test/test_video_to_canvas.html
new file mode 100644
index 0000000000..3267dc63f5
--- /dev/null
+++ b/dom/media/test/test_video_to_canvas.html
@@ -0,0 +1,68 @@
+<!DOCTYPE HTML>
+<html>
+<!--
+https://bugzilla.mozilla.org/show_bug.cgi?id=486646
+-->
+
+<head>
+ <title>Test for Bug 486646</title>
+ <script type="text/javascript" src="/MochiKit/MochiKit.js"></script>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+ <script type="text/javascript" src="manifest.js"></script>
+</head>
+<body>
+<script class="testbody" type="text/javascript">
+
+var manager = new MediaTestManager;
+
+function loaded(e) {
+ var v = e.target;
+ ok(v.readyState >= v.HAVE_CURRENT_DATA,
+ "readyState must be >= HAVE_CURRENT_DATA for " + v._name);
+
+ var canvas = document.createElement("canvas");
+ canvas.width = v.videoWidth;
+ canvas.height = v.videoHeight;
+ document.body.appendChild(canvas);
+ var ctx = canvas.getContext("2d");
+ try {
+ ctx.drawImage(v, 0, 0);
+ ok(true, "Shouldn't throw exception while drawing to canvas from video for " + v._name);
+ } catch (ex) {
+ ok(false, "Shouldn't throw exception while drawing to canvas from video for " + v._name);
+ }
+
+ v._finished = true;
+ v.remove();
+ manager.finished(v.token);
+}
+
+function startTest(test, token) {
+ var type = getMajorMimeType(test.type);
+ if (type != "video")
+ return;
+
+ var v = document.createElement('video');
+ v.token = token;
+ manager.started(token);
+ v.src = test.name;
+ v._name = test.name;
+ v._finished = false;
+ v.autoplay = true;
+ v.style.display = "none";
+ v.addEventListener("ended", loaded);
+ document.body.appendChild(v);
+}
+
+SimpleTest.waitForExplicitFinish();
+SpecialPowers.pushPrefEnv({"set": [["media.cache_size", 40000]]}, beginTest);
+function beginTest() {
+ manager.runTests(gSmallTests, startTest);
+}
+
+</script>
+</pre>
+
+</body>
+</html>
diff --git a/dom/media/test/test_volume.html b/dom/media/test/test_volume.html
new file mode 100644
index 0000000000..3d9e5bd91a
--- /dev/null
+++ b/dom/media/test/test_volume.html
@@ -0,0 +1,41 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Media test: volume attribute set</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+
+<video id='v1'></video><audio id='a1'></audio>
+
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+function test(element, value, shouldThrow) {
+ var threw = null;
+ try {
+ element.volume = value;
+ } catch (ex) {
+ threw = ex.name;
+ }
+ is(shouldThrow, threw, "Case: " +element.id+ " setVolume=" + value);
+}
+
+
+var ids = [document.getElementById('v1'), document.getElementById('a1')];
+
+for (let i=0; i<ids.length; i++) {
+ var element = ids[i];
+ test(element, 0.0, null);
+ test(element, 1.0, null);
+ test(element, -0.1, "IndexSizeError");
+ test(element, 1.1, "IndexSizeError");
+ test(element, undefined, "TypeError");
+ test(element, NaN, "TypeError");
+}
+
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_vp9_superframes.html b/dom/media/test/test_vp9_superframes.html
new file mode 100644
index 0000000000..c570561b31
--- /dev/null
+++ b/dom/media/test/test_vp9_superframes.html
@@ -0,0 +1,31 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Test that all VP9 frames are decoded (contains superframes)</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+
+SimpleTest.waitForExplicitFinish();
+
+function test() {
+ var video = document.createElement("video");
+ video.src = "vp9-superframes.webm";
+ video.play();
+ video.addEventListener("ended", function () {
+ let vpq = video.getVideoPlaybackQuality();
+ is(vpq.totalVideoFrames, 120, "totalVideoFrames must contains 120 frames");
+ SimpleTest.finish();
+ });
+}
+
+addLoadEvent(function() {
+ test();
+});
+</script>
+</pre>
+</body>
+</html>
diff --git a/dom/media/test/test_wav_ended1.html b/dom/media/test/test_wav_ended1.html
new file mode 100644
index 0000000000..99f6e38243
--- /dev/null
+++ b/dom/media/test/test_wav_ended1.html
@@ -0,0 +1,43 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Wave Media test: ended</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<pre id="test">
+<script class="testbody" type="text/javascript">
+// Test if the ended event works correctly.
+var endPassed = false;
+var completed = false;
+
+function startTest() {
+ if (completed)
+ return;
+ var v = document.getElementById('v');
+ v.play();
+}
+
+function playbackEnded() {
+ if (completed)
+ return;
+
+ var v = document.getElementById('v');
+ completed = true;
+ ok(v.currentTime >= 0.9 && v.currentTime <= 1.1,
+ "Checking currentTime at end: " + v.currentTime);
+ ok(v.ended, "Checking playback has ended");
+ SimpleTest.finish();
+}
+
+SimpleTest.waitForExplicitFinish();
+</script>
+</pre>
+<audio id='v'
+ onloadedmetadata='return startTest();'
+ onended='return playbackEnded();'>
+ <source type='audio/x-wav' src='r11025_s16_c1.wav'>
+</audio>
+</body>
+</html>
diff --git a/dom/media/test/test_wav_ended2.html b/dom/media/test/test_wav_ended2.html
new file mode 100644
index 0000000000..0c172139f0
--- /dev/null
+++ b/dom/media/test/test_wav_ended2.html
@@ -0,0 +1,62 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+ <title>Wave Media test: ended and replaying</title>
+ <script src="/tests/SimpleTest/SimpleTest.js"></script>
+ <link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
+</head>
+<body>
+<!-- try with autoplay and no v.play in starttest, also with both -->
+<pre id="test">
+<script class="testbody" type="text/javascript">
+// Test if audio can be replayed after ended.
+var completed = false;
+var playingCount = 0;
+var endCount = 0;
+
+function startTest() {
+ if (completed)
+ return;
+
+ var v = document.getElementById('v');
+ v.play();
+}
+
+function playbackStarted() {
+ if (completed)
+ return;
+
+ playingCount++;
+}
+
+function playbackEnded() {
+ if (completed)
+ return;
+
+ endCount++;
+ var v = document.getElementById('v');
+ ok(v.currentTime >= 0.9 && v.currentTime <= 1.1,
+ "Checking currentTime at end: " + v.currentTime);
+ ok(v.ended, "Checking playback has ended");
+ ok(playingCount > 0, "Expect at least one playing event");
+ playingCount = 0;
+ if (endCount < 2) {
+ v.play();
+ } else {
+ ok(endCount == 2, "Check playback after ended event");
+ completed = true;
+ SimpleTest.finish();
+ }
+}
+
+SimpleTest.waitForExplicitFinish();
+</script>
+</pre>
+<audio id='v'
+ onloadedmetadata='return startTest();'
+ onplaying='return playbackStarted();'
+ onended='return playbackEnded();'>
+ <source type='audio/x-wav' src='r11025_s16_c1.wav'>
+</audio>
+</body>
+</html>
diff --git a/dom/media/test/tone2s-silence4s-tone2s.opus b/dom/media/test/tone2s-silence4s-tone2s.opus
new file mode 100644
index 0000000000..e7d754926e
--- /dev/null
+++ b/dom/media/test/tone2s-silence4s-tone2s.opus
Binary files differ
diff --git a/dom/media/test/two-xing-header-no-content-length.mp3 b/dom/media/test/two-xing-header-no-content-length.mp3
new file mode 100644
index 0000000000..4d139e2886
--- /dev/null
+++ b/dom/media/test/two-xing-header-no-content-length.mp3
Binary files differ
diff --git a/dom/media/test/two-xing-header-no-content-length.mp3^headers^ b/dom/media/test/two-xing-header-no-content-length.mp3^headers^
new file mode 100644
index 0000000000..abfeb4ce28
--- /dev/null
+++ b/dom/media/test/two-xing-header-no-content-length.mp3^headers^
@@ -0,0 +1,3 @@
+HTTP 200 OK
+Content-Length: invalid
+Cache-Control: no-store
diff --git a/dom/media/test/variable-channel.ogg b/dom/media/test/variable-channel.ogg
new file mode 100644
index 0000000000..77e116889c
--- /dev/null
+++ b/dom/media/test/variable-channel.ogg
Binary files differ
diff --git a/dom/media/test/variable-channel.ogg^headers^ b/dom/media/test/variable-channel.ogg^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/variable-channel.ogg^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/variable-channel.opus b/dom/media/test/variable-channel.opus
new file mode 100644
index 0000000000..76b8991167
--- /dev/null
+++ b/dom/media/test/variable-channel.opus
Binary files differ
diff --git a/dom/media/test/variable-channel.opus^headers^ b/dom/media/test/variable-channel.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/variable-channel.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/variable-preskip.opus b/dom/media/test/variable-preskip.opus
new file mode 100644
index 0000000000..a82e831ce6
--- /dev/null
+++ b/dom/media/test/variable-preskip.opus
Binary files differ
diff --git a/dom/media/test/variable-preskip.opus^headers^ b/dom/media/test/variable-preskip.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/variable-preskip.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/variable-samplerate.ogg b/dom/media/test/variable-samplerate.ogg
new file mode 100644
index 0000000000..0c2726e835
--- /dev/null
+++ b/dom/media/test/variable-samplerate.ogg
Binary files differ
diff --git a/dom/media/test/variable-samplerate.ogg^headers^ b/dom/media/test/variable-samplerate.ogg^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/variable-samplerate.ogg^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/variable-samplerate.opus b/dom/media/test/variable-samplerate.opus
new file mode 100644
index 0000000000..1d15170798
--- /dev/null
+++ b/dom/media/test/variable-samplerate.opus
Binary files differ
diff --git a/dom/media/test/variable-samplerate.opus^headers^ b/dom/media/test/variable-samplerate.opus^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/variable-samplerate.opus^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/vbr-head.mp3 b/dom/media/test/vbr-head.mp3
new file mode 100644
index 0000000000..35f4105491
--- /dev/null
+++ b/dom/media/test/vbr-head.mp3
Binary files differ
diff --git a/dom/media/test/vbr-head.mp3^headers^ b/dom/media/test/vbr-head.mp3^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/vbr-head.mp3^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/vbr.mp3 b/dom/media/test/vbr.mp3
new file mode 100644
index 0000000000..38eb376a97
--- /dev/null
+++ b/dom/media/test/vbr.mp3
Binary files differ
diff --git a/dom/media/test/vbr.mp3^headers^ b/dom/media/test/vbr.mp3^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/vbr.mp3^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/very-short.mp3 b/dom/media/test/very-short.mp3
new file mode 100644
index 0000000000..7dbdb20c3a
--- /dev/null
+++ b/dom/media/test/very-short.mp3
Binary files differ
diff --git a/dom/media/test/video-overhang.ogg b/dom/media/test/video-overhang.ogg
new file mode 100644
index 0000000000..e11b28fb5b
--- /dev/null
+++ b/dom/media/test/video-overhang.ogg
Binary files differ
diff --git a/dom/media/test/video-overhang.ogg^headers^ b/dom/media/test/video-overhang.ogg^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/video-overhang.ogg^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/vp9-short.webm b/dom/media/test/vp9-short.webm
new file mode 100644
index 0000000000..16d32abee3
--- /dev/null
+++ b/dom/media/test/vp9-short.webm
Binary files differ
diff --git a/dom/media/test/vp9-short.webm^headers^ b/dom/media/test/vp9-short.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/vp9-short.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/vp9-superframes.webm b/dom/media/test/vp9-superframes.webm
new file mode 100644
index 0000000000..d695e42357
--- /dev/null
+++ b/dom/media/test/vp9-superframes.webm
Binary files differ
diff --git a/dom/media/test/vp9-superframes.webm^headers^ b/dom/media/test/vp9-superframes.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/vp9-superframes.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/vp9.webm b/dom/media/test/vp9.webm
new file mode 100644
index 0000000000..221877e303
--- /dev/null
+++ b/dom/media/test/vp9.webm
Binary files differ
diff --git a/dom/media/test/vp9.webm^headers^ b/dom/media/test/vp9.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/vp9.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/vp9cake-short.webm b/dom/media/test/vp9cake-short.webm
new file mode 100644
index 0000000000..2d353d98a7
--- /dev/null
+++ b/dom/media/test/vp9cake-short.webm
Binary files differ
diff --git a/dom/media/test/vp9cake-short.webm^headers^ b/dom/media/test/vp9cake-short.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/vp9cake-short.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/vp9cake.webm b/dom/media/test/vp9cake.webm
new file mode 100644
index 0000000000..4ea70ed302
--- /dev/null
+++ b/dom/media/test/vp9cake.webm
Binary files differ
diff --git a/dom/media/test/vp9cake.webm^headers^ b/dom/media/test/vp9cake.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/vp9cake.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/wave_metadata.wav b/dom/media/test/wave_metadata.wav
new file mode 100644
index 0000000000..5e17547c30
--- /dev/null
+++ b/dom/media/test/wave_metadata.wav
Binary files differ
diff --git a/dom/media/test/wave_metadata.wav^headers^ b/dom/media/test/wave_metadata.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/wave_metadata.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/wave_metadata_bad_len.wav b/dom/media/test/wave_metadata_bad_len.wav
new file mode 100644
index 0000000000..b89c4818be
--- /dev/null
+++ b/dom/media/test/wave_metadata_bad_len.wav
Binary files differ
diff --git a/dom/media/test/wave_metadata_bad_len.wav^headers^ b/dom/media/test/wave_metadata_bad_len.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/wave_metadata_bad_len.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/wave_metadata_bad_no_null.wav b/dom/media/test/wave_metadata_bad_no_null.wav
new file mode 100644
index 0000000000..18063048c3
--- /dev/null
+++ b/dom/media/test/wave_metadata_bad_no_null.wav
Binary files differ
diff --git a/dom/media/test/wave_metadata_bad_no_null.wav^headers^ b/dom/media/test/wave_metadata_bad_no_null.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/wave_metadata_bad_no_null.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/wave_metadata_bad_utf8.wav b/dom/media/test/wave_metadata_bad_utf8.wav
new file mode 100644
index 0000000000..b6f2a675b2
--- /dev/null
+++ b/dom/media/test/wave_metadata_bad_utf8.wav
Binary files differ
diff --git a/dom/media/test/wave_metadata_bad_utf8.wav^headers^ b/dom/media/test/wave_metadata_bad_utf8.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/wave_metadata_bad_utf8.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/wave_metadata_unknown_tag.wav b/dom/media/test/wave_metadata_unknown_tag.wav
new file mode 100644
index 0000000000..b19fb5170f
--- /dev/null
+++ b/dom/media/test/wave_metadata_unknown_tag.wav
Binary files differ
diff --git a/dom/media/test/wave_metadata_unknown_tag.wav^headers^ b/dom/media/test/wave_metadata_unknown_tag.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/wave_metadata_unknown_tag.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/wave_metadata_utf8.wav b/dom/media/test/wave_metadata_utf8.wav
new file mode 100644
index 0000000000..352db285bb
--- /dev/null
+++ b/dom/media/test/wave_metadata_utf8.wav
Binary files differ
diff --git a/dom/media/test/wave_metadata_utf8.wav^headers^ b/dom/media/test/wave_metadata_utf8.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/wave_metadata_utf8.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/wavedata_alaw.wav b/dom/media/test/wavedata_alaw.wav
new file mode 100644
index 0000000000..ef090d16e0
--- /dev/null
+++ b/dom/media/test/wavedata_alaw.wav
Binary files differ
diff --git a/dom/media/test/wavedata_alaw.wav^headers^ b/dom/media/test/wavedata_alaw.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/wavedata_alaw.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/wavedata_float.wav b/dom/media/test/wavedata_float.wav
new file mode 100644
index 0000000000..155f891d51
--- /dev/null
+++ b/dom/media/test/wavedata_float.wav
Binary files differ
diff --git a/dom/media/test/wavedata_float.wav^headers^ b/dom/media/test/wavedata_float.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/wavedata_float.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/wavedata_s16.wav b/dom/media/test/wavedata_s16.wav
new file mode 100644
index 0000000000..6a69cd78f6
--- /dev/null
+++ b/dom/media/test/wavedata_s16.wav
Binary files differ
diff --git a/dom/media/test/wavedata_s16.wav^headers^ b/dom/media/test/wavedata_s16.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/wavedata_s16.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/wavedata_s24.wav b/dom/media/test/wavedata_s24.wav
new file mode 100644
index 0000000000..dbdb6aac1e
--- /dev/null
+++ b/dom/media/test/wavedata_s24.wav
Binary files differ
diff --git a/dom/media/test/wavedata_s24.wav^headers^ b/dom/media/test/wavedata_s24.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/wavedata_s24.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/wavedata_u8.wav b/dom/media/test/wavedata_u8.wav
new file mode 100644
index 0000000000..1d895c2ce0
--- /dev/null
+++ b/dom/media/test/wavedata_u8.wav
Binary files differ
diff --git a/dom/media/test/wavedata_u8.wav^headers^ b/dom/media/test/wavedata_u8.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/wavedata_u8.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/wavedata_ulaw.wav b/dom/media/test/wavedata_ulaw.wav
new file mode 100644
index 0000000000..0874face21
--- /dev/null
+++ b/dom/media/test/wavedata_ulaw.wav
Binary files differ
diff --git a/dom/media/test/wavedata_ulaw.wav^headers^ b/dom/media/test/wavedata_ulaw.wav^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/wavedata_ulaw.wav^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/white-3s-black-1s.webm b/dom/media/test/white-3s-black-1s.webm
new file mode 100644
index 0000000000..03dd1dcbda
--- /dev/null
+++ b/dom/media/test/white-3s-black-1s.webm
Binary files differ
diff --git a/dom/media/test/white-3s-black-1s.webm^headers^ b/dom/media/test/white-3s-black-1s.webm^headers^
new file mode 100644
index 0000000000..4030ea1d3d
--- /dev/null
+++ b/dom/media/test/white-3s-black-1s.webm^headers^
@@ -0,0 +1 @@
+Cache-Control: no-store
diff --git a/dom/media/test/white-short.webm b/dom/media/test/white-short.webm
new file mode 100644
index 0000000000..0b718cdf20
--- /dev/null
+++ b/dom/media/test/white-short.webm
Binary files differ