| // Copyright 2013 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "content/browser/media/media_browsertest.h" |
| |
| #include <memory> |
| #include <string_view> |
| |
| #include "base/command_line.h" |
| #include "base/strings/string_number_conversions.h" |
| #include "base/strings/utf_string_conversions.h" |
| #include "build/build_config.h" |
| #include "content/public/browser/gpu_utils.h" |
| #include "content/public/browser/web_contents.h" |
| #include "content/public/common/content_features.h" |
| #include "content/public/common/content_switches.h" |
| #include "content/public/test/browser_test.h" |
| #include "content/public/test/browser_test_utils.h" |
| #include "content/public/test/content_browser_test_utils.h" |
| #include "content/shell/browser/shell.h" |
| #include "content/shell/common/shell_switches.h" |
| #include "media/audio/audio_features.h" |
| #include "media/base/media_switches.h" |
| #include "media/base/supported_types.h" |
| #include "media/base/test_data_util.h" |
| #include "media/media_buildflags.h" |
| #include "media/mojo/services/gpu_mojo_media_client_test_util.h" |
| #include "net/test/embedded_test_server/embedded_test_server.h" |
| #include "url/url_util.h" |
| |
| // Proprietary codecs require acceleration on Android. |
| #if BUILDFLAG(IS_ANDROID) && !BUILDFLAG(ENABLE_FFMPEG_VIDEO_DECODERS) |
| #define REQUIRE_ACCELERATION_ON_ANDROID() \ |
| if (!is_accelerated()) \ |
| return |
| #else |
| #define REQUIRE_ACCELERATION_ON_ANDROID() |
| #endif // BUILDFLAG(IS_ANDROID) |
| |
| namespace content { |
| |
| #if BUILDFLAG(IS_ANDROID) |
| // Title set by android cleaner page after short timeout. |
| const char16_t kClean[] = u"CLEAN"; |
| #endif |
| |
| void MediaBrowserTest::SetUpCommandLine(base::CommandLine* command_line) { |
| command_line->AppendSwitchASCII( |
| switches::kAutoplayPolicy, |
| switches::autoplay::kNoUserGestureRequiredPolicy); |
| command_line->AppendSwitch(switches::kExposeInternalsForTesting); |
| |
| std::vector<base::test::FeatureRef> enabled_features = { |
| #if BUILDFLAG(IS_ANDROID) |
| features::kLogJsConsoleMessages, |
| #endif |
| |
| #if BUILDFLAG(ENABLE_HLS_DEMUXER) && BUILDFLAG(USE_PROPRIETARY_CODECS) |
| media::kBuiltInHlsPlayer, |
| #endif |
| }; |
| |
| std::vector<base::test::FeatureRef> disabled_features = { |
| // Disable fallback after decode error to avoid unexpected test pass on |
| // the fallback path. |
| media::kFallbackAfterDecodeError, |
| |
| #if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) |
| // Disable out of process audio on Linux due to process spawn |
| // failures. http://crbug.com/986021 |
| features::kAudioServiceOutOfProcess, |
| #endif |
| }; |
| |
| scoped_feature_list_.InitWithFeatures(enabled_features, disabled_features); |
| } |
| |
| void MediaBrowserTest::RunMediaTestPage(const std::string& html_page, |
| const base::StringPairs& query_params, |
| const std::string& expected_title, |
| bool http) { |
| GURL gurl; |
| std::string query = media::GetURLQueryString(query_params); |
| std::unique_ptr<net::EmbeddedTestServer> http_test_server; |
| if (http) { |
| http_test_server = std::make_unique<net::EmbeddedTestServer>(); |
| http_test_server->ServeFilesFromSourceDirectory(media::GetTestDataPath()); |
| CHECK(http_test_server->Start()); |
| gurl = http_test_server->GetURL("/" + html_page + "?" + query); |
| } else { |
| gurl = content::GetFileUrlWithQuery(media::GetTestDataFilePath(html_page), |
| query); |
| } |
| std::string final_title = RunTest(gurl, expected_title); |
| EXPECT_EQ(expected_title, final_title); |
| } |
| |
| std::string MediaBrowserTest::RunTest(const GURL& gurl, |
| const std::string& expected_title) { |
| VLOG(0) << "Running test URL: " << gurl; |
| TitleWatcher title_watcher(shell()->web_contents(), |
| base::ASCIIToUTF16(expected_title)); |
| AddTitlesToAwait(&title_watcher); |
| EXPECT_TRUE(NavigateToURL(shell(), gurl)); |
| std::u16string result = title_watcher.WaitAndGetTitle(); |
| |
| CleanupTest(); |
| return base::UTF16ToASCII(result); |
| } |
| |
| void MediaBrowserTest::CleanupTest() { |
| #if BUILDFLAG(IS_ANDROID) |
| // We only do this cleanup on Android, as a workaround for a test-only OOM |
| // bug. See http://crbug.com/727542 |
| const std::u16string cleaner_title = kClean; |
| TitleWatcher clean_title_watcher(shell()->web_contents(), cleaner_title); |
| GURL cleaner_url = content::GetFileUrlWithQuery( |
| media::GetTestDataFilePath("cleaner.html"), ""); |
| EXPECT_TRUE(NavigateToURL(shell(), cleaner_url)); |
| std::u16string cleaner_result = clean_title_watcher.WaitAndGetTitle(); |
| EXPECT_EQ(cleaner_result, cleaner_title); |
| #endif |
| } |
| |
| std::string MediaBrowserTest::EncodeErrorMessage( |
| const std::string& original_message) { |
| url::RawCanonOutputT<char> buffer; |
| url::EncodeURIComponent(original_message, &buffer); |
| return std::string(buffer.view()); |
| } |
| |
| void MediaBrowserTest::AddTitlesToAwait(content::TitleWatcher* title_watcher) { |
| title_watcher->AlsoWaitForTitle(base::ASCIIToUTF16(media::kEndedTitle)); |
| title_watcher->AlsoWaitForTitle(base::ASCIIToUTF16(media::kErrorTitle)); |
| title_watcher->AlsoWaitForTitle(base::ASCIIToUTF16(media::kErrorEventTitle)); |
| title_watcher->AlsoWaitForTitle(base::ASCIIToUTF16(media::kFailedTitle)); |
| } |
| |
| void MediaBrowserTest::PreRunTestOnMainThread() { |
| ContentBrowserTest::PreRunTestOnMainThread(); |
| media::AddSupplementalCodecsForTesting(GetGpuPreferencesFromCommandLine()); |
| } |
| |
| // Tests playback and seeking of an audio or video file. Test starts with |
| // playback then, after X seconds or the ended event fires, seeks near end of |
| // file; see player.html for details. The test completes when either the last |
| // 'ended' or an 'error' event fires. |
| class MediaTest : public testing::WithParamInterface<bool>, |
| public MediaBrowserTest { |
| public: |
| bool is_accelerated() { return GetParam(); } |
| |
| void SetUpCommandLine(base::CommandLine* command_line) override { |
| if (!is_accelerated()) |
| command_line->AppendSwitch(switches::kDisableAcceleratedVideoDecode); |
| MediaBrowserTest::SetUpCommandLine(command_line); |
| } |
| |
| void MaybePlayVideo(std::string_view codec_string, |
| const std::string& file_name) { |
| constexpr char kTestVideoPlaybackScript[] = R"({ |
| const video = document.createElement('video'); |
| video.canPlayType('video/mp4; codecs=$1') === 'probably'; |
| })"; |
| EXPECT_TRUE(NavigateToURL(shell(), GURL(url::kAboutBlankURL))); |
| content::WebContents* web_contents = shell()->web_contents(); |
| bool result = |
| EvalJs(web_contents, JsReplace(kTestVideoPlaybackScript, codec_string)) |
| .ExtractBool(); |
| if (!result) { |
| return; |
| } |
| |
| PlayVideo(file_name); |
| } |
| |
| // Play specified audio over http:// or file:// depending on |http| setting. |
| void PlayAudio(const std::string& media_file, bool http = true) { |
| PlayMedia("audio", media_file, http); |
| } |
| |
| // Play specified video over http:// or file:// depending on |http| setting. |
| void PlayVideo(const std::string& media_file, bool http = true) { |
| PlayMedia("video", media_file, http); |
| } |
| |
| void PlayMedia(const std::string& tag, |
| const std::string& media_file, |
| bool http) { |
| base::StringPairs query_params; |
| query_params.emplace_back(tag, media_file); |
| RunMediaTestPage("player.html", query_params, media::kEndedTitle, http); |
| } |
| |
| void RunErrorMessageTest(const std::string& tag, |
| const std::string& media_file, |
| const std::string& expected_error_substring) { |
| base::StringPairs query_params; |
| query_params.emplace_back(tag, media_file); |
| query_params.emplace_back("error_substr", |
| EncodeErrorMessage(expected_error_substring)); |
| RunMediaTestPage("player.html", query_params, media::kErrorEventTitle, |
| true); |
| } |
| |
| void RunVideoSizeTest(const char* media_file, int width, int height) { |
| std::string expected_title = std::string(media::kEndedTitle) + " " + |
| base::NumberToString(width) + " " + |
| base::NumberToString(height); |
| base::StringPairs query_params; |
| query_params.emplace_back("video", media_file); |
| query_params.emplace_back("sizetest", "true"); |
| RunMediaTestPage("player.html", query_params, expected_title, true); |
| } |
| }; |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBearWebm) { |
| PlayVideo("bear.webm"); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBearWebm_FileProtocol) { |
| PlayVideo("bear.webm", false); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, AudioBearOpusWebm) { |
| PlayAudio("bear-opus.webm"); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, AudioBearOpusMp4) { |
| PlayAudio("bear-opus.mp4"); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, AudioBearOpusOgg) { |
| PlayAudio("bear-opus.ogg"); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, AudioBearOpusOgg_FileProtocol) { |
| PlayAudio("bear-opus.ogg", false); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBearSilentWebm) { |
| PlayVideo("bear_silent.webm"); |
| } |
| |
| // We don't expect android devices to support highbit yet. |
| #if !BUILDFLAG(IS_ANDROID) |
| |
| // TODO(crbug.com/40242077): DEMUXER_ERROR_NO_SUPPORTED_STREAMS error on |
| // Fuchsia Arm64. |
| #if BUILDFLAG(IS_FUCHSIA) && defined(ARCH_CPU_ARM64) |
| #define MAYBE_VideoBearHighBitDepthVP9 DISABLED_VideoBearHighBitDepthVP9 |
| #else |
| #define MAYBE_VideoBearHighBitDepthVP9 VideoBearHighBitDepthVP9 |
| #endif |
| IN_PROC_BROWSER_TEST_P(MediaTest, MAYBE_VideoBearHighBitDepthVP9) { |
| PlayVideo("bear-320x180-hi10p-vp9.webm"); |
| } |
| |
| // TODO(crbug.com/40242077): DEMUXER_ERROR_NO_SUPPORTED_STREAMS error on |
| // Fuchsia Arm64. |
| #if BUILDFLAG(IS_FUCHSIA) && defined(ARCH_CPU_ARM64) |
| #define MAYBE_VideoBear12DepthVP9 DISABLED_VideoBear12DepthVP9 |
| #else |
| #define MAYBE_VideoBear12DepthVP9 VideoBear12DepthVP9 |
| #endif |
| IN_PROC_BROWSER_TEST_P(MediaTest, MAYBE_VideoBear12DepthVP9) { |
| // Hardware decode on does not reliably support 12-bit. |
| if (is_accelerated()) |
| return; |
| PlayVideo("bear-320x180-hi12p-vp9.webm"); |
| } |
| #endif // !BUILDFLAG(IS_ANDROID) |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBearMp4Vp9) { |
| PlayVideo("bear-320x240-v_frag-vp9.mp4"); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, AudioBearFlacMp4) { |
| PlayAudio("bear-flac.mp4"); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, AudioBearFlac192kHzMp4) { |
| PlayAudio("bear-flac-192kHz.mp4"); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBearMovPcmS16be) { |
| PlayAudio("bear_pcm_s16be.mov"); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBearMovPcmS24be) { |
| PlayAudio("bear_pcm_s24be.mov"); |
| } |
| |
| #if BUILDFLAG(USE_PROPRIETARY_CODECS) |
| #if BUILDFLAG(ENABLE_HLS_DEMUXER) |
| |
| // TODO(crbug.com/384342045): Failing on win11-arm64. |
| #if BUILDFLAG(IS_WIN) && defined(ARCH_CPU_ARM64) |
| #define MAYBE_HLSSingleFileBear DISABLED_HLSSingleFileBear |
| #else |
| #define MAYBE_HLSSingleFileBear HLSSingleFileBear |
| #endif |
| IN_PROC_BROWSER_TEST_P(MediaTest, MAYBE_HLSSingleFileBear) { |
| REQUIRE_ACCELERATION_ON_ANDROID(); |
| PlayVideo("bear-1280x720-hls-clear-mpl.m3u8"); |
| } |
| |
| #if BUILDFLAG(IS_WIN) && defined(ARCH_CPU_ARM64) |
| #define MAYBE_HLSSingleWithoutExtension DISABLED_HLSSingleWithoutExtension |
| #else |
| #define MAYBE_HLSSingleWithoutExtension HLSSingleWithoutExtension |
| #endif |
| IN_PROC_BROWSER_TEST_P(MediaTest, MAYBE_HLSSingleWithoutExtension) { |
| REQUIRE_ACCELERATION_ON_ANDROID(); |
| PlayVideo("hls/mp_ts_avc1.hls"); |
| } |
| |
| // TODO(crbug.com/384342045): Failing on win11-arm64. |
| #if BUILDFLAG(IS_WIN) && defined(ARCH_CPU_ARM64) |
| #define MAYBE_HLSMultivariantBitrateBear DISABLED_HLSMultivariantBitrateBear |
| #else |
| #define MAYBE_HLSMultivariantBitrateBear HLSMultivariantBitrateBear |
| #endif |
| IN_PROC_BROWSER_TEST_P(MediaTest, MAYBE_HLSMultivariantBitrateBear) { |
| REQUIRE_ACCELERATION_ON_ANDROID(); |
| PlayVideo("hls/multi-bitrate-multivariant-bear/playlist.m3u8"); |
| } |
| |
| #endif // BUILDFLAG(ENABLE_HLS_DEMUXER) |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBearMp4) { |
| REQUIRE_ACCELERATION_ON_ANDROID(); |
| PlayVideo("bear.mp4"); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBearSilentMp4) { |
| REQUIRE_ACCELERATION_ON_ANDROID(); |
| PlayVideo("bear_silent.mp4"); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBearRotated0) { |
| REQUIRE_ACCELERATION_ON_ANDROID(); |
| RunVideoSizeTest("bear_rotate_0.mp4", 1280, 720); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBearRotated90) { |
| REQUIRE_ACCELERATION_ON_ANDROID(); |
| RunVideoSizeTest("bear_rotate_90.mp4", 720, 1280); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBearRotated180) { |
| REQUIRE_ACCELERATION_ON_ANDROID(); |
| RunVideoSizeTest("bear_rotate_180.mp4", 1280, 720); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBearRotated270) { |
| REQUIRE_ACCELERATION_ON_ANDROID(); |
| RunVideoSizeTest("bear_rotate_270.mp4", 720, 1280); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBear3gpAacH264) { |
| REQUIRE_ACCELERATION_ON_ANDROID(); |
| PlayVideo("bear_h264_aac.3gp"); |
| } |
| |
| #if BUILDFLAG(ENABLE_HEVC_PARSER_AND_HW_DECODER) |
| // HEVC video stream with 8-bit 422 range extension profile |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBearMp4Hevc8bit422) { |
| MaybePlayVideo("hvc1.4.10.L93.9D.08", |
| "quick-brown-fox-1280x720-hevc-rext-8bit-422-no-audio.mp4"); |
| } |
| |
| // HEVC video stream with 8-bit 444 range extension profile |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBearMp4Hevc8bit444) { |
| MaybePlayVideo("hvc1.4.10.L93.9E.08", |
| "quick-brown-fox-1280x720-hevc-rext-8bit-444-no-audio.mp4"); |
| } |
| |
| // HEVC video stream with 10-bit 422 range extension profile |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBearMp4Hevc10bit422) { |
| MaybePlayVideo("hvc1.4.10.L93.9D.08", |
| "quick-brown-fox-1280x720-hevc-rext-10bit-422-no-audio.mp4"); |
| } |
| |
| // HEVC video stream with 10-bit 444 range extension profile |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBearMp4Hevc10bit444) { |
| MaybePlayVideo("hvc1.4.10.L93.9C.08", |
| "quick-brown-fox-1280x720-hevc-rext-10bit-444-no-audio.mp4"); |
| } |
| |
| // HEVC video stream with 8-bit main profile |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBearMp4Hevc8bit) { |
| // TODO(crbug.com/40269930) : For Android, the `canPlayType()` test in |
| // `MaybePlayVideo` should be reporting the correct status for HEVC. The below |
| // `REQUIRE_ACCELERATION_ON_ANDROID` flag is a temporary fix. |
| REQUIRE_ACCELERATION_ON_ANDROID(); |
| MaybePlayVideo("hev1.1.6.L93.90", "bear-1280x720-hevc-no-audio.mp4"); |
| } |
| |
| // HEVC video stream with 10-bit main10 profile |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBearMp4Hevc10bit) { |
| // TODO(crbug.com/40269930) : For Android, the `canPlayType()` test in |
| // `MaybePlayVideo` should be reporting the correct status for HEVC. The below |
| // `REQUIRE_ACCELERATION_ON_ANDROID` flag is a temporary fix. |
| REQUIRE_ACCELERATION_ON_ANDROID(); |
| MaybePlayVideo("hev1.2.4.L93.90", "bear-1280x720-hevc-10bit-no-audio.mp4"); |
| } |
| #endif // BUILDFLAG(ENABLE_HEVC_PARSER_AND_HW_DECODER) |
| |
| #if BUILDFLAG(ENABLE_FFMPEG_VIDEO_DECODERS) |
| |
| // Android devices usually only support baseline, main and high. |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBearHighBitDepthMp4) { |
| PlayVideo("bear-320x180-hi10p.mp4"); |
| } |
| |
| #endif |
| |
| // Android can't reliably load lots of videos on a page. |
| // See http://crbug.com/749265 |
| // TODO(crbug.com/40774322): Flaky on Mac. |
| #if BUILDFLAG(IS_MAC) || BUILDFLAG(IS_ANDROID) |
| #define MAYBE_LoadManyVideos DISABLED_LoadManyVideos |
| #else |
| #define MAYBE_LoadManyVideos LoadManyVideos |
| #endif |
| IN_PROC_BROWSER_TEST_P(MediaTest, MAYBE_LoadManyVideos) { |
| // Only run this test in one configuration. |
| if (is_accelerated()) |
| return; |
| base::StringPairs query_params; |
| RunMediaTestPage("load_many_videos.html", query_params, media::kEndedTitle, |
| true); |
| } |
| #endif // BUILDFLAG(USE_PROPRIETARY_CODECS) |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, AudioBearFlac) { |
| PlayAudio("bear.flac"); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, AudioBearFlacOgg) { |
| PlayAudio("bear-flac.ogg"); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBearWavAlaw) { |
| PlayAudio("bear_alaw.wav"); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBearWavMulaw) { |
| PlayAudio("bear_mulaw.wav"); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBearWavPcm) { |
| PlayAudio("bear_pcm.wav"); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBearWavPcm3kHz) { |
| PlayAudio("bear_3kHz.wav"); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoBearWavPcm192kHz) { |
| PlayAudio("bear_192kHz.wav"); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoTulipWebm) { |
| PlayVideo("tulip2.webm"); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoEbu3213Primary) { |
| PlayVideo("ebu-3213-e-vp9.mp4"); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoErrorMissingResource) { |
| RunErrorMessageTest("video", "nonexistent_file.webm", |
| "MEDIA_ELEMENT_ERROR: Format error"); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoErrorEmptySrcAttribute) { |
| RunErrorMessageTest("video", "", "MEDIA_ELEMENT_ERROR: Empty src attribute"); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, VideoErrorNoSupportedStreams) { |
| RunErrorMessageTest("video", "no_streams.webm", |
| "DEMUXER_ERROR_NO_SUPPORTED_STREAMS: FFmpegDemuxer: no " |
| "supported streams"); |
| } |
| |
| // Covers tear-down when navigating away as opposed to browser exiting. |
| IN_PROC_BROWSER_TEST_P(MediaTest, Navigate) { |
| PlayVideo("bear.webm"); |
| EXPECT_TRUE(NavigateToURL(shell(), GURL(url::kAboutBlankURL))); |
| EXPECT_FALSE(shell()->web_contents()->IsCrashed()); |
| } |
| |
| IN_PROC_BROWSER_TEST_P(MediaTest, AudioOnly_XHE_AAC_MP4) { |
| if (media::IsDecoderSupportedAudioType( |
| {media::AudioCodec::kAAC, media::AudioCodecProfile::kXHE_AAC})) { |
| PlayAudio("noise-xhe-aac.mp4"); |
| } |
| } |
| |
| INSTANTIATE_TEST_SUITE_P(Default, MediaTest, ::testing::Bool()); |
| |
| } // namespace content |