electron_content_client.cc 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219
  1. // Copyright (c) 2014 GitHub, Inc.
  2. // Use of this source code is governed by the MIT license that can be
  3. // found in the LICENSE file.
  4. #include "shell/app/electron_content_client.h"
  5. #include <string>
  6. #include <string_view>
  7. #include <vector>
  8. #include "base/command_line.h"
  9. #include "base/containers/extend.h"
  10. #include "base/files/file_util.h"
  11. #include "base/strings/string_split.h"
  12. #include "content/public/common/content_constants.h"
  13. #include "electron/buildflags/buildflags.h"
  14. #include "electron/fuses.h"
  15. #include "extensions/common/constants.h"
  16. #include "pdf/buildflags.h"
  17. #include "ppapi/buildflags/buildflags.h"
  18. #include "shell/common/options_switches.h"
  19. #include "shell/common/process_util.h"
  20. #include "third_party/widevine/cdm/buildflags.h"
  21. #include "ui/base/l10n/l10n_util.h"
  22. #include "ui/base/resource/resource_bundle.h"
  23. #include "url/url_constants.h"
  24. // In SHARED_INTERMEDIATE_DIR.
  25. #include "widevine_cdm_version.h" // NOLINT(build/include_directory)
  26. #if BUILDFLAG(ENABLE_WIDEVINE)
  27. #include "base/native_library.h"
  28. #include "content/public/common/cdm_info.h"
  29. #include "media/base/video_codecs.h"
  30. #endif // BUILDFLAG(ENABLE_WIDEVINE)
  31. #if BUILDFLAG(ENABLE_PDF_VIEWER)
  32. #include "components/pdf/common/constants.h" // nogncheck
  33. #include "shell/common/electron_constants.h"
  34. #endif // BUILDFLAG(ENABLE_PDF_VIEWER)
  35. #if BUILDFLAG(ENABLE_PLUGINS)
  36. #include "content/public/common/content_plugin_info.h"
  37. #endif // BUILDFLAG(ENABLE_PLUGINS)
  38. namespace electron {
  39. namespace {
  40. enum class WidevineCdmFileCheck {
  41. kNotChecked,
  42. kFound,
  43. kNotFound,
  44. };
  45. #if BUILDFLAG(ENABLE_WIDEVINE)
  46. bool IsWidevineAvailable(
  47. base::FilePath* cdm_path,
  48. std::vector<media::VideoCodec>* codecs_supported,
  49. base::flat_set<media::CdmSessionType>* session_types_supported,
  50. base::flat_set<media::EncryptionMode>* modes_supported) {
  51. static WidevineCdmFileCheck widevine_cdm_file_check =
  52. WidevineCdmFileCheck::kNotChecked;
  53. if (widevine_cdm_file_check == WidevineCdmFileCheck::kNotChecked) {
  54. base::CommandLine* command_line = base::CommandLine::ForCurrentProcess();
  55. *cdm_path = command_line->GetSwitchValuePath(switches::kWidevineCdmPath);
  56. if (!cdm_path->empty()) {
  57. *cdm_path = cdm_path->AppendASCII(
  58. base::GetNativeLibraryName(kWidevineCdmLibraryName));
  59. widevine_cdm_file_check = base::PathExists(*cdm_path)
  60. ? WidevineCdmFileCheck::kFound
  61. : WidevineCdmFileCheck::kNotFound;
  62. }
  63. }
  64. if (widevine_cdm_file_check == WidevineCdmFileCheck::kFound) {
  65. // Add the supported codecs as if they came from the component manifest.
  66. // This list must match the CDM that is being bundled with Chrome.
  67. codecs_supported->push_back(media::VideoCodec::kCodecVP8);
  68. codecs_supported->push_back(media::VideoCodec::kCodecVP9);
  69. #if BUILDFLAG(USE_PROPRIETARY_CODECS)
  70. codecs_supported->push_back(media::VideoCodec::kCodecH264);
  71. #endif // BUILDFLAG(USE_PROPRIETARY_CODECS)
  72. // TODO(crbug.com/767941): Push persistent-license support info here once
  73. // we check in a new CDM that supports it on Linux.
  74. session_types_supported->insert(media::CdmSessionType::kTemporary);
  75. #if BUILDFLAG(IS_CHROMEOS)
  76. session_types_supported->insert(media::CdmSessionType::kPersistentLicense);
  77. #endif // BUILDFLAG(IS_CHROMEOS)
  78. modes_supported->insert(media::EncryptionMode::kCenc);
  79. return true;
  80. }
  81. return false;
  82. }
  83. #endif // BUILDFLAG(ENABLE_WIDEVINE)
  84. } // namespace
  85. ElectronContentClient::ElectronContentClient() = default;
  86. ElectronContentClient::~ElectronContentClient() = default;
  87. std::u16string ElectronContentClient::GetLocalizedString(int message_id) {
  88. return l10n_util::GetStringUTF16(message_id);
  89. }
  90. std::string_view ElectronContentClient::GetDataResource(
  91. int resource_id,
  92. ui::ResourceScaleFactor scale_factor) {
  93. return ui::ResourceBundle::GetSharedInstance().GetRawDataResourceForScale(
  94. resource_id, scale_factor);
  95. }
  96. gfx::Image& ElectronContentClient::GetNativeImageNamed(int resource_id) {
  97. return ui::ResourceBundle::GetSharedInstance().GetNativeImageNamed(
  98. resource_id);
  99. }
  100. base::RefCountedMemory* ElectronContentClient::GetDataResourceBytes(
  101. int resource_id) {
  102. return ui::ResourceBundle::GetSharedInstance().LoadDataResourceBytes(
  103. resource_id);
  104. }
  105. void ElectronContentClient::AddAdditionalSchemes(Schemes* schemes) {
  106. // Browser Process registration happens in
  107. // `api::Protocol::RegisterSchemesAsPrivileged`
  108. //
  109. // Renderer Process registration happens in `RendererClientBase`
  110. //
  111. // We use this for registration to network utility process
  112. if (IsUtilityProcess()) {
  113. const auto& cmd = *base::CommandLine::ForCurrentProcess();
  114. auto append_cli_schemes = [&cmd](auto& appendme, const auto key) {
  115. base::Extend(appendme, base::SplitString(cmd.GetSwitchValueASCII(key),
  116. ",", base::TRIM_WHITESPACE,
  117. base::SPLIT_WANT_NONEMPTY));
  118. };
  119. using namespace switches;
  120. append_cli_schemes(schemes->cors_enabled_schemes, kCORSSchemes);
  121. append_cli_schemes(schemes->csp_bypassing_schemes, kBypassCSPSchemes);
  122. append_cli_schemes(schemes->secure_schemes, kSecureSchemes);
  123. append_cli_schemes(schemes->service_worker_schemes, kServiceWorkerSchemes);
  124. append_cli_schemes(schemes->standard_schemes, kStandardSchemes);
  125. }
  126. if (electron::fuses::IsGrantFileProtocolExtraPrivilegesEnabled()) {
  127. schemes->service_worker_schemes.emplace_back(url::kFileScheme);
  128. }
  129. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  130. schemes->standard_schemes.push_back(extensions::kExtensionScheme);
  131. schemes->savable_schemes.push_back(extensions::kExtensionScheme);
  132. schemes->secure_schemes.push_back(extensions::kExtensionScheme);
  133. schemes->service_worker_schemes.push_back(extensions::kExtensionScheme);
  134. schemes->cors_enabled_schemes.push_back(extensions::kExtensionScheme);
  135. schemes->csp_bypassing_schemes.push_back(extensions::kExtensionScheme);
  136. #endif
  137. }
  138. void ElectronContentClient::AddPlugins(
  139. std::vector<content::ContentPluginInfo>* plugins) {
  140. #if BUILDFLAG(ENABLE_PDF_VIEWER)
  141. static constexpr char kPDFPluginExtension[] = "pdf";
  142. static constexpr char kPDFPluginDescription[] = "Portable Document Format";
  143. content::ContentPluginInfo pdf_info;
  144. pdf_info.is_internal = true;
  145. pdf_info.is_out_of_process = true;
  146. pdf_info.name = kPDFInternalPluginName;
  147. pdf_info.description = kPDFPluginDescription;
  148. // This isn't a real file path; it's just used as a unique identifier.
  149. pdf_info.path = base::FilePath(kPdfPluginPath);
  150. content::WebPluginMimeType pdf_mime_type(
  151. pdf::kInternalPluginMimeType, kPDFPluginExtension, kPDFPluginDescription);
  152. pdf_info.mime_types.push_back(pdf_mime_type);
  153. plugins->push_back(pdf_info);
  154. #endif // BUILDFLAG(ENABLE_PDF_VIEWER)
  155. }
  156. void ElectronContentClient::AddContentDecryptionModules(
  157. std::vector<content::CdmInfo>* cdms,
  158. std::vector<media::CdmHostFilePath>* cdm_host_file_paths) {
  159. if (cdms) {
  160. #if BUILDFLAG(ENABLE_WIDEVINE)
  161. base::FilePath cdm_path;
  162. std::vector<media::VideoCodec> video_codecs_supported;
  163. base::flat_set<media::CdmSessionType> session_types_supported;
  164. base::flat_set<media::EncryptionMode> encryption_modes_supported;
  165. if (IsWidevineAvailable(&cdm_path, &video_codecs_supported,
  166. &session_types_supported,
  167. &encryption_modes_supported)) {
  168. base::CommandLine* command_line = base::CommandLine::ForCurrentProcess();
  169. auto cdm_version_string =
  170. command_line->GetSwitchValueASCII(switches::kWidevineCdmVersion);
  171. // CdmInfo needs |path| to be the actual Widevine library,
  172. // not the adapter, so adjust as necessary. It will be in the
  173. // same directory as the installed adapter.
  174. const base::Version version(cdm_version_string);
  175. DCHECK(version.IsValid());
  176. content::CdmCapability capability(
  177. video_codecs_supported, encryption_modes_supported,
  178. session_types_supported, base::flat_set<media::CdmProxy::Protocol>());
  179. cdms->push_back(content::CdmInfo(
  180. kWidevineCdmDisplayName, kWidevineCdmGuid, version, cdm_path,
  181. kWidevineCdmFileSystemId, capability, kWidevineKeySystem, false));
  182. }
  183. #endif // BUILDFLAG(ENABLE_WIDEVINE)
  184. }
  185. }
  186. } // namespace electron