blob: d789ee040ac51fda8eb0087022329cf8423fab14 [file] [log] [blame]
[email protected]dca589ef2012-01-09 11:53:321// Copyright (c) 2012 The Chromium Authors. All rights reserved.
[email protected]8aa7a412011-11-07 12:33:422// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
Nicholas Verned391eae2017-11-13 02:41:385#include "chrome/browser/plugins/plugin_info_host_impl.h"
[email protected]8aa7a412011-11-07 12:33:426
avib896c712015-12-26 02:10:437#include <stddef.h>
dcheng4af48582016-04-19 00:29:358
wafflescd59e9b2016-08-30 04:34:449#include <algorithm>
dcheng4af48582016-04-19 00:29:3510#include <memory>
dchenge73d8520c2015-12-27 01:19:0911#include <utility>
avib896c712015-12-26 02:10:4312
[email protected]8aa7a412011-11-07 12:33:4213#include "base/bind.h"
rbpotter654f42d2016-12-02 22:17:0714#include "base/memory/singleton.h"
Eugene Kim0b4b4572018-09-01 02:37:5815#include "base/stl_util.h"
[email protected]e309f312013-06-07 21:50:0816#include "base/strings/utf_string_conversions.h"
Eric Seckler8652dcd52018-09-20 10:42:2817#include "base/task/post_task.h"
waffles77255cc2016-08-02 17:25:1218#include "base/task_runner_util.h"
Nico Webereaa08412019-08-14 01:24:3719#include "build/branding_buildflags.h"
avib896c712015-12-26 02:10:4320#include "build/build_config.h"
wfhdfcab2e42014-11-07 06:48:3221#include "chrome/browser/browser_process.h"
peconn5100d432015-09-16 12:03:0822#include "chrome/browser/content_settings/host_content_settings_map_factory.h"
[email protected]6be31d202013-02-01 18:20:5423#include "chrome/browser/plugins/chrome_plugin_service_filter.h"
[email protected]cd33d282012-09-21 12:53:0424#include "chrome/browser/plugins/plugin_finder.h"
[email protected]de75c702012-09-25 23:06:0225#include "chrome/browser/plugins/plugin_metadata.h"
[email protected]601e3c82013-10-03 01:31:2026#include "chrome/browser/plugins/plugin_prefs.h"
raymes978301a22016-09-22 00:54:2827#include "chrome/browser/plugins/plugin_utils.h"
[email protected]8aa7a412011-11-07 12:33:4228#include "chrome/browser/profiles/profile.h"
wfhdfcab2e42014-11-07 06:48:3229#include "chrome/browser/ui/browser_otr_state.h"
Scott Violet6200d332018-02-23 21:29:2330#include "chrome/common/buildflags.h"
wafflescd59e9b2016-08-30 04:34:4431#include "chrome/common/chrome_content_client.h"
tommyclic0c4f942016-09-19 19:35:0032#include "chrome/common/chrome_features.h"
Nicholas Verneb5df4f72017-11-08 09:12:0733#include "chrome/common/plugin.mojom.h"
[email protected]8aa7a412011-11-07 12:33:4234#include "chrome/common/pref_names.h"
waffles77255cc2016-08-02 17:25:1235#include "components/component_updater/component_updater_service.h"
mukai8eaec822014-10-25 17:53:1636#include "components/content_settings/core/browser/content_settings_utils.h"
37#include "components/content_settings/core/browser/host_content_settings_map.h"
mukai077089f2014-09-11 18:41:5238#include "components/content_settings/core/common/content_settings.h"
rbpotter654f42d2016-12-02 22:17:0739#include "components/keyed_service/content/browser_context_keyed_service_shutdown_notifier_factory.h"
Scott Violet9ae82892018-03-01 18:38:1240#include "components/nacl/common/buildflags.h"
Bernhard Bauer19190732017-09-28 09:03:3341#include "components/pref_registry/pref_registry_syncable.h"
brettwb1fc1b82016-02-02 00:19:0842#include "components/prefs/pref_service.h"
Steven Holtef0f46772017-12-11 20:54:5143#include "components/ukm/content/source_url_recorder.h"
Eric Seckler8652dcd52018-09-20 10:42:2844#include "content/public/browser/browser_task_traits.h"
[email protected]3a5180ae2011-12-21 02:39:3845#include "content/public/browser/plugin_service.h"
[email protected]31f376c2012-03-13 16:43:0946#include "content/public/browser/plugin_service_filter.h"
tommycli910b4082017-04-03 17:04:1547#include "content/public/browser/render_frame_host.h"
wfhdfcab2e42014-11-07 06:48:3248#include "content/public/common/content_constants.h"
Scott Violetc8240b02018-03-08 22:03:5949#include "extensions/buildflags/buildflags.h"
wfhdfcab2e42014-11-07 06:48:3250#include "net/base/registry_controlled_domains/registry_controlled_domain.h"
Scott Violet02e38b92018-03-27 23:42:1451#include "ppapi/buildflags/buildflags.h"
Steven Holtef0f46772017-12-11 20:54:5152#include "services/metrics/public/cpp/ukm_builders.h"
Mounir Lamourifd9dcefa2017-07-06 10:26:5553#include "services/metrics/public/cpp/ukm_recorder.h"
[email protected]761fa4702013-07-02 15:25:1554#include "url/gurl.h"
tommyclif2a1e9f2016-10-06 18:34:3655#include "url/origin.h"
[email protected]b9b88a082013-03-21 03:51:0256
brettw00899e62016-11-12 02:10:1757#if BUILDFLAG(ENABLE_EXTENSIONS)
fsamuel8dfa19a2015-05-05 01:00:3958#include "components/guest_view/browser/guest_view_base.h"
Sadrul Habib Chowdhuryf43c33c2014-12-04 15:53:5259#include "extensions/browser/extension_registry.h"
hanxi0d0a1e92014-08-26 18:39:4860#include "extensions/browser/guest_view/web_view/web_view_renderer_state.h"
lfg44c8a68f2015-04-16 16:38:1561#include "extensions/common/constants.h"
Sadrul Habib Chowdhuryf43c33c2014-12-04 15:53:5262#include "extensions/common/extension.h"
63#include "extensions/common/manifest_handlers/webview_info.h"
[email protected]21a20132014-07-16 00:48:5964#endif
65
Nico Weberaf3b00b2017-09-11 17:58:1766#if BUILDFLAG(ENABLE_NACL)
[email protected]e0ff1aa2014-08-16 01:38:1167#include "components/nacl/common/nacl_constants.h"
68#endif
69
[email protected]3a5180ae2011-12-21 02:39:3870using content::PluginService;
[email protected]d7bd3e52013-07-21 04:29:2071using content::WebPluginInfo;
[email protected]3a5180ae2011-12-21 02:39:3872
[email protected]b9b88a082013-03-21 03:51:0273namespace {
74
Daniel Bratell84928c52018-01-16 21:40:3475class PluginInfoHostImplShutdownNotifierFactory
rbpotter654f42d2016-12-02 22:17:0776 : public BrowserContextKeyedServiceShutdownNotifierFactory {
77 public:
Daniel Bratell84928c52018-01-16 21:40:3478 static PluginInfoHostImplShutdownNotifierFactory* GetInstance() {
79 return base::Singleton<PluginInfoHostImplShutdownNotifierFactory>::get();
rbpotter654f42d2016-12-02 22:17:0780 }
81
82 private:
Daniel Bratell84928c52018-01-16 21:40:3483 friend struct base::DefaultSingletonTraits<
84 PluginInfoHostImplShutdownNotifierFactory>;
rbpotter654f42d2016-12-02 22:17:0785
Daniel Bratell84928c52018-01-16 21:40:3486 PluginInfoHostImplShutdownNotifierFactory()
rbpotter654f42d2016-12-02 22:17:0787 : BrowserContextKeyedServiceShutdownNotifierFactory(
Nicholas Verned391eae2017-11-13 02:41:3888 "PluginInfoHostImpl") {}
rbpotter654f42d2016-12-02 22:17:0789
Daniel Bratell84928c52018-01-16 21:40:3490 ~PluginInfoHostImplShutdownNotifierFactory() override {}
rbpotter654f42d2016-12-02 22:17:0791
Daniel Bratell84928c52018-01-16 21:40:3492 DISALLOW_COPY_AND_ASSIGN(PluginInfoHostImplShutdownNotifierFactory);
rbpotter654f42d2016-12-02 22:17:0793};
94
brettw00899e62016-11-12 02:10:1795#if BUILDFLAG(ENABLE_EXTENSIONS)
Sadrul Habib Chowdhuryf43c33c2014-12-04 15:53:5296// Returns whether a request from a plugin to load |resource| from a renderer
97// with process id |process_id| is a request for an internal resource by an app
98// listed in |accessible_resources| in its manifest.
99bool IsPluginLoadingAccessibleResourceInWebView(
100 extensions::ExtensionRegistry* extension_registry,
101 int process_id,
102 const GURL& resource) {
103 extensions::WebViewRendererState* renderer_state =
104 extensions::WebViewRendererState::GetInstance();
105 std::string partition_id;
106 if (!renderer_state->IsGuest(process_id) ||
107 !renderer_state->GetPartitionID(process_id, &partition_id)) {
108 return false;
109 }
110
111 const std::string extension_id = resource.host();
paulmeyerad727fc62015-09-09 15:29:59112 const extensions::Extension* extension = extension_registry->GetExtensionById(
113 extension_id, extensions::ExtensionRegistry::ENABLED);
114 if (!extension || !extensions::WebviewInfo::IsResourceWebviewAccessible(
Nicholas Verned391eae2017-11-13 02:41:38115 extension, partition_id, resource.path())) {
Sadrul Habib Chowdhuryf43c33c2014-12-04 15:53:52116 return false;
117 }
118
119 // Make sure the renderer making the request actually belongs to the
120 // same extension.
121 std::string owner_extension;
122 return renderer_state->GetOwnerInfo(process_id, nullptr, &owner_extension) &&
123 owner_extension == extension_id;
124}
brettw00899e62016-11-12 02:10:17125#endif // BUILDFLAG(ENABLE_EXTENSIONS)
Sadrul Habib Chowdhuryf43c33c2014-12-04 15:53:52126
[email protected]b9b88a082013-03-21 03:51:02127} // namespace
128
Nicholas Verned391eae2017-11-13 02:41:38129PluginInfoHostImpl::Context::Context(int render_process_id, Profile* profile)
[email protected]0a8f3a7f2012-01-17 16:17:19130 : render_process_id_(render_process_id),
brettw00899e62016-11-12 02:10:17131#if BUILDFLAG(ENABLE_EXTENSIONS)
Sadrul Habib Chowdhuryf43c33c2014-12-04 15:53:52132 extension_registry_(extensions::ExtensionRegistry::Get(profile)),
133#endif
Nicholas Verned391eae2017-11-13 02:41:38134 host_content_settings_map_(
135 HostContentSettingsMapFactory::GetForProfile(profile)),
[email protected]601e3c82013-10-03 01:31:20136 plugin_prefs_(PluginPrefs::GetForProfile(profile)) {
[email protected]8aa7a412011-11-07 12:33:42137 allow_outdated_plugins_.Init(prefs::kPluginsAllowOutdated,
[email protected]96a5c342012-12-04 18:14:02138 profile->GetPrefs());
Bernhard Bauer19190732017-09-28 09:03:33139 run_all_flash_in_allow_mode_.Init(prefs::kRunAllFlashInAllowMode,
140 profile->GetPrefs());
[email protected]8aa7a412011-11-07 12:33:42141}
142
Nicholas Verned391eae2017-11-13 02:41:38143PluginInfoHostImpl::Context::~Context() {}
[email protected]0a8f3a7f2012-01-17 16:17:19144
Nicholas Verned391eae2017-11-13 02:41:38145void PluginInfoHostImpl::Context::ShutdownOnUIThread() {
rbpotter654f42d2016-12-02 22:17:07146 DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
rbpotter654f42d2016-12-02 22:17:07147 allow_outdated_plugins_.Destroy();
Bernhard Bauer19190732017-09-28 09:03:33148 run_all_flash_in_allow_mode_.Destroy();
rbpotter654f42d2016-12-02 22:17:07149}
150
Nicholas Verned391eae2017-11-13 02:41:38151PluginInfoHostImpl::PluginInfoHostImpl(int render_process_id, Profile* profile)
Clark DuVall1df2052b2019-08-05 19:58:46152 : context_(render_process_id, profile) {
rbpotter654f42d2016-12-02 22:17:07153 shutdown_notifier_ =
Daniel Bratell84928c52018-01-16 21:40:34154 PluginInfoHostImplShutdownNotifierFactory::GetInstance()
155 ->Get(profile)
156 ->Subscribe(base::Bind(&PluginInfoHostImpl::ShutdownOnUIThread,
157 base::Unretained(this)));
rbpotter654f42d2016-12-02 22:17:07158}
159
Nicholas Verned391eae2017-11-13 02:41:38160void PluginInfoHostImpl::ShutdownOnUIThread() {
rbpotter654f42d2016-12-02 22:17:07161 DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
162 context_.ShutdownOnUIThread();
163 shutdown_notifier_.reset();
[email protected]0a8f3a7f2012-01-17 16:17:19164}
165
Bernhard Bauer19190732017-09-28 09:03:33166// static
Nicholas Verned391eae2017-11-13 02:41:38167void PluginInfoHostImpl::RegisterUserPrefs(
Bernhard Bauer19190732017-09-28 09:03:33168 user_prefs::PrefRegistrySyncable* registry) {
169 registry->RegisterBooleanPref(prefs::kPluginsAllowOutdated, false);
Bernhard Bauer19190732017-09-28 09:03:33170 registry->RegisterBooleanPref(prefs::kRunAllFlashInAllowMode, false);
171}
172
Nicholas Verned391eae2017-11-13 02:41:38173PluginInfoHostImpl::~PluginInfoHostImpl() {}
[email protected]649d1c02012-04-27 02:56:21174
Nicholas Verned391eae2017-11-13 02:41:38175struct PluginInfoHostImpl::GetPluginInfo_Params {
[email protected]60eca4eb2013-12-06 00:02:16176 int render_frame_id;
[email protected]8aa7a412011-11-07 12:33:42177 GURL url;
tommyclif2a1e9f2016-10-06 18:34:36178 url::Origin main_frame_origin;
[email protected]8aa7a412011-11-07 12:33:42179 std::string mime_type;
180};
181
Nicholas Verned391eae2017-11-13 02:41:38182void PluginInfoHostImpl::GetPluginInfo(int32_t render_frame_id,
183 const GURL& url,
184 const url::Origin& origin,
185 const std::string& mime_type,
tzikcd02d3b2018-08-09 21:32:47186 GetPluginInfoCallback callback) {
Clark DuVall1df2052b2019-08-05 19:58:46187 DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
Nicholas Verneb5df4f72017-11-08 09:12:07188 GetPluginInfo_Params params = {render_frame_id, url, origin, mime_type};
Clark DuVall1df2052b2019-08-05 19:58:46189 PluginService::GetInstance()->GetPlugins(
190 base::BindOnce(&PluginInfoHostImpl::PluginsLoaded,
191 weak_factory_.GetWeakPtr(), params, std::move(callback)));
[email protected]8aa7a412011-11-07 12:33:42192}
193
Nicholas Verned391eae2017-11-13 02:41:38194void PluginInfoHostImpl::PluginsLoaded(
[email protected]8aa7a412011-11-07 12:33:42195 const GetPluginInfo_Params& params,
Nicholas Verneb5df4f72017-11-08 09:12:07196 GetPluginInfoCallback callback,
[email protected]dfdfeb72012-06-08 11:29:02197 const std::vector<WebPluginInfo>& plugins) {
Nicholas Verneb5df4f72017-11-08 09:12:07198 chrome::mojom::PluginInfoPtr output = chrome::mojom::PluginInfo::New();
[email protected]79fa3362012-03-28 16:21:11199 // This also fills in |actual_mime_type|.
dcheng4af48582016-04-19 00:29:35200 std::unique_ptr<PluginMetadata> plugin_metadata;
[email protected]60eca4eb2013-12-06 00:02:16201 if (context_.FindEnabledPlugin(params.render_frame_id, params.url,
tommyclif2a1e9f2016-10-06 18:34:36202 params.main_frame_origin, params.mime_type,
waffles77255cc2016-08-02 17:25:12203 &output->status, &output->plugin,
204 &output->actual_mime_type, &plugin_metadata)) {
tommyclie97656b82016-11-30 16:43:00205 context_.DecidePluginStatus(
206 params.url, params.main_frame_origin, output->plugin,
207 plugin_metadata->GetSecurityStatus(output->plugin),
208 plugin_metadata->identifier(), &output->status);
[email protected]79fa3362012-03-28 16:21:11209 }
[email protected]cd33d282012-09-21 12:53:04210
Nicholas Verneb5df4f72017-11-08 09:12:07211 if (output->status == chrome::mojom::PluginStatus::kNotFound) {
waffles77255cc2016-08-02 17:25:12212 // Check to see if the component updater can fetch an implementation.
Clark DuVall1df2052b2019-08-05 19:58:46213 std::unique_ptr<component_updater::ComponentInfo> cus_plugin_info =
214 g_browser_process->component_updater()->GetComponentForMimeType(
215 params.mime_type);
216 ComponentPluginLookupDone(params, std::move(output), std::move(callback),
217 std::move(plugin_metadata),
218 std::move(cus_plugin_info));
waffles77255cc2016-08-02 17:25:12219 } else {
Nicholas Verneb5df4f72017-11-08 09:12:07220 GetPluginInfoFinish(params, std::move(output), std::move(callback),
221 std::move(plugin_metadata));
wfhbe79ad52014-11-07 19:02:07222 }
[email protected]8aa7a412011-11-07 12:33:42223}
224
Nicholas Verned391eae2017-11-13 02:41:38225void PluginInfoHostImpl::Context::DecidePluginStatus(
tommyclie97656b82016-11-30 16:43:00226 const GURL& url,
227 const url::Origin& main_frame_origin,
[email protected]546abf32012-09-14 16:55:09228 const WebPluginInfo& plugin,
tommyclie97656b82016-11-30 16:43:00229 PluginMetadata::SecurityStatus security_status,
230 const std::string& plugin_identifier,
Nicholas Verneb5df4f72017-11-08 09:12:07231 chrome::mojom::PluginStatus* status) const {
tommyclie97656b82016-11-30 16:43:00232 if (security_status == PluginMetadata::SECURITY_STATUS_FULLY_TRUSTED) {
Nicholas Verneb5df4f72017-11-08 09:12:07233 *status = chrome::mojom::PluginStatus::kAllowed;
wfhe53608b2016-04-07 02:18:53234 return;
235 }
236
[email protected]8aa7a412011-11-07 12:33:42237 ContentSetting plugin_setting = CONTENT_SETTING_DEFAULT;
238 bool uses_default_content_setting = true;
[email protected]6a9d1deb2014-01-13 19:39:41239 bool is_managed = false;
tommyclie86b2982015-03-16 20:16:45240 // Check plugin content settings. The primary URL is the top origin URL and
241 // the secondary URL is the plugin URL.
raymes978301a22016-09-22 00:54:28242 PluginUtils::GetPluginContentSetting(
tommyclie97656b82016-11-30 16:43:00243 host_content_settings_map_, plugin, main_frame_origin, url,
244 plugin_identifier, &plugin_setting, &uses_default_content_setting,
245 &is_managed);
tommyclie2b1f1b2015-03-31 22:54:59246
[email protected]8aa7a412011-11-07 12:33:42247 DCHECK(plugin_setting != CONTENT_SETTING_DEFAULT);
tommyclie2b1f1b2015-03-31 22:54:59248 DCHECK(plugin_setting != CONTENT_SETTING_ASK);
[email protected]8aa7a412011-11-07 12:33:42249
Nicholas Verneb5df4f72017-11-08 09:12:07250 if (*status == chrome::mojom::PluginStatus::kFlashHiddenPreferHtml) {
tommyclie97656b82016-11-30 16:43:00251 if (plugin_setting == CONTENT_SETTING_BLOCK) {
Tommy C. Li717a34a2019-03-28 16:24:51252 *status = is_managed ? chrome::mojom::PluginStatus::kBlockedByPolicy
253 : chrome::mojom::PluginStatus::kBlockedNoLoading;
tommyclie97656b82016-11-30 16:43:00254 }
255 return;
256 }
257
bauerbecab4f52017-03-13 20:16:41258#if BUILDFLAG(ENABLE_PLUGINS)
tommyclie86b2982015-03-16 20:16:45259 // Check if the plugin is outdated.
tommyclie97656b82016-11-30 16:43:00260 if (security_status == PluginMetadata::SECURITY_STATUS_OUT_OF_DATE &&
[email protected]9f230ed2012-05-24 11:19:40261 !allow_outdated_plugins_.GetValue()) {
[email protected]8aa7a412011-11-07 12:33:42262 if (allow_outdated_plugins_.IsManaged()) {
Nicholas Verneb5df4f72017-11-08 09:12:07263 *status = chrome::mojom::PluginStatus::kOutdatedDisallowed;
[email protected]8aa7a412011-11-07 12:33:42264 } else {
Nicholas Verneb5df4f72017-11-08 09:12:07265 *status = chrome::mojom::PluginStatus::kOutdatedBlocked;
[email protected]8aa7a412011-11-07 12:33:42266 }
267 return;
268 }
bauerbecab4f52017-03-13 20:16:41269#endif // BUILDFLAG(ENABLE_PLUGINS)
[email protected]ef97ce342012-09-24 20:05:02270
tommyclie86b2982015-03-16 20:16:45271 // Check if the plugin is crashing too much.
[email protected]ef97ce342012-09-24 20:05:02272 if (PluginService::GetInstance()->IsPluginUnstable(plugin.path) &&
Nicholas Verned391eae2017-11-13 02:41:38273 plugin_setting != CONTENT_SETTING_BLOCK && uses_default_content_setting) {
Nicholas Verneb5df4f72017-11-08 09:12:07274 *status = chrome::mojom::PluginStatus::kUnauthorized;
[email protected]ef97ce342012-09-24 20:05:02275 return;
276 }
[email protected]8aa7a412011-11-07 12:33:42277
brettw00899e62016-11-12 02:10:17278#if BUILDFLAG(ENABLE_EXTENSIONS)
Sadrul Habib Chowdhuryf43c33c2014-12-04 15:53:52279 // If an app has explicitly made internal resources available by listing them
280 // in |accessible_resources| in the manifest, then allow them to be loaded by
281 // plugins inside a guest-view.
tommyclie97656b82016-11-30 16:43:00282 if (url.SchemeIs(extensions::kExtensionScheme) && !is_managed &&
lfg44c8a68f2015-04-16 16:38:15283 plugin_setting == CONTENT_SETTING_BLOCK &&
tommyclie97656b82016-11-30 16:43:00284 IsPluginLoadingAccessibleResourceInWebView(extension_registry_,
285 render_process_id_, url)) {
Sadrul Habib Chowdhuryf43c33c2014-12-04 15:53:52286 plugin_setting = CONTENT_SETTING_ALLOW;
287 }
brettw00899e62016-11-12 02:10:17288#endif // BUILDFLAG(ENABLE_EXTENSIONS)
Sadrul Habib Chowdhuryf43c33c2014-12-04 15:53:52289
tommycli608ec89b2016-09-28 16:39:00290 if (plugin_setting == CONTENT_SETTING_DETECT_IMPORTANT_CONTENT ||
291 (plugin_setting == CONTENT_SETTING_ALLOW &&
Bernhard Bauer19190732017-09-28 09:03:33292 !run_all_flash_in_allow_mode_.GetValue())) {
Nicholas Verneb5df4f72017-11-08 09:12:07293 *status = chrome::mojom::PluginStatus::kPlayImportantContent;
tommyclie2b1f1b2015-03-31 22:54:59294 } else if (plugin_setting == CONTENT_SETTING_BLOCK) {
Tommy C. Li717a34a2019-03-28 16:24:51295 *status = is_managed ? chrome::mojom::PluginStatus::kBlockedByPolicy
296 : chrome::mojom::PluginStatus::kBlocked;
[email protected]6a9d1deb2014-01-13 19:39:41297 }
[email protected]aea757d2013-09-27 22:27:38298
brettw00899e62016-11-12 02:10:17299#if BUILDFLAG(ENABLE_EXTENSIONS)
tommycli550b2222015-09-03 21:06:10300 // Allow an embedder of <webview> to block a plugin from being loaded inside
301 // the guest. In order to do this, set the status to 'Unauthorized' here,
302 // and update the status as appropriate depending on the response from the
303 // embedder.
Nicholas Verneb5df4f72017-11-08 09:12:07304 if (*status == chrome::mojom::PluginStatus::kAllowed ||
305 *status == chrome::mojom::PluginStatus::kBlocked ||
306 *status == chrome::mojom::PluginStatus::kPlayImportantContent) {
[email protected]140d6cd92014-08-12 18:26:46307 if (extensions::WebViewRendererState::GetInstance()->IsGuest(
tommycli550b2222015-09-03 21:06:10308 render_process_id_))
Nicholas Verneb5df4f72017-11-08 09:12:07309 *status = chrome::mojom::PluginStatus::kUnauthorized;
[email protected]aea757d2013-09-27 22:27:38310 }
tommycli550b2222015-09-03 21:06:10311#endif
[email protected]8aa7a412011-11-07 12:33:42312}
313
Nicholas Verned391eae2017-11-13 02:41:38314bool PluginInfoHostImpl::Context::FindEnabledPlugin(
[email protected]60eca4eb2013-12-06 00:02:16315 int render_frame_id,
[email protected]8aa7a412011-11-07 12:33:42316 const GURL& url,
tommyclif2a1e9f2016-10-06 18:34:36317 const url::Origin& main_frame_origin,
[email protected]8aa7a412011-11-07 12:33:42318 const std::string& mime_type,
Nicholas Verneb5df4f72017-11-08 09:12:07319 chrome::mojom::PluginStatus* status,
[email protected]dfdfeb72012-06-08 11:29:02320 WebPluginInfo* plugin,
[email protected]4da405672012-10-25 10:43:03321 std::string* actual_mime_type,
dcheng4af48582016-04-19 00:29:35322 std::unique_ptr<PluginMetadata>* plugin_metadata) const {
Nicholas Verneb5df4f72017-11-08 09:12:07323 *status = chrome::mojom::PluginStatus::kAllowed;
bauerb96d29c02015-04-09 08:40:35324
[email protected]8aa7a412011-11-07 12:33:42325 bool allow_wildcard = true;
[email protected]dfdfeb72012-06-08 11:29:02326 std::vector<WebPluginInfo> matching_plugins;
[email protected]8aa7a412011-11-07 12:33:42327 std::vector<std::string> mime_types;
328 PluginService::GetInstance()->GetPluginInfoArray(
329 url, mime_type, allow_wildcard, &matching_plugins, &mime_types);
Nico Webereaa08412019-08-14 01:24:37330#if BUILDFLAG(GOOGLE_CHROME_BRANDING)
Eugene Kim0b4b4572018-09-01 02:37:58331 base::EraseIf(matching_plugins, [&](const WebPluginInfo& info) {
332 return info.path.value() == ChromeContentClient::kNotPresent;
333 });
Nico Webereaa08412019-08-14 01:24:37334#endif // BUILDFLAG(GOOGLE_CHROME_BRANDING)
[email protected]4da405672012-10-25 10:43:03335 if (matching_plugins.empty()) {
Nicholas Verneb5df4f72017-11-08 09:12:07336 *status = chrome::mojom::PluginStatus::kNotFound;
[email protected]4da405672012-10-25 10:43:03337 return false;
338 }
339
[email protected]3a5180ae2011-12-21 02:39:38340 content::PluginServiceFilter* filter =
341 PluginService::GetInstance()->GetFilter();
[email protected]4da405672012-10-25 10:43:03342 size_t i = 0;
343 for (; i < matching_plugins.size(); ++i) {
Clark DuVall1df2052b2019-08-05 19:58:46344 if (!filter ||
345 filter->IsPluginAvailable(render_process_id_, render_frame_id, url,
346 main_frame_origin, &matching_plugins[i])) {
[email protected]4da405672012-10-25 10:43:03347 break;
[email protected]8aa7a412011-11-07 12:33:42348 }
349 }
350
tommyclie86b2982015-03-16 20:16:45351 // If we broke out of the loop, we have found an enabled plugin.
[email protected]4da405672012-10-25 10:43:03352 bool enabled = i < matching_plugins.size();
353 if (!enabled) {
tommyclie86b2982015-03-16 20:16:45354 // Otherwise, we only found disabled plugins, so we take the first one.
[email protected]4da405672012-10-25 10:43:03355 i = 0;
Nicholas Verneb5df4f72017-11-08 09:12:07356 *status = chrome::mojom::PluginStatus::kDisabled;
tommyclic0c4f942016-09-19 19:35:00357
Tommy C. Li58342622019-03-27 23:08:51358 // Special case for Flash: this is our Prefer HTML over Plugins logic.
359 if (matching_plugins[0].name ==
360 base::ASCIIToUTF16(content::kFlashPluginName)) {
Nicholas Verneb5df4f72017-11-08 09:12:07361 *status = chrome::mojom::PluginStatus::kFlashHiddenPreferHtml;
tommyclie97656b82016-11-30 16:43:00362
363 // In the Prefer HTML case, the plugin is actually enabled, but hidden.
364 // It will still be blocked in the body of DecidePluginStatus.
365 enabled = true;
tommyclic0c4f942016-09-19 19:35:00366 }
[email protected]4da405672012-10-25 10:43:03367 }
368
369 *plugin = matching_plugins[i];
370 *actual_mime_type = mime_types[i];
371 if (plugin_metadata)
372 *plugin_metadata = PluginFinder::GetInstance()->GetPluginMetadata(*plugin);
373
374 return enabled;
[email protected]8aa7a412011-11-07 12:33:42375}
376
Nicholas Verned391eae2017-11-13 02:41:38377void PluginInfoHostImpl::ComponentPluginLookupDone(
waffles77255cc2016-08-02 17:25:12378 const GetPluginInfo_Params& params,
Nicholas Verneb5df4f72017-11-08 09:12:07379 chrome::mojom::PluginInfoPtr output,
380 GetPluginInfoCallback callback,
waffles77255cc2016-08-02 17:25:12381 std::unique_ptr<PluginMetadata> plugin_metadata,
waffles77255cc2016-08-02 17:25:12382 std::unique_ptr<component_updater::ComponentInfo> cus_plugin_info) {
383 if (cus_plugin_info) {
Nicholas Verneb5df4f72017-11-08 09:12:07384 output->status = chrome::mojom::PluginStatus::kComponentUpdateRequired;
Sean McAllisterdf807712020-08-13 23:19:09385#if defined(OS_LINUX) || defined(OS_CHROMEOS)
wafflese7759f72016-10-10 23:41:25386 if (cus_plugin_info->version != base::Version("0")) {
Nicholas Verneb5df4f72017-11-08 09:12:07387 output->status = chrome::mojom::PluginStatus::kRestartRequired;
wafflese7759f72016-10-10 23:41:25388 }
thestigd6e438182017-06-01 23:16:27389#endif
Tommy C. Li7c493d192019-03-28 21:50:26390 // Component Updater wouldn't provide a deprecated plugin.
391 bool plugin_is_deprecated = false;
Jeremy Romanec48d7a2018-03-01 17:35:09392 plugin_metadata = std::make_unique<PluginMetadata>(
waffles77255cc2016-08-02 17:25:12393 cus_plugin_info->id, cus_plugin_info->name, false, GURL(), GURL(),
Tommy C. Li7c493d192019-03-28 21:50:26394 base::ASCIIToUTF16(cus_plugin_info->id), std::string(),
395 plugin_is_deprecated);
waffles77255cc2016-08-02 17:25:12396 }
Nicholas Verneb5df4f72017-11-08 09:12:07397 GetPluginInfoFinish(params, std::move(output), std::move(callback),
398 std::move(plugin_metadata));
waffles77255cc2016-08-02 17:25:12399}
400
Nicholas Verned391eae2017-11-13 02:41:38401void PluginInfoHostImpl::GetPluginInfoFinish(
waffles77255cc2016-08-02 17:25:12402 const GetPluginInfo_Params& params,
Nicholas Verneb5df4f72017-11-08 09:12:07403 chrome::mojom::PluginInfoPtr output,
404 GetPluginInfoCallback callback,
405 std::unique_ptr<PluginMetadata> plugin_metadata) {
waffles77255cc2016-08-02 17:25:12406 if (plugin_metadata) {
407 output->group_identifier = plugin_metadata->identifier();
408 output->group_name = plugin_metadata->name();
409 }
410
411 context_.MaybeGrantAccess(output->status, output->plugin.path);
412
Nicholas Verneb5df4f72017-11-08 09:12:07413 if (output->status != chrome::mojom::PluginStatus::kNotFound) {
Will Harris235b85172019-11-07 22:21:37414 ReportMetrics(params.render_frame_id, output->actual_mime_type,
Clark DuVall1df2052b2019-08-05 19:58:46415 params.main_frame_origin);
waffles77255cc2016-08-02 17:25:12416 }
Nicholas Verneb5df4f72017-11-08 09:12:07417 std::move(callback).Run(std::move(output));
waffles77255cc2016-08-02 17:25:12418}
419
Nicholas Verned391eae2017-11-13 02:41:38420void PluginInfoHostImpl::ReportMetrics(int render_frame_id,
421 const base::StringPiece& mime_type,
Steven Holtef0f46772017-12-11 20:54:51422 const url::Origin& main_frame_origin) {
tommycli910b4082017-04-03 17:04:15423 DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
424
425 content::RenderFrameHost* frame = content::RenderFrameHost::FromID(
426 context_.render_process_id(), render_frame_id);
427 content::WebContents* web_contents =
428 content::WebContents::FromRenderFrameHost(frame);
429 // This can occur the web contents has already been closed or navigated away.
430 if (!web_contents)
431 return;
432
433 if (web_contents->GetBrowserContext()->IsOffTheRecord())
434 return;
435
Chris Palmerab5e5b52018-09-28 19:19:30436 if (main_frame_origin.opaque())
tommycli910b4082017-04-03 17:04:15437 return;
438
439 if (mime_type != content::kFlashPluginSwfMimeType &&
440 mime_type != content::kFlashPluginSplMimeType) {
441 return;
442 }
443
Steven Holtef0f46772017-12-11 20:54:51444 ukm::builders::Plugins_FlashInstance(
445 ukm::GetSourceIdForWebContentsDocument(web_contents))
446 .Record(ukm::UkmRecorder::Get());
tommycli910b4082017-04-03 17:04:15447}
448
Nicholas Verned391eae2017-11-13 02:41:38449void PluginInfoHostImpl::Context::MaybeGrantAccess(
Nicholas Verneb5df4f72017-11-08 09:12:07450 chrome::mojom::PluginStatus status,
[email protected]650b2d52013-02-10 03:41:45451 const base::FilePath& path) const {
Nicholas Verneb5df4f72017-11-08 09:12:07452 if (status == chrome::mojom::PluginStatus::kAllowed ||
453 status == chrome::mojom::PluginStatus::kPlayImportantContent) {
[email protected]6be31d202013-02-01 18:20:54454 ChromePluginServiceFilter::GetInstance()->AuthorizePlugin(
455 render_process_id_, path);
456 }
457}
458
Nicholas Verned391eae2017-11-13 02:41:38459bool PluginInfoHostImpl::Context::IsPluginEnabled(
[email protected]4cbbeb42014-07-22 19:29:38460 const content::WebPluginInfo& plugin) const {
461 return plugin_prefs_->IsPluginEnabled(plugin);
462}