Import Cobalt 13.95615

Change-Id: Id63ba05afe04cf6cef5344b716b7f5a2929de0a5
diff --git a/src/base/base_paths_starboard.cc b/src/base/base_paths_starboard.cc
index e02d2f4..b3b6e8e 100644
--- a/src/base/base_paths_starboard.cc
+++ b/src/base/base_paths_starboard.cc
@@ -91,25 +91,21 @@
       return PathProviderStarboard(base::DIR_CACHE, result);
 
     case base::DIR_SYSTEM_FONTS:
-#if SB_API_VERSION >= 4
       if (SbSystemGetPath(kSbSystemPathFontDirectory, path,
                           SB_ARRAY_SIZE_INT(path))) {
         *result = FilePath(path);
         return true;
       }
       DLOG(INFO) << "DIR_SYSTEM_FONTS not defined.";
-#endif  // SB_API_VERSION >= 4
       return false;
 
     case base::DIR_SYSTEM_FONTS_CONFIGURATION:
-#if SB_API_VERSION >= 4
       if (SbSystemGetPath(kSbSystemPathFontConfigurationDirectory, path,
                           SB_ARRAY_SIZE_INT(path))) {
         *result = FilePath(path);
         return true;
       }
       DLOG(INFO) << "DIR_SYSTEM_FONTS_CONFIGURATION not defined.";
-#endif  // SB_API_VERSION >= 4
       return false;
   }
 
diff --git a/src/base/time_starboard.cc b/src/base/time_starboard.cc
index 9225951..75dd719 100644
--- a/src/base/time_starboard.cc
+++ b/src/base/time_starboard.cc
@@ -98,7 +98,7 @@
 
 // static
 TimeTicks TimeTicks::ThreadNow() {
-#if SB_API_VERSION >= 3 && SB_HAS(TIME_THREAD_NOW)
+#if SB_HAS(TIME_THREAD_NOW)
   return TimeTicks(SbTimeGetMonotonicThreadNow());
 #else
   return HighResNow();
@@ -107,7 +107,7 @@
 
 // static
 bool TimeTicks::HasThreadNow() {
-#if SB_API_VERSION >= 3 && SB_HAS(TIME_THREAD_NOW)
+#if SB_HAS(TIME_THREAD_NOW)
   return true;
 #else
   return false;
diff --git a/src/cobalt/base/c_val_collection_timer_stats.h b/src/cobalt/base/c_val_collection_timer_stats.h
index bee81ee..2fdd87a 100644
--- a/src/cobalt/base/c_val_collection_timer_stats.h
+++ b/src/cobalt/base/c_val_collection_timer_stats.h
@@ -71,6 +71,9 @@
     start_time_ = base::TimeTicks();
   }
 
+  // Stops any started timer and this sample is ignored.
+  void Cancel() { start_time_ = base::TimeTicks(); }
+
   // Flush the collection. This causes |CValCollectionStats| to update its stats
   // and clear the entries.
   void Flush() { entry_stats_.Flush(); }
diff --git a/src/cobalt/browser/browser.gyp b/src/cobalt/browser/browser.gyp
index 97fff2f..1629c67 100644
--- a/src/cobalt/browser/browser.gyp
+++ b/src/cobalt/browser/browser.gyp
@@ -96,6 +96,8 @@
         'suspend_fuzzer.h',
         'switches.cc',
         'switches.h',
+        'system_platform_error_handler.cc',
+        'system_platform_error_handler.h',
         'trace_manager.cc',
         'trace_manager.h',
         'url_handler.cc',
diff --git a/src/cobalt/browser/browser_module.cc b/src/cobalt/browser/browser_module.cc
index cfc4a45..870d7b5 100644
--- a/src/cobalt/browser/browser_module.cc
+++ b/src/cobalt/browser/browser_module.cc
@@ -14,6 +14,7 @@
 
 #include "cobalt/browser/browser_module.h"
 
+#include <algorithm>
 #include <vector>
 
 #include "base/bind.h"
@@ -252,10 +253,13 @@
       timeout_polling_thread_(kTimeoutPollingThreadName),
       render_timeout_count_(0),
 #endif
+      on_error_retry_count_(0),
       will_quit_(false),
       application_state_(initial_application_state),
       splash_screen_cache_(new SplashScreenCache()),
-      produced_render_tree_(false) {
+      navigation_produced_main_render_tree_(false) {
+  h5vcc_url_handler_.reset(new H5vccURLHandler(this));
+
 #if SB_HAS(CORE_DUMP_HANDLER_SUPPORT)
   SbCoreDumpRegisterHandler(BrowserModule::CoreDumpHandler, this);
   on_error_triggered_count_ = 0;
@@ -330,12 +334,15 @@
 
   fallback_splash_screen_url_ = options.fallback_splash_screen_url;
   // Synchronously construct our WebModule object.
-  NavigateInternal(url);
+  Navigate(url);
   DCHECK(web_module_);
 }
 
 BrowserModule::~BrowserModule() {
   DCHECK_EQ(MessageLoop::current(), self_message_loop_);
+  if (on_error_retry_timer_.IsRunning()) {
+    on_error_retry_timer_.Stop();
+  }
 #if SB_HAS(CORE_DUMP_HANDLER_SUPPORT)
   SbCoreDumpUnregisterHandler(BrowserModule::CoreDumpHandler, this);
 #endif
@@ -343,42 +350,17 @@
 
 void BrowserModule::Navigate(const GURL& url) {
   TRACE_EVENT0("cobalt::browser", "BrowserModule::Navigate()");
+  // Reset the waitable event regardless of the thread. This ensures that the
+  // webdriver won't incorrectly believe that the webmodule has finished loading
+  // when it calls Navigate() and waits for the |web_module_loaded_| signal.
   web_module_loaded_.Reset();
 
-  // Always post this as a task in case this is being called from the WebModule.
-  self_message_loop_->PostTask(
-      FROM_HERE, base::Bind(&BrowserModule::NavigateInternal, weak_this_, url));
-}
-
-void BrowserModule::Reload() {
-  TRACE_EVENT0("cobalt::browser", "BrowserModule::Reload()");
-  DCHECK_EQ(MessageLoop::current(), self_message_loop_);
-  DCHECK(web_module_);
-  web_module_->ExecuteJavascript(
-      "location.reload();",
-      base::SourceLocation("[object BrowserModule]", 1, 1),
-      NULL /* output: succeeded */);
-}
-
-#if SB_HAS(CORE_DUMP_HANDLER_SUPPORT)
-// static
-void BrowserModule::CoreDumpHandler(void* browser_module_as_void) {
-  BrowserModule* browser_module =
-      static_cast<BrowserModule*>(browser_module_as_void);
-  SbCoreDumpLogInteger("BrowserModule.on_error_triggered_count_",
-                       browser_module->on_error_triggered_count_);
-#if defined(COBALT_CHECK_RENDER_TIMEOUT)
-  SbCoreDumpLogInteger("BrowserModule.recovery_mechanism_triggered_count_",
-                       browser_module->recovery_mechanism_triggered_count_);
-  SbCoreDumpLogInteger("BrowserModule.timeout_response_trigger_count_",
-                       browser_module->timeout_response_trigger_count_);
-#endif  // defined(COBALT_CHECK_RENDER_TIMEOUT)
-}
-#endif  // SB_HAS(CORE_DUMP_HANDLER_SUPPORT)
-
-void BrowserModule::NavigateInternal(const GURL& url) {
-  TRACE_EVENT0("cobalt::browser", "BrowserModule::NavigateInternal()");
-  DCHECK_EQ(MessageLoop::current(), self_message_loop_);
+  // Repost to our own message loop if necessary.
+  if (MessageLoop::current() != self_message_loop_) {
+    self_message_loop_->PostTask(
+        FROM_HERE, base::Bind(&BrowserModule::Navigate, weak_this_, url));
+    return;
+  }
 
   // First try the registered handlers (e.g. for h5vcc://). If one of these
   // handles the URL, we don't use the web module.
@@ -386,6 +368,11 @@
     return;
   }
 
+  on_error_url_.clear();
+  if (on_error_retry_timer_.IsRunning()) {
+    on_error_retry_timer_.Stop();
+  }
+
   // Destroy old WebModule first, so we don't get a memory high-watermark after
   // the second WebModule's constructor runs, but before scoped_ptr::reset() is
   // run.
@@ -402,12 +389,15 @@
   const math::Size& viewport_size = GetViewportSize();
 
   DestroySplashScreen();
+  navigation_produced_main_render_tree_ = false;
   base::optional<std::string> key = SplashScreenCache::GetKeyForStartUrl(url);
   if (fallback_splash_screen_url_ ||
       (key && splash_screen_cache_->IsSplashScreenCached(*key))) {
     // Create the splash screen layer.
-    splash_screen_layer_ =
-        render_tree_combiner_->CreateLayer(kSplashScreenZIndex);
+    if (render_tree_combiner_) {
+      splash_screen_layer_ =
+          render_tree_combiner_->CreateLayer(kSplashScreenZIndex);
+    }
 
     splash_screen_.reset(new SplashScreen(
         application_state_,
@@ -470,6 +460,32 @@
   }
 }
 
+void BrowserModule::Reload() {
+  TRACE_EVENT0("cobalt::browser", "BrowserModule::Reload()");
+  DCHECK_EQ(MessageLoop::current(), self_message_loop_);
+  DCHECK(web_module_);
+  web_module_->ExecuteJavascript(
+      "location.reload();",
+      base::SourceLocation("[object BrowserModule]", 1, 1),
+      NULL /* output: succeeded */);
+}
+
+#if SB_HAS(CORE_DUMP_HANDLER_SUPPORT)
+// static
+void BrowserModule::CoreDumpHandler(void* browser_module_as_void) {
+  BrowserModule* browser_module =
+      static_cast<BrowserModule*>(browser_module_as_void);
+  SbCoreDumpLogInteger("BrowserModule.on_error_triggered_count_",
+                       browser_module->on_error_triggered_count_);
+#if defined(COBALT_CHECK_RENDER_TIMEOUT)
+  SbCoreDumpLogInteger("BrowserModule.recovery_mechanism_triggered_count_",
+                       browser_module->recovery_mechanism_triggered_count_);
+  SbCoreDumpLogInteger("BrowserModule.timeout_response_trigger_count_",
+                       browser_module->timeout_response_trigger_count_);
+#endif  // defined(COBALT_CHECK_RENDER_TIMEOUT)
+}
+#endif  // SB_HAS(CORE_DUMP_HANDLER_SUPPORT)
+
 void BrowserModule::OnLoad() {
   TRACE_EVENT0("cobalt::browser", "BrowserModule::OnLoad()");
   // Repost to our own message loop if necessary. This also prevents
@@ -484,6 +500,9 @@
   // changed unless the corresponding benchmark logic is changed as well.
   LOG(INFO) << "Loaded WebModule";
 
+  // Clear |on_error_retry_count_| after a successful load.
+  on_error_retry_count_ = 0;
+
   on_load_event_time_ = base::TimeTicks::Now().ToInternalValue();
   web_module_loaded_.Signal();
 }
@@ -541,15 +560,16 @@
   TRACE_EVENT0("cobalt::browser", "BrowserModule::OnRenderTreeProduced()");
   DCHECK_EQ(MessageLoop::current(), self_message_loop_);
 
-  if (splash_screen_ && !produced_render_tree_) {
+  if (splash_screen_ && !navigation_produced_main_render_tree_) {
     splash_screen_->Shutdown();
   }
-  produced_render_tree_ = true;
-
   if (application_state_ == base::kApplicationStatePreloading ||
       !render_tree_combiner_ || !main_web_module_layer_) {
     return;
   }
+
+  navigation_produced_main_render_tree_ = true;
+
   renderer::Submission renderer_submission(layout_results.render_tree,
                                            layout_results.layout_time);
   renderer_submission.on_rasterized_callback = base::Bind(
@@ -760,16 +780,44 @@
 
 void BrowserModule::OnError(const GURL& url, const std::string& error) {
   TRACE_EVENT0("cobalt::browser", "BrowserModule::OnError()");
+  if (MessageLoop::current() != self_message_loop_) {
+    self_message_loop_->PostTask(
+        FROM_HERE, base::Bind(&BrowserModule::OnError, weak_this_, url, error));
+    return;
+  }
+
 #if SB_HAS(CORE_DUMP_HANDLER_SUPPORT)
   on_error_triggered_count_++;
 #endif
-  LOG(ERROR) << error;
-  std::string url_string = "h5vcc://network-failure";
 
-  // Retry the current URL.
-  url_string += "?retry-url=" + url.spec();
+  on_error_url_ = url.spec();
 
-  Navigate(GURL(url_string));
+  // Start the OnErrorRetry() timer if it isn't already running.
+  // The minimum delay between calls to OnErrorRetry() exponentially grows as
+  // |on_error_retry_count_| increases. |on_error_retry_count_| is reset when
+  // OnLoad() is called.
+  if (!on_error_retry_timer_.IsRunning()) {
+    const int64 kBaseRetryDelayInMilliseconds = 1000;
+    // Cap the max error shift at 10 (1024 * kBaseDelayInMilliseconds)
+    // This results in the minimum delay being capped at ~17 minutes.
+    const int kMaxOnErrorRetryCountShift = 10;
+    int64 min_delay = kBaseRetryDelayInMilliseconds << std::min(
+                          kMaxOnErrorRetryCountShift, on_error_retry_count_);
+    int64 required_delay = std::max(
+        min_delay -
+            (base::TimeTicks::Now() - on_error_retry_time_).InMilliseconds(),
+        static_cast<int64>(0));
+
+    on_error_retry_timer_.Start(
+        FROM_HERE, base::TimeDelta::FromMilliseconds(required_delay), this,
+        &BrowserModule::OnErrorRetry);
+  }
+}
+
+void BrowserModule::OnErrorRetry() {
+  ++on_error_retry_count_;
+  on_error_retry_time_ = base::TimeTicks::Now();
+  TryURLHandlers(GURL("h5vcc://network-failure?retry-url=" + on_error_url_));
 }
 
 bool BrowserModule::FilterKeyEvent(base::Token type,
@@ -1091,9 +1139,6 @@
 #if defined(ENABLE_SCREENSHOT)
   screen_shot_writer_.reset(new ScreenShotWriter(renderer_module_->pipeline()));
 #endif  // defined(ENABLE_SCREENSHOT)
-  // TODO: Pass in dialog closure instead of system window, and initialize
-  // earlier.
-  h5vcc_url_handler_.reset(new H5vccURLHandler(this, system_window_.get()));
 
   media_module_ =
       media::MediaModule::Create(system_window_.get(), GetResourceProvider(),
@@ -1211,6 +1256,12 @@
     FOR_EACH_OBSERVER(LifecycleObserver, lifecycle_observers_,
                       Resume(resource_provider));
   }
+
+  // If no navigate has occurred since the last OnError call, then attempt to
+  // navigate to |on_error_url_| now.
+  if (!on_error_url_.empty()) {
+    Navigate(GURL(on_error_url_));
+  }
 }
 
 math::Size BrowserModule::GetViewportSize() {
diff --git a/src/cobalt/browser/browser_module.h b/src/cobalt/browser/browser_module.h
index 386eb0e..16961ec 100644
--- a/src/cobalt/browser/browser_module.h
+++ b/src/cobalt/browser/browser_module.h
@@ -23,6 +23,7 @@
 #include "base/synchronization/lock.h"
 #include "base/synchronization/waitable_event.h"
 #include "base/threading/thread.h"
+#include "base/timer.h"
 #include "cobalt/account/account_manager.h"
 #include "cobalt/base/application_state.h"
 #include "cobalt/base/message_queue.h"
@@ -34,6 +35,7 @@
 #include "cobalt/browser/screen_shot_writer.h"
 #include "cobalt/browser/splash_screen.h"
 #include "cobalt/browser/suspend_fuzzer.h"
+#include "cobalt/browser/system_platform_error_handler.h"
 #include "cobalt/browser/url_handler.h"
 #include "cobalt/browser/web_module.h"
 #include "cobalt/dom/array_buffer.h"
@@ -107,6 +109,10 @@
   // Reloads web module.
   void Reload();
 
+  SystemPlatformErrorHandler* system_platform_error_handler() {
+    return &system_platform_error_handler_;
+  }
+
   // Adds/removes a URL handler.
   void AddURLHandler(const URLHandler::URLHandlerCallback& callback);
   void RemoveURLHandler(const URLHandler::URLHandlerCallback& callback);
@@ -159,9 +165,6 @@
 #endif  // defined(COBALT_CHECK_RENDER_TIMEOUT)
 #endif  // SB_HAS(CORE_DUMP_HANDLER_SUPPORT)
 
-  // Recreates web module with the given URL.
-  void NavigateInternal(const GURL& url);
-
   // Called when the WebModule's Window.onload event is fired.
   void OnLoad();
 
@@ -214,6 +217,10 @@
   // Error callback for any error that stops the program.
   void OnError(const GURL& url, const std::string& error);
 
+  // OnErrorRetry() runs a retry URL through the URL handlers. It should only be
+  // called by |on_error_retry_timer_|.
+  void OnErrorRetry();
+
   // Filters a key event.
   // Returns true if the event should be passed on to other handlers,
   // false if it was consumed within this function.
@@ -328,6 +335,9 @@
   // The browser module runs on this message loop.
   MessageLoop* const self_message_loop_;
 
+  // Handler for system errors, which is owned by browser module.
+  SystemPlatformErrorHandler system_platform_error_handler_;
+
   // Collection of URL handlers that can potentially handle a URL before
   // using it to initialize a new WebModule.
   URLHandlerCollection url_handlers_;
@@ -448,6 +458,20 @@
   int render_timeout_count_;
 #endif
 
+  // The URL associated with the last OnError() call. It is cleared on the next
+  // call to Navigate().
+  std::string on_error_url_;
+  // The number of OnErrorRetry() calls that have occurred since the last
+  // OnDone() call. This is used to determine the exponential backoff delay
+  // between the call to OnError() and the timer call to OnErrorRetry().
+  int on_error_retry_count_;
+  // The time OnErrorRetry() was last called. This is used to limit how
+  // frequently |on_error_retry_timer_| can call OnErrorRetry().
+  base::TimeTicks on_error_retry_time_;
+  // The timer for the next call to OnErrorRetry(). It is started in OnError()
+  // when it is not already active.
+  base::OneShotTimer<BrowserModule> on_error_retry_timer_;
+
   // Set when the application is about to quit. May be set from a thread other
   // than the one hosting this object, and read from another.
   bool will_quit_;
@@ -474,8 +498,9 @@
   // The splash screen cache.
   scoped_ptr<SplashScreenCache> splash_screen_cache_;
 
-  // Whether or not the main WebModule has produced any render trees yet.
-  bool produced_render_tree_;
+  // Whether or not the main WebModule has produced any render trees yet for the
+  // current navigation.
+  bool navigation_produced_main_render_tree_;
 };
 
 }  // namespace browser
diff --git a/src/cobalt/browser/h5vcc_url_handler.cc b/src/cobalt/browser/h5vcc_url_handler.cc
index 94a8ed9..d02a4a3 100644
--- a/src/cobalt/browser/h5vcc_url_handler.cc
+++ b/src/cobalt/browser/h5vcc_url_handler.cc
@@ -18,6 +18,7 @@
 
 #include "base/bind.h"
 #include "cobalt/browser/browser_module.h"
+#include "cobalt/browser/system_platform_error_handler.h"
 
 namespace cobalt {
 namespace browser {
@@ -75,12 +76,10 @@
 const char kRetryParam[] = "retry-url";
 }  // namespace
 
-H5vccURLHandler::H5vccURLHandler(BrowserModule* browser_module,
-                                 system_window::SystemWindow* system_window)
+H5vccURLHandler::H5vccURLHandler(BrowserModule* browser_module)
     : ALLOW_THIS_IN_INITIALIZER_LIST(URLHandler(
           browser_module,
-          base::Bind(&H5vccURLHandler::HandleURL, base::Unretained(this)))),
-      system_window_(system_window) {}
+          base::Bind(&H5vccURLHandler::HandleURL, base::Unretained(this)))) {}
 
 bool H5vccURLHandler::HandleURL(const GURL& url) {
   bool was_handled = false;
@@ -97,20 +96,21 @@
 }
 
 bool H5vccURLHandler::HandleNetworkFailure() {
-  system_window::SystemWindow::DialogOptions dialog_options;
-  dialog_options.message_code =
-      system_window::SystemWindow::kDialogConnectionError;
-  dialog_options.callback = base::Bind(
-      &H5vccURLHandler::OnNetworkFailureDialogResponse, base::Unretained(this));
-  system_window_->ShowDialog(dialog_options);
+  SystemPlatformErrorHandler::SystemPlatformErrorOptions options;
+  options.error_type = kSbSystemPlatformErrorTypeConnectionError;
+  options.callback =
+      base::Bind(&H5vccURLHandler::OnNetworkFailureSystemPlatformResponse,
+                 base::Unretained(this));
+  browser_module()->system_platform_error_handler()->RaiseSystemPlatformError(
+      options);
   return true;
 }
 
-void H5vccURLHandler::OnNetworkFailureDialogResponse(
-    system_window::SystemWindow::DialogResponse response) {
+void H5vccURLHandler::OnNetworkFailureSystemPlatformResponse(
+    SbSystemPlatformErrorResponse response) {
   const std::string retry_url = GetH5vccUrlQueryParam(url_, kRetryParam);
   // A positive response means we should retry.
-  if (response == system_window::SystemWindow::kDialogPositiveResponse &&
+  if (response == kSbSystemPlatformErrorResponsePositive &&
       retry_url.length() > 0) {
     GURL url(retry_url);
     if (url.is_valid()) {
diff --git a/src/cobalt/browser/h5vcc_url_handler.h b/src/cobalt/browser/h5vcc_url_handler.h
index 203e4a2..6fa46a7 100644
--- a/src/cobalt/browser/h5vcc_url_handler.h
+++ b/src/cobalt/browser/h5vcc_url_handler.h
@@ -17,7 +17,6 @@
 
 #include "cobalt/account/account_manager.h"
 #include "cobalt/browser/url_handler.h"
-#include "cobalt/system_window/system_window.h"
 
 namespace cobalt {
 namespace browser {
@@ -27,19 +26,16 @@
 // handled separately, e.g. by showing a system dialog.
 class H5vccURLHandler : public URLHandler {
  public:
-  explicit H5vccURLHandler(BrowserModule* browser_module,
-                           system_window::SystemWindow* system_window);
+  explicit H5vccURLHandler(BrowserModule* browser_module);
   ~H5vccURLHandler() {}
 
  private:
   bool HandleURL(const GURL& url);
   bool HandleNetworkFailure();
 
-  // Dialog response handlers.
-  void OnNetworkFailureDialogResponse(
-      system_window::SystemWindow::DialogResponse response);
+  void OnNetworkFailureSystemPlatformResponse(
+      SbSystemPlatformErrorResponse response);
 
-  system_window::SystemWindow* system_window_;
   GURL url_;
 };
 
diff --git a/src/cobalt/browser/memory_settings/auto_mem.cc b/src/cobalt/browser/memory_settings/auto_mem.cc
index 2d5df3a..855e8ad 100644
--- a/src/cobalt/browser/memory_settings/auto_mem.cc
+++ b/src/cobalt/browser/memory_settings/auto_mem.cc
@@ -121,7 +121,8 @@
     return setting.Pass();
   }
 
-  setting->set_value(MemorySetting::kStarboardAPI, -1);
+  // This will mark the value as invalid.
+  setting->set_value(MemorySetting::kUnset, -1);
   return setting.Pass();
 }
 
diff --git a/src/cobalt/browser/memory_settings/auto_mem_test.cc b/src/cobalt/browser/memory_settings/auto_mem_test.cc
index 9982a6f..3d6c0dc 100644
--- a/src/cobalt/browser/memory_settings/auto_mem_test.cc
+++ b/src/cobalt/browser/memory_settings/auto_mem_test.cc
@@ -432,6 +432,18 @@
   EXPECT_LE(memory_consumption, kSmallEngineGpuMemorySize);
 }
 
+// Tests that if the gpu memory could not be queried then the resulting
+// max_gpu_bytes will not be valid.
+TEST(AutoMem, NoDefaultGpuMemory) {
+  AutoMemSettings command_line_settings(AutoMemSettings::kTypeCommandLine);
+  AutoMemSettings build_settings(AutoMemSettings::kTypeBuild);
+
+  AutoMem auto_mem(kResolution1080p, command_line_settings,
+                   build_settings);
+
+  EXPECT_FALSE(auto_mem.max_gpu_bytes()->valid());
+}
+
 }  // namespace memory_settings
 }  // namespace browser
 }  // namespace cobalt
diff --git a/src/cobalt/browser/render_tree_combiner.cc b/src/cobalt/browser/render_tree_combiner.cc
index 78fe4be..74727ec 100644
--- a/src/cobalt/browser/render_tree_combiner.cc
+++ b/src/cobalt/browser/render_tree_combiner.cc
@@ -35,6 +35,7 @@
 RenderTreeCombiner::Layer::~Layer() {
   DCHECK(render_tree_combiner_);
   render_tree_combiner_->RemoveLayer(this);
+  render_tree_combiner_->SubmitToRenderer();
 }
 
 void RenderTreeCombiner::Layer::Submit(
diff --git a/src/cobalt/browser/system_platform_error_handler.cc b/src/cobalt/browser/system_platform_error_handler.cc
new file mode 100644
index 0000000..7c1ef7d
--- /dev/null
+++ b/src/cobalt/browser/system_platform_error_handler.cc
@@ -0,0 +1,54 @@
+// Copyright 2017 Google Inc. All Rights Reserved.
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+//     http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include "cobalt/browser/system_platform_error_handler.h"
+
+#include "base/logging.h"
+
+namespace cobalt {
+namespace browser {
+namespace {
+
+void OnSystemPlatformErrorResponse(SbSystemPlatformErrorResponse response,
+                                   void* user_data) {
+  DCHECK(user_data);
+  SystemPlatformErrorHandler* error_handler =
+      static_cast<SystemPlatformErrorHandler*>(user_data);
+  error_handler->HandleSystemPlatformErrorResponse(response);
+}
+
+}  // namespace
+
+void SystemPlatformErrorHandler::RaiseSystemPlatformError(
+    const SystemPlatformErrorOptions& options) {
+  current_system_platform_error_callback_ = options.callback;
+
+  SbSystemPlatformError handle = SbSystemRaisePlatformError(
+      options.error_type, OnSystemPlatformErrorResponse, this);
+  if (!SbSystemPlatformErrorIsValid(handle) &&
+      !current_system_platform_error_callback_.is_null()) {
+    DLOG(WARNING) << "Did not handle error: " << options.error_type;
+    current_system_platform_error_callback_.Reset();
+  }
+}
+
+void SystemPlatformErrorHandler::HandleSystemPlatformErrorResponse(
+    SbSystemPlatformErrorResponse response) {
+  DCHECK(!current_system_platform_error_callback_.is_null());
+  current_system_platform_error_callback_.Run(response);
+  current_system_platform_error_callback_.Reset();
+}
+
+}  // namespace browser
+}  // namespace cobalt
diff --git a/src/cobalt/browser/system_platform_error_handler.h b/src/cobalt/browser/system_platform_error_handler.h
new file mode 100644
index 0000000..64b01a2
--- /dev/null
+++ b/src/cobalt/browser/system_platform_error_handler.h
@@ -0,0 +1,55 @@
+// Copyright 2017 Google Inc. All Rights Reserved.
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+//     http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef COBALT_BROWSER_SYSTEM_PLATFORM_ERROR_HANDLER_H_
+#define COBALT_BROWSER_SYSTEM_PLATFORM_ERROR_HANDLER_H_
+
+#include "base/callback.h"
+#include "starboard/system.h"
+
+namespace cobalt {
+namespace browser {
+
+// This class handles raising system errors to the platform and returning the
+// response to the caller via the provided callback.
+class SystemPlatformErrorHandler {
+ public:
+  // Type of callback to run when the platform responds to a system error.
+  typedef base::Callback<void(SbSystemPlatformErrorResponse response)>
+      SystemPlatformErrorCallback;
+
+  // Options structure for raised system platform errors, including both the
+  // system error being raised and the callback to run with the response.
+  struct SystemPlatformErrorOptions {
+    SbSystemPlatformErrorType error_type;
+    SystemPlatformErrorCallback callback;
+  };
+
+  // Raises a system error with the specified error type and callback.
+  void RaiseSystemPlatformError(const SystemPlatformErrorOptions& options);
+
+  // Called when the platform responds to the system error.
+  void HandleSystemPlatformErrorResponse(
+      SbSystemPlatformErrorResponse response);
+
+ private:
+  // The current platform error callback. Only one platform error callback may
+  // be active at a time.
+  SystemPlatformErrorCallback current_system_platform_error_callback_;
+};
+
+}  // namespace browser
+}  // namespace cobalt
+
+#endif  // COBALT_BROWSER_SYSTEM_PLATFORM_ERROR_HANDLER_H_
diff --git a/src/cobalt/build/build.id b/src/cobalt/build/build.id
index 248bc70..96e2ce3 100644
--- a/src/cobalt/build/build.id
+++ b/src/cobalt/build/build.id
@@ -1 +1 @@
-93830
\ No newline at end of file
+95615
\ No newline at end of file
diff --git a/src/cobalt/build/config/base.gni b/src/cobalt/build/config/base.gni
index 21e6bef..2bc87b6 100644
--- a/src/cobalt/build/config/base.gni
+++ b/src/cobalt/build/config/base.gni
@@ -152,7 +152,7 @@
 #   'stub'        -- Stub graphics rasterization.  A rasterizer object will
 #                    still be available and valid, but it will do nothing.
 if (!defined(rasterizer_type)) {
-  rasterizer_type = "hardware"
+  rasterizer_type = "direct-gles"
 }
 
 # If set to true, will enable support for rendering only the regions of the
diff --git a/src/cobalt/build/config/base.gypi b/src/cobalt/build/config/base.gypi
index 02c5ec2..110e70d 100644
--- a/src/cobalt/build/config/base.gypi
+++ b/src/cobalt/build/config/base.gypi
@@ -174,7 +174,7 @@
     #                    output window.
     #   'stub'        -- Stub graphics rasterization.  A rasterizer object will
     #                    still be available and valid, but it will do nothing.
-    'rasterizer_type%': 'hardware',
+    'rasterizer_type%': 'direct-gles',
 
     # If set to 1, will enable support for rendering only the regions of the
     # display that are modified due to animations, instead of re-rendering the
diff --git a/src/cobalt/dom/document.cc b/src/cobalt/dom/document.cc
index ec42f43..85603a2 100644
--- a/src/cobalt/dom/document.cc
+++ b/src/cobalt/dom/document.cc
@@ -81,6 +81,8 @@
 #if defined(ENABLE_PARTIAL_LAYOUT_CONTROL)
       partial_layout_is_enabled_(true),
 #endif  // defined(ENABLE_PARTIAL_LAYOUT_CONTROL)
+      selector_tree_(new cssom::SelectorTree()),
+      should_recreate_selector_tree_(false),
       navigation_start_clock_(options.navigation_start_clock
                                   ? options.navigation_start_clock
                                   : new base::SystemMonotonicClock()),
@@ -550,28 +552,30 @@
 namespace {
 
 void RemoveRulesFromCSSRuleListFromSelectorTree(
-    cssom::SelectorTree* selector_tree,
-    const scoped_refptr<cssom::CSSRuleList>& css_rule_list) {
+    const scoped_refptr<cssom::CSSRuleList>& css_rule_list,
+    cssom::SelectorTree* maybe_selector_tree) {
   for (unsigned int i = 0; i < css_rule_list->length(); ++i) {
     cssom::CSSRule* rule = css_rule_list->Item(i);
 
     cssom::CSSStyleRule* css_style_rule = rule->AsCSSStyleRule();
     if (css_style_rule && css_style_rule->added_to_selector_tree()) {
-      selector_tree->RemoveRule(css_style_rule);
+      if (maybe_selector_tree) {
+        maybe_selector_tree->RemoveRule(css_style_rule);
+      }
       css_style_rule->set_added_to_selector_tree(false);
     }
 
     cssom::CSSMediaRule* css_media_rule = rule->AsCSSMediaRule();
     if (css_media_rule) {
-      RemoveRulesFromCSSRuleListFromSelectorTree(selector_tree,
-                                                 css_media_rule->css_rules());
+      RemoveRulesFromCSSRuleListFromSelectorTree(css_media_rule->css_rules(),
+                                                 maybe_selector_tree);
     }
   }
 }
 
 void AppendRulesFromCSSRuleListToSelectorTree(
-    cssom::SelectorTree* selector_tree,
-    const scoped_refptr<cssom::CSSRuleList>& css_rule_list) {
+    const scoped_refptr<cssom::CSSRuleList>& css_rule_list,
+    cssom::SelectorTree* selector_tree) {
   for (unsigned int i = 0; i < css_rule_list->length(); ++i) {
     cssom::CSSRule* rule = css_rule_list->Item(i);
 
@@ -584,21 +588,26 @@
     cssom::CSSMediaRule* css_media_rule = rule->AsCSSMediaRule();
     if (css_media_rule) {
       if (css_media_rule->condition_value()) {
-        AppendRulesFromCSSRuleListToSelectorTree(selector_tree,
-                                                 css_media_rule->css_rules());
+        AppendRulesFromCSSRuleListToSelectorTree(css_media_rule->css_rules(),
+                                                 selector_tree);
       } else {
-        RemoveRulesFromCSSRuleListFromSelectorTree(selector_tree,
-                                                   css_media_rule->css_rules());
+        RemoveRulesFromCSSRuleListFromSelectorTree(css_media_rule->css_rules(),
+                                                   selector_tree);
       }
     }
   }
 }
 
-void UpdateSelectorTreeFromCSSStyleSheet(
-    cssom::SelectorTree* selector_tree,
+void AppendRulesFromCSSStyleSheetToSelectorTree(
+    const scoped_refptr<cssom::CSSStyleSheet>& style_sheet,
+    cssom::SelectorTree* selector_tree) {
+  AppendRulesFromCSSRuleListToSelectorTree(style_sheet->css_rules(),
+                                           selector_tree);
+}
+
+void ClearAddedToSelectorTreeFromCSSStyleSheetRules(
     const scoped_refptr<cssom::CSSStyleSheet>& style_sheet) {
-  AppendRulesFromCSSRuleListToSelectorTree(selector_tree,
-                                           style_sheet->css_rules());
+  RemoveRulesFromCSSRuleListFromSelectorTree(style_sheet->css_rules(), NULL);
 }
 
 }  // namespace
@@ -789,16 +798,37 @@
     UpdateStyleSheets();
     UpdateMediaRules();
 
+    // If the selector tree is being recreated, then clear the added state from
+    // the document's style sheets. This will cause them to be added to the new
+    // selector tree.
+    if (should_recreate_selector_tree_) {
+      DLOG(WARNING) << "A style sheet was removed from the document or the "
+                       "document's style sheets have been reordered. This "
+                       "triggers a recreation of the selector tree and should "
+                       "be avoided if possible.";
+      if (user_agent_style_sheet_) {
+        ClearAddedToSelectorTreeFromCSSStyleSheetRules(user_agent_style_sheet_);
+      }
+      for (unsigned int style_sheet_index = 0;
+           style_sheet_index < style_sheets_->length(); ++style_sheet_index) {
+        scoped_refptr<cssom::CSSStyleSheet> css_style_sheet =
+            style_sheets_->Item(style_sheet_index)->AsCSSStyleSheet();
+        ClearAddedToSelectorTreeFromCSSStyleSheetRules(css_style_sheet);
+      }
+      selector_tree_.reset(new cssom::SelectorTree());
+      should_recreate_selector_tree_ = false;
+    }
+
     if (user_agent_style_sheet_) {
-      UpdateSelectorTreeFromCSSStyleSheet(&selector_tree_,
-                                          user_agent_style_sheet_);
+      AppendRulesFromCSSStyleSheetToSelectorTree(user_agent_style_sheet_,
+                                                 selector_tree_.get());
     }
     for (unsigned int style_sheet_index = 0;
          style_sheet_index < style_sheets_->length(); ++style_sheet_index) {
       scoped_refptr<cssom::CSSStyleSheet> css_style_sheet =
           style_sheets_->Item(style_sheet_index)->AsCSSStyleSheet();
-
-      UpdateSelectorTreeFromCSSStyleSheet(&selector_tree_, css_style_sheet);
+      AppendRulesFromCSSStyleSheetToSelectorTree(css_style_sheet,
+                                                 selector_tree_.get());
     }
     scoped_refptr<HTMLHtmlElement> current_html = html();
     if (current_html) {
@@ -914,6 +944,22 @@
          child = child->next_element_sibling()) {
       child->CollectStyleSheetsOfElementAndDescendants(&style_sheet_vector);
     }
+
+    // Check for the removal or reordering of any of the pre-existing style
+    // sheets. In either of these cases, the selector tree must be recreated.
+    if (style_sheets_->length() > style_sheet_vector.size()) {
+      should_recreate_selector_tree_ = true;
+    }
+    for (unsigned int style_sheet_index = 0;
+         !should_recreate_selector_tree_ &&
+         style_sheet_index < style_sheets_->length();
+         ++style_sheet_index) {
+      if (style_sheets_->Item(style_sheet_index)->AsCSSStyleSheet().get() !=
+          style_sheet_vector[style_sheet_index]->AsCSSStyleSheet().get()) {
+        should_recreate_selector_tree_ = true;
+      }
+    }
+
     style_sheets_ = new cssom::StyleSheetList(style_sheet_vector, this);
     are_style_sheets_dirty_ = false;
   }
diff --git a/src/cobalt/dom/document.h b/src/cobalt/dom/document.h
index d0f8853..5291806 100644
--- a/src/cobalt/dom/document.h
+++ b/src/cobalt/dom/document.h
@@ -251,7 +251,7 @@
     return &scripts_to_be_executed_;
   }
 
-  cssom::SelectorTree* selector_tree() { return &selector_tree_; }
+  cssom::SelectorTree* selector_tree() { return selector_tree_.get(); }
 
   // Returns a mapping from keyframes name to CSSKeyframesRule.  This can be
   // used to quickly lookup the @keyframes rule given a string identifier.
@@ -465,7 +465,12 @@
   // List of document observers.
   ObserverList<DocumentObserver> observers_;
   // Selector Tree.
-  cssom::SelectorTree selector_tree_;
+  scoped_ptr<cssom::SelectorTree> selector_tree_;
+  // This is set when the document has a style sheet removed or the order of its
+  // style sheets changed. In this case, it is more straightforward to simply
+  // recreate the selector tree than to attempt to manage updating all of its
+  // internal state.
+  bool should_recreate_selector_tree_;
   // The document's latest sample from the global clock, used for updating
   // animations.
   const scoped_refptr<base::Clock> navigation_start_clock_;
diff --git a/src/cobalt/dom/html_element.cc b/src/cobalt/dom/html_element.cc
index a539d68..6aff330 100644
--- a/src/cobalt/dom/html_element.cc
+++ b/src/cobalt/dom/html_element.cc
@@ -588,26 +588,28 @@
 }
 
 void HTMLElement::InvalidateMatchingRulesRecursively() {
-  if (!matching_rules_valid_) {
-    return;
-  }
+  InvalidateMatchingRulesRecursivelyInternal(true /*is_initial_invalidation*/);
+}
 
-  matching_rules_valid_ = false;
+void HTMLElement::InvalidateMatchingRulesRecursivelyInternal(
+    bool is_initial_invalidation) {
+  if (matching_rules_valid_) {
+    // Move |matching_rules_| into |old_matching_rules_|. This is used for
+    // determining whether or not the matching rules actually changed when they
+    // are updated.
+    old_matching_rules_.swap(matching_rules_);
 
-  // Move |matching_rules_| into |old_matching_rules_|. This is used for
-  // determining whether or not the matching rules actually changed when they
-  // are updated.
-  old_matching_rules_.swap(matching_rules_);
-
-  matching_rules_.clear();
-  rule_matching_state_.matching_nodes.clear();
-  rule_matching_state_.descendant_potential_nodes.clear();
-  rule_matching_state_.following_sibling_potential_nodes.clear();
-  for (int pseudo_element_type = 0; pseudo_element_type < kMaxPseudoElementType;
-       ++pseudo_element_type) {
-    if (pseudo_elements_[pseudo_element_type]) {
-      pseudo_elements_[pseudo_element_type]->ClearMatchingRules();
+    matching_rules_.clear();
+    rule_matching_state_.matching_nodes.clear();
+    rule_matching_state_.descendant_potential_nodes.clear();
+    rule_matching_state_.following_sibling_potential_nodes.clear();
+    for (int pseudo_element_type = 0;
+         pseudo_element_type < kMaxPseudoElementType; ++pseudo_element_type) {
+      if (pseudo_elements_[pseudo_element_type]) {
+        pseudo_elements_[pseudo_element_type]->ClearMatchingRules();
+      }
     }
+    matching_rules_valid_ = false;
   }
 
   // Invalidate matching rules on all children.
@@ -615,18 +617,22 @@
        element = element->next_element_sibling()) {
     HTMLElement* html_element = element->AsHTMLElement();
     if (html_element) {
-      html_element->InvalidateMatchingRulesRecursively();
+      html_element->InvalidateMatchingRulesRecursivelyInternal(
+          false /*is_initial_invalidation*/);
     }
   }
 
-  // Invalidate matching rules on all following siblings if sibling combinators
-  // are used.
-  if (node_document()->selector_tree()->has_sibling_combinators()) {
+  // Invalidate matching rules on all following siblings if this is the initial
+  // invalidation and sibling combinators are used; if this is not the initial
+  // invalidation, then these will already be handled by a previous call.
+  if (is_initial_invalidation &&
+      node_document()->selector_tree()->has_sibling_combinators()) {
     for (Element* element = next_element_sibling(); element;
          element = element->next_element_sibling()) {
       HTMLElement* html_element = element->AsHTMLElement();
       if (html_element) {
-        html_element->InvalidateMatchingRulesRecursively();
+        html_element->InvalidateMatchingRulesRecursivelyInternal(
+            false /*is_initial_invalidation*/);
       }
     }
   }
@@ -1216,7 +1222,6 @@
   if (!matching_rules_valid_) {
     dom_stat_tracker_->OnUpdateMatchingRules();
     UpdateMatchingRules(this);
-    matching_rules_valid_ = true;
 
     // Check for whether the matching rules have changed. If they have, then a
     // new computed style must be generated from them.
diff --git a/src/cobalt/dom/html_element.h b/src/cobalt/dom/html_element.h
index 5806cfc..21fb9ee 100644
--- a/src/cobalt/dom/html_element.h
+++ b/src/cobalt/dom/html_element.h
@@ -190,8 +190,8 @@
   }
   // Returns the rule matching state of this element.
   RuleMatchingState* rule_matching_state() { return &rule_matching_state_; }
-  // Invalidates the matching rules and rule matching state in this element and
-  // its descendants.
+  // Invalidates the matching rules and rule matching state in this element,
+  // its descendants and its siblings.
   void InvalidateMatchingRulesRecursively();
 
   // Computed style related methods.
@@ -263,6 +263,7 @@
     return descendant_computed_styles_valid_;
   }
   bool matching_rules_valid() const { return matching_rules_valid_; }
+  void set_matching_rules_valid() { matching_rules_valid_ = true; }
 
   // Returns whether the element has been designated.
   //   https://www.w3.org/TR/selectors4/#hover-pseudo
@@ -317,6 +318,11 @@
   // directionality does not invalidate the computed style.
   void SetDirectionality(const std::string& value);
 
+  // Invalidates the matching rules and rule matching state in this element and
+  // its descendants. In the case where this is the the initial invalidation,
+  // it will also invalidate the rule matching state of its siblings.
+  void InvalidateMatchingRulesRecursivelyInternal(bool is_initial_invalidation);
+
   // Clear the list of active background images, and notify the animated image
   // tracker to stop the animations.
   void ClearActiveBackgroundImages();
@@ -374,12 +380,11 @@
   cssom::RulesWithCascadePrecedence old_matching_rules_;
   cssom::RulesWithCascadePrecedence matching_rules_;
   RuleMatchingState rule_matching_state_;
+  bool matching_rules_valid_;
 
   // This contains information about the boxes generated from the element.
   scoped_ptr<LayoutBoxes> layout_boxes_;
 
-  bool matching_rules_valid_;
-
   scoped_ptr<PseudoElement> pseudo_elements_[kMaxPseudoElementType];
   base::WeakPtr<DOMStringMap> dataset_;
 
diff --git a/src/cobalt/dom/html_media_element.cc b/src/cobalt/dom/html_media_element.cc
index 51200ba..779f7d3 100644
--- a/src/cobalt/dom/html_media_element.cc
+++ b/src/cobalt/dom/html_media_element.cc
@@ -1528,7 +1528,7 @@
   EndProcessingMediaPlayerCallback();
 }
 
-void HTMLMediaElement::TimeChanged(bool eos_played) {
+void HTMLMediaElement::TimeChanged() {
   DCHECK(player_);
   if (!player_) {
     return;
@@ -1554,9 +1554,8 @@
   // When the current playback position reaches the end of the media resource
   // when the direction of playback is forwards, then the user agent must follow
   // these steps:
-  eos_played |=
-      !SbDoubleIsNan(dur) && (0.0f != dur) && now >= dur && playback_rate_ > 0;
-  if (eos_played) {
+  if (!SbDoubleIsNan(dur) && (0.0f != dur) && now >= dur &&
+      playback_rate_ > 0) {
     // If the media element has a loop attribute specified and does not have a
     // current media controller,
     if (loop()) {
diff --git a/src/cobalt/dom/html_media_element.h b/src/cobalt/dom/html_media_element.h
index 31314e3..b3d1f0d 100644
--- a/src/cobalt/dom/html_media_element.h
+++ b/src/cobalt/dom/html_media_element.h
@@ -231,7 +231,7 @@
   // WebMediaPlayerClient methods
   void NetworkStateChanged() OVERRIDE;
   void ReadyStateChanged() OVERRIDE;
-  void TimeChanged(bool eos_played) OVERRIDE;
+  void TimeChanged() OVERRIDE;
   void DurationChanged() OVERRIDE;
   void OutputModeChanged() OVERRIDE;
   void PlaybackStateChanged() OVERRIDE;
diff --git a/src/cobalt/dom/rule_matching.cc b/src/cobalt/dom/rule_matching.cc
index 1437827..938d78f 100644
--- a/src/cobalt/dom/rule_matching.cc
+++ b/src/cobalt/dom/rule_matching.cc
@@ -683,6 +683,8 @@
       }
     }
   }
+
+  current_element->set_matching_rules_valid();
 }
 
 scoped_refptr<Element> QuerySelector(Node* node, const std::string& selectors,
diff --git a/src/cobalt/dom/rule_matching_test.cc b/src/cobalt/dom/rule_matching_test.cc
index 73f9fdb..6a562fc 100644
--- a/src/cobalt/dom/rule_matching_test.cc
+++ b/src/cobalt/dom/rule_matching_test.cc
@@ -244,11 +244,14 @@
   head_->set_inner_html("<style>:focus {}</style>");
   body_->set_inner_html("<div tabIndex=\"-1\"/>");
   body_->first_element_child()->AsHTMLElement()->Focus();
+  // This is required because RunFocusingSteps() earlies out as a result of the
+  // document not having a browsing context.
+  root_->InvalidateMatchingRulesRecursively();
   UpdateAllMatchingRules();
 
   cssom::RulesWithCascadePrecedence* matching_rules =
       body_->first_element_child()->AsHTMLElement()->matching_rules();
-  ASSERT_EQ(2, matching_rules->size());
+  ASSERT_EQ(1, matching_rules->size());
   EXPECT_EQ(GetDocumentStyleSheet(0)->css_rules()->Item(0),
             (*matching_rules)[0].first);
 }
@@ -705,5 +708,103 @@
   EXPECT_EQ(0, node_list->length());
 }
 
+TEST_F(RuleMatchingTest, StyleElementRemoval) {
+  head_->set_inner_html("<style>* {}</style>");
+  body_->set_inner_html("<div/>");
+  UpdateAllMatchingRules();
+  head_->set_inner_html("<style/>");
+  UpdateAllMatchingRules();
+
+  cssom::RulesWithCascadePrecedence* matching_rules =
+      body_->first_element_child()->AsHTMLElement()->matching_rules();
+  ASSERT_EQ(0, matching_rules->size());
+}
+
+TEST_F(RuleMatchingTest, StyleElementAddition) {
+  head_->set_inner_html("<style/>");
+  body_->set_inner_html("<div/>");
+  UpdateAllMatchingRules();
+  head_->set_inner_html("<style>* {}</style>");
+  UpdateAllMatchingRules();
+
+  cssom::RulesWithCascadePrecedence* matching_rules =
+      body_->first_element_child()->AsHTMLElement()->matching_rules();
+  ASSERT_EQ(1, matching_rules->size());
+}
+
+TEST_F(RuleMatchingTest, StyleElementReorderingOneMatching) {
+  scoped_refptr<HTMLElement> div1 =
+      document_->CreateElement("div")->AsHTMLElement();
+  div1->set_inner_html("<style/>");
+
+  scoped_refptr<HTMLElement> div2 =
+      document_->CreateElement("div")->AsHTMLElement();
+  div2->set_inner_html("<style>* {}</style>");
+
+  body_->set_inner_html("<div/>");
+  head_->AppendChild(div1);
+  head_->AppendChild(div2);
+
+  UpdateAllMatchingRules();
+
+  cssom::RulesWithCascadePrecedence* matching_rules =
+      head_->first_element_child()->AsHTMLElement()->matching_rules();
+  ASSERT_EQ(1, matching_rules->size());
+  EXPECT_NE(GetDocumentStyleSheet(0)->css_rules()->Item(0),
+            (*matching_rules)[0].first);
+  EXPECT_EQ(GetDocumentStyleSheet(1)->css_rules()->Item(0),
+            (*matching_rules)[0].first);
+
+  head_->RemoveChild(div2);
+  head_->InsertBefore(div2, div1);
+
+  UpdateAllMatchingRules();
+
+  matching_rules =
+      head_->first_element_child()->AsHTMLElement()->matching_rules();
+  ASSERT_EQ(1, matching_rules->size());
+  EXPECT_EQ(GetDocumentStyleSheet(0)->css_rules()->Item(0),
+            (*matching_rules)[0].first);
+  EXPECT_NE(GetDocumentStyleSheet(1)->css_rules()->Item(0),
+            (*matching_rules)[0].first);
+}
+
+TEST_F(RuleMatchingTest, StyleElementReorderingTwoMatching) {
+  scoped_refptr<HTMLElement> div1 =
+      document_->CreateElement("div")->AsHTMLElement();
+  div1->set_inner_html("<style>* {}</style>");
+
+  scoped_refptr<HTMLElement> div2 =
+      document_->CreateElement("div")->AsHTMLElement();
+  div2->set_inner_html("<style>* {}</style>");
+
+  body_->set_inner_html("<div/>");
+  head_->AppendChild(div1);
+  head_->AppendChild(div2);
+
+  UpdateAllMatchingRules();
+
+  cssom::RulesWithCascadePrecedence* matching_rules =
+      head_->first_element_child()->AsHTMLElement()->matching_rules();
+  ASSERT_EQ(2, matching_rules->size());
+  EXPECT_EQ(GetDocumentStyleSheet(0)->css_rules()->Item(0),
+            (*matching_rules)[0].first);
+  EXPECT_NE(GetDocumentStyleSheet(1)->css_rules()->Item(0),
+            (*matching_rules)[0].first);
+
+  head_->RemoveChild(div2);
+  head_->InsertBefore(div2, div1);
+
+  UpdateAllMatchingRules();
+
+  matching_rules =
+      head_->first_element_child()->AsHTMLElement()->matching_rules();
+  ASSERT_EQ(2, matching_rules->size());
+  EXPECT_EQ(GetDocumentStyleSheet(0)->css_rules()->Item(0),
+            (*matching_rules)[0].first);
+  EXPECT_NE(GetDocumentStyleSheet(1)->css_rules()->Item(0),
+            (*matching_rules)[0].first);
+}
+
 }  // namespace dom
 }  // namespace cobalt
diff --git a/src/cobalt/dom/window.cc b/src/cobalt/dom/window.cc
index fab5c8c..67cd077 100644
--- a/src/cobalt/dom/window.cc
+++ b/src/cobalt/dom/window.cc
@@ -16,6 +16,7 @@
 
 #include <algorithm>
 
+#include "base/base64.h"
 #include "base/bind.h"
 #include "base/bind_helpers.h"
 #include "cobalt/base/polymorphic_downcast.h"
@@ -288,6 +289,18 @@
 
 scoped_refptr<Crypto> Window::crypto() const { return crypto_; }
 
+std::string Window::Btoa(const std::string& string_to_encode) {
+  std::string output;
+  base::Base64Encode(string_to_encode, &output);
+  return output;
+}
+
+std::string Window::Atob(const std::string& encoded_string) {
+  std::string output;
+  base::Base64Decode(encoded_string, &output);
+  return output;
+}
+
 int Window::SetTimeout(const WindowTimers::TimerCallbackArg& handler,
                        int timeout) {
   DLOG_IF(WARNING, timeout < 0)
diff --git a/src/cobalt/dom/window.h b/src/cobalt/dom/window.h
index 9cd4cb6..0a60269 100644
--- a/src/cobalt/dom/window.h
+++ b/src/cobalt/dom/window.h
@@ -228,6 +228,11 @@
   //   https://www.w3.org/TR/WebCryptoAPI/#crypto-interface
   scoped_refptr<Crypto> crypto() const;
 
+  // base64 encoding and decoding
+  std::string Btoa(const std::string& string_to_encode);
+
+  std::string Atob(const std::string& encoded_string);
+
   // Web API: WindowTimers (implements)
   //   https://www.w3.org/TR/html5/webappapis.html#timers
   //
diff --git a/src/cobalt/dom/window.idl b/src/cobalt/dom/window.idl
index d04fb7a..8749653 100644
--- a/src/cobalt/dom/window.idl
+++ b/src/cobalt/dom/window.idl
@@ -36,6 +36,10 @@
   // the user agent
   readonly attribute Navigator navigator;
 
+  // base64 encoding and decoding
+  DOMString btoa(DOMString string_to_encode);
+  DOMString atob(DOMString encoded_string);
+
   // Custom, not in any spec.
   //
   readonly attribute Console console;
diff --git a/src/cobalt/layout/box.cc b/src/cobalt/layout/box.cc
index 34a3853..ff4edae 100644
--- a/src/cobalt/layout/box.cc
+++ b/src/cobalt/layout/box.cc
@@ -1638,20 +1638,21 @@
   }
 }
 
-void Box::ApplyTransformActionToCoordinate(TransformAction action,
+bool Box::ApplyTransformActionToCoordinate(TransformAction action,
                                            math::Vector2dF* coordinate) const {
   std::vector<math::Vector2dF> coordinate_vector;
   coordinate_vector.push_back(*coordinate);
-  ApplyTransformActionToCoordinates(action, &coordinate_vector);
+  bool result = ApplyTransformActionToCoordinates(action, &coordinate_vector);
   *coordinate = coordinate_vector[0];
+  return result;
 }
 
-void Box::ApplyTransformActionToCoordinates(
+bool Box::ApplyTransformActionToCoordinates(
     TransformAction action, std::vector<math::Vector2dF>* coordinates) const {
   const scoped_refptr<cssom::PropertyValue>& transform =
       computed_style()->transform();
   if (transform == cssom::KeywordValue::GetNone()) {
-    return;
+    return true;
   }
 
   // The border box offset is calculated in two steps because we want to
@@ -1673,6 +1674,11 @@
   if (!matrix.IsIdentity()) {
     if (action == kEnterTransform) {
       matrix = matrix.Inverse();
+      // The matrix is not invertible. Return that applying the transform
+      // failed.
+      if (matrix.IsZeros()) {
+        return false;
+      }
     }
 
     for (std::vector<math::Vector2dF>::iterator coordinate_iter =
@@ -1702,6 +1708,7 @@
       coordinate += containing_block_offset_from_root_as_float;
     }
   }
+  return true;
 }
 
 }  // namespace layout
diff --git a/src/cobalt/layout/box.h b/src/cobalt/layout/box.h
index 51c1758..e90a9f1 100644
--- a/src/cobalt/layout/box.h
+++ b/src/cobalt/layout/box.h
@@ -572,9 +572,11 @@
                               RenderSequence other_render_sequence);
 
   // Applies the specified transform action to the provided coordinates.
-  void ApplyTransformActionToCoordinate(TransformAction action,
+  // Returns false if the transform is not invertible and the action requires
+  // it being inverted.
+  bool ApplyTransformActionToCoordinate(TransformAction action,
                                         math::Vector2dF* coordinate) const;
-  void ApplyTransformActionToCoordinates(
+  bool ApplyTransformActionToCoordinates(
       TransformAction action, std::vector<math::Vector2dF>* coordinates) const;
 
  protected:
diff --git a/src/cobalt/layout/box_generator.cc b/src/cobalt/layout/box_generator.cc
index dc00344..18bf26b 100644
--- a/src/cobalt/layout/box_generator.cc
+++ b/src/cobalt/layout/box_generator.cc
@@ -17,6 +17,7 @@
 #include <string>
 
 #include "base/bind.h"
+#include "base/debug/trace_event.h"
 #include "base/memory/scoped_ptr.h"
 #include "cobalt/cssom/computed_style.h"
 #include "cobalt/cssom/css_computed_style_declaration.h"
@@ -64,6 +65,7 @@
 scoped_refptr<render_tree::Image> GetVideoFrame(
     const scoped_refptr<ShellVideoFrameProvider>& frame_provider,
     render_tree::ResourceProvider* resource_provider) {
+  TRACE_EVENT0("cobalt::layout", "GetVideoFrame()");
   SbDecodeTarget decode_target = frame_provider->GetCurrentSbDecodeTarget();
   if (SbDecodeTargetIsValid(decode_target)) {
 #if SB_HAS(GRAPHICS)
diff --git a/src/cobalt/layout/layout_manager.cc b/src/cobalt/layout/layout_manager.cc
index 7f0cc65..8f38f9e 100644
--- a/src/cobalt/layout/layout_manager.cc
+++ b/src/cobalt/layout/layout_manager.cc
@@ -116,9 +116,7 @@
 void UpdateCamera(
     float width_to_height_aspect_ratio, scoped_refptr<input::Camera3D> camera,
     float max_horizontal_fov_rad, float max_vertical_fov_rad,
-    render_tree::MatrixTransform3DNode::Builder* transform_node_builder,
-    base::TimeDelta time) {
-  UNREFERENCED_PARAMETER(time);
+    render_tree::MatrixTransform3DNode::Builder* transform_node_builder) {
   float vertical_fov_rad =
       std::min(max_vertical_fov_rad,
                2 * static_cast<float>(atan(tan(max_horizontal_fov_rad * 0.5f) /
diff --git a/src/cobalt/layout/replaced_box.cc b/src/cobalt/layout/replaced_box.cc
index c61322a..6c3e322 100644
--- a/src/cobalt/layout/replaced_box.cc
+++ b/src/cobalt/layout/replaced_box.cc
@@ -231,9 +231,7 @@
 }
 
 void AnimateVideoImage(const ReplacedBox::ReplaceImageCB& replace_image_cb,
-                       ImageNode::Builder* image_node_builder,
-                       base::TimeDelta time) {
-  UNREFERENCED_PARAMETER(time);
+                       ImageNode::Builder* image_node_builder) {
   DCHECK(!replace_image_cb.is_null());
   DCHECK(image_node_builder);
 
@@ -250,14 +248,27 @@
 void AnimateVideoWithLetterboxing(
     const ReplacedBox::ReplaceImageCB& replace_image_cb,
     math::SizeF destination_size,
-    CompositionNode::Builder* composition_node_builder, base::TimeDelta time) {
-  UNREFERENCED_PARAMETER(time);
-
+    CompositionNode::Builder* composition_node_builder) {
   DCHECK(!replace_image_cb.is_null());
   DCHECK(composition_node_builder);
 
   scoped_refptr<render_tree::Image> image = replace_image_cb.Run();
 
+  // If the image hasn't changed, then no need to change anything else.  The
+  // image should be the first child (see AddLetterboxedImageToRenderTree()).
+  if (!composition_node_builder->children().empty()) {
+    render_tree::ImageNode* existing_image_node =
+        base::polymorphic_downcast<render_tree::ImageNode*>(
+            composition_node_builder->GetChild(0)->get());
+    if (existing_image_node->data().source.get() == image.get()) {
+      return;
+    }
+  }
+
+  // Reset the composition node from whatever it was before, we will recreate
+  // it anew in each animation frame.
+  *composition_node_builder = CompositionNode::Builder();
+
   // TODO: Detect better when the intrinsic video size is used for the
   //   node size, and trigger a re-layout from the media element when the size
   //   changes.
diff --git a/src/cobalt/layout/topmost_event_target.cc b/src/cobalt/layout/topmost_event_target.cc
index 97e84fe..f825db7 100644
--- a/src/cobalt/layout/topmost_event_target.cc
+++ b/src/cobalt/layout/topmost_event_target.cc
@@ -81,8 +81,12 @@
   if (layout_boxes) {
     const Box* box = layout_boxes->boxes().front();
     if (box->computed_style() && box->IsTransformed()) {
-      box->ApplyTransformActionToCoordinate(Box::kEnterTransform,
-                                            &element_coordinate);
+      // Early out if the transform cannot be applied. This can occur if the
+      // transform matrix is not invertible.
+      if (!box->ApplyTransformActionToCoordinate(Box::kEnterTransform,
+                                                 &element_coordinate)) {
+        return;
+      }
     }
 
     scoped_refptr<dom::HTMLElement> html_element = element->AsHTMLElement();
diff --git a/src/cobalt/layout_tests/layout_tests.cc b/src/cobalt/layout_tests/layout_tests.cc
index 77b0def..f377d78 100644
--- a/src/cobalt/layout_tests/layout_tests.cc
+++ b/src/cobalt/layout_tests/layout_tests.cc
@@ -90,7 +90,7 @@
   scoped_refptr<render_tree::animations::AnimateNode> animate_node =
       new render_tree::animations::AnimateNode(layout_results.render_tree);
   scoped_refptr<render_tree::Node> animated_tree =
-      animate_node->Apply(layout_results.layout_time).animated;
+      animate_node->Apply(layout_results.layout_time).animated->source();
 
   bool results =
       pixel_tester.TestTree(animated_tree, GetParam().base_file_path);
diff --git a/src/cobalt/layout_tests/web_platform_tests.cc b/src/cobalt/layout_tests/web_platform_tests.cc
index 5421370..01788fd 100644
--- a/src/cobalt/layout_tests/web_platform_tests.cc
+++ b/src/cobalt/layout_tests/web_platform_tests.cc
@@ -281,6 +281,7 @@
   std::string json_results = RunWebPlatformTest(test_url, &got_results);
   EXPECT_TRUE(got_results);
   std::vector<TestResult> results = ParseResults(json_results);
+  bool failed_at_least_once = false;
   for (size_t i = 0; i < results.size(); ++i) {
     const WebPlatformTestInfo& test_info = GetParam();
     const TestResult& test_result = results[i];
@@ -290,6 +291,15 @@
     if (it != test_info.exceptions.end()) {
       should_pass = !should_pass;
     }
+    // If expected to fail but current subtest did not fail, wait to report
+    // the entire test failed after the last subtest passed and none failed.
+    if (!should_pass && test_result.status == WebPlatformTestInfo::kPass &&
+        (i != results.size() - 1 || failed_at_least_once)) {
+      should_pass = true;
+    } else {
+      failed_at_least_once = failed_at_least_once ||
+                             test_result.status == WebPlatformTestInfo::kFail;
+    }
     EXPECT_PRED_FORMAT2(CheckResult, should_pass, test_result);
   }
 }
diff --git a/src/cobalt/loader/embedded_resources/you_tube_logo.png b/src/cobalt/loader/embedded_resources/you_tube_logo.png
deleted file mode 100644
index 14563d0..0000000
--- a/src/cobalt/loader/embedded_resources/you_tube_logo.png
+++ /dev/null
Binary files differ
diff --git a/src/cobalt/loader/embedded_resources/youtube_splash_screen.css b/src/cobalt/loader/embedded_resources/youtube_splash_screen.css
deleted file mode 100644
index 81ff7cb..0000000
--- a/src/cobalt/loader/embedded_resources/youtube_splash_screen.css
+++ /dev/null
@@ -1,158 +0,0 @@
-/*
- * Copyright 2015 Google Inc. All Rights Reserved.
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- *     http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
-*/
-
-body {
-  overflow: hidden;
-  font-size: 1.4815vh;  /* Corresponds to 16px at 1080p. */
-}
-
-#splash {
-  background-color: #e62117;
-  background-image: url("h5vcc-embedded://you_tube_logo.png");
-  background-position: center center;
-  background-repeat: no-repeat;
-  background-size: 50%;
-  height: 100%;
-  left: 0;
-  position: absolute;
-  top: 0;
-  width: 100%;
-}
-
-#loading {
-  position: absolute;
-  top: 52em;
-  width: 100%;
-}
-
-#spinner {
-  /* The spinner starts with display set to none, and JavaScript will set this
-     to 'block' after some time has passed, if the splash screen is still
-     visible. */
-  display: none;
-
-  height: 5.33em;
-  margin: 0 auto;
-  position: relative;
-  width: 5.33em;
-}
-
-.dot {
-  background-color: #cbcbcb;
-  border-radius: 50%;
-  height: 1.17em;
-  position: absolute;
-  width: 1.17em;
-}
-
-@keyframes fade1 {
-  0%,100% {opacity: 0}
-  50% {opacity: 1}
-}
-
-@keyframes fade2 {
-  0%,100% {opacity: .25}
-  37.5% {opacity: 1}
-  87.5% {opacity: 0}
-}
-
-@keyframes fade3 {
-  0%,100% {opacity: .5}
-  25% {opacity: 1}
-  75% {opacity: 0}
-}
-
-@keyframes fade4 {
-  0%,100% {opacity: .75}
-  12.5% {opacity: 1}
-  62.5% {opacity: 0}
-}
-
-@keyframes fade5 {
-  0%,100% {opacity: 1}
-  50% {opacity: 0}
-}
-
-@keyframes fade6 {
-  0%,100% {opacity: .75}
-  37.5% {opacity: 0}
-  87.5% {opacity: 1}
-}
-
-@keyframes fade7 {
-  0%,100% {opacity: .5}
-  25% {opacity: 0}
-  75% {opacity: 1}
-}
-
-@keyframes fade8 {
-  0%,100% {opacity: .25}
-  12.5% {opacity: 0}
-  62.5% {opacity: 1}
-}
-
-#dot1 {
-  animation: fade8 .72s infinite ease;
-  left: 0;
-  top: 2.09em;
-}
-
-#dot2 {
-  animation: fade7 .72s infinite ease;
-  left: .61em;
-  top: .61em;
-}
-
-#dot3 {
-  animation: fade6 .72s infinite ease;
-  left: 2.09em;
-  top: 0;
-}
-
-#dot4 {
-  animation: fade5 .72s infinite ease;
-  right: .61em;
-  top: .61em;
-}
-
-#dot5 {
-  animation: fade4 .72s infinite ease;
-  right: 0;
-  top: 2.09em;
-}
-
-#dot6 {
-  animation: fade3 .72s infinite ease;
-  bottom: .61em;
-  right: .61em;
-}
-
-#dot7 {
-  animation: fade2 .72s infinite ease;
-  bottom: 0;
-  left: 2.09em;
-}
-
-#dot8 {
-  animation: fade1 .72s infinite ease;
-  bottom: .61em;
-  left: .61em;
-}
-
-.hidden {
-  height: 0;
-  visibility: hidden;
-}
diff --git a/src/cobalt/loader/embedded_resources/youtube_splash_screen.html b/src/cobalt/loader/embedded_resources/youtube_splash_screen.html
index d40bf9b..0c704e5 100644
--- a/src/cobalt/loader/embedded_resources/youtube_splash_screen.html
+++ b/src/cobalt/loader/embedded_resources/youtube_splash_screen.html
@@ -1,6 +1,6 @@
 <!DOCTYPE html>
 <!--
-  Copyright 2015 Google Inc. All Rights Reserved.
+  Copyright 2017 Google Inc. All Rights Reserved.
 
   Licensed under the Apache License, Version 2.0 (the "License");
   you may not use this file except in compliance with the License.
@@ -15,34 +15,205 @@
   limitations under the License.
 -->
 <html>
+  <head>
+      <meta http-equiv="Content-Security-Policy" content="default-src 'none';
+      script-src 'unsafe-inline';
+      style-src 'unsafe-inline';
+      img-src 'self' data:;">
+<style>
+#background {
+  background-color: #282828;
+  margin: 0;
+  height: 100vh;
+  width: 100vw;
+  transition: 200ms;
+  transition-delay: 500ms;
+}
+#lozengeContainer {
+  position: fixed;
+  z-index: 1;
+  top: 50%;
+  left: 25%;
+  transform: translate(0, -50%);
+  width: 33%;
+  transition: all 0.5s cubic-bezier(0.4, 0, 0.2, 1);
+  transition-delay: 0ms;
+}
+#lozengeContainer.move {
+  transform: translate(-51%, -50%);
 
-<head>
-  <meta http-equiv="Content-Security-Policy" content="default-src 'none';
-      script-src h5vcc-embedded://*/splash_screen.js;
-      style-src h5vcc-embedded://*/youtube_splash_screen.css;
-      img-src h5vcc-embedded://*/you_tube_logo.png;">
-<link rel="stylesheet" type="text/css"
-    href="h5vcc-embedded://youtube_splash_screen.css">
+}
+#lozengeBg {
+  position: absolute;
+  z-index: -1;
+  background-color: #282828;
+}
+
+
+#wordmarkContainer {
+  position: absolute;
+  z-index: -1;
+  top: 50%;
+  left: 25%;
+  transform: translate(0, -50%);
+  width: 33%;
+  transition: all 0.5s cubic-bezier(0.4, 0, 0.2, 1);
+  transition-delay: 0ms;
+}
+#wordmarkContainer.move {
+  transform: translate(54%, -50%);
+  z-index: 0;
+}
+#wordmark {
+  background-size: contain;
+}
+
+#spinnerContainer {
+  position: fixed;
+  top: 71.5%;
+  left: 50%;
+  width: 166px;
+  height: 166px;
+  opacity: .2;
+  margin: 0 0 0 -83px;
+  display: none;
+}
+
+#spinnerContainer.active {
+  display: block;
+  animation: loading 5s linear infinite;
+}
+
+#loader {
+  border-radius: 50%;
+  color: #ffffff;
+  position: relative;
+  width: 166px;
+  height: 166px;
+  box-shadow: inset 0 0 0 15px;
+}
+
+.beforeafter {
+  background-color: #282828;
+  position: absolute;
+  top: -1px;
+  width: 84px;
+  height: 169px;
+}
+#loadbefore {
+  left: -1px;
+  transform-origin: 83px 83px;
+  animation: loading 1000ms cubic-bezier(0.4, 0.0, 0.2, 1) infinite 750ms;
+}
+
+#loadafter {
+  left: 83px;
+  transform-origin: 0 83px;
+  animation: loading 1000ms cubic-bezier(0.4, 0.0, 0.2, 1) infinite;
+}
+
+@keyframes loading {
+  0% {
+    transform: rotate(0deg);
+  }
+  100% {
+    transform: rotate(360deg);
+  }
+}
+
+@media only screen and (min-device-width : 1921px){
+  #lozengeBg {
+    width: 960px;
+    height: 425px;
+  }
+  #lozenge {
+    background: transparent url('') no-repeat right;
+    width: 1267px;
+    height: 425px;
+  }
+  #wordmark {
+    background: transparent url('') no-repeat;
+    width: 1267px;
+    height: 385px;
+  }
+}
+
+@media only screen and (min-device-width : 1281px) and (max-device-width : 1920px) {
+  #lozengeBg {
+    width: 480px;
+    height: 212px;
+  }
+  #lozenge {
+    background: transparent url('') no-repeat right;
+    width: 633px;
+    height: 212px;
+  }
+  #wordmark {
+    background: transparent url('') no-repeat;
+    width: 633px;
+    height: 192px;
+  }
+  #loader {
+    transform: scale(0.5, 0.5);
+  }
+}
+@media only screen and (max-device-width : 1280px) {
+  #lozengeBg {
+    width: 320px;
+    height: 141px;
+  }
+  #lozenge {
+    background: transparent url('') no-repeat right;
+    width: 422px;
+    height: 141px;
+  }
+  #wordmark {
+    background: transparent url('') no-repeat;
+    width: 422px;
+    height: 128px;
+  }
+  #loader {
+    transform: scale(0.33, 0.33);
+  }
+}
+</style>
+
 </head>
 
 <body>
-  <div id="splash">
-    <img src="h5vcc-embedded://you_tube_logo.png" class="hidden">
+<div id="background">
+<div id="lozengeContainer">
+  <div id="lozengeBg"></div>
+  <div id="lozenge"></div>
+</div>
+<div id="wordmarkContainer">
+  <div id="wordmark"></div>
+</div>
+<div id="spinnerContainer" class="active">
+  <div id="loader">
+    <div id="loadbefore" class="beforeafter"></div>
+    <div id="loadafter" class="beforeafter"></div>
   </div>
-  <div id="loading">
-    <div id="spinner">
-      <div class="dot" id="dot1"></div>
-      <div class="dot" id="dot2"></div>
-      <div class="dot" id="dot3"></div>
-      <div class="dot" id="dot4"></div>
-      <div class="dot" id="dot5"></div>
-      <div class="dot" id="dot6"></div>
-      <div class="dot" id="dot7"></div>
-      <div class="dot" id="dot8"></div>
-    </div>
-  </div>
-  <script type="text/javascript" src="h5vcc-embedded://splash_screen.js">
-  </script>
-</body>
+</div>
+</div>
+<script>
+  function backgroundTransitionEnd(e) {
+    if (e.target == e.currentTarget) {
+      window.close();
+    }
+  }
 
+  function expand() {
+    document.getElementById('lozengeContainer').className = 'move';
+    document.getElementById('wordmarkContainer').className = 'move';
+    document.getElementById('spinnerContainer').classList.remove("active");
+    document.getElementById('background').addEventListener('transitionend',
+        backgroundTransitionEnd);
+    document.getElementById('background').style.opacity = 0;
+    return '';
+  }
+
+  window.onbeforeunload = expand;
+</script>
+</body>
 </html>
diff --git a/src/cobalt/loader/image/image.h b/src/cobalt/loader/image/image.h
index 2b72010..740c9b5 100644
--- a/src/cobalt/loader/image/image.h
+++ b/src/cobalt/loader/image/image.h
@@ -140,10 +140,7 @@
       scoped_refptr<FrameProvider> frame_provider,
       const math::RectF& destination_rect,
       const math::Matrix3F& local_transform,
-      render_tree::ImageNode::Builder* image_node_builder,
-      base::TimeDelta time) {
-    UNREFERENCED_PARAMETER(time);
-
+      render_tree::ImageNode::Builder* image_node_builder) {
     image_node_builder->source = frame_provider->GetFrame();
     image_node_builder->destination_rect = destination_rect;
     image_node_builder->local_transform = local_transform;
diff --git a/src/cobalt/loader/image/image_decoder_starboard.cc b/src/cobalt/loader/image/image_decoder_starboard.cc
index 1183eea..49d52a3 100644
--- a/src/cobalt/loader/image/image_decoder_starboard.cc
+++ b/src/cobalt/loader/image/image_decoder_starboard.cc
@@ -23,7 +23,7 @@
 #include "starboard/decode_target.h"
 #include "starboard/image.h"
 
-#if SB_TRUE && SB_HAS(GRAPHICS)
+#if SB_HAS(GRAPHICS)
 
 namespace cobalt {
 namespace loader {
@@ -35,11 +35,7 @@
     : ImageDataDecoder(resource_provider),
       mime_type_(mime_type),
       format_(format),
-#if SB_TRUE
       provider_(resource_provider->GetSbDecodeTargetGraphicsContextProvider()),
-#else   // #if SB_TRUE
-      provider_(resource_provider->GetSbDecodeTargetProvider()),
-#endif  // #if SB_TRUE
       target_(kSbDecodeTargetInvalid) {
   TRACE_EVENT0("cobalt::loader::image",
                "ImageDecoderStarboard::ImageDecoderStarboard()");
@@ -75,6 +71,6 @@
 }  // namespace loader
 }  // namespace cobalt
 
-#endif  // SB_TRUE && SB_HAS(GRAPHICS)
+#endif  // SB_HAS(GRAPHICS)
 
 #endif  // #if defined(STARBOARD)
diff --git a/src/cobalt/loader/loader.gyp b/src/cobalt/loader/loader.gyp
index 6a07f01..cf037bf 100644
--- a/src/cobalt/loader/loader.gyp
+++ b/src/cobalt/loader/loader.gyp
@@ -184,10 +184,8 @@
         '<(input_directory)/cobalt_splash_screen.css',
         '<(input_directory)/cobalt_splash_screen.html',
         '<(input_directory)/equirectangular_40_40.msh',
-        '<(input_directory)/youtube_splash_screen.css',
         '<(input_directory)/youtube_splash_screen.html',
         '<(input_directory)/splash_screen.js',
-        '<(input_directory)/you_tube_logo.png',
       ],
       'actions': [
         {
diff --git a/src/cobalt/loader/unencoded_data_urls/lozenge_1080.png b/src/cobalt/loader/unencoded_data_urls/lozenge_1080.png
new file mode 100644
index 0000000..1e131c4
--- /dev/null
+++ b/src/cobalt/loader/unencoded_data_urls/lozenge_1080.png
Binary files differ
diff --git a/src/cobalt/loader/unencoded_data_urls/lozenge_2160.png b/src/cobalt/loader/unencoded_data_urls/lozenge_2160.png
new file mode 100644
index 0000000..49f2670
--- /dev/null
+++ b/src/cobalt/loader/unencoded_data_urls/lozenge_2160.png
Binary files differ
diff --git a/src/cobalt/loader/unencoded_data_urls/lozenge_720.png b/src/cobalt/loader/unencoded_data_urls/lozenge_720.png
new file mode 100644
index 0000000..1a10d3f
--- /dev/null
+++ b/src/cobalt/loader/unencoded_data_urls/lozenge_720.png
Binary files differ
diff --git a/src/cobalt/loader/unencoded_data_urls/wordmark_1080.png b/src/cobalt/loader/unencoded_data_urls/wordmark_1080.png
new file mode 100644
index 0000000..1c153f1
--- /dev/null
+++ b/src/cobalt/loader/unencoded_data_urls/wordmark_1080.png
Binary files differ
diff --git a/src/cobalt/loader/unencoded_data_urls/wordmark_2160.png b/src/cobalt/loader/unencoded_data_urls/wordmark_2160.png
new file mode 100644
index 0000000..3d106c5
--- /dev/null
+++ b/src/cobalt/loader/unencoded_data_urls/wordmark_2160.png
Binary files differ
diff --git a/src/cobalt/loader/unencoded_data_urls/wordmark_720.png b/src/cobalt/loader/unencoded_data_urls/wordmark_720.png
new file mode 100644
index 0000000..7fcb9f1
--- /dev/null
+++ b/src/cobalt/loader/unencoded_data_urls/wordmark_720.png
Binary files differ
diff --git a/src/cobalt/math/matrix3_f.cc b/src/cobalt/math/matrix3_f.cc
index 2f1f19d..929340e 100644
--- a/src/cobalt/math/matrix3_f.cc
+++ b/src/cobalt/math/matrix3_f.cc
@@ -84,6 +84,12 @@
   return matrix;
 }
 
+bool Matrix3F::IsZeros() const {
+  return data_[M00] == 0.0f && data_[M01] == 0.0f && data_[M02] == 0.0f &&
+         data_[M10] == 0.0f && data_[M11] == 0.0f && data_[M12] == 0.0f &&
+         data_[M20] == 0.0f && data_[M21] == 0.0f && data_[M22] == 0.0f;
+}
+
 bool Matrix3F::IsIdentity() const {
   return data_[M00] == 1.0f && data_[M01] == 0.0f && data_[M02] == 0.0f &&
          data_[M10] == 0.0f && data_[M11] == 1.0f && data_[M12] == 0.0f &&
diff --git a/src/cobalt/math/matrix3_f.h b/src/cobalt/math/matrix3_f.h
index d4560a6..c2e8c63 100644
--- a/src/cobalt/math/matrix3_f.h
+++ b/src/cobalt/math/matrix3_f.h
@@ -39,6 +39,7 @@
     data_[8] = m22;
   }
 
+  bool IsZeros() const;
   bool IsIdentity() const;
 
   bool IsEqual(const Matrix3F& rhs) const;
diff --git a/src/cobalt/media/player/web_media_player.h b/src/cobalt/media/player/web_media_player.h
index beee2aa..b3b5cb6 100644
--- a/src/cobalt/media/player/web_media_player.h
+++ b/src/cobalt/media/player/web_media_player.h
@@ -194,7 +194,7 @@
  public:
   virtual void NetworkStateChanged() = 0;
   virtual void ReadyStateChanged() = 0;
-  virtual void TimeChanged(bool eos_played) = 0;
+  virtual void TimeChanged() = 0;
   virtual void DurationChanged() = 0;
   virtual void OutputModeChanged() = 0;
   virtual void PlaybackStateChanged() = 0;
diff --git a/src/cobalt/media/player/web_media_player_impl.cc b/src/cobalt/media/player/web_media_player_impl.cc
index c8cf80a..bd0d14b 100644
--- a/src/cobalt/media/player/web_media_player_impl.cc
+++ b/src/cobalt/media/player/web_media_player_impl.cc
@@ -590,8 +590,7 @@
   // Update our paused time.
   if (state_.paused) state_.paused_time = pipeline_->GetMediaTime();
 
-  const bool eos_played = false;
-  GetClient()->TimeChanged(eos_played);
+  GetClient()->TimeChanged();
 }
 
 void WebMediaPlayerImpl::OnPipelineEnded(PipelineStatus status) {
@@ -600,9 +599,7 @@
     OnPipelineError(status);
     return;
   }
-
-  const bool eos_played = true;
-  GetClient()->TimeChanged(eos_played);
+  GetClient()->TimeChanged();
 }
 
 void WebMediaPlayerImpl::OnPipelineError(PipelineStatus error) {
diff --git a/src/cobalt/media/sandbox/web_media_player_helper.cc b/src/cobalt/media/sandbox/web_media_player_helper.cc
index 1909a5c..58618a2 100644
--- a/src/cobalt/media/sandbox/web_media_player_helper.cc
+++ b/src/cobalt/media/sandbox/web_media_player_helper.cc
@@ -35,7 +35,7 @@
   // WebMediaPlayerClient methods
   void NetworkStateChanged() OVERRIDE {}
   void ReadyStateChanged() OVERRIDE {}
-  void TimeChanged(bool) OVERRIDE {}
+  void TimeChanged() OVERRIDE {}
   void DurationChanged() OVERRIDE {}
   void OutputModeChanged() OVERRIDE {}
   void PlaybackStateChanged() OVERRIDE {}
@@ -48,8 +48,9 @@
 #endif  // defined(COBALT_MEDIA_SOURCE_2016)
   std::string SourceURL() const OVERRIDE { return ""; }
 #if defined(COBALT_MEDIA_SOURCE_2016)
-  void EncryptedMediaInitDataEncountered(EmeInitDataType, const unsigned char*,
-                                         unsigned) OVERRIDE {}
+  void EncryptedMediaInitDataEncountered(EmeInitDataType init_data_type,
+                                         const unsigned char* init_data,
+                                         unsigned init_data_length) OVERRIDE {}
 #endif  // defined(COBALT_MEDIA_SOURCE_2016)
 };
 
diff --git a/src/cobalt/render_tree/animations/animate_node.cc b/src/cobalt/render_tree/animations/animate_node.cc
index 58c11b4..46435b5 100644
--- a/src/cobalt/render_tree/animations/animate_node.cc
+++ b/src/cobalt/render_tree/animations/animate_node.cc
@@ -66,7 +66,8 @@
                       TraverseList* traverse_list)
       : animation_map_(animation_map),
         traverse_list_(traverse_list),
-        expiry_(-base::TimeDelta::Max()) {}
+        expiry_(-base::TimeDelta::Max()),
+        depends_on_time_expiry_(-base::TimeDelta::Max()) {}
 
   void Visit(animations::AnimateNode* animate) OVERRIDE;
   void Visit(CompositionNode* composition) OVERRIDE { VisitNode(composition); }
@@ -117,6 +118,10 @@
   // The time after which all animations will have completed and be constant.
   base::TimeDelta expiry_;
 
+  // Similar to |expiry_| but accumulated only for animations whose callback
+  // depends on the time parameter.
+  base::TimeDelta depends_on_time_expiry_;
+
   friend class AnimateNode;
 };
 
@@ -146,6 +151,9 @@
 
   // Update our expiry in accordance with the sub-AnimateNode's expiry.
   expiry_ = std::max(expiry_, animate->expiry());
+
+  depends_on_time_expiry_ =
+      std::max(depends_on_time_expiry_, animate->depends_on_time_expiry());
 }
 
 template <typename T>
@@ -208,8 +216,10 @@
 void AnimateNode::TraverseListBuilder::AddToTraverseList(
     Node* node, AnimateNode::Builder::InternalMap::const_iterator found) {
   if (found != animation_map_.end()) {
-    traverse_list_->push_back(TraverseListEntry(node, found->second));
+    traverse_list_->push_back(TraverseListEntry(node, found->second, false));
     expiry_ = std::max(expiry_, found->second->GetExpiry());
+    depends_on_time_expiry_ = std::max(depends_on_time_expiry_,
+                                       found->second->GetDependsOnTimeExpiry());
   } else {
     traverse_list_->push_back(TraverseListEntry(node));
   }
@@ -304,7 +314,9 @@
   TraverseListEntry current_entry = AdvanceIterator(node);
 
   DCHECK(current_entry.animations);
-  ProcessAnimatedNodeBounds(current_entry, node);
+  if (current_entry.did_animate_previously) {
+    ProcessAnimatedNodeBounds(current_entry, node);
+  }
 }
 
 template <typename T>
@@ -336,7 +348,7 @@
   }
   transform_ = old_transform;
 
-  if (current_entry.animations) {
+  if (current_entry.did_animate_previously) {
     ProcessAnimatedNodeBounds(current_entry, node);
   }
 }
@@ -376,7 +388,8 @@
 // tree.
 class AnimateNode::ApplyVisitor : public NodeVisitor {
  public:
-  ApplyVisitor(const TraverseList& traverse_list, base::TimeDelta time_offset);
+  ApplyVisitor(const TraverseList& traverse_list, base::TimeDelta time_offset,
+               const base::optional<base::TimeDelta>& snapshot_time);
 
   void Visit(animations::AnimateNode* /* animate */) OVERRIDE {
     // An invariant of AnimateNodes is that they should never contain descendant
@@ -405,7 +418,9 @@
   // As we compute the animated nodes, we create a new traverse list that leads
   // to the newly created animated nodes.  This can be used afterwards to
   // calculate the bounding boxes around the active animated nodes.
-  TraverseList* animated_traverse_list() { return &animated_traverse_list_; }
+  const TraverseList& animated_traverse_list() const {
+    return animated_traverse_list_;
+  }
 
  private:
   template <typename T>
@@ -415,8 +430,8 @@
   typename base::enable_if<ChildIterator<T>::has_children>::type VisitNode(
       T* node);
   template <typename T>
-  scoped_refptr<Node> ApplyAnimations(const TraverseListEntry& entry,
-                                      typename T::Builder* builder);
+  scoped_refptr<T> ApplyAnimations(const TraverseListEntry& entry,
+                                   typename T::Builder* builder);
   TraverseListEntry AdvanceIterator(Node* node);
 
   // The time offset to be passed in to individual animations.
@@ -435,11 +450,18 @@
 
   // An iterator pointing to the next valid render tree node to visit.
   TraverseList::const_iterator iterator_;
+
+  // Time at which the existing source render tree was created/last animated
+  // at.
+  base::optional<base::TimeDelta> snapshot_time_;
 };
 
-AnimateNode::ApplyVisitor::ApplyVisitor(const TraverseList& traverse_list,
-                                        base::TimeDelta time_offset)
-    : time_offset_(time_offset), traverse_list_(traverse_list) {
+AnimateNode::ApplyVisitor::ApplyVisitor(
+    const TraverseList& traverse_list, base::TimeDelta time_offset,
+    const base::optional<base::TimeDelta>& snapshot_time)
+    : time_offset_(time_offset),
+      traverse_list_(traverse_list),
+      snapshot_time_(snapshot_time) {
   animated_traverse_list_.reserve(traverse_list.size());
   iterator_ = traverse_list_.begin();
 }
@@ -452,9 +474,18 @@
   // have animations.
   DCHECK(current_entry.animations);
   typename T::Builder builder(node->data());
-  animated_ = ApplyAnimations<T>(current_entry, &builder);
+  scoped_refptr<T> animated = ApplyAnimations<T>(current_entry, &builder);
+  // If nothing ends up getting animated, then just re-use the existing node.
+  bool did_animate = false;
+  if (animated->data() == node->data()) {
+    animated_ = node;
+  } else {
+    animated_ = animated.get();
+    did_animate = true;
+  }
+
   animated_traverse_list_.push_back(
-      TraverseListEntry(animated_, current_entry.animations));
+      TraverseListEntry(animated_, current_entry.animations, did_animate));
 }
 
 template <typename T>
@@ -464,7 +495,7 @@
 
   size_t animated_traverse_list_index = animated_traverse_list_.size();
   animated_traverse_list_.push_back(
-      TraverseListEntry(NULL, current_entry.animations));
+      TraverseListEntry(NULL, current_entry.animations, false));
 
   // Traverse the child nodes, but only the ones that are on the
   // |traverse_list_|.  In particular, the next node we are allowed to visit
@@ -482,41 +513,60 @@
       // If one of our children is next up on the path to animation, traverse
       // into it.
       child->Accept(this);
-      // Traversing into the child means that it was animated, and so replaced
-      // by an animated node while it was visited.  Thus, replace it in the
-      // current node's child list with its animated version.
-      child_iterator.ReplaceCurrent(animated_);
-      children_modified = true;
+      if (animated_ != child) {
+        // Traversing into the child and seeing |animated_| emerge from the
+        // traversal equal to something other than |child| means that the child
+        // was animated, and so replaced by an animated node while it was
+        // visited.  Thus, replace it in the current node's child list with its
+        // animated version.
+        child_iterator.ReplaceCurrent(animated_);
+        children_modified = true;
+      }
     }
 
     child_iterator.Next();
   }
 
+  base::optional<typename T::Builder> builder;
+  if (children_modified) {
+    // Reuse the modified Builder object from child traversal if one of
+    // our children was animated.
+    builder.emplace(child_iterator.TakeReplacedChildrenBuilder());
+  }
+
+  bool did_animate = false;
   if (current_entry.animations) {
-    base::optional<typename T::Builder> builder;
-    if (children_modified) {
-      // Reuse the modified Builder object from child traversal if one of
-      // our children was animated.
-      builder.emplace(child_iterator.TakeReplacedChildrenBuilder());
-    } else {
+    if (!builder) {
       // Create a fresh copy of the Builder object for this animated node, to
       // be passed into the animations.
       builder.emplace(node->data());
     }
-    animated_ = ApplyAnimations<T>(current_entry, &(*builder));
+    typename T::Builder original_builder(*builder);
+    scoped_refptr<T> animated = ApplyAnimations<T>(current_entry, &(*builder));
+    if (!(original_builder == *builder)) {
+      did_animate = true;
+    }
+    // If the data didn't actually change, then no animation took place and
+    // so we should note this by not modifying the original render tree node.
+    animated_ = animated->data() == node->data() ? node : animated.get();
   } else {
-    // If there were no animations targeting this node directly, then its
-    // children must have been modified since otherwise it wouldn't be in
-    // the traverse list.
-    DCHECK(children_modified);
-    animated_ = new T(child_iterator.TakeReplacedChildrenBuilder());
+    // If there were no animations targeting this node directly, it may still
+    // need to be animated if its children are animated, which will be the
+    // case if |builder| is populated.
+    if (builder) {
+      animated_ = new T(std::move(*builder));
+    } else {
+      animated_ = node;
+    }
   }
 
   animated_traverse_list_[animated_traverse_list_index].node = animated_;
+  animated_traverse_list_[animated_traverse_list_index].did_animate_previously =
+      did_animate;
 }
 
 template <typename T>
-scoped_refptr<Node> AnimateNode::ApplyVisitor::ApplyAnimations(
+scoped_refptr<T> AnimateNode::ApplyVisitor::ApplyAnimations(
     const TraverseListEntry& entry, typename T::Builder* builder) {
   TRACE_EVENT0("cobalt::renderer",
                "AnimateNode::ApplyVisitor::ApplyAnimations()");
@@ -525,11 +575,16 @@
       base::polymorphic_downcast<const AnimationList<T>*>(
           entry.animations.get());
 
-  // Iterate through each animation applying them one at a time.
-  for (typename AnimationList<T>::InternalList::const_iterator iter =
-           typed_node_animations->data().animations.begin();
-       iter != typed_node_animations->data().animations.end(); ++iter) {
-    iter->Run(builder, time_offset_);
+  // Only execute the animation updates on nodes that have not expired.
+  if (!snapshot_time_ ||
+      typed_node_animations->data().expiry >= *snapshot_time_) {
+    TRACE_EVENT0("cobalt::renderer", "Running animation callbacks");
+    // Iterate through each animation applying them one at a time.
+    for (typename AnimationList<T>::InternalList::const_iterator iter =
+             typed_node_animations->data().animations.begin();
+         iter != typed_node_animations->data().animations.end(); ++iter) {
+      iter->Run(builder, time_offset_);
+    }
   }
 
   return new T(*builder);
@@ -559,9 +614,9 @@
 class AnimateNode::RefCountedTraversalList
     : public base::RefCounted<RefCountedTraversalList> {
  public:
-  explicit RefCountedTraversalList(TraverseList* to_swap_in) {
-    traverse_list_.swap(*to_swap_in);
-  }
+  explicit RefCountedTraversalList(const TraverseList& traverse_list)
+      : traverse_list_(traverse_list) {}
+
   const TraverseList& traverse_list() const { return traverse_list_; }
 
  private:
@@ -594,24 +649,41 @@
 
 AnimateNode::AnimateResults AnimateNode::Apply(base::TimeDelta time_offset) {
   TRACE_EVENT0("cobalt::renderer", "AnimateNode::Apply()");
+  if (snapshot_time_) {
+    // Assume we are always animating forward.
+    DCHECK_LE(*snapshot_time_, time_offset);
+  }
+
   AnimateResults results;
   if (traverse_list_.empty()) {
-    results.animated = source_;
+    results.animated = this;
     // There are no animations, so there is no bounding rectangle, so setup the
     // bounding box function to trivially return an empty rectangle.
     results.get_animation_bounds_since =
         base::Bind(&ReturnTrivialEmptyRectBound);
   } else {
-    ApplyVisitor apply_visitor(traverse_list_, time_offset);
+    ApplyVisitor apply_visitor(traverse_list_, time_offset, snapshot_time_);
     source_->Accept(&apply_visitor);
-    results.animated = apply_visitor.animated();
 
-    // Setup a function for returning
+    // Setup a function for returning the bounds on the regions modified by
+    // animations given a specific starting point in time ("since").  This
+    // can be used by rasterizers to determine which regions need to be
+    // re-drawn or not.
     results.get_animation_bounds_since = base::Bind(
         &GetAnimationBoundsSince,
         scoped_refptr<RefCountedTraversalList>(new RefCountedTraversalList(
             apply_visitor.animated_traverse_list())),
-        time_offset, results.animated);
+        time_offset, apply_visitor.animated());
+
+    if (apply_visitor.animated() == source()) {
+      // If no animations were actually applied, indicate this by returning
+      // this exact node as the animated node.
+      results.animated = this;
+    } else {
+      results.animated = new AnimateNode(apply_visitor.animated_traverse_list(),
+                                         apply_visitor.animated(), expiry_,
+                                         depends_on_time_expiry_, time_offset);
+    }
   }
   return results;
 }
@@ -638,6 +710,7 @@
   }
 
   expiry_ = traverse_list_builder.expiry_;
+  depends_on_time_expiry_ = traverse_list_builder.depends_on_time_expiry_;
 }
 
 }  // namespace animations
diff --git a/src/cobalt/render_tree/animations/animate_node.h b/src/cobalt/render_tree/animations/animate_node.h
index 70ed9ba..c71e620 100644
--- a/src/cobalt/render_tree/animations/animate_node.h
+++ b/src/cobalt/render_tree/animations/animate_node.h
@@ -20,6 +20,7 @@
 
 #include "base/containers/small_map.h"
 #include "base/memory/ref_counted.h"
+#include "base/optional.h"
 #include "cobalt/math/rect_f.h"
 #include "cobalt/render_tree/animations/animation_list.h"
 #include "cobalt/render_tree/movable.h"
@@ -80,12 +81,30 @@
     }
 
     template <typename T>
+    void Add(
+        const scoped_refptr<T>& target_node,
+        const typename Animation<T>::TimeIndependentFunction& single_animation,
+        base::TimeDelta expiry) {
+      AddInternal(target_node,
+                  scoped_refptr<AnimationListBase>(
+                      new AnimationList<T>(single_animation, expiry)));
+    }
+
+    template <typename T>
     void Add(const scoped_refptr<T>& target_node,
              const typename Animation<T>::Function& single_animation) {
       AddInternal(target_node,
                   scoped_refptr<AnimationListBase>(new AnimationList<T>(
                       single_animation, base::TimeDelta::Max())));
     }
+    template <typename T>
+    void Add(const scoped_refptr<T>& target_node,
+             const typename Animation<T>::TimeIndependentFunction&
+                 single_animation) {
+      AddInternal(target_node,
+                  scoped_refptr<AnimationListBase>(new AnimationList<T>(
+                      single_animation, base::TimeDelta::Max())));
+    }
 
     // Merge all mappings from another AnimateNode::Builder into this one.
     // There cannot be any keys that are in both the merge target and source.
@@ -140,8 +159,11 @@
 
   struct AnimateResults {
     // The animated render tree, which is guaranteed to not contain any
-    // AnimateNodes.
-    scoped_refptr<Node> animated;
+    // AnimateNodes.  Note that it is returned as an AnimateNode...  The actual
+    // animated render tree can be obtained via |animated->source()|, the
+    // AnimateNode however allows one to animate the animated node so that
+    // it can be checked if anything has actually been animated or not.
+    scoped_refptr<AnimateNode> animated;
 
     // Can be called in order to return a bounding rectangle around all
     // nodes that are actively animated.  The parameter specifies a "since"
@@ -152,6 +174,9 @@
     base::Callback<math::RectF(base::TimeDelta)> get_animation_bounds_since;
   };
   // Apply the animations to the sub render tree with the given |time_offset|.
+  // Note that if |animated| in the results is equivalent to |this| on which
+  // Apply() is called, then no animations have actually taken place, and a
+  // re-render can be avoided.
   AnimateResults Apply(base::TimeDelta time_offset);
 
   // Returns the sub-tree for which the animations apply to.
@@ -163,6 +188,12 @@
   // all x, y >= expiry().
   const base::TimeDelta& expiry() const { return expiry_; }
 
+  // Similar to |expiry()|, but returns the expiration for all animations whose
+  // callback function actually depends on the time parameter passed into them.
+  const base::TimeDelta& depends_on_time_expiry() const {
+    return depends_on_time_expiry_;
+  }
+
  private:
   // The compiled node animation list is a sequence of nodes that are either
   // animated themselves, or on the path to an animated node.  Only nodes in
@@ -170,12 +201,19 @@
   // list, complete with animations to be applied to the given node, if any.
   struct TraverseListEntry {
     TraverseListEntry(Node* node,
-                      const scoped_refptr<AnimationListBase>& animations)
-        : node(node), animations(animations) {}
-    explicit TraverseListEntry(Node* node) : node(node) {}
+                      const scoped_refptr<AnimationListBase>& animations,
+                      bool did_animate_previously)
+        : node(node),
+          animations(animations),
+          did_animate_previously(did_animate_previously) {}
+    explicit TraverseListEntry(Node* node)
+        : node(node), did_animate_previously(false) {}
 
     Node* node;
     scoped_refptr<AnimationListBase> animations;
+    // Used when checking animation bounds to see which nodes were actually
+    // animated and whose bounds we need to accumulate.
+    bool did_animate_previously;
   };
   typedef std::vector<TraverseListEntry> TraverseList;
 
@@ -192,6 +230,18 @@
   // rectangle for all active animations.
   class BoundsVisitor;
 
+  // This private constructor is used by AnimateNode::Apply() to create a new
+  // AnimateNode that matches the resulting animated render tree.
+  AnimateNode(const TraverseList& traverse_list, scoped_refptr<Node> source,
+              const base::TimeDelta& expiry,
+              const base::TimeDelta& depends_on_time_expiry,
+              const base::TimeDelta& snapshot_time)
+      : traverse_list_(traverse_list),
+        source_(source),
+        expiry_(expiry),
+        depends_on_time_expiry_(depends_on_time_expiry),
+        snapshot_time_(snapshot_time) {}
+
   void CommonInit(const Builder::InternalMap& node_animation_map,
                   const scoped_refptr<Node>& source);
 
@@ -205,6 +255,12 @@
   TraverseList traverse_list_;
   scoped_refptr<Node> source_;
   base::TimeDelta expiry_;
+  base::TimeDelta depends_on_time_expiry_;
+  // Time when the |source_| render tree was last animated, if known.  This
+  // will get set when Apply() is called to produce a new AnimateNode that can
+  // then be Apply()d again.  It can be used during the second apply to check
+  // if some animations have expired.
+  base::optional<base::TimeDelta> snapshot_time_;
 };
 
 }  // namespace animations
diff --git a/src/cobalt/render_tree/animations/animate_node_test.cc b/src/cobalt/render_tree/animations/animate_node_test.cc
index 7f539b9..b19733e 100644
--- a/src/cobalt/render_tree/animations/animate_node_test.cc
+++ b/src/cobalt/render_tree/animations/animate_node_test.cc
@@ -81,10 +81,10 @@
   scoped_refptr<AnimateNode> with_animations =
       CreateSingleAnimation(text_node, base::Bind(&AnimateText));
 
-  scoped_refptr<Node> animated_render_tree =
+  scoped_refptr<AnimateNode> animated_render_tree =
       with_animations->Apply(base::TimeDelta::FromSeconds(1)).animated;
   TextNode* animated_text_node =
-      dynamic_cast<TextNode*>(animated_render_tree.get());
+      dynamic_cast<TextNode*>(animated_render_tree->source().get());
   EXPECT_TRUE(animated_text_node);
   EXPECT_EQ(ColorRGBA(.5f, .5f, .5f), animated_text_node->data().color);
 }
@@ -101,10 +101,10 @@
   scoped_refptr<AnimateNode> with_animations =
       CreateSingleAnimation(image_node, base::Bind(&AnimateImage));
 
-  scoped_refptr<Node> animated_render_tree =
+  scoped_refptr<AnimateNode> animated_render_tree =
       with_animations->Apply(base::TimeDelta::FromSeconds(1)).animated;
   ImageNode* animated_image_node =
-      dynamic_cast<ImageNode*>(animated_render_tree.get());
+      dynamic_cast<ImageNode*>(animated_render_tree->source().get());
   EXPECT_TRUE(animated_image_node);
   EXPECT_TRUE(animated_image_node);
   EXPECT_EQ(RectF(2.0f, 2.0f), animated_image_node->data().destination_rect);
@@ -122,10 +122,10 @@
   scoped_refptr<AnimateNode> with_animations =
       CreateSingleAnimation(rect_node, base::Bind(&AnimateRect));
 
-  scoped_refptr<Node> animated_render_tree =
+  scoped_refptr<AnimateNode> animated_render_tree =
       with_animations->Apply(base::TimeDelta::FromSeconds(1)).animated;
   RectNode* animated_rect_node =
-      dynamic_cast<RectNode*>(animated_render_tree.get());
+      dynamic_cast<RectNode*>(animated_render_tree->source().get());
   EXPECT_TRUE(animated_rect_node);
   EXPECT_TRUE(animated_rect_node);
   EXPECT_EQ(RectF(2.0f, 2.0f), animated_rect_node->data().rect);
@@ -147,26 +147,29 @@
   scoped_refptr<AnimateNode> with_animations =
       CreateSingleAnimation(image_node, base::Bind(&AnimateImageAddWithTime));
 
-  scoped_refptr<Node> animated_render_tree;
+  scoped_refptr<AnimateNode> animated_render_tree;
   ImageNode* animated_image_node;
 
   animated_render_tree =
       with_animations->Apply(base::TimeDelta::FromSeconds(1)).animated;
-  animated_image_node = dynamic_cast<ImageNode*>(animated_render_tree.get());
+  animated_image_node =
+      dynamic_cast<ImageNode*>(animated_render_tree->source().get());
   EXPECT_TRUE(animated_image_node);
   EXPECT_TRUE(animated_image_node);
   EXPECT_FLOAT_EQ(3.0f, animated_image_node->data().destination_rect.width());
 
   animated_render_tree =
       with_animations->Apply(base::TimeDelta::FromSeconds(2)).animated;
-  animated_image_node = dynamic_cast<ImageNode*>(animated_render_tree.get());
+  animated_image_node =
+      dynamic_cast<ImageNode*>(animated_render_tree->source().get());
   EXPECT_TRUE(animated_image_node);
   EXPECT_TRUE(animated_image_node);
   EXPECT_FLOAT_EQ(5.0f, animated_image_node->data().destination_rect.width());
 
   animated_render_tree =
       with_animations->Apply(base::TimeDelta::FromSeconds(4)).animated;
-  animated_image_node = dynamic_cast<ImageNode*>(animated_render_tree.get());
+  animated_image_node =
+      dynamic_cast<ImageNode*>(animated_render_tree->source().get());
   EXPECT_TRUE(animated_image_node);
   EXPECT_TRUE(animated_image_node);
   EXPECT_FLOAT_EQ(9.0f, animated_image_node->data().destination_rect.width());
@@ -202,11 +205,11 @@
   scoped_refptr<AnimateNode> with_animations(
       new AnimateNode(animations_builder, image_node));
 
-  scoped_refptr<Node> animated_render_tree =
+  scoped_refptr<AnimateNode> animated_render_tree =
       with_animations->Apply(base::TimeDelta::FromSeconds(3)).animated;
 
   ImageNode* animated_image_node =
-      dynamic_cast<ImageNode*>(animated_render_tree.get());
+      dynamic_cast<ImageNode*>(animated_render_tree->source().get());
   EXPECT_TRUE(animated_image_node);
 
   EXPECT_FLOAT_EQ(21.0f, animated_image_node->data().destination_rect.width());
@@ -228,11 +231,11 @@
   scoped_refptr<AnimateNode> with_animations =
       CreateSingleAnimation(transform_node, base::Bind(&AnimateTransform));
 
-  scoped_refptr<Node> animated_render_tree =
+  scoped_refptr<AnimateNode> animated_render_tree =
       with_animations->Apply(base::TimeDelta::FromSeconds(1)).animated;
 
   MatrixTransformNode* animated_transform_node =
-      dynamic_cast<MatrixTransformNode*>(animated_render_tree.get());
+      dynamic_cast<MatrixTransformNode*>(animated_render_tree->source().get());
   EXPECT_TRUE(animated_transform_node);
 
   EXPECT_EQ(math::TranslateMatrix(2.0f, 2.0f),
@@ -260,11 +263,11 @@
   scoped_refptr<AnimateNode> with_animations =
       new AnimateNode(animation_builder, composition_node);
 
-  scoped_refptr<Node> animated_render_tree =
+  scoped_refptr<AnimateNode> animated_render_tree =
       with_animations->Apply(base::TimeDelta::FromSeconds(1)).animated;
 
   CompositionNode* animated_composition_node =
-      dynamic_cast<CompositionNode*>(animated_render_tree.get());
+      dynamic_cast<CompositionNode*>(animated_render_tree->source().get());
   EXPECT_TRUE(animated_composition_node);
 
   ImageNode* animated_image_node_a = dynamic_cast<ImageNode*>(
@@ -291,11 +294,11 @@
   scoped_refptr<AnimateNode> with_animations =
       CreateSingleAnimation(transform_node, base::Bind(&AnimateTransform));
 
-  scoped_refptr<Node> animated_render_tree =
+  scoped_refptr<AnimateNode> animated_render_tree =
       with_animations->Apply(base::TimeDelta::FromSeconds(1)).animated;
 
   MatrixTransformNode* animated_transform_node =
-      dynamic_cast<MatrixTransformNode*>(animated_render_tree.get());
+      dynamic_cast<MatrixTransformNode*>(animated_render_tree->source().get());
   EXPECT_TRUE(animated_transform_node);
 
   // Check that the matrix transform node is animated.
@@ -328,11 +331,11 @@
   scoped_refptr<AnimateNode> with_animations =
       new AnimateNode(transform_node_b);
 
-  scoped_refptr<Node> animated_render_tree =
+  scoped_refptr<AnimateNode> animated_render_tree =
       with_animations->Apply(base::TimeDelta::FromSeconds(1)).animated;
 
   MatrixTransformNode* animated_transform_node =
-      dynamic_cast<MatrixTransformNode*>(animated_render_tree.get());
+      dynamic_cast<MatrixTransformNode*>(animated_render_tree->source().get());
   EXPECT_TRUE(animated_transform_node);
 
   // Check that the image node is animated.
diff --git a/src/cobalt/render_tree/animations/animation_list.h b/src/cobalt/render_tree/animations/animation_list.h
index 78f1e65..b51144d 100644
--- a/src/cobalt/render_tree/animations/animation_list.h
+++ b/src/cobalt/render_tree/animations/animation_list.h
@@ -17,6 +17,7 @@
 
 #include <list>
 
+#include "base/bind.h"
 #include "base/callback.h"
 #include "base/memory/ref_counted.h"
 #include "base/time.h"
@@ -68,6 +69,16 @@
   // and ultimately add that to a AnimateNode::Builder object so that it can be
   // mapped to a specific TextNode that it should be applied to.
   typedef base::Callback<void(typename T::Builder*, base::TimeDelta)> Function;
+  typedef base::Callback<void(typename T::Builder*)> TimeIndependentFunction;
+
+  // Helper function to convert from time independent function to a time
+  // dependent function.
+  static void CallTimeIndependentFunction(
+      const TimeIndependentFunction& time_independent_function,
+      typename T::Builder* builder, base::TimeDelta time) {
+    UNREFERENCED_PARAMETER(time);
+    time_independent_function.Run(builder);
+  }
 };
 
 // The AnimationListBase is used so that we can acquire a non-template handle
@@ -77,6 +88,7 @@
 class AnimationListBase : public base::RefCountedThreadSafe<AnimationListBase> {
  public:
   virtual base::TimeDelta GetExpiry() const = 0;
+  virtual base::TimeDelta GetDependsOnTimeExpiry() const = 0;
 
  protected:
   virtual ~AnimationListBase() {}
@@ -100,18 +112,36 @@
   struct Builder {
     DECLARE_AS_MOVABLE(Builder);
 
-    Builder() : expiry(base::TimeDelta::Max()) {}
+    Builder() : expiry(base::TimeDelta::Max()),
+                depends_on_time_expiry(base::TimeDelta::Max()) {}
     explicit Builder(Moved moved) { animations.swap(moved->animations); }
     explicit Builder(const typename Animation<T>::Function& single_animation,
                      base::TimeDelta expiry)
-        : expiry(expiry) {
+        : expiry(expiry), depends_on_time_expiry(expiry) {
       animations.push_back(single_animation);
     }
+    explicit Builder(
+        const typename Animation<T>::TimeIndependentFunction& single_animation,
+        base::TimeDelta expiry)
+        : expiry(expiry),
+          // Since this animation is time independent, we mark its
+          // time-dependent expiration to be already expired.
+          depends_on_time_expiry(-base::TimeDelta::Max()) {
+      // Transform from a time independent function to a time dependent
+      // function, since we store all functions as time dependent functions.
+      animations.push_back(base::Bind(Animation<T>::CallTimeIndependentFunction,
+                                      single_animation));
+    }
 
     InternalList animations;
     // When do the animations expire?  base::TimeDelta::Max() implies that they
     // never expire.
     base::TimeDelta expiry;
+
+    // Similar to |expiry|, but only set for animations that depend on the
+    // time parameter passed into their animation callback functions.
+    // Optimizations can be performed for animations that don't depend on this.
+    base::TimeDelta depends_on_time_expiry;
   };
 
   explicit AnimationList(typename Builder::Moved builder) : data_(builder) {}
@@ -122,10 +152,17 @@
       const typename Animation<T>::Function& single_animation,
       base::TimeDelta expiry)
       : data_(single_animation, expiry) {}
+  explicit AnimationList(
+      const typename Animation<T>::TimeIndependentFunction& single_animation,
+      base::TimeDelta expiry)
+      : data_(single_animation, expiry) {}
 
   const Builder& data() const { return data_; }
 
   base::TimeDelta GetExpiry() const OVERRIDE { return data_.expiry; }
+  base::TimeDelta GetDependsOnTimeExpiry() const OVERRIDE {
+    return data_.depends_on_time_expiry;
+  }
 
  private:
   ~AnimationList() OVERRIDE {}
diff --git a/src/cobalt/render_tree/blur_filter.h b/src/cobalt/render_tree/blur_filter.h
index 5f6fbe7..cbe95a0 100644
--- a/src/cobalt/render_tree/blur_filter.h
+++ b/src/cobalt/render_tree/blur_filter.h
@@ -29,6 +29,10 @@
   // |blur_sigma| must be non-negative.
   explicit BlurFilter(float blur_sigma) { set_blur_sigma(blur_sigma); }
 
+  bool operator==(const BlurFilter& other) const {
+    return blur_sigma_ == other.blur_sigma_;
+  }
+
   void set_blur_sigma(float blur_sigma) {
     DCHECK_LE(0.0f, blur_sigma);
     blur_sigma_ = blur_sigma;
diff --git a/src/cobalt/render_tree/border.h b/src/cobalt/render_tree/border.h
index 761c99f..9b72706 100644
--- a/src/cobalt/render_tree/border.h
+++ b/src/cobalt/render_tree/border.h
@@ -53,6 +53,11 @@
   Border(const BorderSide& left, const BorderSide& right, const BorderSide& top,
          const BorderSide& bottom);
 
+  bool operator==(const Border& other) const {
+    return left == other.left && right == other.right && top == other.top &&
+           bottom == other.bottom;
+  }
+
   BorderSide left;
   BorderSide right;
   BorderSide top;
diff --git a/src/cobalt/render_tree/brush.h b/src/cobalt/render_tree/brush.h
index b5be829..841ca6c 100644
--- a/src/cobalt/render_tree/brush.h
+++ b/src/cobalt/render_tree/brush.h
@@ -47,6 +47,10 @@
  public:
   explicit SolidColorBrush(const ColorRGBA& color) : color_(color) {}
 
+  bool operator==(const SolidColorBrush& other) const {
+    return color_ == other.color_;
+  }
+
   // A type-safe branching.
   void Accept(BrushVisitor* visitor) const OVERRIDE;
 
@@ -70,6 +74,10 @@
   ColorStop(float position, const ColorRGBA& color)
       : position(position), color(color) {}
 
+  bool operator==(const ColorStop& other) const {
+    return position == other.position && color == other.color;
+  }
+
   float position;
   ColorRGBA color;
 };
@@ -134,12 +142,20 @@
                             source_color, dest_color)
       {}
 
+  bool operator==(const LinearGradientBrush& other) const {
+    return data_ == other.data_;
+  }
+
   struct Data {
     Data();
     Data(const math::PointF& source, const math::PointF& dest,
          const ColorStopList& color_stops);
     Data(const math::PointF& source, const math::PointF& dest);
 
+    bool operator==(const Data& other) const {
+      return source_ == other.source_ && dest_ == other.dest_ &&
+             color_stops_ == other.color_stops_;
+    }
     math::PointF source_;
     math::PointF dest_;
 
@@ -203,6 +219,11 @@
                       float radius_y, const ColorRGBA& source_color,
                       const ColorRGBA& dest_color);
 
+  bool operator==(const RadialGradientBrush& other) const {
+    return center_ == other.center_ && radius_x_ == other.radius_x_ &&
+           radius_y_ == other.radius_y_ && color_stops_ == other.color_stops_;
+  }
+
   // A type-safe branching.
   void Accept(BrushVisitor* visitor) const OVERRIDE;
 
@@ -228,6 +249,36 @@
 
 scoped_ptr<Brush> CloneBrush(const Brush* brush);
 
+class EqualsBrushVisitor : public BrushVisitor {
+ public:
+  explicit EqualsBrushVisitor(Brush* brush_a)
+      : brush_a_(brush_a), result_(false) {}
+
+  bool result() const { return result_; }
+
+  void Visit(const SolidColorBrush* solid_color_brush) OVERRIDE {
+    result_ =
+        brush_a_->GetTypeId() == base::GetTypeId<SolidColorBrush>() &&
+        *static_cast<const SolidColorBrush*>(brush_a_) == *solid_color_brush;
+  }
+
+  void Visit(const LinearGradientBrush* linear_gradient_brush) OVERRIDE {
+    result_ = brush_a_->GetTypeId() == base::GetTypeId<LinearGradientBrush>() &&
+              *static_cast<const LinearGradientBrush*>(brush_a_) ==
+                  *linear_gradient_brush;
+  }
+
+  void Visit(const RadialGradientBrush* radial_gradient_brush) OVERRIDE {
+    result_ = brush_a_->GetTypeId() == base::GetTypeId<RadialGradientBrush>() &&
+              *static_cast<const RadialGradientBrush*>(brush_a_) ==
+                  *radial_gradient_brush;
+  }
+
+ private:
+  Brush* brush_a_;
+  bool result_;
+};
+
 }  // namespace render_tree
 }  // namespace cobalt
 
diff --git a/src/cobalt/render_tree/composition_node.h b/src/cobalt/render_tree/composition_node.h
index a6175a0..d39da3f 100644
--- a/src/cobalt/render_tree/composition_node.h
+++ b/src/cobalt/render_tree/composition_node.h
@@ -72,6 +72,10 @@
       children_.swap(moved->children_);
     }
 
+    bool operator==(const Builder& other) const {
+      return offset_ == other.offset_ && children_ == other.children_;
+    }
+
     // Add a child node to the list of children we are building.  When a
     // CompositionNode is constructed from this CompositionNode::Builder, it
     // will have as children all nodes who were passed into the builder via this
@@ -106,6 +110,9 @@
   explicit CompositionNode(Builder::Moved builder)
       : data_(builder), cached_bounds_(ComputeBounds()) {}
 
+  explicit CompositionNode(Builder&& builder)
+      : data_(builder.Pass()), cached_bounds_(ComputeBounds()) {}
+
   void Accept(NodeVisitor* visitor) OVERRIDE;
   math::RectF GetBounds() const OVERRIDE;
 
diff --git a/src/cobalt/render_tree/filter_node.h b/src/cobalt/render_tree/filter_node.h
index dc3df7c..612d76a 100644
--- a/src/cobalt/render_tree/filter_node.h
+++ b/src/cobalt/render_tree/filter_node.h
@@ -51,6 +51,13 @@
 
     math::RectF GetBounds() const;
 
+    bool operator==(const Builder& other) const {
+      return source == other.source && opacity_filter == other.opacity_filter &&
+             viewport_filter == other.viewport_filter &&
+             blur_filter == other.blur_filter &&
+             map_to_mesh_filter == other.map_to_mesh_filter;
+    }
+
     // The source tree, which will be used as the input to the filters specified
     // in this FilterNode.
     scoped_refptr<render_tree::Node> source;
diff --git a/src/cobalt/render_tree/image_node.h b/src/cobalt/render_tree/image_node.h
index ccd4773..7b5f048 100644
--- a/src/cobalt/render_tree/image_node.h
+++ b/src/cobalt/render_tree/image_node.h
@@ -36,6 +36,12 @@
             const math::RectF& destination_rect,
             const math::Matrix3F& local_transform);
 
+    bool operator==(const Builder& other) const {
+      return source == other.source &&
+             destination_rect == other.destination_rect &&
+             local_transform == other.local_transform;
+    }
+
     // A source of pixels. May be smaller or larger than layed out image.
     // The class does not own the image, it merely refers it from a resource
     // pool.
diff --git a/src/cobalt/render_tree/map_to_mesh_filter.h b/src/cobalt/render_tree/map_to_mesh_filter.h
index 060b080..a5c248f 100644
--- a/src/cobalt/render_tree/map_to_mesh_filter.h
+++ b/src/cobalt/render_tree/map_to_mesh_filter.h
@@ -47,6 +47,12 @@
  public:
   struct Builder {
     Builder() {}
+    bool operator==(const Builder& other) const {
+      return resolution_matched_meshes_ == other.resolution_matched_meshes_ &&
+             left_eye_default_mesh_ == other.left_eye_default_mesh_ &&
+             right_eye_default_mesh_ == other.right_eye_default_mesh_;
+    }
+
     void SetDefaultMeshes(
         const scoped_refptr<render_tree::Mesh>& left_eye_mesh,
         const scoped_refptr<render_tree::Mesh>& right_eye_mesh) {
@@ -113,6 +119,10 @@
     DCHECK(stereo_mode != kLeftRightUnadjustedTextureCoords);
   }
 
+  bool operator==(const MapToMeshFilter& other) const {
+    return stereo_mode_ == other.stereo_mode_ && data_ == other.data_;
+  }
+
   StereoMode stereo_mode() const { return stereo_mode_; }
 
   // The omission of the |resolution| parameter will yield the default
diff --git a/src/cobalt/render_tree/matrix_transform_3d_node.h b/src/cobalt/render_tree/matrix_transform_3d_node.h
index bbb517a..3732b68 100644
--- a/src/cobalt/render_tree/matrix_transform_3d_node.h
+++ b/src/cobalt/render_tree/matrix_transform_3d_node.h
@@ -37,6 +37,10 @@
     Builder(const scoped_refptr<Node>& source, const glm::mat4& transform)
         : source(source), transform(transform) {}
 
+    bool operator==(const Builder& other) const {
+      return source == other.source && transform == other.transform;
+    }
+
     // The subtree that will be rendered with |transform| applied to it.
     scoped_refptr<Node> source;
 
diff --git a/src/cobalt/render_tree/matrix_transform_node.h b/src/cobalt/render_tree/matrix_transform_node.h
index 3baafcc..93d9685 100644
--- a/src/cobalt/render_tree/matrix_transform_node.h
+++ b/src/cobalt/render_tree/matrix_transform_node.h
@@ -34,6 +34,10 @@
     Builder(const scoped_refptr<Node>& source, const math::Matrix3F& transform)
         : source(source), transform(transform) {}
 
+    bool operator==(const Builder& other) const {
+      return source == other.source && transform == other.transform;
+    }
+
     // The subtree that will be rendered with |transform| applied to it.
     scoped_refptr<Node> source;
 
diff --git a/src/cobalt/render_tree/opacity_filter.h b/src/cobalt/render_tree/opacity_filter.h
index 93e8718..113c909 100644
--- a/src/cobalt/render_tree/opacity_filter.h
+++ b/src/cobalt/render_tree/opacity_filter.h
@@ -27,6 +27,10 @@
  public:
   explicit OpacityFilter(float opacity) { set_opacity(opacity); }
 
+  bool operator==(const OpacityFilter& other) const {
+    return opacity_ == other.opacity_;
+  }
+
   void set_opacity(float opacity) {
     DCHECK_LE(0.0f, opacity);
     DCHECK_GE(1.0f, opacity);
diff --git a/src/cobalt/render_tree/punch_through_video_node.h b/src/cobalt/render_tree/punch_through_video_node.h
index 3683d6a..f96da9a 100644
--- a/src/cobalt/render_tree/punch_through_video_node.h
+++ b/src/cobalt/render_tree/punch_through_video_node.h
@@ -45,6 +45,10 @@
     Builder(const math::RectF& rect, const SetBoundsCB& set_bounds_cb)
         : rect(rect), set_bounds_cb(set_bounds_cb) {}
 
+    bool operator==(const Builder& other) const {
+      return rect == other.rect && set_bounds_cb.Equals(other.set_bounds_cb);
+    }
+
     // The destination rectangle (size includes border).
     math::RectF rect;
     const SetBoundsCB set_bounds_cb;
diff --git a/src/cobalt/render_tree/rect_node.cc b/src/cobalt/render_tree/rect_node.cc
index 17b046e..b628303 100644
--- a/src/cobalt/render_tree/rect_node.cc
+++ b/src/cobalt/render_tree/rect_node.cc
@@ -84,6 +84,23 @@
       border(moved->border.Pass()),
       rounded_corners(moved->rounded_corners.Pass()) {}
 
+bool RectNode::Builder::operator==(const Builder& other) const {
+  bool brush_equals = !background_brush && !other.background_brush;
+  if (background_brush && other.background_brush) {
+    EqualsBrushVisitor brush_equals_visitor(background_brush.get());
+    other.background_brush->Accept(&brush_equals_visitor);
+    brush_equals = brush_equals_visitor.result();
+  }
+  bool border_equals = (!border && !other.border) ||
+                       (border && other.border && *border == *other.border);
+  bool rounded_corners_equals = (!rounded_corners && !other.rounded_corners) ||
+                                (rounded_corners && other.rounded_corners &&
+                                 *rounded_corners == *other.rounded_corners);
+
+  return rect == other.rect && brush_equals && border_equals &&
+         rounded_corners_equals;
+}
+
 void RectNode::Accept(NodeVisitor* visitor) { visitor->Visit(this); }
 
 math::RectF RectNode::GetBounds() const { return data_.rect; }
diff --git a/src/cobalt/render_tree/rect_node.h b/src/cobalt/render_tree/rect_node.h
index a257cd8..c70fd44 100644
--- a/src/cobalt/render_tree/rect_node.h
+++ b/src/cobalt/render_tree/rect_node.h
@@ -52,6 +52,8 @@
     explicit Builder(const Builder& other);
     explicit Builder(Moved moved);
 
+    bool operator==(const Builder& other) const;
+
     // The destination rectangle (size includes border).
     math::RectF rect;
 
diff --git a/src/cobalt/render_tree/rect_shadow_node.h b/src/cobalt/render_tree/rect_shadow_node.h
index 072c80b..8ff107e 100644
--- a/src/cobalt/render_tree/rect_shadow_node.h
+++ b/src/cobalt/render_tree/rect_shadow_node.h
@@ -38,6 +38,11 @@
             float spread)
         : rect(rect), shadow(shadow), inset(inset), spread(spread) {}
 
+    bool operator==(const Builder& other) const {
+      return rect == other.rect && rounded_corners == other.rounded_corners &&
+             shadow == other.shadow && inset == other.inset &&
+             spread == other.spread;
+    }
     // The destination rectangle.
     math::RectF rect;
 
diff --git a/src/cobalt/render_tree/rounded_corners.h b/src/cobalt/render_tree/rounded_corners.h
index 7646f65..ef3bf38 100644
--- a/src/cobalt/render_tree/rounded_corners.h
+++ b/src/cobalt/render_tree/rounded_corners.h
@@ -81,6 +81,12 @@
         bottom_right(bottom_right),
         bottom_left(bottom_left) {}
 
+  bool operator==(const RoundedCorners& other) const {
+    return top_left == other.top_left && top_right == other.top_right &&
+           bottom_right == other.bottom_right &&
+           bottom_left == other.bottom_left;
+  }
+
   RoundedCorners Inset(float left, float top, float right, float bottom) const {
     return RoundedCorners(
         top_left.Inset(left, top), top_right.Inset(right, top),
diff --git a/src/cobalt/render_tree/shadow.h b/src/cobalt/render_tree/shadow.h
index caefb8a..057c30a 100644
--- a/src/cobalt/render_tree/shadow.h
+++ b/src/cobalt/render_tree/shadow.h
@@ -32,6 +32,11 @@
          const ColorRGBA& color)
       : offset(offset), blur_sigma(blur_sigma), color(color) {}
 
+  bool operator==(const Shadow& other) const {
+    return offset == other.offset && blur_sigma == other.blur_sigma &&
+           color == other.color;
+  }
+
   // Since the blur parameters represent standard deviations, most of the
   // blur is contained within 3 of them, so report that as the extent of the
   // blur.
diff --git a/src/cobalt/render_tree/text_node.h b/src/cobalt/render_tree/text_node.h
index 2dde330..3d35b43 100644
--- a/src/cobalt/render_tree/text_node.h
+++ b/src/cobalt/render_tree/text_node.h
@@ -46,6 +46,11 @@
             const scoped_refptr<GlyphBuffer>& glyph_buffer,
             const ColorRGBA& color);
 
+    bool operator==(const Builder& other) const {
+      return offset == other.offset && glyph_buffer == other.glyph_buffer &&
+             color == other.color && shadows == other.shadows;
+    }
+
     math::Vector2dF offset;
 
     // All of the glyph data needed to render the text.
diff --git a/src/cobalt/render_tree/viewport_filter.h b/src/cobalt/render_tree/viewport_filter.h
index 92f3d58..e4a0afa 100644
--- a/src/cobalt/render_tree/viewport_filter.h
+++ b/src/cobalt/render_tree/viewport_filter.h
@@ -33,6 +33,11 @@
                           const RoundedCorners& rounded_corners)
       : viewport_(viewport), rounded_corners_(rounded_corners) {}
 
+  bool operator==(const ViewportFilter& other) const {
+    return viewport_ == other.viewport_ &&
+           rounded_corners_ == other.rounded_corners_;
+  }
+
   const math::RectF& viewport() const { return viewport_; }
 
   bool has_rounded_corners() const { return !!rounded_corners_; }
diff --git a/src/cobalt/renderer/glimp_shaders/glsl/fragment_skia_texture_masked_texture_domain.glsl b/src/cobalt/renderer/glimp_shaders/glsl/fragment_skia_texture_masked_texture_domain.glsl
index 168df28..6db8fa8 100644
--- a/src/cobalt/renderer/glimp_shaders/glsl/fragment_skia_texture_masked_texture_domain.glsl
+++ b/src/cobalt/renderer/glimp_shaders/glsl/fragment_skia_texture_masked_texture_domain.glsl
@@ -7,7 +7,7 @@
 varying vec2 vMatrixCoord_Stage0;

 varying vec2 vMatrixCoord_Stage1;

 

-void main() 

+void main()

 {

   vec4 output_Stage0;

   {

diff --git a/src/cobalt/renderer/pipeline.cc b/src/cobalt/renderer/pipeline.cc
index 3f8d7dd..8827ef4 100644
--- a/src/cobalt/renderer/pipeline.cc
+++ b/src/cobalt/renderer/pipeline.cc
@@ -82,18 +82,23 @@
       submission_disposal_thread_("Rasterizer Submission Disposal"),
       submit_even_if_render_tree_is_unchanged_(
           submit_even_if_render_tree_is_unchanged),
-      last_render_animations_active_(false),
+      last_did_rasterize_(false),
+      last_animations_expired_(true),
       rasterize_periodic_timer_("Renderer.Rasterize.Duration",
                                 kRasterizePeriodicTimerEntriesPerUpdate,
                                 false /*enable_entry_list_c_val*/),
-      ALLOW_THIS_IN_INITIALIZER_LIST(rasterize_animations_interval_timer_(
-          "Renderer.Rasterize.AnimationsInterval",
-          kRasterizeAnimationsTimerMaxEntries, true /*enable_entry_list_c_val*/,
-          base::Bind(&Pipeline::FrameStatsOnFlushCallback,
-                     base::Unretained(this)))),
       rasterize_animations_timer_("Renderer.Rasterize.Animations",
                                   kRasterizeAnimationsTimerMaxEntries,
                                   true /*enable_entry_list_c_val*/),
+      ALLOW_THIS_IN_INITIALIZER_LIST(rasterize_periodic_interval_timer_(
+          "Renderer.Rasterize.DurationInterval",
+          kRasterizeAnimationsTimerMaxEntries, true /*enable_entry_list_c_val*/,
+          base::Bind(&Pipeline::FrameStatsOnFlushCallback,
+                     base::Unretained(this)))),
+      rasterize_animations_interval_timer_(
+          "Renderer.Rasterize.AnimationsInterval",
+          kRasterizeAnimationsTimerMaxEntries,
+          true /*enable_entry_list_c_val*/),
       new_render_tree_rasterize_count_(
           "Count.Renderer.Rasterize.NewRenderTree", 0,
           "Total number of new render trees rasterized."),
@@ -136,7 +141,7 @@
       clear_on_shutdown_mode_(clear_on_shutdown_mode),
       enable_fps_stdout_(options.enable_fps_stdout),
       enable_fps_overlay_(options.enable_fps_overlay),
-      fps_overlay_updated_(false) {
+      fps_overlay_update_pending_(false) {
   TRACE_EVENT0("cobalt::renderer", "Pipeline::Pipeline()");
   // The actual Pipeline can be constructed from any thread, but we want
   // rasterizer_thread_checker_ to be associated with the rasterizer thread,
@@ -170,6 +175,7 @@
   // must be destroyed before we shutdown the rasterizer thread since it may
   // contain references to render tree nodes and resources.
   last_render_tree_ = NULL;
+  last_animated_render_tree_ = NULL;
 
   // Submit a shutdown task to the rasterizer thread so that it can shutdown
   // anything that must be shutdown from that thread.
@@ -263,18 +269,19 @@
   DCHECK(rasterizer_thread_checker_.CalledOnValidThread());
   TRACE_EVENT0("cobalt::renderer", "Pipeline::RasterizeCurrentTree()");
 
-  base::TimeTicks now = base::TimeTicks::Now();
-  Submission submission = submission_queue_->GetCurrentSubmission(now);
+  base::TimeTicks start_rasterize_time = base::TimeTicks::Now();
+  Submission submission =
+      submission_queue_->GetCurrentSubmission(start_rasterize_time);
 
   bool is_new_render_tree = submission.render_tree != last_render_tree_;
   bool has_render_tree_changed =
-      last_render_animations_active_ || is_new_render_tree;
+      !last_animations_expired_ || is_new_render_tree;
 
   // If our render tree hasn't changed from the one that was previously
   // rendered and it's okay on this system to not flip the display buffer
   // frequently, then we can just not do anything here.
-  if (!fps_overlay_updated_ && !submit_even_if_render_tree_is_unchanged_ &&
-      !has_render_tree_changed) {
+  if (!fps_overlay_update_pending_ &&
+      !submit_even_if_render_tree_is_unchanged_ && !has_render_tree_changed) {
     return;
   }
 
@@ -283,77 +290,93 @@
   render_tree::animations::AnimateNode* animate_node =
       base::polymorphic_downcast<render_tree::animations::AnimateNode*>(
           submission.render_tree.get());
-  bool are_animations_active = animate_node->expiry() > submission.time_offset;
 
-  // If animations are going from being inactive to active, then set the c_val
-  // prior to starting the animation so that it's in the correct state while the
-  // tree is being rendered.
-  // Also, start the interval timer now. While the first entry only captures a
-  // partial interval, it's recorded to include the duration of the first
-  // submission. All subsequent entries will record a full interval.
-  if (!last_render_animations_active_ && are_animations_active) {
-    has_active_animations_c_val_ = true;
-    rasterize_animations_interval_timer_.Start(now);
-  }
+  // Rasterize the last submitted render tree.
+  bool did_rasterize =
+      RasterizeSubmissionToRenderTarget(submission, render_target_);
+
+  bool animations_expired = animate_node->expiry() <= submission.time_offset;
+
+  UpdateRasterizeStats(did_rasterize, animations_expired, is_new_render_tree,
+                       start_rasterize_time, base::TimeTicks::Now());
+
+  last_did_rasterize_ = did_rasterize;
+  last_animations_expired_ = animations_expired;
+}
+
+void Pipeline::UpdateRasterizeStats(bool did_rasterize, bool animations_expired,
+                                    bool is_new_render_tree,
+                                    base::TimeTicks start_time,
+                                    base::TimeTicks end_time) {
+  bool last_animations_active =
+      !last_animations_expired_ && last_did_rasterize_;
+  bool animations_active = !animations_expired && did_rasterize;
 
   // The rasterization is only timed with the periodic timer when the render
   // tree has changed. This ensures that the frames being timed are consistent
   // between platforms that submit unchanged trees and those that don't.
-  bool should_run_periodic_timer = has_render_tree_changed;
-
-  // The rasterization is only timed with the animations timer when there are
-  // animations to track. This applies when animations were active during either
-  // the last rasterization or the current one. The reason for including the
-  // last one is that if animations have just expired, then this rasterization
-  // produces the final state of the animated tree.
-  bool should_run_animations_timer =
-      last_render_animations_active_ || are_animations_active;
-
-  if (should_run_periodic_timer) {
-    rasterize_periodic_timer_.Start(now);
-  }
-  if (should_run_animations_timer) {
-    rasterize_animations_timer_.Start(now);
+  if (did_rasterize) {
+    rasterize_periodic_timer_.Start(start_time);
+    rasterize_periodic_timer_.Stop(end_time);
   }
 
-  // Rasterize the last submitted render tree.
-  RasterizeSubmissionToRenderTarget(submission, render_target_);
+  if (last_animations_active || animations_active) {
+    // The rasterization is only timed with the animations timer when there are
+    // animations to track. This applies when animations were active during
+    // either the last rasterization or the current one. The reason for
+    // including the last one is that if animations have just expired, then this
+    // rasterization produces the final state of the animated tree.
+    if (did_rasterize) {
+      rasterize_animations_timer_.Start(start_time);
+      rasterize_animations_timer_.Stop(end_time);
+    }
 
-  // Update now with the post-submission time.
-  now = base::TimeTicks::Now();
+    // If animations are going from being inactive to active, then set the c_val
+    // prior to starting the animation so that it's in the correct state while
+    // the tree is being rendered.
+    // Also, start the interval timer now. While the first entry only captures a
+    // partial interval, it's recorded to include the duration of the first
+    // submission. All subsequent entries will record a full interval.
+    if (!last_animations_active && animations_active) {
+      has_active_animations_c_val_ = true;
+      rasterize_periodic_interval_timer_.Start(start_time);
+      rasterize_animations_interval_timer_.Start(start_time);
+    }
 
-  if (should_run_periodic_timer) {
-    rasterize_periodic_timer_.Stop(now);
-  }
-  if (should_run_animations_timer) {
-    rasterize_animations_interval_timer_.Stop(now);
-    rasterize_animations_timer_.Stop(now);
+    if (!did_rasterize) {
+      // If we didn't actually rasterize anything, don't count this sample.
+      rasterize_periodic_interval_timer_.Cancel();
+      rasterize_animations_interval_timer_.Cancel();
+    } else {
+      rasterize_periodic_interval_timer_.Stop(end_time);
+      rasterize_animations_interval_timer_.Stop(end_time);
+    }
+
     // If animations are active, then they are guaranteed at least one more
     // interval. Start the timer to record its duration.
-    if (are_animations_active) {
-      rasterize_animations_interval_timer_.Start(now);
+    if (animations_active) {
+      rasterize_periodic_interval_timer_.Start(end_time);
+      rasterize_animations_interval_timer_.Start(end_time);
+    }
+
+    // Check for if the animations are starting or ending.
+    if (!last_animations_active && animations_active) {
+      animations_start_time_ = end_time.ToInternalValue();
+    } else if (last_animations_active && !animations_active) {
+      animations_end_time_ = end_time.ToInternalValue();
+      has_active_animations_c_val_ = false;
+      rasterize_animations_interval_timer_.Flush();
+      rasterize_animations_timer_.Flush();
     }
   }
 
   if (is_new_render_tree) {
     ++new_render_tree_rasterize_count_;
-    new_render_tree_rasterize_time_ = now.ToInternalValue();
+    new_render_tree_rasterize_time_ = end_time.ToInternalValue();
   }
-
-  // Check for if the animations are starting or ending.
-  if (!last_render_animations_active_ && are_animations_active) {
-    animations_start_time_ = now.ToInternalValue();
-  } else if (last_render_animations_active_ && !are_animations_active) {
-    animations_end_time_ = now.ToInternalValue();
-    has_active_animations_c_val_ = false;
-    rasterize_animations_interval_timer_.Flush();
-    rasterize_animations_timer_.Flush();
-  }
-
-  last_render_animations_active_ = are_animations_active;
 }
 
-void Pipeline::RasterizeSubmissionToRenderTarget(
+bool Pipeline::RasterizeSubmissionToRenderTarget(
     const Submission& submission,
     const scoped_refptr<backend::RenderTarget>& render_target) {
   TRACE_EVENT0("cobalt::renderer",
@@ -364,14 +387,17 @@
   // |previous_animated_area_|.
   if (submission.render_tree != last_render_tree_) {
     last_render_tree_ = submission.render_tree;
+    last_animated_render_tree_ = NULL;
     previous_animated_area_ = base::nullopt;
     last_render_time_ = base::nullopt;
   }
 
   // Animate the render tree using the submitted animations.
   render_tree::animations::AnimateNode* animate_node =
-      base::polymorphic_downcast<render_tree::animations::AnimateNode*>(
-          submission.render_tree.get());
+      last_animated_render_tree_
+          ? last_animated_render_tree_.get()
+          : base::polymorphic_downcast<render_tree::animations::AnimateNode*>(
+                submission.render_tree.get());
 
   // Some animations require a GL graphics context to be current.  Specifically,
   // a call to SbPlayerGetCurrentFrame() may be made to get the current video
@@ -381,6 +407,13 @@
   render_tree::animations::AnimateNode::AnimateResults results =
       animate_node->Apply(submission.time_offset);
 
+  if (results.animated == last_animated_render_tree_ &&
+      !submit_even_if_render_tree_is_unchanged_ &&
+      !fps_overlay_update_pending_) {
+    return false;
+  }
+  last_animated_render_tree_ = results.animated;
+
   // Calculate a bounding box around the active animations.  Union it with the
   // bounding box around active animations from the previous frame, and we get
   // a scissor rectangle marking the dirty regions of the screen.
@@ -393,9 +426,11 @@
   }
   previous_animated_area_ = rounded_bounds;
 
-  scoped_refptr<render_tree::Node> submit_tree = results.animated;
+  scoped_refptr<render_tree::Node> submit_tree = results.animated->source();
   if (enable_fps_overlay_ && fps_overlay_) {
-    submit_tree = fps_overlay_->AnnotateRenderTreeWithOverlay(results.animated);
+    submit_tree =
+        fps_overlay_->AnnotateRenderTreeWithOverlay(results.animated->source());
+    fps_overlay_update_pending_ = false;
   }
 
   // Rasterize the animated render tree.
@@ -408,6 +443,8 @@
   }
 
   last_render_time_ = submission.time_offset;
+
+  return true;
 }
 
 void Pipeline::InitializeRasterizerThread(
@@ -499,7 +536,8 @@
   render_tree::animations::AnimateNode::AnimateResults results =
       animate_node->Apply(submission.time_offset);
 
-  std::string tree_dump = render_tree::DumpRenderTreeToString(results.animated);
+  std::string tree_dump =
+      render_tree::DumpRenderTreeToString(results.animated->source());
   if (message.empty() || message == "undefined") {
     // If no filename was specified, send output to the console.
     LOG(INFO) << tree_dump.c_str();
@@ -533,6 +571,7 @@
   }
 
   enable_fps_overlay_ = !enable_fps_overlay_;
+  fps_overlay_update_pending_ = enable_fps_overlay_;
 }
 #endif  // #if defined(ENABLE_DEBUG_CONSOLE)
 
@@ -575,7 +614,7 @@
     }
 
     fps_overlay_->UpdateOverlay(flush_results);
-    fps_overlay_updated_ = true;
+    fps_overlay_update_pending_ = true;
   }
 
   if (enable_fps_stdout_) {
diff --git a/src/cobalt/renderer/pipeline.h b/src/cobalt/renderer/pipeline.h
index e304e59..49308c9 100644
--- a/src/cobalt/renderer/pipeline.h
+++ b/src/cobalt/renderer/pipeline.h
@@ -118,10 +118,17 @@
 
   // Rasterize the animated |render_tree_submission| to |render_target|,
   // applying the time_offset in the submission to the animations.
-  void RasterizeSubmissionToRenderTarget(
+  // Returns true only if a rasterization actually took place.
+  bool RasterizeSubmissionToRenderTarget(
       const Submission& render_tree_submission,
       const scoped_refptr<backend::RenderTarget>& render_target);
 
+  // Updates the rasterizer timer stats according to the |start_time| and
+  // |end_time| of the most recent rasterize call.
+  void UpdateRasterizeStats(bool did_rasterize, bool are_animations_active,
+                            bool is_new_render_tree, base::TimeTicks start_time,
+                            base::TimeTicks end_time);
+
   // This method is executed on the rasterizer thread and is responsible for
   // constructing the rasterizer.
   void InitializeRasterizerThread(
@@ -193,6 +200,10 @@
 
   // Keeps track of the last rendered animated render tree.
   scoped_refptr<render_tree::Node> last_render_tree_;
+
+  scoped_refptr<render_tree::animations::AnimateNode>
+      last_animated_render_tree_;
+
   // Keeps track of the area of the screen that animations previously existed
   // within, so that we can know which regions of the screens would be dirty
   // next frame.
@@ -201,22 +212,31 @@
   base::optional<base::TimeDelta> last_render_time_;
   // Keep track of whether the last rendered tree had active animations. This
   // allows us to skip rasterizing that render tree if we see it again and it
-  // did not have active animations.
-  bool last_render_animations_active_;
+  // did have expired animations.
+  bool last_animations_expired_;
+
+  // Did a rasterization take place in the last frame?
+  bool last_did_rasterize_;
 
   // Timer tracking the amount of time spent in
   // |RasterizeSubmissionToRenderTarget| when the render tree has changed.
   // The tracking is flushed when the max count is hit.
   base::CValCollectionTimerStats<base::CValPublic> rasterize_periodic_timer_;
+  // Timer tracking the amount of time spent in
+  // |RasterizeSubmissionToRenderTarget| while animations are active. The
+  // tracking is flushed when the animations expire.
+  base::CValCollectionTimerStats<base::CValDebug> rasterize_animations_timer_;
+
+  // Accumulates render tree rasterization interval times but does not flush
+  // them until the maximum number of samples is gathered.
+  base::CValCollectionTimerStats<base::CValPublic>
+      rasterize_periodic_interval_timer_;
+
   // Timer tracking the amount of time between calls to
   // |RasterizeSubmissionToRenderTarget| while animations are active. The
   // tracking is flushed when the animations expire.
   base::CValCollectionTimerStats<base::CValPublic>
       rasterize_animations_interval_timer_;
-  // Timer tracking the amount of time spent in
-  // |RasterizeSubmissionToRenderTarget| while animations are active. The
-  // tracking is flushed when the animations expire.
-  base::CValCollectionTimerStats<base::CValDebug> rasterize_animations_timer_;
 
   // The total number of new render trees that have been rasterized.
   base::CVal<int> new_render_tree_rasterize_count_;
@@ -254,7 +274,9 @@
   bool enable_fps_overlay_;
 
   base::optional<FpsOverlay> fps_overlay_;
-  bool fps_overlay_updated_;
+
+  // True if the overlay has been updated and it needs to be re-rasterized.
+  bool fps_overlay_update_pending_;
 };
 
 }  // namespace renderer
diff --git a/src/cobalt/renderer/rasterizer/egl/draw_rect_shadow_blur.cc b/src/cobalt/renderer/rasterizer/egl/draw_rect_shadow_blur.cc
index fcabd2d..3261089 100644
--- a/src/cobalt/renderer/rasterizer/egl/draw_rect_shadow_blur.cc
+++ b/src/cobalt/renderer/rasterizer/egl/draw_rect_shadow_blur.cc
@@ -42,41 +42,55 @@
 
 void SetBlurRRectUniforms(const ShaderFragmentColorBlurRrects& shader,
     math::RectF rect, render_tree::RoundedCorners corners, float sigma) {
-  // Ensure a minimum radius for each corner to avoid division by zero.
-  const float kMinSize = 0.01f;
+  const float kBlurExtentInPixels = kBlurExtentInSigmas * sigma;
+  const float kOffsetScale = kBlurDistance / kBlurExtentInPixels;
 
+  // Ensure a minimum radius for each corner to avoid division by zero.
+  // NOTE: The rounded rect is already specified in terms of sigma.
+  const float kMinSize = 0.01f * kOffsetScale;
   rect.Outset(kMinSize, kMinSize);
   corners = corners.Inset(-kMinSize, -kMinSize, -kMinSize, -kMinSize);
   corners = corners.Normalize(rect);
 
-  // Specify the blur extent size and the (min.y, max.y) for the rect.
-  const float kBlurExtentInPixels = kBlurExtentInSigmas * sigma;
-  GL_CALL(glUniform3f(shader.u_blur_extent(),
-                      kBlurExtentInPixels, rect.y(), rect.bottom()));
+  // A normalized rounded rect should have at least one Y value which the
+  // corners do not cross.
+  const float kCenterY =
+      0.5f * (rect.y() + std::max(corners.top_left.vertical,
+                                  corners.top_right.vertical)) +
+      0.5f * (rect.bottom() - std::max(corners.bottom_left.vertical,
+                                       corners.bottom_right.vertical));
 
-  // Set the "start" and "scale" values so that (pos - start) * scale is in the
-  // first quadrant and normalized. Then specify "radius" so that normalized *
-  // radius + start specifies a point on the respective corner.
-  GL_CALL(glUniform4f(shader.u_spread_start_x(),
+  // The blur extent is (blur_size.y, rect_min.y, rect_max.y, rect_center.y).
+  GL_CALL(glUniform4f(shader.u_blur_extent(),
+                      kBlurExtentInPixels * kOffsetScale,
+                      rect.y(), rect.bottom(), kCenterY));
+
+  // The blur rounded rect is split into top and bottom halves.
+  // The "start" values represent (left_start.xy, right_start.xy).
+  // The "scale" values represent (left_radius.x, 1 / left_radius.y,
+  //   right_radius.x, 1 / right_radius.y). The sign of the scale value helps
+  //   to translate between position and corner offset values, where the corner
+  //   offset is positive if the position is inside the rounded corner.
+  GL_CALL(glUniform4f(shader.u_blur_start_top(),
                       rect.x() + corners.top_left.horizontal,
-                      rect.right() - corners.top_right.horizontal,
-                      rect.x() + corners.bottom_left.horizontal,
-                      rect.right() - corners.bottom_right.horizontal));
-  GL_CALL(glUniform4f(shader.u_spread_start_y(),
                       rect.y() + corners.top_left.vertical,
-                      rect.y() + corners.top_right.vertical,
+                      rect.right() - corners.top_right.horizontal,
+                      rect.y() + corners.top_right.vertical));
+  GL_CALL(glUniform4f(shader.u_blur_start_bottom(),
+                      rect.x() + corners.bottom_left.horizontal,
                       rect.bottom() - corners.bottom_left.vertical,
+                      rect.right() - corners.bottom_right.horizontal,
                       rect.bottom() - corners.bottom_right.vertical));
-  GL_CALL(glUniform4f(shader.u_spread_scale_y(),
-                      -1.0f / corners.top_left.vertical,
-                      -1.0f / corners.top_right.vertical,
-                      1.0f / corners.bottom_left.vertical,
-                      1.0f / corners.bottom_right.vertical));
-  GL_CALL(glUniform4f(shader.u_spread_radius_x(),
+  GL_CALL(glUniform4f(shader.u_blur_scale_top(),
                       -corners.top_left.horizontal,
+                      -1.0f / corners.top_left.vertical,
                       corners.top_right.horizontal,
+                      -1.0f / corners.top_right.vertical));
+  GL_CALL(glUniform4f(shader.u_blur_scale_bottom(),
                       -corners.bottom_left.horizontal,
-                      corners.bottom_right.horizontal));
+                      1.0f / corners.bottom_left.vertical,
+                      corners.bottom_right.horizontal,
+                      1.0f / corners.bottom_right.vertical));
 }
 }  // namespace
 
@@ -89,10 +103,12 @@
 }
 
 DrawRectShadowBlur::VertexAttributesRound::VertexAttributesRound(
-    float x, float y, const RCorner& init) {
+    float x, float y, float offset_scale, const RCorner& rcorner) {
   position[0] = x;
   position[1] = y;
-  rcorner_scissor = RCorner(position, init);
+  offset[0] = x * offset_scale;
+  offset[1] = y * offset_scale;
+  rcorner_scissor = RCorner(position, rcorner);
 }
 
 DrawRectShadowBlur::DrawRectShadowBlur(GraphicsState* graphics_state,
@@ -117,12 +133,10 @@
   OptionalRoundedCorners scaled_base_corners(base_corners);
   if (scaled_base_corners) {
     scaled_base_corners = scaled_base_corners->Scale(scale.x(), scale.y());
-    scaled_base_corners = scaled_base_corners->Normalize(scaled_base_rect);
   }
   spread_rect_.Scale(scale.x(), scale.y());
   if (spread_corners_) {
     spread_corners_ = spread_corners_->Scale(scale.x(), scale.y());
-    spread_corners_ = spread_corners_->Normalize(spread_rect_);
   }
 
   // The blur algorithms used by the shaders do not produce good results with
@@ -168,13 +182,6 @@
     SetupVertexShader(graphics_state, program->GetVertexShader());
     SetFragmentUniforms(program->GetFragmentShader().u_color(),
                         program->GetFragmentShader().u_scale_add());
-    float sigma_scale = kBlurDistance / (kBlurExtentInSigmas * blur_sigma_);
-    GL_CALL(glUniform2f(program->GetFragmentShader().u_sigma_scale(),
-                        sigma_scale, sigma_scale));
-    // Pre-calculate the scale values to calculate the normalized gaussian.
-    GL_CALL(glUniform2f(program->GetFragmentShader().u_gaussian_scale(),
-                        -1.0f / (2.0f * blur_sigma_ * blur_sigma_),
-                        1.0f / (kSqrt2 * kSqrtPi * blur_sigma_)));
     SetBlurRRectUniforms(program->GetFragmentShader(),
                          spread_rect_, *spread_corners_, blur_sigma_);
     GL_CALL(glDrawElements(GL_TRIANGLES, indices_.size(), GL_UNSIGNED_SHORT,
@@ -234,6 +241,10 @@
       sizeof(VertexAttributesRound), vertex_buffer_ +
       offsetof(VertexAttributesRound, position));
   graphics_state->VertexAttribPointer(
+      shader.a_offset(), 2, GL_FLOAT, GL_FALSE,
+      sizeof(VertexAttributesRound), vertex_buffer_ +
+      offsetof(VertexAttributesRound, offset));
+  graphics_state->VertexAttribPointer(
       shader.a_rcorner(), 4, GL_FLOAT, GL_FALSE,
       sizeof(VertexAttributesRound), vertex_buffer_ +
       offsetof(VertexAttributesRound, rcorner_scissor));
@@ -310,10 +321,9 @@
 
 void DrawRectShadowBlur::SetGeometry(GraphicsState* graphics_state,
     const math::RectF& inner_rect, const math::RectF& outer_rect) {
-  // Express offset in terms of blur sigma for the shader.
+  // The spread rect and offsets should be expressed in terms of sigma for the
+  // shader.
   float offset_scale = kBlurDistance / (kBlurExtentInSigmas * blur_sigma_);
-
-  // The spread rect should also be expressed in terms of sigma.
   spread_rect_.Scale(offset_scale, offset_scale);
 
   // The box shadow is a triangle strip covering the area between outer rect
@@ -360,21 +370,15 @@
 
 void DrawRectShadowBlur::SetGeometry(GraphicsState* graphics_state,
     const RRectAttributes (&rrect)[8]) {
+  // The spread rect and offsets should be expressed in terms of sigma for the
+  // shader.
+  float offset_scale = kBlurDistance / (kBlurExtentInSigmas * blur_sigma_);
+  spread_rect_.Scale(offset_scale, offset_scale);
+  *spread_corners_ = spread_corners_->Scale(offset_scale, offset_scale);
+
   // The shadowed area is already split into quads.
   for (int i = 0; i < arraysize(rrect); ++i) {
-    uint16_t vert = static_cast<uint16_t>(attributes_round_.size());
-    const math::RectF& bounds = rrect[i].bounds;
-    const RCorner& rcorner = rrect[i].rcorner;
-    attributes_round_.emplace_back(bounds.x(), bounds.y(), rcorner);
-    attributes_round_.emplace_back(bounds.right(), bounds.y(), rcorner);
-    attributes_round_.emplace_back(bounds.x(), bounds.bottom(), rcorner);
-    attributes_round_.emplace_back(bounds.right(), bounds.bottom(), rcorner);
-    indices_.emplace_back(vert);
-    indices_.emplace_back(vert + 1);
-    indices_.emplace_back(vert + 2);
-    indices_.emplace_back(vert + 1);
-    indices_.emplace_back(vert + 2);
-    indices_.emplace_back(vert + 3);
+    AddQuad(rrect[i].bounds, offset_scale, rrect[i].rcorner);
   }
 
   graphics_state->ReserveVertexData(
@@ -385,6 +389,12 @@
 void DrawRectShadowBlur::SetGeometry(GraphicsState* graphics_state,
     const RRectAttributes (&rrect_outer)[4],
     const RRectAttributes (&rrect_inner)[8]) {
+  // The spread rect and offsets should be expressed in terms of sigma for the
+  // shader.
+  float offset_scale = kBlurDistance / (kBlurExtentInSigmas * blur_sigma_);
+  spread_rect_.Scale(offset_scale, offset_scale);
+  *spread_corners_ = spread_corners_->Scale(offset_scale, offset_scale);
+
   // Draw the area between the inner rect and outer rect using the outer rect's
   // rounded corners. The inner quads already exclude the inscribed rectangle.
   for (int i = 0; i < arraysize(rrect_inner); ++i) {
@@ -392,19 +402,7 @@
       math::RectF rect = math::IntersectRects(
           rrect_inner[i].bounds, rrect_outer[o].bounds);
       if (!rect.IsEmpty()) {
-        // Use two triangles to draw the intersection.
-        const RCorner& rcorner = rrect_outer[o].rcorner;
-        uint16_t vert = static_cast<uint16_t>(attributes_round_.size());
-        attributes_round_.emplace_back(rect.x(), rect.y(), rcorner);
-        attributes_round_.emplace_back(rect.right(), rect.y(), rcorner);
-        attributes_round_.emplace_back(rect.x(), rect.bottom(), rcorner);
-        attributes_round_.emplace_back(rect.right(), rect.bottom(), rcorner);
-        indices_.emplace_back(vert);
-        indices_.emplace_back(vert + 1);
-        indices_.emplace_back(vert + 2);
-        indices_.emplace_back(vert + 1);
-        indices_.emplace_back(vert + 2);
-        indices_.emplace_back(vert + 3);
+        AddQuad(rect, offset_scale, rrect_outer[o].rcorner);
       }
     }
   }
@@ -414,6 +412,21 @@
   graphics_state->ReserveVertexIndices(indices_.size());
 }
 
+void DrawRectShadowBlur::AddQuad(const math::RectF& rect, float scale,
+    const RCorner& rcorner) {
+  uint16_t vert = static_cast<uint16_t>(attributes_round_.size());
+  attributes_round_.emplace_back(rect.x(), rect.y(), scale, rcorner);
+  attributes_round_.emplace_back(rect.right(), rect.y(), scale, rcorner);
+  attributes_round_.emplace_back(rect.x(), rect.bottom(), scale, rcorner);
+  attributes_round_.emplace_back(rect.right(), rect.bottom(), scale, rcorner);
+  indices_.emplace_back(vert);
+  indices_.emplace_back(vert + 1);
+  indices_.emplace_back(vert + 2);
+  indices_.emplace_back(vert + 1);
+  indices_.emplace_back(vert + 2);
+  indices_.emplace_back(vert + 3);
+}
+
 }  // namespace egl
 }  // namespace rasterizer
 }  // namespace renderer
diff --git a/src/cobalt/renderer/rasterizer/egl/draw_rect_shadow_blur.h b/src/cobalt/renderer/rasterizer/egl/draw_rect_shadow_blur.h
index 107af85..6ba4e85 100644
--- a/src/cobalt/renderer/rasterizer/egl/draw_rect_shadow_blur.h
+++ b/src/cobalt/renderer/rasterizer/egl/draw_rect_shadow_blur.h
@@ -73,8 +73,10 @@
   };
 
   struct VertexAttributesRound {
-    VertexAttributesRound(float x, float y, const RCorner& init);
+    VertexAttributesRound(float x, float y, float offset_scale,
+                          const RCorner& rcorner);
     float position[2];
+    float offset[2];
     RCorner rcorner_scissor;
   };
 
@@ -95,6 +97,7 @@
   void SetGeometry(GraphicsState* graphics_state,
                    const RRectAttributes (&rrect_outer)[4],
                    const RRectAttributes (&rrect_inner)[8]);
+  void AddQuad(const math::RectF& rect, float scale, const RCorner& rcorner);
 
   math::RectF spread_rect_;
   OptionalRoundedCorners spread_corners_;
diff --git a/src/cobalt/renderer/rasterizer/egl/hardware_rasterizer.cc b/src/cobalt/renderer/rasterizer/egl/hardware_rasterizer.cc
index bcd8e49..6043afa 100644
--- a/src/cobalt/renderer/rasterizer/egl/hardware_rasterizer.cc
+++ b/src/cobalt/renderer/rasterizer/egl/hardware_rasterizer.cc
@@ -300,6 +300,8 @@
           purge_skia_font_caches_on_destruction)) {
 }
 
+HardwareRasterizer::~HardwareRasterizer() {}
+
 void HardwareRasterizer::Submit(
     const scoped_refptr<render_tree::Node>& render_tree,
     const scoped_refptr<backend::RenderTarget>& render_target,
diff --git a/src/cobalt/renderer/rasterizer/egl/hardware_rasterizer.h b/src/cobalt/renderer/rasterizer/egl/hardware_rasterizer.h
index cd7eb6e..fd3c144 100644
--- a/src/cobalt/renderer/rasterizer/egl/hardware_rasterizer.h
+++ b/src/cobalt/renderer/rasterizer/egl/hardware_rasterizer.h
@@ -52,6 +52,7 @@
                               int scratch_surface_cache_size_in_bytes,
                               int offscreen_target_cache_size_in_bytes,
                               bool purge_skia_font_caches_on_destruction);
+  virtual ~HardwareRasterizer();
 
   void Submit(const scoped_refptr<render_tree::Node>& render_tree,
               const scoped_refptr<backend::RenderTarget>& render_target,
diff --git a/src/cobalt/renderer/rasterizer/egl/render_tree_node_visitor.cc b/src/cobalt/renderer/rasterizer/egl/render_tree_node_visitor.cc
index 72b7da7..677f941 100644
--- a/src/cobalt/renderer/rasterizer/egl/render_tree_node_visitor.cc
+++ b/src/cobalt/renderer/rasterizer/egl/render_tree_node_visitor.cc
@@ -64,8 +64,12 @@
 }
 
 bool IsOnlyScaleAndTranslate(const math::Matrix3F& matrix) {
-  return matrix(2, 0) == 0 && matrix(2, 1) == 0 && matrix(2, 2) == 1 &&
-         matrix(0, 1) == 0 && matrix(1, 0) == 0;
+  const float kEpsilon = 0.0001f;
+  return std::abs(matrix(0, 1)) < kEpsilon &&
+         std::abs(matrix(1, 0)) < kEpsilon &&
+         std::abs(matrix(2, 0)) < kEpsilon &&
+         std::abs(matrix(2, 1)) < kEpsilon &&
+         std::abs(matrix(2, 2) - 1.0f) < kEpsilon;
 }
 
 math::Matrix3F GetTexcoordTransform(
@@ -672,15 +676,28 @@
   // Get a suitable cache of the render tree node if one exists, or allocate
   // a new offscreen target if possible. The OffscreenTargetErrorFunction will
   // determine whether any caches are fit for use.
-  *out_content_cached = offscreen_target_manager_->GetCachedOffscreenTarget(
-      node, base::Bind(&OffscreenTargetErrorFunction, mapped_bounds),
-      out_target_info);
-  if (!(*out_content_cached)) {
-    offscreen_target_manager_->AllocateOffscreenTarget(node,
-        content_size, mapped_bounds, out_target_info);
+
+  // Do not cache rotating nodes since these will result in inappropriate
+  // reuse of offscreen targets. Transforms that are rotations of angles in
+  // the first quadrant will produce the same mapped rect sizes as angles in
+  // the other 3 quadrants. Also avoid caching reflections.
+  bool allow_caching = IsOnlyScaleAndTranslate(draw_state_.transform) &&
+                       draw_state_.transform(0, 0) > 0.0f &&
+                       draw_state_.transform(1, 1) > 0.0f;
+  if (allow_caching) {
+    *out_content_cached = offscreen_target_manager_->GetCachedOffscreenTarget(
+        node, base::Bind(&OffscreenTargetErrorFunction, mapped_bounds),
+        out_target_info);
+    if (!(*out_content_cached)) {
+      offscreen_target_manager_->AllocateOffscreenTarget(node,
+          content_size, mapped_bounds, out_target_info);
+    } else {
+      // Maintain the size of the cached contents to avoid scaling artifacts.
+      content_size = out_target_info->region.size();
+    }
   } else {
-    // Maintain the size of the cached contents to avoid scaling artifacts.
-    content_size = out_target_info->region.size();
+    *out_content_cached = false;
+    out_target_info->framebuffer = nullptr;
   }
 
   // If no offscreen target could be allocated, then the render tree node will
diff --git a/src/cobalt/renderer/rasterizer/egl/shaders/fragment_color_blur_rrects.glsl b/src/cobalt/renderer/rasterizer/egl/shaders/fragment_color_blur_rrects.glsl
index efc8747..ffbee99 100644
--- a/src/cobalt/renderer/rasterizer/egl/shaders/fragment_color_blur_rrects.glsl
+++ b/src/cobalt/renderer/rasterizer/egl/shaders/fragment_color_blur_rrects.glsl
@@ -14,26 +14,19 @@
 

 precision mediump float;

 

-// The rounded spread rect is represented in a way to optimize calculation of

-// the extents. Each element of a vec4 represents a corner's value -- order

-// is top left, top right, bottom left, bottom right. Extents for each corner

-// can be calculated as:

-//   extents_x = start_x + radius_x * sqrt(1 - scaled_y^2) where

-//   scaled_y = clamp((pos.yyyy - start_y) * scale_y, 0.0, 1.0)

-// To simplify handling left vs right and top vs bottom corners, the sign of

-// scale_y and radius_x handles negation as needed.

-uniform vec4 u_spread_start_x;

-uniform vec4 u_spread_start_y;

-uniform vec4 u_spread_scale_y;

-uniform vec4 u_spread_radius_x;

+// The blur rounded rect is split into top and bottom halves.

+// The "start" values represent (left_start.xy, right_start.xy).

+// The "scale" values represent (left_radius.x, 1 / left_radius.y,

+//   right_radius.x, 1 / right_radius.y). The sign of the scale value helps

+//   to translate between position and corner offset values, where the corner

+//   offset is positive if the position is inside the rounded corner.

+uniform vec4 u_blur_start_top;

+uniform vec4 u_blur_start_bottom;

+uniform vec4 u_blur_scale_top;

+uniform vec4 u_blur_scale_bottom;

 

-// The blur extent specifies (3 * sigma, min_rect_y, max_rect_y). This is used

-// to clamp the interval over which integration should be evaluated.

-uniform vec3 u_blur_extent;

-

-// The gaussian scale uniform is used to simplify calculation of the gaussian

-// function at a particular point.

-uniform vec2 u_gaussian_scale;

+// The blur extent specifies (blur_size, min_rect_y, max_rect_y, center_rect_y).

+uniform vec4 u_blur_extent;

 

 // The scale_add uniform is used to switch the shader between generating

 // outset shadows and inset shadows. It impacts the shadow gradient and

@@ -44,39 +37,25 @@
 

 uniform vec4 u_color;

 

-// Blur calculations happen in terms in sigma distances. Use sigma_scale to

-// translate pixel distances into sigma distances.

-uniform vec2 u_sigma_scale;

-

 varying vec2 v_offset;

 varying vec4 v_rcorner;

 

 #include "function_is_outside_rcorner.inc"

 #include "function_gaussian_integral.inc"

 

-vec2 GetXExtents(float y) {

-  // Use x^2 / a^2 + y^2 / b^2 = 1 to solve for the x value of each rounded

-  // corner at the given y.

-  vec4 scaled = clamp((y - u_spread_start_y) * u_spread_scale_y, 0.0, 1.0);

-  vec4 root = sqrt(1.0 - scaled * scaled);

-  vec4 extent = u_spread_start_x + u_spread_radius_x * root;

-

-  // If the y value was before a corner started, then the calculated extent

-  // would equal the unrounded rectangle's extents (since negative values were

-  // clamped to 0 in the above calculation). So smaller extents (i.e. extents

-  // closer to the rectangle center), represent the relevant corners' extents.

-  return vec2(max(extent.x, extent.z), min(extent.y, extent.w));

-}

-

 float GetXBlur(float x, float y) {

-  // Get the integral over the interval occupied by the rectangle.

-  vec2 pos = (GetXExtents(y) - x) * u_sigma_scale;

-  return GaussianIntegral(pos);

-}

+  // Solve for X of the rounded corners at the given Y based on the equation

+  // for an ellipse: x^2 / a^2 + y^2 / b^2 = 1.

+  vec4 corner_start =

+      (y < u_blur_extent.w) ? u_blur_start_top : u_blur_start_bottom;

+  vec4 corner_scale =

+      (y < u_blur_extent.w) ? u_blur_scale_top : u_blur_scale_bottom;

+  vec2 scaled = clamp((y - corner_start.yw) * corner_scale.yw, 0.0, 1.0);

+  vec2 root = sqrt(1.0 - scaled * scaled);

+  vec2 extent_x = corner_start.xz + corner_scale.xz * root;

 

-vec3 GetGaussian(vec3 offset) {

-  // Evaluate the gaussian at the given offsets.

-  return exp(offset * offset * u_gaussian_scale.x);

+  // Get the integral over the interval occupied by the rectangle.

+  return GaussianIntegral(extent_x - x);

 }

 

 float GetBlur(vec2 pos) {

@@ -111,12 +90,13 @@
   vec3 xblur2 = vec3(GetXBlur(pos.x, pos2.x),

                      GetXBlur(pos.x, pos2.y),

                      GetXBlur(pos.x, pos2.z));

-  vec3 yblur1 = GetGaussian(offset1) * weight;

-  vec3 yblur2 = GetGaussian(offset2) * weight;

+  vec3 yblur1 = exp(-offset1 * offset1) * weight;

+  vec3 yblur2 = exp(-offset2 * offset2) * weight;

 

-  // Since each yblur value should be scaled by u_gaussian_scale.y, save some

-  // cycles and multiply the sum by it.

-  return (dot(xblur1, yblur1) + dot(xblur2, yblur2)) * u_gaussian_scale.y;

+  // Since each yblur value should be normalized by kNormalizeGaussian, just

+  // scale the sum by it.

+  const float kNormalizeGaussian = 0.564189584;  // 1 / sqrt(pi)

+  return (dot(xblur1, yblur1) + dot(xblur2, yblur2)) * kNormalizeGaussian;

 }

 

 void main() {

diff --git a/src/cobalt/renderer/rasterizer/egl/shaders/vertex_offset_rcorner.glsl b/src/cobalt/renderer/rasterizer/egl/shaders/vertex_offset_rcorner.glsl
index 3d9ff46..fbcead7 100644
--- a/src/cobalt/renderer/rasterizer/egl/shaders/vertex_offset_rcorner.glsl
+++ b/src/cobalt/renderer/rasterizer/egl/shaders/vertex_offset_rcorner.glsl
@@ -15,6 +15,7 @@
 uniform vec4 u_clip_adjustment;

 uniform mat3 u_view_matrix;

 attribute vec2 a_position;

+attribute vec2 a_offset;

 attribute vec4 a_rcorner;

 varying vec2 v_offset;

 varying vec4 v_rcorner;

@@ -23,6 +24,6 @@
   vec3 pos2d = u_view_matrix * vec3(a_position, 1);

   gl_Position = vec4(pos2d.xy * u_clip_adjustment.xy +

                      u_clip_adjustment.zw, 0, pos2d.z);

-  v_offset = a_position;

+  v_offset = a_offset;

   v_rcorner = a_rcorner;

 }

diff --git a/src/cobalt/renderer/rasterizer/lib/external_rasterizer.cc b/src/cobalt/renderer/rasterizer/lib/external_rasterizer.cc
index e1cb238..00dc1ee 100644
--- a/src/cobalt/renderer/rasterizer/lib/external_rasterizer.cc
+++ b/src/cobalt/renderer/rasterizer/lib/external_rasterizer.cc
@@ -27,6 +27,7 @@
 #include "cobalt/renderer/backend/egl/graphics_context.h"
 #include "cobalt/renderer/backend/egl/render_target.h"
 #include "cobalt/renderer/rasterizer/common/find_node.h"
+#include "cobalt/renderer/rasterizer/egl/hardware_rasterizer.h"
 #include "cobalt/renderer/rasterizer/lib/exported/graphics.h"
 #include "cobalt/renderer/rasterizer/lib/exported/video.h"
 #include "cobalt/renderer/rasterizer/skia/hardware_image.h"
@@ -154,8 +155,12 @@
 
   backend::GraphicsContextEGL* graphics_context_;
 
+#if defined(COBALT_FORCE_DIRECT_GLES_RASTERIZER)
+  egl::HardwareRasterizer hardware_rasterizer_;
+#else
   skia::HardwareRasterizer hardware_rasterizer_;
-  skia::HardwareRasterizer::Options options_;
+#endif
+  Rasterizer::Options options_;
 
   // The main offscreen render target to use when rendering UI or rectangular
   // video.
@@ -178,7 +183,7 @@
                                int skia_atlas_width, int skia_atlas_height,
                                int skia_cache_size_in_bytes,
                                int scratch_surface_cache_size_in_bytes,
-                               int surface_cache_size_in_bytes,
+                               int rasterizer_gpu_cache_size_in_bytes,
                                bool purge_skia_font_caches_on_destruction)
     : graphics_context_(
           base::polymorphic_downcast<backend::GraphicsContextEGL*>(
@@ -186,11 +191,12 @@
       hardware_rasterizer_(
           graphics_context, skia_atlas_width, skia_atlas_height,
           skia_cache_size_in_bytes, scratch_surface_cache_size_in_bytes,
-          surface_cache_size_in_bytes, purge_skia_font_caches_on_destruction),
+          rasterizer_gpu_cache_size_in_bytes,
+          purge_skia_font_caches_on_destruction),
       video_projection_type_(kCbLibVideoProjectionTypeNone),
       video_stereo_mode_(render_tree::StereoMode::kMono),
       video_texture_rgb_(0) {
-  options_.flags = skia::HardwareRasterizer::kSubmitFlags_Clear;
+  options_.flags = Rasterizer::kSubmitFlags_Clear;
   graphics_context_->MakeCurrent();
 
   // TODO: Import the correct size for this and any other textures from the lib
@@ -397,12 +403,14 @@
 ExternalRasterizer::ExternalRasterizer(
     backend::GraphicsContext* graphics_context, int skia_atlas_width,
     int skia_atlas_height, int skia_cache_size_in_bytes,
-    int scratch_surface_cache_size_in_bytes, int surface_cache_size_in_bytes,
+    int scratch_surface_cache_size_in_bytes,
+    int rasterizer_gpu_cache_size_in_bytes,
     bool purge_skia_font_caches_on_destruction)
     : impl_(new Impl(
           graphics_context, skia_atlas_width, skia_atlas_height,
           skia_cache_size_in_bytes, scratch_surface_cache_size_in_bytes,
-          surface_cache_size_in_bytes, purge_skia_font_caches_on_destruction)) {
+          rasterizer_gpu_cache_size_in_bytes,
+          purge_skia_font_caches_on_destruction)) {
 }
 
 ExternalRasterizer::~ExternalRasterizer() {}
diff --git a/src/cobalt/renderer/rasterizer/lib/external_rasterizer.h b/src/cobalt/renderer/rasterizer/lib/external_rasterizer.h
index 9ad1fd0..d2f32e6 100644
--- a/src/cobalt/renderer/rasterizer/lib/external_rasterizer.h
+++ b/src/cobalt/renderer/rasterizer/lib/external_rasterizer.h
@@ -37,7 +37,7 @@
                      int skia_atlas_width, int skia_atlas_height,
                      int skia_cache_size_in_bytes,
                      int scratch_surface_cache_size_in_bytes,
-                     int surface_cache_size_in_bytes,
+                     int rasterizer_gpu_cache_size_in_bytes,
                      bool purge_skia_font_caches_on_destruction);
   virtual ~ExternalRasterizer();
 
diff --git a/src/cobalt/renderer/rasterizer/lib/lib.gyp b/src/cobalt/renderer/rasterizer/lib/lib.gyp
index 0b5a2c5..53a5e51 100644
--- a/src/cobalt/renderer/rasterizer/lib/lib.gyp
+++ b/src/cobalt/renderer/rasterizer/lib/lib.gyp
@@ -27,6 +27,7 @@
        'dependencies': [
          '<(DEPTH)/base/base.gyp:base',
          '<(DEPTH)/cobalt/render_tree/render_tree.gyp:render_tree',
+         '<(DEPTH)/cobalt/renderer/rasterizer/egl/rasterizer.gyp:hardware_rasterizer',
          '<(DEPTH)/cobalt/renderer/rasterizer/skia/common.gyp:common',
          '<(DEPTH)/cobalt/renderer/rasterizer/skia/rasterizer.gyp:hardware_rasterizer',
          '<(DEPTH)/cobalt/renderer/rasterizer/skia/skia/skia.gyp:skia',
diff --git a/src/cobalt/renderer/rasterizer/lib/renderer_module_default_options_lib.cc b/src/cobalt/renderer/rasterizer/lib/renderer_module_default_options_lib.cc
index f98baf7..2b2ea4b 100644
--- a/src/cobalt/renderer/rasterizer/lib/renderer_module_default_options_lib.cc
+++ b/src/cobalt/renderer/rasterizer/lib/renderer_module_default_options_lib.cc
@@ -28,7 +28,11 @@
           options.skia_glyph_texture_atlas_dimensions.height(),
           options.skia_cache_size_in_bytes,
           options.scratch_surface_cache_size_in_bytes,
+#if defined(COBALT_FORCE_DIRECT_GLES_RASTERIZER)
+          options.offscreen_target_cache_size_in_bytes,
+#else
           options.surface_cache_size_in_bytes,
+#endif
           options.purge_skia_font_caches_on_destruction));
 }
 }  // namespace
diff --git a/src/cobalt/renderer/submission_queue.cc b/src/cobalt/renderer/submission_queue.cc
index 20a1e39..b7b861e 100644
--- a/src/cobalt/renderer/submission_queue.cc
+++ b/src/cobalt/renderer/submission_queue.cc
@@ -89,8 +89,11 @@
         base::polymorphic_downcast<render_tree::animations::AnimateNode*>(
             submission_queue_.front().render_tree.get());
 
-    if (animate_node->expiry() <= submission_time(now) &&
-        animate_node->expiry() <=
+    // Check the expiration of only animations that depend on the time
+    // parameter, since they are the only ones that will be affected by snapping
+    // time.
+    if (animate_node->depends_on_time_expiry() <= submission_time(now) &&
+        animate_node->depends_on_time_expiry() <=
             latest_to_submission_time + render_time(now)) {
       to_submission_time_in_ms_.SnapToTarget();
     }
diff --git a/src/cobalt/system_window/system_window.cc b/src/cobalt/system_window/system_window.cc
index 3a21efb..1e2c38f 100644
--- a/src/cobalt/system_window/system_window.cc
+++ b/src/cobalt/system_window/system_window.cc
@@ -211,51 +211,6 @@
   }
 }
 
-void OnDialogClose(SbSystemPlatformErrorResponse response, void* user_data) {
-  DCHECK(user_data);
-  SystemWindow* system_window = static_cast<SystemWindow*>(user_data);
-  system_window->HandleDialogClose(response);
-}
-
-void SystemWindow::ShowDialog(const SystemWindow::DialogOptions& options) {
-  SbSystemPlatformErrorType error_type =
-      kSbSystemPlatformErrorTypeConnectionError;
-  switch (options.message_code) {
-    case kDialogConnectionError:
-      error_type = kSbSystemPlatformErrorTypeConnectionError;
-      break;
-    default:
-      NOTREACHED();
-      break;
-  }
-
-  SbSystemPlatformError handle =
-      SbSystemRaisePlatformError(error_type, OnDialogClose, this);
-  if (SbSystemPlatformErrorIsValid(handle)) {
-    current_dialog_callback_ = options.callback;
-  } else {
-    DLOG(WARNING) << "Failed to notify user of error: " << options.message_code;
-  }
-}
-
-void SystemWindow::HandleDialogClose(SbSystemPlatformErrorResponse response) {
-  DCHECK(!current_dialog_callback_.is_null());
-  switch (response) {
-    case kSbSystemPlatformErrorResponsePositive:
-      current_dialog_callback_.Run(kDialogPositiveResponse);
-      break;
-    case kSbSystemPlatformErrorResponseNegative:
-      current_dialog_callback_.Run(kDialogNegativeResponse);
-      break;
-    case kSbSystemPlatformErrorResponseCancel:
-      current_dialog_callback_.Run(kDialogCancelResponse);
-      break;
-    default:
-      DLOG(WARNING) << "Unrecognized dialog response: " << response;
-      break;
-  }
-}
-
 void HandleInputEvent(const SbEvent* event) {
   if (event->type != kSbEventTypeInput) {
     return;
diff --git a/src/cobalt/system_window/system_window.h b/src/cobalt/system_window/system_window.h
index b9f6017..121313c 100644
--- a/src/cobalt/system_window/system_window.h
+++ b/src/cobalt/system_window/system_window.h
@@ -15,17 +15,12 @@
 #ifndef COBALT_SYSTEM_WINDOW_SYSTEM_WINDOW_H_
 #define COBALT_SYSTEM_WINDOW_SYSTEM_WINDOW_H_
 
-#include <string>
-
-#include "base/callback.h"
-#include "base/memory/scoped_ptr.h"
 #include "base/optional.h"
 #include "cobalt/base/event_dispatcher.h"
 #include "cobalt/math/size.h"
 #include "cobalt/system_window/input_event.h"
 #include "starboard/input.h"
 #include "starboard/key.h"
-#include "starboard/system.h"
 
 namespace cobalt {
 namespace system_window {
@@ -36,38 +31,10 @@
 // create a display render target for a graphics system.
 class SystemWindow {
  public:
-  // Enumeration of possible responses for the dialog callback.
-  enum DialogResponse {
-    kDialogPositiveResponse,
-    kDialogNegativeResponse,
-    kDialogCancelResponse
-  };
-
-  // Type of callback to run when user closes a dialog.
-  typedef base::Callback<void(DialogResponse response)> DialogCallback;
-
-  // Enumeration of possible message codes for a dialog.
-  enum DialogMessageCode {
-    kDialogConnectionError
-  };
-
-  // Options structure for dialog creation. It is expected that each platform
-  // will implement a modal dialog with possible support for:
-  // A message code specifying the text to be displayed, which should be
-  // localized according to the platform.
-  // A callback indicating the user's response: positive, negative or cancel.
-  struct DialogOptions {
-    DialogMessageCode message_code;
-    DialogCallback callback;
-  };
-
   SystemWindow(base::EventDispatcher* event_dispatcher,
                const base::optional<math::Size>& window_size);
   ~SystemWindow();
 
-  // Launches a system dialog.
-  void ShowDialog(const DialogOptions& options);
-
   // Returns the dimensions of the window.
   math::Size GetWindowSize() const;
 
@@ -92,9 +59,6 @@
   // Handles a single Starboard input event, dispatching any appropriate events.
   void HandleInputEvent(const SbInputData& data);
 
-  // Called when the user closes the dialog.
-  void HandleDialogClose(SbSystemPlatformErrorResponse response);
-
  private:
   void UpdateModifiers(SbKey key, bool pressed);
   InputEvent::Modifiers GetModifiers();
@@ -107,9 +71,6 @@
   SbWindow window_;
 
   bool key_down_;
-
-  // The current dialog callback. Only one dialog may be open at a time.
-  DialogCallback current_dialog_callback_;
 };
 
 }  // namespace system_window
diff --git a/src/cobalt/xhr/xml_http_request.cc b/src/cobalt/xhr/xml_http_request.cc
index c1774a4..4c2e893 100644
--- a/src/cobalt/xhr/xml_http_request.cc
+++ b/src/cobalt/xhr/xml_http_request.cc
@@ -163,7 +163,8 @@
       error_(false),
       sent_(false),
       stop_timeout_(false),
-      upload_complete_(false) {
+      upload_complete_(false),
+      active_requests_count_(0) {
   DCHECK(settings_);
   dom::GlobalStats::GetInstance()->Add(this);
   xhr_id_ = ++s_xhr_sequence_num_;
@@ -365,8 +366,9 @@
   // Step 9
   sent_ = true;
   // Now that a send is happening, prevent this object
-  // from being collected until it's complete or aborted.
-  PreventGarbageCollection();
+  // from being collected until it's complete or aborted
+  // if no currently active request has called it before.
+  IncrementActiveRequests();
   FireProgressEvent(this, base::Tokens::loadstart());
   if (!upload_complete_) {
     FireProgressEvent(upload_, base::Tokens::loadstart());
@@ -677,7 +679,7 @@
     ChangeState(kDone);
     UpdateProgress();
     // Undo the ref we added in Send()
-    AllowGarbageCollection();
+    DecrementActiveRequests();
   } else {
     HandleRequestError(kNetworkError);
   }
@@ -775,7 +777,7 @@
   FireProgressEvent(this, base::Tokens::loadend());
 
   fetch_callback_.reset();
-  AllowGarbageCollection();
+  DecrementActiveRequests();
 }
 
 void XMLHttpRequest::OnTimeout() {
@@ -856,6 +858,21 @@
   }
 }
 
+void XMLHttpRequest::IncrementActiveRequests() {
+  if (active_requests_count_ == 0) {
+    PreventGarbageCollection();
+  }
+  active_requests_count_++;
+}
+
+void XMLHttpRequest::DecrementActiveRequests() {
+  DCHECK_GT(active_requests_count_, 0);
+  active_requests_count_--;
+  if (active_requests_count_ == 0) {
+    AllowGarbageCollection();
+  }
+}
+
 void XMLHttpRequest::PreventGarbageCollection() {
   settings_->global_environment()->PreventGarbageCollection(
       make_scoped_refptr(this));
diff --git a/src/cobalt/xhr/xml_http_request.h b/src/cobalt/xhr/xml_http_request.h
index ab9a5e4..209292a 100644
--- a/src/cobalt/xhr/xml_http_request.h
+++ b/src/cobalt/xhr/xml_http_request.h
@@ -218,6 +218,13 @@
   void AllowGarbageCollection();
   void StartRequest(const std::string& request_body);
 
+  // The following two methods are used to determine if garbage collection is
+  // needed. It is legal to reuse XHR and send a new request in last request's
+  // onload event listener. We should not allow garbage collection until
+  // the last request is fetched.
+  void IncrementActiveRequests();
+  void DecrementActiveRequests();
+
   // Accessors / mutators for testing.
   const GURL& request_url() const { return request_url_; }
   bool error() const { return error_; }
@@ -271,6 +278,7 @@
   bool sent_;
   bool stop_timeout_;
   bool upload_complete_;
+  int active_requests_count_;
 
   static bool verbose_;
   // Unique ID for debugging.
diff --git a/src/nb/nb.gyp b/src/nb/nb.gyp
index 8c50a66..b33d946 100644
--- a/src/nb/nb.gyp
+++ b/src/nb/nb.gyp
@@ -60,6 +60,8 @@
             'scoped_ptr.h',
             'simple_thread.cc',
             'simple_thread.h',
+            'simple_profiler.cc',
+            'simple_profiler.h',
             'std_allocator.h',
             'string_interner.cc',
             'string_interner.h',
@@ -104,6 +106,7 @@
             'multipart_allocator_test.cc',
             'rewindable_vector_test.cc',
             'run_all_unittests.cc',
+            'simple_profiler_test.cc',
             'std_allocator_test.cc',
             'string_interner_test.cc',
             'test_thread.h',
diff --git a/src/nb/simple_profiler.cc b/src/nb/simple_profiler.cc
new file mode 100644
index 0000000..6b313aa
--- /dev/null
+++ b/src/nb/simple_profiler.cc
@@ -0,0 +1,187 @@
+/*

+ * Copyright 2017 Google Inc. All Rights Reserved.

+ *

+ * Licensed under the Apache License, Version 2.0 (the "License");

+ * you may not use this file except in compliance with the License.

+ * You may obtain a copy of the License at

+ *

+ *     http://www.apache.org/licenses/LICENSE-2.0

+ *

+ * Unless required by applicable law or agreed to in writing, software

+ * distributed under the License is distributed on an "AS IS" BASIS,

+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.

+ * See the License for the specific language governing permissions and

+ * limitations under the License.

+ */

+

+#include <algorithm>

+#include <string>

+#include <vector>

+

+#include "nb/rewindable_vector.h"

+#include "nb/simple_profiler.h"

+#include "nb/thread_local_object.h"

+#include "starboard/atomic.h"

+#include "starboard/once.h"

+#include "starboard/string.h"

+#include "starboard/time.h"

+

+namespace nb {

+namespace {

+

+class SimpleProfilerManager {

+ public:

+  struct Entry {

+    Entry() : name(nullptr), time_delta(0), indent_value(0) {}

+    Entry(const char* n, SbTimeMonotonic dt, int ind)

+        : name(n), time_delta(dt), indent_value(ind) {}

+    Entry(const Entry& e) = default;

+    const char* name;

+    SbTimeMonotonic time_delta;

+    int indent_value;

+  };

+

+  using MessageHandlerFunction = SimpleProfiler::MessageHandlerFunction;

+  using ClockFunction = SimpleProfiler::ClockFunction;

+  SimpleProfilerManager()

+      : default_enabled_(true), message_handler_(nullptr),

+        clock_function_(nullptr) {}

+

+  int BeginInstance(const char* name) {

+    Data& d = ThreadLocal();

+    if (!d.enable_flag) {

+      return -1;

+    }

+    SbTimeMonotonic now = NowTime();

+    // SbAtomicMemoryBarrier() to Keep order of operations so clock doesn't

+    // get sampled out of order.

+    SbAtomicMemoryBarrier();

+    d.profiles.emplace_back(name, now, d.instance_count);

+    ++d.instance_count;

+    return d.profiles.size() - 1;

+  }

+

+  void Output(const Entry& entry, std::stringstream* sstream) {

+    for (auto i = 0; i < entry.indent_value; ++i) {

+      (*sstream) << ' ';

+    }

+    (*sstream) << entry.name << ": " << entry.time_delta << "us\n";

+  }

+

+  void FinishInstance(int index) {

+    Data& d = ThreadLocal();

+    if (!d.enable_flag || index < 0) {

+      return;

+    }

+

+    --d.instance_count;

+    Entry& entry = d.profiles[static_cast<size_t>(index)];

+    entry.time_delta = NowTime() - entry.time_delta;

+    // SbAtomicMemoryBarrier() to Keep order of operations so clock doesn't

+    // get sampled out of order.

+    SbAtomicMemoryBarrier();

+

+    if (d.instance_count == 0) {

+      std::stringstream ss;

+      for (auto it = d.profiles.begin(); it != d.profiles.end(); ++it) {

+        Output(*it, &ss);

+      }

+      d.profiles.clear();

+      HandleMessage(ss.str());

+    }

+  }

+  bool ThreadLocalEnabled() { return ThreadLocal().enable_flag; }

+  void SetThreadLocalEnabled(bool value) {

+    Data& d = ThreadLocal();

+    d.enable_flag = value;

+    d.enable_flag_set = true;

+  }

+

+  bool IsEnabled() const {

+    const Data& d = ThreadLocal();

+    if (d.enable_flag_set) {

+      return d.enable_flag;

+    }

+    return default_enabled_;

+  }

+  void SetDefaultEnabled(bool value) { default_enabled_ = value; }

+

+  void SetMessageHandler(MessageHandlerFunction handler) {

+    message_handler_ = handler;

+  }

+

+  void SetClockFunction(ClockFunction fcn) {

+    clock_function_ = fcn;

+  }

+

+  SbTimeMonotonic NowTime() {

+    if (!clock_function_) {

+      return SbTimeGetMonotonicNow();

+    } else {

+      return clock_function_();

+    }

+  }

+

+ private:

+  struct Data {

+    bool enable_flag = true;

+    bool enable_flag_set = false;

+    int instance_count = 0;

+    std::vector<Entry> profiles;

+  };

+

+  void HandleMessage(const std::string& str) {

+    if (message_handler_) {

+      message_handler_(str.c_str());

+    } else {

+      SbLogRaw(str.c_str());

+    }

+  }

+

+  Data& ThreadLocal() { return *thread_local_.GetOrCreate(); }

+  const Data& ThreadLocal() const { return *thread_local_.GetOrCreate(); }

+  mutable nb::ThreadLocalObject<Data> thread_local_;

+  bool default_enabled_;

+  MessageHandlerFunction message_handler_;

+  ClockFunction clock_function_;

+};

+

+SB_ONCE_INITIALIZE_FUNCTION(SimpleProfilerManager, GetSimpleProfilerManager);

+

+}  // namespace

+

+SimpleProfiler::EnableScope::EnableScope(bool enabled) {

+  SimpleProfilerManager* mgr = GetSimpleProfilerManager();

+  prev_enabled_ = mgr->ThreadLocalEnabled();

+  mgr->SetThreadLocalEnabled(enabled);

+}

+

+SimpleProfiler::EnableScope::~EnableScope() {

+  GetSimpleProfilerManager()->SetThreadLocalEnabled(prev_enabled_);

+}

+

+SimpleProfiler::SimpleProfiler(const char* name) {

+  momento_index_ = GetSimpleProfilerManager()->BeginInstance(name);

+}

+

+SimpleProfiler::~SimpleProfiler() {

+  GetSimpleProfilerManager()->FinishInstance(momento_index_);

+}

+

+void SimpleProfiler::SetEnabledByDefault(bool value) {

+  GetSimpleProfilerManager()->SetDefaultEnabled(value);

+}

+

+bool SimpleProfiler::IsEnabled() {

+  return GetSimpleProfilerManager()->IsEnabled();

+}

+

+void SimpleProfiler::SetLoggingFunction(MessageHandlerFunction fcn) {

+  GetSimpleProfilerManager()->SetMessageHandler(fcn);

+}

+

+void SimpleProfiler::SetClockFunction(ClockFunction fcn) {

+  GetSimpleProfilerManager()->SetClockFunction(fcn);

+}

+

+}  // namespace nb

diff --git a/src/nb/simple_profiler.h b/src/nb/simple_profiler.h
new file mode 100644
index 0000000..2c9e4f4
--- /dev/null
+++ b/src/nb/simple_profiler.h
@@ -0,0 +1,105 @@
+/*

+ * Copyright 2017 Google Inc. All Rights Reserved.

+ *

+ * Licensed under the Apache License, Version 2.0 (the "License");

+ * you may not use this file except in compliance with the License.

+ * You may obtain a copy of the License at

+ *

+ *     http://www.apache.org/licenses/LICENSE-2.0

+ *

+ * Unless required by applicable law or agreed to in writing, software

+ * distributed under the License is distributed on an "AS IS" BASIS,

+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.

+ * See the License for the specific language governing permissions and

+ * limitations under the License.

+ */

+

+#ifndef NB_SIMPLE_PROFILER_H_

+#define NB_SIMPLE_PROFILER_H_

+

+#include <string>

+

+#include "starboard/time.h"

+#include "starboard/types.h"

+

+namespace nb {

+

+// SimpleProfiler is useful for development. It will allow the developer

+// to track where the CPU spends most of it's time in a call chain.

+// CPU time is tracked via thread-local-storage. When the last SimpleProfiler

+// in the thread is destroyed then a printout of the timing will be printed

+// to the output log.

+//

+// By default, SimpleProfiler will generate output whenever it's used. To

+// selectively profile a thread see class SimpleProfiler::EnableScope.

+//

+// Example:

+//  void Foo() {

+//    SimpleProfiler profile(__FUNCTION__);

+//    Bar();

+//    Baz();

+//  }

+//  void Bar() {

+//    SimpleProfiler profile(__FUNCTION__);

+//    // ... do something expensive ...

+//    Qux();

+//  }

+//  void Baz() {

+//    SimpleProfiler profile(__FUNCTION__);

+//    // ... do something cheap ...

+//  }

+//  void Qux() {

+//    SimpleProfiler profile(__FUNCTION__);

+//    // ... do something nearly free ...

+//  }

+//

+//  Outputs: "Foo: 25us\n"

+//           " Bar: 20us\n"

+//           "  Qux: 1us\n"

+//           " Baz: 4us\n"

+class SimpleProfiler {

+ public:

+  explicit SimpleProfiler(const char* name);

+  ~SimpleProfiler();

+  // EnableScope can be used to enable and disable SimpleProfiler in the

+  // thread. A scoped object is used so that SimpleProfiler

+  // constructor / destructor order is maintained in relation to

+  // enabling / disabling.

+  // Example:

+  //  // Assume SimpleProfiler was globally disabled by default.

+  //  void Foo() {

+  //     SimpleProfiler::ThreadScope enable_scope(true);  // enabled in scope.

+  //     SimpleProfiler profile(__FUNCTION__);

+  //     DoWork();

+  //  }

+  class EnableScope {

+   public:

+    explicit EnableScope(bool enabled);

+    ~EnableScope();

+   private:

+    bool prev_enabled_;

+  };

+  // If SetThreadLocalEnableFlag() isn't explicitly set by the thread

+  // then this |input| value will control whether the SimpleProfiler

+  // is active or not. For best results, set this value as early as

+  // possible during program execution.

+  static void SetEnabledByDefault(bool value);

+  // Is SimpleProfiler enabled? If Get/SetThreadLocalEnableFlag() isn't

+  // set then this will return an enabled by default flag, which defaults

+  // to true.

+  static bool IsEnabled();

+  typedef void (*MessageHandlerFunction)(const char* msg);

+  // Useful for tests. Setting to nullptr will reset the behavior to

+  // default functionality.

+  static void SetLoggingFunction(MessageHandlerFunction fcn);

+  // Useful for tests. Setting to nullptr will reset the behavior to

+  // default functionality.

+  typedef SbTimeMonotonic (*ClockFunction)();

+  static void SetClockFunction(ClockFunction fcn);

+ private:

+  int momento_index_;

+};

+

+}  // namespace nb

+

+#endif  // NB_SIMPLE_PROFILER_H_

diff --git a/src/nb/simple_profiler_test.cc b/src/nb/simple_profiler_test.cc
new file mode 100644
index 0000000..41b5cee
--- /dev/null
+++ b/src/nb/simple_profiler_test.cc
@@ -0,0 +1,95 @@
+/*

+ * Copyright 2017 Google Inc. All Rights Reserved.

+ *

+ * Licensed under the Apache License, Version 2.0 (the "License");

+ * you may not use this file except in compliance with the License.

+ * You may obtain a copy of the License at

+ *

+ *     http://www.apache.org/licenses/LICENSE-2.0

+ *

+ * Unless required by applicable law or agreed to in writing, software

+ * distributed under the License is distributed on an "AS IS" BASIS,

+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.

+ * See the License for the specific language governing permissions and

+ * limitations under the License.

+ */

+

+#include "nb/simple_profiler.h"

+

+#include <map>

+#include <string>

+#include <vector>

+

+#include "starboard/configuration.h"

+#include "testing/gtest/include/gtest/gtest.h"

+

+namespace nb {

+class SimpleProfilerTest : public ::testing::Test {

+ public:

+  SimpleProfilerTest() {}

+  virtual void SetUp() SB_OVERRIDE {

+    s_log_buffer_.clear();

+    SimpleProfiler::SetLoggingFunction(TestLogFunction);

+  }

+  virtual void TearDown() SB_OVERRIDE {

+    SimpleProfiler::SetLoggingFunction(nullptr);

+    SimpleProfiler::SetClockFunction(nullptr);

+    s_log_buffer_.clear();

+  }

+

+  std::string GetProfilerOutput() { return s_log_buffer_; }

+

+ private:

+  static std::string s_log_buffer_;

+  static void TestLogFunction(const char* value) { s_log_buffer_ = value; }

+};

+std::string SimpleProfilerTest::s_log_buffer_;

+

+struct CallChain {

+  static void Foo() {

+    SimpleProfiler profile("Foo");

+    Bar();

+    Qux();

+  }

+

+  static void Bar() {

+    SimpleProfiler profile("Bar");

+    Baz();

+  }

+

+  static void Baz() { SimpleProfiler profile("Baz"); }

+  static void Qux() { SimpleProfiler profile("Qux"); }

+};

+

+TEST_F(SimpleProfilerTest, IsEnabledByDefault) {

+  EXPECT_TRUE(SimpleProfiler::IsEnabled());

+}

+

+SbTimeMonotonic NullTime() {

+  return SbTimeMonotonic(0);

+}

+// Tests the expectation that SimpleProfiler can be used in a call

+// chain and will generate the expected string.

+TEST_F(SimpleProfilerTest, CallChain) {

+  SimpleProfiler::SetClockFunction(NullTime);

+  CallChain::Foo();

+  std::string profiler_log = GetProfilerOutput();

+

+  std::string expected_output =

+      "Foo: 0us\n"

+      " Bar: 0us\n"

+      "  Baz: 0us\n"

+      " Qux: 0us\n";

+

+  EXPECT_EQ(expected_output, profiler_log) << " actual output:\n"

+                                           << profiler_log;

+}

+

+TEST_F(SimpleProfilerTest, EnableScopeDisabled) {

+  SimpleProfiler::EnableScope enable(false);

+  CallChain::Foo();

+  std::string profiler_log = GetProfilerOutput();

+  EXPECT_TRUE(profiler_log.empty());

+}

+

+}  // namespace nb

diff --git a/src/net/dial/dial_http_server.cc b/src/net/dial/dial_http_server.cc
index a1a44c4..482d27d 100644
--- a/src/net/dial/dial_http_server.cc
+++ b/src/net/dial/dial_http_server.cc
@@ -76,9 +76,6 @@
   // get the port information
   int ret = http_server_->GetLocalAddress(addr);
 
-#if defined(OS_STARBOARD)
-
-#if SB_API_VERSION >= 4
   if (ret != 0) {
     return ERR_FAILED;
   }
@@ -101,24 +98,6 @@
   }
 
   return ERR_FAILED;
-#else
-  SbSocketAddress address;
-  ret |= SbSocketGetLocalInterfaceAddress(&address) ? 0 : -1;
-  address.port = addr->port();
-  return (ret == 0 && addr->FromSbSocketAddress(&address)) ? OK : ERR_FAILED;
-#endif  // SB_API_VERSION >= 4
-
-#else
-  // Now get the IPAddress of the network card.
-  SockaddrStorage sock_addr;
-  struct sockaddr_in *in = (struct sockaddr_in *)sock_addr.addr;
-  ret |= lb_get_local_ip_address(&in->sin_addr);
-  in->sin_family = AF_INET;
-  in->sin_port = htons(addr->port());
-
-  return (ret == 0 && addr->FromSockAddr(sock_addr.addr, sock_addr.addr_len))
-      ? OK : ERR_FAILED;
-#endif
 }
 
 void DialHttpServer::OnHttpRequest(int conn_id,
diff --git a/src/starboard/linux/x64x11/skia/atomic_public.h b/src/starboard/linux/x64x11/skia/atomic_public.h
new file mode 100644
index 0000000..048b3c9
--- /dev/null
+++ b/src/starboard/linux/x64x11/skia/atomic_public.h
@@ -0,0 +1,20 @@
+// Copyright 2017 Google Inc. All Rights Reserved.
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+//     http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef STARBOARD_LINUX_X64X11_SKIA_ATOMIC_PUBLIC_H_
+#define STARBOARD_LINUX_X64X11_SKIA_ATOMIC_PUBLIC_H_
+
+#include "starboard/linux/shared/atomic_public.h"
+
+#endif  // STARBOARD_LINUX_X64X11_SKIA_ATOMIC_PUBLIC_H_
diff --git a/src/starboard/linux/x64x11/skia/configuration_public.h b/src/starboard/linux/x64x11/skia/configuration_public.h
new file mode 100644
index 0000000..ad03114
--- /dev/null
+++ b/src/starboard/linux/x64x11/skia/configuration_public.h
@@ -0,0 +1,28 @@
+// Copyright 2017 Google Inc. All Rights Reserved.
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+//     http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+// The Starboard configuration for Desktop X86 Linux configured for the future
+// starboard API.
+
+#ifndef STARBOARD_LINUX_X64X11_SKIA_CONFIGURATION_PUBLIC_H_
+#define STARBOARD_LINUX_X64X11_SKIA_CONFIGURATION_PUBLIC_H_
+
+// This is not a released configuration, so it should implement the
+// experimental API version to validate trunk's viability.
+#define SB_API_VERSION SB_EXPERIMENTAL_API_VERSION
+
+// Include the X64X11 Linux configuration.
+#include "starboard/linux/x64x11/configuration_public.h"
+
+#endif  // STARBOARD_LINUX_X64X11_SKIA_CONFIGURATION_PUBLIC_H_
diff --git a/src/starboard/linux/x64x11/skia/gyp_configuration.gypi b/src/starboard/linux/x64x11/skia/gyp_configuration.gypi
new file mode 100644
index 0000000..06b67cc
--- /dev/null
+++ b/src/starboard/linux/x64x11/skia/gyp_configuration.gypi
@@ -0,0 +1,41 @@
+# Copyright 2016 Google Inc. All Rights Reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+#     http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+
+{
+  'variables': {
+    # Use the skia hardware rasterizer.
+    'rasterizer_type': 'hardware',
+  },
+  'target_defaults': {
+    'default_configuration': 'linux-x64x11-skia_debug',
+    'configurations': {
+      'linux-x64x11-skia_debug': {
+        'inherit_from': ['debug_base'],
+      },
+      'linux-x64x11-skia_devel': {
+        'inherit_from': ['devel_base'],
+      },
+      'linux-x64x11-skia_qa': {
+        'inherit_from': ['qa_base'],
+      },
+      'linux-x64x11-skia_gold': {
+        'inherit_from': ['gold_base'],
+      },
+    }, # end of configurations
+  },
+
+  'includes': [
+    '../gyp_configuration.gypi',
+  ],
+}
diff --git a/src/starboard/linux/x64x11/skia/gyp_configuration.py b/src/starboard/linux/x64x11/skia/gyp_configuration.py
new file mode 100644
index 0000000..45b4461
--- /dev/null
+++ b/src/starboard/linux/x64x11/skia/gyp_configuration.py
@@ -0,0 +1,27 @@
+# Copyright 2016 Google Inc. All Rights Reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+#     http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+"""Starboard Linux X64 X11 future platform configuration for gyp_cobalt."""
+
+import logging
+
+# Import the shared Linux platform configuration.
+from starboard.linux.shared import gyp_configuration
+
+
+def CreatePlatformConfig():
+  try:
+    return gyp_configuration.PlatformConfig('linux-x64x11-skia')
+  except RuntimeError as e:
+    logging.critical(e)
+    return None
diff --git a/src/starboard/linux/x64x11/skia/starboard_platform.gyp b/src/starboard/linux/x64x11/skia/starboard_platform.gyp
new file mode 100644
index 0000000..528d1cd
--- /dev/null
+++ b/src/starboard/linux/x64x11/skia/starboard_platform.gyp
@@ -0,0 +1,33 @@
+# Copyright 2016 Google Inc. All Rights Reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+#     http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+{
+  'includes': [
+    '../starboard_platform.gypi'
+  ],
+  'targets': [
+    {
+      'target_name': 'starboard_platform',
+      'type': 'static_library',
+      'sources': ['<@(starboard_platform_sources)'],
+      'defines': [
+        # This must be defined when building Starboard, and must not when
+        # building Starboard client code.
+        'STARBOARD_IMPLEMENTATION',
+      ],
+      'dependencies': [
+        '<@(starboard_platform_dependencies)',
+      ],
+    },
+  ],
+}
diff --git a/src/starboard/linux/x64x11/skia/thread_types_public.h b/src/starboard/linux/x64x11/skia/thread_types_public.h
new file mode 100644
index 0000000..5e74da8
--- /dev/null
+++ b/src/starboard/linux/x64x11/skia/thread_types_public.h
@@ -0,0 +1,20 @@
+// Copyright 2017 Google Inc. All Rights Reserved.
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+//     http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef STARBOARD_LINUX_X64X11_SKIA_THREAD_TYPES_PUBLIC_H_
+#define STARBOARD_LINUX_X64X11_SKIA_THREAD_TYPES_PUBLIC_H_
+
+#include "starboard/linux/shared/thread_types_public.h"
+
+#endif  // STARBOARD_LINUX_X64X11_SKIA_THREAD_TYPES_PUBLIC_H_
diff --git a/src/starboard/nplb/audio_sink_helpers.cc b/src/starboard/nplb/audio_sink_helpers.cc
index 981c0e1..51aa218 100644
--- a/src/starboard/nplb/audio_sink_helpers.cc
+++ b/src/starboard/nplb/audio_sink_helpers.cc
@@ -151,7 +151,7 @@
 bool AudioSinkTestEnvironment::WaitUntilAllFramesAreConsumed() {
   ScopedLock lock(mutex_);
   SbTimeMonotonic start = SbTimeGetMonotonicNow();
-  while (frames_appended_ == frames_consumed_) {
+  while (frames_appended_ != frames_consumed_) {
     SbTime time_elapsed = SbTimeGetMonotonicNow() - start;
     if (time_elapsed >= kTimeToTry) {
       return false;
diff --git a/src/starboard/nplb/system_get_total_gpu_memory_test.cc b/src/starboard/nplb/system_get_total_gpu_memory_test.cc
index 0cd7d9b..c39814c 100644
--- a/src/starboard/nplb/system_get_total_gpu_memory_test.cc
+++ b/src/starboard/nplb/system_get_total_gpu_memory_test.cc
@@ -23,7 +23,7 @@
   if (SbSystemHasCapability(kSbSystemCapabilityCanQueryGPUMemoryStats)) {
     // If we claim to have GPU memory reporting capabilities, then this value
     // should be larger than 0.
-    EXPECT_LT(0, SbSystemGetTotalCPUMemory());
+    EXPECT_LT(0, SbSystemGetTotalGPUMemory());
   }
 }
 
diff --git a/src/starboard/raspi/2/skia/atomic_public.h b/src/starboard/raspi/2/skia/atomic_public.h
new file mode 100644
index 0000000..dca0903
--- /dev/null
+++ b/src/starboard/raspi/2/skia/atomic_public.h
@@ -0,0 +1,20 @@
+// Copyright 2017 Google Inc. All Rights Reserved.
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+//     http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef STARBOARD_RASPI_2_SKIA_ATOMIC_PUBLIC_H_
+#define STARBOARD_RASPI_2_SKIA_ATOMIC_PUBLIC_H_
+
+#include "starboard/raspi/2/atomic_public.h"
+
+#endif  // STARBOARD_RASPI_2_SKIA_ATOMIC_PUBLIC_H_
diff --git a/src/starboard/raspi/2/skia/configuration_public.h b/src/starboard/raspi/2/skia/configuration_public.h
new file mode 100644
index 0000000..6c65ff5
--- /dev/null
+++ b/src/starboard/raspi/2/skia/configuration_public.h
@@ -0,0 +1,25 @@
+// Copyright 2017 Google Inc. All Rights Reserved.
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+//     http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+// The Starboard configuration for Raspberry PI 2 Raspbian.
+
+// Other source files should never include this header directly, but should
+// include the generic "starboard/configuration.h" instead.
+
+#ifndef STARBOARD_RASPI_2_SKIA_CONFIGURATION_PUBLIC_H_
+#define STARBOARD_RASPI_2_SKIA_CONFIGURATION_PUBLIC_H_
+
+#include "starboard/raspi/2/configuration_public.h"
+
+#endif  // STARBOARD_RASPI_2_SKIA_CONFIGURATION_PUBLIC_H_
diff --git a/src/starboard/raspi/2/skia/gyp_configuration.gypi b/src/starboard/raspi/2/skia/gyp_configuration.gypi
new file mode 100644
index 0000000..21c8ea4
--- /dev/null
+++ b/src/starboard/raspi/2/skia/gyp_configuration.gypi
@@ -0,0 +1,43 @@
+# Copyright 2017 Google Inc. All Rights Reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+#     http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+
+{
+  'variables': {
+    # Use the skia hardware rasterizer.
+    'rasterizer_type': 'hardware',
+  },
+
+  'target_defaults': {
+    'default_configuration': 'raspi-2-skia_debug',
+    'configurations': {
+      'raspi-2-skia_debug': {
+        'inherit_from': ['debug_base'],
+      },
+      'raspi-2-skia_devel': {
+        'inherit_from': ['devel_base'],
+      },
+      'raspi-2-skia_qa': {
+        'inherit_from': ['qa_base'],
+      },
+      'raspi-2-skia_gold': {
+        'inherit_from': ['gold_base'],
+      },
+    }, # end of configurations
+  },
+
+  'includes': [
+    '../architecture.gypi',
+    '../../shared/gyp_configuration.gypi',
+  ],
+}
diff --git a/src/starboard/raspi/2/skia/gyp_configuration.py b/src/starboard/raspi/2/skia/gyp_configuration.py
new file mode 100644
index 0000000..ccf358f
--- /dev/null
+++ b/src/starboard/raspi/2/skia/gyp_configuration.py
@@ -0,0 +1,32 @@
+# Copyright 2017 Google Inc. All Rights Reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+#     http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+"""Starboard Raspberry Pi 2 platform configuration for gyp_cobalt."""
+
+import logging
+import os
+import sys
+
+_SCRIPT_DIR = os.path.abspath(os.path.dirname(__file__))
+sys.path.insert(0, os.path.join(_SCRIPT_DIR, '../..'))
+
+# pylint: disable=g-import-not-at-top
+from shared.gyp_configuration import RaspiPlatformConfig
+
+
+def CreatePlatformConfig():
+  try:
+    return RaspiPlatformConfig('raspi-2-skia')
+  except RuntimeError as e:
+    logging.critical(e)
+    return None
diff --git a/src/starboard/raspi/2/skia/starboard_platform.gyp b/src/starboard/raspi/2/skia/starboard_platform.gyp
new file mode 100644
index 0000000..a304ac6
--- /dev/null
+++ b/src/starboard/raspi/2/skia/starboard_platform.gyp
@@ -0,0 +1,18 @@
+# Copyright 2017 Google Inc. All Rights Reserved.
+#
+# Licensed under the Apache License, Version 2.0 (the "License");
+# you may not use this file except in compliance with the License.
+# You may obtain a copy of the License at
+#
+#     http://www.apache.org/licenses/LICENSE-2.0
+#
+# Unless required by applicable law or agreed to in writing, software
+# distributed under the License is distributed on an "AS IS" BASIS,
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+# See the License for the specific language governing permissions and
+# limitations under the License.
+{
+  'includes': [
+    '../../shared/starboard_platform.gypi',
+  ],
+}
diff --git a/src/starboard/raspi/2/skia/thread_types_public.h b/src/starboard/raspi/2/skia/thread_types_public.h
new file mode 100644
index 0000000..98cf1fb
--- /dev/null
+++ b/src/starboard/raspi/2/skia/thread_types_public.h
@@ -0,0 +1,20 @@
+// Copyright 2017 Google Inc. All Rights Reserved.
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+//     http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#ifndef STARBOARD_RASPI_2_SKIA_THREAD_TYPES_PUBLIC_H_
+#define STARBOARD_RASPI_2_SKIA_THREAD_TYPES_PUBLIC_H_
+
+#include "starboard/raspi/2/thread_types_public.h"
+
+#endif  // STARBOARD_RASPI_2_SKIA_THREAD_TYPES_PUBLIC_H_
diff --git a/src/starboard/shared/starboard/player/filter/audio_renderer_impl_internal.cc b/src/starboard/shared/starboard/player/filter/audio_renderer_impl_internal.cc
index 2ad5415..c219c59 100644
--- a/src/starboard/shared/starboard/player/filter/audio_renderer_impl_internal.cc
+++ b/src/starboard/shared/starboard/player/filter/audio_renderer_impl_internal.cc
@@ -86,6 +86,8 @@
       audio_sink_(kSbAudioSinkInvalid),
       can_accept_more_data_(true),
       process_audio_data_scheduled_(false),
+      process_audio_data_closure_(
+          Bind(&AudioRendererImpl::ProcessAudioData, this)),
       decoder_needs_full_reset_(false),
       audio_frame_tracker_(audio_frame_tracker.Pass()) {
   SB_DCHECK(decoder_ != NULL);
@@ -177,6 +179,12 @@
     // TODO: Remove SetPlaybackRate() support from audio sink as it only need to
     // support play/pause.
     audio_sink_->SetPlaybackRate(playback_rate_ > 0.0 ? 1.0 : 0.0);
+    if (playback_rate_ > 0.0) {
+      if (process_audio_data_scheduled_) {
+        Remove(process_audio_data_closure_);
+      }
+      ProcessAudioData();
+    }
   }
 }
 
@@ -353,9 +361,7 @@
   ++pending_decoder_outputs_;
 
   if (process_audio_data_scheduled_) {
-    // A ProcessAudioData() callback has been scheduled and we should let it
-    // process the output.
-    return;
+    Remove(process_audio_data_closure_);
   }
 
   process_audio_data_scheduled_ = true;
@@ -381,7 +387,7 @@
       const SbTimeMonotonic delay = kMaxCachedFrames * kSbTimeSecond /
                                     decoder_->GetSamplesPerSecond() / 4;
       process_audio_data_scheduled_ = true;
-      Schedule(Bind(&AudioRendererImpl::ProcessAudioData, this), delay);
+      Schedule(process_audio_data_closure_, delay);
       return;
     }
 
@@ -419,6 +425,12 @@
     }
   }
 
+  if (seeking_.load() || playback_rate_ == 0.0) {
+    process_audio_data_scheduled_ = true;
+    Schedule(process_audio_data_closure_, 5 * kSbTimeMillisecond);
+    return;
+  }
+
   int64_t frames_in_buffer =
       frames_sent_to_sink_.load() - frames_consumed_by_sink_.load();
   if (kMaxCachedFrames - frames_in_buffer < kFrameAppendUnit &&
@@ -431,13 +443,21 @@
       delay = frames_to_delay * kSbTimeSecond / decoder_->GetSamplesPerSecond();
     }
     process_audio_data_scheduled_ = true;
-    Schedule(Bind(&AudioRendererImpl::ProcessAudioData, this), delay);
+    Schedule(process_audio_data_closure_, delay);
   }
 }
 
 bool AudioRendererImpl::AppendAudioToFrameBuffer() {
   SB_DCHECK(BelongsToCurrentThread());
 
+  if (seeking_.load() && time_stretcher_.IsQueueFull()) {
+    seeking_.store(false);
+  }
+
+  if (seeking_.load() || playback_rate_ == 0.0) {
+    return false;
+  }
+
   int frames_in_buffer = static_cast<int>(frames_sent_to_sink_.load() -
                                           frames_consumed_by_sink_.load());
 
@@ -447,18 +467,13 @@
 
   int offset_to_append = frames_sent_to_sink_.load() % kMaxCachedFrames;
 
-  // When |playback_rate_| is 0, try to fill the buffer with playback rate as 1.
-  // Otherwise the preroll will never finish.
-  float playback_rate_to_fill =
-      playback_rate_ == 0.0 ? 1.f : static_cast<float>(playback_rate_);
   scoped_refptr<DecodedAudio> decoded_audio =
-      time_stretcher_.Read(kFrameAppendUnit, playback_rate_to_fill);
+      time_stretcher_.Read(kFrameAppendUnit, playback_rate_);
   SB_DCHECK(decoded_audio);
   if (decoded_audio->frames() == 0 && eos_state_.load() == kEOSDecoded) {
     eos_state_.store(kEOSSentToSink);
   }
-  audio_frame_tracker_->AddFrames(decoded_audio->frames(),
-                                  playback_rate_to_fill);
+  audio_frame_tracker_->AddFrames(decoded_audio->frames(), playback_rate_);
   // TODO: Support kSbMediaAudioFrameStorageTypePlanar.
   decoded_audio->SwitchFormatTo(sink_sample_type_,
                                 kSbMediaAudioFrameStorageTypeInterleaved);
@@ -482,12 +497,6 @@
 
   frames_sent_to_sink_.fetch_add(frames_appended);
 
-  int64_t preroll_frames =
-      decoder_->GetSamplesPerSecond() * kPrerollTime / kSbTimeSecond;
-  if (seeking_.load() && frames_sent_to_sink_.load() > preroll_frames) {
-    seeking_.store(false);
-  }
-
   return frames_appended > 0;
 }
 
diff --git a/src/starboard/shared/starboard/player/filter/audio_renderer_impl_internal.h b/src/starboard/shared/starboard/player/filter/audio_renderer_impl_internal.h
index f1b7730..cd36913 100644
--- a/src/starboard/shared/starboard/player/filter/audio_renderer_impl_internal.h
+++ b/src/starboard/shared/starboard/player/filter/audio_renderer_impl_internal.h
@@ -44,10 +44,6 @@
 // |AudioDecoder| interface, rather than a platform specific implementation.
 class AudioRendererImpl : public AudioRenderer, private JobQueue::JobOwner {
  public:
-  // Preroll is considered as finished after either the amount of audio caches
-  // exceeds kPrerollTime or if EOS is reached.
-  static const size_t kPrerollTime = kSbTimeSecond / 4;
-
   AudioRendererImpl(
       scoped_ptr<AudioDecoder> decoder,
       const SbMediaAudioHeader& audio_header,
@@ -146,6 +142,7 @@
 
   bool can_accept_more_data_;
   bool process_audio_data_scheduled_;
+  Closure process_audio_data_closure_;
 
   // Our owner will attempt to seek to pts 0 when playback begins.  In
   // general, seeking could require a full reset of the underlying decoder on
diff --git a/src/starboard/shared/starboard/player/filter/audio_renderer_impl_internal_test.cc b/src/starboard/shared/starboard/player/filter/audio_renderer_impl_internal_test.cc
index 517c3ef..4037ee2 100644
--- a/src/starboard/shared/starboard/player/filter/audio_renderer_impl_internal_test.cc
+++ b/src/starboard/shared/starboard/player/filter/audio_renderer_impl_internal_test.cc
@@ -185,19 +185,16 @@
   const int kFramesPerBuffer = 1024;
 
   int frames_written = 0;
-  int preroll_frames = kDefaultSamplesPerSecond *
-                       AudioRendererImpl::kPrerollTime / kSbTimeSecond;
 
-  while (frames_written <= preroll_frames) {
-    SbMediaTime pts = frames_written / kDefaultSamplesPerSecond;
+  while (audio_renderer_->IsSeekingInProgress()) {
+    SbMediaTime pts =
+        frames_written * kSbMediaTimeSecond / kDefaultSamplesPerSecond;
     WriteSample(CreateInputBuffer(pts));
     CallConsumedCB();
     SendDecoderOutput(CreateDecodedAudio(pts, kFramesPerBuffer));
     frames_written += kFramesPerBuffer;
   }
 
-  EXPECT_FALSE(audio_renderer_->IsSeekingInProgress());
-
   WriteEndOfStream();
 
   EXPECT_EQ(audio_renderer_->GetCurrentTime(), 0);
@@ -210,9 +207,11 @@
   SbMediaTime media_time = audio_renderer_->GetCurrentTime();
 
   while (!audio_renderer_->IsEndOfStreamPlayed()) {
-    SbThreadSleep(AudioRendererImpl::kPrerollTime);
+    SbThreadSleep(kSbTimeMillisecond);
     SbMediaTime new_media_time = audio_renderer_->GetCurrentTime();
-    EXPECT_GT(new_media_time, media_time);
+    // TODO: Replace it with EXPECT_GT once audio time reporting is more
+    //       accurate.
+    EXPECT_GE(new_media_time, media_time);
     media_time = new_media_time;
   }
 }
@@ -229,24 +228,16 @@
   const int kFramesPerBuffer = 1024;
 
   int frames_written = 0;
-  int preroll_frames = kDefaultSamplesPerSecond *
-                       AudioRendererImpl::kPrerollTime / kSbTimeSecond *
-                       kPlaybackRate * 2;
 
-  while (frames_written <= preroll_frames) {
-    SbMediaTime pts = frames_written / kDefaultSamplesPerSecond;
+  while (audio_renderer_->IsSeekingInProgress()) {
+    SbMediaTime pts =
+        frames_written * kSbMediaTimeSecond / kDefaultSamplesPerSecond;
     WriteSample(CreateInputBuffer(pts));
     CallConsumedCB();
     SendDecoderOutput(CreateDecodedAudio(pts, kFramesPerBuffer));
     frames_written += kFramesPerBuffer;
-
-    if (!audio_renderer_->IsSeekingInProgress()) {
-      break;
-    }
   }
 
-  EXPECT_FALSE(audio_renderer_->IsSeekingInProgress());
-
   WriteEndOfStream();
 
   EXPECT_EQ(audio_renderer_->GetCurrentTime(), 0);
@@ -259,9 +250,11 @@
   SbMediaTime media_time = audio_renderer_->GetCurrentTime();
 
   while (!audio_renderer_->IsEndOfStreamPlayed()) {
-    SbThreadSleep(AudioRendererImpl::kPrerollTime);
+    SbThreadSleep(kSbTimeMillisecond);
     SbMediaTime new_media_time = audio_renderer_->GetCurrentTime();
-    EXPECT_GT(new_media_time, media_time);
+    // TODO: Replace it with EXPECT_GT once audio time reporting is more
+    //       accurate.
+    EXPECT_GE(new_media_time, media_time);
     media_time = new_media_time;
   }
 }
@@ -271,31 +264,30 @@
 
   const int kFramesPerBuffer = 1024;
 
-  int frames_written = 0;
-  int preroll_frames = kDefaultSamplesPerSecond *
-                       AudioRendererImpl::kPrerollTime / kSbTimeSecond;
-
   audio_renderer_->Play();
 
-  while (frames_written <= preroll_frames) {
-    SbMediaTime pts = frames_written / kDefaultSamplesPerSecond;
+  int frames_written = 0;
+
+  while (audio_renderer_->IsSeekingInProgress()) {
+    SbMediaTime pts =
+        frames_written * kSbMediaTimeSecond / kDefaultSamplesPerSecond;
     WriteSample(CreateInputBuffer(pts));
     CallConsumedCB();
     SendDecoderOutput(CreateDecodedAudio(pts, kFramesPerBuffer));
     frames_written += kFramesPerBuffer;
   }
 
-  EXPECT_FALSE(audio_renderer_->IsSeekingInProgress());
-
   WriteEndOfStream();
   SendDecoderOutput(new DecodedAudio);
 
   SbMediaTime media_time = audio_renderer_->GetCurrentTime();
 
   while (!audio_renderer_->IsEndOfStreamPlayed()) {
-    SbThreadSleep(AudioRendererImpl::kPrerollTime);
+    SbThreadSleep(kSbTimeMillisecond);
     SbMediaTime new_media_time = audio_renderer_->GetCurrentTime();
-    EXPECT_GT(new_media_time, media_time);
+    // TODO: Replace it with EXPECT_GT once audio time reporting is more
+    //       accurate.
+    EXPECT_GE(new_media_time, media_time);
     media_time = new_media_time;
   }
 }
@@ -354,22 +346,19 @@
   const int kFramesPerBuffer = 1024;
 
   int frames_written = 0;
-  int preroll_frames = kDefaultSamplesPerSecond *
-                       AudioRendererImpl::kPrerollTime / kSbTimeSecond;
 
-  while (frames_written <= preroll_frames) {
-    SbMediaTime pts = frames_written / kDefaultSamplesPerSecond;
+  while (audio_renderer_->IsSeekingInProgress()) {
+    SbMediaTime pts =
+        frames_written * kSbMediaTimeSecond / kDefaultSamplesPerSecond;
     WriteSample(CreateInputBuffer(pts));
     CallConsumedCB();
     SendDecoderOutput(CreateDecodedAudio(pts, kFramesPerBuffer / 2));
     frames_written += kFramesPerBuffer / 2;
-    pts = frames_written / kDefaultSamplesPerSecond;
+    pts = frames_written * kSbMediaTimeSecond / kDefaultSamplesPerSecond;
     SendDecoderOutput(CreateDecodedAudio(pts, kFramesPerBuffer / 2));
     frames_written += kFramesPerBuffer / 2;
   }
 
-  EXPECT_FALSE(audio_renderer_->IsSeekingInProgress());
-
   WriteEndOfStream();
 
   EXPECT_EQ(audio_renderer_->GetCurrentTime(), 0);
@@ -382,9 +371,11 @@
   SbMediaTime media_time = audio_renderer_->GetCurrentTime();
 
   while (!audio_renderer_->IsEndOfStreamPlayed()) {
-    SbThreadSleep(AudioRendererImpl::kPrerollTime);
+    SbThreadSleep(kSbTimeMillisecond);
     SbMediaTime new_media_time = audio_renderer_->GetCurrentTime();
-    EXPECT_GT(new_media_time, media_time);
+    // TODO: Replace it with EXPECT_GT once audio time reporting is more
+    //       accurate.
+    EXPECT_GE(new_media_time, media_time);
     media_time = new_media_time;
   }
 }
@@ -395,20 +386,21 @@
   const int kFramesPerBuffer = 1024;
 
   int frames_written = 0;
-  int preroll_frames = kDefaultSamplesPerSecond *
-                       AudioRendererImpl::kPrerollTime / kSbTimeSecond;
 
-  while (frames_written <= preroll_frames) {
-    SbMediaTime pts = frames_written / kDefaultSamplesPerSecond;
+  while (audio_renderer_->IsSeekingInProgress()) {
+    SbMediaTime pts =
+        frames_written * kSbMediaTimeSecond / kDefaultSamplesPerSecond;
+    SbMediaTime output_pts = pts;
     WriteSample(CreateInputBuffer(pts));
     CallConsumedCB();
-    frames_written += kFramesPerBuffer;
+    frames_written += kFramesPerBuffer / 2;
+    pts = frames_written * kSbMediaTimeSecond / kDefaultSamplesPerSecond;
+    WriteSample(CreateInputBuffer(pts));
+    CallConsumedCB();
+    frames_written += kFramesPerBuffer / 2;
+    SendDecoderOutput(CreateDecodedAudio(output_pts, kFramesPerBuffer));
   }
 
-  SendDecoderOutput(CreateDecodedAudio(0, frames_written));
-
-  EXPECT_FALSE(audio_renderer_->IsSeekingInProgress());
-
   WriteEndOfStream();
 
   EXPECT_EQ(audio_renderer_->GetCurrentTime(), 0);
@@ -421,13 +413,16 @@
   SbMediaTime media_time = audio_renderer_->GetCurrentTime();
 
   while (!audio_renderer_->IsEndOfStreamPlayed()) {
-    SbThreadSleep(AudioRendererImpl::kPrerollTime);
+    SbThreadSleep(kSbTimeMillisecond);
     SbMediaTime new_media_time = audio_renderer_->GetCurrentTime();
-    EXPECT_GT(new_media_time, media_time);
+    // TODO: Replace it with EXPECT_GT once audio time reporting is more
+    //       accurate.
+    EXPECT_GE(new_media_time, media_time);
     media_time = new_media_time;
   }
 }
 
+// TODO: Implement test for Seek()
 TEST_F(AudioRendererImplTest, Seek) {}
 
 }  // namespace
diff --git a/src/starboard/shared/starboard/player/filter/wsola_internal.cc b/src/starboard/shared/starboard/player/filter/wsola_internal.cc
index f1f32f5..3742d0a 100644
--- a/src/starboard/shared/starboard/player/filter/wsola_internal.cc
+++ b/src/starboard/shared/starboard/player/filter/wsola_internal.cc
@@ -33,6 +33,12 @@
 #include "starboard/memory.h"
 #include "starboard/shared/internal_only.h"
 
+// TODO: Detect Neon on ARM platform and enable SIMD.
+#if SB_IS(ARCH_X86)
+#define USE_SIMD 1
+#include <xmmintrin.h>
+#endif  // SB_IS(ARCH_X86)
+
 namespace starboard {
 namespace shared {
 namespace starboard {
@@ -68,9 +74,53 @@
   SB_DCHECK(frame_offset_a + num_frames <= a->frames());
   SB_DCHECK(frame_offset_b + num_frames <= b->frames());
 
-  SbMemorySet(dot_product, 0, sizeof(*dot_product) * a->channels());
   const float* a_frames = reinterpret_cast<const float*>(a->buffer());
   const float* b_frames = reinterpret_cast<const float*>(b->buffer());
+
+// SIMD optimized variants can provide a massive speedup to this operation.
+#if defined(USE_SIMD)
+  const int rem = num_frames % 4;
+  const int last_index = num_frames - rem;
+  const int channels = a->channels();
+  for (int ch = 0; ch < channels; ++ch) {
+    const float* a_src = a_frames + frame_offset_a * a->channels() + ch;
+    const float* b_src = b_frames + frame_offset_b * b->channels() + ch;
+
+#if SB_IS(ARCH_X86)
+    // First sum all components.
+    __m128 m_sum = _mm_setzero_ps();
+    for (int s = 0; s < last_index; s += 4) {
+      m_sum = _mm_add_ps(
+          m_sum, _mm_mul_ps(_mm_loadu_ps(a_src + s), _mm_loadu_ps(b_src + s)));
+    }
+
+    // Reduce to a single float for this channel. Sadly, SSE1,2 doesn't have a
+    // horizontal sum function, so we have to condense manually.
+    m_sum = _mm_add_ps(_mm_movehl_ps(m_sum, m_sum), m_sum);
+    _mm_store_ss(dot_product + ch,
+                 _mm_add_ss(m_sum, _mm_shuffle_ps(m_sum, m_sum, 1)));
+#elif SB_IS(ARCH_ARM)
+    // First sum all components.
+    float32x4_t m_sum = vmovq_n_f32(0);
+    for (int s = 0; s < last_index; s += 4)
+      m_sum = vmlaq_f32(m_sum, vld1q_f32(a_src + s), vld1q_f32(b_src + s));
+
+    // Reduce to a single float for this channel.
+    float32x2_t m_half = vadd_f32(vget_high_f32(m_sum), vget_low_f32(m_sum));
+    dot_product[ch] = vget_lane_f32(vpadd_f32(m_half, m_half), 0);
+#endif  // SB_IS(ARCH_X86)
+  }
+
+  if (!rem) {
+    return;
+  }
+  num_frames = rem;
+  frame_offset_a += last_index;
+  frame_offset_b += last_index;
+#else   // defined(USE_SIMD)
+  SbMemorySet(dot_product, 0, sizeof(*dot_product) * a->channels());
+#endif  // defined(USE_SIMD)
+
   for (int k = 0; k < a->channels(); ++k) {
     const float* ch_a = a_frames + frame_offset_a * a->channels() + k;
     const float* ch_b = b_frames + frame_offset_b * b->channels() + k;
diff --git a/src/starboard/shared/starboard/player/job_queue.h b/src/starboard/shared/starboard/player/job_queue.h
index 361d82e..53b7576 100644
--- a/src/starboard/shared/starboard/player/job_queue.h
+++ b/src/starboard/shared/starboard/player/job_queue.h
@@ -53,6 +53,7 @@
     void Schedule(Closure closure, SbTimeMonotonic delay = 0) {
       job_queue_->Schedule(closure, this, delay);
     }
+    void Remove(Closure closure) { job_queue_->Remove(closure); }
     void CancelPendingJobs() { job_queue_->RemoveJobsByToken(this); }
 
    private:
diff --git a/src/starboard/shared/uwp/application_uwp.cc b/src/starboard/shared/uwp/application_uwp.cc
index 6ebcda9..acc7057 100644
--- a/src/starboard/shared/uwp/application_uwp.cc
+++ b/src/starboard/shared/uwp/application_uwp.cc
@@ -26,9 +26,11 @@
 
 #include "starboard/event.h"
 #include "starboard/log.h"
+#include "starboard/mutex.h"
 #include "starboard/shared/starboard/application.h"
 #include "starboard/shared/starboard/audio_sink/audio_sink_internal.h"
 #include "starboard/shared/starboard/player/video_frame_internal.h"
+#include "starboard/shared/uwp/async_utils.h"
 #include "starboard/shared/uwp/window_internal.h"
 #include "starboard/shared/win32/thread_private.h"
 #include "starboard/shared/win32/wchar_utils.h"
@@ -42,6 +44,7 @@
 using starboard::shared::starboard::CommandLine;
 using starboard::shared::uwp::ApplicationUwp;
 using starboard::shared::uwp::GetArgvZero;
+using starboard::shared::uwp::WaitForResult;
 using starboard::shared::win32::stringToPlatformString;
 using starboard::shared::win32::wchar_tToUTF8;
 using starboard::shared::starboard::player::VideoFrame;
@@ -59,6 +62,13 @@
 using Windows::Foundation::TimeSpan;
 using Windows::Foundation::TypedEventHandler;
 using Windows::Foundation::Uri;
+using Windows::Media::Protection::HdcpProtection;
+using Windows::Media::Protection::HdcpSession;
+using Windows::Media::Protection::HdcpSetProtectionResult;
+using Windows::Networking::Connectivity::ConnectionProfile;
+using Windows::Networking::Connectivity::NetworkConnectivityLevel;
+using Windows::Networking::Connectivity::NetworkInformation;
+using Windows::Networking::Connectivity::NetworkStatusChangedEventHandler;
 using Windows::System::Threading::ThreadPoolTimer;
 using Windows::System::Threading::TimerElapsedHandler;
 using Windows::System::UserAuthenticationStatus;
@@ -203,7 +213,7 @@
 
 ref class App sealed : public IFrameworkView {
  public:
-  App() : previously_activated_(false) {}
+  App() : previously_activated_(false), has_internet_access_(false) {}
 
   // IFrameworkView methods.
   virtual void Initialize(CoreApplicationView^ applicationView) {
@@ -215,7 +225,37 @@
     applicationView->Activated +=
         ref new TypedEventHandler<CoreApplicationView^, IActivatedEventArgs^>(
             this, &App::OnActivated);
+
+    has_internet_access_ = HasInternetAccess();
+
+    SB_LOG(INFO) << "Has internet access? " << std::boolalpha
+                 << has_internet_access_;
+
+    NetworkInformation::NetworkStatusChanged +=
+        ref new NetworkStatusChangedEventHandler(this, &App::OnNetworkChanged);
   }
+
+  void OnNetworkChanged(Platform::Object^ sender) {
+    SB_UNREFERENCED_PARAMETER(sender);
+    bool has_internet_access = HasInternetAccess();
+
+    if (has_internet_access == has_internet_access_) {
+      return;
+    }
+
+    SB_LOG(INFO) << "NetworkChanged.  Has internet access? " << std::boolalpha
+      << has_internet_access;
+
+    has_internet_access_ = has_internet_access;
+
+    const SbEventType network_event =
+        (has_internet_access ? kSbEventTypeNetworkConnect
+                             : kSbEventTypeNetworkDisconnect);
+
+    ApplicationUwp::Get()->Inject(
+        new ApplicationUwp::Event(network_event, nullptr, nullptr));
+  }
+
   virtual void SetWindow(CoreWindow^ window) {
     ApplicationUwp::Get()->SetCoreWindow(window);
     window->KeyUp += ref new TypedEventHandler<CoreWindow^, KeyEventArgs^>(
@@ -363,7 +403,33 @@
     }
   }
 
+  bool HasInternetAccess() {
+    ConnectionProfile^ connection_profile =
+        NetworkInformation::GetInternetConnectionProfile();
+
+    if (connection_profile == nullptr) {
+      return false;
+    }
+    const NetworkConnectivityLevel connectivity_level =
+        connection_profile->GetNetworkConnectivityLevel();
+
+    switch (connectivity_level) {
+      case NetworkConnectivityLevel::InternetAccess:
+      case NetworkConnectivityLevel::ConstrainedInternetAccess:
+        return true;
+      case NetworkConnectivityLevel::None:
+      case NetworkConnectivityLevel::LocalAccess:
+        return false;
+    }
+    SB_NOTREACHED() << "Unknown network connectivity level found "
+                    << sbwin32::platformStringToString(
+                            connectivity_level.ToString());
+
+    return false;
+  }
+
   bool previously_activated_;
+  bool has_internet_access_;
   // Only valid if previously_activated_ is true
   ActivationKind previous_activation_kind_;
   std::vector<std::string> args_;
@@ -448,6 +514,23 @@
   return false;
 }
 
+Windows::Media::Protection::HdcpSession^ ApplicationUwp::GetHdcpSession() {
+  if (!hdcp_session_) {
+    hdcp_session_ = ref new HdcpSession();
+  }
+  return hdcp_session_;
+}
+
+void ApplicationUwp::ResetHdcpSession() {
+  // delete will call the destructor, but not free memory.
+  // The destructor is called explicitly so that HDCP session can be
+  // torn down immediately.
+  if (hdcp_session_) {
+    delete hdcp_session_;
+    hdcp_session_ = nullptr;
+  }
+}
+
 void ApplicationUwp::Inject(Application::Event* event) {
   RunInMainThreadAsync([this, event]() {
     bool result = DispatchAndDelete(event);
@@ -529,6 +612,54 @@
   return stringToPlatformString(localized_strings_.GetString(id, fallback));
 }
 
+bool ApplicationUwp::IsHdcpOn() {
+  ::starboard::ScopedLock lock(hdcp_session_mutex_);
+
+  return GetHdcpSession()->IsEffectiveProtectionAtLeast(HdcpProtection::On);
+}
+
+bool ApplicationUwp::TurnOnHdcp() {
+  HdcpSetProtectionResult protection_result;
+  {
+    ::starboard::ScopedLock lock(hdcp_session_mutex_);
+
+    protection_result =
+      WaitForResult(GetHdcpSession()->SetDesiredMinProtectionAsync(
+        HdcpProtection::On));
+  }
+
+  if (IsHdcpOn()) {
+    return true;
+  }
+
+  // If the operation did not have intended result, log something.
+  switch (protection_result) {
+  case HdcpSetProtectionResult::Success:
+    SB_LOG(INFO) << "Successfully set HDCP.";
+    break;
+  case HdcpSetProtectionResult::NotSupported:
+    SB_LOG(INFO) << "HDCP is not supported.";
+    break;
+  case HdcpSetProtectionResult::TimedOut:
+    SB_LOG(INFO) << "Setting HDCP timed out.";
+    break;
+  case HdcpSetProtectionResult::UnknownFailure:
+    SB_LOG(INFO) << "Unknown failure returned while setting HDCP.";
+    break;
+  }
+
+  return false;
+}
+
+bool ApplicationUwp::TurnOffHdcp() {
+  {
+    ::starboard::ScopedLock lock(hdcp_session_mutex_);
+    ResetHdcpSession();
+  }
+  bool success = !SbMediaIsOutputProtected();
+  return success;
+}
+
 void ApplicationUwp::AcceptFrame(SbPlayer player,
                                  const scoped_refptr<VideoFrame>& frame,
                                  int x,
diff --git a/src/starboard/shared/uwp/application_uwp.h b/src/starboard/shared/uwp/application_uwp.h
index fda18e2..e62305c 100644
--- a/src/starboard/shared/uwp/application_uwp.h
+++ b/src/starboard/shared/uwp/application_uwp.h
@@ -32,6 +32,8 @@
 namespace shared {
 namespace uwp {
 
+using Windows::Media::Protection::HdcpSession;
+
 // Returns win32's GetModuleFileName(). For cases where we'd like an argv[0].
 std::string GetArgvZero();
 
@@ -97,6 +99,12 @@
 
   Platform::String^ GetString(const char* id, const char* fallback) const;
 
+  bool IsHdcpOn();
+  // Returns true on success.
+  bool TurnOnHdcp();
+  // Returns true on success.
+  bool TurnOffHdcp();
+
  private:
   // --- Application overrides ---
   bool IsStartImmediate() SB_OVERRIDE { return false; }
@@ -116,6 +124,11 @@
                    int width,
                    int height) SB_OVERRIDE;
 
+  // These two functions should only be called while holding
+  // |hdcp_session_mutex_|.
+  Windows::Media::Protection::HdcpSession^ GetHdcpSession();
+  void ResetHdcpSession();
+
   // The single open window, if any.
   SbWindow window_;
   Platform::Agile<Windows::UI::Core::CoreWindow> core_window_;
@@ -123,9 +136,13 @@
   shared::starboard::LocalizedStrings localized_strings_;
 
   Mutex mutex_;
-  // Locked by mutex_
+  // |timer_event_map_| is locked by |mutex_|.
   std::unordered_map<SbEventId, Windows::System::Threading::ThreadPoolTimer^>
     timer_event_map_;
+
+  // |hdcp_session_| is locked by |hdcp_session_mutex_|.
+  Mutex hdcp_session_mutex_;
+  Windows::Media::Protection::HdcpSession^ hdcp_session_;
 };
 
 }  // namespace uwp
diff --git a/src/starboard/shared/uwp/async_utils.h b/src/starboard/shared/uwp/async_utils.h
index 7e01f82..ad4d4e7 100644
--- a/src/starboard/shared/uwp/async_utils.h
+++ b/src/starboard/shared/uwp/async_utils.h
@@ -34,6 +34,7 @@
   concurrency::create_task(operation,
                            task_continuation_context::use_arbitrary())
       .then([&event](TResult result) {
+        SB_UNREFERENCED_PARAMETER(result);
         BOOL success = SetEvent(event);
         SB_DCHECK(success);
       }, task_continuation_context::use_arbitrary());
diff --git a/src/starboard/shared/uwp/media_is_output_protected.cc b/src/starboard/shared/uwp/media_is_output_protected.cc
new file mode 100644
index 0000000..117f65a
--- /dev/null
+++ b/src/starboard/shared/uwp/media_is_output_protected.cc
@@ -0,0 +1,21 @@
+// Copyright 2017 Google Inc. All Rights Reserved.

+//

+// Licensed under the Apache License, Version 2.0 (the "License");

+// you may not use this file except in compliance with the License.

+// You may obtain a copy of the License at

+//

+//     http://www.apache.org/licenses/LICENSE-2.0

+//

+// Unless required by applicable law or agreed to in writing, software

+// distributed under the License is distributed on an "AS IS" BASIS,

+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.

+// See the License for the specific language governing permissions and

+// limitations under the License.

+

+#include "starboard/media.h"

+

+#include "starboard/shared/uwp/application_uwp.h"

+

+bool SbMediaIsOutputProtected() {

+  return starboard::shared::uwp::ApplicationUwp::Get()->IsHdcpOn();

+}

diff --git a/src/starboard/shared/uwp/media_set_output_protection.cc b/src/starboard/shared/uwp/media_set_output_protection.cc
new file mode 100644
index 0000000..1a3cc31
--- /dev/null
+++ b/src/starboard/shared/uwp/media_set_output_protection.cc
@@ -0,0 +1,50 @@
+// Copyright 2017 Google Inc. All Rights Reserved.

+//

+// Licensed under the Apache License, Version 2.0 (the "License");

+// you may not use this file except in compliance with the License.

+// You may obtain a copy of the License at

+//

+//     http://www.apache.org/licenses/LICENSE-2.0

+//

+// Unless required by applicable law or agreed to in writing, software

+// distributed under the License is distributed on an "AS IS" BASIS,

+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.

+// See the License for the specific language governing permissions and

+// limitations under the License.

+

+#include "starboard/media.h"

+

+#include "starboard/log.h"

+#include "starboard/shared/uwp/application_uwp.h"

+#include "starboard/time.h"

+

+using starboard::shared::uwp::ApplicationUwp;

+

+bool SbMediaSetOutputProtection(bool should_enable_dhcp) {

+  bool is_hdcp_on = SbMediaIsOutputProtected();

+

+  if (is_hdcp_on == should_enable_dhcp) {

+    return true;

+  }

+

+  SB_LOG(INFO) << "Attempting to "

+               << (should_enable_dhcp ? "enable" : "disable")

+               << " output protection.  Current status: "

+               << (is_hdcp_on ? "enabled" : "disabled");

+  SbTimeMonotonic tick = SbTimeGetMonotonicNow();

+

+  bool hdcp_success = false;

+  if (should_enable_dhcp) {

+    hdcp_success = ApplicationUwp::Get()->TurnOnHdcp();

+  } else {

+    hdcp_success = ApplicationUwp::Get()->TurnOffHdcp();

+  }

+

+  is_hdcp_on = (hdcp_success ? should_enable_dhcp : !should_enable_dhcp);

+

+  SbTimeMonotonic tock = SbTimeGetMonotonicNow();

+  SB_LOG(INFO) << "Output protection is "

+    << (is_hdcp_on ? "enabled" : "disabled") << ".  Toggling HDCP took "

+    << (tock - tick) / kSbTimeMillisecond << " milliseconds.";

+  return hdcp_success;

+}

diff --git a/src/starboard/shared/uwp/starboard_platform.gypi b/src/starboard/shared/uwp/starboard_platform.gypi
index 85f0f72..8d082dc 100644
--- a/src/starboard/shared/uwp/starboard_platform.gypi
+++ b/src/starboard/shared/uwp/starboard_platform.gypi
@@ -19,6 +19,8 @@
       'application_uwp.h',
       'async_utils.h',
       'get_home_directory.cc',
+      'media_is_output_protected.cc',
+      'media_set_output_protection.cc',
       'system_clear_platform_error.cc',
       'system_get_device_type.cc',
       'system_get_property.cc',
diff --git a/src/starboard/shared/win32/drm_system_playready.cc b/src/starboard/shared/win32/drm_system_playready.cc
index 38fb628..bfa8dcd 100644
--- a/src/starboard/shared/win32/drm_system_playready.cc
+++ b/src/starboard/shared/win32/drm_system_playready.cc
@@ -184,8 +184,9 @@
       }
 
       if (item.second->IsHDCPRequired()) {
-        // TODO: Enforce HDCP
-        // if (!is_hdcp_enabled()) { return kFailure; }
+        if (!SbMediaSetOutputProtection(true)) {
+          return kFailure;
+        }
       }
 
       return kSuccess;
diff --git a/src/starboard/shared/win32/media_is_output_protected.cc b/src/starboard/shared/win32/media_is_output_protected.cc
index 213e5f5..03e6642 100644
--- a/src/starboard/shared/win32/media_is_output_protected.cc
+++ b/src/starboard/shared/win32/media_is_output_protected.cc
@@ -1,21 +1,20 @@
-// Copyright 2017 Google Inc. All Rights Reserved.
-//
-// Licensed under the Apache License, Version 2.0 (the "License");
-// you may not use this file except in compliance with the License.
-// You may obtain a copy of the License at
-//
-//     http://www.apache.org/licenses/LICENSE-2.0
-//
-// Unless required by applicable law or agreed to in writing, software
-// distributed under the License is distributed on an "AS IS" BASIS,
-// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
-// See the License for the specific language governing permissions and
-// limitations under the License.
-
-#include "starboard/media.h"
-
-bool SbMediaIsOutputProtected() {
-  // Pretend that HDCP is always on.
-  // TODO: Fix this with proper HDCP query.
-  return true;
-}
+// Copyright 2017 Google Inc. All Rights Reserved.

+//

+// Licensed under the Apache License, Version 2.0 (the "License");

+// you may not use this file except in compliance with the License.

+// You may obtain a copy of the License at

+//

+//     http://www.apache.org/licenses/LICENSE-2.0

+//

+// Unless required by applicable law or agreed to in writing, software

+// distributed under the License is distributed on an "AS IS" BASIS,

+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.

+// See the License for the specific language governing permissions and

+// limitations under the License.

+

+#include "starboard/media.h"

+

+bool SbMediaIsOutputProtected() {

+  SB_NOTIMPLEMENTED();

+  return false;

+}

diff --git a/src/starboard/shared/x11/window_internal.h b/src/starboard/shared/x11/window_internal.h
index 5dcc153..6740a9e 100644
--- a/src/starboard/shared/x11/window_internal.h
+++ b/src/starboard/shared/x11/window_internal.h
@@ -15,6 +15,7 @@
 #ifndef STARBOARD_SHARED_X11_WINDOW_INTERNAL_H_
 #define STARBOARD_SHARED_X11_WINDOW_INTERNAL_H_
 
+#include <X11/extensions/Xrender.h>
 #include <X11/Xlib.h>
 
 #include "starboard/configuration.h"
@@ -22,18 +23,12 @@
 #include "starboard/shared/starboard/player/video_frame_internal.h"
 #include "starboard/window.h"
 
-#if SB_TRUE || SB_IS(PLAYER_PUNCHED_OUT)
-#include <X11/extensions/Xrender.h>
-#endif  // SB_TRUE ||
-        // SB_IS(PLAYER_PUNCHED_OUT)
-
 struct SbWindowPrivate {
   SbWindowPrivate(Display* display, const SbWindowOptions* options);
   ~SbWindowPrivate();
 
   Window window;
 
-#if SB_TRUE || SB_IS(PLAYER_PUNCHED_OUT)
   typedef ::starboard::shared::starboard::player::VideoFrame VideoFrame;
 
   // Composites graphics and the given video frame video for this window. In
@@ -80,8 +75,6 @@
 
   // A cached XRender Picture wrapper for |gl_window|.
   Picture gl_picture;
-#endif  // SB_TRUE ||
-        // SB_IS(PLAYER_PUNCHED_OUT)
 
   // The display that this window was created from.
   Display* display;
diff --git a/src/starboard/win/shared/gyp_configuration.gypi b/src/starboard/win/shared/gyp_configuration.gypi
index 6b18f49..c112351 100644
--- a/src/starboard/win/shared/gyp_configuration.gypi
+++ b/src/starboard/win/shared/gyp_configuration.gypi
@@ -139,6 +139,11 @@
                 # Enable some warnings, even those that are disabled by default.
                 # See https://msdn.microsoft.com/en-us/library/23k5d385.aspx
                 'WarningLevel': '4',
+                'AdditionalOptions': [
+                  # Warn if an enumeration value is unhandled in switch (C4062).
+                  # This warning is off by default, so it must be turned on explicitly.
+                  '/w44062',
+                ],
               },
             },
           }],
diff --git a/src/starboard/win/shared/starboard_platform.gypi b/src/starboard/win/shared/starboard_platform.gypi
index 42dacc9..2a4362a 100644
--- a/src/starboard/win/shared/starboard_platform.gypi
+++ b/src/starboard/win/shared/starboard_platform.gypi
@@ -246,7 +246,6 @@
         '<(DEPTH)/starboard/shared/stub/cryptography_transform.cc',
         '<(DEPTH)/starboard/shared/stub/image_decode.cc',
         '<(DEPTH)/starboard/shared/stub/image_is_decode_supported.cc',
-        '<(DEPTH)/starboard/shared/stub/media_set_output_protection.cc',
         '<(DEPTH)/starboard/shared/stub/system_get_stack.cc',
         '<(DEPTH)/starboard/shared/stub/system_get_total_gpu_memory.cc',
         '<(DEPTH)/starboard/shared/stub/system_get_used_gpu_memory.cc',
diff --git a/src/starboard/win/win32/starboard_platform.gypi b/src/starboard/win/win32/starboard_platform.gypi
index 2335dbd..b96b643 100644
--- a/src/starboard/win/win32/starboard_platform.gypi
+++ b/src/starboard/win/win32/starboard_platform.gypi
@@ -33,6 +33,8 @@
       '<(DEPTH)/starboard/shared/starboard/system_request_unpause.cc',
       '<(DEPTH)/starboard/shared/stub/decode_target_get_info.cc',
       '<(DEPTH)/starboard/shared/stub/decode_target_release.cc',
+      '<(DEPTH)/starboard/shared/stub/media_is_output_protected.cc',
+      '<(DEPTH)/starboard/shared/stub/media_set_output_protection.cc',
       '<(DEPTH)/starboard/win/shared/system_get_path.cc',
       '<@(uwp_incompatible_win32)',
     ],
diff --git a/src/third_party/angle/src/libANGLE/renderer/d3d/SurfaceD3D.cpp b/src/third_party/angle/src/libANGLE/renderer/d3d/SurfaceD3D.cpp
index 4f10d28..7559686 100644
--- a/src/third_party/angle/src/libANGLE/renderer/d3d/SurfaceD3D.cpp
+++ b/src/third_party/angle/src/libANGLE/renderer/d3d/SurfaceD3D.cpp
@@ -107,8 +107,7 @@
             out = sizeof(mDXGIBuffer);
             hr = static_cast<ID3D11DeviceChild*>(mD3DTexture)->
                 GetPrivateData(kCobaltDxgiBuffer, &out, &mDXGIBuffer);
-            ASSERT(SUCCEEDED(hr));
-            if (mDXGIBuffer != nullptr) {
+            if (SUCCEEDED(hr) && mDXGIBuffer != nullptr) {
               mDXGIBuffer->AddRef();
             }
             break;
@@ -148,8 +147,11 @@
 
         D3D11_TEXTURE2D_DESC texture_desc;
         d3Texture->GetDesc(&texture_desc);
-        if ((texture_desc.BindFlags & D3D11_BIND_RENDER_TARGET) == 0)
+
+        if (texture_desc.Format == DXGI_FORMAT_NV12)
         {
+            // NV12 textures cannot be rendered to,
+            // so don't proceed to making a swap chain.
             return egl::Error(EGL_SUCCESS);
         }
     }
diff --git a/src/third_party/libevent/kqueue.c b/src/third_party/libevent/kqueue.c
index 556b73c..bcad213 100644
--- a/src/third_party/libevent/kqueue.c
+++ b/src/third_party/libevent/kqueue.c
@@ -30,6 +30,22 @@
 #include "config.h"
 #endif
 
+#ifdef STARBOARD
+#include "libevent-starboard.h"
+
+#include <assert.h>
+#include <sys/types.h>
+#include <sys/event.h>
+#include <sys/socket.h>
+
+// Use libevent's local compatibility  versions of these.
+#include "third_party/libevent/compat/sys/queue.h"
+#include "third_party/libevent/compat/sys/_libevent_time.h"
+
+// Include Starboard poems after all system headers.
+#include "starboard/client_porting/poem/stdio_poem.h"
+#include "starboard/client_porting/poem/string_poem.h"
+#else  // STARBOARD
 #define _GNU_SOURCE 1
 
 #include <sys/types.h>
@@ -50,6 +66,7 @@
 #ifdef HAVE_INTTYPES_H
 #include <inttypes.h>
 #endif
+#endif  // STARBOARD
 
 /* Some platforms apparently define the udata field of struct kevent as
  * intptr_t, whereas others define it as void*.  There doesn't seem to be an
@@ -62,6 +79,9 @@
 
 #include "event.h"
 #include "event-internal.h"
+#ifndef STARBOARD
+#include "evsignal.h"
+#endif  // STARBOARD
 #include "log.h"
 
 #define EVLIST_X_KQINKERNEL	0x1000
@@ -148,11 +168,12 @@
 	 * stick an error in events[0].  If kqueue is broken, then
 	 * kevent will fail.
 	 */
+   // https://github.com/azat/libevent/commit/a9f6f32e7773347334149d89bc58bf9b984e1714
 	if (kevent(kq,
 		kqueueop->changes, 1, kqueueop->events, NEVENT, NULL) != 1 ||
 	    kqueueop->events[0].ident != -1 ||
-	    kqueueop->events[0].flags != EV_ERROR) {
-		event_warn("%s: detected broken kqueue; not using.", __func__);
+	    !(kqueueop->events[0].flags & EV_ERROR)) {
+    event_warn("%s: detected broken kqueue; not using.", __func__);
 		free(kqueueop->changes);
 		free(kqueueop->events);
 		free(kqueueop);
@@ -234,7 +255,6 @@
 	kqop->nchanges = 0;
 	if (res == -1) {
 		if (errno != EINTR) {
-                        event_warn("kevent");
 			return (-1);
 		}
 
diff --git a/src/third_party/libevent/libevent.gyp b/src/third_party/libevent/libevent.gyp
index 3b9f4ed..d62a6c9 100644
--- a/src/third_party/libevent/libevent.gyp
+++ b/src/third_party/libevent/libevent.gyp
@@ -24,6 +24,7 @@
             'evrpc.c',
             'evutil.c',
             'http.c',
+            'kqueue.c',
             'log.c',
             'poll.c',
             'select.c',
@@ -54,10 +55,22 @@
               'include_dirs': [ 'starboard' ],
               'conditions': [
                 [ 'sb_libevent_method == "epoll"', {
-                  'sources!': [ 'poll.c' ],
+                  'sources!': [
+                    'kqueue.c',
+                    'poll.c',
+                  ],
                 }],
                 [ 'sb_libevent_method == "poll"', {
-                  'sources!': [ 'epoll.c' ],
+                  'sources!': [
+                    'epoll.c',
+                    'kqueue.c',
+                  ],
+                }],
+                [ 'sb_libevent_method == "kqueue"', {
+                  'sources!': [
+                    'epoll.c',
+                    'poll.c',
+                  ],
                 }],
                 [ 'target_os == "linux"', {
                   'sources': [ 'epoll_sub.c' ],
@@ -74,6 +87,10 @@
                   'include_dirs': [ 'starboard/linux' ],
                   }
                 ],
+                [ 'target_os == "ios"', {
+                  'include_dirs': [ 'starboard/darwin' ],
+                  }
+                ],
                 [ 'target_os == "orbis"', {
                   'include_dirs': [ 'starboard/ps4' ],
                   }
diff --git a/src/third_party/libevent/starboard/darwin/libevent-starboard.h b/src/third_party/libevent/starboard/darwin/libevent-starboard.h
new file mode 100644
index 0000000..e1d5a42
--- /dev/null
+++ b/src/third_party/libevent/starboard/darwin/libevent-starboard.h
@@ -0,0 +1,281 @@
+/*
+ * Copyright 2017 Google Inc. All Rights Reserved.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+#ifndef _LIBEVENT_STARBOARD_H_
+#define _LIBEVENT_STARBOARD_H_
+
+#include "starboard/atomic.h"
+#include "starboard/types.h"
+
+#include <errno.h>
+#include <sys/time.h>
+#include <time.h>
+#include <unistd.h>
+
+/* Define if clock_gettime is available in libc */
+/* #undef _EVENT_DNS_USE_CPU_CLOCK_FOR_ID */
+
+/* Define is no secure id variant is available */
+#define _EVENT_DNS_USE_GETTIMEOFDAY_FOR_ID 1
+
+/* Define to 1 if you have the `clock_gettime' function. */
+/* #undef _EVENT_HAVE_CLOCK_GETTIME */
+
+/* Define if /dev/poll is available */
+/* #undef _EVENT_HAVE_DEVPOLL */
+
+/* Define to 1 if you have the <dlfcn.h> header file. */
+#define _EVENT_HAVE_DLFCN_H 1
+
+/* Define if your system supports the epoll system calls */
+/* #undef _EVENT_HAVE_EPOLL */
+
+/* Define to 1 if you have the `epoll_ctl' function. */
+/* #undef _EVENT_HAVE_EPOLL_CTL */
+
+/* Define if your system supports event ports */
+/* #undef _EVENT_HAVE_EVENT_PORTS */
+
+/* Define to 1 if you have the `fcntl' function. */
+#define _EVENT_HAVE_FCNTL 1
+
+/* Define to 1 if you have the <fcntl.h> header file. */
+#define _EVENT_HAVE_FCNTL_H 1
+
+/* Define to 1 if the system has the type `fd_mask'. */
+#define _EVENT_HAVE_FD_MASK 1
+
+/* Define to 1 if you have the `getaddrinfo' function. */
+#define _EVENT_HAVE_GETADDRINFO 1
+
+/* Define to 1 if you have the `getegid' function. */
+#define _EVENT_HAVE_GETEGID 1
+
+/* Define to 1 if you have the `geteuid' function. */
+#define _EVENT_HAVE_GETEUID 1
+
+/* Define to 1 if you have the `getnameinfo' function. */
+#define _EVENT_HAVE_GETNAMEINFO 1
+
+/* Define to 1 if you have the `gettimeofday' function. */
+#define _EVENT_HAVE_GETTIMEOFDAY 1
+
+/* Define to 1 if you have the `inet_ntop' function. */
+#define _EVENT_HAVE_INET_NTOP 1
+
+/* Define to 1 if you have the <inttypes.h> header file. */
+#define _EVENT_HAVE_INTTYPES_H 1
+
+/* Define to 1 if you have the `issetugid' function. */
+#define _EVENT_HAVE_ISSETUGID 1
+
+/* Define to 1 if you have the `kqueue' function. */
+#define _EVENT_HAVE_KQUEUE 1
+
+/* Define to 1 if you have the `nsl' library (-lnsl). */
+/* #undef _EVENT_HAVE_LIBNSL */
+
+/* Define to 1 if you have the `resolv' library (-lresolv). */
+#define _EVENT_HAVE_LIBRESOLV 1
+
+/* Define to 1 if you have the `rt' library (-lrt). */
+/* #undef _EVENT_HAVE_LIBRT */
+
+/* Define to 1 if you have the `socket' library (-lsocket). */
+/* #undef _EVENT_HAVE_LIBSOCKET */
+
+/* Define to 1 if you have the <memory.h> header file. */
+#define _EVENT_HAVE_MEMORY_H 1
+
+/* Define to 1 if you have the <netinet/in6.h> header file. */
+/* #undef _EVENT_HAVE_NETINET_IN6_H */
+
+/* Define to 1 if you have the `poll' function. */
+#define _EVENT_HAVE_POLL 1
+
+/* Define to 1 if you have the <poll.h> header file. */
+#define _EVENT_HAVE_POLL_H 1
+
+/* Define to 1 if you have the `port_create' function. */
+/* #undef _EVENT_HAVE_PORT_CREATE */
+
+/* Define to 1 if you have the <port.h> header file. */
+/* #undef _EVENT_HAVE_PORT_H */
+
+/* Define to 1 if you have the `select' function. */
+#define _EVENT_HAVE_SELECT 1
+
+/* Define if F_SETFD is defined in <fcntl.h> */
+#define _EVENT_HAVE_SETFD 1
+
+/* Define to 1 if you have the `sigaction' function. */
+#define _EVENT_HAVE_SIGACTION 1
+
+/* Define to 1 if you have the `signal' function. */
+#define _EVENT_HAVE_SIGNAL 1
+
+/* Define to 1 if you have the <signal.h> header file. */
+#define _EVENT_HAVE_SIGNAL_H 1
+
+/* Define to 1 if you have the <stdarg.h> header file. */
+#define _EVENT_HAVE_STDARG_H 1
+
+/* Define to 1 if you have the <stdint.h> header file. */
+#define _EVENT_HAVE_STDINT_H 1
+
+/* Define to 1 if you have the <stdlib.h> header file. */
+#define _EVENT_HAVE_STDLIB_H 1
+
+/* Define to 1 if you have the <strings.h> header file. */
+#define _EVENT_HAVE_STRINGS_H 1
+
+/* Define to 1 if you have the <string.h> header file. */
+#define _EVENT_HAVE_STRING_H 1
+
+/* Define to 1 if you have the `strlcpy' function. */
+#define _EVENT_HAVE_STRLCPY 1
+
+/* Define to 1 if you have the `strsep' function. */
+#define _EVENT_HAVE_STRSEP 1
+
+/* Define to 1 if you have the `strtok_r' function. */
+#define _EVENT_HAVE_STRTOK_R 1
+
+/* Define to 1 if you have the `strtoll' function. */
+#define _EVENT_HAVE_STRTOLL 1
+
+/* Define to 1 if the system has the type `struct in6_addr'. */
+#define _EVENT_HAVE_STRUCT_IN6_ADDR 1
+
+/* Define to 1 if you have the <sys/devpoll.h> header file. */
+/* #undef _EVENT_HAVE_SYS_DEVPOLL_H */
+
+/* Define to 1 if you have the <sys/epoll.h> header file. */
+/* #undef _EVENT_HAVE_SYS_EPOLL_H */
+
+/* Define to 1 if you have the <sys/event.h> header file. */
+#define _EVENT_HAVE_SYS_EVENT_H 1
+
+/* Define to 1 if you have the <sys/ioctl.h> header file. */
+#define _EVENT_HAVE_SYS_IOCTL_H 1
+
+/* Define to 1 if you have the <sys/param.h> header file. */
+#define _EVENT_HAVE_SYS_PARAM_H 1
+
+/* Define to 1 if you have the <sys/queue.h> header file. */
+#define _EVENT_HAVE_SYS_QUEUE_H 1
+
+/* Define to 1 if you have the <sys/select.h> header file. */
+#define _EVENT_HAVE_SYS_SELECT_H 1
+
+/* Define to 1 if you have the <sys/socket.h> header file. */
+#define _EVENT_HAVE_SYS_SOCKET_H 1
+
+/* Define to 1 if you have the <sys/stat.h> header file. */
+#define _EVENT_HAVE_SYS_STAT_H 1
+
+/* Define to 1 if you have the <sys/time.h> header file. */
+#define _EVENT_HAVE_SYS_TIME_H 1
+
+/* Define to 1 if you have the <sys/types.h> header file. */
+#define _EVENT_HAVE_SYS_TYPES_H 1
+
+/* Define if TAILQ_FOREACH is defined in <sys/queue.h> */
+#define _EVENT_HAVE_TAILQFOREACH 1
+
+/* Define if timeradd is defined in <sys/time.h> */
+#define _EVENT_HAVE_TIMERADD 1
+
+/* Define if timerclear is defined in <sys/time.h> */
+#define _EVENT_HAVE_TIMERCLEAR 1
+
+/* Define if timercmp is defined in <sys/time.h> */
+#define _EVENT_HAVE_TIMERCMP 1
+
+/* Define if timerisset is defined in <sys/time.h> */
+#define _EVENT_HAVE_TIMERISSET 1
+
+/* Define to 1 if the system has the type `uint16_t'. */
+#define _EVENT_HAVE_UINT16_T 1
+
+/* Define to 1 if the system has the type `uint32_t'. */
+#define _EVENT_HAVE_UINT32_T 1
+
+/* Define to 1 if the system has the type `uint64_t'. */
+#define _EVENT_HAVE_UINT64_T 1
+
+/* Define to 1 if the system has the type `uint8_t'. */
+#define _EVENT_HAVE_UINT8_T 1
+
+/* Define to 1 if you have the <unistd.h> header file. */
+#define _EVENT_HAVE_UNISTD_H 1
+
+/* Define to 1 if you have the `vasprintf' function. */
+#define _EVENT_HAVE_VASPRINTF 1
+
+/* Define if kqueue works correctly with pipes */
+#define HAVE_WORKING_KQUEUE 1
+#define _EVENT_HAVE_WORKING_KQUEUE 1
+
+/* Name of package */
+#define _EVENT_PACKAGE "libevent"
+
+/* Define to the address where bug reports for this package should be sent. */
+#define _EVENT_PACKAGE_BUGREPORT ""
+
+/* Define to the full name of this package. */
+#define _EVENT_PACKAGE_NAME ""
+
+/* Define to the full name and version of this package. */
+#define _EVENT_PACKAGE_STRING ""
+
+/* Define to the one symbol short name of this package. */
+#define _EVENT_PACKAGE_TARNAME ""
+
+/* Define to the version of this package. */
+#define _EVENT_PACKAGE_VERSION ""
+
+/* Define to 1 if you have the ANSI C header files. */
+#define _EVENT_STDC_HEADERS 1
+
+/* Define to 1 if you can safely include both <sys/time.h> and <time.h>. */
+#define _EVENT_TIME_WITH_SYS_TIME 1
+
+/* Version number of package */
+#define _EVENT_VERSION "1.4.13-stable"
+
+/* Define to appropriate substitue if compiler doesnt have __func__ */
+/* #undef _EVENT___func__ */
+
+/* Define to empty if `const' does not conform to ANSI C. */
+/* #undef _EVENT_const */
+
+/* Define to `__inline__' or `__inline' if that's what the C compiler
+   calls it, or to nothing if 'inline' is not supported under any name.  */
+#ifndef _EVENT___cplusplus
+/* #undef _EVENT_inline */
+#endif
+
+/* Define to `int' if <sys/types.h> does not define. */
+/* #undef _EVENT_pid_t */
+
+/* Define to `unsigned int' if <sys/types.h> does not define. */
+/* #undef _EVENT_size_t */
+
+/* Define to unsigned int if you dont have it */
+/* #undef _EVENT_socklen_t */
+
+#endif  // _LIBEVENT_STARBOARD_H_
diff --git a/src/third_party/mozjs/mfbt/Assertions.h b/src/third_party/mozjs/mfbt/Assertions.h
index a520892..5e3f391 100644
--- a/src/third_party/mozjs/mfbt/Assertions.h
+++ b/src/third_party/mozjs/mfbt/Assertions.h
@@ -154,7 +154,7 @@
  * in release builds as well as debug builds.  But if the failure is one that
  * should be debugged and fixed, MOZ_ASSERT is generally preferable.
  */
-#if defined(STARBOARD) && (SB_API_VERSION >= 4)
+#if defined(STARBOARD)
 #  define MOZ_CRASH() \
      do { \
        *((volatile int*) NULL) = 123; \
diff --git a/src/third_party/openssl/openssl/crypto/evp/e_aes.c b/src/third_party/openssl/openssl/crypto/evp/e_aes.c
index fb0d026..96b6934 100644
--- a/src/third_party/openssl/openssl/crypto/evp/e_aes.c
+++ b/src/third_party/openssl/openssl/crypto/evp/e_aes.c
@@ -112,7 +112,7 @@
 
 #  define MAXBITCHUNK     ((size_t)1<<(sizeof(size_t)*8-4))
 
-#if defined(OPENSSL_SYS_STARBOARD) && SB_API_VERSION >= 4
+#if defined(OPENSSL_SYS_STARBOARD)
 static bool sb_translate_mode(int flags, SbCryptographyBlockCipherMode *mode) {
   switch (flags & EVP_CIPH_MODE) {
     case EVP_CIPH_CBC_MODE:
@@ -338,7 +338,7 @@
     return 1;                                \
   }
 
-#else  // defined(OPENSSL_SYS_STARBOARD) && SB_API_VERSION >= 4
+#else  // defined(OPENSSL_SYS_STARBOARD)
 
 static inline bool sb_has_stream(EVP_CIPHER_CTX *context) {
   return false;
@@ -409,7 +409,7 @@
 
 #define SB_TRY_CIPHER(context, out, in, len)
 
-#endif  // defined(OPENSSL_SYS_STARBOARD) && SB_API_VERSION >= 4
+#endif  // defined(OPENSSL_SYS_STARBOARD)
 
 #  ifdef VPAES_ASM
 int vpaes_set_encrypt_key(const unsigned char *userKey, int bits,
@@ -963,11 +963,11 @@
 static int aes_gcm_cleanup(EVP_CIPHER_CTX *c)
 {
     EVP_AES_GCM_CTX *gctx = c->cipher_data;
-#if defined(OPENSSL_SYS_STARBOARD) && SB_API_VERSION >= 4
+#if defined(OPENSSL_SYS_STARBOARD)
     SbCryptographyDestroyTransformer(gctx->gcm.gcm_transformer);
     SbCryptographyDestroyTransformer(gctx->gcm.ctr_transformer);
     SbCryptographyDestroyTransformer(gctx->gcm.ecb_transformer);
-#endif  // defined(OPENSSL_SYS_STARBOARD) && SB_API_VERSION >= 4
+#endif  // defined(OPENSSL_SYS_STARBOARD)
     OPENSSL_cleanse(&gctx->gcm, sizeof(gctx->gcm));
     if (gctx->iv != c->iv)
         OPENSSL_free(gctx->iv);
@@ -1003,7 +1003,7 @@
         gctx->taglen = -1;
         gctx->iv_gen = 0;
         gctx->tls_aad_len = -1;
-#if defined(OPENSSL_SYS_STARBOARD) && SB_API_VERSION >= 4
+#if defined(OPENSSL_SYS_STARBOARD)
         gctx->ctr = NULL;
         gctx->gcm.gcm_transformer = kSbCryptographyInvalidTransformer;
         gctx->gcm.ctr_transformer = kSbCryptographyInvalidTransformer;
diff --git a/src/third_party/openssl/openssl/crypto/evp/evp.h b/src/third_party/openssl/openssl/crypto/evp/evp.h
index 6e50fa7..7f03387 100644
--- a/src/third_party/openssl/openssl/crypto/evp/evp.h
+++ b/src/third_party/openssl/openssl/crypto/evp/evp.h
@@ -436,7 +436,7 @@
     int buf_len;                /* number we have left */
     unsigned char oiv[EVP_MAX_IV_LENGTH]; /* original iv */
     unsigned char iv[EVP_MAX_IV_LENGTH]; /* working iv */
-#if defined(OPENSSL_SYS_STARBOARD) && SB_API_VERSION >= 4
+#if defined(OPENSSL_SYS_STARBOARD)
     // A handle to a transformer for the full stream cipher. Intermediate state
     // is kept internally to the transformer, so fields like |iv| and |num| are
     // not used.
@@ -444,7 +444,7 @@
     // This is not used for GCM. The GCM transformer is inside the GCM context,
     // which is stored as the |cipher_data| for the GCM cipher.
     SbCryptographyTransformer stream_transformer;
-#endif  // defined(STARBOARD) && SB_API_VERSION >= 4
+#endif  // defined(STARBOARD)
     unsigned char buf[EVP_MAX_BLOCK_LENGTH]; /* saved partial block */
     int num;                    /* used by cfb/ofb/ctr mode */
     void *app_data;             /* application stuff */
diff --git a/src/third_party/openssl/openssl/crypto/evp/evp_enc.c b/src/third_party/openssl/openssl/crypto/evp/evp_enc.c
index 3021e52..8e85db5 100644
--- a/src/third_party/openssl/openssl/crypto/evp/evp_enc.c
+++ b/src/third_party/openssl/openssl/crypto/evp/evp_enc.c
@@ -548,9 +548,9 @@
 
 int EVP_CIPHER_CTX_cleanup(EVP_CIPHER_CTX *c)
 {
-#if defined(OPENSSL_SYS_STARBOARD) && SB_API_VERSION >= 4
+#if defined(OPENSSL_SYS_STARBOARD)
     SbCryptographyDestroyTransformer(c->stream_transformer);
-#endif  // defined(OPENSSL_SYS_STARBOARD) && SB_API_VERSION >= 4
+#endif  // defined(OPENSSL_SYS_STARBOARD)
 #ifndef OPENSSL_FIPS
     if (c->cipher != NULL) {
         if (c->cipher->cleanup && !c->cipher->cleanup(c))
diff --git a/src/third_party/openssl/openssl/crypto/modes/gcm128.c b/src/third_party/openssl/openssl/crypto/modes/gcm128.c
index 1df77da..18433b1 100644
--- a/src/third_party/openssl/openssl/crypto/modes/gcm128.c
+++ b/src/third_party/openssl/openssl/crypto/modes/gcm128.c
@@ -87,7 +87,7 @@
 } while(0)
 
 
-#if defined(OPENSSL_SYS_STARBOARD) && SB_API_VERSION >= 4
+#if defined(OPENSSL_SYS_STARBOARD)
 static const int kBlockSizeBits = 128;
 static const int kBlockSizeBytes = 16;
 static inline void sb_block128(GCM128_CONTEXT *ctx,
@@ -120,7 +120,7 @@
     (*ctr)(in, out, blocks, key, ivec);
   }
 }
-#else  // defined(OPENSSL_SYS_STARBOARD) && SB_API_VERSION >= 4
+#else  // defined(OPENSSL_SYS_STARBOARD)
 static inline void sb_block128(GCM128_CONTEXT *ctx,
                                block128_f block,
                                const unsigned char in[16],
@@ -140,7 +140,7 @@
   SB_UNREFERENCED_PARAMETER(ctx);
   (*ctr)(in, out, blocks, key, ivec);
 }
-#endif  // defined(OPENSSL_SYS_STARBOARD) && SB_API_VERSION >= 4
+#endif  // defined(OPENSSL_SYS_STARBOARD)
 
 /*-
  * Even though permitted values for TABLE_BITS are 8, 4 and 1, it should
diff --git a/src/third_party/openssl/openssl/crypto/modes/modes_lcl.h b/src/third_party/openssl/openssl/crypto/modes/modes_lcl.h
index db4bf70..421b0d8 100644
--- a/src/third_party/openssl/openssl/crypto/modes/modes_lcl.h
+++ b/src/third_party/openssl/openssl/crypto/modes/modes_lcl.h
@@ -121,7 +121,7 @@
     block128_f block;
     void *key;
 
-#if defined(OPENSSL_SYS_STARBOARD) && SB_API_VERSION >= 4
+#if defined(OPENSSL_SYS_STARBOARD)
     // Whether in decrypt or encrypt mode.
     int encrypt;
 
@@ -145,7 +145,7 @@
     // mode. If valid, then this is used instead of any other state in the
     // context.
     SbCryptographyTransformer gcm_transformer;
-#endif  // defined(OPENSSL_SYS_STARBOARD) && SB_API_VERSION >= 4
+#endif  // defined(OPENSSL_SYS_STARBOARD)
 };
 
 struct xts128_context {