blob: 8f9bfa5cc9c66c99f62e4b7a8ef0c1e6ab1fcc72 [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"
gabb15e19072016-05-11 20:45:4119#include "base/threading/thread_task_runner_handle.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"
nzolghadrd87a308d2016-12-07 15:45:5643#include "components/rappor/rappor_service_impl.h"
Steven Holtef0f46772017-12-11 20:54:5144#include "components/ukm/content/source_url_recorder.h"
Eric Seckler8652dcd52018-09-20 10:42:2845#include "content/public/browser/browser_task_traits.h"
[email protected]3a5180ae2011-12-21 02:39:3846#include "content/public/browser/plugin_service.h"
[email protected]31f376c2012-03-13 16:43:0947#include "content/public/browser/plugin_service_filter.h"
tommycli910b4082017-04-03 17:04:1548#include "content/public/browser/render_frame_host.h"
wfhdfcab2e42014-11-07 06:48:3249#include "content/public/common/content_constants.h"
Scott Violetc8240b02018-03-08 22:03:5950#include "extensions/buildflags/buildflags.h"
wfhdfcab2e42014-11-07 06:48:3251#include "net/base/registry_controlled_domains/registry_controlled_domain.h"
Scott Violet02e38b92018-03-27 23:42:1452#include "ppapi/buildflags/buildflags.h"
Steven Holtef0f46772017-12-11 20:54:5153#include "services/metrics/public/cpp/ukm_builders.h"
Mounir Lamourifd9dcefa2017-07-06 10:26:5554#include "services/metrics/public/cpp/ukm_recorder.h"
[email protected]761fa4702013-07-02 15:25:1555#include "url/gurl.h"
tommyclif2a1e9f2016-10-06 18:34:3656#include "url/origin.h"
[email protected]b9b88a082013-03-21 03:51:0257
brettw00899e62016-11-12 02:10:1758#if BUILDFLAG(ENABLE_EXTENSIONS)
fsamuel8dfa19a2015-05-05 01:00:3959#include "components/guest_view/browser/guest_view_base.h"
Sadrul Habib Chowdhuryf43c33c2014-12-04 15:53:5260#include "extensions/browser/extension_registry.h"
hanxi0d0a1e92014-08-26 18:39:4861#include "extensions/browser/guest_view/web_view/web_view_renderer_state.h"
lfg44c8a68f2015-04-16 16:38:1562#include "extensions/common/constants.h"
Sadrul Habib Chowdhuryf43c33c2014-12-04 15:53:5263#include "extensions/common/extension.h"
64#include "extensions/common/manifest_handlers/webview_info.h"
[email protected]21a20132014-07-16 00:48:5965#endif
66
Nico Weberaf3b00b2017-09-11 17:58:1767#if BUILDFLAG(ENABLE_NACL)
[email protected]e0ff1aa2014-08-16 01:38:1168#include "components/nacl/common/nacl_constants.h"
69#endif
70
[email protected]3a5180ae2011-12-21 02:39:3871using content::PluginService;
[email protected]d7bd3e52013-07-21 04:29:2072using content::WebPluginInfo;
[email protected]3a5180ae2011-12-21 02:39:3873
[email protected]b9b88a082013-03-21 03:51:0274namespace {
75
Daniel Bratell84928c52018-01-16 21:40:3476class PluginInfoHostImplShutdownNotifierFactory
rbpotter654f42d2016-12-02 22:17:0777 : public BrowserContextKeyedServiceShutdownNotifierFactory {
78 public:
Daniel Bratell84928c52018-01-16 21:40:3479 static PluginInfoHostImplShutdownNotifierFactory* GetInstance() {
80 return base::Singleton<PluginInfoHostImplShutdownNotifierFactory>::get();
rbpotter654f42d2016-12-02 22:17:0781 }
82
83 private:
Daniel Bratell84928c52018-01-16 21:40:3484 friend struct base::DefaultSingletonTraits<
85 PluginInfoHostImplShutdownNotifierFactory>;
rbpotter654f42d2016-12-02 22:17:0786
Daniel Bratell84928c52018-01-16 21:40:3487 PluginInfoHostImplShutdownNotifierFactory()
rbpotter654f42d2016-12-02 22:17:0788 : BrowserContextKeyedServiceShutdownNotifierFactory(
Nicholas Verned391eae2017-11-13 02:41:3889 "PluginInfoHostImpl") {}
rbpotter654f42d2016-12-02 22:17:0790
Daniel Bratell84928c52018-01-16 21:40:3491 ~PluginInfoHostImplShutdownNotifierFactory() override {}
rbpotter654f42d2016-12-02 22:17:0792
Daniel Bratell84928c52018-01-16 21:40:3493 DISALLOW_COPY_AND_ASSIGN(PluginInfoHostImplShutdownNotifierFactory);
rbpotter654f42d2016-12-02 22:17:0794};
95
brettw00899e62016-11-12 02:10:1796#if BUILDFLAG(ENABLE_EXTENSIONS)
Sadrul Habib Chowdhuryf43c33c2014-12-04 15:53:5297// Returns whether a request from a plugin to load |resource| from a renderer
98// with process id |process_id| is a request for an internal resource by an app
99// listed in |accessible_resources| in its manifest.
100bool IsPluginLoadingAccessibleResourceInWebView(
101 extensions::ExtensionRegistry* extension_registry,
102 int process_id,
103 const GURL& resource) {
104 extensions::WebViewRendererState* renderer_state =
105 extensions::WebViewRendererState::GetInstance();
106 std::string partition_id;
107 if (!renderer_state->IsGuest(process_id) ||
108 !renderer_state->GetPartitionID(process_id, &partition_id)) {
109 return false;
110 }
111
112 const std::string extension_id = resource.host();
paulmeyerad727fc62015-09-09 15:29:59113 const extensions::Extension* extension = extension_registry->GetExtensionById(
114 extension_id, extensions::ExtensionRegistry::ENABLED);
115 if (!extension || !extensions::WebviewInfo::IsResourceWebviewAccessible(
Nicholas Verned391eae2017-11-13 02:41:38116 extension, partition_id, resource.path())) {
Sadrul Habib Chowdhuryf43c33c2014-12-04 15:53:52117 return false;
118 }
119
120 // Make sure the renderer making the request actually belongs to the
121 // same extension.
122 std::string owner_extension;
123 return renderer_state->GetOwnerInfo(process_id, nullptr, &owner_extension) &&
124 owner_extension == extension_id;
125}
brettw00899e62016-11-12 02:10:17126#endif // BUILDFLAG(ENABLE_EXTENSIONS)
Sadrul Habib Chowdhuryf43c33c2014-12-04 15:53:52127
[email protected]b9b88a082013-03-21 03:51:02128} // namespace
129
Nicholas Verned391eae2017-11-13 02:41:38130PluginInfoHostImpl::Context::Context(int render_process_id, Profile* profile)
[email protected]0a8f3a7f2012-01-17 16:17:19131 : render_process_id_(render_process_id),
[email protected]df02aca2012-02-09 21:03:20132 resource_context_(profile->GetResourceContext()),
brettw00899e62016-11-12 02:10:17133#if BUILDFLAG(ENABLE_EXTENSIONS)
Sadrul Habib Chowdhuryf43c33c2014-12-04 15:53:52134 extension_registry_(extensions::ExtensionRegistry::Get(profile)),
135#endif
Nicholas Verned391eae2017-11-13 02:41:38136 host_content_settings_map_(
137 HostContentSettingsMapFactory::GetForProfile(profile)),
[email protected]601e3c82013-10-03 01:31:20138 plugin_prefs_(PluginPrefs::GetForProfile(profile)) {
[email protected]8aa7a412011-11-07 12:33:42139 allow_outdated_plugins_.Init(prefs::kPluginsAllowOutdated,
[email protected]96a5c342012-12-04 18:14:02140 profile->GetPrefs());
[email protected]6a1c98e02012-10-24 21:49:43141 allow_outdated_plugins_.MoveToThread(
Eric Seckler8652dcd52018-09-20 10:42:28142 base::CreateSingleThreadTaskRunnerWithTraits(
143 {content::BrowserThread::IO}));
Bernhard Bauer19190732017-09-28 09:03:33144 run_all_flash_in_allow_mode_.Init(prefs::kRunAllFlashInAllowMode,
145 profile->GetPrefs());
146 run_all_flash_in_allow_mode_.MoveToThread(
Eric Seckler8652dcd52018-09-20 10:42:28147 base::CreateSingleThreadTaskRunnerWithTraits(
148 {content::BrowserThread::IO}));
[email protected]8aa7a412011-11-07 12:33:42149}
150
Nicholas Verned391eae2017-11-13 02:41:38151PluginInfoHostImpl::Context::~Context() {}
[email protected]0a8f3a7f2012-01-17 16:17:19152
Nicholas Verned391eae2017-11-13 02:41:38153void PluginInfoHostImpl::Context::ShutdownOnUIThread() {
rbpotter654f42d2016-12-02 22:17:07154 DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
rbpotter654f42d2016-12-02 22:17:07155 allow_outdated_plugins_.Destroy();
Bernhard Bauer19190732017-09-28 09:03:33156 run_all_flash_in_allow_mode_.Destroy();
rbpotter654f42d2016-12-02 22:17:07157}
158
Nicholas Verned391eae2017-11-13 02:41:38159PluginInfoHostImpl::PluginInfoHostImpl(int render_process_id, Profile* profile)
160 : context_(render_process_id, profile),
tommycli910b4082017-04-03 17:04:15161 main_thread_task_runner_(base::ThreadTaskRunnerHandle::Get()),
Nicholas Verneb5df4f72017-11-08 09:12:07162 binding_(this) {
rbpotter654f42d2016-12-02 22:17:07163 shutdown_notifier_ =
Daniel Bratell84928c52018-01-16 21:40:34164 PluginInfoHostImplShutdownNotifierFactory::GetInstance()
165 ->Get(profile)
166 ->Subscribe(base::Bind(&PluginInfoHostImpl::ShutdownOnUIThread,
167 base::Unretained(this)));
rbpotter654f42d2016-12-02 22:17:07168}
169
Nicholas Verned391eae2017-11-13 02:41:38170void PluginInfoHostImpl::ShutdownOnUIThread() {
rbpotter654f42d2016-12-02 22:17:07171 DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
172 context_.ShutdownOnUIThread();
173 shutdown_notifier_.reset();
[email protected]0a8f3a7f2012-01-17 16:17:19174}
175
Nicholas Verned391eae2017-11-13 02:41:38176void PluginInfoHostImplTraits::Destruct(const PluginInfoHostImpl* impl) {
Eric Seckler8652dcd52018-09-20 10:42:28177 base::PostTaskWithTraits(
178 FROM_HERE, {content::BrowserThread::IO},
Nicholas Verned391eae2017-11-13 02:41:38179 base::BindOnce(&PluginInfoHostImpl::DestructOnBrowserThread,
180 base::Unretained(impl)));
Nicholas Verne4ec6cda2017-11-11 11:36:25181}
182
Nicholas Verned391eae2017-11-13 02:41:38183void PluginInfoHostImpl::DestructOnBrowserThread() const {
Nicholas Verneb5df4f72017-11-08 09:12:07184 binding_.Close();
[email protected]8aa7a412011-11-07 12:33:42185 // Destroy on the UI thread because we contain a |PrefMember|.
186 content::BrowserThread::DeleteOnUIThread::Destruct(this);
187}
188
Bernhard Bauer19190732017-09-28 09:03:33189// static
Nicholas Verned391eae2017-11-13 02:41:38190void PluginInfoHostImpl::RegisterUserPrefs(
Bernhard Bauer19190732017-09-28 09:03:33191 user_prefs::PrefRegistrySyncable* registry) {
192 registry->RegisterBooleanPref(prefs::kPluginsAllowOutdated, false);
Bernhard Bauer19190732017-09-28 09:03:33193 registry->RegisterBooleanPref(prefs::kRunAllFlashInAllowMode, false);
194}
195
Nicholas Verned391eae2017-11-13 02:41:38196PluginInfoHostImpl::~PluginInfoHostImpl() {}
[email protected]649d1c02012-04-27 02:56:21197
Nicholas Verned391eae2017-11-13 02:41:38198struct PluginInfoHostImpl::GetPluginInfo_Params {
[email protected]60eca4eb2013-12-06 00:02:16199 int render_frame_id;
[email protected]8aa7a412011-11-07 12:33:42200 GURL url;
tommyclif2a1e9f2016-10-06 18:34:36201 url::Origin main_frame_origin;
[email protected]8aa7a412011-11-07 12:33:42202 std::string mime_type;
203};
204
Nicholas Verned391eae2017-11-13 02:41:38205void PluginInfoHostImpl::OnPluginInfoHostRequest(
Nicholas Verneb5df4f72017-11-08 09:12:07206 chrome::mojom::PluginInfoHostAssociatedRequest request) {
207 binding_.Bind(std::move(request));
208}
209
Nicholas Verned391eae2017-11-13 02:41:38210void PluginInfoHostImpl::GetPluginInfo(int32_t render_frame_id,
211 const GURL& url,
212 const url::Origin& origin,
213 const std::string& mime_type,
tzikcd02d3b2018-08-09 21:32:47214 GetPluginInfoCallback callback) {
Nicholas Verneb5df4f72017-11-08 09:12:07215 GetPluginInfo_Params params = {render_frame_id, url, origin, mime_type};
Nicholas Verned391eae2017-11-13 02:41:38216 PluginService::GetInstance()->GetPlugins(base::BindOnce(
217 &PluginInfoHostImpl::PluginsLoaded, this, params, std::move(callback)));
[email protected]8aa7a412011-11-07 12:33:42218}
219
Nicholas Verned391eae2017-11-13 02:41:38220void PluginInfoHostImpl::PluginsLoaded(
[email protected]8aa7a412011-11-07 12:33:42221 const GetPluginInfo_Params& params,
Nicholas Verneb5df4f72017-11-08 09:12:07222 GetPluginInfoCallback callback,
[email protected]dfdfeb72012-06-08 11:29:02223 const std::vector<WebPluginInfo>& plugins) {
Nicholas Verneb5df4f72017-11-08 09:12:07224 chrome::mojom::PluginInfoPtr output = chrome::mojom::PluginInfo::New();
[email protected]79fa3362012-03-28 16:21:11225 // This also fills in |actual_mime_type|.
dcheng4af48582016-04-19 00:29:35226 std::unique_ptr<PluginMetadata> plugin_metadata;
[email protected]60eca4eb2013-12-06 00:02:16227 if (context_.FindEnabledPlugin(params.render_frame_id, params.url,
tommyclif2a1e9f2016-10-06 18:34:36228 params.main_frame_origin, params.mime_type,
waffles77255cc2016-08-02 17:25:12229 &output->status, &output->plugin,
230 &output->actual_mime_type, &plugin_metadata)) {
tommyclie97656b82016-11-30 16:43:00231 context_.DecidePluginStatus(
232 params.url, params.main_frame_origin, output->plugin,
233 plugin_metadata->GetSecurityStatus(output->plugin),
234 plugin_metadata->identifier(), &output->status);
[email protected]79fa3362012-03-28 16:21:11235 }
[email protected]cd33d282012-09-21 12:53:04236
Nicholas Verneb5df4f72017-11-08 09:12:07237 if (output->status == chrome::mojom::PluginStatus::kNotFound) {
waffles77255cc2016-08-02 17:25:12238 // Check to see if the component updater can fetch an implementation.
239 base::PostTaskAndReplyWithResult(
240 main_thread_task_runner_.get(), FROM_HERE,
Nicholas Verneb5df4f72017-11-08 09:12:07241 base::BindOnce(
waffles77255cc2016-08-02 17:25:12242 &component_updater::ComponentUpdateService::GetComponentForMimeType,
243 base::Unretained(g_browser_process->component_updater()),
244 params.mime_type),
Nicholas Verned391eae2017-11-13 02:41:38245 base::BindOnce(&PluginInfoHostImpl::ComponentPluginLookupDone, this,
246 params, std::move(output), std::move(callback),
Nicholas Verneb5df4f72017-11-08 09:12:07247 std::move(plugin_metadata)));
waffles77255cc2016-08-02 17:25:12248 } else {
Nicholas Verneb5df4f72017-11-08 09:12:07249 GetPluginInfoFinish(params, std::move(output), std::move(callback),
250 std::move(plugin_metadata));
wfhbe79ad52014-11-07 19:02:07251 }
[email protected]8aa7a412011-11-07 12:33:42252}
253
Nicholas Verned391eae2017-11-13 02:41:38254void PluginInfoHostImpl::Context::DecidePluginStatus(
tommyclie97656b82016-11-30 16:43:00255 const GURL& url,
256 const url::Origin& main_frame_origin,
[email protected]546abf32012-09-14 16:55:09257 const WebPluginInfo& plugin,
tommyclie97656b82016-11-30 16:43:00258 PluginMetadata::SecurityStatus security_status,
259 const std::string& plugin_identifier,
Nicholas Verneb5df4f72017-11-08 09:12:07260 chrome::mojom::PluginStatus* status) const {
tommyclie97656b82016-11-30 16:43:00261 if (security_status == PluginMetadata::SECURITY_STATUS_FULLY_TRUSTED) {
Nicholas Verneb5df4f72017-11-08 09:12:07262 *status = chrome::mojom::PluginStatus::kAllowed;
wfhe53608b2016-04-07 02:18:53263 return;
264 }
265
[email protected]8aa7a412011-11-07 12:33:42266 ContentSetting plugin_setting = CONTENT_SETTING_DEFAULT;
267 bool uses_default_content_setting = true;
[email protected]6a9d1deb2014-01-13 19:39:41268 bool is_managed = false;
tommyclie86b2982015-03-16 20:16:45269 // Check plugin content settings. The primary URL is the top origin URL and
270 // the secondary URL is the plugin URL.
raymes978301a22016-09-22 00:54:28271 PluginUtils::GetPluginContentSetting(
tommyclie97656b82016-11-30 16:43:00272 host_content_settings_map_, plugin, main_frame_origin, url,
273 plugin_identifier, &plugin_setting, &uses_default_content_setting,
274 &is_managed);
tommyclie2b1f1b2015-03-31 22:54:59275
[email protected]8aa7a412011-11-07 12:33:42276 DCHECK(plugin_setting != CONTENT_SETTING_DEFAULT);
tommyclie2b1f1b2015-03-31 22:54:59277 DCHECK(plugin_setting != CONTENT_SETTING_ASK);
[email protected]8aa7a412011-11-07 12:33:42278
Nicholas Verneb5df4f72017-11-08 09:12:07279 if (*status == chrome::mojom::PluginStatus::kFlashHiddenPreferHtml) {
tommyclie97656b82016-11-30 16:43:00280 if (plugin_setting == CONTENT_SETTING_BLOCK) {
Tommy C. Li717a34a2019-03-28 16:24:51281 *status = is_managed ? chrome::mojom::PluginStatus::kBlockedByPolicy
282 : chrome::mojom::PluginStatus::kBlockedNoLoading;
tommyclie97656b82016-11-30 16:43:00283 }
284 return;
285 }
286
bauerbecab4f52017-03-13 20:16:41287#if BUILDFLAG(ENABLE_PLUGINS)
tommyclie86b2982015-03-16 20:16:45288 // Check if the plugin is outdated.
tommyclie97656b82016-11-30 16:43:00289 if (security_status == PluginMetadata::SECURITY_STATUS_OUT_OF_DATE &&
[email protected]9f230ed2012-05-24 11:19:40290 !allow_outdated_plugins_.GetValue()) {
[email protected]8aa7a412011-11-07 12:33:42291 if (allow_outdated_plugins_.IsManaged()) {
Nicholas Verneb5df4f72017-11-08 09:12:07292 *status = chrome::mojom::PluginStatus::kOutdatedDisallowed;
[email protected]8aa7a412011-11-07 12:33:42293 } else {
Nicholas Verneb5df4f72017-11-08 09:12:07294 *status = chrome::mojom::PluginStatus::kOutdatedBlocked;
[email protected]8aa7a412011-11-07 12:33:42295 }
296 return;
297 }
bauerbecab4f52017-03-13 20:16:41298#endif // BUILDFLAG(ENABLE_PLUGINS)
[email protected]ef97ce342012-09-24 20:05:02299
tommyclie86b2982015-03-16 20:16:45300 // Check if the plugin is crashing too much.
[email protected]ef97ce342012-09-24 20:05:02301 if (PluginService::GetInstance()->IsPluginUnstable(plugin.path) &&
Nicholas Verned391eae2017-11-13 02:41:38302 plugin_setting != CONTENT_SETTING_BLOCK && uses_default_content_setting) {
Nicholas Verneb5df4f72017-11-08 09:12:07303 *status = chrome::mojom::PluginStatus::kUnauthorized;
[email protected]ef97ce342012-09-24 20:05:02304 return;
305 }
[email protected]8aa7a412011-11-07 12:33:42306
brettw00899e62016-11-12 02:10:17307#if BUILDFLAG(ENABLE_EXTENSIONS)
Sadrul Habib Chowdhuryf43c33c2014-12-04 15:53:52308 // If an app has explicitly made internal resources available by listing them
309 // in |accessible_resources| in the manifest, then allow them to be loaded by
310 // plugins inside a guest-view.
tommyclie97656b82016-11-30 16:43:00311 if (url.SchemeIs(extensions::kExtensionScheme) && !is_managed &&
lfg44c8a68f2015-04-16 16:38:15312 plugin_setting == CONTENT_SETTING_BLOCK &&
tommyclie97656b82016-11-30 16:43:00313 IsPluginLoadingAccessibleResourceInWebView(extension_registry_,
314 render_process_id_, url)) {
Sadrul Habib Chowdhuryf43c33c2014-12-04 15:53:52315 plugin_setting = CONTENT_SETTING_ALLOW;
316 }
brettw00899e62016-11-12 02:10:17317#endif // BUILDFLAG(ENABLE_EXTENSIONS)
Sadrul Habib Chowdhuryf43c33c2014-12-04 15:53:52318
tommycli608ec89b2016-09-28 16:39:00319 if (plugin_setting == CONTENT_SETTING_DETECT_IMPORTANT_CONTENT ||
320 (plugin_setting == CONTENT_SETTING_ALLOW &&
Bernhard Bauer19190732017-09-28 09:03:33321 !run_all_flash_in_allow_mode_.GetValue())) {
Nicholas Verneb5df4f72017-11-08 09:12:07322 *status = chrome::mojom::PluginStatus::kPlayImportantContent;
tommyclie2b1f1b2015-03-31 22:54:59323 } else if (plugin_setting == CONTENT_SETTING_BLOCK) {
Tommy C. Li717a34a2019-03-28 16:24:51324 *status = is_managed ? chrome::mojom::PluginStatus::kBlockedByPolicy
325 : chrome::mojom::PluginStatus::kBlocked;
[email protected]6a9d1deb2014-01-13 19:39:41326 }
[email protected]aea757d2013-09-27 22:27:38327
brettw00899e62016-11-12 02:10:17328#if BUILDFLAG(ENABLE_EXTENSIONS)
tommycli550b2222015-09-03 21:06:10329 // Allow an embedder of <webview> to block a plugin from being loaded inside
330 // the guest. In order to do this, set the status to 'Unauthorized' here,
331 // and update the status as appropriate depending on the response from the
332 // embedder.
Nicholas Verneb5df4f72017-11-08 09:12:07333 if (*status == chrome::mojom::PluginStatus::kAllowed ||
334 *status == chrome::mojom::PluginStatus::kBlocked ||
335 *status == chrome::mojom::PluginStatus::kPlayImportantContent) {
[email protected]140d6cd92014-08-12 18:26:46336 if (extensions::WebViewRendererState::GetInstance()->IsGuest(
tommycli550b2222015-09-03 21:06:10337 render_process_id_))
Nicholas Verneb5df4f72017-11-08 09:12:07338 *status = chrome::mojom::PluginStatus::kUnauthorized;
[email protected]aea757d2013-09-27 22:27:38339 }
tommycli550b2222015-09-03 21:06:10340#endif
[email protected]8aa7a412011-11-07 12:33:42341}
342
Nicholas Verned391eae2017-11-13 02:41:38343bool PluginInfoHostImpl::Context::FindEnabledPlugin(
[email protected]60eca4eb2013-12-06 00:02:16344 int render_frame_id,
[email protected]8aa7a412011-11-07 12:33:42345 const GURL& url,
tommyclif2a1e9f2016-10-06 18:34:36346 const url::Origin& main_frame_origin,
[email protected]8aa7a412011-11-07 12:33:42347 const std::string& mime_type,
Nicholas Verneb5df4f72017-11-08 09:12:07348 chrome::mojom::PluginStatus* status,
[email protected]dfdfeb72012-06-08 11:29:02349 WebPluginInfo* plugin,
[email protected]4da405672012-10-25 10:43:03350 std::string* actual_mime_type,
dcheng4af48582016-04-19 00:29:35351 std::unique_ptr<PluginMetadata>* plugin_metadata) const {
Nicholas Verneb5df4f72017-11-08 09:12:07352 *status = chrome::mojom::PluginStatus::kAllowed;
bauerb96d29c02015-04-09 08:40:35353
[email protected]8aa7a412011-11-07 12:33:42354 bool allow_wildcard = true;
[email protected]dfdfeb72012-06-08 11:29:02355 std::vector<WebPluginInfo> matching_plugins;
[email protected]8aa7a412011-11-07 12:33:42356 std::vector<std::string> mime_types;
357 PluginService::GetInstance()->GetPluginInfoArray(
358 url, mime_type, allow_wildcard, &matching_plugins, &mime_types);
wafflescd59e9b2016-08-30 04:34:44359#if defined(GOOGLE_CHROME_BUILD)
Eugene Kim0b4b4572018-09-01 02:37:58360 base::EraseIf(matching_plugins, [&](const WebPluginInfo& info) {
361 return info.path.value() == ChromeContentClient::kNotPresent;
362 });
wafflescd59e9b2016-08-30 04:34:44363#endif // defined(GOOGLE_CHROME_BUILD)
[email protected]4da405672012-10-25 10:43:03364 if (matching_plugins.empty()) {
Nicholas Verneb5df4f72017-11-08 09:12:07365 *status = chrome::mojom::PluginStatus::kNotFound;
[email protected]4da405672012-10-25 10:43:03366 return false;
367 }
368
[email protected]3a5180ae2011-12-21 02:39:38369 content::PluginServiceFilter* filter =
370 PluginService::GetInstance()->GetFilter();
[email protected]4da405672012-10-25 10:43:03371 size_t i = 0;
372 for (; i < matching_plugins.size(); ++i) {
Nicholas Verned391eae2017-11-13 02:41:38373 if (!filter || filter->IsPluginAvailable(
374 render_process_id_, render_frame_id, resource_context_,
375 url, main_frame_origin, &matching_plugins[i])) {
[email protected]4da405672012-10-25 10:43:03376 break;
[email protected]8aa7a412011-11-07 12:33:42377 }
378 }
379
tommyclie86b2982015-03-16 20:16:45380 // If we broke out of the loop, we have found an enabled plugin.
[email protected]4da405672012-10-25 10:43:03381 bool enabled = i < matching_plugins.size();
382 if (!enabled) {
tommyclie86b2982015-03-16 20:16:45383 // Otherwise, we only found disabled plugins, so we take the first one.
[email protected]4da405672012-10-25 10:43:03384 i = 0;
Nicholas Verneb5df4f72017-11-08 09:12:07385 *status = chrome::mojom::PluginStatus::kDisabled;
tommyclic0c4f942016-09-19 19:35:00386
Tommy C. Li58342622019-03-27 23:08:51387 // Special case for Flash: this is our Prefer HTML over Plugins logic.
388 if (matching_plugins[0].name ==
389 base::ASCIIToUTF16(content::kFlashPluginName)) {
Nicholas Verneb5df4f72017-11-08 09:12:07390 *status = chrome::mojom::PluginStatus::kFlashHiddenPreferHtml;
tommyclie97656b82016-11-30 16:43:00391
392 // In the Prefer HTML case, the plugin is actually enabled, but hidden.
393 // It will still be blocked in the body of DecidePluginStatus.
394 enabled = true;
tommyclic0c4f942016-09-19 19:35:00395 }
[email protected]4da405672012-10-25 10:43:03396 }
397
398 *plugin = matching_plugins[i];
399 *actual_mime_type = mime_types[i];
400 if (plugin_metadata)
401 *plugin_metadata = PluginFinder::GetInstance()->GetPluginMetadata(*plugin);
402
403 return enabled;
[email protected]8aa7a412011-11-07 12:33:42404}
405
Nicholas Verned391eae2017-11-13 02:41:38406void PluginInfoHostImpl::ComponentPluginLookupDone(
waffles77255cc2016-08-02 17:25:12407 const GetPluginInfo_Params& params,
Nicholas Verneb5df4f72017-11-08 09:12:07408 chrome::mojom::PluginInfoPtr output,
409 GetPluginInfoCallback callback,
waffles77255cc2016-08-02 17:25:12410 std::unique_ptr<PluginMetadata> plugin_metadata,
waffles77255cc2016-08-02 17:25:12411 std::unique_ptr<component_updater::ComponentInfo> cus_plugin_info) {
412 if (cus_plugin_info) {
Nicholas Verneb5df4f72017-11-08 09:12:07413 output->status = chrome::mojom::PluginStatus::kComponentUpdateRequired;
wafflese7759f72016-10-10 23:41:25414#if defined(OS_LINUX)
415 if (cus_plugin_info->version != base::Version("0")) {
Nicholas Verneb5df4f72017-11-08 09:12:07416 output->status = chrome::mojom::PluginStatus::kRestartRequired;
wafflese7759f72016-10-10 23:41:25417 }
thestigd6e438182017-06-01 23:16:27418#endif
Jeremy Romanec48d7a2018-03-01 17:35:09419 plugin_metadata = std::make_unique<PluginMetadata>(
waffles77255cc2016-08-02 17:25:12420 cus_plugin_info->id, cus_plugin_info->name, false, GURL(), GURL(),
thestigd6e438182017-06-01 23:16:27421 base::ASCIIToUTF16(cus_plugin_info->id), std::string());
waffles77255cc2016-08-02 17:25:12422 }
Nicholas Verneb5df4f72017-11-08 09:12:07423 GetPluginInfoFinish(params, std::move(output), std::move(callback),
424 std::move(plugin_metadata));
waffles77255cc2016-08-02 17:25:12425}
426
Nicholas Verned391eae2017-11-13 02:41:38427void PluginInfoHostImpl::GetPluginInfoFinish(
waffles77255cc2016-08-02 17:25:12428 const GetPluginInfo_Params& params,
Nicholas Verneb5df4f72017-11-08 09:12:07429 chrome::mojom::PluginInfoPtr output,
430 GetPluginInfoCallback callback,
431 std::unique_ptr<PluginMetadata> plugin_metadata) {
waffles77255cc2016-08-02 17:25:12432 if (plugin_metadata) {
433 output->group_identifier = plugin_metadata->identifier();
434 output->group_name = plugin_metadata->name();
435 }
436
437 context_.MaybeGrantAccess(output->status, output->plugin.path);
438
Nicholas Verneb5df4f72017-11-08 09:12:07439 if (output->status != chrome::mojom::PluginStatus::kNotFound) {
waffles77255cc2016-08-02 17:25:12440 main_thread_task_runner_->PostTask(
tommycli910b4082017-04-03 17:04:15441 FROM_HERE,
Nicholas Verned391eae2017-11-13 02:41:38442 base::BindOnce(&PluginInfoHostImpl::ReportMetrics, this,
tzik3f7781d2017-04-20 17:09:33443 params.render_frame_id, output->actual_mime_type,
Steven Holtef0f46772017-12-11 20:54:51444 params.url, params.main_frame_origin));
waffles77255cc2016-08-02 17:25:12445 }
Nicholas Verneb5df4f72017-11-08 09:12:07446 std::move(callback).Run(std::move(output));
waffles77255cc2016-08-02 17:25:12447}
448
Nicholas Verned391eae2017-11-13 02:41:38449void PluginInfoHostImpl::ReportMetrics(int render_frame_id,
450 const base::StringPiece& mime_type,
451 const GURL& url,
Steven Holtef0f46772017-12-11 20:54:51452 const url::Origin& main_frame_origin) {
tommycli910b4082017-04-03 17:04:15453 DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
454
455 content::RenderFrameHost* frame = content::RenderFrameHost::FromID(
456 context_.render_process_id(), render_frame_id);
457 content::WebContents* web_contents =
458 content::WebContents::FromRenderFrameHost(frame);
459 // This can occur the web contents has already been closed or navigated away.
460 if (!web_contents)
461 return;
462
463 if (web_contents->GetBrowserContext()->IsOffTheRecord())
464 return;
465
466 rappor::RapporServiceImpl* rappor_service =
467 g_browser_process->rappor_service();
468 if (!rappor_service)
469 return;
Chris Palmerab5e5b52018-09-28 19:19:30470 if (main_frame_origin.opaque())
tommycli910b4082017-04-03 17:04:15471 return;
472
473 if (mime_type != content::kFlashPluginSwfMimeType &&
474 mime_type != content::kFlashPluginSplMimeType) {
475 return;
476 }
477
478 rappor_service->RecordSampleString(
479 "Plugins.FlashOriginUrl", rappor::ETLD_PLUS_ONE_RAPPOR_TYPE,
480 net::registry_controlled_domains::GetDomainAndRegistry(
481 main_frame_origin.GetURL(),
482 net::registry_controlled_domains::INCLUDE_PRIVATE_REGISTRIES));
483 rappor_service->RecordSampleString(
484 "Plugins.FlashUrl", rappor::ETLD_PLUS_ONE_RAPPOR_TYPE,
485 net::registry_controlled_domains::GetDomainAndRegistry(
486 url, net::registry_controlled_domains::INCLUDE_PRIVATE_REGISTRIES));
487
Steven Holtef0f46772017-12-11 20:54:51488 ukm::builders::Plugins_FlashInstance(
489 ukm::GetSourceIdForWebContentsDocument(web_contents))
490 .Record(ukm::UkmRecorder::Get());
tommycli910b4082017-04-03 17:04:15491}
492
Nicholas Verned391eae2017-11-13 02:41:38493void PluginInfoHostImpl::Context::MaybeGrantAccess(
Nicholas Verneb5df4f72017-11-08 09:12:07494 chrome::mojom::PluginStatus status,
[email protected]650b2d52013-02-10 03:41:45495 const base::FilePath& path) const {
Nicholas Verneb5df4f72017-11-08 09:12:07496 if (status == chrome::mojom::PluginStatus::kAllowed ||
497 status == chrome::mojom::PluginStatus::kPlayImportantContent) {
[email protected]6be31d202013-02-01 18:20:54498 ChromePluginServiceFilter::GetInstance()->AuthorizePlugin(
499 render_process_id_, path);
500 }
501}
502
Nicholas Verned391eae2017-11-13 02:41:38503bool PluginInfoHostImpl::Context::IsPluginEnabled(
[email protected]4cbbeb42014-07-22 19:29:38504 const content::WebPluginInfo& plugin) const {
505 return plugin_prefs_->IsPluginEnabled(plugin);
506}