summaryrefslogtreecommitdiffstats
path: root/chrome
diff options
context:
space:
mode:
authorjamesr@chromium.org <jamesr@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2011-02-16 01:09:05 +0000
committerjamesr@chromium.org <jamesr@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98>2011-02-16 01:09:05 +0000
commit52ccd0eafdc55341868e1fb6e30d18a3638553f0 (patch)
tree5df6accfcf1ad5b6be4577846191d8331475686a /chrome
parent8355d3b353688885d69c5e46795029284eba901e (diff)
downloadchromium_src-52ccd0eafdc55341868e1fb6e30d18a3638553f0.zip
chromium_src-52ccd0eafdc55341868e1fb6e30d18a3638553f0.tar.gz
chromium_src-52ccd0eafdc55341868e1fb6e30d18a3638553f0.tar.bz2
Simple scheduling for requestAnimationFrame
This simplifies the schedule for requestAnimationFrame callbacks and always invokes the callbacks from the top of DoDeferredUpdate. Also adds a histogram for catch-up timers so we can more easily quantify the effects of base::Time / base::TimeTicks skew. BUG=none TEST=none Review URL: http://codereview.chromium.org/6519005 git-svn-id: svn://svn.chromium.org/chrome/trunk/src@75043 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'chrome')
-rw-r--r--chrome/renderer/render_widget.cc75
-rw-r--r--chrome/renderer/render_widget.h6
2 files changed, 35 insertions, 46 deletions
diff --git a/chrome/renderer/render_widget.cc b/chrome/renderer/render_widget.cc
index 28ff55f..15850bd 100644
--- a/chrome/renderer/render_widget.cc
+++ b/chrome/renderer/render_widget.cc
@@ -78,7 +78,7 @@ RenderWidget::RenderWidget(RenderThreadBase* render_thread,
suppress_next_char_events_(false),
is_accelerated_compositing_active_(false),
animation_update_pending_(false),
- animation_waiting_for_paint_(false) {
+ animation_task_posted_(false) {
RenderProcess::current()->AddRefProcess();
DCHECK(render_thread_);
}
@@ -481,26 +481,26 @@ void RenderWidget::PaintDebugBorder(const gfx::Rect& rect,
canvas->drawIRect(irect, paint);
}
-void RenderWidget::CallDoDeferredUpdate() {
- DoDeferredUpdate();
-
- if (pending_input_event_ack_.get())
- Send(pending_input_event_ack_.release());
+void RenderWidget::AnimationCallback() {
+ animation_task_posted_ = false;
+ UMA_HISTOGRAM_CUSTOM_COUNTS("Renderer4.AnimationDelayTime", 0, 0, 16, 17);
+ CallDoDeferredUpdate();
}
-void RenderWidget::UpdateAnimationsIfNeeded() {
- if (webwidget_ && !is_hidden() && animation_update_pending_ &&
- !animation_waiting_for_paint_) {
+void RenderWidget::AnimateIfNeeded() {
+ if (animation_update_pending_) {
base::Time now = base::Time::Now();
if (now >= animation_floor_time_) {
- UpdateAnimationsAndFloorTime();
- // If updating the animation caused invalidations, make sure that we paint
- // at least once before we call animate() again.
- // Update layout first as that might cause further invalidations.
- webwidget_->layout();
- if (paint_aggregator_.HasPendingUpdate())
- animation_waiting_for_paint_ = true;
- } else {
+ animation_floor_time_ = now + base::TimeDelta::FromMilliseconds(16);
+ // Set a timer to call us back after 16ms (targetting 60FPS) before
+ // running animation callbacks so that if a callback requests another
+ // we'll be sure to run it at the proper time.
+ MessageLoop::current()->PostDelayedTask(FROM_HERE, NewRunnableMethod(
+ this, &RenderWidget::AnimationCallback), 16);
+ animation_task_posted_ = true;
+ animation_update_pending_ = false;
+ webwidget_->animate();
+ } else if (!animation_task_posted_) {
// This code uses base::Time::Now() to calculate the floor and next fire
// time because javascript's Date object uses base::Time::Now(). The
// message loop uses base::TimeTicks, which on windows can have a
@@ -510,17 +510,20 @@ void RenderWidget::UpdateAnimationsIfNeeded() {
// avoid exposing this delay to javascript, we keep posting delayed
// tasks until base::Time::Now() has advanced far enough.
int64 delay = (animation_floor_time_ - now).InMillisecondsRoundedUp();
+ UMA_HISTOGRAM_CUSTOM_COUNTS("Renderer4.AnimationDelayTime", delay,
+ 0, 16, 17);
+ animation_task_posted_ = true;
MessageLoop::current()->PostDelayedTask(FROM_HERE, NewRunnableMethod(
- this, &RenderWidget::UpdateAnimationsIfNeeded), delay);
+ this, &RenderWidget::AnimationCallback), delay);
}
}
}
-void RenderWidget::UpdateAnimationsAndFloorTime() {
- animation_update_pending_ = false;
- animation_floor_time_ =
- base::Time::Now() + base::TimeDelta::FromMilliseconds(16);
- webwidget_->animate();
+void RenderWidget::CallDoDeferredUpdate() {
+ DoDeferredUpdate();
+
+ if (pending_input_event_ack_.get())
+ Send(pending_input_event_ack_.release());
}
void RenderWidget::DoDeferredUpdate() {
@@ -534,21 +537,7 @@ void RenderWidget::DoDeferredUpdate() {
return;
}
- if (animation_update_pending_) {
- if (animation_waiting_for_paint_) {
- // If we have pending animation updates but need to paint before updating
- // them, post a task to UpdateAnimationsIfNeeded that will either update
- // animations directly (if the animation floor time has passed by the time
- // the function runs) or post a delayed task if the floor time is not yet
- // reached.
- MessageLoop::current()->PostTask(FROM_HERE, NewRunnableMethod(
- this, &RenderWidget::UpdateAnimationsIfNeeded));
- } else if (base::Time::Now() > animation_floor_time_) {
- // Otherwise, if it's time to update the animations go ahead.
- UpdateAnimationsAndFloorTime();
- }
- }
- animation_waiting_for_paint_ = false;
+ AnimateIfNeeded();
// Layout may generate more invalidation. It may also enable the
// GPU acceleration, so make sure to run layout before we send the
@@ -660,9 +649,6 @@ void RenderWidget::DoDeferredUpdate() {
// WebWidgetClient
void RenderWidget::didInvalidateRect(const WebRect& rect) {
- DCHECK(!is_accelerated_compositing_active_ ||
- (rect.x == 0 && rect.y == 0 && rect.width == 1 && rect.height == 1));
-
// We only want one pending DoDeferredUpdate call at any time...
bool update_pending = paint_aggregator_.HasPendingUpdate();
@@ -744,8 +730,11 @@ void RenderWidget::scheduleComposite() {
void RenderWidget::scheduleAnimation() {
if (!animation_update_pending_) {
animation_update_pending_ = true;
- MessageLoop::current()->PostTask(FROM_HERE, NewRunnableMethod(
- this, &RenderWidget::UpdateAnimationsIfNeeded));
+ if (!animation_task_posted_) {
+ animation_task_posted_ = true;
+ MessageLoop::current()->PostTask(FROM_HERE, NewRunnableMethod(
+ this, &RenderWidget::AnimationCallback));
+ }
}
}
diff --git a/chrome/renderer/render_widget.h b/chrome/renderer/render_widget.h
index dac1efa..aecdb29 100644
--- a/chrome/renderer/render_widget.h
+++ b/chrome/renderer/render_widget.h
@@ -159,9 +159,9 @@ class RenderWidget : public IPC::Channel::Listener,
// Paints a border at the given rect for debugging purposes.
void PaintDebugBorder(const gfx::Rect& rect, skia::PlatformCanvas* canvas);
+ void AnimationCallback();
+ void AnimateIfNeeded();
void CallDoDeferredUpdate();
- void UpdateAnimationsIfNeeded();
- void UpdateAnimationsAndFloorTime();
void DoDeferredUpdate();
void DoDeferredClose();
void DoDeferredSetWindowRect(const WebKit::WebRect& pos);
@@ -373,7 +373,7 @@ class RenderWidget : public IPC::Channel::Listener,
base::Time animation_floor_time_;
bool animation_update_pending_;
- bool animation_waiting_for_paint_;
+ bool animation_task_posted_;
DISALLOW_COPY_AND_ASSIGN(RenderWidget);
};