// Copyright 2013 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "chrome/renderer/media/chrome_key_systems.h" #include #include #include "base/logging.h" #include "base/strings/string16.h" #include "base/strings/string_split.h" #include "base/strings/utf_string_conversions.h" #include "chrome/common/render_messages.h" #include "components/cdm/renderer/widevine_key_systems.h" #include "content/public/renderer/render_thread.h" #if defined(OS_ANDROID) #include "components/cdm/renderer/android_key_systems.h" #endif #include "widevine_cdm_version.h" // In SHARED_INTERMEDIATE_DIR. // The following must be after widevine_cdm_version.h. #if defined(WIDEVINE_CDM_AVAILABLE) && defined(WIDEVINE_CDM_MIN_GLIBC_VERSION) #include #include "base/version.h" #endif using content::KeySystemInfo; using content::SupportedCodecs; #if defined(ENABLE_PEPPER_CDMS) static bool IsPepperCdmAvailable( const std::string& pepper_type, std::vector* additional_param_names, std::vector* additional_param_values) { bool is_available = false; content::RenderThread::Get()->Send( new ChromeViewHostMsg_IsInternalPluginAvailableForMimeType( pepper_type, &is_available, additional_param_names, additional_param_values)); return is_available; } // External Clear Key (used for testing). static void AddExternalClearKey( std::vector* concrete_key_systems) { static const char kExternalClearKeyKeySystem[] = "org.chromium.externalclearkey"; static const char kExternalClearKeyDecryptOnlyKeySystem[] = "org.chromium.externalclearkey.decryptonly"; static const char kExternalClearKeyFileIOTestKeySystem[] = "org.chromium.externalclearkey.fileiotest"; static const char kExternalClearKeyInitializeFailKeySystem[] = "org.chromium.externalclearkey.initializefail"; static const char kExternalClearKeyCrashKeySystem[] = "org.chromium.externalclearkey.crash"; static const char kExternalClearKeyPepperType[] = "application/x-ppapi-clearkey-cdm"; std::vector additional_param_names; std::vector additional_param_values; if (!IsPepperCdmAvailable(kExternalClearKeyPepperType, &additional_param_names, &additional_param_values)) { return; } KeySystemInfo info(kExternalClearKeyKeySystem); info.supported_codecs = content::EME_CODEC_WEBM_ALL; #if defined(USE_PROPRIETARY_CODECS) info.supported_codecs |= content::EME_CODEC_MP4_ALL; #endif // defined(USE_PROPRIETARY_CODECS) info.pepper_type = kExternalClearKeyPepperType; concrete_key_systems->push_back(info); // Add support of decrypt-only mode in ClearKeyCdm. info.key_system = kExternalClearKeyDecryptOnlyKeySystem; concrete_key_systems->push_back(info); // A key system that triggers FileIO test in ClearKeyCdm. info.key_system = kExternalClearKeyFileIOTestKeySystem; concrete_key_systems->push_back(info); // A key system that Chrome thinks is supported by ClearKeyCdm, but actually // will be refused by ClearKeyCdm. This is to test the CDM initialization // failure case. info.key_system = kExternalClearKeyInitializeFailKeySystem; concrete_key_systems->push_back(info); // A key system that triggers a crash in ClearKeyCdm. info.key_system = kExternalClearKeyCrashKeySystem; concrete_key_systems->push_back(info); } #if defined(WIDEVINE_CDM_AVAILABLE) // This function finds "codecs" and parses the value into the vector |codecs|. // Converts the codec strings to UTF-8 since we only expect ASCII strings and // this simplifies the rest of the code in this file. void GetSupportedCodecsForPepperCdm( const std::vector& additional_param_names, const std::vector& additional_param_values, std::vector* codecs) { DCHECK(codecs->empty()); DCHECK_EQ(additional_param_names.size(), additional_param_values.size()); for (size_t i = 0; i < additional_param_names.size(); ++i) { if (additional_param_names[i] == base::ASCIIToUTF16(kCdmSupportedCodecsParamName)) { const base::string16& codecs_string16 = additional_param_values[i]; std::string codecs_string; if (!base::UTF16ToUTF8(codecs_string16.c_str(), codecs_string16.length(), &codecs_string)) { DLOG(WARNING) << "Non-UTF-8 codecs string."; // Continue using the best effort conversion. } base::SplitString(codecs_string, kCdmSupportedCodecsValueDelimiter, codecs); break; } } } static void AddPepperBasedWidevine( std::vector* concrete_key_systems) { #if defined(WIDEVINE_CDM_MIN_GLIBC_VERSION) Version glibc_version(gnu_get_libc_version()); DCHECK(glibc_version.IsValid()); if (glibc_version.IsOlderThan(WIDEVINE_CDM_MIN_GLIBC_VERSION)) return; #endif // defined(WIDEVINE_CDM_MIN_GLIBC_VERSION) std::vector additional_param_names; std::vector additional_param_values; if (!IsPepperCdmAvailable(kWidevineCdmPluginMimeType, &additional_param_names, &additional_param_values)) { DVLOG(1) << "Widevine CDM is not currently available."; return; } std::vector codecs; GetSupportedCodecsForPepperCdm(additional_param_names, additional_param_values, &codecs); SupportedCodecs supported_codecs = content::EME_CODEC_NONE; for (size_t i = 0; i < codecs.size(); ++i) { if (codecs[i] == kCdmSupportedCodecVorbis) supported_codecs |= content::EME_CODEC_WEBM_VORBIS; if (codecs[i] == kCdmSupportedCodecVp8) supported_codecs |= content::EME_CODEC_WEBM_VP8; if (codecs[i] == kCdmSupportedCodecVp9) supported_codecs |= content::EME_CODEC_WEBM_VP9; #if defined(USE_PROPRIETARY_CODECS) if (codecs[i] == kCdmSupportedCodecAac) supported_codecs |= content::EME_CODEC_MP4_AAC; if (codecs[i] == kCdmSupportedCodecAvc1) supported_codecs |= content::EME_CODEC_MP4_AVC1; #endif // defined(USE_PROPRIETARY_CODECS) } cdm::AddWidevineWithCodecs(cdm::WIDEVINE, supported_codecs, concrete_key_systems); } #endif // defined(WIDEVINE_CDM_AVAILABLE) #endif // defined(ENABLE_PEPPER_CDMS) void AddChromeKeySystems(std::vector* key_systems_info) { #if defined(ENABLE_PEPPER_CDMS) AddExternalClearKey(key_systems_info); #if defined(WIDEVINE_CDM_AVAILABLE) AddPepperBasedWidevine(key_systems_info); #endif // defined(WIDEVINE_CDM_AVAILABLE) #endif // defined(ENABLE_PEPPER_CDMS) #if defined(OS_ANDROID) cdm::AddAndroidWidevine(key_systems_info); #endif // defined(OS_ANDROID) }