blob: 4a5acc5f53bfe872a605b7fe289e60c478b620ba [file] [log] [blame]
// Copyright 2017 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 "content/browser/renderer_host/render_widget_targeter.h"
#include "base/metrics/histogram_functions.h"
#include "base/metrics/histogram_macros.h"
#include "content/browser/renderer_host/input/one_shot_timeout_monitor.h"
#include "content/browser/renderer_host/render_widget_host_impl.h"
#include "content/browser/renderer_host/render_widget_host_view_base.h"
#include "content/public/browser/site_isolation_policy.h"
#include "third_party/blink/public/platform/web_input_event.h"
#include "ui/events/blink/blink_event_util.h"
namespace content {
namespace {
bool MergeEventIfPossible(const blink::WebInputEvent& event,
ui::WebScopedInputEvent* blink_event) {
if (!blink::WebInputEvent::IsTouchEventType(event.GetType()) &&
!blink::WebInputEvent::IsGestureEventType(event.GetType()) &&
ui::CanCoalesce(event, **blink_event)) {
ui::Coalesce(event, blink_event->get());
return true;
}
return false;
}
gfx::PointF ComputeEventLocation(const blink::WebInputEvent& event) {
if (blink::WebInputEvent::IsMouseEventType(event.GetType()) ||
event.GetType() == blink::WebInputEvent::kMouseWheel) {
return static_cast<const blink::WebMouseEvent&>(event).PositionInWidget();
}
if (blink::WebInputEvent::IsTouchEventType(event.GetType())) {
return static_cast<const blink::WebTouchEvent&>(event)
.touches[0]
.PositionInWidget();
}
if (blink::WebInputEvent::IsGestureEventType(event.GetType()))
return static_cast<const blink::WebGestureEvent&>(event).PositionInWidget();
return gfx::PointF();
}
} // namespace
class TracingUmaTracker {
public:
TracingUmaTracker(const char* metric_name, const char* tracing_category)
: id_(next_id_++),
start_time_(base::TimeTicks::Now()),
metric_name_(metric_name),
tracing_category_(tracing_category) {
TRACE_EVENT_ASYNC_BEGIN0(
tracing_category_, metric_name_,
TRACE_ID_WITH_SCOPE(metric_name_, TRACE_ID_LOCAL(id_)));
}
~TracingUmaTracker() = default;
TracingUmaTracker(TracingUmaTracker&& tracker) = default;
TracingUmaTracker& operator=(TracingUmaTracker&& tracker) = default;
void Stop() {
TRACE_EVENT_ASYNC_END0(
tracing_category_, metric_name_,
TRACE_ID_WITH_SCOPE(metric_name_, TRACE_ID_LOCAL(id_)));
UmaHistogramTimes(metric_name_, base::TimeTicks::Now() - start_time_);
}
private:
const int id_;
const base::TimeTicks start_time_;
// These variables must be string literals and live for the duration
// of the program since tracing stores pointers.
const char* metric_name_;
const char* tracing_category_;
static int next_id_;
DISALLOW_COPY_AND_ASSIGN(TracingUmaTracker);
};
int TracingUmaTracker::next_id_ = 1;
RenderWidgetTargetResult::RenderWidgetTargetResult() = default;
RenderWidgetTargetResult::RenderWidgetTargetResult(
const RenderWidgetTargetResult&) = default;
RenderWidgetTargetResult::RenderWidgetTargetResult(
RenderWidgetHostViewBase* in_view,
bool in_should_query_view,
base::Optional<gfx::PointF> in_location,
bool in_latched_target)
: view(in_view),
should_query_view(in_should_query_view),
target_location(in_location),
latched_target(in_latched_target) {}
RenderWidgetTargetResult::~RenderWidgetTargetResult() = default;
RenderWidgetTargeter::TargetingRequest::TargetingRequest() = default;
RenderWidgetTargeter::TargetingRequest::TargetingRequest(
TargetingRequest&& request) = default;
RenderWidgetTargeter::TargetingRequest& RenderWidgetTargeter::TargetingRequest::
operator=(TargetingRequest&&) = default;
RenderWidgetTargeter::TargetingRequest::~TargetingRequest() = default;
RenderWidgetTargeter::RenderWidgetTargeter(Delegate* delegate)
: delegate_(delegate), weak_ptr_factory_(this) {
DCHECK(delegate_);
}
RenderWidgetTargeter::~RenderWidgetTargeter() = default;
void RenderWidgetTargeter::FindTargetAndDispatch(
RenderWidgetHostViewBase* root_view,
const blink::WebInputEvent& event,
const ui::LatencyInfo& latency) {
DCHECK(blink::WebInputEvent::IsMouseEventType(event.GetType()) ||
event.GetType() == blink::WebInputEvent::kMouseWheel ||
blink::WebInputEvent::IsTouchEventType(event.GetType()) ||
(blink::WebInputEvent::IsGestureEventType(event.GetType()) &&
(static_cast<const blink::WebGestureEvent&>(event).SourceDevice() ==
blink::WebGestureDevice::kWebGestureDeviceTouchscreen ||
static_cast<const blink::WebGestureEvent&>(event).SourceDevice() ==
blink::WebGestureDevice::kWebGestureDeviceTouchpad)));
if (request_in_flight_) {
if (!requests_.empty()) {
auto& request = requests_.back();
if (MergeEventIfPossible(event, &request.event))
return;
}
TargetingRequest request;
request.root_view = root_view->GetWeakPtr();
request.event = ui::WebInputEventTraits::Clone(event);
request.latency = latency;
request.tracker = std::make_unique<TracingUmaTracker>(
"Event.AsyncTargeting.TimeInQueue", "input,latency");
requests_.push(std::move(request));
return;
}
RenderWidgetTargetResult result =
delegate_->FindTargetSynchronously(root_view, event);
RenderWidgetHostViewBase* target = result.view;
auto* event_ptr = &event;
async_depth_ = 0;
// TODO(kenrb, wjmaclean): Asynchronous hit tests don't work properly with
// GuestViews, so rely on the synchronous result.
// See https://crbug.com/802378.
if (result.should_query_view && !target->IsRenderWidgetHostViewGuest()) {
// TODO(kenrb, sadrul): When all event types support asynchronous hit
// testing, we should be able to have FindTargetSynchronously return the
// view and location to use for the renderer hit test query.
// Currently it has to return the surface hit test target, for event types
// that ignore |result.should_query_view|, and therefore we have to use
// root_view and the original event location for the initial query.
QueryClient(root_view, root_view, *event_ptr, latency,
ComputeEventLocation(event), nullptr, gfx::PointF());
} else {
FoundTarget(root_view, target, *event_ptr, latency, result.target_location,
result.latched_target);
}
}
void RenderWidgetTargeter::ViewWillBeDestroyed(RenderWidgetHostViewBase* view) {
unresponsive_views_.erase(view);
}
void RenderWidgetTargeter::QueryClient(
RenderWidgetHostViewBase* root_view,
RenderWidgetHostViewBase* target,
const blink::WebInputEvent& event,
const ui::LatencyInfo& latency,
const gfx::PointF& target_location,
RenderWidgetHostViewBase* last_request_target,
const gfx::PointF& last_target_location) {
DCHECK(!request_in_flight_);
auto* target_client = target->host()->input_target_client();
// |target_client| may not be set yet for this |target| on Mac, need to
// understand why this happens. https://crbug.com/859492
if (!target_client) {
FoundTarget(root_view, target, event, latency, target_location, false);
return;
}
request_in_flight_ = true;
async_depth_++;
TracingUmaTracker tracker("Event.AsyncTargeting.ResponseTime",
"input,latency");
async_hit_test_timeout_.reset(new OneShotTimeoutMonitor(
base::BindOnce(
&RenderWidgetTargeter::AsyncHitTestTimedOut,
weak_ptr_factory_.GetWeakPtr(), root_view->GetWeakPtr(),
target->GetWeakPtr(), target_location,
last_request_target ? last_request_target->GetWeakPtr() : nullptr,
last_target_location, ui::WebInputEventTraits::Clone(event), latency),
async_hit_test_timeout_delay_));
target_client->FrameSinkIdAt(
gfx::ToCeiledPoint(target_location),
base::BindOnce(&RenderWidgetTargeter::FoundFrameSinkId,
weak_ptr_factory_.GetWeakPtr(), root_view->GetWeakPtr(),
target->GetWeakPtr(),
ui::WebInputEventTraits::Clone(event), latency,
++last_request_id_, target_location, std::move(tracker)));
}
void RenderWidgetTargeter::FlushEventQueue() {
while (!request_in_flight_ && !requests_.empty()) {
auto request = std::move(requests_.front());
requests_.pop();
// The root-view has gone away. Ignore this event, and try to process the
// next event.
if (!request.root_view) {
continue;
}
request.tracker->Stop();
FindTargetAndDispatch(request.root_view.get(), *request.event,
request.latency);
}
}
void RenderWidgetTargeter::FoundFrameSinkId(
base::WeakPtr<RenderWidgetHostViewBase> root_view,
base::WeakPtr<RenderWidgetHostViewBase> target,
ui::WebScopedInputEvent event,
const ui::LatencyInfo& latency,
uint32_t request_id,
const gfx::PointF& target_location,
TracingUmaTracker tracker,
const viz::FrameSinkId& frame_sink_id) {
tracker.Stop();
if (request_id != last_request_id_ || !request_in_flight_) {
// This is a response to a request that already timed out, so the event
// should have already been dispatched. Mark the renderer as responsive
// and otherwise ignore this response.
unresponsive_views_.erase(target.get());
return;
}
request_in_flight_ = false;
async_hit_test_timeout_.reset(nullptr);
auto* view = delegate_->FindViewFromFrameSinkId(frame_sink_id);
if (!view)
view = target.get();
// If a client was asked to find a target, then it is necessary to keep
// asking the clients until a client claims an event for itself.
if (view == target.get() ||
unresponsive_views_.find(view) != unresponsive_views_.end()) {
FoundTarget(root_view.get(), view, *event, latency, target_location, false);
} else {
gfx::PointF location = target_location;
target->TransformPointToCoordSpaceForView(location, view, &location);
QueryClient(root_view.get(), view, *event, latency, location, target.get(),
target_location);
}
}
void RenderWidgetTargeter::FoundTarget(
RenderWidgetHostViewBase* root_view,
RenderWidgetHostViewBase* target,
const blink::WebInputEvent& event,
const ui::LatencyInfo& latency,
const base::Optional<gfx::PointF>& target_location,
bool latched_target) {
if (SiteIsolationPolicy::UseDedicatedProcessesForAllSites() &&
!latched_target) {
UMA_HISTOGRAM_COUNTS_100("Event.AsyncTargeting.AsyncClientDepth",
async_depth_);
}
// RenderWidgetHostViewMac can be deleted asynchronously, in which case the
// View will be valid but there will no longer be a RenderWidgetHostImpl.
if (!root_view || !root_view->GetRenderWidgetHost())
return;
delegate_->DispatchEventToTarget(root_view, target, event, latency,
target_location);
FlushEventQueue();
}
void RenderWidgetTargeter::AsyncHitTestTimedOut(
base::WeakPtr<RenderWidgetHostViewBase> current_request_root_view,
base::WeakPtr<RenderWidgetHostViewBase> current_request_target,
const gfx::PointF& current_target_location,
base::WeakPtr<RenderWidgetHostViewBase> last_request_target,
const gfx::PointF& last_target_location,
ui::WebScopedInputEvent event,
const ui::LatencyInfo& latency) {
DCHECK(request_in_flight_);
request_in_flight_ = false;
if (!current_request_root_view)
return;
// Mark view as unresponsive so further events will not be sent to it.
if (current_request_target)
unresponsive_views_.insert(current_request_target.get());
if (current_request_root_view.get() == current_request_target.get()) {
// When a request to the top-level frame times out then the event gets
// sent there anyway. It will trigger the hung renderer dialog if the
// renderer fails to process it.
FoundTarget(current_request_root_view.get(),
current_request_root_view.get(), *event, latency,
current_target_location, false);
} else {
FoundTarget(current_request_root_view.get(), last_request_target.get(),
*event, latency, last_target_location, false);
}
}
} // namespace content