diff --git a/DEPS b/DEPS
index 09df101..c3ba0b6 100644
--- a/DEPS
+++ b/DEPS
@@ -299,15 +299,15 @@
   # Three lines of non-changing comments so that
   # the commit queue can handle CLs rolling Skia
   # and whatever else without interference from each other.
-  'skia_revision': '8943ec85c41cc1dd976785f2d29f712952fc2e92',
+  'skia_revision': 'b368746d696a72bff291dddf7ab1492f5d72267f',
   # Three lines of non-changing comments so that
   # the commit queue can handle CLs rolling V8
   # and whatever else without interference from each other.
-  'v8_revision': '2227d8bbb76f71e31edca694815522149547c305',
+  'v8_revision': '6eadb14bb7a7908c7c0b86a0381260430b6c81fc',
   # Three lines of non-changing comments so that
   # the commit queue can handle CLs rolling ANGLE
   # and whatever else without interference from each other.
-  'angle_revision': '2072aea9998c4170bff1811a5013605b99c66547',
+  'angle_revision': '0f3aaebf742f0ac16360a1191d75f99843d1b255',
   # Three lines of non-changing comments so that
   # the commit queue can handle CLs rolling SwiftShader
   # and whatever else without interference from each other.
@@ -434,11 +434,11 @@
   # Three lines of non-changing comments so that
   # the commit queue can handle CLs rolling libavif
   # and whatever else without interference from each other.
-  'libavif_revision': 'bfb84ad6f9791be5b1d2d68e43bf1c79fc17b06c',
+  'libavif_revision': '96b8450ca5e5acf173645aafc5c5e91959dfe0c1',
   # Three lines of non-changing comments so that
   # the commit queue can handle CLs rolling nearby
   # and whatever else without interference from each other.
-  'nearby_revision': '73c1fba668a86dd003fa2ac6af86ab7d89c09908',
+  'nearby_revision': 'b3f85342261db10bbeb2e5a07c7e54b616309b9e',
   # Three lines of non-changing comments so that
   # the commit queue can handle CLs rolling securemessage
   # and whatever else without interference from each other.
@@ -478,7 +478,7 @@
 
   # If you change this, also update the libc++ revision in
   # //buildtools/deps_revisions.gni.
-  'libcxx_revision':       '19ffb9c00636bcabb6c6ce76ccaa4370583d3649',
+  'libcxx_revision':       '2fc3d704672fbd3e85fad8492d39e02d49412891',
 
   # GN CIPD package version.
   'gn_version': 'git_revision:5e19d2fb166fbd4f6f32147fbb2f497091a54ad8',
@@ -1205,7 +1205,7 @@
   },
 
   'src/third_party/depot_tools':
-    Var('chromium_git') + '/chromium/tools/depot_tools.git' + '@' + 'e1c8efebe0a3cce42ca46d6057b6d4bd909ad203',
+    Var('chromium_git') + '/chromium/tools/depot_tools.git' + '@' + '18359bb9a19ee9b31a33fae83aa49ced770a9b81',
 
   'src/third_party/devtools-frontend/src':
     Var('chromium_git') + '/devtools/devtools-frontend' + '@' + Var('devtools_frontend_revision'),
@@ -1640,7 +1640,7 @@
   },
 
   'src/third_party/perfetto':
-    Var('android_git') + '/platform/external/perfetto.git' + '@' + '165c172f3a6e30aa98495eaef67d5042cb4133a8',
+    Var('android_git') + '/platform/external/perfetto.git' + '@' + 'aa7e493dcf9bf77707735b98723b916e08561bbe',
 
   'src/third_party/perl': {
       'url': Var('chromium_git') + '/chromium/deps/perl.git' + '@' + '6f3e5028eb65d0b4c5fdd792106ac4c84eee1eb3',
@@ -1822,10 +1822,10 @@
     Var('chromium_git') + '/external/khronosgroup/webgl.git' + '@' + 'd1b65aa5a88f6efd900604dfcda840154e9f16e2',
 
   'src/third_party/webgpu-cts/src':
-    Var('chromium_git') + '/external/github.com/gpuweb/cts.git' + '@' + 'b99b81d6c5f615447fbdb469fa720808faf998b1',
+    Var('chromium_git') + '/external/github.com/gpuweb/cts.git' + '@' + '216c8e78167b6643afb2dbb170f1501471971112',
 
   'src/third_party/webrtc':
-    Var('webrtc_git') + '/src.git' + '@' + '597a2ba41aaa0b7ce09c8e0db94060cd6af7a52d',
+    Var('webrtc_git') + '/src.git' + '@' + '919b79b7efc34a059cafef7e0901c467b230d69a',
 
   # Wuffs' canonical repository is at github.com/google/wuffs, but we use
   # Skia's mirror of Wuffs, the same as in upstream Skia's DEPS file.
@@ -1895,7 +1895,7 @@
     Var('chromium_git') + '/v8/v8.git' + '@' +  Var('v8_revision'),
 
   'src-internal': {
-    'url': 'https://chrome-internal.googlesource.com/chrome/src-internal.git@5232f9de253ba0a63dfdd8c7965d59fb4cb2b315',
+    'url': 'https://chrome-internal.googlesource.com/chrome/src-internal.git@94562e58e6e22f0f8b5c1dd89ab590f94494bf85',
     'condition': 'checkout_src_internal',
   },
 
diff --git a/android_webview/java/src/org/chromium/android_webview/common/ProductionSupportedFlagList.java b/android_webview/java/src/org/chromium/android_webview/common/ProductionSupportedFlagList.java
index 24dee108..fbc78d2cc 100644
--- a/android_webview/java/src/org/chromium/android_webview/common/ProductionSupportedFlagList.java
+++ b/android_webview/java/src/org/chromium/android_webview/common/ProductionSupportedFlagList.java
@@ -189,6 +189,9 @@
             Flag.baseFeature(AutofillFeatures.AUTOFILL_SERVER_BEHAVIORS,
                     "When enabled, Autofill will request experimental "
                             + "predictions from the Autofill API."),
+            Flag.baseFeature(AutofillFeatures.AUTOFILL_SUPPORT_POOR_MANS_PLACEHOLDER,
+                    "When enabled, Autofill will infer labels from artificial placeholders, "
+                            + "placed on top of input fields using CSS."),
             Flag.baseFeature(FeatureConstants.KEYBOARD_ACCESSORY_PAYMENT_VIRTUAL_CARD_FEATURE,
                     "When enabled, merchant bound virtual cards will be offered in the keyboard "
                             + "accessory."),
diff --git a/android_webview/javatests/src/org/chromium/android_webview/test/AcceptLanguageTest.java b/android_webview/javatests/src/org/chromium/android_webview/test/AcceptLanguageTest.java
index def3b07..0ad75e76 100644
--- a/android_webview/javatests/src/org/chromium/android_webview/test/AcceptLanguageTest.java
+++ b/android_webview/javatests/src/org/chromium/android_webview/test/AcceptLanguageTest.java
@@ -4,11 +4,9 @@
 
 package org.chromium.android_webview.test;
 
-import android.os.Build;
 import android.os.LocaleList;
 import android.support.test.InstrumentationRegistry;
 
-import androidx.annotation.RequiresApi;
 import androidx.test.filters.SmallTest;
 
 import org.junit.After;
@@ -21,7 +19,6 @@
 import org.chromium.android_webview.AwContents;
 import org.chromium.android_webview.test.util.JSUtils;
 import org.chromium.base.test.util.Feature;
-import org.chromium.base.test.util.MinAndroidSdkLevel;
 import org.chromium.net.test.EmbeddedTestServer;
 
 import java.util.Locale;
@@ -80,20 +77,12 @@
 
     private boolean isSingleLocale(String lang, String country) {
         String languageTag = String.format("%s-%s", lang, country);
-        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
-            // In N+, multiple locales can be set.
-            return languageTag.equals(LocaleList.getDefault().toLanguageTags());
-        } else {
-            return languageTag.equals(Locale.getDefault().toLanguageTag());
-        }
+        // In N+, multiple locales can be set.
+        return languageTag.equals(LocaleList.getDefault().toLanguageTags());
     }
 
     private void setSingleLocale(String lang, String country) {
-        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
-            LocaleList.setDefault(new LocaleList(new Locale(lang, country)));
-        } else {
-            Locale.setDefault(new Locale(lang, country));
-        }
+        LocaleList.setDefault(new LocaleList(new Locale(lang, country)));
     }
 
     private void setLocaleForTesting(String lang, String country) {
@@ -154,8 +143,6 @@
      */
     @Test
     @SmallTest
-    @MinAndroidSdkLevel(Build.VERSION_CODES.N)
-    @RequiresApi(Build.VERSION_CODES.N)
     @Feature({"AndroidWebView"})
     public void testAcceptLanguagesWithenUS() throws Throwable {
         LocaleList.setDefault(new LocaleList(new Locale("ko", "KR")));
diff --git a/android_webview/javatests/src/org/chromium/android_webview/test/devui/DeveloperUiTest.java b/android_webview/javatests/src/org/chromium/android_webview/test/devui/DeveloperUiTest.java
index 24d7953..0abc802 100644
--- a/android_webview/javatests/src/org/chromium/android_webview/test/devui/DeveloperUiTest.java
+++ b/android_webview/javatests/src/org/chromium/android_webview/test/devui/DeveloperUiTest.java
@@ -34,7 +34,6 @@
 import android.content.Context;
 import android.content.Intent;
 import android.content.SharedPreferences;
-import android.os.Build;
 import android.provider.Settings;
 import android.support.test.InstrumentationRegistry;
 import android.view.View;
@@ -46,7 +45,6 @@
 
 import org.hamcrest.Matcher;
 import org.junit.After;
-import org.junit.Assume;
 import org.junit.Before;
 import org.junit.Rule;
 import org.junit.Test;
@@ -178,9 +176,6 @@
     @MediumTest
     @Feature({"AndroidWebView"})
     public void testMenuOptions_switchProvider_shownOnNougat() throws Throwable {
-        Assume.assumeTrue("This test verifies behavior introduced in Nougat and above",
-                Build.VERSION.SDK_INT >= Build.VERSION_CODES.N);
-
         launchHomeFragment();
 
         openOptionsMenu();
@@ -192,19 +187,6 @@
     @Test
     @MediumTest
     @Feature({"AndroidWebView"})
-    public void testMenuOptions_switchProvider_notShown() throws Throwable {
-        Assume.assumeTrue("This test verifies pre-Nougat behavior",
-                Build.VERSION.SDK_INT < Build.VERSION_CODES.N);
-
-        launchHomeFragment();
-
-        openOptionsMenu();
-        onView(withId(R.id.options_menu_switch_provider)).check(doesNotExist());
-    }
-
-    @Test
-    @MediumTest
-    @Feature({"AndroidWebView"})
     public void testMenuOptions_reportBug() throws Throwable {
         launchHomeFragment();
 
diff --git a/android_webview/javatests/src/org/chromium/android_webview/test/devui/HomeFragmentTest.java b/android_webview/javatests/src/org/chromium/android_webview/test/devui/HomeFragmentTest.java
index 52c7602..20f740c 100644
--- a/android_webview/javatests/src/org/chromium/android_webview/test/devui/HomeFragmentTest.java
+++ b/android_webview/javatests/src/org/chromium/android_webview/test/devui/HomeFragmentTest.java
@@ -40,7 +40,6 @@
 
 import org.junit.After;
 import org.junit.Assert;
-import org.junit.Assume;
 import org.junit.Before;
 import org.junit.Rule;
 import org.junit.Test;
@@ -242,9 +241,6 @@
     @MediumTest
     @Feature({"AndroidWebView"})
     public void testDifferentWebViewPackageError_bannerMessage_postNougat() throws Throwable {
-        Assume.assumeTrue("This test verifies behavior introduced in Nougat and above",
-                Build.VERSION.SDK_INT >= Build.VERSION_CODES.N);
-
         Context context = InstrumentationRegistry.getTargetContext();
         // Inject a dummy PackageInfo as the current WebView package to make sure it will always be
         // different from the test's app package.
@@ -271,9 +267,6 @@
     @Feature({"AndroidWebView"})
     // Test the dialog shown when the WebView package error message is clicked.
     public void testDifferentWebViewPackageError_dialog_postNougat() throws Throwable {
-        Assume.assumeTrue("This test verifies behavior introduced in Nougat and above",
-                Build.VERSION.SDK_INT >= Build.VERSION_CODES.N);
-
         Context context = InstrumentationRegistry.getTargetContext();
         // Inject a dummy PackageInfo as the current WebView package to make sure it will always be
         // different from the test's app package.
@@ -298,65 +291,4 @@
                 .perform(click());
         intended(IntentMatchers.hasAction(Settings.ACTION_WEBVIEW_SETTINGS));
     }
-
-    @Test
-    @MediumTest
-    @Feature({"AndroidWebView"})
-    // Test that error message is shown when system's WebView provider package is different from dev
-    // UI's on a preNougat android versions (where WebView provider can't be changed).
-    public void testDifferentWebViewPackageError_bannerMessage_preNougat() throws Throwable {
-        Assume.assumeTrue("This test verifies pre-Nougat behavior",
-                Build.VERSION.SDK_INT < Build.VERSION_CODES.N);
-
-        Context context = InstrumentationRegistry.getTargetContext();
-        // Inject a dummy PackageInfo as the current WebView package to make sure it will always be
-        // different from the test's app package.
-        WebViewPackageHelper.setCurrentWebViewPackageForTesting(FAKE_WEBVIEW_PACKAGE);
-        launchHomeFragment();
-
-        String expectedErrorMessage = String.format(Locale.US,
-                WebViewPackageError.DIFFERENT_WEBVIEW_PROVIDER_ERROR_MESSAGE,
-                WebViewPackageHelper.loadLabel(context));
-        ViewUtils.waitForView(withId(R.id.main_error_view));
-        onView(withId(R.id.main_error_view)).check(matches(isDisplayed()));
-        onView(withId(R.id.error_text)).check(matches(withText(expectedErrorMessage)));
-        // Since the current provider is set to a fake package not an actual installed WebView
-        // provider, the UI shouldn't offer opening current WebView provider dev UI. It should not
-        // offer to change system WebView provider because this is not supported on pre-Nougat
-        // android versions.
-        onView(withId(R.id.action_button)).check(matches(not(isDisplayed())));
-    }
-
-    @Test
-    @MediumTest
-    @Feature({"AndroidWebView"})
-    // Test the dialog shown when the WebView package error message is clicked (where WebView
-    // provider can't be changed).
-    public void testDifferentWebViewPackageError_dialog_preNougat() throws Throwable {
-        Assume.assumeTrue("This test verifies pre-Nougat behavior",
-                Build.VERSION.SDK_INT < Build.VERSION_CODES.N);
-
-        Context context = InstrumentationRegistry.getTargetContext();
-        // Inject a dummy PackageInfo as the current WebView package to make sure it will always be
-        // different from the test's app package.
-        WebViewPackageHelper.setCurrentWebViewPackageForTesting(FAKE_WEBVIEW_PACKAGE);
-        launchHomeFragment();
-
-        String dialogExpectedMessage = String.format(Locale.US,
-                WebViewPackageError.DIFFERENT_WEBVIEW_PROVIDER_DIALOG_MESSAGE,
-                WebViewPackageHelper.loadLabel(context));
-        ViewUtils.waitForView(withId(R.id.main_error_view));
-        onView(withId(R.id.main_error_view)).perform(click());
-        ViewUtils.waitForView(withText(dialogExpectedMessage));
-        onView(withText(dialogExpectedMessage)).check(matches(isDisplayed()));
-        // Since the current provider is set to a fake package not an actual installed WebView
-        // provider, the UI shouldn't offer opening current WebView provider dev UI. It should not
-        // offer to change system WebView provider because this is not supported on pre-Nougat
-        // android versions.
-        //
-        // There should be no buttons in the Dialog.
-        onView(withId(android.R.id.button1)).check(matches(not(isDisplayed())));
-        onView(withId(android.R.id.button2)).check(matches(not(isDisplayed())));
-        onView(withId(android.R.id.button3)).check(matches(not(isDisplayed())));
-    }
 }
diff --git a/ash/ambient/util/ambient_util.h b/ash/ambient/util/ambient_util.h
index 054a171a..9292674 100644
--- a/ash/ambient/util/ambient_util.h
+++ b/ash/ambient/util/ambient_util.h
@@ -33,10 +33,12 @@
 // Ambient mode uses non-standard colors for some text and the media icon, so
 // provides a wrapper for |AshColorProvider::GetContentLayerColor|. This is
 // currently only supported for primary and secondary text and icons.
+// TODO(b/262012604) rework to use ui::ColorProvider.
 ASH_EXPORT SkColor
 GetContentLayerColor(AshColorProvider::ContentLayerType content_layer_type,
                      bool dark_mode_enable);
 // Version of the above that uses AshColorProvider::IsDarkModeEnabled().
+// TODO(b/262012604) rework to use ui::ColorProvider.
 ASH_EXPORT SkColor
 GetContentLayerColor(AshColorProvider::ContentLayerType content_layer_type);
 
diff --git a/ash/app_list/views/app_list_view_pixeltest.cc b/ash/app_list/views/app_list_view_pixeltest.cc
index 2dec915..103efa2 100644
--- a/ash/app_list/views/app_list_view_pixeltest.cc
+++ b/ash/app_list/views/app_list_view_pixeltest.cc
@@ -83,6 +83,8 @@
   SetUpAnswerCardResult(results, 1, 1);
   test_helper->GetProductivityLauncherSearchView()
       ->OnSearchResultContainerResultsChanged();
+  // OnSearchResultContainerResultsChanged will schedule show animations().
+  base::RunLoop().RunUntilIdle();
 
   HideCursor();
   EXPECT_TRUE(GetPixelDiffer()->CompareUiComponentsOnPrimaryScreen(
diff --git a/ash/metrics/login_unlock_throughput_recorder.cc b/ash/metrics/login_unlock_throughput_recorder.cc
index 9bc43c8..1fa227ace 100644
--- a/ash/metrics/login_unlock_throughput_recorder.cc
+++ b/ash/metrics/login_unlock_throughput_recorder.cc
@@ -28,6 +28,7 @@
 #include "ui/compositor/layer.h"
 #include "ui/compositor/total_animation_throughput_reporter.h"
 #include "ui/views/animation/bounds_animator.h"
+#include "ui/views/animation/bounds_animator_observer.h"
 
 namespace ash {
 namespace {
@@ -351,6 +352,32 @@
   ReportLogin(start, data);
 }
 
+void LoginUnlockThroughputRecorder::OnArcOptedIn() {
+  arc_opt_in_time_ = base::TimeTicks::Now();
+}
+
+void LoginUnlockThroughputRecorder::OnArcAppListReady() {
+  if (arc_app_list_ready_reported_)
+    return;
+
+  // |Ash.ArcAppInitialAppsInstallDuration| histogram is only reported for
+  // the first user session after they opted into the ARC++.
+  // |arc_opt_in_time_| will only have value if user opted in into the ARC++
+  // in this session (in this browser instance).
+  if (arc_opt_in_time_.has_value()) {
+    const auto duration = base::TimeTicks::Now() - arc_opt_in_time_.value();
+    UmaHistogramCustomTimes("Ash.ArcAppInitialAppsInstallDuration", duration,
+                            base::Seconds(1) /* min */,
+                            base::Hours(1) /* max */, 100 /* buckets */);
+  }
+
+  arc_app_list_ready_reported_ = true;
+}
+
+bool LoginUnlockThroughputRecorder::NeedReportArcAppListReady() const {
+  return arc_opt_in_time_.has_value() && !arc_app_list_ready_reported_;
+}
+
 void LoginUnlockThroughputRecorder::ScheduleWaitForShelfAnimationEnd() {
   ShelfView* shelf_view =
       RootWindowController::ForWindow(
diff --git a/ash/metrics/login_unlock_throughput_recorder.h b/ash/metrics/login_unlock_throughput_recorder.h
index 7925dba..46d3fe5 100644
--- a/ash/metrics/login_unlock_throughput_recorder.h
+++ b/ash/metrics/login_unlock_throughput_recorder.h
@@ -17,6 +17,7 @@
 #include "base/time/time.h"
 #include "cc/metrics/frame_sequence_metrics.h"
 #include "chromeos/ash/components/login/login_state/login_state.h"
+#include "third_party/abseil-cpp/absl/types/optional.h"
 #include "ui/compositor/total_animation_throughput_reporter.h"
 
 namespace ui {
@@ -70,6 +71,16 @@
   // This is called when the list of shelf icons is updated.
   void UpdateShelfIconList(const ShelfModel* model);
 
+  // This is called when ARC++ becomes enabled.
+  void OnArcOptedIn();
+
+  // This is called when list of ARC++ apps is updated.
+  void OnArcAppListReady();
+
+  // This is true if we need to report Ash.ArcAppInitialAppsInstallDuration
+  // histogram in this session but it has not been reported yet.
+  bool NeedReportArcAppListReady() const;
+
   void ResetScopedThroughputReporterBlockerForTesting();
 
   const ui::TotalAnimationThroughputReporter*
@@ -137,6 +148,10 @@
 
   bool user_logged_in_ = false;
 
+  bool arc_app_list_ready_reported_ = false;
+
+  absl::optional<base::TimeTicks> arc_opt_in_time_;
+
   base::WeakPtr<ui::TotalAnimationThroughputReporter>
       login_animation_throughput_reporter_;
 
diff --git a/ash/projector/projector_controller_impl.cc b/ash/projector/projector_controller_impl.cc
index 51ef9c4..f670cc7c 100644
--- a/ash/projector/projector_controller_impl.cc
+++ b/ash/projector/projector_controller_impl.cc
@@ -507,7 +507,6 @@
   DCHECK(client_->GetSpeechRecognitionAvailability().IsAvailable());
 
   client_->StopSpeechRecognition();
-  is_speech_recognition_on_ = false;
 }
 
 void ProjectorControllerImpl::OnContainerFolderCreated(
diff --git a/ash/public/cpp/BUILD.gn b/ash/public/cpp/BUILD.gn
index e112fbd..8ca459cc 100644
--- a/ash/public/cpp/BUILD.gn
+++ b/ash/public/cpp/BUILD.gn
@@ -342,8 +342,8 @@
     "update_types.h",
     "view_shadow.cc",
     "view_shadow.h",
-    "views_text_services_context_menu_impl.cc",
-    "views_text_services_context_menu_impl.h",
+    "views_text_services_context_menu_ash.cc",
+    "views_text_services_context_menu_ash.h",
     "vm_camera_mic_constants.cc",
     "vm_camera_mic_constants.h",
     "wallpaper/google_photos_wallpaper_params.cc",
diff --git a/ash/public/cpp/views_text_services_context_menu_impl.cc b/ash/public/cpp/views_text_services_context_menu_ash.cc
similarity index 66%
rename from ash/public/cpp/views_text_services_context_menu_impl.cc
rename to ash/public/cpp/views_text_services_context_menu_ash.cc
index 74e87ca..fb262a8 100644
--- a/ash/public/cpp/views_text_services_context_menu_impl.cc
+++ b/ash/public/cpp/views_text_services_context_menu_ash.cc
@@ -2,7 +2,7 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
-#include "ash/public/cpp/views_text_services_context_menu_impl.h"
+#include "ash/public/cpp/views_text_services_context_menu_ash.h"
 
 #include "ash/public/cpp/clipboard_history_controller.h"
 #include "chromeos/crosapi/mojom/clipboard_history.mojom.h"
@@ -13,16 +13,25 @@
 
 namespace ash {
 
-ViewsTextServicesContextMenuImpl::ViewsTextServicesContextMenuImpl(
+ViewsTextServicesContextMenuAsh::ViewsTextServicesContextMenuAsh(
     ui::SimpleMenuModel* menu,
     views::Textfield* client)
     : views::ViewsTextServicesContextMenuBase(menu, client) {
-  AddClipboardHistoryMenuOption(menu);
+  // If the menu has a paste option, add a clipboard history option as well.
+  const absl::optional<size_t> paste_index =
+      menu->GetIndexOfCommandId(ui::TouchEditable::kPaste);
+
+  if (!paste_index.has_value())
+    return;
+
+  const size_t target_index = paste_index.value() + 1;
+  menu->InsertItemAt(target_index, IDS_APP_SHOW_CLIPBOARD_HISTORY,
+                     l10n_util::GetStringUTF16(IDS_APP_SHOW_CLIPBOARD_HISTORY));
 }
 
-ViewsTextServicesContextMenuImpl::~ViewsTextServicesContextMenuImpl() = default;
+ViewsTextServicesContextMenuAsh::~ViewsTextServicesContextMenuAsh() = default;
 
-bool ViewsTextServicesContextMenuImpl::GetAcceleratorForCommandId(
+bool ViewsTextServicesContextMenuAsh::GetAcceleratorForCommandId(
     int command_id,
     ui::Accelerator* accelerator) const {
   if (command_id == IDS_APP_SHOW_CLIPBOARD_HISTORY) {
@@ -34,35 +43,34 @@
       command_id, accelerator);
 }
 
-bool ViewsTextServicesContextMenuImpl::IsCommandIdChecked(
-    int command_id) const {
+bool ViewsTextServicesContextMenuAsh::IsCommandIdChecked(int command_id) const {
   if (command_id == IDS_APP_SHOW_CLIPBOARD_HISTORY)
     return true;
 
   return ViewsTextServicesContextMenuBase::IsCommandIdChecked(command_id);
 }
 
-bool ViewsTextServicesContextMenuImpl::IsCommandIdEnabled(
-    int command_id) const {
+bool ViewsTextServicesContextMenuAsh::IsCommandIdEnabled(int command_id) const {
   if (command_id == IDS_APP_SHOW_CLIPBOARD_HISTORY)
     return ClipboardHistoryController::Get()->CanShowMenu();
 
   return ViewsTextServicesContextMenuBase::IsCommandIdEnabled(command_id);
 }
 
-void ViewsTextServicesContextMenuImpl::ExecuteCommand(int command_id,
-                                                      int event_flags) {
+void ViewsTextServicesContextMenuAsh::ExecuteCommand(int command_id,
+                                                     int event_flags) {
   if (command_id == IDS_APP_SHOW_CLIPBOARD_HISTORY) {
     auto* clipboard_history_controller = ClipboardHistoryController::Get();
 
     // Calculate the menu source type from `event_flags`.
     ui::MenuSourceType source_type;
-    if (event_flags & ui::EF_LEFT_MOUSE_BUTTON)
+    if (event_flags & ui::EF_LEFT_MOUSE_BUTTON) {
       source_type = ui::MENU_SOURCE_MOUSE;
-    else if (event_flags & ui::EF_FROM_TOUCH)
+    } else if (event_flags & ui::EF_FROM_TOUCH) {
       source_type = ui::MENU_SOURCE_TOUCH;
-    else
+    } else {
       source_type = ui::MENU_SOURCE_KEYBOARD;
+    }
 
     clipboard_history_controller->ShowMenu(
         client()->GetCaretBounds(), source_type,
@@ -74,26 +82,11 @@
   ViewsTextServicesContextMenuBase::ExecuteCommand(command_id, event_flags);
 }
 
-bool ViewsTextServicesContextMenuImpl::SupportsCommand(int command_id) const {
+bool ViewsTextServicesContextMenuAsh::SupportsCommand(int command_id) const {
   if (command_id == IDS_APP_SHOW_CLIPBOARD_HISTORY)
     return true;
 
   return ViewsTextServicesContextMenuBase::SupportsCommand(command_id);
 }
 
-void ViewsTextServicesContextMenuImpl::AddClipboardHistoryMenuOption(
-    ui::SimpleMenuModel* menu) {
-  const absl::optional<size_t> index_of_paste =
-      menu->GetIndexOfCommandId(ui::TouchEditable::kPaste);
-
-  // Only add the clipboard history menu option when having the menu option
-  // for paste.
-  if (!index_of_paste.has_value())
-    return;
-
-  const size_t target_index = index_of_paste.value() + 1;
-  menu->InsertItemAt(target_index, IDS_APP_SHOW_CLIPBOARD_HISTORY,
-                     l10n_util::GetStringUTF16(IDS_APP_SHOW_CLIPBOARD_HISTORY));
-}
-
 }  // namespace ash
diff --git a/ash/public/cpp/views_text_services_context_menu_ash.h b/ash/public/cpp/views_text_services_context_menu_ash.h
new file mode 100644
index 0000000..b0b243fb
--- /dev/null
+++ b/ash/public/cpp/views_text_services_context_menu_ash.h
@@ -0,0 +1,42 @@
+// Copyright 2021 The Chromium Authors
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#ifndef ASH_PUBLIC_CPP_VIEWS_TEXT_SERVICES_CONTEXT_MENU_ASH_H_
+#define ASH_PUBLIC_CPP_VIEWS_TEXT_SERVICES_CONTEXT_MENU_ASH_H_
+
+#include "ash/public/cpp/ash_public_export.h"
+#include "ui/views/controls/views_text_services_context_menu_base.h"
+
+namespace views {
+class Textfield;
+}
+
+namespace ash {
+
+// This class implements support for adding and handling text service items in
+// ChromeOS system UI textfields and ash-chrome browser native textfields (i.e.,
+// the omnibox but not the WebUI embedded in the browser).
+class ASH_PUBLIC_EXPORT ViewsTextServicesContextMenuAsh
+    : public views::ViewsTextServicesContextMenuBase {
+ public:
+  ViewsTextServicesContextMenuAsh(ui::SimpleMenuModel* menu,
+                                  views::Textfield* client);
+  ViewsTextServicesContextMenuAsh(const ViewsTextServicesContextMenuAsh&) =
+      delete;
+  ViewsTextServicesContextMenuAsh& operator=(
+      const ViewsTextServicesContextMenuAsh&) = delete;
+  ~ViewsTextServicesContextMenuAsh() override;
+
+  // ViewsTextServicesContextMenuBase:
+  bool GetAcceleratorForCommandId(int command_id,
+                                  ui::Accelerator* accelerator) const override;
+  bool IsCommandIdChecked(int command_id) const override;
+  bool IsCommandIdEnabled(int command_id) const override;
+  void ExecuteCommand(int command_id, int event_flags) override;
+  bool SupportsCommand(int command_id) const override;
+};
+
+}  // namespace ash
+
+#endif  // ASH_PUBLIC_CPP_VIEWS_TEXT_SERVICES_CONTEXT_MENU_ASH_H_
diff --git a/ash/public/cpp/views_text_services_context_menu_impl.h b/ash/public/cpp/views_text_services_context_menu_impl.h
deleted file mode 100644
index 8856872..0000000
--- a/ash/public/cpp/views_text_services_context_menu_impl.h
+++ /dev/null
@@ -1,46 +0,0 @@
-// Copyright 2021 The Chromium Authors
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#ifndef ASH_PUBLIC_CPP_VIEWS_TEXT_SERVICES_CONTEXT_MENU_IMPL_H_
-#define ASH_PUBLIC_CPP_VIEWS_TEXT_SERVICES_CONTEXT_MENU_IMPL_H_
-
-#include "ash/public/cpp/ash_public_export.h"
-#include "ui/views/controls/views_text_services_context_menu_base.h"
-
-namespace views {
-class Textfield;
-}
-
-namespace ash {
-
-// This class supports the text context menu with the exclusive functions under
-// the CrOS environment.
-class ASH_PUBLIC_EXPORT ViewsTextServicesContextMenuImpl
-    : public views::ViewsTextServicesContextMenuBase {
- public:
-  ViewsTextServicesContextMenuImpl(ui::SimpleMenuModel* menu,
-                                   views::Textfield* client);
-  ViewsTextServicesContextMenuImpl(const ViewsTextServicesContextMenuImpl&) =
-      delete;
-  ViewsTextServicesContextMenuImpl& operator=(
-      const ViewsTextServicesContextMenuImpl&) = delete;
-  ~ViewsTextServicesContextMenuImpl() override;
-
-  // ViewsTextServicesContextMenuBase:
-  bool GetAcceleratorForCommandId(int command_id,
-                                  ui::Accelerator* accelerator) const override;
-  bool IsCommandIdChecked(int command_id) const override;
-  bool IsCommandIdEnabled(int command_id) const override;
-  void ExecuteCommand(int command_id, int event_flags) override;
-  bool SupportsCommand(int command_id) const override;
-
- private:
-  // Adds the menu option which shows the clipboard history menu after
-  // activation.
-  void AddClipboardHistoryMenuOption(ui::SimpleMenuModel* menu);
-};
-
-}  // namespace ash
-
-#endif  // ASH_PUBLIC_CPP_VIEWS_TEXT_SERVICES_CONTEXT_MENU_IMPL_H_
diff --git a/ash/quick_pair/pairing/fast_pair/fast_pair_pairer_impl.cc b/ash/quick_pair/pairing/fast_pair/fast_pair_pairer_impl.cc
index 8670cbcb..6cbe14a 100644
--- a/ash/quick_pair/pairing/fast_pair/fast_pair_pairer_impl.cc
+++ b/ash/quick_pair/pairing/fast_pair/fast_pair_pairer_impl.cc
@@ -131,7 +131,9 @@
   // If this is a v1 pairing, we pass off the responsibility to the Bluetooth
   // pairing dialog, and will listen for the
   // BluetoothAdapter::Observer::DevicePairedChanged event before firing the
-  // |paired_callback|.
+  // |paired_callback|. V1 devices only support the "initial pairing" protocol,
+  // not the "retroactive" or "subsequent" pairing protocols, so only
+  // "initial pairing" metrics are emitted to here.
   if (device_->version().value() == DeviceFastPairVersion::kV1) {
     RecordInitialSuccessFunnelFlow(
         FastPairInitialSuccessFunnelEvent::kV1DeviceDetected);
@@ -479,8 +481,11 @@
   // be.
   if (!ShouldBeEnabledForLoginStatus(
           Shell::Get()->session_controller()->login_status())) {
-    RecordInitialSuccessFunnelFlow(
-        FastPairInitialSuccessFunnelEvent::kGuestModeDetected);
+    if (device_->protocol == Protocol::kFastPairInitial) {
+      RecordInitialSuccessFunnelFlow(
+          FastPairInitialSuccessFunnelEvent::kGuestModeDetected);
+    }
+
     QP_LOG(VERBOSE) << __func__ << ": No logged in user to save account key to";
     std::move(pairing_procedure_complete_).Run(device_);
     return;
@@ -538,8 +543,12 @@
     QP_LOG(INFO) << __func__
                  << ": Device is already saved, skipping write account key. "
                     "Pairing procedure complete.";
-    RecordInitialSuccessFunnelFlow(
-        FastPairInitialSuccessFunnelEvent::kDeviceAlreadyAssociatedToAccount);
+
+    if (device_->protocol == Protocol::kFastPairInitial) {
+      RecordInitialSuccessFunnelFlow(
+          FastPairInitialSuccessFunnelEvent::kDeviceAlreadyAssociatedToAccount);
+    }
+
     std::move(pairing_procedure_complete_).Run(device_);
     return;
   }
@@ -558,8 +567,11 @@
   RAND_bytes(account_key.data(), account_key.size());
   account_key[0] = 0x04;
 
-  RecordInitialSuccessFunnelFlow(
-      FastPairInitialSuccessFunnelEvent::kPreparingToWriteAccountKey);
+  if (device_->protocol == Protocol::kFastPairInitial) {
+    RecordInitialSuccessFunnelFlow(
+        FastPairInitialSuccessFunnelEvent::kPreparingToWriteAccountKey);
+  }
+
   fast_pair_gatt_service_client_->WriteAccountKey(
       account_key, fast_pair_handshake_->fast_pair_data_encryptor(),
       base::BindOnce(&FastPairPairerImpl::OnWriteAccountKey,
@@ -600,8 +612,12 @@
   QP_LOG(INFO)
       << __func__
       << ": Account key written to device. Pairing procedure complete.";
-  RecordInitialSuccessFunnelFlow(
-      FastPairInitialSuccessFunnelEvent::kAccountKeyWritten);
+
+  if (device_->protocol == Protocol::kFastPairInitial) {
+    RecordInitialSuccessFunnelFlow(
+        FastPairInitialSuccessFunnelEvent::kAccountKeyWritten);
+  }
+
   std::move(pairing_procedure_complete_).Run(device_);
 }
 
diff --git a/ash/quick_pair/pairing/fast_pair/fast_pair_pairer_impl_unittest.cc b/ash/quick_pair/pairing/fast_pair/fast_pair_pairer_impl_unittest.cc
index 52e76b9..1ef93c520 100644
--- a/ash/quick_pair/pairing/fast_pair/fast_pair_pairer_impl_unittest.cc
+++ b/ash/quick_pair/pairing/fast_pair/fast_pair_pairer_impl_unittest.cc
@@ -2494,5 +2494,34 @@
   EXPECT_EQ(GetPairFailure(), PairFailure::kCreateBondTimeout);
 }
 
+TEST_F(FastPairPairerImplTest, RetroactiveNotLoggedToInitial) {
+  Login(user_manager::UserType::USER_TYPE_REGULAR);
+  fast_pair_repository_.SetOptInStatus(
+      nearby::fastpair::OptInStatus::STATUS_OPTED_OUT);
+  base::test::ScopedFeatureList feature_list;
+  feature_list.InitWithFeatures(
+      /*enabled_features=*/{features::kFastPairSavedDevices},
+      /*disabled_features=*/{features::kFastPairSavedDevicesStrictOptIn});
+  CreateMockDevice(DeviceFastPairVersion::kHigherThanV1,
+                   /*protocol=*/Protocol::kFastPairRetroactive);
+  // When pairing starts, if the classic address can't be resolved to
+  // a device then we pair via address.
+  SetGetDeviceNullptr();
+  CreatePairer();
+  fake_fast_pair_handshake_->InvokeCallback();
+  EXPECT_CALL(pairing_procedure_complete_, Run);
+  RunWriteAccountKeyCallback();
+  histogram_tester().ExpectTotalCount(
+      kWriteAccountKeyCharacteristicResultMetric, 1);
+  EXPECT_EQ(histogram_tester().GetBucketCount(
+                kInitialSuccessFunnelMetric,
+                FastPairInitialSuccessFunnelEvent::kPreparingToWriteAccountKey),
+            0);
+  EXPECT_EQ(histogram_tester().GetBucketCount(
+                kInitialSuccessFunnelMetric,
+                FastPairInitialSuccessFunnelEvent::kAccountKeyWritten),
+            0);
+}
+
 }  // namespace quick_pair
 }  // namespace ash
diff --git a/ash/quick_pair/pairing/retroactive_pairing_detector_impl.cc b/ash/quick_pair/pairing/retroactive_pairing_detector_impl.cc
index a0813d76..5c53335 100644
--- a/ash/quick_pair/pairing/retroactive_pairing_detector_impl.cc
+++ b/ash/quick_pair/pairing/retroactive_pairing_detector_impl.cc
@@ -127,20 +127,18 @@
 
 void RetroactivePairingDetectorImpl::OnDevicePaired(
     scoped_refptr<Device> device) {
-  // When a device is paired to via Fast Pair, we save the device's classic
-  // pairing address here so when we get the the BluetoothAdapter's
-  // |DevicePairedChanged| fired, we can determine if it was the one we already
-  // have paired to. The classic address is assigned to the Device during the
-  // initial Fast Pair pairing protocol during the key exchange, and if it
-  // doesn't exist, then it wasn't properly paired during initial Fast Pair
+  // The classic address is assigned to the Device during the
+  // initial Fast Pair pairing protocol and if it doesn't exist,
+  // then it wasn't properly paired during initial Fast Pair
   // pairing.
   if (!device->classic_address())
     return;
 
-  // Sometimes we might encounter the case where |DevicePairedChanged| fires
-  // before FastPair's |OnDevicePaired|, and if that is the case and a device
-  // has been inserted in the |potential_retroactive_addresses_|, we need
-  // to remove it.
+  // The Bluetooth Adapter system event `DevicePairedChanged` fires before
+  // Fast Pair's `OnDevicePaired`, and a Fast Pair pairing is expected to have
+  // both events. If a device is Fast Paired, it is already inserted in the
+  // |potential_retroactive_addresses_| in `DevicePairedChanged`; we need to
+  // remove it to prevent a false positive.
   if (base::Contains(potential_retroactive_addresses_,
                      device->classic_address().value())) {
     QP_LOG(VERBOSE)
@@ -151,9 +149,6 @@
     RemoveDeviceInformation(device->classic_address().value());
     return;
   }
-
-  QP_LOG(INFO) << __func__ << ": Storing Fast Pair device address";
-  fast_pair_addresses_.insert(device->classic_address().value());
 }
 
 void RetroactivePairingDetectorImpl::DevicePairedChanged(
@@ -166,17 +161,19 @@
   // This event fires whenever a device pairing has changed with the adapter.
   // If the |new_paired_status| is false, it means a device was unpaired with
   // the adapter, so we early return since it would not be a device to
-  // retroactively pair to. If the device that was paired to that fires this
-  // event is a device we just paired to with Fast Pair, then we early return
-  // since it also wouldn't be one to retroactively pair to. We want to only
-  // continue our check here if we have a newly paired device that was paired
-  // with classic Bluetooth pairing.
-  const std::string& classic_address = device->GetAddress();
-  if (!new_paired_status ||
-      base::Contains(fast_pair_addresses_, classic_address)) {
+  // retroactively pair to.
+  if (!new_paired_status) {
     return;
   }
 
+  // Both classic paired and Fast paired devices call this function, so we
+  // have to add the device to |potential_retroactive_addresses_|. We expect
+  // devices paired via Fast Pair to always call `OnDevicePaired` after calling
+  // this function, which will remove the device from
+  // |potential_retroactive_addresses_|.
+  const std::string& classic_address = device->GetAddress();
+  potential_retroactive_addresses_.insert(classic_address);
+
   // In order to confirm that this device is a retroactive pairing, we need to
   // first check if it has already been saved to the user's account. If it has
   // already been saved, we don't want to prompt the user to save a device
@@ -192,18 +189,12 @@
     bool is_device_saved_to_account) {
   if (is_device_saved_to_account) {
     QP_LOG(INFO) << __func__ << ": device already saved to user's account";
+    RemoveDeviceInformation(classic_address);
     return;
   }
 
   QP_LOG(VERBOSE) << __func__ << ": device = " << classic_address;
 
-  // The device pairing just changed, and this means that it was just
-  // classically paired. Because we have now verified that it not saved to the
-  // user's account, we can continue verifying this device for the retroactive
-  // pairing scenario by checking if the Message Stream contains the model id
-  // and ble address.
-  potential_retroactive_addresses_.insert(classic_address);
-
   // Attempt to retrieve a MessageStream instance immediately, if it was
   // already connected.
   MessageStream* message_stream =
diff --git a/ash/quick_pair/pairing/retroactive_pairing_detector_impl.h b/ash/quick_pair/pairing/retroactive_pairing_detector_impl.h
index 74212eb..67b038a 100644
--- a/ash/quick_pair/pairing/retroactive_pairing_detector_impl.h
+++ b/ash/quick_pair/pairing/retroactive_pairing_detector_impl.h
@@ -133,10 +133,6 @@
 
   void RemoveDeviceInformation(const std::string& device_address);
 
-  // The classic pairing addresses of Fast Pair devices that we have already
-  // paired to.
-  base::flat_set<std::string> fast_pair_addresses_;
-
   // The classic pairing addresses of potential Retroactive Pair supported
   // devices that are found in the adapter. We have to store them and wait for a
   // MessageStream instance to be created for the device in order to fully
diff --git a/ash/quick_pair/pairing/retroactive_pairing_detector_unittest.cc b/ash/quick_pair/pairing/retroactive_pairing_detector_unittest.cc
index ac04d4f1..2b2ff53 100644
--- a/ash/quick_pair/pairing/retroactive_pairing_detector_unittest.cc
+++ b/ash/quick_pair/pairing/retroactive_pairing_detector_unittest.cc
@@ -240,22 +240,6 @@
   std::unique_ptr<RetroactivePairingDetector> retroactive_pairing_detector_;
 };
 
-TEST_F(RetroactivePairingDetectorTest, DevicedPaired_FastPair) {
-  Login(user_manager::UserType::USER_TYPE_REGULAR);
-  fast_pair_repository_.SetOptInStatus(
-      nearby::fastpair::OptInStatus::STATUS_OPTED_IN);
-  base::RunLoop().RunUntilIdle();
-  CreateRetroactivePairingDetector();
-
-  EXPECT_FALSE(retroactive_pair_found_);
-
-  PairFastPairDeviceWithFastPair(kTestDeviceAddress);
-  PairFastPairDeviceWithClassicBluetooth(
-      /*new_paired_status=*/true, kTestDeviceAddress);
-
-  EXPECT_FALSE(retroactive_pair_found_);
-}
-
 TEST_F(RetroactivePairingDetectorTest,
        DevicedPaired_FastPair_BluetoothEventFiresFirst) {
   Login(user_manager::UserType::USER_TYPE_REGULAR);
@@ -273,6 +257,46 @@
   EXPECT_FALSE(retroactive_pair_found_);
 }
 
+// Regression test for b/261041950
+TEST_F(RetroactivePairingDetectorTest,
+       FastPairPairingEventCalledDuringBluetoothAdapterPairingEvent) {
+  Login(user_manager::UserType::USER_TYPE_REGULAR);
+  fast_pair_repository_.SetOptInStatus(
+      nearby::fastpair::OptInStatus::STATUS_OPTED_IN);
+  CreateRetroactivePairingDetector();
+
+  EXPECT_FALSE(retroactive_pair_found_);
+
+  SetMessageStream(kModelIdBleAddressBytes);
+
+  // Simulate the Bluetooth Adapter event firing, with the callback to
+  // `IsDeviceSavedToAccount` delayed.
+  fast_pair_repository_.SetIsDeviceSavedToAccountCallbackDelayed(
+      /*is_delayed=*/true);
+  PairFastPairDeviceWithClassicBluetooth(
+      /*new_paired_status=*/true, kTestDeviceAddress);
+
+  // Simulate the Fast Pair pairing event firing during the Bluetooth Adapter
+  // pairing event call stack. The Bluetooth Adapter system
+  // event response has not finished completing because of the delay set in
+  // `SetIsDeviceSavedToAccountCallbackDelayed`.
+  PairFastPairDeviceWithFastPair(kTestDeviceAddress);
+
+  // Trigger the callback to check the repository after the Fast Pair pairing
+  // event fires. This will conclude the BluetoothAdapter pairing event call
+  // stack.
+  fast_pair_repository_.TriggerIsDeviceSavedToAccountCallback();
+
+  // Simulate data being received via Message Stream for the device. It should
+  // not be detected since the Fast Pair event has been fired, removing it
+  // as a possible retroactive device.
+  fake_socket_->TriggerReceiveCallback();
+  NotifyMessageStreamConnected(kTestDeviceAddress);
+  base::RunLoop().RunUntilIdle();
+
+  EXPECT_FALSE(retroactive_pair_found_);
+}
+
 TEST_F(RetroactivePairingDetectorTest, DeviceUnpaired) {
   Login(user_manager::UserType::USER_TYPE_REGULAR);
   fast_pair_repository_.SetOptInStatus(
diff --git a/ash/quick_pair/repository/fake_fast_pair_repository.cc b/ash/quick_pair/repository/fake_fast_pair_repository.cc
index ca1d6bb..1d5e0e3 100644
--- a/ash/quick_pair/repository/fake_fast_pair_repository.cc
+++ b/ash/quick_pair/repository/fake_fast_pair_repository.cc
@@ -178,6 +178,11 @@
 void FakeFastPairRepository::IsDeviceSavedToAccount(
     const std::string& mac_address,
     IsDeviceSavedToAccountCallback callback) {
+  if (saved_to_account_callback_is_delayed_) {
+    saved_to_account_callback_ = std::move(callback);
+    return;
+  }
+
   if (base::Contains(saved_mac_addresses_, mac_address)) {
     std::move(callback).Run(true);
     return;
@@ -186,5 +191,9 @@
   std::move(callback).Run(false);
 }
 
+void FakeFastPairRepository::TriggerIsDeviceSavedToAccountCallback() {
+  std::move(saved_to_account_callback_).Run(false);
+}
+
 }  // namespace quick_pair
 }  // namespace ash
diff --git a/ash/quick_pair/repository/fake_fast_pair_repository.h b/ash/quick_pair/repository/fake_fast_pair_repository.h
index 8d042e6..e055d23 100644
--- a/ash/quick_pair/repository/fake_fast_pair_repository.h
+++ b/ash/quick_pair/repository/fake_fast_pair_repository.h
@@ -89,9 +89,19 @@
   void IsDeviceSavedToAccount(const std::string& mac_address,
                               IsDeviceSavedToAccountCallback callback) override;
 
+  // `SetIsDeviceSavedToAccountCallbackDelay` and
+  // `TriggerIsDeviceSavedToAccountCallback` are used together to control when
+  // the callback is triggered.
+  void TriggerIsDeviceSavedToAccountCallback();
+  void SetIsDeviceSavedToAccountCallbackDelayed(bool is_delayed) {
+    saved_to_account_callback_is_delayed_ = is_delayed;
+  }
+
  private:
   static void SetInstance(FastPairRepository* instance);
 
+  IsDeviceSavedToAccountCallback saved_to_account_callback_;
+  bool saved_to_account_callback_is_delayed_ = false;
   nearby::fastpair::OptInStatus status_ =
       nearby::fastpair::OptInStatus::STATUS_UNKNOWN;
   std::vector<nearby::fastpair::FastPairDevice> devices_;
diff --git a/ash/shelf/swipe_home_to_overview_controller_unittest.cc b/ash/shelf/swipe_home_to_overview_controller_unittest.cc
index 668179d..f81f9914 100644
--- a/ash/shelf/swipe_home_to_overview_controller_unittest.cc
+++ b/ash/shelf/swipe_home_to_overview_controller_unittest.cc
@@ -24,9 +24,11 @@
 #include "base/test/metrics/histogram_tester.h"
 #include "base/test/simple_test_tick_clock.h"
 #include "third_party/abseil-cpp/absl/types/optional.h"
+#include "ui/compositor/compositor.h"
 #include "ui/compositor/layer.h"
 #include "ui/compositor/layer_animator.h"
 #include "ui/compositor/scoped_animation_duration_scale_mode.h"
+#include "ui/compositor/test/layer_animation_stopped_waiter.h"
 #include "ui/compositor/test/test_utils.h"
 #include "ui/gfx/geometry/point_f.h"
 #include "ui/gfx/geometry/rect_f.h"
@@ -106,17 +108,12 @@
   }
 
   void WaitForHomeLauncherAnimationToFinish() {
-    auto* compositor =
-        Shell::GetPrimaryRootWindowController()->GetHost()->compositor();
-    // Wait until home launcher animation finishes.
-    while (GetAppListTestHelper()
-               ->GetAppListView()
-               ->GetWidget()
-               ->GetLayer()
-               ->GetAnimator()
-               ->is_animating()) {
-      EXPECT_TRUE(ui::WaitForNextFrameToBePresented(compositor));
-    }
+    ui::LayerAnimationStoppedWaiter animation_waiter;
+    ui::Layer* app_list_layer =
+        GetAppListTestHelper()->GetAppListView()->GetWidget()->GetLayer();
+    animation_waiter.Wait(app_list_layer);
+
+    ui::Compositor* compositor = app_list_layer->GetCompositor();
 
     // Ensure there is one more frame presented after animation finishes
     // to allow animation throughput data is passed from cc to ui.
diff --git a/ash/shell.cc b/ash/shell.cc
index dfdabe87..5a3baa3 100644
--- a/ash/shell.cc
+++ b/ash/shell.cc
@@ -100,7 +100,7 @@
 #include "ash/public/cpp/shell_window_ids.h"
 #include "ash/public/cpp/system_sounds_delegate.h"
 #include "ash/public/cpp/tab_cluster/tab_cluster_ui_controller.h"
-#include "ash/public/cpp/views_text_services_context_menu_impl.h"
+#include "ash/public/cpp/views_text_services_context_menu_ash.h"
 #include "ash/quick_pair/keyed_service/quick_pair_mediator.h"
 #include "ash/rgb_keyboard/rgb_keyboard_manager.h"
 #include "ash/root_window_controller.h"
@@ -1536,8 +1536,8 @@
       base::BindRepeating(
           [](ui::SimpleMenuModel* menu_model, views::Textfield* textfield)
               -> std::unique_ptr<views::ViewsTextServicesContextMenu> {
-            return std::make_unique<ViewsTextServicesContextMenuImpl>(
-                menu_model, textfield);
+            return std::make_unique<ViewsTextServicesContextMenuAsh>(menu_model,
+                                                                     textfield);
           }));
 
   for (auto& observer : shell_observers_)
diff --git a/ash/style/dark_light_mode_controller_impl.cc b/ash/style/dark_light_mode_controller_impl.cc
index 56a15eb..4b2c9b0 100644
--- a/ash/style/dark_light_mode_controller_impl.cc
+++ b/ash/style/dark_light_mode_controller_impl.cc
@@ -134,6 +134,7 @@
                                         !IsDarkModeEnabled());
   active_user_pref_service_->CommitPendingWrite();
   NotifyColorModeChanges();
+  SystemNudgeController::RecordNudgeAction(NudgeCatalogName::kDarkLightMode);
 
   // Updates showing logic of educational nudge on toggling the entry points of
   // dark/light mode.
diff --git a/ash/system/camera/autozoom_controller_impl.cc b/ash/system/camera/autozoom_controller_impl.cc
index dfdff6b..b3ace6a 100644
--- a/ash/system/camera/autozoom_controller_impl.cc
+++ b/ash/system/camera/autozoom_controller_impl.cc
@@ -66,6 +66,7 @@
   SetState(state_ == cros::mojom::CameraAutoFramingState::OFF
                ? cros::mojom::CameraAutoFramingState::ON_SINGLE
                : cros::mojom::CameraAutoFramingState::OFF);
+  SystemNudgeController::RecordNudgeAction(NudgeCatalogName::kAutozoom);
 }
 
 void AutozoomControllerImpl::AddObserver(AutozoomObserver* observer) {
diff --git a/ash/system/do_not_disturb_notification_controller.cc b/ash/system/do_not_disturb_notification_controller.cc
index fed5bde5..8a40391 100644
--- a/ash/system/do_not_disturb_notification_controller.cc
+++ b/ash/system/do_not_disturb_notification_controller.cc
@@ -22,7 +22,6 @@
 
 using message_center::MessageCenter;
 
-const char kDoNotDisturbNotificationId[] = "do_not_disturb";
 const char kDoNotDisturbNotifierId[] =
     "ash.do_not_disturb_notification_controller";
 
@@ -36,6 +35,10 @@
   MessageCenter::Get()->RemoveObserver(this);
 }
 
+// static
+const char DoNotDisturbNotificationController::kDoNotDisturbNotificationId[] =
+    "do_not_disturb";
+
 std::unique_ptr<message_center::Notification>
 DoNotDisturbNotificationController::CreateNotification() {
   message_center::RichNotificationData optional_fields;
diff --git a/ash/system/do_not_disturb_notification_controller.h b/ash/system/do_not_disturb_notification_controller.h
index d4e513f8..a52232a 100644
--- a/ash/system/do_not_disturb_notification_controller.h
+++ b/ash/system/do_not_disturb_notification_controller.h
@@ -30,6 +30,8 @@
 
   ~DoNotDisturbNotificationController() override;
 
+  static const char kDoNotDisturbNotificationId[];
+
   // message_center::MessageCenterObserver:
   void OnQuietModeChanged(bool in_quiet_mode) override;
 
diff --git a/ash/system/do_not_disturb_notification_controller_unittest.cc b/ash/system/do_not_disturb_notification_controller_unittest.cc
index 5a5b207..bb0051a 100644
--- a/ash/system/do_not_disturb_notification_controller_unittest.cc
+++ b/ash/system/do_not_disturb_notification_controller_unittest.cc
@@ -15,8 +15,6 @@
 
 using message_center::MessageCenter;
 
-const char kDoNotDisturbNotificationId[] = "do_not_disturb";
-
 }  // namespace
 
 class DoNotDisturbNotificationControllerTest
@@ -42,7 +40,7 @@
 
   bool IsDoNotDisturbNotificationPresent() {
     return MessageCenter::Get()->FindNotificationById(
-        kDoNotDisturbNotificationId);
+        DoNotDisturbNotificationController::kDoNotDisturbNotificationId);
   }
 
  private:
@@ -87,8 +85,8 @@
   ASSERT_TRUE(IsDoNotDisturbNotificationPresent());
 
   // Simulate a click on the notification's "Turn off" button.
-  auto* notification =
-      message_center->FindNotificationById(kDoNotDisturbNotificationId);
+  auto* notification = message_center->FindNotificationById(
+      DoNotDisturbNotificationController::kDoNotDisturbNotificationId);
   notification->delegate()->Click(0, absl::nullopt);
   EXPECT_FALSE(IsDoNotDisturbNotificationPresent());
   EXPECT_FALSE(message_center->IsQuietMode());
diff --git a/ash/system/message_center/session_state_notification_blocker.cc b/ash/system/message_center/session_state_notification_blocker.cc
index 9c5e130..eb87c26 100644
--- a/ash/system/message_center/session_state_notification_blocker.cc
+++ b/ash/system/message_center/session_state_notification_blocker.cc
@@ -6,12 +6,11 @@
 
 #include "ash/session/session_controller_impl.h"
 #include "ash/shell.h"
-#include "ash/system/message_center/ash_message_center_lock_screen_controller.h"
+#include "ash/system/do_not_disturb_notification_controller.h"
 #include "ash/system/power/battery_notification.h"
 #include "base/containers/contains.h"
 #include "ui/message_center/message_center.h"
 #include "ui/message_center/public/cpp/notification.h"
-#include "ui/message_center/public/cpp/notification_types.h"
 #include "ui/message_center/public/cpp/notifier_id.h"
 
 using session_manager::SessionState;
@@ -96,6 +95,15 @@
   if (Shell::Get()->session_controller()->IsRunningInAppMode())
     return false;
 
+  // Do not show the "Do not disturb" notification if there is no active
+  // session.
+  if (notification.id() ==
+          DoNotDisturbNotificationController::kDoNotDisturbNotificationId &&
+      Shell::Get()->session_controller()->GetSessionState() !=
+          SessionState::ACTIVE) {
+    return false;
+  }
+
   if (notification.id() == BatteryNotification::kNotificationId)
     return true;
 
diff --git a/ash/system/message_center/session_state_notification_blocker_unittest.cc b/ash/system/message_center/session_state_notification_blocker_unittest.cc
index 2c9ab2a..d65199dd 100644
--- a/ash/system/message_center/session_state_notification_blocker_unittest.cc
+++ b/ash/system/message_center/session_state_notification_blocker_unittest.cc
@@ -9,9 +9,9 @@
 #include "ash/constants/ash_features.h"
 #include "ash/constants/notifier_catalogs.h"
 #include "ash/session/test_session_controller_client.h"
+#include "ash/system/do_not_disturb_notification_controller.h"
 #include "ash/system/power/battery_notification.h"
 #include "ash/test/ash_test_base.h"
-#include "base/command_line.h"
 #include "base/strings/utf_string_conversions.h"
 #include "base/test/scoped_feature_list.h"
 #include "ui/message_center/message_center.h"
@@ -95,6 +95,19 @@
     return blocker_->ShouldShowNotificationAsPopup(notification);
   }
 
+  bool ShouldShowDoNotDisturbNotification() {
+    message_center::Notification notification(
+        message_center::NOTIFICATION_TYPE_SIMPLE,
+        DoNotDisturbNotificationController::kDoNotDisturbNotificationId,
+        u"chromeos-title", u"chromeos-message", ui::ImageModel(),
+        u"chromeos-source", GURL(),
+        message_center::NotifierId(
+            message_center::NotifierType::SYSTEM_COMPONENT, "test-notifier",
+            NotificationCatalogName::kDoNotDisturb),
+        message_center::RichNotificationData(), nullptr);
+    return blocker_->ShouldShowNotification(notification);
+  }
+
   void SetLockedState(bool locked) {
     GetSessionControllerClient()->SetSessionState(
         locked ? SessionState::LOCKED : SessionState::ACTIVE);
@@ -261,5 +274,27 @@
   SessionStateNotificationBlocker::SetUseLoginNotificationDelayForTest(false);
 }
 
+TEST_P(SessionStateNotificationBlockerTest, DoNotDisturbNotification) {
+  // OOBE.
+  GetSessionControllerClient()->SetSessionState(SessionState::OOBE);
+  EXPECT_FALSE(ShouldShowDoNotDisturbNotification());
+
+  // Login screen.
+  GetSessionControllerClient()->SetSessionState(SessionState::LOGIN_PRIMARY);
+  EXPECT_FALSE(ShouldShowDoNotDisturbNotification());
+
+  // Logged in as a normal user.
+  SimulateUserLogin("user@test.com");
+  EXPECT_TRUE(ShouldShowDoNotDisturbNotification());
+
+  // Lock.
+  SetLockedState(true);
+  EXPECT_FALSE(ShouldShowDoNotDisturbNotification());
+
+  // Unlock.
+  SetLockedState(false);
+  EXPECT_TRUE(ShouldShowDoNotDisturbNotification());
+}
+
 }  // namespace
 }  // namespace ash
diff --git a/ash/system/microphone_mute/microphone_mute_notification_controller.cc b/ash/system/microphone_mute/microphone_mute_notification_controller.cc
index 7a14527..f90baeb 100644
--- a/ash/system/microphone_mute/microphone_mute_notification_controller.cc
+++ b/ash/system/microphone_mute/microphone_mute_notification_controller.cc
@@ -9,6 +9,7 @@
 #include <vector>
 
 #include "ash/constants/notifier_catalogs.h"
+#include "ash/public/cpp/new_window_delegate.h"
 #include "ash/public/cpp/notification_utils.h"
 #include "ash/public/cpp/sensor_disabled_notification_delegate.h"
 #include "ash/shell.h"
@@ -21,10 +22,14 @@
 #include "components/vector_icons/vector_icons.h"
 #include "ui/base/l10n/l10n_util.h"
 #include "ui/message_center/message_center.h"
+#include "ui/message_center/public/cpp/notification_delegate.h"
 
 namespace ash {
 namespace {
 
+// TODO(b/244529735): Replace the generic support URL with the final one.
+const char kLearnMoreUrl[] = "https://www.support.google.com/chromebook";
+
 void SetMicrophoneNotificationVisible(const bool visible) {
   PrivacyHubNotificationController* const privacy_hub_notification_controller =
       Shell::Get()->system_notification_controller()->privacy_hub();
@@ -132,12 +137,30 @@
   notification_data.remove_on_click = true;
 
   scoped_refptr<message_center::NotificationDelegate> delegate;
-  // Don't show a button to unmute device if the microphone was muted by a HW
-  // mute switch, as in that case unmute action would not work.
-  if (!mic_muted_by_mute_switch_) {
+
+  if (mic_muted_by_mute_switch_) {
+    // If microphone is muted by the hardware(HW) switch, show the 'Learn more'
+    // button, pointing to the instructions how to unmute the system (unmute
+    // can't be done programmatically).
+    notification_data.buttons.emplace_back(
+        l10n_util::GetStringUTF16(IDS_ASH_LEARN_MORE));
+    delegate =
+        base::MakeRefCounted<message_center::HandleNotificationClickDelegate>(
+            base::BindRepeating([](absl::optional<int> button_index) {
+              if (!button_index) {
+                return;
+              }
+
+              NewWindowDelegate::GetPrimary()->OpenUrl(
+                  GURL(kLearnMoreUrl),
+                  NewWindowDelegate::OpenUrlFrom::kUserInteraction,
+                  NewWindowDelegate::Disposition::kNewForegroundTab);
+            }));
+  } else {
+    // If microphone is muted by the software(SW) switch, add the unmute
+    // button to the notification.
     notification_data.buttons.emplace_back(l10n_util::GetStringUTF16(
         IDS_MICROPHONE_MUTED_NOTIFICATION_ACTION_BUTTON));
-
     delegate =
         base::MakeRefCounted<message_center::HandleNotificationClickDelegate>(
             base::BindRepeating([](absl::optional<int> button_index) {
diff --git a/ash/system/microphone_mute/microphone_mute_notification_controller_unittest.cc b/ash/system/microphone_mute/microphone_mute_notification_controller_unittest.cc
index aff8c4d2..8c96c27 100644
--- a/ash/system/microphone_mute/microphone_mute_notification_controller_unittest.cc
+++ b/ash/system/microphone_mute/microphone_mute_notification_controller_unittest.cc
@@ -8,6 +8,7 @@
 
 #include "ash/constants/ash_features.h"
 #include "ash/public/cpp/sensor_disabled_notification_delegate.h"
+#include "ash/public/cpp/test/test_new_window_delegate.h"
 #include "ash/public/cpp/test/test_system_tray_client.h"
 #include "ash/strings/grit/ash_strings.h"
 #include "ash/system/microphone_mute/microphone_mute_notification_controller.h"
@@ -17,8 +18,12 @@
 #include "base/strings/utf_string_conversions.h"
 #include "base/test/metrics/histogram_tester.h"
 #include "base/test/scoped_feature_list.h"
+#include "chromeos/ash/components/audio/cras_audio_handler.h"
+#include "chromeos/ash/components/dbus/audio/cras_audio_client.h"
 #include "chromeos/ash/components/dbus/audio/fake_cras_audio_client.h"
+#include "testing/gmock/include/gmock/gmock.h"
 #include "third_party/abseil-cpp/absl/types/optional.h"
+#include "ui/aura/window.h"
 #include "ui/base/l10n/l10n_util.h"
 #include "ui/message_center/message_center.h"
 #include "ui/message_center/public/cpp/notification.h"
@@ -26,6 +31,8 @@
 
 namespace ash {
 
+namespace {
+
 class FakeSensorDisabledNotificationDelegate
     : public SensorDisabledNotificationDelegate {
  public:
@@ -48,16 +55,32 @@
   std::vector<std::u16string> apps_accessing_microphone_;
 };
 
+class MockNewWindowDelegate : public testing::NiceMock<TestNewWindowDelegate> {
+ public:
+  // TestNewWindowDelegate:
+  MOCK_METHOD(void,
+              OpenUrl,
+              (const GURL& url, OpenUrlFrom from, Disposition disposition),
+              (override));
+};
+
+}  // namespace
+
 class MicrophoneMuteNotificationControllerTest : public AshTestBase {
  public:
   MicrophoneMuteNotificationControllerTest() {
     scoped_feature_list_.InitAndEnableFeature(features::kMicMuteNotifications);
+    auto delegate = std::make_unique<MockNewWindowDelegate>();
+    new_window_delegate_ = delegate.get();
+    window_delegate_provider_ =
+        std::make_unique<TestNewWindowDelegateProvider>(std::move(delegate));
   }
   ~MicrophoneMuteNotificationControllerTest() override = default;
 
   // AshTestBase:
   void SetUp() override {
     AshTestBase::SetUp();
+
     controller_ = std::make_unique<MicrophoneMuteNotificationController>();
     delegate_ = std::make_unique<FakeSensorDisabledNotificationDelegate>();
   }
@@ -137,11 +160,15 @@
     return histogram_tester_;
   }
 
+  MockNewWindowDelegate& new_window_delegate() { return *new_window_delegate_; }
+
  private:
   const base::HistogramTester histogram_tester_;
   base::test::ScopedFeatureList scoped_feature_list_;
   std::unique_ptr<MicrophoneMuteNotificationController> controller_;
   std::unique_ptr<FakeSensorDisabledNotificationDelegate> delegate_;
+  MockNewWindowDelegate* new_window_delegate_ = nullptr;
+  std::unique_ptr<TestNewWindowDelegateProvider> window_delegate_provider_;
 };
 
 TEST_F(MicrophoneMuteNotificationControllerTest, SimpleMuteUnMute) {
@@ -269,7 +296,8 @@
   EXPECT_FALSE(GetNotification());
 }
 
-TEST_F(MicrophoneMuteNotificationControllerTest, MuteNotificationActionButton) {
+TEST_F(MicrophoneMuteNotificationControllerTest,
+       SwMuteNotificationActionButton) {
   MuteMicrophone();
   LaunchApp(u"junior");
   SetNumberOfActiveInputStreams(1);
@@ -296,7 +324,7 @@
             1);
 }
 
-TEST_F(MicrophoneMuteNotificationControllerTest, MuteNotificationActionBody) {
+TEST_F(MicrophoneMuteNotificationControllerTest, SwMuteNotificationActionBody) {
   MuteMicrophone();
   LaunchApp(u"junior");
   SetNumberOfActiveInputStreams(1);
@@ -325,23 +353,45 @@
 }
 
 TEST_F(MicrophoneMuteNotificationControllerTest,
-       NoNotificationActionButtonIfMutedByHwSwitch) {
+       HwMuteNotificationActionButton) {
   SetMicrophoneMuteSwitchState(/*muted=*/true);
 
   LaunchApp(u"junior");
   SetNumberOfActiveInputStreams(1);
 
-  // The mute notification should not have an action button if device is muted
-  // by a mute switch.
+  // The mute notification should have a "Learn more" button.
   message_center::Notification* notification = GetNotification();
   ASSERT_TRUE(notification);
-  EXPECT_EQ(0u, notification->buttons().size());
+  EXPECT_EQ(1u, notification->buttons().size());
+
+  // Clicking the "Learn more" button should open a new Chrome tab with the
+  // support link.
+  EXPECT_CALL(new_window_delegate(), OpenUrl).Times(1);
+  ClickOnNotificationButton();
+
+  EXPECT_TRUE(chromeos::CrasAudioHandler::Get()->IsInputMuted());
 
   SetMicrophoneMuteSwitchState(/*muted=*/false);
   ASSERT_FALSE(chromeos::CrasAudioHandler::Get()->IsInputMuted());
   EXPECT_FALSE(GetNotification());
 }
 
+TEST_F(MicrophoneMuteNotificationControllerTest, HwMuteNotificationActionBody) {
+  SetMicrophoneMuteSwitchState(/*muted=*/true);
+  LaunchApp(u"junior");
+  SetNumberOfActiveInputStreams(1);
+
+  message_center::Notification* notification = GetNotification();
+  ASSERT_TRUE(notification);
+  EXPECT_EQ(1u, notification->buttons().size());
+
+  ClickOnNotificationBody();
+
+  // Check that clicking the body has no effect and notification disappears.
+  EXPECT_TRUE(chromeos::CrasAudioHandler::Get()->IsInputMuted());
+  EXPECT_FALSE(GetNotification());
+}
+
 TEST_F(MicrophoneMuteNotificationControllerTest,
        TogglingMuteSwitchRemovesNotificationActionButton) {
   // Mute microphone, and activate an audio input stream.
@@ -353,12 +403,18 @@
   message_center::Notification* notification = GetNotification();
   ASSERT_TRUE(notification);
   EXPECT_EQ(1u, notification->buttons().size());
+  EXPECT_EQ(l10n_util::GetStringUTF16(
+                IDS_MICROPHONE_MUTED_NOTIFICATION_ACTION_BUTTON),
+            notification->buttons()[0].title);
 
-  // Toggle microphone mute switch and verify the action button disappears.
+  // Toggle microphone mute switch and verify that new notification appears with
+  // a "Learn more" button.
   SetMicrophoneMuteSwitchState(/*muted=*/true);
   notification = GetNotification();
   ASSERT_TRUE(notification);
-  EXPECT_EQ(0u, notification->buttons().size());
+  EXPECT_EQ(1u, notification->buttons().size());
+  EXPECT_EQ(l10n_util::GetStringUTF16(IDS_ASH_LEARN_MORE),
+            notification->buttons()[0].title);
 
   SetMicrophoneMuteSwitchState(/*muted=*/false);
   ASSERT_FALSE(chromeos::CrasAudioHandler::Get()->IsInputMuted());
diff --git a/ash/system/notification_center/notification_center_test_api.cc b/ash/system/notification_center/notification_center_test_api.cc
index 07d14d64..5088263b 100644
--- a/ash/system/notification_center/notification_center_test_api.cc
+++ b/ash/system/notification_center/notification_center_test_api.cc
@@ -14,6 +14,7 @@
 #include "ash/system/notification_center/notification_center_view.h"
 #include "ash/system/notification_center/notification_list_view.h"
 #include "ash/system/notification_center/stacked_notification_bar.h"
+#include "ash/system/unified/notification_counter_view.h"
 #include "ash/system/unified/unified_system_tray.h"
 #include "base/strings/string_number_conversions.h"
 #include "ui/base/models/image_model.h"
@@ -84,6 +85,12 @@
   return notification_center_tray_->GetVisible();
 }
 
+bool NotificationCenterTestApi::IsDoNotDisturbIconShown() {
+  return notification_center_tray_->notification_icons_controller_
+      ->quiet_mode_view()
+      ->GetVisible();
+}
+
 views::View* NotificationCenterTestApi::GetNotificationViewForId(
     const std::string& id) {
   // Ensure this api is only called when the notification list view exists, i.e.
diff --git a/ash/system/notification_center/notification_center_test_api.h b/ash/system/notification_center/notification_center_test_api.h
index b9573d6..9c2fe57 100644
--- a/ash/system/notification_center/notification_center_test_api.h
+++ b/ash/system/notification_center/notification_center_test_api.h
@@ -61,10 +61,14 @@
   // otherwise.
   bool IsPopupShown(const std::string& id);
 
-  // Returns true if `NotificationCenterTray`` is showing in the shelf, false
+  // Returns true if `NotificationCenterTray` is showing in the shelf, false
   // otherwise.
   bool IsTrayShown();
 
+  // Returns true if `QuietModeView` is showing in the `NotificationCenterTray`,
+  // false otherwise.
+  bool IsDoNotDisturbIconShown();
+
   // Returns the notification view associated with the provided notification id.
   // Should be only used when the notifications bubble is open.
   views::View* GetNotificationViewForId(const std::string& id);
diff --git a/ash/system/notification_center/notification_center_tray.h b/ash/system/notification_center/notification_center_tray.h
index d5324ec..be8560b 100644
--- a/ash/system/notification_center/notification_center_tray.h
+++ b/ash/system/notification_center/notification_center_tray.h
@@ -13,7 +13,6 @@
 #include "ash/system/tray/tray_background_view.h"
 #include "ash/system/unified/notification_icons_controller.h"
 #include "ui/base/metadata/metadata_header_macros.h"
-#include "ui/message_center/message_center.h"
 #include "ui/message_center/message_center_observer.h"
 #include "ui/message_center/message_center_types.h"
 
diff --git a/ash/system/notification_center/notification_center_tray_unittest.cc b/ash/system/notification_center/notification_center_tray_unittest.cc
index 4f0ea38..9d4366d2 100644
--- a/ash/system/notification_center/notification_center_tray_unittest.cc
+++ b/ash/system/notification_center/notification_center_tray_unittest.cc
@@ -199,6 +199,31 @@
   EXPECT_FALSE(test_api()->IsBubbleShown());
 }
 
+// Tests that visibility of the Do not disturb icon changes with Do not disturb
+// mode.
+TEST_F(NotificationCenterTrayTest, DoNotDisturbIconVisibility) {
+  // Test the case where the tray is not initially visible.
+  ASSERT_FALSE(test_api()->IsTrayShown());
+  EXPECT_FALSE(test_api()->IsDoNotDisturbIconShown());
+  message_center::MessageCenter::Get()->SetQuietMode(true);
+  EXPECT_TRUE(test_api()->IsTrayShown());
+  EXPECT_TRUE(test_api()->IsDoNotDisturbIconShown());
+  message_center::MessageCenter::Get()->SetQuietMode(false);
+  EXPECT_FALSE(test_api()->IsTrayShown());
+  EXPECT_FALSE(test_api()->IsDoNotDisturbIconShown());
+
+  // Test the case where the tray is initially visible.
+  test_api()->AddNotification();
+  ASSERT_TRUE(test_api()->IsTrayShown());
+  EXPECT_FALSE(test_api()->IsDoNotDisturbIconShown());
+  message_center::MessageCenter::Get()->SetQuietMode(true);
+  EXPECT_TRUE(test_api()->IsTrayShown());
+  EXPECT_TRUE(test_api()->IsDoNotDisturbIconShown());
+  message_center::MessageCenter::Get()->SetQuietMode(false);
+  EXPECT_TRUE(test_api()->IsTrayShown());
+  EXPECT_FALSE(test_api()->IsDoNotDisturbIconShown());
+}
+
 // TODO(b/252875025):
 // Add following test cases as we add relevant functionality:
 // - Focus Change dismissing bubble
diff --git a/ash/system/time/time_tray_item_view.cc b/ash/system/time/time_tray_item_view.cc
index fe84631..a070e851 100644
--- a/ash/system/time/time_tray_item_view.cc
+++ b/ash/system/time/time_tray_item_view.cc
@@ -22,7 +22,7 @@
                                      : TimeView::ClockLayout::VERTICAL_CLOCK;
   time_view_ = AddChildView(std::make_unique<TimeView>(
       clock_layout, Shell::Get()->system_tray_model()->clock(), type));
-  time_view_->SetTextColor(kColorAshIconColorPrimary);
+  time_view_->SetTextColorId(kColorAshIconColorPrimary);
 }
 
 TimeTrayItemView::~TimeTrayItemView() = default;
diff --git a/ash/system/time/time_view.cc b/ash/system/time/time_view.cc
index 3d676fb..017702bb 100644
--- a/ash/system/time/time_view.cc
+++ b/ash/system/time/time_view.cc
@@ -194,10 +194,28 @@
   Layout();
 }
 
-void TimeView::SetTextColor(ui::ColorId color_id,
+void TimeView::SetTextColorId(ui::ColorId color_id,
+                              bool auto_color_readability_enabled) {
+  auto set_color_id = [&](views::Label* label) {
+    label->SetEnabledColorId(color_id);
+    label->SetAutoColorReadabilityEnabled(auto_color_readability_enabled);
+  };
+
+  switch (type_) {
+    case kTime:
+      set_color_id(horizontal_label_);
+      set_color_id(vertical_label_hours_);
+      set_color_id(vertical_label_minutes_);
+      return;
+    case kDate:
+      set_color_id(horizontal_label_date_);
+  }
+}
+
+void TimeView::SetTextColor(SkColor color,
                             bool auto_color_readability_enabled) {
   auto set_color = [&](views::Label* label) {
-    label->SetEnabledColorId(color_id);
+    label->SetEnabledColor(color);
     label->SetAutoColorReadabilityEnabled(auto_color_readability_enabled);
   };
 
diff --git a/ash/system/time/time_view.h b/ash/system/time/time_view.h
index 898627e..1419fe9 100644
--- a/ash/system/time/time_view.h
+++ b/ash/system/time/time_view.h
@@ -76,9 +76,12 @@
   // Updates clock layout.
   void UpdateClockLayout(ClockLayout clock_layout);
 
-  // Updates the time text color.
-  void SetTextColor(ui::ColorId color_id,
-                    bool auto_color_readability_enabled = false);
+  // Updates the time text color id.
+  void SetTextColorId(ui::ColorId color_id,
+                      bool auto_color_readability_enabled = false);
+
+  // Updates the text color.
+  void SetTextColor(SkColor color, bool auto_color_readability_enabled = false);
 
   // Updates the time text fontlist.
   void SetTextFont(const gfx::FontList& font_list);
diff --git a/ash/system/tray/system_nudge.cc b/ash/system/tray/system_nudge.cc
index 6e55e7e..32d8b77f 100644
--- a/ash/system/tray/system_nudge.cc
+++ b/ash/system/tray/system_nudge.cc
@@ -189,9 +189,6 @@
   const std::u16string accessibility_text = GetAccessibilityText();
   if (!accessibility_text.empty())
     nudge_view_->GetViewAccessibility().AnnounceText(accessibility_text);
-
-  base::UmaHistogramEnumeration("Ash.NotifierFramework.Nudge.ShownCount",
-                                params_.catalog_name);
 }
 
 void SystemNudge::Close() {
diff --git a/ash/system/tray/system_nudge.h b/ash/system/tray/system_nudge.h
index e89cefd8..c786cfc 100644
--- a/ash/system/tray/system_nudge.h
+++ b/ash/system/tray/system_nudge.h
@@ -66,6 +66,8 @@
 
   views::Widget* widget() { return widget_.get(); }
 
+  NudgeCatalogName catalog_name() { return params_.catalog_name; }
+
  protected:
   // Each SystemNudge subclass must override these methods to customize
   // their nudge by creating a label and getting an icon specific to the feature
diff --git a/ash/system/tray/system_nudge_controller.cc b/ash/system/tray/system_nudge_controller.cc
index 330e33dc..aa54c08 100644
--- a/ash/system/tray/system_nudge_controller.cc
+++ b/ash/system/tray/system_nudge_controller.cc
@@ -10,6 +10,8 @@
 #include "ash/shell.h"
 #include "ash/system/tray/system_nudge.h"
 #include "base/bind.h"
+#include "base/metrics/histogram_functions.h"
+#include "base/strings/strcat.h"
 #include "base/time/time.h"
 #include "ui/compositor/layer.h"
 #include "ui/compositor/scoped_animation_duration_scale_mode.h"
@@ -27,6 +29,19 @@
     gfx::Tween::LINEAR;
 constexpr gfx::Tween::Type kNudgeFadeScalingAnimationTweenType =
     gfx::Tween::LINEAR_OUT_SLOW_IN;
+
+constexpr char NotifierFrameworkNudgeHistogram[] =
+    "Ash.NotifierFramework.Nudge";
+
+// Used in histogram names.
+std::string GetNudgeTimeToActionRange(const base::TimeDelta& time) {
+  if (time <= base::Minutes(1))
+    return "Within1m";
+  if (time <= base::Hours(1))
+    return "Within1h";
+  return "WithinSession";
+}
+
 }  // namespace
 
 // A class for observing the nudge fade out animation. Once the fade
@@ -67,6 +82,31 @@
   hide_nudge_animation_observer_.reset();
 }
 
+// static
+void SystemNudgeController::RecordNudgeAction(NudgeCatalogName catalog_name) {
+  auto& nudge_registry = GetNudgeRegistry();
+  auto it = std::find_if(
+      std::begin(nudge_registry), std::end(nudge_registry),
+      [catalog_name](
+          const std::pair<NudgeCatalogName, base::TimeTicks> registry_entry) {
+        return catalog_name == registry_entry.first;
+      });
+
+  // Don't record "TimeToAction" metric if the nudge hasn't been shown.
+  if (it == std::end(nudge_registry))
+    return;
+
+  const base::TimeDelta delta = base::TimeTicks::Now() - (*it).second;
+  const std::string time_range = GetNudgeTimeToActionRange(delta);
+
+  base::UmaHistogramEnumeration(
+      base::StrCat({NotifierFrameworkNudgeHistogram, ".TimeToAction.",
+                    time_range.c_str()}),
+      catalog_name);
+
+  nudge_registry.erase(it);
+}
+
 void SystemNudgeController::ShowNudge() {
   if (nudge_ && !nudge_->widget()->IsClosed()) {
     hide_nudge_timer_.AbandonAndStop();
@@ -77,6 +117,7 @@
   nudge_ = CreateSystemNudge();
   nudge_->Show();
   StartFadeAnimation(/*show=*/true);
+  RecordNudgeShown(nudge_->catalog_name());
 
   // Start a timer to close the nudge after a set amount of time.
   hide_nudge_timer_.Start(FROM_HERE, kNudgeShowTime,
@@ -92,10 +133,22 @@
   hide_nudge_timer_.FireNow();
 }
 
+void SystemNudgeController::ResetNudgeRegistryForTesting() {
+  GetNudgeRegistry().clear();
+}
+
 void SystemNudgeController::HideNudge() {
   StartFadeAnimation(/*show=*/false);
 }
 
+// static
+std::vector<std::pair<NudgeCatalogName, base::TimeTicks>>&
+SystemNudgeController::GetNudgeRegistry() {
+  static auto nudge_registry =
+      std::vector<std::pair<NudgeCatalogName, base::TimeTicks>>();
+  return nudge_registry;
+}
+
 void SystemNudgeController::StartFadeAnimation(bool show) {
   // Clean any pending animation observer.
   hide_nudge_animation_observer_.reset();
@@ -142,4 +195,23 @@
   }
 }
 
+void SystemNudgeController::RecordNudgeShown(NudgeCatalogName catalog_name) {
+  base::UmaHistogramEnumeration("Ash.NotifierFramework.Nudge.ShownCount",
+                                catalog_name);
+  auto& nudge_registry = GetNudgeRegistry();
+
+  auto it = std::find_if(
+      std::begin(nudge_registry), std::end(nudge_registry),
+      [catalog_name](
+          const std::pair<NudgeCatalogName, base::TimeTicks> registry_entry) {
+        return catalog_name == registry_entry.first;
+      });
+
+  if (it == std::end(nudge_registry)) {
+    nudge_registry.emplace_back(catalog_name, base::TimeTicks::Now());
+  } else {
+    (*it).second = base::TimeTicks::Now();
+  }
+}
+
 }  // namespace ash
diff --git a/ash/system/tray/system_nudge_controller.h b/ash/system/tray/system_nudge_controller.h
index b07a38c..85223ece 100644
--- a/ash/system/tray/system_nudge_controller.h
+++ b/ash/system/tray/system_nudge_controller.h
@@ -8,6 +8,7 @@
 #include <memory>
 
 #include "ash/ash_export.h"
+#include "ash/constants/notifier_catalogs.h"
 #include "base/memory/weak_ptr.h"
 #include "base/timer/timer.h"
 #include "ui/compositor/layer_animation_observer.h"
@@ -28,6 +29,10 @@
   SystemNudgeController& operator=(const SystemNudgeController&) = delete;
   virtual ~SystemNudgeController();
 
+  // Records Nudge "TimeToAction" metric, which tracks the time from when a
+  // nudge was shown to when the action the nudge informs of was performed.
+  static void RecordNudgeAction(NudgeCatalogName catalog_name);
+
   // Shows the nudge widget.
   void ShowNudge();
 
@@ -40,6 +45,10 @@
   // Get the system nudge for testing purpose.
   SystemNudge* GetSystemNudgeForTesting() { return nudge_.get(); }
 
+  // Resets the `nudge_registry` object that records the time a nudge was last
+  // shown.
+  void ResetNudgeRegistryForTesting();
+
  protected:
   // Concrete subclasses must implement this method to return a
   // SystemNudge that creates a label and specifies an icon specific
@@ -50,9 +59,17 @@
   void HideNudge();
 
  private:
+  // Returns the registry which keeps track of when a nudge was last shown.
+  static std::vector<std::pair<NudgeCatalogName, base::TimeTicks>>&
+  GetNudgeRegistry();
+
   // Begins the animation for fading in or fading out the nudge.
   void StartFadeAnimation(bool show);
 
+  // Records the time a nudge was last shown and stores it in the
+  // `nudge_registry`.
+  void RecordNudgeShown(NudgeCatalogName catalog_name);
+
   // Contextual nudge which shows a view.
   std::unique_ptr<SystemNudge> nudge_;
 
diff --git a/ash/system/tray/system_nudge_unittest.cc b/ash/system/tray/system_nudge_unittest.cc
index 5f87de40e..66644ff1 100644
--- a/ash/system/tray/system_nudge_unittest.cc
+++ b/ash/system/tray/system_nudge_unittest.cc
@@ -9,9 +9,11 @@
 #include "ash/public/cpp/shelf_types.h"
 #include "ash/shelf/shelf.h"
 #include "ash/shelf/shelf_layout_manager.h"
+#include "ash/system/tray/system_nudge_controller.h"
 #include "ash/system/tray/system_nudge_label.h"
 #include "ash/test/ash_test_base.h"
 #include "base/test/metrics/histogram_tester.h"
+#include "base/test/task_environment.h"
 #include "ui/gfx/vector_icon_types.h"
 
 namespace ash {
@@ -26,12 +28,22 @@
 
 constexpr char kNudgeName[] = "TestSystemNudge";
 
+constexpr NudgeCatalogName kTestCatalogName =
+    NudgeCatalogName::kTestCatalogName;
+
+constexpr char kNudgeShownCount[] = "Ash.NotifierFramework.Nudge.ShownCount";
+constexpr char kNudgeTimeToActionWithin1m[] =
+    "Ash.NotifierFramework.Nudge.TimeToAction.Within1m";
+constexpr char kNudgeTimeToActionWithin1h[] =
+    "Ash.NotifierFramework.Nudge.TimeToAction.Within1h";
+constexpr char kNudgeTimeToActionWithinSession[] =
+    "Ash.NotifierFramework.Nudge.TimeToAction.WithinSession";
+
 gfx::VectorIcon kEmptyIcon;
 
 class TestSystemNudge : public SystemNudge {
  public:
-  explicit TestSystemNudge(
-      NudgeCatalogName catalog_name = NudgeCatalogName::kTestCatalogName)
+  explicit TestSystemNudge(NudgeCatalogName catalog_name = kTestCatalogName)
       : SystemNudge(kNudgeName,
                     catalog_name,
                     kIconSize,
@@ -54,14 +66,26 @@
   }
 };
 
-constexpr char kNudgeShownCountHistogramName[] =
-    "Ash.NotifierFramework.Nudge.ShownCount";
+class TestSystemNudgeController : public SystemNudgeController {
+ public:
+  TestSystemNudgeController() = default;
+  TestSystemNudgeController(const TestSystemNudgeController&) = delete;
+  TestSystemNudgeController& operator=(const TestSystemNudgeController&) =
+      delete;
+  ~TestSystemNudgeController() override = default;
+
+  // SystemNudgeController:
+  std::unique_ptr<SystemNudge> CreateSystemNudge() override {
+    return std::make_unique<TestSystemNudge>(kTestCatalogName);
+  }
+};
 
 }  // namespace
 
 class SystemNudgeTest : public AshTestBase {
  public:
-  SystemNudgeTest() = default;
+  SystemNudgeTest()
+      : AshTestBase(base::test::TaskEnvironment::TimeSource::MOCK_TIME) {}
   SystemNudgeTest(const SystemNudgeTest&) = delete;
   SystemNudgeTest& operator=(const SystemNudgeTest&) = delete;
   ~SystemNudgeTest() override = default;
@@ -107,20 +131,54 @@
 
 TEST_F(SystemNudgeTest, ShownCountMetric) {
   base::HistogramTester histogram_tester;
+  auto nudge_controller = std::make_unique<TestSystemNudgeController>();
 
-  const NudgeCatalogName catalog_name_1 = static_cast<NudgeCatalogName>(1);
-  const NudgeCatalogName catalog_name_2 = static_cast<NudgeCatalogName>(2);
-  TestSystemNudge nudge_1(catalog_name_1);
-  TestSystemNudge nudge_2(catalog_name_2);
+  histogram_tester.ExpectBucketCount(kNudgeShownCount, kTestCatalogName, 0);
 
-  nudge_1.Show();
-  histogram_tester.ExpectBucketCount(kNudgeShownCountHistogramName,
-                                     catalog_name_1, 1);
+  nudge_controller->ShowNudge();
+  histogram_tester.ExpectBucketCount(kNudgeShownCount, kTestCatalogName, 1);
 
-  nudge_2.Show();
-  nudge_2.Show();
-  histogram_tester.ExpectBucketCount(kNudgeShownCountHistogramName,
-                                     catalog_name_2, 2);
+  nudge_controller->ShowNudge();
+  nudge_controller->ShowNudge();
+  histogram_tester.ExpectBucketCount(kNudgeShownCount, kTestCatalogName, 3);
+}
+
+TEST_F(SystemNudgeTest, TimeToActionMetric) {
+  base::HistogramTester histogram_tester;
+  auto nudge_controller = std::make_unique<TestSystemNudgeController>();
+  nudge_controller->ResetNudgeRegistryForTesting();
+
+  // Metric is not recorded if nudge has not been shown.
+  SystemNudgeController::RecordNudgeAction(kTestCatalogName);
+  histogram_tester.ExpectBucketCount(kNudgeTimeToActionWithin1m,
+                                     kTestCatalogName, 0);
+
+  // Metric is recorded after nudge is shown.
+  nudge_controller->ShowNudge();
+  task_environment()->FastForwardBy(base::Seconds(1));
+  SystemNudgeController::RecordNudgeAction(kTestCatalogName);
+  histogram_tester.ExpectBucketCount(kNudgeTimeToActionWithin1m,
+                                     kTestCatalogName, 1);
+
+  // Metric is not recorded if the nudge action is performed again without
+  // another nudge being shown.
+  SystemNudgeController::RecordNudgeAction(kTestCatalogName);
+  histogram_tester.ExpectBucketCount(kNudgeTimeToActionWithin1m,
+                                     kTestCatalogName, 1);
+
+  // Metric is recorded with appropriate time range after showing nudge again
+  // and waiting the time to fall into the next time bucket.
+  nudge_controller->ShowNudge();
+  task_environment()->FastForwardBy(base::Minutes(2));
+  SystemNudgeController::RecordNudgeAction(kTestCatalogName);
+  histogram_tester.ExpectBucketCount(kNudgeTimeToActionWithin1h,
+                                     kTestCatalogName, 1);
+
+  nudge_controller->ShowNudge();
+  task_environment()->FastForwardBy(base::Hours(2));
+  SystemNudgeController::RecordNudgeAction(kTestCatalogName);
+  histogram_tester.ExpectBucketCount(kNudgeTimeToActionWithinSession,
+                                     kTestCatalogName, 1);
 }
 
 TEST_F(SystemNudgeTest, NudgePositionChangeWhenShelfAutoHide) {
diff --git a/ash/system/unified/notification_counter_view.cc b/ash/system/unified/notification_counter_view.cc
index 75cc1ae..5eae9db7 100644
--- a/ash/system/unified/notification_counter_view.cc
+++ b/ash/system/unified/notification_counter_view.cc
@@ -4,19 +4,14 @@
 
 #include "ash/system/unified/notification_counter_view.h"
 
-#include <algorithm>
-
-#include "ash/public/cpp/vm_camera_mic_constants.h"
 #include "ash/resources/vector_icons/vector_icons.h"
 #include "ash/session/session_controller_impl.h"
 #include "ash/shell.h"
 #include "ash/strings/grit/ash_strings.h"
 #include "ash/style/ash_color_id.h"
-#include "ash/style/ash_color_provider.h"
 #include "ash/system/message_center/ash_message_center_lock_screen_controller.h"
 #include "ash/system/message_center/message_center_utils.h"
 #include "ash/system/tray/tray_constants.h"
-#include "ash/system/tray/tray_utils.h"
 #include "ash/system/unified/notification_icons_controller.h"
 #include "base/i18n/number_formatting.h"
 #include "ui/base/l10n/l10n_util.h"
@@ -25,11 +20,9 @@
 #include "ui/gfx/canvas.h"
 #include "ui/gfx/font_list.h"
 #include "ui/gfx/image/canvas_image_source.h"
-#include "ui/gfx/vector_icon_utils.h"
 #include "ui/message_center/message_center.h"
 #include "ui/views/border.h"
 #include "ui/views/controls/image_view.h"
-#include "ui/views/controls/label.h"
 #include "ui/views/controls/separator.h"
 
 namespace ash {
@@ -196,11 +189,9 @@
 QuietModeView::~QuietModeView() = default;
 
 void QuietModeView::Update() {
-  // TODO(yamaguchi): Add this check when new style of the system tray is
-  // implemented, so that icon resizing will not happen here.
-  // DCHECK_EQ(kTrayIconSize,
-  //     gfx::GetDefaultSizeOfVectorIcon(kSystemTrayDoNotDisturbIcon));
-  if (message_center::MessageCenter::Get()->IsQuietMode()) {
+  if (message_center::MessageCenter::Get()->IsQuietMode() &&
+      Shell::Get()->session_controller()->GetSessionState() ==
+          session_manager::SessionState::ACTIVE) {
     image_view()->SetImage(ui::ImageModel::FromVectorIcon(
         kSystemTrayDoNotDisturbIcon, kColorAshIconColorPrimary));
     SetVisible(true);
diff --git a/ash/system/unified/notification_counter_view.h b/ash/system/unified/notification_counter_view.h
index 40b1063..1d6337a 100644
--- a/ash/system/unified/notification_counter_view.h
+++ b/ash/system/unified/notification_counter_view.h
@@ -7,7 +7,6 @@
 
 #include "ash/ash_export.h"
 #include "ash/system/tray/tray_item_view.h"
-#include "base/scoped_observation.h"
 
 namespace session_manager {
 enum class SessionState;
diff --git a/ash/system/unified/notification_counter_view_unittest.cc b/ash/system/unified/notification_counter_view_unittest.cc
index 3ee91d09..8fe1b164 100644
--- a/ash/system/unified/notification_counter_view_unittest.cc
+++ b/ash/system/unified/notification_counter_view_unittest.cc
@@ -12,8 +12,6 @@
 #include "ash/test/ash_test_base.h"
 #include "base/strings/string_number_conversions.h"
 #include "base/test/scoped_feature_list.h"
-#include "testing/gtest/include/gtest/gtest.h"
-#include "ui/gfx/image/image.h"
 #include "ui/message_center/message_center.h"
 #include "ui/message_center/public/cpp/notification.h"
 #include "ui/message_center/public/cpp/notification_types.h"
@@ -43,7 +41,8 @@
 class NotificationCounterViewTest : public AshTestBase,
                                     public testing::WithParamInterface<bool> {
  public:
-  NotificationCounterViewTest() = default;
+  NotificationCounterViewTest()
+      : AshTestBase(base::test::TaskEnvironment::TimeSource::MOCK_TIME) {}
   NotificationCounterViewTest(const NotificationCounterViewTest&) = delete;
   NotificationCounterViewTest& operator=(const NotificationCounterViewTest&) =
       delete;
@@ -72,6 +71,15 @@
                                      ->notification_counter_view();
   }
 
+  QuietModeView* GetDoNotDisturbIconView() {
+    auto* status_area_widget = GetPrimaryShelf()->status_area_widget();
+    return IsQsRevampEnabled()
+               ? status_area_widget->notification_center_tray()
+                     ->notification_icons_controller_->quiet_mode_view()
+               : status_area_widget->unified_system_tray()
+                     ->notification_icons_controller_->quiet_mode_view();
+  }
+
  private:
   std::unique_ptr<base::test::ScopedFeatureList> scoped_feature_list_;
 };
@@ -166,4 +174,20 @@
   EXPECT_FALSE(GetNotificationCounterView()->GetVisible());
 }
 
+TEST_P(NotificationCounterViewTest, DoNotDisturbIconVisibility) {
+  ASSERT_FALSE(GetDoNotDisturbIconView()->GetVisible());
+
+  // Turn on Do not disturb mode.
+  message_center::MessageCenter::Get()->SetQuietMode(true);
+  EXPECT_TRUE(GetDoNotDisturbIconView()->GetVisible());
+
+  // Show the lock screen.
+  BlockUserSession(BLOCKED_BY_LOCK_SCREEN);
+  EXPECT_FALSE(GetDoNotDisturbIconView()->GetVisible());
+
+  // Log in.
+  UnblockUserSession();
+  EXPECT_TRUE(GetDoNotDisturbIconView()->GetVisible());
+}
+
 }  // namespace ash
diff --git a/ash/system/unified/notification_icons_controller.cc b/ash/system/unified/notification_icons_controller.cc
index e783eb5..dad44cf 100644
--- a/ash/system/unified/notification_icons_controller.cc
+++ b/ash/system/unified/notification_icons_controller.cc
@@ -175,15 +175,14 @@
   notification_counter_view_ = tray_container->AddChildView(
       std::make_unique<NotificationCounterView>(shelf_, /*controller=*/this));
 
-  // `quiet_mode_view_` and `separator_` are only shown in the
-  // `UnifiedSystemTray` with the QsRevamp feature disabled. Once kQsRevamp
-  // launches `quiet_mode_view_` will remain in the `UnifiedSystemTray`.
-  // The `separator_` will not be needed because the icons related to this
-  // controller will have their own dedicated tray button.
-  if (!features::IsQsRevampEnabled()) {
-    quiet_mode_view_ =
-        tray_container->AddChildView(std::make_unique<QuietModeView>(shelf_));
+  quiet_mode_view_ =
+      tray_container->AddChildView(std::make_unique<QuietModeView>(shelf_));
 
+  // `separator_` is only shown in the `UnifiedSystemTray` with the QsRevamp
+  // feature disabled. The `separator_` will not be needed once kQsRevamp
+  // launches because the icons related to this controller will have their own
+  // dedicated tray button.
+  if (!features::IsQsRevampEnabled()) {
     separator_ = tray_container->AddChildView(
         std::make_unique<SeparatorTrayItemView>(shelf_));
   }
@@ -222,8 +221,7 @@
 
 void NotificationIconsController::UpdateNotificationIndicators() {
   notification_counter_view_->Update();
-  if (!features::IsQsRevampEnabled())
-    quiet_mode_view_->Update();
+  quiet_mode_view_->Update();
 }
 
 void NotificationIconsController::OnSystemTrayButtonSizeChanged(
diff --git a/ash/system/unified/unified_system_tray.cc b/ash/system/unified/unified_system_tray.cc
index a4a7a6c..d41f53d 100644
--- a/ash/system/unified/unified_system_tray.cc
+++ b/ash/system/unified/unified_system_tray.cc
@@ -36,7 +36,6 @@
 #include "ash/system/tray/tray_background_view.h"
 #include "ash/system/tray/tray_constants.h"
 #include "ash/system/tray/tray_container.h"
-#include "ash/system/tray/tray_event_filter.h"
 #include "ash/system/unified/camera_mic_tray_item_view.h"
 #include "ash/system/unified/current_locale_view.h"
 #include "ash/system/unified/date_tray.h"
@@ -49,7 +48,6 @@
 #include "ash/system/unified/unified_system_tray_bubble.h"
 #include "ash/system/unified/unified_system_tray_model.h"
 #include "ash/system/unified/unified_system_tray_view.h"
-#include "base/debug/stack_trace.h"
 #include "base/memory/scoped_refptr.h"
 #include "base/metrics/histogram_functions.h"
 #include "base/metrics/histogram_macros.h"
@@ -63,8 +61,6 @@
 #include "ui/display/display.h"
 #include "ui/display/screen.h"
 #include "ui/gfx/geometry/point.h"
-#include "ui/message_center/message_center.h"
-#include "ui/message_center/notification_view_controller.h"
 #include "ui/views/controls/image_view.h"
 
 namespace ash {
@@ -214,12 +210,6 @@
 
   AddTrayItemToContainer(std::make_unique<ScreenCaptureTrayItemView>(shelf));
 
-  if (features::IsQsRevampEnabled()) {
-    quiet_mode_view_ =
-        tray_container()->AddChildView(std::make_unique<QuietModeView>(shelf));
-    tray_items_.push_back(quiet_mode_view_);
-  }
-
   if (!features::IsQsRevampEnabled()) {
     tray_items_.push_back(
         notification_icons_controller_->notification_counter_view());
@@ -275,11 +265,13 @@
 
   ShelfConfig::Get()->AddObserver(this);
   Shell::Get()->tablet_mode_controller()->AddObserver(this);
+  message_center::MessageCenter::Get()->AddObserver(this);
 }
 
 UnifiedSystemTray::~UnifiedSystemTray() {
   Shell::Get()->tablet_mode_controller()->RemoveObserver(this);
   ShelfConfig::Get()->RemoveObserver(this);
+  message_center::MessageCenter::Get()->RemoveObserver(this);
 
   DestroyBubbles();
 
@@ -513,6 +505,12 @@
   UpdateLayout();
 }
 
+void UnifiedSystemTray::OnQuietModeChanged(bool in_quiet_mode) {
+  if (!features::IsQsRevampEnabled()) {
+    notification_icons_controller_->UpdateNotificationIndicators();
+  }
+}
+
 void UnifiedSystemTray::OnDateTrayActionPerformed(const ui::Event& event) {
   if (!bubble_)
     ShowBubble();
@@ -720,9 +718,6 @@
 }
 
 void UnifiedSystemTray::UpdateNotificationInternal() {
-  if (features::IsQsRevampEnabled())
-    quiet_mode_view_->Update();
-
   // Limit update frequency in order to avoid flashing when 2 updates are
   // incoming in a very short period of time. It happens when ARC++ apps
   // creating bundled notifications.
diff --git a/ash/system/unified/unified_system_tray.h b/ash/system/unified/unified_system_tray.h
index 67040800..0840971 100644
--- a/ash/system/unified/unified_system_tray.h
+++ b/ash/system/unified/unified_system_tray.h
@@ -13,8 +13,6 @@
 #include "ash/public/cpp/accelerators.h"
 #include "ash/public/cpp/shelf_config.h"
 #include "ash/public/cpp/tablet_mode_observer.h"
-#include "ash/system/status_area_widget.h"
-#include "ash/system/time/time_view.h"
 #include "ash/system/tray/tray_background_view.h"
 #include "ash/system/unified/unified_system_tray_controller.h"
 #include "ash/system/unified/unified_system_tray_model.h"
@@ -49,7 +47,6 @@
 class NotificationIconsController;
 class PrivacyIndicatorsTrayItemView;
 class PrivacyScreenToastController;
-class QuietModeView;
 class Shelf;
 class TrayBubbleView;
 class TrayItemView;
@@ -74,7 +71,8 @@
     : public TrayBackgroundView,
       public ShelfConfig::Observer,
       public UnifiedSystemTrayController::Observer,
-      public TabletModeObserver {
+      public TabletModeObserver,
+      public message_center::MessageCenterObserver {
  public:
   class Observer : public base::CheckedObserver {
    public:
@@ -216,6 +214,9 @@
   void OnTabletModeStarted() override;
   void OnTabletModeEnded() override;
 
+  // message_center::MessageCenterObserver:
+  void OnQuietModeChanged(bool in_quiet_mode) override;
+
   // Gets called when an action is performed on the `DateTray`.
   void OnDateTrayActionPerformed(const ui::Event& event);
 
@@ -309,7 +310,6 @@
 
   NetworkTrayView* network_tray_view_ = nullptr;
   ChannelIndicatorView* channel_indicator_view_ = nullptr;
-  QuietModeView* quiet_mode_view_ = nullptr;
 
   // Contains all tray items views added to tray_container().
   std::list<TrayItemView*> tray_items_;
diff --git a/ash/system/unified/unified_system_tray_unittest.cc b/ash/system/unified/unified_system_tray_unittest.cc
index 1f1a6d9..d5f266bc 100644
--- a/ash/system/unified/unified_system_tray_unittest.cc
+++ b/ash/system/unified/unified_system_tray_unittest.cc
@@ -6,7 +6,6 @@
 
 #include "ash/accessibility/accessibility_controller_impl.h"
 #include "ash/constants/ash_features.h"
-#include "ash/ime/ime_controller_impl.h"
 #include "ash/public/cpp/test/shell_test_api.h"
 #include "ash/root_window_controller.h"
 #include "ash/shelf/shelf.h"
@@ -21,10 +20,8 @@
 #include "ash/system/time/time_tray_item_view.h"
 #include "ash/system/time/time_view.h"
 #include "ash/system/unified/ime_mode_view.h"
-#include "ash/system/unified/notification_counter_view.h"
 #include "ash/system/unified/unified_slider_bubble_controller.h"
 #include "ash/system/unified/unified_system_tray_bubble.h"
-#include "ash/system/unified/unified_system_tray_controller.h"
 #include "ash/system/unified/unified_system_tray_view.h"
 #include "ash/test/ash_test_base.h"
 #include "ash/wm/tablet_mode/tablet_mode_controller.h"
@@ -37,7 +34,6 @@
 #include "ui/base/l10n/l10n_util.h"
 #include "ui/display/display.h"
 #include "ui/display/screen.h"
-#include "ui/events/event.h"
 #include "ui/events/event_constants.h"
 #include "ui/message_center/message_center.h"
 
@@ -149,10 +145,6 @@
     return GetPrimaryUnifiedSystemTray()->ime_mode_view_;
   }
 
-  QuietModeView* quiet_mode_view() {
-    return GetPrimaryUnifiedSystemTray()->quiet_mode_view_;
-  }
-
  private:
   int id_ = 0;
   base::test::ScopedFeatureList feature_list_;
@@ -685,23 +677,6 @@
             ShelfConfig::Get()->GetShelfControlButtonColor(widget));
 }
 
-// Tests that the `quiet_mode_view_` is visible based on the system's quiet mode
-// setting.
-TEST_P(UnifiedSystemTrayTest, QuietModeViewVisibility) {
-  // `quiet_mode_view_` does not exist in `unified_system_tray_` if QsRevamp is
-  // not enabled. It is owned by `notification_icons_controller` in that case.
-  if (!IsQsRevampEnabled())
-    return;
-
-  auto* message_center = message_center::MessageCenter::Get();
-
-  message_center->SetQuietMode(false);
-  EXPECT_FALSE(quiet_mode_view()->GetVisible());
-
-  message_center->SetQuietMode(true);
-  EXPECT_TRUE(quiet_mode_view()->GetVisible());
-}
-
 // Tests that the bubble automatically hides if it is visible when another
 // bubble becomes visible, and otherwise does not automatically show or hide.
 TEST_P(UnifiedSystemTrayTest, BubbleHideBehavior) {
diff --git a/ash/webui/diagnostics_ui/resources/keyboard_tester.html b/ash/webui/diagnostics_ui/resources/keyboard_tester.html
index 9f3baef..7717a3b 100644
--- a/ash/webui/diagnostics_ui/resources/keyboard_tester.html
+++ b/ash/webui/diagnostics_ui/resources/keyboard_tester.html
@@ -1,13 +1,13 @@
 <style include="diagnostics-shared">
   @media (min-width: 600px) {
     :host {
-      --cr-dialog-width: 416px;
+      --cr-dialog-width: 504px;
     }
   }
 
   @media (min-width: 768px) {
     :host {
-      --cr-dialog-width: 512px;
+      --cr-dialog-width: 636px;
     }
   }
 
diff --git a/ash/webui/personalization_app/personalization_app_ui.cc b/ash/webui/personalization_app/personalization_app_ui.cc
index 8551a4c..84ca5aa6 100644
--- a/ash/webui/personalization_app/personalization_app_ui.cc
+++ b/ash/webui/personalization_app/personalization_app_ui.cc
@@ -307,7 +307,7 @@
       "script-src chrome://resources chrome://test chrome://webui-test "
       "'self';");
 
-  // TODO(crbug.com/1098690): Trusted Type Polymer
+  // TODO(crbug.com/1400799): Enable TrustedTypes.
   source->DisableTrustedTypesCSP();
 
   AddResources(source);
diff --git a/base/allocator/partition_allocator/BUILD.gn b/base/allocator/partition_allocator/BUILD.gn
index 0a6bb25..d8e0a94 100644
--- a/base/allocator/partition_allocator/BUILD.gn
+++ b/base/allocator/partition_allocator/BUILD.gn
@@ -393,6 +393,7 @@
     "BACKUP_REF_PTR_POISON_OOB_PTR=$backup_ref_ptr_poison_oob_ptr",
     "PUT_REF_COUNT_IN_PREVIOUS_SLOT=$put_ref_count_in_previous_slot",
     "USE_ASAN_BACKUP_REF_PTR=$use_asan_backup_ref_ptr",
+    "USE_ASAN_UNOWNED_PTR=$use_asan_unowned_ptr",
     "ENABLE_GWP_ASAN_SUPPORT=$_enable_gwp_asan_support",
 
     # Not to be used directly - instead use
diff --git a/base/allocator/partition_allocator/partition_alloc.gni b/base/allocator/partition_allocator/partition_alloc.gni
index cf598e18..88915bd5 100644
--- a/base/allocator/partition_allocator/partition_alloc.gni
+++ b/base/allocator/partition_allocator/partition_alloc.gni
@@ -83,6 +83,9 @@
   # research. ASAN BRP is not supported in Chromium-external builds.
   use_asan_backup_ref_ptr =
       build_with_chromium && is_asan && (is_win || is_android || is_linux)
+
+  # Use probe-on-destruct unowned ptr detection with ASAN.
+  use_asan_unowned_ptr = false
 }
 
 # AsanBackupRefPtr is not supported outside Chromium. The implementation is
@@ -96,6 +99,11 @@
     !enable_backup_ref_ptr_support || !use_asan_backup_ref_ptr,
     "Both BackupRefPtr and AsanBackupRefPtr can't be enabled at the same time")
 
+# AsanBackupRefPtr and AsanUnownedPtr are mutually exclusive variants of raw_ptr.
+assert(
+    !use_asan_unowned_ptr || !use_asan_backup_ref_ptr,
+    "Both AsanUnownedPtr and AsanBackupRefPtr can't be enabled at the same time")
+
 assert(!use_asan_backup_ref_ptr || is_asan,
        "AsanBackupRefPtr requires AddressSanitizer")
 
@@ -151,6 +159,7 @@
   enable_backup_ref_ptr_support = false
   enable_mte_checked_ptr_support = false
   use_asan_backup_ref_ptr = false
+  use_asan_unowned_ptr = false
   put_ref_count_in_previous_slot = false
   enable_backup_ref_ptr_slow_checks = false
   enable_dangling_raw_ptr_checks = false
@@ -187,6 +196,9 @@
     enable_backup_ref_ptr_support || !backup_ref_ptr_poison_oob_ptr,
     "Can't enable poisoning for OOB pointers if BackupRefPtr isn't enabled at all")
 
+assert(!use_asan_unowned_ptr || is_asan,
+       "AsanUnownedPtr requires AddressSanitizer")
+
 declare_args() {
   enable_pkeys = is_linux && target_cpu == "x64"
 }
diff --git a/base/memory/raw_ptr.h b/base/memory/raw_ptr.h
index f2273362..4191f0e4 100644
--- a/base/memory/raw_ptr.h
+++ b/base/memory/raw_ptr.h
@@ -817,6 +817,90 @@
 };
 #endif  // BUILDFLAG(USE_ASAN_BACKUP_REF_PTR)
 
+#if BUILDFLAG(USE_ASAN_UNOWNED_PTR)
+
+struct AsanUnownedPtrImpl {
+  // Wraps a pointer.
+  template <typename T>
+  static PA_ALWAYS_INLINE T* WrapRawPtr(T* ptr) {
+    return ptr;
+  }
+
+  // Notifies the allocator when a wrapped pointer is being removed or replaced.
+  template <typename T>
+  static PA_ALWAYS_INLINE void ReleaseWrappedPtr(T* wrapped_ptr) {
+    ProbeForLowSeverityLifetimeIssue(wrapped_ptr);
+  }
+
+  // Unwraps the pointer, while asserting that memory hasn't been freed. The
+  // function is allowed to crash on nullptr.
+  template <typename T>
+  static PA_ALWAYS_INLINE T* SafelyUnwrapPtrForDereference(T* wrapped_ptr) {
+    // ASAN will catch use of dereferenced ptr without additional probing.
+    return wrapped_ptr;
+  }
+
+  // Unwraps the pointer, while asserting that memory hasn't been freed. The
+  // function must handle nullptr gracefully.
+  template <typename T>
+  static PA_ALWAYS_INLINE T* SafelyUnwrapPtrForExtraction(T* wrapped_ptr) {
+    ProbeForLowSeverityLifetimeIssue(wrapped_ptr);
+    return wrapped_ptr;
+  }
+
+  // Unwraps the pointer, without making an assertion on whether memory was
+  // freed or not.
+  template <typename T>
+  static PA_ALWAYS_INLINE T* UnsafelyUnwrapPtrForComparison(T* wrapped_ptr) {
+    return wrapped_ptr;
+  }
+
+  // Upcasts the wrapped pointer.
+  template <typename To, typename From>
+  static PA_ALWAYS_INLINE constexpr To* Upcast(From* wrapped_ptr) {
+    static_assert(std::is_convertible<From*, To*>::value,
+                  "From must be convertible to To.");
+    // Note, this cast may change the address if upcasting to base that lies in
+    // the middle of the derived object.
+    return wrapped_ptr;
+  }
+
+  // Advance the wrapped pointer by `delta_elems`.
+  template <typename T,
+            typename Z,
+            typename = std::enable_if_t<offset_type<Z>, void>>
+  static PA_ALWAYS_INLINE T* Advance(T* wrapped_ptr, Z delta_elems) {
+    return wrapped_ptr + delta_elems;
+  }
+
+  template <typename T>
+  static PA_ALWAYS_INLINE ptrdiff_t GetDeltaElems(T* wrapped_ptr1,
+                                                  T* wrapped_ptr2) {
+    return wrapped_ptr1 - wrapped_ptr2;
+  }
+
+  // Returns a copy of a wrapped pointer, without making an assertion on whether
+  // memory was freed or not.
+  template <typename T>
+  static PA_ALWAYS_INLINE T* Duplicate(T* wrapped_ptr) {
+    return wrapped_ptr;
+  }
+
+  template <typename T>
+  static void ProbeForLowSeverityLifetimeIssue(T* wrapped_ptr) {
+    if (wrapped_ptr) {
+      reinterpret_cast<const volatile uint8_t*>(wrapped_ptr)[0];
+    }
+  }
+
+  // This is for accounting only, used by unit tests.
+  static PA_ALWAYS_INLINE void IncrementSwapCountForTest() {}
+  static PA_ALWAYS_INLINE void IncrementLessCountForTest() {}
+  static PA_ALWAYS_INLINE void IncrementPointerToMemberOperatorCountForTest() {}
+};
+
+#endif  // BUILDFLAG(USE_ASAN_UNOWNED_PTR)
+
 template <class Super>
 struct RawPtrCountingImplWrapperForTest
     : public raw_ptr_traits::RawPtrTypeToImpl<Super>::Impl {
@@ -985,6 +1069,10 @@
   using Impl = internal::BackupRefPtrImpl</*AllowDangling=*/true>;
 #elif BUILDFLAG(USE_ASAN_BACKUP_REF_PTR)
   using Impl = internal::AsanBackupRefPtrImpl;
+#elif BUILDFLAG(USE_ASAN_UNOWNED_PTR)
+  // No special bookkeeping required for this case, just treat these
+  // as ordinary pointers.
+  using Impl = internal::RawPtrNoOpImpl;
 #elif defined(PA_ENABLE_MTE_CHECKED_PTR_SUPPORT_WITH_64_BITS_POINTERS)
   using Impl = internal::MTECheckedPtrImpl<
       internal::MTECheckedPtrImplPartitionAllocSupport>;
@@ -999,6 +1087,8 @@
   using Impl = internal::BackupRefPtrImpl</*AllowDangling=*/false>;
 #elif BUILDFLAG(USE_ASAN_BACKUP_REF_PTR)
   using Impl = internal::AsanBackupRefPtrImpl;
+#elif BUILDFLAG(USE_ASAN_UNOWNED_PTR)
+  using Impl = internal::AsanUnownedPtrImpl;
 #elif defined(PA_ENABLE_MTE_CHECKED_PTR_SUPPORT_WITH_64_BITS_POINTERS)
   using Impl = internal::MTECheckedPtrImpl<
       internal::MTECheckedPtrImplPartitionAllocSupport>;
@@ -1060,7 +1150,7 @@
   static_assert(raw_ptr_traits::IsSupportedType<T>::value,
                 "raw_ptr<T> doesn't work with this kind of pointee type T");
 
-#if BUILDFLAG(ENABLE_BACKUP_REF_PTR_SUPPORT)
+#if BUILDFLAG(ENABLE_BACKUP_REF_PTR_SUPPORT) || BUILDFLAG(USE_ASAN_UNOWNED_PTR)
   // BackupRefPtr requires a non-trivial default constructor, destructor, etc.
   constexpr PA_ALWAYS_INLINE raw_ptr() noexcept : wrapped_ptr_(nullptr) {}
 
@@ -1103,7 +1193,8 @@
     wrapped_ptr_ = nullptr;
   }
 
-#else  // BUILDFLAG(ENABLE_BACKUP_REF_PTR_SUPPORT)
+#else  // BUILDFLAG(ENABLE_BACKUP_REF_PTR_SUPPORT) ||
+       // BUILDFLAG(USE_ASAN_UNOWNED_PTR)
 
   // raw_ptr can be trivially default constructed (leaving |wrapped_ptr_|
   // uninitialized).  This is needed for compatibility with raw pointers.
@@ -1124,7 +1215,8 @@
 
   PA_ALWAYS_INLINE ~raw_ptr() noexcept = default;
 
-#endif  // BUILDFLAG(ENABLE_BACKUP_REF_PTR_SUPPORT)
+#endif  // BUILDFLAG(ENABLE_BACKUP_REF_PTR_SUPPORT) ||
+        // BUILDFLAG(USE_ASAN_UNOWNED_PTR)
 
   // Deliberately implicit, because raw_ptr is supposed to resemble raw ptr.
   // NOLINTNEXTLINE(google-explicit-constructor)
diff --git a/base/memory/raw_ptr_unittest.cc b/base/memory/raw_ptr_unittest.cc
index 034232ca..9ce62e1 100644
--- a/base/memory/raw_ptr_unittest.cc
+++ b/base/memory/raw_ptr_unittest.cc
@@ -56,7 +56,8 @@
 static_assert(sizeof(raw_ptr<std::string>) == sizeof(std::string*),
               "raw_ptr shouldn't add memory overhead");
 
-#if !BUILDFLAG(ENABLE_BACKUP_REF_PTR_SUPPORT)
+#if !BUILDFLAG(ENABLE_BACKUP_REF_PTR_SUPPORT) && \
+    !BUILDFLAG(USE_ASAN_UNOWNED_PTR)
 // |is_trivially_copyable| assertion means that arrays/vectors of raw_ptr can
 // be copied by memcpy.
 static_assert(std::is_trivially_copyable<raw_ptr<void>>::value,
diff --git a/base/memory/raw_ref.h b/base/memory/raw_ref.h
index 71eb6b0..669acee 100644
--- a/base/memory/raw_ref.h
+++ b/base/memory/raw_ref.h
@@ -83,6 +83,9 @@
 #if BUILDFLAG(USE_ASAN_BACKUP_REF_PTR)
       std::is_same_v<Impl, internal::AsanBackupRefPtrImpl> ||
 #endif  // BUILDFLAG(USE_ASAN_BACKUP_REF_PTR)
+#if BUILDFLAG(USE_ASAN_UNOWNED_PTR)
+      std::is_same_v<Impl, internal::AsanUnownedPtrImpl> ||
+#endif  // BUILDFLAG(USE_ASAN_UNOWNED_PTR)
       std::is_same_v<Impl, internal::RawPtrNoOpImpl>;
 
  public:
diff --git a/base/win/sid.cc b/base/win/sid.cc
index 4e48268c..f0904ac3 100644
--- a/base/win/sid.cc
+++ b/base/win/sid.cc
@@ -7,25 +7,56 @@
 #include <windows.h>
 
 #include <sddl.h>
+#include <stdint.h>
 #include <stdlib.h>
 
+#include <algorithm>
 #include <iterator>
+#include <map>
 #include <utility>
 
 #include "base/check.h"
-#include "base/notreached.h"
+#include "base/no_destructor.h"
 #include "base/rand_util.h"
+#include "base/strings/string_util_win.h"
 #include "base/win/scoped_handle.h"
 #include "base/win/scoped_localalloc.h"
 #include "base/win/windows_version.h"
+#include "third_party/boringssl/src/include/openssl/crypto.h"
+#include "third_party/boringssl/src/include/openssl/sha.h"
 
-namespace base {
-namespace win {
+namespace base::win {
 
 namespace {
 
-absl::optional<long>  // NOLINT(runtime/int)
-WellKnownCapabilityToRid(WellKnownCapability capability) {
+template <typename Iterator>
+Sid FromSubAuthorities(const SID_IDENTIFIER_AUTHORITY& identifier_authority,
+                       size_t sub_authority_count,
+                       Iterator sub_authorities) {
+  DCHECK(sub_authority_count <= SID_MAX_SUB_AUTHORITIES);
+  BYTE sid_buffer[SECURITY_MAX_SID_SIZE];
+  SID* sid = reinterpret_cast<SID*>(sid_buffer);
+  sid->Revision = SID_REVISION;
+  sid->SubAuthorityCount = static_cast<UCHAR>(sub_authority_count);
+  sid->IdentifierAuthority = identifier_authority;
+  for (size_t index = 0; index < sub_authority_count; ++index) {
+    sid->SubAuthority[index] = static_cast<DWORD>(*sub_authorities++);
+  }
+  DCHECK(::IsValidSid(sid));
+  return *Sid::FromPSID(sid);
+}
+
+Sid FromSubAuthorities(const SID_IDENTIFIER_AUTHORITY& identifier_authority,
+                       std::initializer_list<int32_t> sub_authorities) {
+  return FromSubAuthorities(identifier_authority, sub_authorities.size(),
+                            sub_authorities.begin());
+}
+
+Sid FromNtAuthority(std::initializer_list<int32_t> sub_authorities) {
+  return FromSubAuthorities(SECURITY_NT_AUTHORITY, sub_authorities);
+}
+
+int32_t WellKnownCapabilityToRid(WellKnownCapability capability) {
   switch (capability) {
     case WellKnownCapability::kInternetClient:
       return SECURITY_CAPABILITY_INTERNET_CLIENT;
@@ -52,97 +83,6 @@
     case WellKnownCapability::kContacts:
       return SECURITY_CAPABILITY_CONTACTS;
   }
-  return absl::nullopt;
-}
-
-absl::optional<WELL_KNOWN_SID_TYPE> WellKnownSidToEnum(WellKnownSid sid) {
-  switch (sid) {
-    case WellKnownSid::kNull:
-      return WinNullSid;
-    case WellKnownSid::kWorld:
-      return WinWorldSid;
-    case WellKnownSid::kCreatorOwner:
-      return WinCreatorOwnerSid;
-    case WellKnownSid::kNetwork:
-      return WinNetworkSid;
-    case WellKnownSid::kBatch:
-      return WinBatchSid;
-    case WellKnownSid::kInteractive:
-      return WinInteractiveSid;
-    case WellKnownSid::kService:
-      return WinServiceSid;
-    case WellKnownSid::kAnonymous:
-      return WinAnonymousSid;
-    case WellKnownSid::kSelf:
-      return WinSelfSid;
-    case WellKnownSid::kAuthenticatedUser:
-      return WinAuthenticatedUserSid;
-    case WellKnownSid::kRestricted:
-      return WinRestrictedCodeSid;
-    case WellKnownSid::kLocalSystem:
-      return WinLocalSystemSid;
-    case WellKnownSid::kLocalService:
-      return WinLocalServiceSid;
-    case WellKnownSid::kNetworkService:
-      return WinNetworkServiceSid;
-    case WellKnownSid::kBuiltinAdministrators:
-      return WinBuiltinAdministratorsSid;
-    case WellKnownSid::kBuiltinUsers:
-      return WinBuiltinUsersSid;
-    case WellKnownSid::kBuiltinGuests:
-      return WinBuiltinGuestsSid;
-    case WellKnownSid::kUntrustedLabel:
-      return WinUntrustedLabelSid;
-    case WellKnownSid::kLowLabel:
-      return WinLowLabelSid;
-    case WellKnownSid::kMediumLabel:
-      return WinMediumLabelSid;
-    case WellKnownSid::kHighLabel:
-      return WinHighLabelSid;
-    case WellKnownSid::kSystemLabel:
-      return WinSystemLabelSid;
-    case WellKnownSid::kWriteRestricted:
-      return WinWriteRestrictedCodeSid;
-    case WellKnownSid::kCreatorOwnerRights:
-      return WinCreatorOwnerRightsSid;
-    case WellKnownSid::kAllApplicationPackages:
-      return WinBuiltinAnyPackageSid;
-    case WellKnownSid::kAllRestrictedApplicationPackages:
-      // This should be handled by FromKnownSid.
-      NOTREACHED();
-      break;
-  }
-  return absl::nullopt;
-}
-
-absl::optional<Sid> FromSubAuthorities(
-    PSID_IDENTIFIER_AUTHORITY identifier_authority,
-    BYTE sub_authority_count,
-    PDWORD sub_authorities) {
-  BYTE sid[SECURITY_MAX_SID_SIZE];
-  if (!::InitializeSid(sid, identifier_authority, sub_authority_count))
-    return absl::nullopt;
-
-  for (DWORD index = 0; index < sub_authority_count; ++index) {
-    PDWORD sub_authority = ::GetSidSubAuthority(sid, index);
-    *sub_authority = sub_authorities[index];
-  }
-  return Sid::FromPSID(sid);
-}
-
-template <typename T>
-absl::optional<std::vector<Sid>> FromVector(
-    const std::vector<T>& values,
-    absl::optional<Sid> (*create_sid)(T)) {
-  std::vector<Sid> converted_sids;
-  converted_sids.reserve(values.size());
-  for (T value : values) {
-    auto sid = create_sid(value);
-    if (!sid)
-      return absl::nullopt;
-    converted_sids.push_back(std::move(*sid));
-  }
-  return converted_sids;
 }
 
 }  // namespace
@@ -154,94 +94,128 @@
 }
 
 absl::optional<Sid> Sid::FromKnownCapability(WellKnownCapability capability) {
-  absl::optional<long> capability_rid =  // NOLINT(runtime/int)
-      WellKnownCapabilityToRid(capability);
-  if (!capability_rid)
-    return absl::nullopt;
-  SID_IDENTIFIER_AUTHORITY capability_authority = {
-      SECURITY_APP_PACKAGE_AUTHORITY};
-  DWORD sub_authorities[] = {SECURITY_CAPABILITY_BASE_RID,
-                             static_cast<DWORD>(*capability_rid)};
-  return FromSubAuthorities(&capability_authority, std::size(sub_authorities),
-                            sub_authorities);
+  int32_t capability_rid = WellKnownCapabilityToRid(capability);
+  return FromSubAuthorities(SECURITY_APP_PACKAGE_AUTHORITY,
+                            {SECURITY_CAPABILITY_BASE_RID, capability_rid});
 }
 
 absl::optional<Sid> Sid::FromNamedCapability(const wchar_t* capability_name) {
-  DCHECK_GE(GetVersion(), Version::WIN10);
-
-  if (!capability_name || !*capability_name)
-    return absl::nullopt;
-
-  typedef decltype(
-      ::DeriveCapabilitySidsFromName)* DeriveCapabilitySidsFromNameFunc;
-  static const DeriveCapabilitySidsFromNameFunc derive_capability_sids =
-      []() -> DeriveCapabilitySidsFromNameFunc {
-    HMODULE module = GetModuleHandle(L"api-ms-win-security-base-l1-2-2.dll");
-    if (!module)
-      return nullptr;
-
-    return reinterpret_cast<DeriveCapabilitySidsFromNameFunc>(
-        ::GetProcAddress(module, "DeriveCapabilitySidsFromName"));
-  }();
-  if (!derive_capability_sids)
-    return absl::nullopt;
-
-  // Pre-reserve some space for SID deleters.
-  std::vector<ScopedLocalAlloc> deleter_list;
-  deleter_list.reserve(16);
-
-  PSID* capability_groups = nullptr;
-  DWORD capability_group_count = 0;
-  PSID* capability_sids = nullptr;
-  DWORD capability_sid_count = 0;
-
-  if (!derive_capability_sids(capability_name, &capability_groups,
-                              &capability_group_count, &capability_sids,
-                              &capability_sid_count)) {
+  if (!capability_name || !*capability_name) {
     return absl::nullopt;
   }
+  static const base::NoDestructor<std::map<std::wstring, WellKnownCapability>>
+      known_capabilities(
+          {{L"INTERNETCLIENT", WellKnownCapability::kInternetClient},
+           {L"INTERNETCLIENTSERVER",
+            WellKnownCapability::kInternetClientServer},
+           {L"PRIVATENETWORKCLIENTSERVER",
+            WellKnownCapability::kPrivateNetworkClientServer},
+           {L"PICTURESLIBRARY", WellKnownCapability::kPicturesLibrary},
+           {L"VIDEOSLIBRARY", WellKnownCapability::kVideosLibrary},
+           {L"MUSICLIBRARY", WellKnownCapability::kMusicLibrary},
+           {L"DOCUMENTSLIBRARY", WellKnownCapability::kDocumentsLibrary},
+           {L"ENTERPRISEAUTHENTICATION",
+            WellKnownCapability::kEnterpriseAuthentication},
+           {L"SHAREDUSERCERTIFICATES",
+            WellKnownCapability::kSharedUserCertificates},
+           {L"REMOVABLESTORAGE", WellKnownCapability::kRemovableStorage},
+           {L"APPOINTMENTS", WellKnownCapability::kAppointments},
+           {L"CONTACTS", WellKnownCapability::kContacts}});
 
-  deleter_list.emplace_back(capability_groups);
-  deleter_list.emplace_back(capability_sids);
-
-  for (DWORD i = 0; i < capability_group_count; ++i) {
-    deleter_list.emplace_back(capability_groups[i]);
+  std::wstring cap_upper = base::ToUpperASCII(capability_name);
+  auto known_cap = known_capabilities->find(cap_upper);
+  if (known_cap != known_capabilities->end()) {
+    return FromKnownCapability(known_cap->second);
   }
-  for (DWORD i = 0; i < capability_sid_count; ++i) {
-    deleter_list.emplace_back(capability_sids[i]);
-  }
+  CRYPTO_library_init();
+  static_assert((SHA256_DIGEST_LENGTH / sizeof(DWORD)) ==
+                SECURITY_APP_PACKAGE_RID_COUNT);
+  DWORD rids[(SHA256_DIGEST_LENGTH / sizeof(DWORD)) + 2];
+  rids[0] = SECURITY_CAPABILITY_BASE_RID;
+  rids[1] = SECURITY_CAPABILITY_APP_RID;
 
-  if (capability_sid_count < 1)
-    return absl::nullopt;
-
-  return FromPSID(capability_sids[0]);
+  SHA256(reinterpret_cast<const uint8_t*>(cap_upper.c_str()),
+         cap_upper.size() * sizeof(wchar_t),
+         reinterpret_cast<uint8_t*>(&rids[2]));
+  return FromSubAuthorities(SECURITY_APP_PACKAGE_AUTHORITY, std::size(rids),
+                            rids);
 }
 
 absl::optional<Sid> Sid::FromKnownSid(WellKnownSid type) {
-  if (type == WellKnownSid::kAllRestrictedApplicationPackages) {
-    SID_IDENTIFIER_AUTHORITY package_authority = {
-        SECURITY_APP_PACKAGE_AUTHORITY};
-    DWORD sub_authorities[] = {SECURITY_APP_PACKAGE_BASE_RID,
-                               SECURITY_BUILTIN_PACKAGE_ANY_RESTRICTED_PACKAGE};
-    return FromSubAuthorities(&package_authority, 2, sub_authorities);
+  switch (type) {
+    case WellKnownSid::kNull:
+      return FromSubAuthorities(SECURITY_NULL_SID_AUTHORITY,
+                                {SECURITY_NULL_RID});
+    case WellKnownSid::kWorld:
+      return FromSubAuthorities(SECURITY_WORLD_SID_AUTHORITY,
+                                {SECURITY_WORLD_RID});
+    case WellKnownSid::kCreatorOwner:
+      return FromSubAuthorities(SECURITY_CREATOR_SID_AUTHORITY,
+                                {SECURITY_CREATOR_OWNER_RID});
+    case WellKnownSid::kCreatorOwnerRights:
+      return FromSubAuthorities(SECURITY_CREATOR_SID_AUTHORITY,
+                                {SECURITY_CREATOR_OWNER_RIGHTS_RID});
+    case WellKnownSid::kNetwork:
+      return FromNtAuthority({SECURITY_NETWORK_RID});
+    case WellKnownSid::kBatch:
+      return FromNtAuthority({SECURITY_BATCH_RID});
+    case WellKnownSid::kInteractive:
+      return FromNtAuthority({SECURITY_INTERACTIVE_RID});
+    case WellKnownSid::kService:
+      return FromNtAuthority({SECURITY_SERVICE_RID});
+    case WellKnownSid::kAnonymous:
+      return FromNtAuthority({SECURITY_ANONYMOUS_LOGON_RID});
+    case WellKnownSid::kSelf:
+      return FromNtAuthority({SECURITY_PRINCIPAL_SELF_RID});
+    case WellKnownSid::kAuthenticatedUser:
+      return FromNtAuthority({SECURITY_AUTHENTICATED_USER_RID});
+    case WellKnownSid::kRestricted:
+      return FromNtAuthority({SECURITY_RESTRICTED_CODE_RID});
+    case WellKnownSid::kWriteRestricted:
+      return FromNtAuthority({SECURITY_WRITE_RESTRICTED_CODE_RID});
+    case WellKnownSid::kLocalSystem:
+      return FromNtAuthority({SECURITY_LOCAL_SYSTEM_RID});
+    case WellKnownSid::kLocalService:
+      return FromNtAuthority({SECURITY_LOCAL_SERVICE_RID});
+    case WellKnownSid::kNetworkService:
+      return FromNtAuthority({SECURITY_NETWORK_SERVICE_RID});
+    case WellKnownSid::kBuiltinAdministrators:
+      return FromNtAuthority(
+          {SECURITY_BUILTIN_DOMAIN_RID, DOMAIN_ALIAS_RID_ADMINS});
+    case WellKnownSid::kBuiltinUsers:
+      return FromNtAuthority(
+          {SECURITY_BUILTIN_DOMAIN_RID, DOMAIN_ALIAS_RID_USERS});
+    case WellKnownSid::kBuiltinGuests:
+      return FromNtAuthority(
+          {SECURITY_BUILTIN_DOMAIN_RID, DOMAIN_ALIAS_RID_GUESTS});
+    case WellKnownSid::kUntrustedLabel:
+      return FromIntegrityLevel(SECURITY_MANDATORY_UNTRUSTED_RID);
+    case WellKnownSid::kLowLabel:
+      return FromIntegrityLevel(SECURITY_MANDATORY_LOW_RID);
+    case WellKnownSid::kMediumLabel:
+      return FromIntegrityLevel(SECURITY_MANDATORY_MEDIUM_RID);
+    case WellKnownSid::kHighLabel:
+      return FromIntegrityLevel(SECURITY_MANDATORY_HIGH_RID);
+    case WellKnownSid::kSystemLabel:
+      return FromIntegrityLevel(SECURITY_MANDATORY_SYSTEM_RID);
+    case WellKnownSid::kAllApplicationPackages:
+      return FromSubAuthorities(SECURITY_APP_PACKAGE_AUTHORITY,
+                                {SECURITY_APP_PACKAGE_BASE_RID,
+                                 SECURITY_BUILTIN_PACKAGE_ANY_PACKAGE});
+    case WellKnownSid::kAllRestrictedApplicationPackages:
+      return FromSubAuthorities(
+          SECURITY_APP_PACKAGE_AUTHORITY,
+          {SECURITY_APP_PACKAGE_BASE_RID,
+           SECURITY_BUILTIN_PACKAGE_ANY_RESTRICTED_PACKAGE});
   }
-
-  BYTE sid[SECURITY_MAX_SID_SIZE];
-  DWORD size_sid = SECURITY_MAX_SID_SIZE;
-  absl::optional<WELL_KNOWN_SID_TYPE> known_sid = WellKnownSidToEnum(type);
-  if (!known_sid)
-    return absl::nullopt;
-  if (!::CreateWellKnownSid(*known_sid, nullptr, sid, &size_sid))
-    return absl::nullopt;
-
-  return Sid(sid, size_sid);
 }
 
 absl::optional<Sid> Sid::FromSddlString(const wchar_t* sddl_sid) {
   PSID psid = nullptr;
   if (!::ConvertStringSidToSid(sddl_sid, &psid))
     return absl::nullopt;
-  return FromPSID(TakeLocalAlloc(psid).get());
+  auto psid_alloc = TakeLocalAlloc(psid);
+  return FromPSID(psid_alloc.get());
 }
 
 absl::optional<Sid> Sid::FromPSID(PSID sid) {
@@ -252,37 +226,62 @@
 }
 
 absl::optional<Sid> Sid::GenerateRandomSid() {
-  SID_IDENTIFIER_AUTHORITY package_authority = {SECURITY_NULL_SID_AUTHORITY};
   DWORD sub_authorities[4] = {};
   RandBytes(&sub_authorities, sizeof(sub_authorities));
-  return FromSubAuthorities(&package_authority, _countof(sub_authorities),
-                            sub_authorities);
+  return FromSubAuthorities(SECURITY_NULL_SID_AUTHORITY,
+                            std::size(sub_authorities), sub_authorities);
 }
 
 absl::optional<Sid> Sid::FromIntegrityLevel(DWORD integrity_level) {
-  SID_IDENTIFIER_AUTHORITY package_authority = {
-      SECURITY_MANDATORY_LABEL_AUTHORITY};
-  return FromSubAuthorities(&package_authority, 1, &integrity_level);
+  return FromSubAuthorities(SECURITY_MANDATORY_LABEL_AUTHORITY, 1,
+                            &integrity_level);
 }
 
 absl::optional<std::vector<Sid>> Sid::FromSddlStringVector(
     const std::vector<const wchar_t*>& sddl_sids) {
-  return FromVector(sddl_sids, Sid::FromSddlString);
+  std::vector<Sid> converted_sids;
+  converted_sids.reserve(sddl_sids.size());
+  for (const wchar_t* sddl_sid : sddl_sids) {
+    absl::optional<Sid> sid = FromSddlString(sddl_sid);
+    if (!sid)
+      return absl::nullopt;
+    converted_sids.push_back(std::move(*sid));
+  }
+  return converted_sids;
 }
 
 absl::optional<std::vector<Sid>> Sid::FromNamedCapabilityVector(
     const std::vector<const wchar_t*>& capability_names) {
-  return FromVector(capability_names, Sid::FromNamedCapability);
+  std::vector<Sid> converted_sids;
+  converted_sids.reserve(capability_names.size());
+  for (const wchar_t* name : capability_names) {
+    absl::optional<Sid> sid = FromNamedCapability(name);
+    if (!sid)
+      return absl::nullopt;
+    converted_sids.push_back(std::move(*sid));
+  }
+  return converted_sids;
 }
 
 absl::optional<std::vector<Sid>> Sid::FromKnownCapabilityVector(
     const std::vector<WellKnownCapability>& capabilities) {
-  return FromVector(capabilities, Sid::FromKnownCapability);
+  std::vector<Sid> sids;
+  std::transform(capabilities.cbegin(), capabilities.cend(),
+                 std::back_inserter(sids),
+                 [](WellKnownCapability known_cap) -> Sid {
+                   return *Sid::FromKnownCapability(known_cap);
+                 });
+  return sids;
 }
 
 absl::optional<std::vector<Sid>> Sid::FromKnownSidVector(
-    const std::vector<WellKnownSid>& sids) {
-  return FromVector(sids, Sid::FromKnownSid);
+    const std::vector<WellKnownSid>& known_sids) {
+  std::vector<Sid> sids;
+  std::transform(known_sids.cbegin(), known_sids.cend(),
+                 std::back_inserter(sids), [](WellKnownSid known_sid) -> Sid {
+                   return *Sid::FromKnownSid(known_sid);
+                 });
+  return sids;
 }
 
 Sid::Sid(Sid&& sid) = default;
@@ -299,7 +298,8 @@
   LPWSTR sid = nullptr;
   if (!::ConvertSidToStringSid(GetPSID(), &sid))
     return absl::nullopt;
-  return TakeLocalAlloc(sid).get();
+  auto sid_ptr = TakeLocalAlloc(sid);
+  return sid_ptr.get();
 }
 
 Sid Sid::Clone() const {
@@ -318,5 +318,4 @@
   return !(operator==(sid));
 }
 
-}  // namespace win
-}  // namespace base
+}  // namespace base::win
diff --git a/base/win/sid_unittest.cc b/base/win/sid_unittest.cc
index 70bd201..98ec8b2b 100644
--- a/base/win/sid_unittest.cc
+++ b/base/win/sid_unittest.cc
@@ -74,16 +74,49 @@
   return TestSidVector(Sid::FromSddlStringVector(sddl), sddl);
 }
 
+bool EqualNamedCapSid(const Sid& sid, const std::wstring& capability_name) {
+  typedef decltype(::DeriveCapabilitySidsFromName)*
+      DeriveCapabilitySidsFromNameFunc;
+  static const DeriveCapabilitySidsFromNameFunc derive_capability_sids =
+      []() -> DeriveCapabilitySidsFromNameFunc {
+    HMODULE module = GetModuleHandle(L"api-ms-win-security-base-l1-2-2.dll");
+    CHECK(module);
+    return reinterpret_cast<DeriveCapabilitySidsFromNameFunc>(
+        ::GetProcAddress(module, "DeriveCapabilitySidsFromName"));
+  }();
+  CHECK(derive_capability_sids);
+
+  // Pre-reserve some space for SID deleters.
+  std::vector<base::win::ScopedLocalAlloc> deleter_list;
+  deleter_list.reserve(16);
+
+  PSID* capability_groups = nullptr;
+  DWORD capability_group_count = 0;
+  PSID* capability_sids = nullptr;
+  DWORD capability_sid_count = 0;
+
+  CHECK(derive_capability_sids(capability_name.c_str(), &capability_groups,
+                               &capability_group_count, &capability_sids,
+                               &capability_sid_count));
+  deleter_list.emplace_back(capability_groups);
+  deleter_list.emplace_back(capability_sids);
+
+  for (DWORD i = 0; i < capability_group_count; ++i) {
+    deleter_list.emplace_back(capability_groups[i]);
+  }
+  for (DWORD i = 0; i < capability_sid_count; ++i) {
+    deleter_list.emplace_back(capability_sids[i]);
+  }
+
+  CHECK_GE(capability_sid_count, 1U);
+  return sid.Equal(capability_sids[0]);
+}
+
 struct KnownCapabilityTestEntry {
   WellKnownCapability capability;
   const wchar_t* sddl_sid;
 };
 
-struct NamedCapabilityTestEntry {
-  const wchar_t* capability_name;
-  const wchar_t* sddl_sid;
-};
-
 struct KnownSidTestEntry {
   WellKnownSid sid;
   WELL_KNOWN_SID_TYPE well_known_sid;
@@ -135,28 +168,25 @@
 }
 
 TEST(SidTest, NamedCapability) {
-  if (GetVersion() < Version::WIN10_RS2)
-    return;
+  const std::wstring capabilities[] = {L"InternetClient",
+                                       L"InternetClientServer",
+                                       L"PrivateNetworkClientServer",
+                                       L"PicturesLibrary",
+                                       L"VideosLibrary",
+                                       L"MusicLibrary",
+                                       L"DocumentsLibrary",
+                                       L"EnterpriseAuthentication",
+                                       L"SharedUserCertificates",
+                                       L"RemovableStorage",
+                                       L"Appointments",
+                                       L"Contacts",
+                                       L"registryRead",
+                                       L"lpacCryptoServices"};
 
-  EXPECT_FALSE(Sid::FromNamedCapability(nullptr));
-  EXPECT_FALSE(Sid::FromNamedCapability(L""));
-
-  const NamedCapabilityTestEntry capabilities[] = {
-      {L"internetClient", L"S-1-15-3-1"},
-      {L"internetClientServer", L"S-1-15-3-2"},
-      {L"registryRead",
-       L"S-1-15-3-1024-1065365936-1281604716-3511738428-"
-       "1654721687-432734479-3232135806-4053264122-3456934681"},
-      {L"lpacCryptoServices",
-       L"S-1-15-3-1024-3203351429-2120443784-2872670797-"
-       "1918958302-2829055647-4275794519-765664414-2751773334"},
-      {L"enterpriseAuthentication", L"S-1-15-3-8"},
-      {L"privateNetworkClientServer", L"S-1-15-3-3"}};
-
-  for (auto capability : capabilities) {
-    EXPECT_TRUE(EqualSid(Sid::FromNamedCapability(capability.capability_name),
-                         capability.sddl_sid))
-        << "Named Capability: " << capability.sddl_sid;
+  for (const std::wstring& capability : capabilities) {
+    EXPECT_TRUE(EqualNamedCapSid(*Sid::FromNamedCapability(capability.c_str()),
+                                 capability))
+        << "Named Capability: " << capability;
   }
 }
 
@@ -254,28 +284,24 @@
 }
 
 TEST(SidTest, FromNamedCapabilityVector) {
-  if (GetVersion() < Version::WIN10_RS2)
-    return;
-  std::vector<const wchar_t*> capabilities = {L"internetClient",
-                                              L"internetClientServer",
+  std::vector<const wchar_t*> capabilities = {L"InternetClient",
+                                              L"InternetClientServer",
+                                              L"PrivateNetworkClientServer",
+                                              L"PicturesLibrary",
+                                              L"VideosLibrary",
+                                              L"MusicLibrary",
+                                              L"DocumentsLibrary",
+                                              L"EnterpriseAuthentication",
+                                              L"SharedUserCertificates",
+                                              L"RemovableStorage",
+                                              L"Appointments",
+                                              L"Contacts",
                                               L"registryRead",
-                                              L"lpacCryptoServices",
-                                              L"enterpriseAuthentication",
-                                              L"privateNetworkClientServer"};
-  std::vector<const wchar_t*> sddl_caps = {
-      L"S-1-15-3-1",
-      L"S-1-15-3-2",
-      L"S-1-15-3-1024-1065365936-1281604716-3511738428-1654721687-432734479-"
-      L"3232135806-4053264122-3456934681",
-      L"S-1-15-3-1024-3203351429-2120443784-2872670797-1918958302-2829055647-"
-      L"4275794519-765664414-2751773334",
-      L"S-1-15-3-8",
-      L"S-1-15-3-3"};
-  ASSERT_TRUE(
-      TestSidVector(Sid::FromNamedCapabilityVector(capabilities), sddl_caps));
-  ASSERT_FALSE(Sid::FromNamedCapabilityVector({L""}));
-  ASSERT_FALSE(Sid::FromNamedCapabilityVector({L"abc", nullptr}));
-  ASSERT_TRUE(Sid::FromNamedCapabilityVector({}));
+                                              L"lpacCryptoServices"};
+
+  ASSERT_TRUE(ranges::equal(*Sid::FromNamedCapabilityVector(capabilities),
+                            capabilities, EqualNamedCapSid));
+  EXPECT_EQ(Sid::FromNamedCapabilityVector({})->size(), 0U);
 }
 
 TEST(SidTest, FromKnownCapabilityVector) {
diff --git a/buildtools/deps_revisions.gni b/buildtools/deps_revisions.gni
index 44e04b9..249ad4f2 100644
--- a/buildtools/deps_revisions.gni
+++ b/buildtools/deps_revisions.gni
@@ -5,5 +5,5 @@
 declare_args() {
   # Used to cause full rebuilds on libc++ rolls. This should be kept in sync
   # with the libcxx_revision vars in //DEPS.
-  libcxx_revision = "19ffb9c00636bcabb6c6ce76ccaa4370583d3649"
+  libcxx_revision = "2fc3d704672fbd3e85fad8492d39e02d49412891"
 }
diff --git a/chrome/android/chrome_java_resources.gni b/chrome/android/chrome_java_resources.gni
index 4978bf9..e1cfd7c 100644
--- a/chrome/android/chrome_java_resources.gni
+++ b/chrome/android/chrome_java_resources.gni
@@ -366,6 +366,8 @@
   "java/res/drawable/adaptive_toolbar_preference_header.xml",
   "java/res/drawable/arrow_down.xml",
   "java/res/drawable/arrow_up.xml",
+  "java/res/drawable/bg_circle_new_tab_button_detached.xml",
+  "java/res/drawable/bg_circle_new_tab_button_folio.xml",
   "java/res/drawable/bg_tabstrip_tab_divider.xml",
   "java/res/drawable/bg_white_dialog.xml",
   "java/res/drawable/bookmark_save_flow_ripple.xml",
diff --git a/chrome/android/chrome_test_java_sources.gni b/chrome/android/chrome_test_java_sources.gni
index e507315..63d7231 100644
--- a/chrome/android/chrome_test_java_sources.gni
+++ b/chrome/android/chrome_test_java_sources.gni
@@ -294,7 +294,6 @@
   "javatests/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceRecognitionHandlerTest.java",
   "javatests/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchConsentBottomSheetTest.java",
   "javatests/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchConsentControllerTest.java",
-  "javatests/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchConsentModalTest.java",
   "javatests/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchConsentUiRenderTest.java",
   "javatests/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchServiceRenderTest.java",
   "javatests/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchServiceTest.java",
diff --git a/chrome/android/features/start_surface/java/src/org/chromium/chrome/features/start_surface/StartSurface.java b/chrome/android/features/start_surface/java/src/org/chromium/chrome/features/start_surface/StartSurface.java
index 63282f9..17d91d5 100644
--- a/chrome/android/features/start_surface/java/src/org/chromium/chrome/features/start_surface/StartSurface.java
+++ b/chrome/android/features/start_surface/java/src/org/chromium/chrome/features/start_surface/StartSurface.java
@@ -184,6 +184,12 @@
     void setLaunchOrigin(@NewTabPageLaunchOrigin int launchOrigin);
 
     /**
+     * Resets the scroll position. This is called when Start surface is showing but not via back
+     * operations.
+     */
+    void resetScrollPosition();
+
+    /**
      * Called by the TabSwitcherLayout when the system back button is pressed.
      * @return Whether or not the TabSwitcher consumed the event.
      */
diff --git a/chrome/android/features/start_surface/java/src/org/chromium/chrome/features/start_surface/StartSurfaceCoordinator.java b/chrome/android/features/start_surface/java/src/org/chromium/chrome/features/start_surface/StartSurfaceCoordinator.java
index f51bf5b..d9760264 100644
--- a/chrome/android/features/start_surface/java/src/org/chromium/chrome/features/start_surface/StartSurfaceCoordinator.java
+++ b/chrome/android/features/start_surface/java/src/org/chromium/chrome/features/start_surface/StartSurfaceCoordinator.java
@@ -508,6 +508,11 @@
     }
 
     @Override
+    public void resetScrollPosition() {
+        mStartSurfaceMediator.resetScrollPosition();
+    }
+
+    @Override
     public boolean onBackPressed() {
         return mStartSurfaceMediator.onBackPressed();
     }
diff --git a/chrome/android/features/start_surface/java/src/org/chromium/chrome/features/start_surface/StartSurfaceMediator.java b/chrome/android/features/start_surface/java/src/org/chromium/chrome/features/start_surface/StartSurfaceMediator.java
index 5e81873..a3a6aeb 100644
--- a/chrome/android/features/start_surface/java/src/org/chromium/chrome/features/start_surface/StartSurfaceMediator.java
+++ b/chrome/android/features/start_surface/java/src/org/chromium/chrome/features/start_surface/StartSurfaceMediator.java
@@ -512,14 +512,6 @@
         mIsHomepageShown = true;
         notifyShowStateChange();
 
-        // TODO(crbug.com/1347089): When entering the Start surface by tapping back button or other
-        // back gestures, we shouldn't reset the scrolling position. Maybe we could add a boolean
-        // |mResetPosition| and set it as false only when this method is called because of back
-        // actions.
-        mPropertyModel.set(RESET_TASK_SURFACE_HEADER_SCROLL_POSITION, true);
-        mPropertyModel.set(RESET_FEED_SURFACE_SCROLL_POSITION, true);
-        StartSurfaceUserData.getInstance().saveFeedInstanceState(null);
-
         mIsIncognito = mTabModelSelector.isIncognitoSelected();
         mPropertyModel.set(IS_INCOGNITO, mIsIncognito);
         setMVTilesVisibility(!mIsIncognito);
@@ -680,6 +672,14 @@
         }
     }
 
+    void resetScrollPosition() {
+        if (mPropertyModel == null) return;
+
+        mPropertyModel.set(RESET_TASK_SURFACE_HEADER_SCROLL_POSITION, true);
+        mPropertyModel.set(RESET_FEED_SURFACE_SCROLL_POSITION, true);
+        StartSurfaceUserData.getInstance().saveFeedInstanceState(null);
+    }
+
     // TODO(crbug.com/1115757): After crrev.com/c/2315823, Overview state and Startsurface state are
     // two different things, audit the wording usage and see if we can rename this method to
     // setStartSurfaceStateInternal.
@@ -687,10 +687,7 @@
         if (mStartSurfaceState == StartSurfaceState.SHOWING_HOMEPAGE) {
             // When entering the Start surface by tapping home button or new tab page, we need to
             // reset the scrolling position.
-            mPropertyModel.set(RESET_TASK_SURFACE_HEADER_SCROLL_POSITION, true);
-            mPropertyModel.set(RESET_FEED_SURFACE_SCROLL_POSITION, true);
-            StartSurfaceUserData.getInstance().saveFeedInstanceState(null);
-
+            resetScrollPosition();
         } else if (mStartSurfaceState == StartSurfaceState.SHOWING_TABSWITCHER) {
             maybeDestroyFeedPlaceholder();
             // Set secondary surface visible to make sure tab list recyclerview is updated in time
diff --git a/chrome/android/features/start_surface/javatests/src/org/chromium/chrome/features/start_surface/StartSurfaceTest.java b/chrome/android/features/start_surface/javatests/src/org/chromium/chrome/features/start_surface/StartSurfaceTest.java
index 00bf6708..6db1704f 100644
--- a/chrome/android/features/start_surface/javatests/src/org/chromium/chrome/features/start_surface/StartSurfaceTest.java
+++ b/chrome/android/features/start_surface/javatests/src/org/chromium/chrome/features/start_surface/StartSurfaceTest.java
@@ -667,8 +667,6 @@
     @Feature({"StartSurface"})
     @CommandLineFlags.Add({START_SURFACE_TEST_SINGLE_ENABLED_PARAMS})
     public void testShow_SingleAsHomepage_ResetScrollPosition() {
-        assumeTrue("https://crbug.com/1385547 - Disable for NoInstant.", mUseInstantStart);
-        
         if (!mImmediateReturn) {
             StartSurfaceTestUtils.pressHomePageButton(mActivityTestRule.getActivity());
         }
@@ -699,6 +697,40 @@
     @Test
     @MediumTest
     @Feature({"StartSurface"})
+    @EnableFeatures(ChromeFeatureList.START_SURFACE_REFACTOR)
+    @CommandLineFlags.Add({START_SURFACE_TEST_SINGLE_ENABLED_PARAMS})
+    public void testShow_SingleAsHomepage_DoNotResetScrollPositionFromBack() {
+        assumeTrue(mImmediateReturn);
+
+        ChromeTabbedActivity cta = mActivityTestRule.getActivity();
+        StartSurfaceTestUtils.waitForStartSurfaceVisible(
+                mLayoutChangedCallbackHelper, mCurrentlyActiveLayout, cta);
+        TabUiTestHelper.verifyTabModelTabCount(cta, 1, 0);
+
+        // Scroll the toolbar.
+        StartSurfaceTestUtils.scrollToolbarAndVerify(cta);
+        AppBarLayout taskSurfaceHeader = cta.findViewById(R.id.task_surface_header);
+        assertNotEquals(taskSurfaceHeader.getBottom(), taskSurfaceHeader.getHeight());
+
+        // Verifies the case of scrolling Start surface ->  MV tile -> tapping back ->
+        // Start surface. The Start surface should not reset its scroll position.
+        StartSurfaceTestUtils.launchFirstMVTile(cta, 1);
+        Assert.assertEquals("The launched tab should have the launch type FROM_START_SURFACE",
+                TabLaunchType.FROM_START_SURFACE,
+                cta.getActivityTabProvider().get().getLaunchType());
+        StartSurfaceTestUtils.pressBack(mActivityTestRule);
+        // Back gesture on the tab should take us back to the start surface homepage.
+        StartSurfaceTestUtils.waitForStartSurfaceVisible(
+                mLayoutChangedCallbackHelper, mCurrentlyActiveLayout, cta);
+
+        // The Start surface should not reset its scroll position.
+        CriteriaHelper.pollInstrumentationThread(
+                () -> taskSurfaceHeader.getBottom() != taskSurfaceHeader.getHeight());
+    }
+
+    @Test
+    @MediumTest
+    @Feature({"StartSurface"})
     @CommandLineFlags.Add({START_SURFACE_TEST_SINGLE_ENABLED_PARAMS})
     public void singleAsHomepage_PressHomeButtonWillKeepTab() {
         if (!mImmediateReturn) {
diff --git a/chrome/android/features/tab_ui/java/src/org/chromium/chrome/browser/tasks/tab_management/TabUiThemeProvider.java b/chrome/android/features/tab_ui/java/src/org/chromium/chrome/browser/tasks/tab_management/TabUiThemeProvider.java
index 6b60a8e..89486d6 100644
--- a/chrome/android/features/tab_ui/java/src/org/chromium/chrome/browser/tasks/tab_management/TabUiThemeProvider.java
+++ b/chrome/android/features/tab_ui/java/src/org/chromium/chrome/browser/tasks/tab_management/TabUiThemeProvider.java
@@ -60,6 +60,15 @@
     }
 
     /**
+     * Returns the semantic color value that corresponds to colorPrimaryContainer.
+     *
+     * @param context {@link Context} used to retrieve color.
+     */
+    public static @ColorInt int getDefaultContainerColor(Context context) {
+        return MaterialColors.getColor(context, R.attr.colorPrimaryContainer, TAG);
+    }
+
+    /**
      * Returns the color for the tab strip background.
      *
      * @param context {@link Context} used to retrieve color.
diff --git a/chrome/android/java/res/drawable/bg_circle_new_tab_button_detached.xml b/chrome/android/java/res/drawable/bg_circle_new_tab_button_detached.xml
new file mode 100644
index 0000000..cad27e1
--- /dev/null
+++ b/chrome/android/java/res/drawable/bg_circle_new_tab_button_detached.xml
@@ -0,0 +1,15 @@
+<?xml version="1.0" encoding="utf-8"?>
+<!--
+Copyright 2022 The Chromium Authors
+Use of this source code is governed by a BSD-style license that can be
+found in the LICENSE file.
+-->
+
+<shape xmlns:android="http://schemas.android.com/apk/res/android"
+    android:shape="oval">
+<!--  Set New tab button background to 38 dp diameter for tab strip redesign detached-->
+  <size
+    android:width="38dp"
+    android:height="38dp" />
+  <solid android:color="@color/modern_white" />
+</shape>
\ No newline at end of file
diff --git a/chrome/android/java/res/drawable/bg_circle_new_tab_button_folio.xml b/chrome/android/java/res/drawable/bg_circle_new_tab_button_folio.xml
new file mode 100644
index 0000000..a4063dc
--- /dev/null
+++ b/chrome/android/java/res/drawable/bg_circle_new_tab_button_folio.xml
@@ -0,0 +1,15 @@
+<?xml version="1.0" encoding="utf-8"?>
+<!--
+Copyright 2022 The Chromium Authors
+Use of this source code is governed by a BSD-style license that can be
+found in the LICENSE file.
+-->
+
+<shape xmlns:android="http://schemas.android.com/apk/res/android"
+    android:shape="oval">
+  <!--  Set New tab button background to 36 dp diameter for tab strip redesign folio-->
+  <size
+      android:width="36dp"
+      android:height="36dp" />
+  <solid android:color="@color/modern_white" />
+</shape>
\ No newline at end of file
diff --git a/chrome/android/java/src/org/chromium/chrome/browser/ChromeTabbedActivity.java b/chrome/android/java/src/org/chromium/chrome/browser/ChromeTabbedActivity.java
index d488e73..f340ea6 100644
--- a/chrome/android/java/src/org/chromium/chrome/browser/ChromeTabbedActivity.java
+++ b/chrome/android/java/src/org/chromium/chrome/browser/ChromeTabbedActivity.java
@@ -2600,6 +2600,9 @@
             layoutTypeToShow = LayoutType.START_SURFACE;
             if (state == StartSurfaceState.SHOWING_PREVIOUS) {
                 ReturnToChromeUtil.recordBackNavigationToStart("FromTab");
+            } else {
+                // Resets the scroll position when Start is showing not via back operations.
+                mStartSurfaceSupplier.get().resetScrollPosition();
             }
         }
 
diff --git a/chrome/android/java/src/org/chromium/chrome/browser/compositor/layouts/components/CompositorButton.java b/chrome/android/java/src/org/chromium/chrome/browser/compositor/layouts/components/CompositorButton.java
index 0f73e16..ddb7face 100644
--- a/chrome/android/java/src/org/chromium/chrome/browser/compositor/layouts/components/CompositorButton.java
+++ b/chrome/android/java/src/org/chromium/chrome/browser/compositor/layouts/components/CompositorButton.java
@@ -64,6 +64,7 @@
     private final CompositorOnClickHandler mClickHandler;
 
     protected int mResource;
+    protected int mBackgroundResource;
 
     private int mPressedResource;
     private int mIncognitoResource;
diff --git a/chrome/android/java/src/org/chromium/chrome/browser/compositor/layouts/components/TintedCompositorButton.java b/chrome/android/java/src/org/chromium/chrome/browser/compositor/layouts/components/TintedCompositorButton.java
index 64e3f50..279fa435 100644
--- a/chrome/android/java/src/org/chromium/chrome/browser/compositor/layouts/components/TintedCompositorButton.java
+++ b/chrome/android/java/src/org/chromium/chrome/browser/compositor/layouts/components/TintedCompositorButton.java
@@ -21,6 +21,10 @@
     private @ColorRes int mPressedTintResource;
     private @ColorRes int mIncognitoTintResource;
     private @ColorRes int mIncognitoPressedTintResource;
+    private @ColorInt int mBackgroundDefaultTint;
+    private @ColorInt int mBackgroundPressedTint;
+    private @ColorInt int mBackgroundIncognitoTint;
+    private @ColorInt int mBackgroundIncognitoPressedTint;
 
     public TintedCompositorButton(
             Context context, float width, float height, CompositorOnClickHandler clickHandler) {
@@ -32,7 +36,6 @@
     public TintedCompositorButton(Context context, float width, float height,
             CompositorOnClickHandler clickHandler, @DrawableRes int resource) {
         super(context, width, height, clickHandler);
-
         mContext = context;
         mResource = resource;
     }
@@ -62,6 +65,20 @@
     }
 
     /**
+     * @param backgroundResource The default Android resource.
+     */
+    public void setBackgroundResourceId(@DrawableRes int backgroundResource) {
+        mBackgroundResource = backgroundResource;
+    }
+
+    /**
+     * @return The Android resource that represents button background.
+     */
+    public int getBackgroundResourceId() {
+        return mBackgroundResource;
+    }
+
+    /**
      * A set of Android resources to supply to the compositor.
      * @param defaultTint           The default tint resource.
      * @param pressedTint           The pressed tint resource.
@@ -77,15 +94,42 @@
     }
 
     /**
-     * @return The tint (color value, NOT the resource Id) depending on the state of the button and
-     *         the tab (incognito or not).
+     * A set of Android color to supply to the compositor.
+     * @param backgroundDefaultTint           The default background tint.
+     * @param backgroundPressedTint           The pressed background tint.
+     * @param backgroundIncognitoTint         The incognito background tint.
+     * @param backgroundIncognitoPressedTint  The incognito pressed background tint.
+     */
+    public void setBackgroundTint(@ColorInt int backgroundDefaultTint,
+            @ColorInt int backgroundPressedTint, @ColorInt int backgroundIncognitoTint,
+            @ColorInt int backgroundIncognitoPressedTint) {
+        mBackgroundDefaultTint = backgroundDefaultTint;
+        mBackgroundPressedTint = backgroundPressedTint;
+        mBackgroundIncognitoTint = backgroundIncognitoTint;
+        mBackgroundIncognitoPressedTint = backgroundIncognitoPressedTint;
+    }
+
+    /**
+     * @return The icon tint (color value, NOT the resource Id) depending on the state of the button
+     *         and the tab (incognito or not).
      */
     public @ColorInt int getTint() {
         int tint = isIncognito() ? mIncognitoTintResource : mDefaultTintResource;
         if (isPressed()) {
             tint = isIncognito() ? mIncognitoPressedTintResource : mPressedTintResource;
         }
-
         return AppCompatResources.getColorStateList(mContext, tint).getDefaultColor();
     }
+
+    /**
+     * @return The button background tint (color value, NOT the resource Id) depending on the state
+     *         of the button and the tab.
+     */
+    public @ColorInt int getBackgroundTint() {
+        int tint = isIncognito() ? mBackgroundIncognitoTint : mBackgroundDefaultTint;
+        if (isPressed()) {
+            tint = isIncognito() ? mBackgroundIncognitoPressedTint : mBackgroundPressedTint;
+        }
+        return tint;
+    }
 }
diff --git a/chrome/android/java/src/org/chromium/chrome/browser/compositor/overlays/strip/StripLayoutHelper.java b/chrome/android/java/src/org/chromium/chrome/browser/compositor/overlays/strip/StripLayoutHelper.java
index 23a5836..7923f06 100644
--- a/chrome/android/java/src/org/chromium/chrome/browser/compositor/overlays/strip/StripLayoutHelper.java
+++ b/chrome/android/java/src/org/chromium/chrome/browser/compositor/overlays/strip/StripLayoutHelper.java
@@ -51,10 +51,14 @@
 import org.chromium.chrome.browser.tabmodel.TabModelUtils;
 import org.chromium.chrome.browser.tasks.tab_groups.TabGroupModelFilter;
 import org.chromium.chrome.browser.tasks.tab_management.TabUiFeatureUtilities;
+import org.chromium.chrome.browser.tasks.tab_management.TabUiThemeProvider;
+import org.chromium.components.browser_ui.styles.ChromeColors;
+import org.chromium.components.browser_ui.styles.SemanticColorUtils;
 import org.chromium.components.browser_ui.widget.animation.Interpolators;
 import org.chromium.content_public.browser.UiThreadTaskTraits;
 import org.chromium.ui.base.DeviceFormFactor;
 import org.chromium.ui.base.LocalizationUtils;
+import org.chromium.ui.util.ColorUtils;
 
 import java.util.ArrayList;
 import java.util.List;
@@ -128,14 +132,22 @@
     private static final float REORDER_EDGE_SCROLL_START_MIN_DP = 87.4f;
     private static final float REORDER_EDGE_SCROLL_START_MAX_DP = 18.4f;
     private static final float NEW_TAB_BUTTON_Y_OFFSET_DP = 10.f;
+    private static final float NEW_TAB_BUTTON_BACKGROUND_Y_OFFSET_DP = 0.f;
     private static final float NEW_TAB_BUTTON_CLICK_SLOP_DP = 12.f;
     private static final float NEW_TAB_BUTTON_WIDTH_DP = 24.f;
     private static final float NEW_TAB_BUTTON_HEIGHT_DP = 24.f;
+    private static final float NEW_TAB_BUTTON_BACKGROUND_WIDTH_DP_FOLIO = 36.f;
+    private static final float NEW_TAB_BUTTON_BACKGROUND_HEIGHT_DP_FOLIO = 36.f;
+    private static final float NEW_TAB_BUTTON_BACKGROUND_WIDTH_DP_DETACHED = 38.f;
+    private static final float NEW_TAB_BUTTON_BACKGROUND_HEIGHT_DP_DETACHED = 38.f;
     private static final float NEW_TAB_BUTTON_PADDING_DP = 24.f;
     private static final float NEW_TAB_BUTTON_TOUCH_TARGET_OFFSET = 12.f;
     private static final float FOLIO_ATTACHED_BOTTOM_MARGIN_DP = 0.f;
     private static final float FOLIO_ANIM_INTERMEDIATE_MARGIN_DP = -12.f;
     private static final float FOLIO_DETACHED_BOTTOM_MARGIN_DP = 4.f;
+    private static final float NEW_TAB_BUTTON_DESIRED_TOUCH_TARGET_SIZE = 48.f;
+    private static final float NEW_TAB_BUTTON_DEFAULT_PRESSED_OPACITY = 0.2f;
+    private static final float NEW_TAB_BUTTON_DARK_DETACHED_OPACITY = 0.15f;
     static final float BACKGROUND_TAB_BRIGHTNESS_DEFAULT = 1.f;
     static final float BACKGROUND_TAB_BRIGHTNESS_DIMMED = 0.65f;
     static final float DIVIDER_HIDDEN_OPACITY = 0.f;
@@ -242,7 +254,15 @@
     public StripLayoutHelper(Context context, LayoutUpdateHost updateHost,
             LayoutRenderHost renderHost, boolean incognito, CompositorButton modelSelectorButton) {
         mTabOverlapWidth = TAB_OVERLAP_WIDTH_DP;
-        mNewTabButtonWidth = NEW_TAB_BUTTON_WIDTH_DP;
+        if (ChromeFeatureList.sTabStripRedesign.isEnabled()) {
+            if (TabUiFeatureUtilities.isTabStripFolioEnabled()) {
+                mNewTabButtonWidth = NEW_TAB_BUTTON_BACKGROUND_WIDTH_DP_FOLIO;
+            } else {
+                mNewTabButtonWidth = NEW_TAB_BUTTON_BACKGROUND_WIDTH_DP_DETACHED;
+            }
+        } else {
+            mNewTabButtonWidth = NEW_TAB_BUTTON_WIDTH_DP;
+        }
         mModelSelectorButton = modelSelectorButton;
         mTabStripImpEnabled = ChromeFeatureList.sTabStripImprovements.isEnabled();
 
@@ -264,14 +284,78 @@
                 handleNewTabClick();
             }
         };
-        mNewTabButton = new TintedCompositorButton(context, NEW_TAB_BUTTON_WIDTH_DP,
-                NEW_TAB_BUTTON_HEIGHT_DP, newTabClickHandler, R.drawable.ic_new_tab_button);
 
-        mNewTabButton.setTintResources(R.color.new_tab_button_tint_list,
-                R.color.new_tab_button_pressed_tint_list, R.color.modern_white,
-                R.color.default_icon_color_blue_light);
+        if (ChromeFeatureList.sTabStripRedesign.isEnabled()) {
+            // Set new tab button background resource based on which TSR is enabled, background has
+            // different size.
+            if (TabUiFeatureUtilities.isTabStripFolioEnabled()) {
+                // Tab strip redesign folio enabled, bg size 36 * 36.
+                mNewTabButton = new TintedCompositorButton(context,
+                        NEW_TAB_BUTTON_BACKGROUND_WIDTH_DP_FOLIO,
+                        NEW_TAB_BUTTON_BACKGROUND_HEIGHT_DP_FOLIO, newTabClickHandler,
+                        R.drawable.ic_new_tab_button);
+                mNewTabButton.setBackgroundResourceId(R.drawable.bg_circle_new_tab_button_folio);
+            } else {
+                // Tab strip redesign detached enabled, bg size 38 * 38.
+                mNewTabButton = new TintedCompositorButton(context,
+                        NEW_TAB_BUTTON_BACKGROUND_WIDTH_DP_DETACHED,
+                        NEW_TAB_BUTTON_BACKGROUND_HEIGHT_DP_DETACHED, newTabClickHandler,
+                        R.drawable.ic_new_tab_button);
+                mNewTabButton.setBackgroundResourceId(R.drawable.bg_circle_new_tab_button_detached);
+            }
+
+            // Primary container for default bg color.
+            int defaultBackgroundTint = TabUiThemeProvider.getDefaultContainerColor(context);
+
+            // Primary @ 20% for default pressed bg color.
+            int pressedBackgroundTint = androidx.core.graphics.ColorUtils.setAlphaComponent(
+                    SemanticColorUtils.getDefaultIconColorAccent1(context),
+                    (int) (NEW_TAB_BUTTON_DEFAULT_PRESSED_OPACITY * 255));
+
+            // Surface 2 baseline for folio, surface 3 baseline for detached incognito bg color.
+            int incognitoBackgroundTint = TabUiFeatureUtilities.isTabStripFolioEnabled()
+                    ? context.getResources().getColor(R.color.default_bg_color_dark_elev_2_baseline)
+                    : context.getResources().getColor(
+                            R.color.default_bg_color_dark_elev_3_baseline);
+
+            // surface 5 baseline for incognito pressed bg color
+            int incognitoBackgroundPressedTint =
+                    context.getResources().getColor(R.color.default_bg_color_dark_elev_5_baseline);
+
+            // Tab strip redesign new tab button night mode bg color.
+            if (ColorUtils.inNightMode(context)) {
+                // Surface 1 for folio night mode bg color.
+                if (TabUiFeatureUtilities.isTabStripFolioEnabled()) {
+                    defaultBackgroundTint =
+                            ChromeColors.getSurfaceColor(context, R.dimen.default_elevation_1);
+                } else {
+                    // Primary @ 15% for detached night mode bg color.
+                    defaultBackgroundTint = androidx.core.graphics.ColorUtils.setAlphaComponent(
+                            SemanticColorUtils.getDefaultIconColorAccent1(context),
+                            (int) (NEW_TAB_BUTTON_DARK_DETACHED_OPACITY * 255));
+                }
+                // Surface 5 for pressed night mode bg color.
+                pressedBackgroundTint =
+                        ChromeColors.getSurfaceColor(context, R.dimen.default_elevation_5);
+            }
+            mNewTabButton.setBackgroundTint(defaultBackgroundTint, pressedBackgroundTint,
+                    incognitoBackgroundTint, incognitoBackgroundPressedTint);
+
+            // No pressed state color change for new tab button icon when TSR enabled.
+            mNewTabButton.setTintResources(R.color.default_icon_color_tint_list,
+                    R.color.default_icon_color_tint_list, R.color.modern_white,
+                    R.color.modern_white);
+            mNewTabButton.setY(NEW_TAB_BUTTON_BACKGROUND_Y_OFFSET_DP);
+        } else {
+            // when TSR disabled
+            mNewTabButton = new TintedCompositorButton(context, NEW_TAB_BUTTON_WIDTH_DP,
+                    NEW_TAB_BUTTON_HEIGHT_DP, newTabClickHandler, R.drawable.ic_new_tab_button);
+            mNewTabButton.setTintResources(R.color.new_tab_button_tint_list,
+                    R.color.new_tab_button_pressed_tint_list, R.color.modern_white,
+                    R.color.default_icon_color_blue_light);
+            mNewTabButton.setY(NEW_TAB_BUTTON_Y_OFFSET_DP);
+        }
         mNewTabButton.setIncognito(incognito);
-        mNewTabButton.setY(NEW_TAB_BUTTON_Y_OFFSET_DP);
         mNewTabButton.setClickSlop(NEW_TAB_BUTTON_CLICK_SLOP_DP);
         Resources res = context.getResources();
         mNewTabButton.setAccessibilityDescription(
@@ -352,6 +436,11 @@
      */
     public float getNewTabButtonTouchTargetOffset() {
         boolean isRtl = LocalizationUtils.isLayoutRtl();
+        if (ChromeFeatureList.sTabStripRedesign.isEnabled()) {
+            float newTabButtonTouchTargetOffsetTSR =
+                    (NEW_TAB_BUTTON_DESIRED_TOUCH_TARGET_SIZE - mNewTabButtonWidth) / 2;
+            return isRtl ? newTabButtonTouchTargetOffsetTSR : -newTabButtonTouchTargetOffsetTSR;
+        }
         return isRtl ? NEW_TAB_BUTTON_TOUCH_TARGET_OFFSET : -NEW_TAB_BUTTON_TOUCH_TARGET_OFFSET;
     }
 
@@ -1583,7 +1672,7 @@
         // 2. Get offset from strip stacker.
         float offset = mStripStacker.computeNewTabButtonOffset(mStripTabs, mTabOverlapWidth,
                 mLeftMargin, mRightMargin, mWidth, mNewTabButtonWidth,
-                NEW_TAB_BUTTON_TOUCH_TARGET_OFFSET, mCachedTabWidth, animate);
+                Math.abs(getNewTabButtonTouchTargetOffset()), mCachedTabWidth, animate);
 
         // 3. Hide the new tab button if it's not visible on the screen.
         boolean isRtl = LocalizationUtils.isLayoutRtl();
@@ -1601,7 +1690,6 @@
         } else {
             mNewTabButton.setX(offset);
         }
-
         return null;
     }
 
diff --git a/chrome/android/java/src/org/chromium/chrome/browser/compositor/scene_layer/TabStripSceneLayer.java b/chrome/android/java/src/org/chromium/chrome/browser/compositor/scene_layer/TabStripSceneLayer.java
index 99d4af1..94d2b2b9 100644
--- a/chrome/android/java/src/org/chromium/chrome/browser/compositor/scene_layer/TabStripSceneLayer.java
+++ b/chrome/android/java/src/org/chromium/chrome/browser/compositor/scene_layer/TabStripSceneLayer.java
@@ -133,13 +133,12 @@
         CompositorButton modelSelectorButton = layoutHelper.getModelSelectorButton();
         boolean newTabButtonVisible = newTabButton.isVisible();
         boolean modelSelectorButtonVisible = modelSelectorButton.isVisible();
-
         TabStripSceneLayerJni.get().updateNewTabButton(mNativePtr, TabStripSceneLayer.this,
-                newTabButton.getResourceId(), newTabButton.getX() * mDpToPx,
-                newTabButton.getY() * mDpToPx, newTabButton.getWidth() * mDpToPx,
-                newTabButton.getHeight() * mDpToPx,
+                newTabButton.getResourceId(), newTabButton.getBackgroundResourceId(),
+                newTabButton.getX() * mDpToPx, newTabButton.getY() * mDpToPx,
                 layoutHelper.getNewTabBtnTouchTargetOffset() * mDpToPx, newTabButtonVisible,
-                newTabButton.getTint(), newTabButton.getOpacity(), resourceManager);
+                newTabButton.getTint(), newTabButton.getBackgroundTint(), newTabButton.getOpacity(),
+                resourceManager);
 
         TabStripSceneLayerJni.get().updateModelSelectorButton(mNativePtr, TabStripSceneLayer.this,
                 modelSelectorButton.getResourceId(), modelSelectorButton.getX() * mDpToPx,
@@ -218,8 +217,8 @@
         void updateStripScrim(long nativeTabStripSceneLayer, TabStripSceneLayer caller, float x,
                 float y, float width, float height, int color, float alpha);
         void updateNewTabButton(long nativeTabStripSceneLayer, TabStripSceneLayer caller,
-                int resourceId, float x, float y, float width, float height,
-                float touchTargetOffset, boolean visible, int tint, float buttonAlpha,
+                int resourceId, int backgroundResourceId, float x, float y, float touchTargetOffset,
+                boolean visible, int tint, int backgroundTint, float buttonAlpha,
                 ResourceManager resourceManager);
         void updateModelSelectorButton(long nativeTabStripSceneLayer, TabStripSceneLayer caller,
                 int resourceId, float x, float y, float width, float height, boolean incognito,
diff --git a/chrome/android/java/src/org/chromium/chrome/browser/media/PictureInPictureActivity.java b/chrome/android/java/src/org/chromium/chrome/browser/media/PictureInPictureActivity.java
index a0878cf1..2b17cad 100644
--- a/chrome/android/java/src/org/chromium/chrome/browser/media/PictureInPictureActivity.java
+++ b/chrome/android/java/src/org/chromium/chrome/browser/media/PictureInPictureActivity.java
@@ -365,13 +365,22 @@
 
     private class InitiatorTabObserver extends EmptyTabObserver {
         @Override
+        public void onClosingStateChanged(Tab tab, boolean closing) {
+            if (closing) {
+                PictureInPictureActivity.this.onExitPictureInPicture(/*closeByNative=*/false);
+            }
+        }
+
+        @Override
         public void onDestroyed(Tab tab) {
-            if (tab.isClosing()) PictureInPictureActivity.this.finish();
+            if (tab.isClosing()) {
+                PictureInPictureActivity.this.onExitPictureInPicture(/*closeByNative=*/false);
+            }
         }
 
         @Override
         public void onCrash(Tab tab) {
-            PictureInPictureActivity.this.finish();
+            PictureInPictureActivity.this.onExitPictureInPicture(/*closeByNative=*/false);
         }
     }
 
@@ -465,7 +474,7 @@
         // or InitiatorTab has been destroyed by user or crashed.
         if (mNativeOverlayWindowAndroid != sPendingNativeOverlayWindowAndroid
                 || TabUtils.getActivity(mInitiatorTab) == null) {
-            this.finish();
+            onExitPictureInPicture(/*closeByNative=*/false);
             return;
         }
         sPendingNativeOverlayWindowAndroid = 0;
@@ -497,14 +506,35 @@
     }
 
     @Override
-    public void onStop() {
-        super.onStop();
-        if (mCompositorView != null) mCompositorView.destroy();
+    @RequiresApi(api = Build.VERSION_CODES.O)
+    public void onPictureInPictureModeChanged(
+            boolean isInPictureInPictureMode, Configuration newConfig) {
+        super.onPictureInPictureModeChanged(isInPictureInPictureMode, newConfig);
+        if (isInPictureInPictureMode) return;
+        PictureInPictureActivityJni.get().onBackToTab(mNativeOverlayWindowAndroid);
+        onExitPictureInPicture(/*closeByNative=*/false);
     }
 
     @Override
-    public void onDestroy() {
-        super.onDestroy();
+    protected ActivityWindowAndroid createWindowAndroid() {
+        return new ActivityWindowAndroid(
+                this, /* listenToActivityState= */ true, getIntentRequestTracker());
+    }
+
+    @CalledByNative
+    public void close() {
+        onExitPictureInPicture(/*closeByNative=*/true);
+    }
+
+    private void onExitPictureInPicture(boolean closeByNative) {
+        if (!closeByNative && mNativeOverlayWindowAndroid != 0) {
+            PictureInPictureActivityJni.get().destroy(mNativeOverlayWindowAndroid);
+        }
+
+        if (mCompositorView != null) {
+            mCompositorView.destroy();
+            mCompositorView = null;
+        }
 
         if (mMediaSessionReceiver != null) {
             unregisterReceiver(mMediaSessionReceiver);
@@ -516,26 +546,7 @@
             mInitiatorTab = null;
         }
         mTabObserver = null;
-    }
 
-    @Override
-    @RequiresApi(api = Build.VERSION_CODES.O)
-    public void onPictureInPictureModeChanged(
-            boolean isInPictureInPictureMode, Configuration newConfig) {
-        super.onPictureInPictureModeChanged(isInPictureInPictureMode, newConfig);
-        if (isInPictureInPictureMode) return;
-        PictureInPictureActivityJni.get().onBackToTab(mNativeOverlayWindowAndroid);
-        this.finish();
-    }
-
-    @Override
-    protected ActivityWindowAndroid createWindowAndroid() {
-        return new ActivityWindowAndroid(
-                this, /* listenToActivityState= */ true, getIntentRequestTracker());
-    }
-
-    @CalledByNative
-    public void close() {
         this.finish();
     }
 
@@ -682,7 +693,7 @@
             PictureInPictureActivity pipActivity = (PictureInPictureActivity) activity;
             if (nativeOverlayWindowAndroid == pipActivity.getNativeOverlayWindowAndroid()) {
                 pipActivity.resetNativeOverlayWindowAndroid();
-                pipActivity.finish();
+                pipActivity.onExitPictureInPicture(/*closeByNative=*/true);
             }
         }
     }
diff --git a/chrome/android/java/src/org/chromium/chrome/browser/quickactionsearchwidget/QuickActionSearchWidgetProvider.java b/chrome/android/java/src/org/chromium/chrome/browser/quickactionsearchwidget/QuickActionSearchWidgetProvider.java
index 3a7c250..032140e70 100644
--- a/chrome/android/java/src/org/chromium/chrome/browser/quickactionsearchwidget/QuickActionSearchWidgetProvider.java
+++ b/chrome/android/java/src/org/chromium/chrome/browser/quickactionsearchwidget/QuickActionSearchWidgetProvider.java
@@ -11,7 +11,10 @@
 import android.content.Intent;
 import android.content.pm.PackageManager;
 import android.net.Uri;
+import android.os.Build;
 import android.os.Bundle;
+import android.util.ArrayMap;
+import android.util.SizeF;
 import android.widget.RemoteViews;
 
 import androidx.annotation.NonNull;
@@ -32,6 +35,9 @@
 import org.chromium.chrome.browser.ui.searchactivityutils.SearchActivityPreferencesManager.SearchActivityPreferences;
 import org.chromium.components.embedder_support.util.UrlConstants;
 
+import java.util.ArrayList;
+import java.util.Map;
+
 /**
  * {@link AppWidgetProvider} for a widget that provides an entry point for users to quickly perform
  * actions in Chrome.
@@ -45,15 +51,10 @@
             extends QuickActionSearchWidgetProvider {
         @Override
         @NonNull
-        RemoteViews getRemoteViews(@NonNull Context context,
-                @NonNull SearchActivityPreferences prefs, @NonNull AppWidgetManager manager,
-                int widgetId) {
-            Bundle options = manager.getAppWidgetOptions(widgetId);
-            return getDelegate().createSearchWidgetRemoteViews(context, prefs,
-                    getPortraitModeTargetAreaWidth(options),
-                    getPortraitModeTargetAreaHeight(options),
-                    getLandscapeModeTargetAreaWidth(options),
-                    getLandscapeModeTargetAreaHeight(options));
+        RemoteViews createWidget(@NonNull Context context, @NonNull SearchActivityPreferences prefs,
+                int areaWidthDp, int areaHeightDp) {
+            return getDelegate().createSearchWidgetRemoteViews(
+                    context, prefs, areaWidthDp, areaHeightDp);
         }
     }
 
@@ -85,15 +86,10 @@
             extends QuickActionSearchWidgetProvider {
         @Override
         @NonNull
-        RemoteViews getRemoteViews(@NonNull Context context,
-                @NonNull SearchActivityPreferences prefs, @NonNull AppWidgetManager manager,
-                int widgetId) {
-            Bundle options = manager.getAppWidgetOptions(widgetId);
-            return getDelegate().createDinoWidgetRemoteViews(context, prefs,
-                    getPortraitModeTargetAreaWidth(options),
-                    getPortraitModeTargetAreaHeight(options),
-                    getLandscapeModeTargetAreaWidth(options),
-                    getLandscapeModeTargetAreaHeight(options));
+        RemoteViews createWidget(@NonNull Context context, @NonNull SearchActivityPreferences prefs,
+                int areaWidthDp, int areaHeightDp) {
+            return getDelegate().createDinoWidgetRemoteViews(
+                    context, prefs, areaWidthDp, areaHeightDp);
         }
     }
 
@@ -130,8 +126,8 @@
 
         for (int index = 0; index < widgetIds.length; index++) {
             int widgetId = widgetIds[index];
-            manager.updateAppWidget(
-                    widgetId, getRemoteViews(context, preferences, manager, widgetId));
+            Bundle options = manager.getAppWidgetOptions(widgetId);
+            manager.updateAppWidget(widgetId, getRemoteViews(context, preferences, options));
         }
     }
 
@@ -182,18 +178,95 @@
     }
 
     /**
-     * Acquire screen orientation specific layouts that will be applied to the
-     * widget.
-     * The two layouts represent screen orientations in Landscape and Portrait mode.
+     * Construct the widget for specific dimensions.
      *
      * @param context Current context.
-     * @param manager The AppWidgetManager instance to query widget info.
-     * @param widgetId The widget to get the delegate for.
+     * @param prefs Widget settings and feature availability.
+     * @param areaWidthDp The width of the widget area, expressed in Dp.
+     * @param areaHeightDp The height of the widget area, expressed in Dp.
+     * @return RemoteViews description for a single widget layout.
      */
     @VisibleForTesting(otherwise = VisibleForTesting.PROTECTED)
-    abstract @NonNull RemoteViews getRemoteViews(@NonNull Context context,
-            @NonNull SearchActivityPreferences prefs, @NonNull AppWidgetManager manager,
-            int widgetId);
+    @NonNull
+    abstract RemoteViews createWidget(@NonNull Context context,
+            @NonNull SearchActivityPreferences prefs, int areaWidthDp, int areaHeightDp);
+
+    /**
+     * Acquire the RemoteViews that represent the widget.
+     *
+     * @param context Current context.
+     * @param prefs Widget settings and feature availability.
+     * @param options Options bundle passed by AppWidgetManager.
+     */
+    @VisibleForTesting(otherwise = VisibleForTesting.PROTECTED)
+    @NonNull
+    RemoteViews getRemoteViews(@NonNull Context context, @NonNull SearchActivityPreferences prefs,
+            @NonNull Bundle options) {
+        var views = getSizeMappedRemoteViews(context, prefs, options);
+        if (views != null) {
+            return views;
+        }
+        return getOrientationSpecificRemoteViews(context, prefs, options);
+    }
+
+    /**
+     * Acquire screen orientation specific layouts that will be applied to the
+     * widget.
+     *
+     * @param context Current context.
+     * @param prefs Widget settings and feature availability.
+     * @param options Widget parameters passed by the AppWidgetManager.
+     * @return RemoteViews describing widget for landscape and portrait screen orientations.
+     */
+    @VisibleForTesting(otherwise = VisibleForTesting.PROTECTED)
+    @NonNull
+    RemoteViews getOrientationSpecificRemoteViews(@NonNull Context context,
+            @NonNull SearchActivityPreferences prefs, @NonNull Bundle options) {
+        var portraitViews = createWidget(context, prefs, getPortraitModeTargetAreaWidth(options),
+                getPortraitModeTargetAreaHeight(options));
+
+        var landscapeViews = createWidget(context, prefs, getLandscapeModeTargetAreaWidth(options),
+                getLandscapeModeTargetAreaHeight(options));
+
+        return new RemoteViews(landscapeViews, portraitViews);
+    }
+
+    /**
+     * Acquire size-specific layouts that will be applied to the widget.
+     *
+     * @param context Current context.
+     * @param prefs Widget settings and feature availability.
+     * @param options Widget parameters passed by the AppWidgetManager.
+     * @return RemoteViews describing widget for all sizes requested by the AppWidgetManager, or
+     *         null, if the AppWidgetManager did not specify the sizes.
+     */
+    @VisibleForTesting(otherwise = VisibleForTesting.PROTECTED)
+    @Nullable
+    RemoteViews getSizeMappedRemoteViews(@NonNull Context context,
+            @NonNull SearchActivityPreferences prefs, @NonNull Bundle options) {
+        // On Android S and above, attempt to build widget from supplied array of sizes.
+        // This is reserved to Android S because appropriate RemoteViews constructor may not be
+        // available.
+        // Note that the creation may still fail, if the launcher is unable to offer appropriate
+        // details.
+        // Check for supported system version.
+        if (Build.VERSION.SDK_INT < Build.VERSION_CODES.S) {
+            return null;
+        }
+
+        ArrayList<SizeF> sizes =
+                options.getParcelableArrayList(AppWidgetManager.OPTION_APPWIDGET_SIZES);
+        if (sizes == null || sizes.isEmpty()) {
+            return null;
+        }
+        Map<SizeF, RemoteViews> mappings = new ArrayMap<>();
+
+        for (var size : sizes) {
+            mappings.put(size,
+                    createWidget(context, prefs, (int) size.getWidth(), (int) size.getHeight()));
+        }
+        return new RemoteViews(mappings);
+    }
 
     /**
      * This function initializes the QuickActionSearchWidgetProvider component. Namely, this
diff --git a/chrome/android/java/src/org/chromium/chrome/browser/share/LensUtils.java b/chrome/android/java/src/org/chromium/chrome/browser/share/LensUtils.java
index 9bd0e52..2219902b 100644
--- a/chrome/android/java/src/org/chromium/chrome/browser/share/LensUtils.java
+++ b/chrome/android/java/src/org/chromium/chrome/browser/share/LensUtils.java
@@ -158,7 +158,7 @@
     public static boolean isGoogleLensFeatureEnabledOnTablet() {
         return ChromeFeatureList.getFieldTrialParamByFeatureAsBoolean(
                 ChromeFeatureList.CONTEXT_MENU_SEARCH_WITH_GOOGLE_LENS, ENABLE_ON_TABLET_PARAM_NAME,
-                false);
+                true);
     }
 
     /**
diff --git a/chrome/android/javatests/src/org/chromium/chrome/browser/media/PictureInPictureActivityTest.java b/chrome/android/javatests/src/org/chromium/chrome/browser/media/PictureInPictureActivityTest.java
index 6fd7128..4f3e32b 100644
--- a/chrome/android/javatests/src/org/chromium/chrome/browser/media/PictureInPictureActivityTest.java
+++ b/chrome/android/javatests/src/org/chromium/chrome/browser/media/PictureInPictureActivityTest.java
@@ -297,12 +297,22 @@
         testExitOn(activity, () -> activity.close());
     }
 
+    @Test
+    @MediumTest
+    @MinAndroidSdkLevel(Build.VERSION_CODES.O)
+    @Restriction(RESTRICTION_TYPE_NON_LOW_END_DEVICE)
+    public void testNotifyNativeWhenTabClose() throws Throwable {
+        PictureInPictureActivity activity = startPictureInPictureActivity();
+        testExitOn(activity, () -> mTab.setClosing(/*closing=*/true));
+        verify(mNativeMock, times(1)).destroy(NATIVE_OVERLAY);
+    }
+
     private WebContents getWebContents() {
         return mActivityTestRule.getActivity().getCurrentWebContents();
     }
 
     private void testExitOn(Activity activity, Runnable runnable) throws Throwable {
-        runnable.run();
+        TestThreadUtils.runOnUiThreadBlocking(() -> runnable.run());
 
         CriteriaHelper.pollUiThread(() -> {
             Criteria.checkThat(activity == null || activity.isDestroyed(), Matchers.is(true));
diff --git a/chrome/android/javatests/src/org/chromium/chrome/browser/metrics/StartupLoadingMetricsTest.java b/chrome/android/javatests/src/org/chromium/chrome/browser/metrics/StartupLoadingMetricsTest.java
index ca0d2d3..455405e 100644
--- a/chrome/android/javatests/src/org/chromium/chrome/browser/metrics/StartupLoadingMetricsTest.java
+++ b/chrome/android/javatests/src/org/chromium/chrome/browser/metrics/StartupLoadingMetricsTest.java
@@ -25,7 +25,6 @@
 import org.chromium.base.metrics.RecordHistogram;
 import org.chromium.base.test.util.CommandLineFlags;
 import org.chromium.base.test.util.CriteriaHelper;
-import org.chromium.base.test.util.DisableIf;
 import org.chromium.base.test.util.DisabledTest;
 import org.chromium.base.test.util.DoNotBatch;
 import org.chromium.base.test.util.JniMocker;
@@ -223,7 +222,6 @@
      */
     @Test
     @LargeTest
-    @DisableIf.Build(supported_abis_includes = "x86", message = "https://crbug.com/1062055")
     public void testErrorPageNotRecorded() throws Exception {
         runAndWaitForPageLoadMetricsRecorded(
                 () -> mTabbedActivityTestRule.startMainActivityWithURL(mErrorPage));
diff --git a/chrome/android/javatests/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchConsentControllerTest.java b/chrome/android/javatests/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchConsentControllerTest.java
index ee759ab..263036b 100644
--- a/chrome/android/javatests/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchConsentControllerTest.java
+++ b/chrome/android/javatests/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchConsentControllerTest.java
@@ -111,29 +111,11 @@
 
     @Test
     @MediumTest
-    @DisableFeatures(ChromeFeatureList.ASSISTANT_CONSENT_MODAL)
     public void testNoBottomSheetControllerAvailable() {
         ChromeTabbedActivity cta = mActivityTestRule.getActivity();
         TestThreadUtils.runOnUiThreadBlocking(() -> {
-            AssistantVoiceSearchConsentController.show(cta.getWindowAndroid(),
-                    mSharedPreferencesManager,
-                    () -> {}, null, cta.getWindowAndroid().getModalDialogManager(), mCallback);
-        });
-        Mockito.verify(mCallback, Mockito.timeout(1000)).onResult(false);
-    }
-
-    @Test
-    @MediumTest
-    @EnableFeatures(ChromeFeatureList.ASSISTANT_CONSENT_MODAL)
-    public void testNoModalDialogManagerAvailable() {
-        ChromeTabbedActivity cta = mActivityTestRule.getActivity();
-        TestThreadUtils.runOnUiThreadBlocking(() -> {
-            AssistantVoiceSearchConsentController.show(cta.getWindowAndroid(),
-                    mSharedPreferencesManager,
-                    ()
-                            -> {},
-                    cta.getRootUiCoordinatorForTesting().getBottomSheetController(), null,
-                    mCallback);
+            AssistantVoiceSearchConsentController.show(
+                    cta.getWindowAndroid(), mSharedPreferencesManager, () -> {}, null, mCallback);
         });
         Mockito.verify(mCallback, Mockito.timeout(1000)).onResult(false);
     }
diff --git a/chrome/android/javatests/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchConsentModalTest.java b/chrome/android/javatests/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchConsentModalTest.java
deleted file mode 100644
index 9b47402..0000000
--- a/chrome/android/javatests/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchConsentModalTest.java
+++ /dev/null
@@ -1,132 +0,0 @@
-// Copyright 2022 The Chromium Authors
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-package org.chromium.chrome.browser.omnibox.voice;
-
-import static androidx.test.espresso.Espresso.onView;
-import static androidx.test.espresso.action.ViewActions.click;
-import static androidx.test.espresso.matcher.ViewMatchers.withId;
-
-import androidx.test.filters.MediumTest;
-
-import org.junit.After;
-import org.junit.Before;
-import org.junit.Rule;
-import org.junit.Test;
-import org.junit.runner.RunWith;
-import org.mockito.Mock;
-import org.mockito.Mockito;
-import org.mockito.junit.MockitoJUnit;
-import org.mockito.junit.MockitoRule;
-
-import org.chromium.base.test.util.CommandLineFlags;
-import org.chromium.chrome.R;
-import org.chromium.chrome.browser.flags.ChromeSwitches;
-import org.chromium.chrome.test.ChromeJUnit4ClassRunner;
-import org.chromium.chrome.test.ChromeTabbedActivityTestRule;
-import org.chromium.content_public.browser.test.util.TestThreadUtils;
-import org.chromium.ui.base.WindowAndroid;
-import org.chromium.ui.modaldialog.DialogDismissalCause;
-import org.chromium.ui.modaldialog.ModalDialogManager;
-
-/** Tests for AssistantVoiceSearchConsentDialog */
-@RunWith(ChromeJUnit4ClassRunner.class)
-@CommandLineFlags.Add({ChromeSwitches.DISABLE_FIRST_RUN_EXPERIENCE})
-public class AssistantVoiceSearchConsentModalTest {
-    @Rule
-    public ChromeTabbedActivityTestRule mActivityTestRule = new ChromeTabbedActivityTestRule();
-
-    @Rule
-    public MockitoRule mMockitoRule = MockitoJUnit.rule();
-
-    @Mock
-    AssistantVoiceSearchConsentUi.Observer mObserver;
-
-    ModalDialogManager mModalDialogManager;
-    AssistantVoiceSearchConsentModal mModal;
-
-    @Before
-    public void setUp() {
-        mActivityTestRule.startMainActivityOnBlankPage();
-
-        TestThreadUtils.runOnUiThreadBlocking(() -> {
-            WindowAndroid wa = mActivityTestRule.getActivity().getWindowAndroid();
-            mModalDialogManager = wa.getModalDialogManager();
-            mModal = new AssistantVoiceSearchConsentModal(
-                    wa.getContext().get(), mModalDialogManager);
-        });
-    }
-
-    @After
-    public void tearDown() {}
-
-    private void showModal() {
-        TestThreadUtils.runOnUiThreadBlocking(() -> { mModal.show(mObserver); });
-    }
-
-    @Test
-    @MediumTest
-    public void testAcceptButton() {
-        showModal();
-
-        onView(withId(org.chromium.chrome.R.id.positive_button)).perform(click());
-
-        Mockito.verify(mObserver, Mockito.timeout(1000)).onConsentAccepted();
-    }
-
-    @Test
-    @MediumTest
-    public void testRejectButton() {
-        showModal();
-
-        onView(withId(org.chromium.chrome.R.id.negative_button)).perform(click());
-
-        Mockito.verify(mObserver, Mockito.timeout(1000)).onConsentRejected();
-    }
-
-    @Test
-    @MediumTest
-    public void testLearnMoreButton() {
-        showModal();
-
-        onView(withId(R.id.avs_consent_ui_learn_more)).perform(click());
-
-        Mockito.verify(mObserver, Mockito.timeout(1000)).onLearnMoreClicked();
-    }
-
-    @Test
-    @MediumTest
-    public void testCancel() {
-        showModal();
-
-        TestThreadUtils.runOnUiThreadBlocking(() -> {
-            mModalDialogManager.dismissAllDialogs(
-                    DialogDismissalCause.NAVIGATE_BACK_OR_TOUCH_OUTSIDE);
-        });
-
-        Mockito.verify(mObserver, Mockito.timeout(1000)).onConsentCanceled();
-    }
-
-    @Test
-    @MediumTest
-    public void testDismiss() {
-        showModal();
-
-        TestThreadUtils.runOnUiThreadBlocking(() -> { mModal.dismiss(); });
-
-        Mockito.verifyNoMoreInteractions(mObserver);
-    }
-
-    @Test
-    @MediumTest
-    public void testNonUserCancel() {
-        showModal();
-
-        TestThreadUtils.runOnUiThreadBlocking(() -> {
-            mModalDialogManager.dismissAllDialogs(DialogDismissalCause.ACTION_ON_CONTENT);
-        });
-
-        Mockito.verify(mObserver, Mockito.timeout(1000)).onNonUserCancel();
-    }
-}
diff --git a/chrome/android/junit/src/org/chromium/chrome/browser/compositor/overlays/strip/StripLayoutHelperTest.java b/chrome/android/junit/src/org/chromium/chrome/browser/compositor/overlays/strip/StripLayoutHelperTest.java
index 4978ed65..f69c169 100644
--- a/chrome/android/junit/src/org/chromium/chrome/browser/compositor/overlays/strip/StripLayoutHelperTest.java
+++ b/chrome/android/junit/src/org/chromium/chrome/browser/compositor/overlays/strip/StripLayoutHelperTest.java
@@ -123,7 +123,8 @@
         when(mModelSelectorBtn.isVisible()).thenReturn(true);
         when(mTabGroupModelFilter.hasOtherRelatedTabs(any())).thenReturn(false);
 
-        mActivity = Robolectric.buildActivity(Activity.class).setup().get();
+        mActivity = Robolectric.setupActivity(Activity.class);
+        mActivity.setTheme(org.chromium.chrome.R.style.Theme_BrowserUI);
         TabUiFeatureUtilities.setTabMinWidthForTesting(190.f);
     }
 
@@ -1450,6 +1451,7 @@
 
     @Test
     @Feature("Tab Groups on Tab Strip")
+    @Features.DisableFeatures(ChromeFeatureList.TAB_STRIP_REDESIGN)
     public void testReorder_ExtraMinScroll() {
         // Mock 3 tabs. Group the first two tabs.
         initializeTest(false, false, true, 0, 3);
@@ -1478,7 +1480,7 @@
 
         // Assert: New tab button position before starting tab closure.
         mStripLayoutHelper.updateLayout(TIMESTAMP);
-        assertEquals("Unexpected starting newTabButton position.", 764.f,
+        assertEquals("Unexpected starting newTabButton position.", 743.f,
                 mStripLayoutHelper.getNewTabButton().getX(), 0.0f);
 
         // Act: Call on close tab button handler.
@@ -1500,7 +1502,7 @@
                 mStripLayoutHelper.getStripLayoutTabs().length);
         assertTrue("MultiStepAnimations should still be running.",
                 mStripLayoutHelper.isMultiStepCloseAnimationsRunning());
-        assertEquals("NewTabButton should not have moved.", 764.f,
+        assertEquals("NewTabButton should not have moved.", 743.f,
                 mStripLayoutHelper.getNewTabButton().getX(), 0.0f);
 
         // Act: End next set of animations to apply final values.
@@ -1508,14 +1510,14 @@
 
         // Assert: Animations completed. The tab width is not resized and drawX does not change.
         float expectedDrawX =
-                -460.f; // Since we are focused on the last tab, start tabs are off screen.
+                -474.f; // Since we are focused on the last tab, start tabs are off screen.
         StripLayoutTab[] updatedTabs = mStripLayoutHelper.getStripLayoutTabs();
         for (StripLayoutTab stripTab : updatedTabs) {
             assertEquals("Unexpected tab width after resize.", 156.f, stripTab.getWidth(), 0.0);
             assertEquals("Unexpected tab position.", expectedDrawX, stripTab.getDrawX(), 0.0);
             expectedDrawX += (TAB_WIDTH_MEDIUM - TAB_OVERLAP_WIDTH);
         }
-        assertEquals("NewTabButton should not have moved.", 764.f,
+        assertEquals("NewTabButton should not have moved.", 743.f,
                 mStripLayoutHelper.getNewTabButton().getX(), 0.0f);
         assertFalse("MultiStepAnimations should have stopped running.",
                 mStripLayoutHelper.isMultiStepCloseAnimationsRunning());
@@ -1535,7 +1537,7 @@
 
         // Assert: New tab button position before starting tab closure.
         mStripLayoutHelper.updateLayout(TIMESTAMP);
-        assertEquals("Unexpected starting newTabButton position.", 764.f,
+        assertEquals("Unexpected starting newTabButton position.", 743.f,
                 mStripLayoutHelper.getNewTabButton().getX(), 0.0f);
 
         // Act: Call on close tab button handler.
@@ -1556,7 +1558,7 @@
         assertEquals(expectedTabCount, mStripLayoutHelper.getStripLayoutTabs().length);
         assertTrue("MultiStepAnimations should still be running.",
                 mStripLayoutHelper.isMultiStepCloseAnimationsRunning());
-        assertEquals("NewTabButton should not have moved.", 764.f,
+        assertEquals("NewTabButton should not have moved.", 743.f,
                 mStripLayoutHelper.getNewTabButton().getX(), 0.0f);
 
         // Act: Set animation time forward by 250ms for next set of animations.
@@ -1565,7 +1567,7 @@
         // Assert: Animations completed. The tab width is resized, tab.drawX is changed and
         // newTabButton.drawX is also changed.
         float expectedDrawX = 0.f;
-        float expectedWidthAfterResize = 265.f;
+        float expectedWidthAfterResize = 262.f;
         StripLayoutTab[] updatedTabs = mStripLayoutHelper.getStripLayoutTabs();
         for (int i = 0; i < updatedTabs.length; i++) {
             StripLayoutTab stripTab = updatedTabs[i];
@@ -1574,7 +1576,7 @@
             assertEquals("Unexpected tab position.", expectedDrawX, stripTab.getDrawX(), 0.0);
             expectedDrawX += (expectedWidthAfterResize - TAB_OVERLAP_WIDTH);
         }
-        assertEquals("NewTabButton position is incorrect.", 759.f,
+        assertEquals("NewTabButton position is incorrect.", 743.f,
                 mStripLayoutHelper.getNewTabButton().getX(), 0.0f);
         assertFalse("MultiStepAnimations should have ended.",
                 mStripLayoutHelper.isMultiStepCloseAnimationsRunning());
diff --git a/chrome/android/junit/src/org/chromium/chrome/browser/quickactionsearchwidget/QuickActionSearchWidgetProviderTest.java b/chrome/android/junit/src/org/chromium/chrome/browser/quickactionsearchwidget/QuickActionSearchWidgetProviderTest.java
index 9cdaf6a..3a9587f 100644
--- a/chrome/android/junit/src/org/chromium/chrome/browser/quickactionsearchwidget/QuickActionSearchWidgetProviderTest.java
+++ b/chrome/android/junit/src/org/chromium/chrome/browser/quickactionsearchwidget/QuickActionSearchWidgetProviderTest.java
@@ -6,6 +6,8 @@
 
 import static org.mockito.Mockito.any;
 import static org.mockito.Mockito.anyInt;
+import static org.mockito.Mockito.clearInvocations;
+import static org.mockito.Mockito.eq;
 import static org.mockito.Mockito.times;
 import static org.mockito.Mockito.verify;
 import static org.mockito.Mockito.when;
@@ -13,8 +15,9 @@
 import android.appwidget.AppWidgetManager;
 import android.content.Context;
 import android.content.Intent;
+import android.os.Build;
 import android.os.Bundle;
-import android.widget.RemoteViews;
+import android.util.SizeF;
 
 import androidx.test.core.app.ApplicationProvider;
 import androidx.test.filters.SmallTest;
@@ -25,81 +28,202 @@
 import org.junit.runner.RunWith;
 import org.mockito.Mock;
 import org.mockito.Mockito;
-import org.mockito.MockitoAnnotations;
 import org.mockito.junit.MockitoJUnit;
 import org.mockito.junit.MockitoRule;
+import org.robolectric.annotation.Config;
+import org.robolectric.shadows.ShadowLog;
 
 import org.chromium.base.test.BaseRobolectricTestRunner;
-import org.chromium.chrome.browser.firstrun.FirstRunStatus;
-import org.chromium.chrome.browser.ui.quickactionsearchwidget.QuickActionSearchWidgetProviderDelegate;
+import org.chromium.chrome.browser.quickactionsearchwidget.QuickActionSearchWidgetProvider.QuickActionSearchWidgetProviderDino;
 import org.chromium.chrome.browser.ui.searchactivityutils.SearchActivityPreferencesManager.SearchActivityPreferences;
 
+import java.util.ArrayList;
+import java.util.Arrays;
+
 /**
  * Tests for the (@link QuickActionSearchWidgetProvider}.
  */
 @RunWith(BaseRobolectricTestRunner.class)
+@Config(manifest = Config.NONE, shadows = {ShadowLog.class})
 public class QuickActionSearchWidgetProviderTest {
+    private static final int WIDGET_A_ID = 123;
+    private static final int WIDGET_B_ID = 987;
     // These are random unique identifiers, the value of these numbers have no special meaning.
     // The number of identifiers has no particular meaning either.
-    private static final int[] WIDGET_IDS = {1, 2};
+    private static final int[] WIDGET_IDS = {WIDGET_A_ID, WIDGET_B_ID};
+    public @Rule MockitoRule mMockitoRule = MockitoJUnit.rule();
+    private @Mock AppWidgetManager mAppWidgetManagerMock;
 
-    /**
-     * A sub class of {@link QuickActionSearchWidgetProvider} for testing, since
-     * QuickActionSearchWidgetProvider is abstract.
-     */
-    private class TestProvider extends QuickActionSearchWidgetProvider {
-        @Override
-        RemoteViews getRemoteViews(Context context, SearchActivityPreferences prefs,
-                AppWidgetManager manager, int widgetId) {
-            return mRemoteViews;
-        }
-
-        @Override
-        protected QuickActionSearchWidgetProviderDelegate getDelegate() {
-            return mDelegateMock;
-        }
-    }
-
-    @Rule
-    public MockitoRule mMockitoRule = MockitoJUnit.rule();
-    @Mock
-    private AppWidgetManager mAppWidgetManagerMock;
-    @Mock
-    private QuickActionSearchWidgetProviderDelegate mDelegateMock;
-    @Mock
-    private Bundle mBundleMock;
-    @Mock
-    private RemoteViews mRemoteViews;
-
+    private SearchActivityPreferences mPreferences;
     private QuickActionSearchWidgetProvider mWidgetProvider;
     private Context mContext;
+    private Intent mIntent;
+    private Bundle mOptionsWidgetA;
+    private Bundle mOptionsWidgetB;
 
     @Before
     public void setUp() {
-        FirstRunStatus.setFirstRunFlowComplete(true);
-        MockitoAnnotations.initMocks(this);
+        ShadowLog.stream = System.out;
         mContext = Mockito.spy(ApplicationProvider.getApplicationContext());
+        mOptionsWidgetA = new Bundle();
+        mOptionsWidgetB = new Bundle();
+        mPreferences = new SearchActivityPreferences(
+                "Search Engine", "https://search.engine.com", true, true, true);
 
         // Inflate an actual RemoteViews to avoid stubbing internal methods or making
         // any other assumptions about the class.
-        mWidgetProvider = Mockito.spy(new TestProvider());
+        mWidgetProvider = Mockito.spy(new QuickActionSearchWidgetProviderDino());
         when(mContext.getSystemService(Context.APPWIDGET_SERVICE))
                 .thenReturn(mAppWidgetManagerMock);
-        when(mAppWidgetManagerMock.getAppWidgetOptions(anyInt())).thenReturn(mBundleMock);
+        when(mAppWidgetManagerMock.getAppWidgetOptions(eq(WIDGET_A_ID)))
+                .thenReturn(mOptionsWidgetA);
+        when(mAppWidgetManagerMock.getAppWidgetOptions(eq(WIDGET_B_ID)))
+                .thenReturn(mOptionsWidgetB);
+
+        // Blanket intent that defines which widget IDs we would be testing here.
+        mIntent = new Intent();
+        mIntent.putExtra(AppWidgetManager.EXTRA_APPWIDGET_IDS, WIDGET_IDS);
+    }
+
+    private void updateReportedWidgetSizes(Bundle options, SizeF portrait, SizeF landscape) {
+        // - Portrait mode is narrow and tall, hence MIN_WIDTH and MAX_HEIGHT.
+        // - Landscape mode is wide and short, hence MAX_WIDTH and MIN_HEIGHT.
+        options.putInt(AppWidgetManager.OPTION_APPWIDGET_MIN_WIDTH, (int) portrait.getWidth());
+        options.putInt(AppWidgetManager.OPTION_APPWIDGET_MAX_HEIGHT, (int) portrait.getHeight());
+        options.putInt(AppWidgetManager.OPTION_APPWIDGET_MAX_WIDTH, (int) landscape.getWidth());
+        options.putInt(AppWidgetManager.OPTION_APPWIDGET_MIN_HEIGHT, (int) landscape.getHeight());
     }
 
     @Test
     @SmallTest
-    public void testAppWidgetUpdateInvokesUpdateWidgets() {
-        Intent appWidgetUpdateIntent = new Intent(AppWidgetManager.ACTION_APPWIDGET_UPDATE);
-        appWidgetUpdateIntent.putExtra(AppWidgetManager.EXTRA_APPWIDGET_IDS, WIDGET_IDS);
+    public void testAppWidgetInstallationCreatesWidgets() {
+        mIntent.setAction(AppWidgetManager.ACTION_APPWIDGET_UPDATE);
+        updateReportedWidgetSizes(mOptionsWidgetA, new SizeF(80, 80), new SizeF(400, 40));
+        updateReportedWidgetSizes(mOptionsWidgetB, new SizeF(30, 10), new SizeF(100, 30));
+        mWidgetProvider.onReceive(mContext, mIntent);
 
-        mWidgetProvider.onReceive(mContext, appWidgetUpdateIntent);
+        // There are 2 fake widgets to be created, and each should be created for portrait and
+        // landscape screen orientation.
+        // Widget A, Portrait.
+        verify(mWidgetProvider).createWidget(any(), any(), eq(80), eq(80));
+        // Widget A, Landscape.
+        verify(mWidgetProvider).createWidget(any(), any(), eq(400), eq(40));
+        // Widget B, Portrait.
+        verify(mWidgetProvider).createWidget(any(), any(), eq(30), eq(10));
+        // Widget B, Landscape.
+        verify(mWidgetProvider).createWidget(any(), any(), eq(100), eq(30));
+        // 4 total, no more.
+        verify(mWidgetProvider, times(4)).createWidget(any(), any(), anyInt(), anyInt());
+    }
 
-        verify(mWidgetProvider, times(1)).onUpdate(mContext, mAppWidgetManagerMock, WIDGET_IDS);
-        verify(mWidgetProvider, times(1)).onUpdate(any(), any(), any());
+    @Test
+    @SmallTest
+    public void testAppWidgetResizeUpdatesWidgets() {
+        updateReportedWidgetSizes(mOptionsWidgetA, new SizeF(80, 80), new SizeF(400, 40));
+        updateReportedWidgetSizes(mOptionsWidgetB, new SizeF(30, 10), new SizeF(100, 30));
+        mIntent.setAction(AppWidgetManager.ACTION_APPWIDGET_UPDATE);
+        mWidgetProvider.onReceive(mContext, mIntent);
+
+        verify(mWidgetProvider).createWidget(any(), any(), eq(80), eq(80));
+        verify(mWidgetProvider).createWidget(any(), any(), eq(400), eq(40));
+        verify(mWidgetProvider).createWidget(any(), any(), eq(30), eq(10));
+        verify(mWidgetProvider).createWidget(any(), any(), eq(100), eq(30));
+        verify(mWidgetProvider, times(4)).createWidget(any(), any(), anyInt(), anyInt());
+
+        clearInvocations(mWidgetProvider);
+
+        // OPTIONS_CHANGED specifies which particular widget needs to be updated. Make sure we
+        // reflect that here. Below, we flip the sizes so that Widget B uses sizes of Widget A and
+        // vice versa.
+        when(mAppWidgetManagerMock.getAppWidgetOptions(eq(WIDGET_A_ID)))
+                .thenReturn(mOptionsWidgetB);
+        when(mAppWidgetManagerMock.getAppWidgetOptions(eq(WIDGET_B_ID)))
+                .thenReturn(mOptionsWidgetA);
+
+        // First, resize Widget A with old Widget B size specs.
+        mIntent.setAction(AppWidgetManager.ACTION_APPWIDGET_OPTIONS_CHANGED);
+        mIntent.putExtra(AppWidgetManager.EXTRA_APPWIDGET_ID, WIDGET_A_ID);
+        mIntent.putExtra(AppWidgetManager.EXTRA_APPWIDGET_OPTIONS, mOptionsWidgetB);
+        mWidgetProvider.onReceive(mContext, mIntent);
+
+        verify(mWidgetProvider).createWidget(any(), any(), eq(30), eq(10));
+        verify(mWidgetProvider).createWidget(any(), any(), eq(100), eq(30));
+        verify(mWidgetProvider, times(2)).createWidget(any(), any(), anyInt(), anyInt());
+        clearInvocations(mWidgetProvider);
+
+        // Next, resize Widget B with old Widget A size specs.
+        mIntent.putExtra(AppWidgetManager.EXTRA_APPWIDGET_ID, WIDGET_B_ID);
+        mIntent.putExtra(AppWidgetManager.EXTRA_APPWIDGET_OPTIONS, mOptionsWidgetA);
+        mWidgetProvider.onReceive(mContext, mIntent);
+        verify(mWidgetProvider).createWidget(any(), any(), eq(80), eq(80));
+        verify(mWidgetProvider).createWidget(any(), any(), eq(400), eq(40));
+        verify(mWidgetProvider, times(2)).createWidget(any(), any(), anyInt(), anyInt());
+    }
+
+    @Test
+    @SmallTest
+    @Config(sdk = Build.VERSION_CODES.S)
+    public void testCreateWidgetsFromFallbackValues_missingSizes() {
+        updateReportedWidgetSizes(mOptionsWidgetA, new SizeF(80, 80), new SizeF(400, 40));
+        mWidgetProvider.getRemoteViews(mContext, mPreferences, mOptionsWidgetA);
 
         // There are 2 fake widgets that we work with, so expect both being evaluated
-        verify(mWidgetProvider, times(2)).getRemoteViews(any(), any(), any(), anyInt());
+        // One for portrait mode (max_width, min_height) and
+        // One for landscape mode (min_width, max_height).
+        verify(mWidgetProvider).createWidget(any(), any(), eq(400), eq(40));
+        verify(mWidgetProvider).createWidget(any(), any(), eq(80), eq(80));
+        verify(mWidgetProvider, times(2)).createWidget(any(), any(), anyInt(), anyInt());
+    }
+
+    @Test
+    @SmallTest
+    @Config(sdk = Build.VERSION_CODES.S)
+    public void testCreateWidgetFromFallbackValues_emptySizes() {
+        updateReportedWidgetSizes(mOptionsWidgetA, new SizeF(80, 80), new SizeF(400, 40));
+        // Lastly, set the empty array of sizes.
+        mOptionsWidgetA.putParcelableArrayList(
+                AppWidgetManager.OPTION_APPWIDGET_SIZES, new ArrayList<SizeF>());
+        mWidgetProvider.getRemoteViews(mContext, mPreferences, mOptionsWidgetA);
+
+        // There are 2 fake widgets that we work with, so expect both being evaluated
+        // One for portrait mode (max_width, min_height) and
+        // One for landscape mode (min_width, max_height).
+        verify(mWidgetProvider).createWidget(any(), any(), eq(400), eq(40));
+        verify(mWidgetProvider).createWidget(any(), any(), eq(80), eq(80));
+        verify(mWidgetProvider, times(2)).createWidget(any(), any(), anyInt(), anyInt());
+    }
+
+    @Test
+    @SmallTest
+    @Config(sdk = Build.VERSION_CODES.S)
+    public void testCreateWidgetFromSizeSpecs() {
+        updateReportedWidgetSizes(mOptionsWidgetA, new SizeF(80, 80), new SizeF(400, 40));
+        // Lastly, set a different array of sizes and confirm it is used instead.
+        mOptionsWidgetA.putParcelableArrayList(AppWidgetManager.OPTION_APPWIDGET_SIZES,
+                new ArrayList<SizeF>(Arrays.asList(new SizeF(50, 50))));
+        mWidgetProvider.getRemoteViews(mContext, mPreferences, mOptionsWidgetA);
+
+        // Only one call is expected here, because we declare only one size in our list.
+        verify(mWidgetProvider).createWidget(any(), any(), eq(50), eq(50));
+        verify(mWidgetProvider, times(1)).createWidget(any(), any(), anyInt(), anyInt());
+    }
+
+    @Test
+    @SmallTest
+    @Config(sdk = Build.VERSION_CODES.R)
+    public void testCreateWidgetFromLegacyMeasurements() {
+        updateReportedWidgetSizes(mOptionsWidgetA, new SizeF(80, 80), new SizeF(400, 40));
+        // Define new sizes. These must be ignored, because RemoteViews constructor is not
+        // supported.
+        mOptionsWidgetA.putParcelableArrayList(AppWidgetManager.OPTION_APPWIDGET_SIZES,
+                new ArrayList<SizeF>(Arrays.asList(new SizeF(50, 50))));
+        mWidgetProvider.getRemoteViews(mContext, mPreferences, mOptionsWidgetA);
+
+        // There are 2 fake widgets that we work with, so expect both being evaluated
+        // One for portrait mode (max_width, min_height) and
+        // One for landscape mode (min_width, max_height).
+        verify(mWidgetProvider).createWidget(any(), any(), eq(400), eq(40));
+        verify(mWidgetProvider).createWidget(any(), any(), eq(80), eq(80));
+        verify(mWidgetProvider, times(2)).createWidget(any(), any(), anyInt(), anyInt());
     }
 }
diff --git a/chrome/android/junit/src/org/chromium/chrome/browser/share/LensUtilsTest.java b/chrome/android/junit/src/org/chromium/chrome/browser/share/LensUtilsTest.java
index 1d63bb67..812ce7f 100644
--- a/chrome/android/junit/src/org/chromium/chrome/browser/share/LensUtilsTest.java
+++ b/chrome/android/junit/src/org/chromium/chrome/browser/share/LensUtilsTest.java
@@ -114,9 +114,9 @@
     }
 
     @Test
-    public void isGoogleLensFeatureEnabled_tabletDisabled() {
+    public void isGoogleLensFeatureEnabled_tabletEnabledByDefault() {
         configureFeature(ChromeFeatureList.CONTEXT_MENU_SEARCH_WITH_GOOGLE_LENS);
-        Assert.assertFalse("Feature incorrectly enabled when Lens on tablet was disabled",
+        Assert.assertTrue("Feature incorrectly disabled when Lens on tablet was enabled",
                 LensUtils.isGoogleLensFeatureEnabledOnTablet());
     }
 
diff --git a/chrome/app/generated_resources.grd b/chrome/app/generated_resources.grd
index 5eccb867..19b282f 100644
--- a/chrome/app/generated_resources.grd
+++ b/chrome/app/generated_resources.grd
@@ -6674,6 +6674,9 @@
       <message name="IDS_NTP_CUSTOMIZE_NO_BACKGROUND_LABEL" desc="The label for the 'Classic Chrome' tile in the customization menu on the New Tab Page">
         Classic Chrome
       </message>
+      <message name="IDS_NTP_CUSTOMIZE_UPLOADED_IMAGE_LABEL" desc="The label for the 'Uploaded image' tile in the customization menu on the New Tab Page side panel.">
+        Uploaded image
+      </message>
       <message name="IDS_NTP_CUSTOMIZE_UPLOAD_FROM_DEVICE_LABEL" desc="The label for the 'Upload from device' tile in the customization menu on the New Tab Page">
         Upload from device
       </message>
@@ -13069,6 +13072,12 @@
       <message name="IDS_WEBAUTHN_ERROR_NO_TRANSPORTS_DESCRIPTION" desc="Description in the dialog shown when the user could not sign in to a web site, because their computer did not support any of the hardware-based authentication mechanisms desired by the web site.">
         This device doesn't support the type of security key requested by this website
       </message>
+      <message name="IDS_WEBAUTHN_ERROR_NO_PASSKEYS_TITLE" desc="Title of the dialog shown when the user could not sign in to a web site because their computer did not have any of the passkeys registered with the site.">
+        No passkeys available
+      </message>
+      <message name="IDS_WEBAUTHN_ERROR_NO_PASSKEYS_DESCRIPTION" desc="Description in the dialog shown when the user could not sign in to a web site because their computer did not have any of the passkeys registered with the site.">
+        There aren't any passkeys for <ph name="APP_NAME">$1<ex>google.com</ex></ph> on this device
+      </message>
       <message name="IDS_WEBAUTHN_BLUETOOTH_POWER_ON_AUTO_TITLE" desc="Title of the dialog informing the user that Chrome needs to turn on Bluetooth by itself so that security keys can be used over Bluetooth.">
         Turn on Bluetooth?
       </message>
diff --git a/chrome/app/generated_resources_grd/IDS_NTP_CUSTOMIZE_UPLOADED_IMAGE_LABEL.png.sha1 b/chrome/app/generated_resources_grd/IDS_NTP_CUSTOMIZE_UPLOADED_IMAGE_LABEL.png.sha1
new file mode 100644
index 0000000..ee9f695
--- /dev/null
+++ b/chrome/app/generated_resources_grd/IDS_NTP_CUSTOMIZE_UPLOADED_IMAGE_LABEL.png.sha1
@@ -0,0 +1 @@
+e91b85af596088e04af1f5e35e42e39dd5173dac
\ No newline at end of file
diff --git a/chrome/app/generated_resources_grd/IDS_WEBAUTHN_ERROR_NO_PASSKEYS_DESCRIPTION.png.sha1 b/chrome/app/generated_resources_grd/IDS_WEBAUTHN_ERROR_NO_PASSKEYS_DESCRIPTION.png.sha1
new file mode 100644
index 0000000..68e115f
--- /dev/null
+++ b/chrome/app/generated_resources_grd/IDS_WEBAUTHN_ERROR_NO_PASSKEYS_DESCRIPTION.png.sha1
@@ -0,0 +1 @@
+d66e3b8f10c2f6b6ecb34b20b33ff19a3028450d
\ No newline at end of file
diff --git a/chrome/app/generated_resources_grd/IDS_WEBAUTHN_ERROR_NO_PASSKEYS_TITLE.png.sha1 b/chrome/app/generated_resources_grd/IDS_WEBAUTHN_ERROR_NO_PASSKEYS_TITLE.png.sha1
new file mode 100644
index 0000000..68e115f
--- /dev/null
+++ b/chrome/app/generated_resources_grd/IDS_WEBAUTHN_ERROR_NO_PASSKEYS_TITLE.png.sha1
@@ -0,0 +1 @@
+d66e3b8f10c2f6b6ecb34b20b33ff19a3028450d
\ No newline at end of file
diff --git a/chrome/browser/BUILD.gn b/chrome/browser/BUILD.gn
index 44c56811..a6f5f77d 100644
--- a/chrome/browser/BUILD.gn
+++ b/chrome/browser/BUILD.gn
@@ -5590,6 +5590,8 @@
       "lacros/task_manager_lacros.h",
       "lacros/ui_metric_recorder_lacros.cc",
       "lacros/ui_metric_recorder_lacros.h",
+      "lacros/views_text_services_context_menu_lacros.cc",
+      "lacros/views_text_services_context_menu_lacros.h",
       "lacros/vpn_extension_tracker_lacros.cc",
       "lacros/vpn_extension_tracker_lacros.h",
       "lacros/web_app_provider_bridge_lacros.cc",
@@ -5661,6 +5663,7 @@
       "//ui/chromeos/strings:strings_grit",
       "//ui/chromeos/styles:cros_styles_views",
       "//ui/platform_window",
+      "//ui/strings:ui_strings_grit",
     ]
 
     if (enable_plugins) {
@@ -6393,6 +6396,8 @@
       "enterprise/idle/idle_service.h",
       "enterprise/idle/idle_service_factory.cc",
       "enterprise/idle/idle_service_factory.h",
+      "enterprise/idle/idle_timeout_policy_handler.cc",
+      "enterprise/idle/idle_timeout_policy_handler.h",
       "enterprise/remote_commands/rotate_attestation_credential_job.cc",
       "enterprise/remote_commands/rotate_attestation_credential_job.h",
       "enterprise/signals/signals_aggregator_factory.cc",
diff --git a/chrome/browser/about_flags.cc b/chrome/browser/about_flags.cc
index a4b2e2e..0810df9 100644
--- a/chrome/browser/about_flags.cc
+++ b/chrome/browser/about_flags.cc
@@ -4190,9 +4190,6 @@
      flag_descriptions::kVoiceButtonInTopToolbarName,
      flag_descriptions::kVoiceButtonInTopToolbarDescription, kOsAndroid,
      FEATURE_VALUE_TYPE(chrome::android::kVoiceButtonInTopToolbar)},
-    {"assistant-consent-modal", flag_descriptions::kAssistantConsentModalName,
-     flag_descriptions::kAssistantConsentModalDescription, kOsAndroid,
-     FEATURE_VALUE_TYPE(chrome::android::kAssistantConsentModal)},
     {"assistant-consent-simplified-text",
      flag_descriptions::kAssistantConsentSimplifiedTextName,
      flag_descriptions::kAssistantConsentSimplifiedTextDescription, kOsAndroid,
@@ -9642,6 +9639,11 @@
               entry.internal_name)) {
     return true;
   }
+  // Skip lacros-selection if it is controlled by LacrosSelection policy.
+  if (!strcmp(kLacrosSelectionInternalName, entry.internal_name)) {
+    return crosapi::browser_util::GetCachedLacrosSelectionPolicy() !=
+           crosapi::browser_util::LacrosSelectionPolicy::kUserChoice;
+  }
 
   if (!strcmp(kPreferDcheckInternalName, entry.internal_name)) {
     return !crosapi::browser_util::IsLacrosAllowedToBeEnabled();
diff --git a/chrome/browser/accessibility/page_colors_factory.cc b/chrome/browser/accessibility/page_colors_factory.cc
index 429b3166..cbc932a 100644
--- a/chrome/browser/accessibility/page_colors_factory.cc
+++ b/chrome/browser/accessibility/page_colors_factory.cc
@@ -47,10 +47,11 @@
   return true;
 }
 
-KeyedService* PageColorsFactory::BuildServiceInstanceFor(
+std::unique_ptr<KeyedService>
+PageColorsFactory::BuildServiceInstanceForBrowserContext(
     content::BrowserContext* context) const {
-  std::unique_ptr<PageColors> page_colors = std::make_unique<PageColors>(
+  auto page_colors = std::make_unique<PageColors>(
       Profile::FromBrowserContext(context)->GetPrefs());
   page_colors->Init();
-  return page_colors.release();
+  return page_colors;
 }
diff --git a/chrome/browser/accessibility/page_colors_factory.h b/chrome/browser/accessibility/page_colors_factory.h
index e560f3a..1022b84a 100644
--- a/chrome/browser/accessibility/page_colors_factory.h
+++ b/chrome/browser/accessibility/page_colors_factory.h
@@ -28,8 +28,8 @@
   // BrowserContextKeyedServiceFactory:
   content::BrowserContext* GetBrowserContextToUse(
       content::BrowserContext* context) const override;
-  KeyedService* BuildServiceInstanceFor(
-      content::BrowserContext* profile) const override;
+  std::unique_ptr<KeyedService> BuildServiceInstanceForBrowserContext(
+      content::BrowserContext* context) const override;
   bool ServiceIsCreatedWithBrowserContext() const override;
   void RegisterProfilePrefs(
       user_prefs::PrefRegistrySyncable* registry) override;
diff --git a/chrome/browser/android/compositor/scene_layer/tab_strip_scene_layer.cc b/chrome/browser/android/compositor/scene_layer/tab_strip_scene_layer.cc
index af57558..881ef38 100644
--- a/chrome/browser/android/compositor/scene_layer/tab_strip_scene_layer.cc
+++ b/chrome/browser/android/compositor/scene_layer/tab_strip_scene_layer.cc
@@ -18,6 +18,7 @@
 
 using base::android::JavaParamRef;
 using base::android::JavaRef;
+bool tab_strip_redesign_enabled;
 
 namespace android {
 
@@ -28,12 +29,14 @@
       scrollable_strip_layer_(cc::Layer::Create()),
       scrim_layer_(cc::SolidColorLayer::Create()),
       new_tab_button_(cc::UIResourceLayer::Create()),
+      new_tab_button_background_(cc::UIResourceLayer::Create()),
       left_fade_(cc::UIResourceLayer::Create()),
       right_fade_(cc::UIResourceLayer::Create()),
       model_selector_button_(cc::UIResourceLayer::Create()),
       write_index_(0),
       content_tree_(nullptr) {
   new_tab_button_->SetIsDrawable(true);
+  new_tab_button_background_->SetIsDrawable(true);
   model_selector_button_->SetIsDrawable(true);
   left_fade_->SetIsDrawable(true);
   right_fade_->SetIsDrawable(true);
@@ -45,6 +48,8 @@
   scrollable_strip_layer_->SetIsDrawable(true);
   const bool tab_strip_improvements_enabled =
       base::FeatureList::IsEnabled(chrome::android::kTabStripImprovements);
+  tab_strip_redesign_enabled =
+      base::FeatureList::IsEnabled(chrome::android::kTabStripRedesign);
   if (!tab_strip_improvements_enabled) {
     scrollable_strip_layer_->AddChild(new_tab_button_);
   }
@@ -56,6 +61,9 @@
   tab_strip_layer_->AddChild(right_fade_);
   tab_strip_layer_->AddChild(model_selector_button_);
   if (tab_strip_improvements_enabled) {
+    if (tab_strip_redesign_enabled) {
+      tab_strip_layer_->AddChild(new_tab_button_background_);
+    }
     tab_strip_layer_->AddChild(new_tab_button_);
   }
   tab_strip_layer_->AddChild(scrim_layer_);
@@ -163,13 +171,13 @@
     JNIEnv* env,
     const JavaParamRef<jobject>& jobj,
     jint resource_id,
+    jint bg_resource_id,
     jfloat x,
     jfloat y,
-    jfloat width,
-    jfloat height,
     jfloat touch_target_offset,
     jboolean visible,
     jint tint,
+    jint background_tint,
     jfloat button_alpha,
     const JavaParamRef<jobject>& jresource_manager) {
   ui::ResourceManager* resource_manager =
@@ -178,16 +186,40 @@
       resource_manager->GetStaticResourceWithTint(resource_id, tint);
 
   new_tab_button_->SetUIResourceId(button_resource->ui_resource()->id());
-  float left_offset = (width - button_resource->size().width()) / 2;
-  float top_offset = (height - button_resource->size().height()) / 2;
   // The touch target for the new tab button is skewed towards the end of the
   // strip. This ensures that the view itself is correctly aligned without
   // adjusting the touch target.
-  left_offset += touch_target_offset;
-  new_tab_button_->SetPosition(gfx::PointF(x + left_offset, y + top_offset));
+  float left_offset = touch_target_offset;
+
   new_tab_button_->SetBounds(button_resource->size());
   new_tab_button_->SetHideLayerAndSubtree(!visible);
   new_tab_button_->SetOpacity(button_alpha);
+
+  // Set Tab Strip Redesign new tab button background
+  if (tab_strip_redesign_enabled) {
+    ui::Resource* button_background_resource =
+        resource_manager->GetStaticResourceWithTint(bg_resource_id,
+                                                    background_tint, true);
+    float background_left_offset = (button_background_resource->size().width() -
+                                    button_resource->size().width()) /
+                                   2;
+    float background_top_offset = (button_background_resource->size().height() -
+                                   button_resource->size().height()) /
+                                  2;
+    new_tab_button_background_->SetUIResourceId(
+        button_background_resource->ui_resource()->id());
+    new_tab_button_background_->SetPosition(gfx::PointF(x + left_offset, y));
+
+    new_tab_button_background_->SetBounds(button_background_resource->size());
+    new_tab_button_background_->SetHideLayerAndSubtree(!visible);
+    new_tab_button_background_->SetOpacity(button_alpha);
+    new_tab_button_->SetPosition(
+        gfx::PointF(background_left_offset, background_top_offset));
+    new_tab_button_background_->AddChild(new_tab_button_);
+  } else {
+    // Only show new tab button icon when TSR is disabled
+    new_tab_button_->SetPosition(gfx::PointF(x + left_offset, y));
+  }
 }
 
 void TabStripSceneLayer::UpdateModelSelectorButton(
diff --git a/chrome/browser/android/compositor/scene_layer/tab_strip_scene_layer.h b/chrome/browser/android/compositor/scene_layer/tab_strip_scene_layer.h
index ef96dce..6a84473 100644
--- a/chrome/browser/android/compositor/scene_layer/tab_strip_scene_layer.h
+++ b/chrome/browser/android/compositor/scene_layer/tab_strip_scene_layer.h
@@ -69,13 +69,13 @@
       JNIEnv* env,
       const base::android::JavaParamRef<jobject>& jobj,
       jint resource_id,
+      jint bg_resource_id,
       jfloat x,
       jfloat y,
-      jfloat width,
-      jfloat height,
       jfloat touch_target_offset,
       jboolean visible,
       jint tint,
+      jint background_tint,
       jfloat button_alpha,
       const base::android::JavaParamRef<jobject>& jresource_manager);
 
@@ -152,6 +152,7 @@
   scoped_refptr<cc::Layer> scrollable_strip_layer_;
   scoped_refptr<cc::SolidColorLayer> scrim_layer_;
   scoped_refptr<cc::UIResourceLayer> new_tab_button_;
+  scoped_refptr<cc::UIResourceLayer> new_tab_button_background_;
   scoped_refptr<cc::UIResourceLayer> left_fade_;
   scoped_refptr<cc::UIResourceLayer> right_fade_;
   scoped_refptr<cc::UIResourceLayer> model_selector_button_;
diff --git a/chrome/browser/ash/BUILD.gn b/chrome/browser/ash/BUILD.gn
index 77f999a..5de5844 100644
--- a/chrome/browser/ash/BUILD.gn
+++ b/chrome/browser/ash/BUILD.gn
@@ -523,6 +523,8 @@
     "arc/session/arc_demo_mode_preference_handler.h",
     "arc/session/arc_disk_space_monitor.cc",
     "arc/session/arc_disk_space_monitor.h",
+    "arc/session/arc_initial_optin_notifier.cc",
+    "arc/session/arc_initial_optin_notifier.h",
     "arc/session/arc_play_store_enabled_preference_handler.cc",
     "arc/session/arc_play_store_enabled_preference_handler.h",
     "arc/session/arc_provisioning_result.cc",
@@ -2315,6 +2317,8 @@
     "policy/handlers/device_wifi_allowed_handler.h",
     "policy/handlers/lacros_availability_policy_handler.cc",
     "policy/handlers/lacros_availability_policy_handler.h",
+    "policy/handlers/lacros_selection_policy_handler.cc",
+    "policy/handlers/lacros_selection_policy_handler.h",
     "policy/handlers/lock_to_single_user_manager.cc",
     "policy/handlers/lock_to_single_user_manager.h",
     "policy/handlers/minimum_version_policy_handler.cc",
diff --git a/chrome/browser/ash/app_list/arc/DEPS b/chrome/browser/ash/app_list/arc/DEPS
new file mode 100644
index 0000000..b118d9f
--- /dev/null
+++ b/chrome/browser/ash/app_list/arc/DEPS
@@ -0,0 +1,5 @@
+specific_include_rules = {
+  "arc_app_list_prefs\.cc": [
+     "+ash",
+  ],
+}
diff --git a/chrome/browser/ash/app_list/arc/arc_app_list_prefs.cc b/chrome/browser/ash/app_list/arc/arc_app_list_prefs.cc
index 0faafbfc..56f8471 100644
--- a/chrome/browser/ash/app_list/arc/arc_app_list_prefs.cc
+++ b/chrome/browser/ash/app_list/arc/arc_app_list_prefs.cc
@@ -18,6 +18,8 @@
 #include "ash/components/arc/session/connection_holder.h"
 #include "ash/constants/ash_features.h"
 #include "ash/constants/ash_switches.h"
+#include "ash/metrics/login_unlock_throughput_recorder.h"
+#include "ash/shell.h"
 #include "base/bind.h"
 #include "base/check.h"
 #include "base/containers/contains.h"
@@ -43,6 +45,7 @@
 #include "chrome/browser/ash/login/session/user_session_manager.h"
 #include "chrome/browser/image_decoder/image_decoder.h"
 #include "chrome/browser/profiles/profile.h"
+#include "chrome/browser/profiles/profile_manager.h"
 #include "chrome/browser/ui/ash/shelf/chrome_shelf_controller.h"
 #include "chrome/common/chrome_features.h"
 #include "chrome/grit/generated_resources.h"
@@ -376,6 +379,44 @@
   update.Get().Remove(kDeprecatePackagePrefsSystem);
 }
 
+ash::LoginUnlockThroughputRecorder* GetLoginRecorder() {
+  return ash::Shell::HasInstance()
+             ? ash::Shell::Get()->login_unlock_throughput_recorder()
+             : nullptr;
+}
+
+void OnArcAppListRefreshed(Profile* profile) {
+  if (!arc::IsArcPlayStoreEnabledForProfile(profile))
+    return;
+
+  ash::LoginUnlockThroughputRecorder* throughput_recorder = GetLoginRecorder();
+  if (!throughput_recorder || !throughput_recorder->NeedReportArcAppListReady())
+    return;
+
+  DCHECK_EQ(ProfileManager::GetPrimaryUserProfile(), profile);
+  auto* prefs = ArcAppListPrefs::Get(profile);
+  if (!prefs)
+    return;
+
+  const std::vector<std::string> app_ids = prefs->GetAppIds();
+  int launchable = 0;
+  int ready = 0;
+  int error = 0;
+  for (const auto& app_id : app_ids) {
+    std::unique_ptr<ArcAppListPrefs::AppInfo> app_info = prefs->GetApp(app_id);
+    if (app_info) {
+      if (app_info->launchable)
+        ++launchable;
+
+      if (app_info->ready)
+        ++ready;
+    } else {
+      ++error;
+    }
+  }
+  if (ready + error >= launchable)
+    throughput_recorder->OnArcAppListReady();
+}
 }  // namespace
 
 // static
@@ -1589,6 +1630,7 @@
       }
     }
   }
+  OnArcAppListRefreshed(profile_);
 }
 
 void ArcAppListPrefs::RemoveApp(const std::string& app_id) {
@@ -1784,6 +1826,7 @@
       MaybeSetDefaultAppLoadingTimeout();
     }
   }
+  OnArcAppListRefreshed(profile_);
 }
 
 void ArcAppListPrefs::DetectDefaultAppAvailability() {
diff --git a/chrome/browser/ash/arc/session/arc_initial_optin_notifier.cc b/chrome/browser/ash/arc/session/arc_initial_optin_notifier.cc
new file mode 100644
index 0000000..ea28e28
--- /dev/null
+++ b/chrome/browser/ash/arc/session/arc_initial_optin_notifier.cc
@@ -0,0 +1,107 @@
+// Copyright 2022 The Chromium Authors
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include "chrome/browser/ash/arc/session/arc_initial_optin_notifier.h"
+
+#include <string>
+
+#include "ash/components/arc/arc_util.h"
+#include "ash/metrics/login_unlock_throughput_recorder.h"
+#include "ash/shell.h"
+#include "base/bind.h"
+#include "base/logging.h"
+#include "chrome/browser/ash/arc/arc_util.h"
+#include "chrome/browser/ash/arc/session/arc_session_manager.h"
+#include "chrome/browser/profiles/profile.h"
+#include "components/keyed_service/content/browser_context_dependency_manager.h"
+#include "content/public/browser/browser_thread.h"
+
+namespace arc {
+
+namespace {
+
+class ArcInitialOptInNotifierFactory
+    : public BrowserContextKeyedServiceFactory {
+ public:
+  ArcInitialOptInNotifierFactory();
+
+  ArcInitialOptInNotifierFactory(const ArcInitialOptInNotifierFactory&) =
+      delete;
+  ArcInitialOptInNotifierFactory& operator=(
+      const ArcInitialOptInNotifierFactory&) = delete;
+
+  ~ArcInitialOptInNotifierFactory() override = default;
+
+  KeyedService* BuildServiceInstanceFor(
+      content::BrowserContext* browser_context) const override {
+    return new ArcInitialOptInNotifier(browser_context);
+  }
+
+  // static
+  static ArcInitialOptInNotifier* GetForBrowserContext(
+      content::BrowserContext* context) {
+    return static_cast<ArcInitialOptInNotifier*>(
+        GetInstance()->GetServiceForBrowserContext(context, true));
+  }
+
+  // static
+  static ArcInitialOptInNotifierFactory* GetInstance() {
+    return base::Singleton<ArcInitialOptInNotifierFactory>::get();
+  }
+};
+
+ArcInitialOptInNotifierFactory::ArcInitialOptInNotifierFactory()
+    : BrowserContextKeyedServiceFactory(
+          "ArcInitialOptInNotifierFactory",
+          BrowserContextDependencyManager::GetInstance()) {}
+
+}  // anonymous namespace
+
+// static
+ArcInitialOptInNotifier* ArcInitialOptInNotifier::GetForProfile(
+    Profile* profile) {
+  return ArcInitialOptInNotifierFactory::GetForBrowserContext(profile);
+}
+
+ArcInitialOptInNotifier::ArcInitialOptInNotifier(
+    content::BrowserContext* context) {
+  ArcSessionManager* arc_session_manager = arc::ArcSessionManager::Get();
+  // arc::ArcSessionManager might not be set in tests.
+  if (arc_session_manager)
+    arc_session_manager->AddObserver(this);
+}
+
+ArcInitialOptInNotifier::~ArcInitialOptInNotifier() {
+  ArcSessionManager* arc_session_manager = arc::ArcSessionManager::Get();
+  // arc::ArcSessionManager may be released first.
+  if (arc_session_manager)
+    arc_session_manager->RemoveObserver(this);
+}
+
+void ArcInitialOptInNotifier::OnArcInitialStart() {
+  if (!IsArcPlayAutoInstallDisabled())
+    return;
+
+  LOG(WARNING) << "kArcDisablePlayAutoInstall flag is set. Force Arc apps "
+                  "loaded metric.";
+  ash::LoginUnlockThroughputRecorder* throughput_recorder =
+      ash::Shell::HasInstance()
+          ? ash::Shell::Get()->login_unlock_throughput_recorder()
+          : nullptr;
+  if (throughput_recorder) {
+    throughput_recorder->OnArcAppListReady();
+  }
+}
+
+void ArcInitialOptInNotifier::OnArcOptInUserAction() {
+  DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
+  ash::LoginUnlockThroughputRecorder* throughput_recorder =
+      ash::Shell::HasInstance()
+          ? ash::Shell::Get()->login_unlock_throughput_recorder()
+          : nullptr;
+  if (throughput_recorder)
+    throughput_recorder->OnArcOptedIn();
+}
+
+}  // namespace arc
diff --git a/chrome/browser/ash/arc/session/arc_initial_optin_notifier.h b/chrome/browser/ash/arc/session/arc_initial_optin_notifier.h
new file mode 100644
index 0000000..84b23b8
--- /dev/null
+++ b/chrome/browser/ash/arc/session/arc_initial_optin_notifier.h
@@ -0,0 +1,45 @@
+// Copyright 2022 The Chromium Authors
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#ifndef CHROME_BROWSER_ASH_ARC_SESSION_ARC_INITIAL_OPTIN_NOTIFIER_H_
+#define CHROME_BROWSER_ASH_ARC_SESSION_ARC_INITIAL_OPTIN_NOTIFIER_H_
+
+#include "base/memory/weak_ptr.h"
+#include "chrome/browser/ash/arc/session/arc_session_manager_observer.h"
+#include "components/keyed_service/core/keyed_service.h"
+
+class Profile;
+
+namespace content {
+class BrowserContext;
+}
+
+namespace arc {
+
+// Observes Arc session manager opt-in status, and notifies metrics.
+class ArcInitialOptInNotifier : public ArcSessionManagerObserver,
+                                public KeyedService {
+ public:
+  // Returns singleton instance for the given Profile.
+  static ArcInitialOptInNotifier* GetForProfile(Profile* profile);
+
+  explicit ArcInitialOptInNotifier(content::BrowserContext* context);
+
+  ArcInitialOptInNotifier(const ArcInitialOptInNotifier&) = delete;
+  ArcInitialOptInNotifier& operator=(const ArcInitialOptInNotifier&) = delete;
+
+  ~ArcInitialOptInNotifier() override;
+
+  // ArcSessionManagerObserver:
+  void OnArcInitialStart() override;
+  void OnArcOptInUserAction() override;
+
+ private:
+  // Must be the last member.
+  base::WeakPtrFactory<ArcInitialOptInNotifier> weak_ptr_factory_{this};
+};
+
+}  // namespace arc
+
+#endif  // CHROME_BROWSER_ASH_ARC_SESSION_ARC_INITIAL_OPTIN_NOTIFIER_H_
diff --git a/chrome/browser/ash/arc/session/arc_service_launcher.cc b/chrome/browser/ash/arc/session/arc_service_launcher.cc
index 00859b20..9be86d5 100644
--- a/chrome/browser/ash/arc/session/arc_service_launcher.cc
+++ b/chrome/browser/ash/arc/session/arc_service_launcher.cc
@@ -83,6 +83,7 @@
 #include "chrome/browser/ash/arc/screen_capture/arc_screen_capture_bridge.h"
 #include "chrome/browser/ash/arc/session/arc_demo_mode_preference_handler.h"
 #include "chrome/browser/ash/arc/session/arc_disk_space_monitor.h"
+#include "chrome/browser/ash/arc/session/arc_initial_optin_notifier.h"
 #include "chrome/browser/ash/arc/session/arc_play_store_enabled_preference_handler.h"
 #include "chrome/browser/ash/arc/session/arc_session_manager.h"
 #include "chrome/browser/ash/arc/sharesheet/arc_sharesheet_bridge.h"
@@ -310,6 +311,7 @@
   apps::ArcAppsFactory::GetForProfile(profile);
   ash::ApkWebAppService::Get(profile);
   ash::app_restore::AppRestoreArcTaskHandler::GetForProfile(profile);
+  ArcInitialOptInNotifier::GetForProfile(profile);
 
   if (arc::IsArcVmEnabled()) {
     // ARCVM-only services.
diff --git a/chrome/browser/ash/arc/session/arc_session_manager.cc b/chrome/browser/ash/arc/session/arc_session_manager.cc
index 0988602..5385f13 100644
--- a/chrome/browser/ash/arc/session/arc_session_manager.cc
+++ b/chrome/browser/ash/arc/session/arc_session_manager.cc
@@ -1243,6 +1243,9 @@
   switch (result) {
     case ArcRequirementChecker::RequirementCheckResult::kOk:
       VLOG(1) << "Starting ARC for first sign in.";
+      for (auto& observer : observer_list_)
+        observer.OnArcOptInUserAction();
+
       StartArc();
       break;
     case ArcRequirementChecker::RequirementCheckResult::
diff --git a/chrome/browser/ash/arc/session/arc_session_manager_observer.h b/chrome/browser/ash/arc/session/arc_session_manager_observer.h
index a4bcda3..f13f7f2 100644
--- a/chrome/browser/ash/arc/session/arc_session_manager_observer.h
+++ b/chrome/browser/ash/arc/session/arc_session_manager_observer.h
@@ -18,6 +18,11 @@
   // is represented by "arc.enabled" preference, is updated.
   virtual void OnArcPlayStoreEnabledChanged(bool enabled) {}
 
+  // Called with when user opted-in for ARC and ARC is going to be created for
+  // this user. I.e. successful user action triggered ARC user instance
+  // initialization.
+  virtual void OnArcOptInUserAction() {}
+
   // Called to notify that checking of Android management status started
   // during the opt-in flow.
   virtual void OnArcOptInManagementCheckStarted() {}
diff --git a/chrome/browser/ash/crosapi/browser_loader.cc b/chrome/browser/ash/crosapi/browser_loader.cc
index 90a0b2c5..839ccdb 100644
--- a/chrome/browser/ash/crosapi/browser_loader.cc
+++ b/chrome/browser/ash/crosapi/browser_loader.cc
@@ -206,21 +206,19 @@
     return;
   }
 
-  // If the user has specified to force using stateful or rootfs lacros-chrome
-  // binary, force the selection.
-  const base::CommandLine* cmdline = base::CommandLine::ForCurrentProcess();
-  if (cmdline->HasSwitch(browser_util::kLacrosSelectionSwitch)) {
+  // If the LacrosSelection policy or the user have specified to force using
+  // stateful or rootfs lacros-chrome binary, force the selection.
+  if (absl::optional<browser_util::LacrosSelection> lacros_selection =
+          browser_util::DetermineLacrosSelection()) {
     // TODO(crbug.com/1293250): We should check the version compatibility here,
     // too.
-    auto value =
-        cmdline->GetSwitchValueASCII(browser_util::kLacrosSelectionSwitch);
-    if (value == browser_util::kLacrosSelectionRootfs) {
-      LoadRootfsLacros(std::move(callback));
-      return;
-    }
-    if (value == browser_util::kLacrosSelectionStateful) {
-      LoadStatefulLacros(std::move(callback));
-      return;
+    switch (lacros_selection.value()) {
+      case browser_util::LacrosSelection::kRootfs:
+        LoadRootfsLacros(std::move(callback));
+        return;
+      case browser_util::LacrosSelection::kStateful:
+        LoadStatefulLacros(std::move(callback));
+        return;
     }
   }
 
diff --git a/chrome/browser/ash/crosapi/browser_loader_unittest.cc b/chrome/browser/ash/crosapi/browser_loader_unittest.cc
index 034d6cafa22..fc6c330 100644
--- a/chrome/browser/ash/crosapi/browser_loader_unittest.cc
+++ b/chrome/browser/ash/crosapi/browser_loader_unittest.cc
@@ -10,13 +10,17 @@
 #include "base/run_loop.h"
 #include "base/strings/utf_string_conversions.h"
 #include "base/test/bind.h"
+#include "base/test/scoped_command_line.h"
 #include "base/test/scoped_feature_list.h"
+#include "base/test/test_future.h"
 #include "chrome/browser/ash/crosapi/browser_util.h"
 #include "chrome/browser/browser_process.h"
 #include "chrome/browser/component_updater/fake_cros_component_manager.h"
 #include "chrome/test/base/browser_process_platform_part_test_api_chromeos.h"
 #include "chromeos/ash/components/dbus/upstart/fake_upstart_client.h"
 #include "components/component_updater/mock_component_updater_service.h"
+#include "components/policy/core/common/policy_map.h"
+#include "components/policy/policy_constants.h"
 #include "components/update_client/update_client.h"
 #include "content/public/test/browser_task_environment.h"
 #include "testing/gtest/include/gtest/gtest.h"
@@ -32,6 +36,33 @@
 constexpr char kLacrosMounterUpstartJob[] = "lacros_2dmounter";
 constexpr char kLacrosUnmounterUpstartJob[] = "lacros_2dunmounter";
 
+// This implementation of RAII for LacrosSelection is to make it easy reset
+// the state between runs.
+class ScopedLacrosSelectionCache {
+ public:
+  explicit ScopedLacrosSelectionCache(
+      browser_util::LacrosSelectionPolicy lacros_selection) {
+    SetLacrosSelection(lacros_selection);
+  }
+  ScopedLacrosSelectionCache(const ScopedLacrosSelectionCache&) = delete;
+  ScopedLacrosSelectionCache& operator=(const ScopedLacrosSelectionCache&) =
+      delete;
+  ~ScopedLacrosSelectionCache() {
+    browser_util::ClearLacrosSelectionCacheForTest();
+  }
+
+ private:
+  void SetLacrosSelection(
+      browser_util::LacrosSelectionPolicy lacros_selection) {
+    policy::PolicyMap policy;
+    policy.Set(policy::key::kLacrosSelection, policy::POLICY_LEVEL_MANDATORY,
+               policy::POLICY_SCOPE_USER, policy::POLICY_SOURCE_CLOUD,
+               base::Value(GetLacrosSelectionPolicyName(lacros_selection)),
+               /*external_data_fetcher=*/nullptr);
+    browser_util::CacheLacrosSelection(policy);
+  }
+};
+
 }  // namespace
 
 class BrowserLoaderTest : public testing::Test {
@@ -229,4 +260,70 @@
   EXPECT_TRUE(callback_called);
 }
 
+TEST_F(BrowserLoaderTest, OnLoadSelectionPolicyIsRootfs) {
+  ScopedLacrosSelectionCache cache(
+      browser_util::LacrosSelectionPolicy::kRootfs);
+  base::test::ScopedCommandLine command_line;
+  command_line.GetProcessCommandLine()->AppendSwitchASCII(
+      browser_util::kLacrosSelectionSwitch,
+      browser_util::kLacrosSelectionStateful);
+
+  base::test::TestFuture<base::FilePath, LacrosSelection, base::Version> future;
+  browser_loader_->Load(future.GetCallback<const base::FilePath&,
+                                           LacrosSelection, base::Version>());
+
+  const LacrosSelection selection = future.Get<1>();
+  EXPECT_EQ(selection, LacrosSelection::kRootfs);
+}
+
+TEST_F(BrowserLoaderTest, OnLoadSelectionPolicyIsStateful) {
+  ScopedLacrosSelectionCache cache(
+      browser_util::LacrosSelectionPolicy::kStateful);
+  base::test::ScopedCommandLine command_line;
+  command_line.GetProcessCommandLine()->AppendSwitchASCII(
+      browser_util::kLacrosSelectionSwitch,
+      browser_util::kLacrosSelectionRootfs);
+
+  base::test::TestFuture<base::FilePath, LacrosSelection, base::Version> future;
+  browser_loader_->Load(future.GetCallback<const base::FilePath&,
+                                           LacrosSelection, base::Version>());
+
+  const LacrosSelection selection = future.Get<1>();
+  EXPECT_EQ(selection, LacrosSelection::kStateful);
+}
+
+TEST_F(BrowserLoaderTest,
+       OnLoadSelectionPolicyIsUserChoiceAndCommandLineIsRootfs) {
+  ScopedLacrosSelectionCache cache(
+      browser_util::LacrosSelectionPolicy::kUserChoice);
+  base::test::ScopedCommandLine command_line;
+  command_line.GetProcessCommandLine()->AppendSwitchASCII(
+      browser_util::kLacrosSelectionSwitch,
+      browser_util::kLacrosSelectionRootfs);
+
+  base::test::TestFuture<base::FilePath, LacrosSelection, base::Version> future;
+  browser_loader_->Load(future.GetCallback<const base::FilePath&,
+                                           LacrosSelection, base::Version>());
+
+  const LacrosSelection selection = future.Get<1>();
+  EXPECT_EQ(selection, LacrosSelection::kRootfs);
+}
+
+TEST_F(BrowserLoaderTest,
+       OnLoadSelectionPolicyIsUserChoiceAndCommandLineIsStateful) {
+  ScopedLacrosSelectionCache cache(
+      browser_util::LacrosSelectionPolicy::kUserChoice);
+  base::test::ScopedCommandLine command_line;
+  command_line.GetProcessCommandLine()->AppendSwitchASCII(
+      browser_util::kLacrosSelectionSwitch,
+      browser_util::kLacrosSelectionStateful);
+
+  base::test::TestFuture<base::FilePath, LacrosSelection, base::Version> future;
+  browser_loader_->Load(future.GetCallback<const base::FilePath&,
+                                           LacrosSelection, base::Version>());
+
+  const LacrosSelection selection = future.Get<1>();
+  EXPECT_EQ(selection, LacrosSelection::kStateful);
+}
+
 }  // namespace crosapi
diff --git a/chrome/browser/ash/crosapi/browser_util.cc b/chrome/browser/ash/crosapi/browser_util.cc
index f1e7a51..1e9ac8b 100644
--- a/chrome/browser/ash/crosapi/browser_util.cc
+++ b/chrome/browser/ash/crosapi/browser_util.cc
@@ -61,12 +61,16 @@
 absl::optional<LacrosDataBackwardMigrationMode>
     g_lacros_data_backward_migration_mode;
 
+// At session start the value for LacrosSelection logic is applied and the
+// result is stored in this variable which is used after that as a cache.
+absl::optional<LacrosSelectionPolicy> g_lacros_selection_cache;
+
 // The rootfs lacros-chrome metadata keys.
 constexpr char kLacrosMetadataContentKey[] = "content";
 constexpr char kLacrosMetadataVersionKey[] = "version";
 
 // The conversion map for LacrosAvailability policy data. The values must match
-// the ones from policy_templates.json.
+// the ones from LacrosAvailability.yaml.
 constexpr auto kLacrosAvailabilityMap =
     base::MakeFixedFlatMap<base::StringPiece, LacrosAvailability>({
         {"user_choice", LacrosAvailability::kUserChoice},
@@ -90,6 +94,15 @@
          LacrosDataBackwardMigrationMode::kKeepAll},
     });
 
+// The conversion map for LacrosSelection policy data. The values must match
+// the ones from LacrosSelection.yaml.
+constexpr auto kLacrosSelectionPolicyMap =
+    base::MakeFixedFlatMap<base::StringPiece, LacrosSelectionPolicy>({
+        {"user_choice", LacrosSelectionPolicy::kUserChoice},
+        {"rootfs", LacrosSelectionPolicy::kRootfs},
+        {"stateful", LacrosSelectionPolicy::kStateful},
+    });
+
 // Some account types require features that aren't yet supported by lacros.
 // See https://crbug.com/1080693
 bool IsUserTypeAllowed(const User* user) {
@@ -904,6 +917,51 @@
       value ? value->GetString() : base::StringPiece());
 }
 
+void CacheLacrosSelection(const policy::PolicyMap& map) {
+  if (g_lacros_selection_cache.has_value()) {
+    // Some browser tests might call this multiple times.
+    LOG(ERROR) << "Trying to cache LacrosSelection and the value was set";
+    return;
+  }
+
+  const base::Value* value =
+      map.GetValue(policy::key::kLacrosSelection, base::Value::Type::STRING);
+  g_lacros_selection_cache = ParseLacrosSelectionPolicy(
+      value ? value->GetString() : base::StringPiece());
+}
+
+LacrosSelectionPolicy GetCachedLacrosSelectionPolicy() {
+  if (g_lacros_selection_cache)
+    return g_lacros_selection_cache.value();
+
+  return LacrosSelectionPolicy::kUserChoice;
+}
+
+absl::optional<LacrosSelection> DetermineLacrosSelection() {
+  switch (GetCachedLacrosSelectionPolicy()) {
+    case LacrosSelectionPolicy::kRootfs:
+      return LacrosSelection::kRootfs;
+    case LacrosSelectionPolicy::kStateful:
+      return LacrosSelection::kStateful;
+    case LacrosSelectionPolicy::kUserChoice:
+      break;
+  }
+
+  const base::CommandLine* cmdline = base::CommandLine::ForCurrentProcess();
+
+  if (!cmdline->HasSwitch(browser_util::kLacrosSelectionSwitch))
+    return absl::nullopt;
+
+  auto value =
+      cmdline->GetSwitchValueASCII(browser_util::kLacrosSelectionSwitch);
+  if (value == browser_util::kLacrosSelectionRootfs)
+    return LacrosSelection::kRootfs;
+  if (value == browser_util::kLacrosSelectionStateful)
+    return LacrosSelection::kStateful;
+
+  return absl::nullopt;
+}
+
 ComponentInfo GetLacrosComponentInfoForChannel(version_info::Channel channel) {
   // We default to the Dev component for UNKNOWN channels.
   static const auto kChannelToComponentInfoMap =
@@ -978,6 +1036,10 @@
   g_lacros_data_backward_migration_mode.reset();
 }
 
+void ClearLacrosSelectionCacheForTest() {
+  g_lacros_selection_cache.reset();
+}
+
 MigrationMode GetMigrationMode(const user_manager::User* user,
                                PolicyInitState policy_init_state) {
   if (base::FeatureList::IsEnabled(
@@ -1110,6 +1172,16 @@
   return absl::nullopt;
 }
 
+absl::optional<LacrosSelectionPolicy> ParseLacrosSelectionPolicy(
+    base::StringPiece value) {
+  auto* it = kLacrosSelectionPolicyMap.find(value);
+  if (it != kLacrosSelectionPolicyMap.end())
+    return it->second;
+
+  LOG(ERROR) << "Unknown LacrosSelection policy value is passed: " << value;
+  return absl::nullopt;
+}
+
 base::StringPiece GetLacrosAvailabilityPolicyName(LacrosAvailability value) {
   for (const auto& entry : kLacrosAvailabilityMap) {
     if (entry.second == value)
@@ -1142,6 +1214,16 @@
   return base::StringPiece();
 }
 
+base::StringPiece GetLacrosSelectionPolicyName(LacrosSelectionPolicy value) {
+  for (const auto& entry : kLacrosSelectionPolicyMap) {
+    if (entry.second == value)
+      return entry.first;
+  }
+
+  NOTREACHED();
+  return base::StringPiece();
+}
+
 bool IsAshBrowserSyncEnabled() {
   // Turn off sync from Ash if Lacros is enabled and Ash web browser is
   // disabled.
diff --git a/chrome/browser/ash/crosapi/browser_util.h b/chrome/browser/ash/crosapi/browser_util.h
index 5113159..4dfec38 100644
--- a/chrome/browser/ash/crosapi/browser_util.h
+++ b/chrome/browser/ash/crosapi/browser_util.h
@@ -80,6 +80,20 @@
   kLacrosOnly = 4
 };
 
+// Represents the policy indicating which Lacros browser to launch, named
+// LacrosSelection. The values shall be consistent with the controlling
+// policy. Unlike `LacrosSelection` representing which lacros to select,
+// `LacrosSelectionPolicy` represents how to decide which lacros to select.
+enum class LacrosSelectionPolicy {
+  // Indicates that the user decides which Lacros browser to launch: rootfs or
+  // stateful.
+  kUserChoice = 0,
+  // Indicates that rootfs Lacros will always be launched.
+  kRootfs = 1,
+  // Indicates that stateful Lacros will always be launched.
+  kStateful = 2,
+};
+
 // Represents the different options available for lacros selection.
 enum class LacrosSelection {
   kRootfs = 0,
@@ -351,6 +365,20 @@
 // LacrosDataBackwardMigrationMode policy.
 void CacheLacrosDataBackwardMigrationMode(const policy::PolicyMap& map);
 
+// To be called at primary user login, to cache the policy value for
+// LacrosSelection policy. The effective value of the policy does not
+// change for the duration of the user session, so cached value shall be
+// checked.
+void CacheLacrosSelection(const policy::PolicyMap& map);
+
+// Returns cached value of LacrosSelection policy. See `CacheLacrosSelection`
+// for details.
+LacrosSelectionPolicy GetCachedLacrosSelectionPolicy();
+
+// Returns lacros selection option according to LarcrosSelectionPolicy and
+// lacros-selection flag. Returns nullopt if there is no preference.
+absl::optional<LacrosSelection> DetermineLacrosSelection();
+
 // Returns the lacros ComponentInfo for a given channel.
 ComponentInfo GetLacrosComponentInfoForChannel(version_info::Channel channel);
 
@@ -385,6 +413,9 @@
 // Clears the cached value for LacrosDataBackwardMigrationMode.
 void ClearLacrosDataBackwardMigrationModeCacheForTest();
 
+// Clears the cached value for LacrosSelection policy.
+void ClearLacrosSelectionCacheForTest();
+
 bool IsProfileMigrationEnabled(const AccountId& account_id);
 
 // Returns true if the profile migration can run, but not yet completed.
@@ -458,7 +489,13 @@
 absl::optional<LacrosAvailability> ParseLacrosAvailability(
     base::StringPiece value);
 
-// Returns the policy value name from the given value.
+// Parses the string representation of LacrosSelection policy value into the
+// enum value. Returns nullopt on unknown value.
+absl::optional<LacrosSelectionPolicy> ParseLacrosSelectionPolicy(
+    base::StringPiece value);
+
+// Returns the policy LacrosAvailability value name from the given value.
+// Returned StringPiece is guaranteed to never be invalidated
 base::StringPiece GetLacrosAvailabilityPolicyName(LacrosAvailability value);
 
 // Parses the string representation of LacrosDataBackwardMigrationMode policy
@@ -470,6 +507,10 @@
 base::StringPiece GetLacrosDataBackwardMigrationModeName(
     LacrosDataBackwardMigrationMode value);
 
+// Returns the LacrosSelection policy value name from the given value. Returned
+// StringPiece is guaranteed to never be invalidated.
+base::StringPiece GetLacrosSelectionPolicyName(LacrosSelectionPolicy value);
+
 // Stores that "Go to files button" on the migration error screen is clicked.
 void SetGotoFilesClicked(PrefService* local_state,
                          const std::string& user_id_hash);
diff --git a/chrome/browser/ash/crosapi/browser_util_unittest.cc b/chrome/browser/ash/crosapi/browser_util_unittest.cc
index 36be553..0cd2a067 100644
--- a/chrome/browser/ash/crosapi/browser_util_unittest.cc
+++ b/chrome/browser/ash/crosapi/browser_util_unittest.cc
@@ -68,6 +68,33 @@
   }
 };
 
+// This implementation of RAII for LacrosSelection is to make it easy reset
+// the state between runs.
+class ScopedLacrosSelectionCache {
+ public:
+  explicit ScopedLacrosSelectionCache(
+      browser_util::LacrosSelectionPolicy lacros_selection) {
+    SetLacrosSelection(lacros_selection);
+  }
+  ScopedLacrosSelectionCache(const ScopedLacrosSelectionCache&) = delete;
+  ScopedLacrosSelectionCache& operator=(const ScopedLacrosSelectionCache&) =
+      delete;
+  ~ScopedLacrosSelectionCache() {
+    browser_util::ClearLacrosSelectionCacheForTest();
+  }
+
+ private:
+  void SetLacrosSelection(
+      browser_util::LacrosSelectionPolicy lacros_selection) {
+    policy::PolicyMap policy;
+    policy.Set(policy::key::kLacrosSelection, policy::POLICY_LEVEL_MANDATORY,
+               policy::POLICY_SCOPE_USER, policy::POLICY_SOURCE_CLOUD,
+               base::Value(GetLacrosSelectionPolicyName(lacros_selection)),
+               /*external_data_fetcher=*/nullptr);
+    browser_util::CacheLacrosSelection(policy);
+  }
+};
+
 }  // namespace
 
 class BrowserUtilTest : public testing::Test {
@@ -1116,4 +1143,64 @@
   EXPECT_EQ(serial_number.value(), expected_serial_number);
 }
 
+TEST_F(BrowserUtilTest, LacrosSelection) {
+  // Neither policy nor command line have any preference on Lacros selection.
+  EXPECT_FALSE(browser_util::DetermineLacrosSelection());
+
+  {
+    // LacrosSelection policy has precedence over command line.
+    ScopedLacrosSelectionCache cache(
+        browser_util::LacrosSelectionPolicy::kRootfs);
+    base::test::ScopedCommandLine cmd_line;
+    cmd_line.GetProcessCommandLine()->AppendSwitchASCII(
+        browser_util::kLacrosSelectionSwitch,
+        browser_util::kLacrosSelectionStateful);
+    EXPECT_EQ(browser_util::DetermineLacrosSelection(),
+              LacrosSelection::kRootfs);
+  }
+
+  {
+    // LacrosSelection policy has precedence over command line.
+    ScopedLacrosSelectionCache cache(
+        browser_util::LacrosSelectionPolicy::kStateful);
+    base::test::ScopedCommandLine cmd_line;
+    cmd_line.GetProcessCommandLine()->AppendSwitchASCII(
+        browser_util::kLacrosSelectionSwitch,
+        browser_util::kLacrosSelectionRootfs);
+    EXPECT_EQ(browser_util::DetermineLacrosSelection(),
+              LacrosSelection::kStateful);
+  }
+
+  {
+    // LacrosSelection allows command line check, but command line is not set.
+    ScopedLacrosSelectionCache cache(
+        browser_util::LacrosSelectionPolicy::kUserChoice);
+    EXPECT_FALSE(browser_util::DetermineLacrosSelection());
+  }
+
+  {
+    // LacrosSelection allows command line check.
+    ScopedLacrosSelectionCache cache(
+        browser_util::LacrosSelectionPolicy::kUserChoice);
+    base::test::ScopedCommandLine cmd_line;
+    cmd_line.GetProcessCommandLine()->AppendSwitchASCII(
+        browser_util::kLacrosSelectionSwitch,
+        browser_util::kLacrosSelectionRootfs);
+    EXPECT_EQ(browser_util::DetermineLacrosSelection(),
+              LacrosSelection::kRootfs);
+  }
+
+  {
+    // LacrosSelection allows command line check.
+    ScopedLacrosSelectionCache cache(
+        browser_util::LacrosSelectionPolicy::kUserChoice);
+    base::test::ScopedCommandLine cmd_line;
+    cmd_line.GetProcessCommandLine()->AppendSwitchASCII(
+        browser_util::kLacrosSelectionSwitch,
+        browser_util::kLacrosSelectionStateful);
+    EXPECT_EQ(browser_util::DetermineLacrosSelection(),
+              LacrosSelection::kStateful);
+  }
+}
+
 }  // namespace crosapi
diff --git a/chrome/browser/ash/net/network_diagnostics/http_request_manager.cc b/chrome/browser/ash/net/network_diagnostics/http_request_manager.cc
index 2bebce7..d951c08 100644
--- a/chrome/browser/ash/net/network_diagnostics/http_request_manager.cc
+++ b/chrome/browser/ash/net/network_diagnostics/http_request_manager.cc
@@ -31,16 +31,18 @@
       semantics {
         sender: "NetworkDiagnosticsRoutines"
         description: "Routines send network traffic (http requests) to "
-        "hosts in order to validate the internet connection on a device."
-        trigger:
-            "A routine makes an http request."
+          "hosts in order to validate the internet connection on a device."
+        trigger: "A routine makes an http request."
         data:
-            "No data other than the path is sent. No user identifier is "
-            "sent along with the data."
+          "No data other than the path is sent. No user identifier is "
+          "sent along with the data."
         destination: WEBSITE
       }
       policy {
         cookies_allowed: NO
+        policy_exception_justification:
+          "No policy defined to enable/disable or limit this request as this "
+          "is on-demand user initiated operation to do the network diagnostics."
       }
   )");
 }
diff --git a/chrome/browser/ash/policy/handlers/lacros_selection_policy_handler.cc b/chrome/browser/ash/policy/handlers/lacros_selection_policy_handler.cc
new file mode 100644
index 0000000..c8bbf43
--- /dev/null
+++ b/chrome/browser/ash/policy/handlers/lacros_selection_policy_handler.cc
@@ -0,0 +1,54 @@
+// Copyright 2022 The Chromium Authors
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include "chrome/browser/ash/policy/handlers/lacros_selection_policy_handler.h"
+
+#if !BUILDFLAG(IS_CHROMEOS_ASH)
+#error This file shall only be used in ash.
+#endif
+
+#include "chrome/common/pref_names.h"
+#include "components/policy/core/browser/policy_error_map.h"
+#include "components/policy/policy_constants.h"
+#include "components/prefs/pref_value_map.h"
+#include "components/strings/grit/components_strings.h"
+
+namespace policy {
+
+LacrosSelectionPolicyHandler::LacrosSelectionPolicyHandler()
+    : TypeCheckingPolicyHandler(key::kLacrosSelection,
+                                base::Value::Type::STRING) {}
+
+bool LacrosSelectionPolicyHandler::CheckPolicySettings(
+    const PolicyMap& policies,
+    PolicyErrorMap* errors) {
+  return GetValue(policies, errors).has_value();
+}
+
+void LacrosSelectionPolicyHandler::ApplyPolicySettings(
+    const PolicyMap& policies,
+    PrefValueMap* prefs) {
+  auto enum_value = GetValue(policies, nullptr);
+  if (enum_value.has_value())
+    prefs->SetInteger(prefs::kLacrosSelection, static_cast<int>(*enum_value));
+}
+
+absl::optional<crosapi::browser_util::LacrosSelectionPolicy>
+LacrosSelectionPolicyHandler::GetValue(const PolicyMap& policies,
+                                       PolicyErrorMap* errors) {
+  const base::Value* value;
+  const bool value_found = CheckAndGetValue(policies, errors, &value) && value;
+  if (!value_found)
+    return absl::nullopt;
+
+  auto parsed =
+      crosapi::browser_util::ParseLacrosSelectionPolicy(value->GetString());
+  if (!parsed.has_value() && errors) {
+    errors->AddError(policy_name(), IDS_POLICY_INVALID_SELECTION_ERROR,
+                     "LacrosSelection value");
+  }
+  return parsed;
+}
+
+}  // namespace policy
diff --git a/chrome/browser/ash/policy/handlers/lacros_selection_policy_handler.h b/chrome/browser/ash/policy/handlers/lacros_selection_policy_handler.h
new file mode 100644
index 0000000..cc867d3
--- /dev/null
+++ b/chrome/browser/ash/policy/handlers/lacros_selection_policy_handler.h
@@ -0,0 +1,40 @@
+// Copyright 2022 The Chromium Authors
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#ifndef CHROME_BROWSER_ASH_POLICY_HANDLERS_LACROS_SELECTION_POLICY_HANDLER_H_
+#define CHROME_BROWSER_ASH_POLICY_HANDLERS_LACROS_SELECTION_POLICY_HANDLER_H_
+
+#include "build/chromeos_buildflags.h"
+#include "chrome/browser/ash/crosapi/browser_util.h"
+#include "components/policy/core/browser/configuration_policy_handler.h"
+#include "third_party/abseil-cpp/absl/types/optional.h"
+
+class PrefValueMap;
+
+namespace policy {
+
+class PolicyMap;
+
+// The handler for LacrosSelection selection policy which maps string-enum
+// policy values to `LacrosSelectionPolicy` enum stored in `PolicyMap`.
+class LacrosSelectionPolicyHandler : public TypeCheckingPolicyHandler {
+ public:
+  LacrosSelectionPolicyHandler();
+
+  // ConfigurationPolicyHandler:
+  bool CheckPolicySettings(const PolicyMap& policies,
+                           PolicyErrorMap* errors) override;
+
+  void ApplyPolicySettings(const PolicyMap& policies,
+                           PrefValueMap* prefs) override;
+
+ private:
+  absl::optional<crosapi::browser_util::LacrosSelectionPolicy> GetValue(
+      const PolicyMap& policies,
+      PolicyErrorMap* errors);
+};
+
+}  // namespace policy
+
+#endif  // CHROME_BROWSER_ASH_POLICY_HANDLERS_LACROS_SELECTION_POLICY_HANDLER_H_
diff --git a/chrome/browser/ash/preferences.cc b/chrome/browser/ash/preferences.cc
index fdc83ae..5b32656 100644
--- a/chrome/browser/ash/preferences.cc
+++ b/chrome/browser/ash/preferences.cc
@@ -146,6 +146,10 @@
   registry->RegisterIntegerPref(
       ::prefs::kLacrosLaunchSwitch,
       static_cast<int>(crosapi::browser_util::LacrosAvailability::kUserChoice));
+  registry->RegisterIntegerPref(
+      ::prefs::kLacrosSelection,
+      static_cast<int>(
+          crosapi::browser_util::LacrosSelectionPolicy::kUserChoice));
   registry->RegisterStringPref(::prefs::kLacrosDataBackwardMigrationMode, "");
   registry->RegisterBooleanPref(prefs::kDeviceSystemWideTracingEnabled, true);
   registry->RegisterBooleanPref(
diff --git a/chrome/browser/ash/system_web_apps/system_web_app_manager.cc b/chrome/browser/ash/system_web_apps/system_web_app_manager.cc
index e74a539..89c62b2 100644
--- a/chrome/browser/ash/system_web_apps/system_web_app_manager.cc
+++ b/chrome/browser/ash/system_web_apps/system_web_app_manager.cc
@@ -368,19 +368,9 @@
   // or `SetSystemAppsForTesting()` has been called.
   if (base::CommandLine::ForCurrentProcess()->HasSwitch(switches::kTestType) &&
       skip_app_installation_in_test_) {
-    base::SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
-        FROM_HERE,
-        base::BindOnce(
-            &SystemWebAppManager::OnAppsSynchronized,
-            weak_ptr_factory_.GetWeakPtr(), should_force_install_apps,
-            install_start_time,
-            /*install_results=*/
-            std::map<GURL,
-                     web_app::ExternallyManagedAppManager::InstallResult>(),
-            /*uninstall_results=*/std::map<GURL, bool>()));
-
-    return;
+    install_options_list.clear();
   }
+
   provider_->externally_managed_app_manager().SynchronizeInstalledApps(
       std::move(install_options_list),
       web_app::ExternalInstallSource::kSystemInstalled,
diff --git a/chrome/browser/background/background_application_list_model_unittest.cc b/chrome/browser/background/background_application_list_model_unittest.cc
index 0b62606f..83af0e8 100644
--- a/chrome/browser/background/background_application_list_model_unittest.cc
+++ b/chrome/browser/background/background_application_list_model_unittest.cc
@@ -101,9 +101,9 @@
   std::string error;
   scoped_refptr<Extension> extension;
 
-  extension = Extension::Create(bogus_file_pathname(name),
-                                extensions::mojom::ManifestLocation::kInternal,
-                                manifest, Extension::NO_FLAGS, &error);
+  extension = Extension::Create(
+      bogus_file_pathname(name), extensions::mojom::ManifestLocation::kInternal,
+      manifest.GetDict(), Extension::NO_FLAGS, &error);
 
   // Cannot ASSERT_* here because that attempts an illegitimate return.
   // Cannot EXPECT_NE here because that assumes non-pointers unlike EXPECT_EQ
diff --git a/chrome/browser/background/background_contents_service.cc b/chrome/browser/background/background_contents_service.cc
index f9d2020..a6a2b9d 100644
--- a/chrome/browser/background/background_contents_service.cc
+++ b/chrome/browser/background/background_contents_service.cc
@@ -641,17 +641,18 @@
   // already an entry for this application, no need to do anything.
   // TODO(atwilson): Verify that this is the desired behavior based on developer
   // feedback (http://crbug.com/47118).
-  DictionaryPrefUpdate update(prefs_, prefs::kRegisteredBackgroundContents);
-  base::Value* pref = update.Get();
+  ScopedDictPrefUpdate update(prefs_, prefs::kRegisteredBackgroundContents);
+  base::Value::Dict& pref = update.Get();
   const std::string& appid = GetParentApplicationId(background_contents);
-  if (pref->FindDictKey(appid))
+  if (pref.FindDict(appid)) {
     return;
+  }
 
   // No entry for this application yet, so add one.
-  base::Value dict(base::Value::Type::DICTIONARY);
-  dict.SetStringKey(kUrlKey, background_contents->GetURL().spec());
-  dict.SetStringKey(kFrameNameKey, contents_map_[appid].frame_name);
-  pref->SetKey(appid, std::move(dict));
+  base::Value::Dict dict;
+  dict.Set(kUrlKey, background_contents->GetURL().spec());
+  dict.Set(kFrameNameKey, contents_map_[appid].frame_name);
+  pref.Set(appid, std::move(dict));
 }
 
 bool BackgroundContentsService::HasRegisteredBackgroundContents(
@@ -669,8 +670,8 @@
     return;
   DCHECK(IsTracked(background_contents));
   const std::string& appid = GetParentApplicationId(background_contents);
-  DictionaryPrefUpdate update(prefs_, prefs::kRegisteredBackgroundContents);
-  update.Get()->RemoveKey(appid);
+  ScopedDictPrefUpdate update(prefs_, prefs::kRegisteredBackgroundContents);
+  update->Remove(appid);
 }
 
 void BackgroundContentsService::ShutdownAssociatedBackgroundContents(
diff --git a/chrome/browser/chrome_browser_interface_binders.cc b/chrome/browser/chrome_browser_interface_binders.cc
index 778626ea..0b97572 100644
--- a/chrome/browser/chrome_browser_interface_binders.cc
+++ b/chrome/browser/chrome_browser_interface_binders.cc
@@ -1034,6 +1034,9 @@
     RegisterWebUIControllerInterfaceBinder<
         metrics_reporter::mojom::PageMetricsHost, TabSearchUI, NewTabPageUI>(
         map);
+  } else {
+    RegisterWebUIControllerInterfaceBinder<
+        metrics_reporter::mojom::PageMetricsHost, NewTabPageUI>(map);
   }
 
   RegisterWebUIControllerInterfaceBinder<
diff --git a/chrome/browser/chrome_content_browser_client.cc b/chrome/browser/chrome_content_browser_client.cc
index 03210d3..10cf5bb 100644
--- a/chrome/browser/chrome_content_browser_client.cc
+++ b/chrome/browser/chrome_content_browser_client.cc
@@ -3298,18 +3298,21 @@
 }
 
 bool ChromeContentBrowserClient::IsSharedStorageAllowed(
+    content::BrowserContext* browser_context,
     content::RenderFrameHost* rfh,
     const url::Origin& top_frame_origin,
     const url::Origin& accessing_origin) {
-  Profile* profile = Profile::FromBrowserContext(rfh->GetBrowserContext());
+  Profile* profile = Profile::FromBrowserContext(browser_context);
   auto* privacy_sandbox_settings =
       PrivacySandboxSettingsFactory::GetForProfile(profile);
   DCHECK(privacy_sandbox_settings);
   bool allowed = privacy_sandbox_settings->IsSharedStorageAllowed(
       top_frame_origin, accessing_origin);
-  content_settings::PageSpecificContentSettings::BrowsingDataAccessed(
-      rfh, blink::StorageKey(accessing_origin),
-      BrowsingDataModel::StorageType::kSharedStorage, !allowed);
+  if (rfh) {
+    content_settings::PageSpecificContentSettings::BrowsingDataAccessed(
+        rfh, blink::StorageKey(accessing_origin),
+        BrowsingDataModel::StorageType::kSharedStorage, !allowed);
+  }
   return allowed;
 }
 
diff --git a/chrome/browser/chrome_content_browser_client.h b/chrome/browser/chrome_content_browser_client.h
index 3284619..ead5dfd 100644
--- a/chrome/browser/chrome_content_browser_client.h
+++ b/chrome/browser/chrome_content_browser_client.h
@@ -321,7 +321,8 @@
       const url::Origin* impression_origin,
       const url::Origin* conversion_origin,
       const url::Origin* reporting_origin) override;
-  bool IsSharedStorageAllowed(content::RenderFrameHost* rfh,
+  bool IsSharedStorageAllowed(content::BrowserContext* browser_context,
+                              content::RenderFrameHost* rfh,
                               const url::Origin& top_frame_origin,
                               const url::Origin& accessing_origin) override;
   bool IsPrivateAggregationAllowed(
diff --git a/chrome/browser/devtools/protocol/cast_handler_unittest.cc b/chrome/browser/devtools/protocol/cast_handler_unittest.cc
index b6cbc67..b0bd35c 100644
--- a/chrome/browser/devtools/protocol/cast_handler_unittest.cc
+++ b/chrome/browser/devtools/protocol/cast_handler_unittest.cc
@@ -83,8 +83,6 @@
 
  protected:
   void EnableHandler() {
-    EXPECT_CALL(*router_, RegisterMediaRoutesObserver(_))
-        .WillOnce(SaveArg<0>(&routes_observer_));
     EXPECT_CALL(*router_, RegisterMediaSinksObserver(_))
         .WillRepeatedly(
             WithArg<0>([this](media_router::MediaSinksObserver* observer) {
@@ -102,7 +100,6 @@
 
   std::unique_ptr<CastHandler> handler_;
   media_router::MockMediaRouter* router_ = nullptr;
-  media_router::MediaRoutesObserver* routes_observer_ = nullptr;
   media_router::MediaSinksObserver* desktop_sinks_observer_ = nullptr;
   media_router::MediaSinksObserver* sinks_observer_ = nullptr;
 };
@@ -201,14 +198,14 @@
 
 TEST_F(CastHandlerTest, StopCasting) {
   sinks_observer_->OnSinksUpdated({sink1, sink2}, {});
-  routes_observer_->OnRoutesUpdated({Route1()});
+  router_->routes_observers().begin()->OnRoutesUpdated({Route1()});
   EXPECT_CALL(*router_, TerminateRoute(kRouteId1));
   EXPECT_TRUE(handler_->StopCasting(kSinkName1).IsSuccess());
 }
 
 TEST_F(CastHandlerTest, StopCastingWithInvalidName) {
   sinks_observer_->OnSinksUpdated({sink1, sink2}, {});
-  routes_observer_->OnRoutesUpdated({Route1()});
+  router_->routes_observers().begin()->OnRoutesUpdated({Route1()});
   // Attempting to stop casting to a sink without a route should fail.
   EXPECT_TRUE(handler_->StopCasting(kSinkName2).IsError());
 }
diff --git a/chrome/browser/download/bubble/download_bubble_controller.cc b/chrome/browser/download/bubble/download_bubble_controller.cc
index c13525e9..4aec509 100644
--- a/chrome/browser/download/bubble/download_bubble_controller.cc
+++ b/chrome/browser/download/bubble/download_bubble_controller.cc
@@ -391,10 +391,12 @@
     case DownloadCommands::DISCARD:
       DownloadItemWarningData::AddWarningActionEvent(
           model->GetDownloadItem(),
-          is_main_view ? DownloadItemWarningData::BUBBLE_MAINPAGE
-                       : DownloadItemWarningData::BUBBLE_SUBPAGE,
-          command == DownloadCommands::KEEP ? DownloadItemWarningData::PROCEED
-                                            : DownloadItemWarningData::DISCARD);
+          is_main_view
+              ? DownloadItemWarningData::WarningSurface::BUBBLE_MAINPAGE
+              : DownloadItemWarningData::WarningSurface::BUBBLE_SUBPAGE,
+          command == DownloadCommands::KEEP
+              ? DownloadItemWarningData::WarningAction::PROCEED
+              : DownloadItemWarningData::WarningAction::DISCARD);
       commands.ExecuteCommand(command);
       break;
     case DownloadCommands::REVIEW:
diff --git a/chrome/browser/download/download_danger_prompt.cc b/chrome/browser/download/download_danger_prompt.cc
index 58f7169d4..8dc335d7 100644
--- a/chrome/browser/download/download_danger_prompt.cc
+++ b/chrome/browser/download/download_danger_prompt.cc
@@ -69,15 +69,16 @@
   DownloadItemWarningData::WarningAction warning_action;
   switch (action) {
     case Action::ACCEPT:
-      warning_action = DownloadItemWarningData::PROCEED;
+      warning_action = DownloadItemWarningData::WarningAction::PROCEED;
       break;
     case Action::CANCEL:
-      warning_action = DownloadItemWarningData::CANCEL;
+      warning_action = DownloadItemWarningData::WarningAction::CANCEL;
       break;
     case Action::DISMISS:
-      warning_action = DownloadItemWarningData::CLOSE;
+      warning_action = DownloadItemWarningData::WarningAction::CLOSE;
       break;
   }
   DownloadItemWarningData::AddWarningActionEvent(
-      download, DownloadItemWarningData::DOWNLOAD_PROMPT, warning_action);
+      download, DownloadItemWarningData::WarningSurface::DOWNLOAD_PROMPT,
+      warning_action);
 }
diff --git a/chrome/browser/download/download_item_warning_data.cc b/chrome/browser/download/download_item_warning_data.cc
index 3c729956..f20a3a5 100644
--- a/chrome/browser/download/download_item_warning_data.cc
+++ b/chrome/browser/download/download_item_warning_data.cc
@@ -46,6 +46,10 @@
   base::UmaHistogramEnumeration(
       "Download.WarningData.SurfaceWithoutWarningShown", surface);
 }
+
+void RecordWarningActionAdded(WarningAction action) {
+  base::UmaHistogramEnumeration("Download.WarningData.ActionAdded", action);
+}
 }  // namespace
 
 // static
@@ -81,6 +85,7 @@
     if (data->warning_first_shown_time_.is_null()) {
       RecordAddWarningActionEventOutcome(
           AddWarningActionEventOutcome::ADDED_WARNING_FIRST_SHOWN);
+      RecordWarningActionAdded(action);
       data->warning_first_shown_time_ = base::Time::Now();
     } else {
       RecordAddWarningActionEventOutcome(
@@ -102,12 +107,15 @@
   int64_t action_latency =
       (base::Time::Now() - data->warning_first_shown_time_).InMilliseconds();
   bool is_terminal_action =
-      (action == PROCEED || action == DISCARD) ? true : false;
+      (action == WarningAction::PROCEED || action == WarningAction::DISCARD)
+          ? true
+          : false;
   DCHECK_NE(WarningAction::SHOWN, action);
   data->action_events_.emplace_back(surface, action, action_latency,
                                     is_terminal_action);
   RecordAddWarningActionEventOutcome(
       AddWarningActionEventOutcome::ADDED_WARNING_ACTION);
+  RecordWarningActionAdded(action);
 }
 
 // static
@@ -116,57 +124,57 @@
     const WarningActionEvent& event) {
   ClientSafeBrowsingReportRequest::DownloadWarningAction action;
   switch (event.surface) {
-    case DownloadItemWarningData::BUBBLE_MAINPAGE:
+    case DownloadItemWarningData::WarningSurface::BUBBLE_MAINPAGE:
       action.set_surface(ClientSafeBrowsingReportRequest::
                              DownloadWarningAction::BUBBLE_MAINPAGE);
       break;
-    case DownloadItemWarningData::BUBBLE_SUBPAGE:
+    case DownloadItemWarningData::WarningSurface::BUBBLE_SUBPAGE:
       action.set_surface(ClientSafeBrowsingReportRequest::
                              DownloadWarningAction::BUBBLE_SUBPAGE);
       break;
-    case DownloadItemWarningData::DOWNLOADS_PAGE:
+    case DownloadItemWarningData::WarningSurface::DOWNLOADS_PAGE:
       action.set_surface(ClientSafeBrowsingReportRequest::
                              DownloadWarningAction::DOWNLOADS_PAGE);
       break;
-    case DownloadItemWarningData::DOWNLOAD_PROMPT:
+    case DownloadItemWarningData::WarningSurface::DOWNLOAD_PROMPT:
       action.set_surface(ClientSafeBrowsingReportRequest::
                              DownloadWarningAction::DOWNLOAD_PROMPT);
       break;
   }
   switch (event.action) {
-    case DownloadItemWarningData::PROCEED:
+    case DownloadItemWarningData::WarningAction::PROCEED:
       action.set_action(
           ClientSafeBrowsingReportRequest::DownloadWarningAction::PROCEED);
       break;
-    case DownloadItemWarningData::DISCARD:
+    case DownloadItemWarningData::WarningAction::DISCARD:
       action.set_action(
           ClientSafeBrowsingReportRequest::DownloadWarningAction::DISCARD);
       break;
-    case DownloadItemWarningData::KEEP:
+    case DownloadItemWarningData::WarningAction::KEEP:
       action.set_action(
           ClientSafeBrowsingReportRequest::DownloadWarningAction::KEEP);
       break;
-    case DownloadItemWarningData::CLOSE:
+    case DownloadItemWarningData::WarningAction::CLOSE:
       action.set_action(
           ClientSafeBrowsingReportRequest::DownloadWarningAction::CLOSE);
       break;
-    case DownloadItemWarningData::CANCEL:
+    case DownloadItemWarningData::WarningAction::CANCEL:
       action.set_action(
           ClientSafeBrowsingReportRequest::DownloadWarningAction::CANCEL);
       break;
-    case DownloadItemWarningData::DISMISS:
+    case DownloadItemWarningData::WarningAction::DISMISS:
       action.set_action(
           ClientSafeBrowsingReportRequest::DownloadWarningAction::DISMISS);
       break;
-    case DownloadItemWarningData::BACK:
+    case DownloadItemWarningData::WarningAction::BACK:
       action.set_action(
           ClientSafeBrowsingReportRequest::DownloadWarningAction::BACK);
       break;
-    case DownloadItemWarningData::OPEN_SUBPAGE:
+    case DownloadItemWarningData::WarningAction::OPEN_SUBPAGE:
       action.set_action(
           ClientSafeBrowsingReportRequest::DownloadWarningAction::OPEN_SUBPAGE);
       break;
-    case DownloadItemWarningData::SHOWN:
+    case DownloadItemWarningData::WarningAction::SHOWN:
       NOTREACHED();
       break;
   }
diff --git a/chrome/browser/download/download_item_warning_data.h b/chrome/browser/download/download_item_warning_data.h
index 84f8838..92c2716 100644
--- a/chrome/browser/download/download_item_warning_data.h
+++ b/chrome/browser/download/download_item_warning_data.h
@@ -24,8 +24,8 @@
   // go/chrome-download-warning-surfaces for details.
   // These values are persisted to logs. Entries should not be renumbered and
   // numeric values should never be reused.
-  enum WarningSurface {
-    // Applicable actions: DISCARD
+  enum class WarningSurface {
+    // Applicable actions: DISCARD, OPEN_SUBPAGE
     BUBBLE_MAINPAGE = 1,
     // Applicable actions: PROCEED, DISCARD, DISMISS, CLOSE, BACK
     BUBBLE_SUBPAGE = 2,
@@ -37,7 +37,9 @@
   };
 
   // Users action on the warning surface.
-  enum WarningAction {
+  // These values are persisted to logs. Entries should not be renumbered and
+  // numeric values should never be reused.
+  enum class WarningAction {
     // The warning is shown. This is a special action that may not be triggered
     // by user. We will use this action as the anchor to track the latency of
     // other actions.
@@ -61,7 +63,8 @@
     // to the bubble main page.
     BACK = 7,
     // The user has opened the subpage from the main page.
-    OPEN_SUBPAGE = 8
+    OPEN_SUBPAGE = 8,
+    kMaxValue = OPEN_SUBPAGE
   };
 
   struct WarningActionEvent {
diff --git a/chrome/browser/download/download_item_warning_data_unittest.cc b/chrome/browser/download/download_item_warning_data_unittest.cc
index ebd0af6..47b130b8 100644
--- a/chrome/browser/download/download_item_warning_data_unittest.cc
+++ b/chrome/browser/download/download_item_warning_data_unittest.cc
@@ -39,13 +39,14 @@
     if (actual_event.surface != expected_surface) {
       success = false;
       ADD_FAILURE() << "Warning action event should have surface "
-                    << expected_surface << ", but found "
-                    << actual_event.surface;
+                    << static_cast<int>(expected_surface) << ", but found "
+                    << static_cast<int>(actual_event.surface);
     }
     if (actual_event.action != expected_action) {
       success = false;
       ADD_FAILURE() << "Warning action event should have action "
-                    << expected_action << ", but found " << actual_event.action;
+                    << static_cast<int>(expected_action) << ", but found "
+                    << static_cast<int>(actual_event.action);
     }
     if (actual_event.action_latency_msec != expected_latency) {
       success = false;
diff --git a/chrome/browser/download/download_ui_model.cc b/chrome/browser/download/download_ui_model.cc
index 123b67a..46de7ef 100644
--- a/chrome/browser/download/download_ui_model.cc
+++ b/chrome/browser/download/download_ui_model.cc
@@ -1780,3 +1780,33 @@
 void DownloadUIModel::DetermineAndSetShouldPreferOpeningInBrowser(
     const base::FilePath& target_path,
     bool is_filetype_handled_safely) {}
+
+std::u16string DownloadUIModel::GetInProgressAccessibleAlertText() const {
+  // Prefer to announce the time remaining, if known.
+  base::TimeDelta remaining;
+  if (TimeRemaining(&remaining)) {
+    // If complete, skip this round: a completion status update is coming soon.
+    if (remaining.is_zero())
+      return std::u16string();
+
+    const std::u16string remaining_string =
+        ui::TimeFormat::Simple(ui::TimeFormat::FORMAT_REMAINING,
+                               ui::TimeFormat::LENGTH_SHORT, remaining);
+    return l10n_util::GetStringFUTF16(
+        IDS_DOWNLOAD_STATUS_TIME_REMAINING_ACCESSIBLE_ALERT, remaining_string);
+  }
+
+  // Time remaining is unknown, try to announce percent remaining.
+  if (PercentComplete() > 0) {
+    DCHECK_LE(PercentComplete(), 100);
+    return l10n_util::GetStringFUTF16Int(
+        IDS_DOWNLOAD_STATUS_PERCENT_COMPLETE_ACCESSIBLE_ALERT,
+        100 - PercentComplete());
+  }
+
+  // Percent remaining is also unknown, announce bytes to download.
+  return l10n_util::GetStringFUTF16(
+      IDS_DOWNLOAD_STATUS_IN_PROGRESS_ACCESSIBLE_ALERT,
+      ui::FormatBytes(GetTotalBytes()),
+      GetFileNameToReportUser().LossyDisplayName());
+}
diff --git a/chrome/browser/download/download_ui_model.h b/chrome/browser/download/download_ui_model.h
index 69103e5..d289774e 100644
--- a/chrome/browser/download/download_ui_model.h
+++ b/chrome/browser/download/download_ui_model.h
@@ -536,6 +536,10 @@
       const base::FilePath& target_path,
       bool is_filetype_handled_safely);
 
+  // Returns the accessible alert text that should be announced when the
+  // download is in progress.
+  virtual std::u16string GetInProgressAccessibleAlertText() const;
+
  protected:
   // Returns the MIME type of the download.
   virtual std::string GetMimeType() const;
diff --git a/chrome/browser/enterprise/idle/action.cc b/chrome/browser/enterprise/idle/action.cc
index 235da753..71d89a7 100644
--- a/chrome/browser/enterprise/idle/action.cc
+++ b/chrome/browser/enterprise/idle/action.cc
@@ -30,10 +30,11 @@
   CloseBrowsersAction() : Action(ActionType::kCloseBrowsers) {}
 
   void Run(Profile* profile, Continuation continuation) override {
-    int minutes = profile->GetPrefs()->GetInteger(prefs::kIdleTimeout);
+    base::TimeDelta timeout =
+        profile->GetPrefs()->GetTimeDelta(prefs::kIdleTimeout);
     continuation_ = std::move(continuation);
     subscription_ = BrowserCloser::GetInstance()->ShowDialogAndCloseBrowsers(
-        profile, base::Minutes(minutes),
+        profile, timeout,
         base::BindOnce(&CloseBrowsersAction::OnCloseFinished,
                        base::Unretained(this)));
   }
diff --git a/chrome/browser/enterprise/idle/idle_service.cc b/chrome/browser/enterprise/idle/idle_service.cc
index 0e749d7..8d112b08 100644
--- a/chrome/browser/enterprise/idle/idle_service.cc
+++ b/chrome/browser/enterprise/idle/idle_service.cc
@@ -33,10 +33,11 @@
 IdleService::~IdleService() = default;
 
 void IdleService::OnIdleTimeoutPrefChanged() {
-  int minutes = profile_->GetPrefs()->GetInteger(prefs::kIdleTimeout);
-  if (minutes > 0) {
+  base::TimeDelta timeout =
+      profile_->GetPrefs()->GetTimeDelta(prefs::kIdleTimeout);
+  if (timeout.is_positive()) {
     // `is_idle_` will auto-update in 1 second, no need to set it here.
-    idle_threshold_ = base::Minutes(minutes);
+    idle_threshold_ = timeout;
     if (!polling_service_observation_.IsObserving()) {
       polling_service_observation_.Observe(
           ui::IdlePollingService::GetInstance());
diff --git a/chrome/browser/enterprise/idle/idle_service_factory.cc b/chrome/browser/enterprise/idle/idle_service_factory.cc
index 625a5c6..cced806 100644
--- a/chrome/browser/enterprise/idle/idle_service_factory.cc
+++ b/chrome/browser/enterprise/idle/idle_service_factory.cc
@@ -36,8 +36,7 @@
 
 void IdleServiceFactory::RegisterProfilePrefs(
     user_prefs::PrefRegistrySyncable* registry) {
-  // TODO(crbug.com/1316551): Use TimeDeltaPref instead.
-  registry->RegisterIntegerPref(prefs::kIdleTimeout, 0);
+  registry->RegisterTimeDeltaPref(prefs::kIdleTimeout, base::TimeDelta());
   registry->RegisterListPref(prefs::kIdleTimeoutActions);
 }
 
diff --git a/chrome/browser/enterprise/idle/idle_timeout_policy_handler.cc b/chrome/browser/enterprise/idle/idle_timeout_policy_handler.cc
new file mode 100644
index 0000000..770e073
--- /dev/null
+++ b/chrome/browser/enterprise/idle/idle_timeout_policy_handler.cc
@@ -0,0 +1,139 @@
+// Copyright 2022 The Chromium Authors
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include "chrome/browser/enterprise/idle/idle_timeout_policy_handler.h"
+
+#include <string>
+
+#include "base/json/values_util.h"
+#include "base/strings/string_number_conversions.h"
+#include "base/time/time.h"
+#include "base/values.h"
+#include "chrome/browser/enterprise/idle/action.h"
+#include "chrome/browser/prefs/session_startup_pref.h"
+#include "chrome/common/pref_names.h"
+#include "components/policy/core/browser/configuration_policy_handler.h"
+#include "components/policy/core/browser/policy_error_map.h"
+#include "components/policy/core/common/policy_map.h"
+#include "components/policy/core/common/schema.h"
+#include "components/policy/policy_constants.h"
+#include "components/prefs/pref_value_map.h"
+#include "components/strings/grit/components_strings.h"
+
+namespace enterprise_idle {
+
+namespace {
+
+// If `other_policy_name` is unset, adds an error to `errors` and returns false.
+bool CheckOtherPolicySet(const policy::PolicyMap& policies,
+                         const std::string& this_policy_name,
+                         const std::string& other_policy_name,
+                         policy::PolicyErrorMap* errors) {
+  if (policies.GetValueUnsafe(other_policy_name))
+    return true;
+
+  errors->AddError(this_policy_name, IDS_POLICY_DEPENDENCY_ERROR_ANY_VALUE,
+                   other_policy_name);
+  return false;
+}
+
+}  // namespace
+
+IdleTimeoutPolicyHandler::IdleTimeoutPolicyHandler()
+    : policy::IntRangePolicyHandler(policy::key::kIdleTimeout,
+                                    prefs::kIdleTimeout,
+                                    1,
+                                    INT_MAX,
+                                    true) {}
+
+IdleTimeoutPolicyHandler::~IdleTimeoutPolicyHandler() = default;
+
+void IdleTimeoutPolicyHandler::ApplyPolicySettings(
+    const policy::PolicyMap& policies,
+    PrefValueMap* prefs) {
+  const base::Value* value =
+      policies.GetValue(policy::key::kIdleTimeout, base::Value::Type::INTEGER);
+  DCHECK(value);
+
+  // Apply a minimum of 1.
+  base::TimeDelta time_delta = base::Minutes(std::max(value->GetInt(), 1));
+  prefs->SetValue(prefs::kIdleTimeout, base::TimeDeltaToValue(time_delta));
+}
+
+bool IdleTimeoutPolicyHandler::CheckPolicySettings(
+    const policy::PolicyMap& policies,
+    policy::PolicyErrorMap* errors) {
+  // Nothing to do if unset.
+  if (!policies.GetValueUnsafe(policy::key::kIdleTimeout))
+    return false;
+
+  // Check that it's an integer, and that it's >= 1.
+  if (!policy::IntRangePolicyHandler::CheckPolicySettings(policies, errors))
+    return false;
+
+  // If IdleTimeoutActions is unset, add an error and do nothing.
+  if (!CheckOtherPolicySet(policies, policy::key::kIdleTimeout,
+                           policy::key::kIdleTimeoutActions, errors)) {
+    return false;
+  }
+
+  return true;
+}
+
+IdleTimeoutActionsPolicyHandler::IdleTimeoutActionsPolicyHandler(
+    policy::Schema schema)
+    : policy::SchemaValidatingPolicyHandler(
+          policy::key::kIdleTimeoutActions,
+          schema.GetKnownProperty(policy::key::kIdleTimeoutActions),
+          policy::SCHEMA_ALLOW_UNKNOWN_AND_INVALID_LIST_ENTRY) {}
+
+IdleTimeoutActionsPolicyHandler::~IdleTimeoutActionsPolicyHandler() = default;
+
+void IdleTimeoutActionsPolicyHandler::ApplyPolicySettings(
+    const policy::PolicyMap& policies,
+    PrefValueMap* prefs) {
+  const base::Value* policy_value = policies.GetValue(
+      policy::key::kIdleTimeoutActions, base::Value::Type::LIST);
+  DCHECK(policy_value);
+
+  // Convert strings to integers (from the ActionType enum).
+  base::Value::List converted_actions;
+  for (const base::Value& action : policy_value->GetList()) {
+    if (!action.is_string())
+      continue;
+    const std::string& name = action.GetString();
+    if (name == "close_browsers")
+      converted_actions.Append(static_cast<int>(ActionType::kCloseBrowsers));
+    else if (name == "show_profile_picker")
+      converted_actions.Append(
+          static_cast<int>(ActionType::kShowProfilePicker));
+    // Silently drop unsupported values.
+  }
+  prefs->SetValue(prefs::kIdleTimeoutActions,
+                  base::Value(std::move(converted_actions)));
+}
+
+bool IdleTimeoutActionsPolicyHandler::CheckPolicySettings(
+    const policy::PolicyMap& policies,
+    policy::PolicyErrorMap* errors) {
+  // Nothing to do if unset.
+  if (!policies.GetValueUnsafe(policy::key::kIdleTimeoutActions))
+    return false;
+
+  // Check that it's a list of strings, and that they're supported enum values.
+  // Unsupported enum values are dropped, with a warning on chrome://policy.
+  if (!policy::SchemaValidatingPolicyHandler::CheckPolicySettings(policies,
+                                                                  errors)) {
+    return false;
+  }
+
+  // If IdleTimeout is unset, add an error and do nothing.
+  if (!CheckOtherPolicySet(policies, policy::key::kIdleTimeoutActions,
+                           policy::key::kIdleTimeout, errors)) {
+    return false;
+  }
+
+  return true;
+}
+}  // namespace enterprise_idle
diff --git a/chrome/browser/enterprise/idle/idle_timeout_policy_handler.h b/chrome/browser/enterprise/idle/idle_timeout_policy_handler.h
new file mode 100644
index 0000000..c40759c
--- /dev/null
+++ b/chrome/browser/enterprise/idle/idle_timeout_policy_handler.h
@@ -0,0 +1,58 @@
+// Copyright 2022 The Chromium Authors
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#ifndef CHROME_BROWSER_ENTERPRISE_IDLE_IDLE_TIMEOUT_POLICY_HANDLER_H_
+#define CHROME_BROWSER_ENTERPRISE_IDLE_IDLE_TIMEOUT_POLICY_HANDLER_H_
+
+#include "components/policy/core/browser/configuration_policy_handler.h"
+
+class PrefValueMap;
+
+namespace policy {
+class PolicyErrorMap;
+class PolicyMap;
+}  // namespace policy
+
+namespace enterprise_idle {
+
+// Handles IdleTimeout policy.
+class IdleTimeoutPolicyHandler : public policy::IntRangePolicyHandler {
+ public:
+  IdleTimeoutPolicyHandler();
+
+  IdleTimeoutPolicyHandler(const IdleTimeoutPolicyHandler&) = delete;
+  IdleTimeoutPolicyHandler& operator=(const IdleTimeoutPolicyHandler&) = delete;
+
+  ~IdleTimeoutPolicyHandler() override;
+
+  // ConfigurationPolicyHandler methods:
+  void ApplyPolicySettings(const policy::PolicyMap& policies,
+                           PrefValueMap* prefs) override;
+  bool CheckPolicySettings(const policy::PolicyMap& policies,
+                           policy::PolicyErrorMap* errors) override;
+};
+
+// Handles IdleTimeoutActions policy.
+class IdleTimeoutActionsPolicyHandler
+    : public policy::SchemaValidatingPolicyHandler {
+ public:
+  explicit IdleTimeoutActionsPolicyHandler(policy::Schema schema);
+
+  IdleTimeoutActionsPolicyHandler(const IdleTimeoutActionsPolicyHandler&) =
+      delete;
+  IdleTimeoutActionsPolicyHandler& operator=(
+      const IdleTimeoutActionsPolicyHandler&) = delete;
+
+  ~IdleTimeoutActionsPolicyHandler() override;
+
+  // ConfigurationPolicyHandler methods:
+  void ApplyPolicySettings(const policy::PolicyMap& policies,
+                           PrefValueMap* prefs) override;
+  bool CheckPolicySettings(const policy::PolicyMap& policies,
+                           policy::PolicyErrorMap* errors) override;
+};
+
+}  // namespace enterprise_idle
+
+#endif  // CHROME_BROWSER_ENTERPRISE_IDLE_IDLE_TIMEOUT_POLICY_HANDLER_H_
diff --git a/chrome/browser/enterprise/idle/idle_timeout_policy_handler_unittest.cc b/chrome/browser/enterprise/idle/idle_timeout_policy_handler_unittest.cc
new file mode 100644
index 0000000..f6b3583
--- /dev/null
+++ b/chrome/browser/enterprise/idle/idle_timeout_policy_handler_unittest.cc
@@ -0,0 +1,272 @@
+// Copyright 2022 The Chromium Authors
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include "chrome/browser/enterprise/idle/idle_timeout_policy_handler.h"
+
+#include <string>
+
+#include "base/json/values_util.h"
+#include "base/ranges/algorithm.h"
+#include "base/strings/utf_string_conversions.h"
+#include "base/time/time.h"
+#include "base/values.h"
+#include "chrome/browser/enterprise/idle/action.h"
+#include "chrome/common/pref_names.h"
+#include "components/policy/core/browser/configuration_policy_handler.h"
+#include "components/policy/core/browser/policy_error_map.h"
+#include "components/policy/core/common/policy_map.h"
+#include "components/policy/core/common/policy_types.h"
+#include "components/policy/policy_constants.h"
+#include "components/prefs/pref_value_map.h"
+#include "components/strings/grit/components_strings.h"
+#include "testing/gmock/include/gmock/gmock.h"
+#include "testing/gtest/include/gtest/gtest.h"
+#include "ui/base/l10n/l10n_util.h"
+
+namespace enterprise_idle {
+
+using base::UTF8ToUTF16;
+
+class IdleTimeoutPolicyHandlerTest : public testing::Test {
+ protected:
+  void SetPolicyValue(const std::string& policy, base::Value value) {
+    policies_.Set(policy, policy::POLICY_LEVEL_MANDATORY,
+                  policy::POLICY_SCOPE_USER, policy::POLICY_SOURCE_PLATFORM,
+                  std::move(value), nullptr);
+  }
+
+  bool CheckPolicySettings() {
+    bool results[] = {
+        timeout_handler_.CheckPolicySettings(policies_, &errors_),
+        actions_handler_.CheckPolicySettings(policies_, &errors_),
+    };
+    return base::ranges::all_of(base::span(results), std::identity{});
+  }
+
+  void ApplyPolicySettings() {
+    timeout_handler_.ApplyPolicySettings(policies_, &prefs_);
+    actions_handler_.ApplyPolicySettings(policies_, &prefs_);
+  }
+
+  void CheckAndApplyPolicySettings() {
+    if (CheckPolicySettings())
+      ApplyPolicySettings();
+  }
+
+  policy::PolicyErrorMap& errors() { return errors_; }
+  PrefValueMap& prefs() { return prefs_; }
+
+ private:
+  policy::PolicyMap policies_;
+  policy::PolicyErrorMap errors_;
+  PrefValueMap prefs_;
+  policy::Schema schema_ = policy::Schema::Wrap(policy::GetChromeSchemaData());
+  IdleTimeoutPolicyHandler timeout_handler_;
+  IdleTimeoutActionsPolicyHandler actions_handler_ =
+      IdleTimeoutActionsPolicyHandler(schema_);
+};
+
+TEST_F(IdleTimeoutPolicyHandlerTest, PoliciesNotSet) {
+  CheckAndApplyPolicySettings();
+
+  // Shouldn't error.
+  EXPECT_EQ(errors().size(), 0U);
+
+  // Prefs should not be set.
+  const base::Value* pref_value;
+  EXPECT_FALSE(prefs().GetValue(prefs::kIdleTimeout, &pref_value));
+  EXPECT_FALSE(prefs().GetValue(prefs::kIdleTimeoutActions, &pref_value));
+}
+
+TEST_F(IdleTimeoutPolicyHandlerTest, JustTimeout) {
+  // IdleTimeout is set, but not IdleTimeoutActions.
+  SetPolicyValue(policy::key::kIdleTimeout, base::Value(15));
+
+  CheckAndApplyPolicySettings();
+
+  // Should have an error.
+  auto expected_error =
+      l10n_util::GetStringFUTF16(IDS_POLICY_DEPENDENCY_ERROR_ANY_VALUE,
+                                 UTF8ToUTF16(policy::key::kIdleTimeoutActions));
+  EXPECT_EQ(errors().size(), 1U);
+  EXPECT_EQ(errors().begin()->second.message, expected_error);
+
+  // Prefs should not be set.
+  const base::Value* pref_value;
+  EXPECT_FALSE(prefs().GetValue(prefs::kIdleTimeout, &pref_value));
+  EXPECT_FALSE(prefs().GetValue(prefs::kIdleTimeoutActions, &pref_value));
+}
+
+TEST_F(IdleTimeoutPolicyHandlerTest, JustActions) {
+  // IdleTimeoutActions is set, but not IdleTimeout.
+  SetPolicyValue(policy::key::kIdleTimeoutActions,
+                 base::Value(base::Value::List()));
+
+  CheckAndApplyPolicySettings();
+
+  // Should have an error.
+  auto expected_error =
+      l10n_util::GetStringFUTF16(IDS_POLICY_DEPENDENCY_ERROR_ANY_VALUE,
+                                 UTF8ToUTF16(policy::key::kIdleTimeout));
+  EXPECT_EQ(errors().size(), 1U);
+  EXPECT_EQ(errors().begin()->second.message, expected_error);
+
+  // Prefs should not be set.
+  const base::Value* pref_value;
+  EXPECT_FALSE(prefs().GetValue(prefs::kIdleTimeout, &pref_value));
+  EXPECT_FALSE(prefs().GetValue(prefs::kIdleTimeoutActions, &pref_value));
+}
+
+TEST_F(IdleTimeoutPolicyHandlerTest, InvalidTimeoutPolicyType) {
+  // Give an integer to a string policy.
+  SetPolicyValue(policy::key::kIdleTimeout, base::Value("invalid"));
+  SetPolicyValue(policy::key::kIdleTimeoutActions,
+                 base::Value(base::Value::List()));
+
+  CheckAndApplyPolicySettings();
+
+  // Should have an error.
+  auto expected_error = l10n_util::GetStringFUTF16(
+      IDS_POLICY_TYPE_ERROR,
+      UTF8ToUTF16(base::Value::GetTypeName(base::Value::Type::INTEGER)));
+  EXPECT_EQ(errors().size(), 1U);
+  EXPECT_EQ(errors().begin()->second.message, expected_error);
+
+  // Prefs should not be set.
+  const base::Value* pref_value;
+  EXPECT_FALSE(prefs().GetValue(prefs::kIdleTimeout, &pref_value));
+  EXPECT_FALSE(prefs().GetValue(prefs::kIdleTimeoutActions, &pref_value));
+}
+
+TEST_F(IdleTimeoutPolicyHandlerTest, InvalidActionsPolicyType) {
+  // Give a string to a string-enum policy.
+  SetPolicyValue(policy::key::kIdleTimeout, base::Value(5));
+  SetPolicyValue(policy::key::kIdleTimeoutActions, base::Value("invalid"));
+
+  CheckAndApplyPolicySettings();
+
+  // Should have an error.
+  auto expected_error = l10n_util::GetStringFUTF16(
+      IDS_POLICY_SCHEMA_VALIDATION_ERROR,
+      u"Policy type mismatch: expected: \"list\", actual: \"string\".");
+  EXPECT_EQ(errors().size(), 1U);
+  EXPECT_EQ(errors().begin()->second.message, expected_error);
+
+  // Prefs should not be set.
+  const base::Value* pref_value;
+  EXPECT_FALSE(prefs().GetValue(prefs::kIdleTimeout, &pref_value));
+  EXPECT_FALSE(prefs().GetValue(prefs::kIdleTimeoutActions, &pref_value));
+}
+
+TEST_F(IdleTimeoutPolicyHandlerTest, InvalidActionWrongType) {
+  // IdleTimeoutActions is a list, but one of the elements is not even a string.
+  SetPolicyValue(policy::key::kIdleTimeout, base::Value(5));
+  base::Value::List list;
+  list.Append("close_browsers");
+  list.Append(34);
+  SetPolicyValue(policy::key::kIdleTimeoutActions,
+                 base::Value(std::move(list)));
+
+  CheckAndApplyPolicySettings();
+
+  // Should have an error.
+  auto expected_error = l10n_util::GetStringFUTF16(
+      IDS_POLICY_ERROR_WITH_PATH,
+      UTF8ToUTF16(policy::key::kIdleTimeoutActions) + u"[1]",
+      l10n_util::GetStringFUTF16(
+          IDS_POLICY_SCHEMA_VALIDATION_ERROR,
+          u"Policy type mismatch: expected: \"string\", actual: \"integer\"."));
+  EXPECT_EQ(errors().size(), 1U);
+  EXPECT_EQ(errors().begin()->second.message, expected_error);
+
+  // Prefs should not be set.
+  const base::Value* pref_value;
+  EXPECT_TRUE(prefs().GetValue(prefs::kIdleTimeout, &pref_value));
+  EXPECT_TRUE(prefs().GetValue(prefs::kIdleTimeoutActions, &pref_value));
+}
+
+TEST_F(IdleTimeoutPolicyHandlerTest, ValidConfiguration) {
+  SetPolicyValue(policy::key::kIdleTimeout, base::Value(15));
+  base::Value::List list;
+  list.Append("close_browsers");
+  list.Append("show_profile_picker");
+  SetPolicyValue(policy::key::kIdleTimeoutActions,
+                 base::Value(std::move(list)));
+
+  CheckAndApplyPolicySettings();
+
+  // Should have no errors.
+  EXPECT_TRUE(errors().empty());
+
+  // Prefs should be set.
+  const base::Value* pref_value;
+  EXPECT_TRUE(prefs().GetValue(prefs::kIdleTimeout, &pref_value));
+  ASSERT_THAT(pref_value, testing::NotNull());
+  EXPECT_EQ(base::TimeDeltaToValue(base::Minutes(15)), *pref_value);
+
+  EXPECT_TRUE(prefs().GetValue(prefs::kIdleTimeoutActions, &pref_value));
+  ASSERT_THAT(pref_value, testing::NotNull());
+  EXPECT_TRUE(pref_value->is_list());
+  EXPECT_THAT(
+      pref_value->GetList(),
+      testing::ElementsAre(static_cast<int>(ActionType::kCloseBrowsers),
+                           static_cast<int>(ActionType::kShowProfilePicker)));
+}
+
+TEST_F(IdleTimeoutPolicyHandlerTest, OneMinuteMinimum) {
+  // Set the policy to 0, which should clamp the pref to 1.
+  SetPolicyValue(policy::key::kIdleTimeout, base::Value(0));
+  SetPolicyValue(policy::key::kIdleTimeoutActions,
+                 base::Value(base::Value::List()));
+
+  CheckAndApplyPolicySettings();
+
+  // Should have an error.
+  auto expected_error =
+      l10n_util::GetStringFUTF16(IDS_POLICY_OUT_OF_RANGE_ERROR, u"0");
+  EXPECT_EQ(errors().size(), 1U);
+  EXPECT_EQ(errors().begin()->second.message, expected_error);
+
+  // Prefs should be set.
+  const base::Value* pref_value;
+  EXPECT_TRUE(prefs().GetValue(prefs::kIdleTimeout, &pref_value));
+  ASSERT_THAT(pref_value, testing::NotNull());
+  EXPECT_EQ(base::TimeDeltaToValue(base::Minutes(1)), *pref_value);
+}
+
+TEST_F(IdleTimeoutPolicyHandlerTest, ActionNotRecognized) {
+  // IdleTimeoutActions is a list, but one of the elements is not recognized
+  // as a valid option. Recognized actions are applied, but not the others.
+  SetPolicyValue(policy::key::kIdleTimeout, base::Value(5));
+  base::Value::List list;
+  list.Append("close_browsers");
+  list.Append("show_profile_picker");
+  list.Append("added_in_future_version_of_chrome");
+  SetPolicyValue(policy::key::kIdleTimeoutActions,
+                 base::Value(std::move(list)));
+
+  CheckAndApplyPolicySettings();
+
+  // Should have an error.
+  auto expected_error = l10n_util::GetStringFUTF16(
+      IDS_POLICY_ERROR_WITH_PATH,
+      UTF8ToUTF16(policy::key::kIdleTimeoutActions) + u"[2]",
+      l10n_util::GetStringFUTF16(IDS_POLICY_SCHEMA_VALIDATION_ERROR,
+                                 u"Invalid value for string"));
+  EXPECT_EQ(errors().size(), 1U);
+  EXPECT_EQ(errors().begin()->second.message, expected_error);
+
+  // Prefs should not be set.
+  const base::Value* pref_value;
+  EXPECT_TRUE(prefs().GetValue(prefs::kIdleTimeout, &pref_value));
+  EXPECT_TRUE(prefs().GetValue(prefs::kIdleTimeoutActions, &pref_value));
+  ASSERT_THAT(pref_value, testing::NotNull());
+  EXPECT_TRUE(pref_value->is_list());
+  EXPECT_THAT(
+      pref_value->GetList(),
+      testing::ElementsAre(static_cast<int>(ActionType::kCloseBrowsers),
+                           static_cast<int>(ActionType::kShowProfilePicker)));
+}
+
+}  // namespace enterprise_idle
diff --git a/chrome/browser/extensions/api/debugger/debugger_apitest.cc b/chrome/browser/extensions/api/debugger/debugger_apitest.cc
index 3417a0681..6a45147 100644
--- a/chrome/browser/extensions/api/debugger/debugger_apitest.cc
+++ b/chrome/browser/extensions/api/debugger/debugger_apitest.cc
@@ -148,10 +148,9 @@
       extension_function_test_utils::RunFunctionAndReturnSingleResult(
           get_targets_function.get(), "[]", browser()));
   EXPECT_TRUE(value->is_list());
-  const base::ListValue& targets = base::Value::AsListValue(*value);
 
   std::string debugger_target_id;
-  for (const base::Value& target_value : targets.GetList()) {
+  for (const base::Value& target_value : value->GetList()) {
     EXPECT_TRUE(target_value.is_dict());
     absl::optional<int> id = target_value.FindIntKey("tabId");
     if (id == tab_id) {
diff --git a/chrome/browser/extensions/api/declarative_net_request/declarative_net_request_unittest.cc b/chrome/browser/extensions/api/declarative_net_request/declarative_net_request_unittest.cc
index d3f04c72..aae884a 100644
--- a/chrome/browser/extensions/api/declarative_net_request/declarative_net_request_unittest.cc
+++ b/chrome/browser/extensions/api/declarative_net_request/declarative_net_request_unittest.cc
@@ -375,11 +375,10 @@
             function.get(), "[]" /* args */, browser_context());
     ASSERT_TRUE(result);
     ASSERT_TRUE(result->is_list());
-    const base::ListValue& ids_value = base::Value::AsListValue(*result);
 
     std::u16string error;
     std::vector<std::string> actual_ids;
-    for (const auto& val : ids_value.GetList())
+    for (const auto& val : result->GetList())
       actual_ids.push_back(val.GetString());
 
     EXPECT_THAT(expected_ids, UnorderedElementsAreArray(actual_ids));
diff --git a/chrome/browser/extensions/api/messaging/native_messaging_launch_from_native_unittest.cc b/chrome/browser/extensions/api/messaging/native_messaging_launch_from_native_unittest.cc
index 389c17f2..99582ef 100644
--- a/chrome/browser/extensions/api/messaging/native_messaging_launch_from_native_unittest.cc
+++ b/chrome/browser/extensions/api/messaging/native_messaging_launch_from_native_unittest.cc
@@ -107,11 +107,9 @@
     EXPECT_TRUE(base::PathService::Get(DIR_TEST_DATA, &path));
 
     std::string error;
-    scoped_refptr<Extension> extension(
-        Extension::Create(path, mojom::ManifestLocation::kInternal,
-                          base::Value::AsDictionaryValue(
-                              base::Value(manifest_builder.BuildDict())),
-                          Extension::NO_FLAGS, &error));
+    scoped_refptr<Extension> extension(Extension::Create(
+        path, mojom::ManifestLocation::kInternal, manifest_builder.BuildDict(),
+        Extension::NO_FLAGS, &error));
     ASSERT_TRUE(extension.get()) << error;
     ExtensionRegistry::Get(&profile_)->AddEnabled(extension);
     extension_id_ = extension->id();
diff --git a/chrome/browser/extensions/api/webrtc_audio_private/webrtc_audio_private_browsertest.cc b/chrome/browser/extensions/api/webrtc_audio_private/webrtc_audio_private_browsertest.cc
index 02ee5ccf..0f7a550 100644
--- a/chrome/browser/extensions/api/webrtc_audio_private/webrtc_audio_private_browsertest.cc
+++ b/chrome/browser/extensions/api/webrtc_audio_private/webrtc_audio_private_browsertest.cc
@@ -142,20 +142,20 @@
   GetAudioDeviceDescriptions(false, &devices);
 
   std::unique_ptr<base::Value> result = InvokeGetSinks();
-  const base::ListValue& sink_list = base::Value::AsListValue(*result);
+  const base::Value::List& sink_list = result->GetList();
 
   std::string result_string;
   JSONWriter::Write(*result, &result_string);
   VLOG(2) << result_string;
 
-  EXPECT_EQ(devices.size(), sink_list.GetList().size());
+  EXPECT_EQ(devices.size(), sink_list.size());
 
   // Iterate through both lists in lockstep and compare. The order
   // should be identical.
   size_t ix = 0;
   AudioDeviceDescriptions::const_iterator it = devices.begin();
-  for (; ix < sink_list.GetList().size() && it != devices.end(); ++ix, ++it) {
-    const base::Value& value = sink_list.GetList()[ix];
+  for (; ix < sink_list.size() && it != devices.end(); ++ix, ++it) {
+    const base::Value& value = sink_list[ix];
     EXPECT_TRUE(value.is_dict());
     const base::Value::Dict& dict = value.GetDict();
     const std::string* sink_id = dict.FindString("sinkId");
diff --git a/chrome/browser/extensions/extension_install_prompt.cc b/chrome/browser/extensions/extension_install_prompt.cc
index f711ce3..855896ac 100644
--- a/chrome/browser/extensions/extension_install_prompt.cc
+++ b/chrome/browser/extensions/extension_install_prompt.cc
@@ -37,6 +37,7 @@
 #include "extensions/common/manifest_constants.h"
 #include "extensions/common/manifest_handlers/icons_handler.h"
 #include "extensions/common/permissions/permission_set.h"
+#include "third_party/abseil-cpp/absl/types/optional.h"
 #include "ui/base/l10n/l10n_util.h"
 #include "ui/base/resource/resource_bundle.h"
 #include "ui/base/ui_base_types.h"
@@ -476,30 +477,27 @@
 // static
 scoped_refptr<Extension>
 ExtensionInstallPrompt::GetLocalizedExtensionForDisplay(
-    const base::DictAdapterForMigration manifest,
+    const base::Value::Dict& manifest,
     int flags,
     const std::string& id,
     const std::string& localized_name,
     const std::string& localized_description,
     std::string* error) {
-  std::unique_ptr<base::DictionaryValue> localized_manifest;
+  absl::optional<base::Value::Dict> localized_manifest;
   if (!localized_name.empty() || !localized_description.empty()) {
-    localized_manifest = base::DictionaryValue::From(
-        base::Value::ToUniquePtrValue(base::Value(manifest.Clone())));
+    localized_manifest = manifest.Clone();
     if (!localized_name.empty()) {
-      localized_manifest->SetStringKey(extensions::manifest_keys::kName,
-                                       localized_name);
+      localized_manifest->Set(extensions::manifest_keys::kName, localized_name);
     }
     if (!localized_description.empty()) {
-      localized_manifest->SetStringKey(extensions::manifest_keys::kDescription,
-                                       localized_description);
+      localized_manifest->Set(extensions::manifest_keys::kDescription,
+                              localized_description);
     }
   }
 
   return Extension::Create(
       base::FilePath(), extensions::mojom::ManifestLocation::kInternal,
-      localized_manifest.get() ? *localized_manifest : manifest, flags, id,
-      error);
+      localized_manifest ? *localized_manifest : manifest, flags, id, error);
 }
 
 ExtensionInstallPrompt::ExtensionInstallPrompt(content::WebContents* contents)
diff --git a/chrome/browser/extensions/extension_install_prompt.h b/chrome/browser/extensions/extension_install_prompt.h
index e67259d..a8feacb 100644
--- a/chrome/browser/extensions/extension_install_prompt.h
+++ b/chrome/browser/extensions/extension_install_prompt.h
@@ -285,7 +285,7 @@
   // Creates a dummy extension from the |manifest|, replacing the name and
   // description with the localizations if provided.
   static scoped_refptr<extensions::Extension> GetLocalizedExtensionForDisplay(
-      const base::DictAdapterForMigration manifest,
+      const base::Value::Dict& manifest,
       int flags,  // Extension::InitFromValueFlags
       const std::string& id,
       const std::string& localized_name,
diff --git a/chrome/browser/flag-metadata.json b/chrome/browser/flag-metadata.json
index b2e5bcd..f792675 100644
--- a/chrome/browser/flag-metadata.json
+++ b/chrome/browser/flag-metadata.json
@@ -2290,7 +2290,7 @@
   {
     "name": "enable-external-keyboards-in-diagnostics-app",
     "owners": [ "//ash/webui/diagnostics_ui/OWNERS" ],
-    "expiry_milestone": 110
+    "expiry_milestone": 120
   },
   {
     "name": "enable-fake-keyboard-heuristic",
@@ -2627,7 +2627,7 @@
   {
     "name": "enable-log-controller-for-diagnostics-app",
     "owners": [ "//ash/webui/diagnostics_ui/OWNERS" ],
-    "expiry_milestone": 110
+    "expiry_milestone": 120
   },
   {
     "name": "enable-logging-js-console-messages",
@@ -3189,7 +3189,7 @@
   {
     "name": "enable-touchpads-in-diagnostics-app",
     "owners": [ "//ash/webui/diagnostics_ui/OWNERS" ],
-    "expiry_milestone": 110
+    "expiry_milestone": 120
   },
   {
     "name": "enable-touchscreen-calibration",
@@ -3201,7 +3201,7 @@
   {
     "name": "enable-touchscreens-in-diagnostics-app",
     "owners": [ "//ash/webui/diagnostics_ui/OWNERS" ],
-    "expiry_milestone": 110
+    "expiry_milestone": 120
   },
   {
     "name": "enable-translate-sub-frames",
diff --git a/chrome/browser/flag_descriptions.cc b/chrome/browser/flag_descriptions.cc
index 8872845..64d529d 100644
--- a/chrome/browser/flag_descriptions.cc
+++ b/chrome/browser/flag_descriptions.cc
@@ -289,10 +289,6 @@
     "grouped into processes based on their URL's site or origin. The default "
     "grouping when enabled is per-site.";
 
-const char kAssistantConsentModalName[] = "AssistantConsentModal";
-const char kAssistantConsentModalDescription[] =
-    "Enables the modal version of the Assistant voice search consent dialog.";
-
 const char kAssistantConsentSimplifiedTextName[] =
     "AssistantConsentSimplifiedText";
 const char kAssistantConsentSimplifiedTextDescription[] =
diff --git a/chrome/browser/flag_descriptions.h b/chrome/browser/flag_descriptions.h
index 752a9f4c..8acf4ea 100644
--- a/chrome/browser/flag_descriptions.h
+++ b/chrome/browser/flag_descriptions.h
@@ -168,9 +168,6 @@
 extern const char kUseCustomMessagesDomainName[];
 extern const char kUseCustomMessagesDomainDescription[];
 
-extern const char kAssistantConsentModalName[];
-extern const char kAssistantConsentModalDescription[];
-
 extern const char kAssistantConsentSimplifiedTextName[];
 extern const char kAssistantConsentSimplifiedTextDescription[];
 
diff --git a/chrome/browser/flags/android/chrome_feature_list.cc b/chrome/browser/flags/android/chrome_feature_list.cc
index 1a39009..d704d449 100644
--- a/chrome/browser/flags/android/chrome_feature_list.cc
+++ b/chrome/browser/flags/android/chrome_feature_list.cc
@@ -174,7 +174,6 @@
     &kAllowNewIncognitoTabIntents,
     &kAndroidScrollOptimizations,
     &kAndroidSearchEngineChoiceNotification,
-    &kAssistantConsentModal,
     &kAssistantConsentSimplifiedText,
     &kAssistantConsentV2,
     &kAssistantIntentExperimentId,
@@ -476,10 +475,6 @@
              "AndroidSearchEngineChoiceNotification",
              base::FEATURE_ENABLED_BY_DEFAULT);
 
-BASE_FEATURE(kAssistantConsentModal,
-             "AssistantConsentModal",
-             base::FEATURE_DISABLED_BY_DEFAULT);
-
 BASE_FEATURE(kAssistantConsentSimplifiedText,
              "AssistantConsentSimplifiedText",
              base::FEATURE_DISABLED_BY_DEFAULT);
diff --git a/chrome/browser/flags/android/chrome_feature_list.h b/chrome/browser/flags/android/chrome_feature_list.h
index c58a6f0..8e91bda3 100644
--- a/chrome/browser/flags/android/chrome_feature_list.h
+++ b/chrome/browser/flags/android/chrome_feature_list.h
@@ -20,7 +20,6 @@
 BASE_DECLARE_FEATURE(kAllowNewIncognitoTabIntents);
 BASE_DECLARE_FEATURE(kAndroidScrollOptimizations);
 BASE_DECLARE_FEATURE(kAndroidSearchEngineChoiceNotification);
-BASE_DECLARE_FEATURE(kAssistantConsentModal);
 BASE_DECLARE_FEATURE(kAssistantConsentSimplifiedText);
 BASE_DECLARE_FEATURE(kAssistantConsentV2);
 BASE_DECLARE_FEATURE(kAssistantIntentExperimentId);
diff --git a/chrome/browser/flags/android/java/src/org/chromium/chrome/browser/flags/ChromeFeatureList.java b/chrome/browser/flags/android/java/src/org/chromium/chrome/browser/flags/ChromeFeatureList.java
index fbfe38a..c4ab08ed 100644
--- a/chrome/browser/flags/android/java/src/org/chromium/chrome/browser/flags/ChromeFeatureList.java
+++ b/chrome/browser/flags/android/java/src/org/chromium/chrome/browser/flags/ChromeFeatureList.java
@@ -195,7 +195,6 @@
     public static final String ANONYMOUS_UPDATE_CHECKS = "AnonymousUpdateChecks";
     public static final String APP_LANGUAGE_PROMPT = "AppLanguagePrompt";
     public static final String APP_LANGUAGE_PROMPT_ULP = "AppLanguagePromptULP";
-    public static final String ASSISTANT_CONSENT_MODAL = "AssistantConsentModal";
     public static final String ASSISTANT_CONSENT_SIMPLIFIED_TEXT = "AssistantConsentSimplifiedText";
     public static final String ASSISTANT_CONSENT_V2 = "AssistantConsentV2";
     public static final String ASSISTANT_INTENT_EXPERIMENT_ID = "AssistantIntentExperimentId";
@@ -660,7 +659,7 @@
     public static final CachedFlag sInstantStart = new CachedFlag(INSTANT_START, false);
     public static final CachedFlag sInterestFeedV2 = new CachedFlag(INTEREST_FEED_V2, true);
     public static final CachedFlag sLensCameraAssistedSearch =
-            new CachedFlag(LENS_CAMERA_ASSISTED_SEARCH, false);
+            new CachedFlag(LENS_CAMERA_ASSISTED_SEARCH, true);
     public static final CachedFlag sNewWindowAppMenu = new CachedFlag(NEW_WINDOW_APP_MENU, true);
     public static final CachedFlag sOmahaMinSdkVersionAndroid =
             new CachedFlag(OMAHA_MIN_SDK_VERSION_ANDROID, false);
diff --git a/chrome/browser/lacros/chrome_browser_main_extra_parts_lacros.cc b/chrome/browser/lacros/chrome_browser_main_extra_parts_lacros.cc
index 71eab4c..be3e1fd 100644
--- a/chrome/browser/lacros/chrome_browser_main_extra_parts_lacros.cc
+++ b/chrome/browser/lacros/chrome_browser_main_extra_parts_lacros.cc
@@ -34,6 +34,7 @@
 #include "chrome/browser/lacros/sync/sync_crosapi_manager_lacros.h"
 #include "chrome/browser/lacros/task_manager_lacros.h"
 #include "chrome/browser/lacros/ui_metric_recorder_lacros.h"
+#include "chrome/browser/lacros/views_text_services_context_menu_lacros.h"
 #include "chrome/browser/lacros/vpn_extension_tracker_lacros.h"
 #include "chrome/browser/lacros/web_app_provider_bridge_lacros.h"
 #include "chrome/browser/lacros/web_page_info_lacros.h"
@@ -49,6 +50,7 @@
 #include "chromeos/startup/browser_params_proxy.h"
 #include "components/arc/common/intent_helper/arc_icon_cache_delegate.h"
 #include "extensions/common/features/feature_session_type.h"
+#include "ui/views/controls/views_text_services_context_menu_chromeos.h"
 
 namespace {
 
@@ -231,4 +233,13 @@
     chrome_kiosk_launch_controller_ =
         std::make_unique<ChromeKioskLaunchControllerLacros>(*profile);
   }
+
+  views::ViewsTextServicesContextMenuChromeos::SetImplFactory(
+      base::BindRepeating(
+          [](ui::SimpleMenuModel* menu_model, views::Textfield* textfield)
+              -> std::unique_ptr<views::ViewsTextServicesContextMenu> {
+            return std::make_unique<
+                crosapi::ViewsTextServicesContextMenuLacros>(menu_model,
+                                                             textfield);
+          }));
 }
diff --git a/chrome/browser/lacros/standalone_browser_test_controller.cc b/chrome/browser/lacros/standalone_browser_test_controller.cc
index e848e1bc..313f529 100644
--- a/chrome/browser/lacros/standalone_browser_test_controller.cc
+++ b/chrome/browser/lacros/standalone_browser_test_controller.cc
@@ -131,8 +131,7 @@
   std::string error;
   auto extension = extensions::Extension::Create(
       base::FilePath(), extensions::mojom::ManifestLocation::kUnpacked,
-      base::Value::AsDictionaryValue(
-          base::Value(CreateVpnExtensionManifest(extension_name))),
+      CreateVpnExtensionManifest(extension_name),
       extensions::Extension::NO_FLAGS, &error);
   if (!error.empty()) {
     std::move(callback).Run(error);
diff --git a/chrome/browser/lacros/views_text_services_context_menu_lacros.cc b/chrome/browser/lacros/views_text_services_context_menu_lacros.cc
new file mode 100644
index 0000000..6a4f95ce
--- /dev/null
+++ b/chrome/browser/lacros/views_text_services_context_menu_lacros.cc
@@ -0,0 +1,123 @@
+// Copyright 2022 The Chromium Authors
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include "chrome/browser/lacros/views_text_services_context_menu_lacros.h"
+
+#include "chrome/app/chrome_command_ids.h"
+#include "chromeos/crosapi/mojom/clipboard_history.mojom.h"
+#include "chromeos/lacros/lacros_service.h"
+#include "ui/base/clipboard/clipboard.h"
+#include "ui/base/data_transfer_policy/data_transfer_endpoint.h"
+#include "ui/base/l10n/l10n_util.h"
+#include "ui/base/pointer/touch_editing_controller.h"
+#include "ui/strings/grit/ui_strings.h"
+#include "ui/views/controls/textfield/textfield.h"
+#include "ui/views/controls/views_text_services_context_menu_base.h"
+
+namespace {
+
+bool IsClipboardHistoryLacrosServiceAvailable() {
+  auto* const service = chromeos::LacrosService::Get();
+  return service && service->IsAvailable<crosapi::mojom::ClipboardHistory>();
+}
+
+bool IsClipboardHistoryEmpty() {
+  ui::DataTransferEndpoint dte(ui::EndpointType::kClipboardHistory);
+  std::vector<std::u16string> types;
+  ui::Clipboard::GetForCurrentThread()->ReadAvailableTypes(
+      ui::ClipboardBuffer::kCopyPaste, &dte, &types);
+  return types.empty();
+}
+
+}  // namespace
+
+namespace crosapi {
+
+ViewsTextServicesContextMenuLacros::ViewsTextServicesContextMenuLacros(
+    ui::SimpleMenuModel* menu,
+    views::Textfield* client)
+    : views::ViewsTextServicesContextMenuBase(menu, client) {
+  if (!IsClipboardHistoryLacrosServiceAvailable())
+    return;
+
+  // If the menu has a paste option, add a clipboard history option as well.
+  const absl::optional<size_t> paste_index =
+      menu->GetIndexOfCommandId(ui::TouchEditable::kPaste);
+
+  if (!paste_index.has_value())
+    return;
+
+  const size_t target_index = paste_index.value() + 1;
+  menu->InsertItemAt(target_index, IDS_APP_SHOW_CLIPBOARD_HISTORY,
+                     l10n_util::GetStringUTF16(IDS_APP_SHOW_CLIPBOARD_HISTORY));
+}
+
+ViewsTextServicesContextMenuLacros::~ViewsTextServicesContextMenuLacros() =
+    default;
+
+bool ViewsTextServicesContextMenuLacros::GetAcceleratorForCommandId(
+    int command_id,
+    ui::Accelerator* accelerator) const {
+  if (command_id == IDS_APP_SHOW_CLIPBOARD_HISTORY) {
+    *accelerator = ui::Accelerator(ui::VKEY_V, ui::EF_COMMAND_DOWN);
+    return true;
+  }
+
+  return ViewsTextServicesContextMenuBase::GetAcceleratorForCommandId(
+      command_id, accelerator);
+}
+
+bool ViewsTextServicesContextMenuLacros::IsCommandIdChecked(
+    int command_id) const {
+  if (command_id == IDS_APP_SHOW_CLIPBOARD_HISTORY)
+    return true;
+
+  return ViewsTextServicesContextMenuBase::IsCommandIdChecked(command_id);
+}
+
+bool ViewsTextServicesContextMenuLacros::IsCommandIdEnabled(
+    int command_id) const {
+  if (command_id == IDS_APP_SHOW_CLIPBOARD_HISTORY) {
+    return IsClipboardHistoryLacrosServiceAvailable() &&
+           !IsClipboardHistoryEmpty();
+  }
+
+  return ViewsTextServicesContextMenuBase::IsCommandIdEnabled(command_id);
+}
+
+void ViewsTextServicesContextMenuLacros::ExecuteCommand(int command_id,
+                                                        int event_flags) {
+  if (command_id == IDS_APP_SHOW_CLIPBOARD_HISTORY) {
+    if (!IsClipboardHistoryLacrosServiceAvailable())
+      return;
+
+    // Calculate the menu source type from `event_flags`.
+    ui::MenuSourceType source_type;
+    if (event_flags & ui::EF_LEFT_MOUSE_BUTTON) {
+      source_type = ui::MENU_SOURCE_MOUSE;
+    } else if (event_flags & ui::EF_FROM_TOUCH) {
+      source_type = ui::MENU_SOURCE_TOUCH;
+    } else {
+      source_type = ui::MENU_SOURCE_KEYBOARD;
+    }
+
+    chromeos::LacrosService::Get()
+        ->GetRemote<mojom::ClipboardHistory>()
+        ->ShowClipboard(
+            client()->GetCaretBounds(), source_type,
+            mojom::ClipboardHistoryControllerShowSource::kTextfieldContextMenu);
+    return;
+  }
+
+  ViewsTextServicesContextMenuBase::ExecuteCommand(command_id, event_flags);
+}
+
+bool ViewsTextServicesContextMenuLacros::SupportsCommand(int command_id) const {
+  if (command_id == IDS_APP_SHOW_CLIPBOARD_HISTORY)
+    return true;
+
+  return ViewsTextServicesContextMenuBase::SupportsCommand(command_id);
+}
+
+}  // namespace crosapi
diff --git a/chrome/browser/lacros/views_text_services_context_menu_lacros.h b/chrome/browser/lacros/views_text_services_context_menu_lacros.h
new file mode 100644
index 0000000..4342bc6b
--- /dev/null
+++ b/chrome/browser/lacros/views_text_services_context_menu_lacros.h
@@ -0,0 +1,45 @@
+// Copyright 2022 The Chromium Authors
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#ifndef CHROME_BROWSER_LACROS_VIEWS_TEXT_SERVICES_CONTEXT_MENU_LACROS_H_
+#define CHROME_BROWSER_LACROS_VIEWS_TEXT_SERVICES_CONTEXT_MENU_LACROS_H_
+
+#include "ui/views/controls/views_text_services_context_menu_base.h"
+
+namespace ui {
+class SimpleMenuModel;
+}  // namespace ui
+
+namespace views {
+class Textfield;
+}  // namespace views
+
+namespace crosapi {
+
+// This class implements support for adding and handling text service items in
+// lacros-chrome browser native textfields (i.e., the omnibox but not the WebUI
+// embedded in the browser).
+class ViewsTextServicesContextMenuLacros
+    : public views::ViewsTextServicesContextMenuBase {
+ public:
+  ViewsTextServicesContextMenuLacros(ui::SimpleMenuModel* menu,
+                                     views::Textfield* client);
+  ViewsTextServicesContextMenuLacros(
+      const ViewsTextServicesContextMenuLacros&) = delete;
+  ViewsTextServicesContextMenuLacros& operator=(
+      const ViewsTextServicesContextMenuLacros&) = delete;
+  ~ViewsTextServicesContextMenuLacros() override;
+
+  // ViewsTextServicesContextMenuBase:
+  bool GetAcceleratorForCommandId(int command_id,
+                                  ui::Accelerator* accelerator) const override;
+  bool IsCommandIdChecked(int command_id) const override;
+  bool IsCommandIdEnabled(int command_id) const override;
+  void ExecuteCommand(int command_id, int event_flags) override;
+  bool SupportsCommand(int command_id) const override;
+};
+
+}  // namespace crosapi
+
+#endif  // CHROME_BROWSER_LACROS_VIEWS_TEXT_SERVICES_CONTEXT_MENU_LACROS_H_
diff --git a/chrome/browser/media/router/mojo/media_router_mojo_impl.cc b/chrome/browser/media/router/mojo/media_router_mojo_impl.cc
index 8779cdb..e4fd52c 100644
--- a/chrome/browser/media/router/mojo/media_router_mojo_impl.cc
+++ b/chrome/browser/media/router/mojo/media_router_mojo_impl.cc
@@ -106,8 +106,8 @@
 
 void MediaRouterMojoImpl::Initialize() {
   DCHECK(!internal_routes_observer_);
-  // The observer calls virtual methods on MediaRouter; it must be created
-  // outside of the ctor
+  // Because observer calls virtual methods on MediaRouter, it must be created
+  // outside of the constructor.
   internal_routes_observer_ =
       std::make_unique<InternalMediaRoutesObserver>(this);
 }
@@ -122,7 +122,7 @@
       std::move(media_route_provider_remote));
   bound_remote.set_disconnect_handler(
       base::BindOnce(&MediaRouterMojoImpl::OnProviderConnectionError,
-                     weak_factory_.GetWeakPtr(), provider_id));
+                     AsWeakPtr(), provider_id));
   media_route_providers_[provider_id] = std::move(bound_remote);
 }
 
@@ -250,9 +250,9 @@
   const mojom::MediaRouteProviderId provider_id = sink->provider_id();
 
   const std::string presentation_id = MediaRouterBase::CreatePresentationId();
-  auto mr_callback = base::BindOnce(
-      &MediaRouterMojoImpl::RouteResponseReceived, weak_factory_.GetWeakPtr(),
-      presentation_id, provider_id, off_the_record, std::move(callback), false);
+  auto mr_callback = base::BindOnce(&MediaRouterMojoImpl::RouteResponseReceived,
+                                    AsWeakPtr(), presentation_id, provider_id,
+                                    off_the_record, std::move(callback), false);
 
   if (source.IsDesktopMirroringSource()) {
     desktop_picker_.Show(
@@ -260,9 +260,9 @@
         {DesktopMediaList::Type::kScreen},
         base::BindRepeating([](content::WebContents* wc) { return true; }),
         base::BindOnce(&MediaRouterMojoImpl::CreateRouteWithSelectedDesktop,
-                       weak_factory_.GetWeakPtr(), provider_id, sink_id,
-                       presentation_id, origin, web_contents, timeout,
-                       off_the_record, std::move(mr_callback)));
+                       AsWeakPtr(), provider_id, sink_id, presentation_id,
+                       origin, web_contents, timeout, off_the_record,
+                       std::move(mr_callback)));
   } else {
     const int frame_tree_node_id =
         web_contents ? web_contents->GetPrimaryMainFrame()->GetFrameTreeNodeId()
@@ -296,9 +296,9 @@
   const int frame_tree_node_id =
       web_contents ? web_contents->GetPrimaryMainFrame()->GetFrameTreeNodeId()
                    : kDefaultFrameTreeNodeId;
-  auto mr_callback = base::BindOnce(
-      &MediaRouterMojoImpl::RouteResponseReceived, weak_factory_.GetWeakPtr(),
-      presentation_id, *provider_id, off_the_record, std::move(callback), true);
+  auto mr_callback = base::BindOnce(&MediaRouterMojoImpl::RouteResponseReceived,
+                                    AsWeakPtr(), presentation_id, *provider_id,
+                                    off_the_record, std::move(callback), true);
   media_route_providers_[*provider_id]->JoinRoute(
       source_id, presentation_id, origin, frame_tree_node_id, timeout,
       off_the_record, std::move(mr_callback));
@@ -313,9 +313,8 @@
         mojom::RouteRequestResultCode::ROUTE_NOT_FOUND);
     return;
   }
-  auto callback =
-      base::BindOnce(&MediaRouterMojoImpl::OnTerminateRouteResult,
-                     weak_factory_.GetWeakPtr(), route_id, *provider_id);
+  auto callback = base::BindOnce(&MediaRouterMojoImpl::OnTerminateRouteResult,
+                                 AsWeakPtr(), route_id, *provider_id);
   media_route_providers_[*provider_id]->TerminateRoute(route_id,
                                                        std::move(callback));
 }
@@ -380,7 +379,7 @@
     return;
   }
   auto callback = base::BindOnce(&MediaRouterMojoImpl::OnMediaControllerCreated,
-                                 weak_factory_.GetWeakPtr(), route_id);
+                                 AsWeakPtr(), route_id);
   media_route_providers_[*provider_id]->CreateMediaRouteController(
       route_id, std::move(controller), std::move(observer),
       std::move(callback));
@@ -562,7 +561,7 @@
 void MediaRouterMojoImpl::RegisterMediaRoutesObserver(
     MediaRoutesObserver* observer) {
   DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
-  bool is_first_observer = !routes_query_.HasObservers();
+  const bool is_first_observer = !routes_query_.HasObservers();
   if (!is_first_observer)
     DCHECK(!routes_query_.HasObserver(observer));
 
@@ -578,19 +577,15 @@
     // MediaRoutesObserver is calling this method from its constructor, and that
     // must complete before invoking its virtual OnRoutesUpdated() method.
     content::GetUIThreadTaskRunner({})->PostTask(
-        FROM_HERE,
-        base::BindOnce(&MediaRouterMojoImpl::NotifyOfExistingRoutesIfRegistered,
-                       weak_factory_.GetWeakPtr(),
-                       // Safe because we check the routes_query_ in the notify
-                       // method before usage.
-                       base::Unretained(observer)));
+        FROM_HERE, base::BindOnce(&MediaRouterMojoImpl::NotifyOfExistingRoutes,
+                                  AsWeakPtr(), observer->AsWeakPtr()));
   }
 }
 
-void MediaRouterMojoImpl::NotifyOfExistingRoutesIfRegistered(
-    MediaRoutesObserver* observer) const {
+void MediaRouterMojoImpl::NotifyOfExistingRoutes(
+    base::WeakPtr<MediaRoutesObserver> observer) const {
   DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
-  if (routes_query_.HasObserver(observer)) {
+  if (observer) {
     observer->OnRoutesUpdated(
         routes_query_.cached_route_list().value_or(std::vector<MediaRoute>{}));
   }
@@ -598,6 +593,7 @@
 
 void MediaRouterMojoImpl::UnregisterMediaRoutesObserver(
     MediaRoutesObserver* observer) {
+  DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
   routes_query_.RemoveObserver(observer);
 }
 
@@ -694,12 +690,14 @@
 
 void MediaRouterMojoImpl::OnRouteAdded(mojom::MediaRouteProviderId provider_id,
                                        const MediaRoute& route) {
+  DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
   routes_query_.AddRouteForProvider(provider_id, route);
   routes_query_.NotifyObservers();
 }
 
 void MediaRouterMojoImpl::SyncStateToMediaRouteProvider(
     mojom::MediaRouteProviderId provider_id) {
+  DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
   const auto& provider = media_route_providers_[provider_id];
   // Sink queries.
   for (const auto& it : sinks_queries_) {
@@ -799,6 +797,7 @@
 
 absl::optional<mojom::MediaRouteProviderId>
 MediaRouterMojoImpl::GetProviderIdForRoute(const MediaRoute::Id& route_id) {
+  DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
   for (const auto& provider_to_routes : routes_query_.providers_to_routes()) {
     const mojom::MediaRouteProviderId provider_id = provider_to_routes.first;
     const std::vector<MediaRoute>& routes = provider_to_routes.second;
@@ -820,6 +819,7 @@
 absl::optional<mojom::MediaRouteProviderId>
 MediaRouterMojoImpl::GetProviderIdForPresentation(
     const std::string& presentation_id) {
+  DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
   for (const auto& provider_to_routes : routes_query_.providers_to_routes()) {
     const mojom::MediaRouteProviderId provider_id = provider_to_routes.first;
     const std::vector<MediaRoute>& routes = provider_to_routes.second;
@@ -972,8 +972,7 @@
             std::move(inner_callback)
                 .Run(route, std::move(connection), error_text, result_code);
           },
-          std::move(mr_callback), weak_factory_.GetWeakPtr(),
-          request.stream_id));
+          std::move(mr_callback), AsWeakPtr(), request.stream_id));
 }
 
 }  // namespace media_router
diff --git a/chrome/browser/media/router/mojo/media_router_mojo_impl.h b/chrome/browser/media/router/mojo/media_router_mojo_impl.h
index b9375b6c..b2b2b16 100644
--- a/chrome/browser/media/router/mojo/media_router_mojo_impl.h
+++ b/chrome/browser/media/router/mojo/media_router_mojo_impl.h
@@ -49,7 +49,9 @@
 enum class MediaRouteProviderWakeReason;
 
 // MediaRouter implementation that delegates calls to a MediaRouteProvider.
-class MediaRouterMojoImpl : public MediaRouterBase, public mojom::MediaRouter {
+class MediaRouterMojoImpl : public MediaRouterBase,
+                            public mojom::MediaRouter,
+                            public base::SupportsWeakPtr<MediaRouterMojoImpl> {
  public:
   MediaRouterMojoImpl(const MediaRouterMojoImpl&) = delete;
   MediaRouterMojoImpl& operator=(const MediaRouterMojoImpl&) = delete;
@@ -215,7 +217,7 @@
   // Represents a query to the MediaRouteProviders for media routes and caches
   // media routes returned by MRPs. Holds observers for the query.
   //
-  // NOTE: If the to-do below for providers_for_routes_ is fixed, then this
+  // NOTE: If the to-do below for providers_to_routes_ is fixed, then this
   // entire class can be replaced with a std::vector<MediaRoute> and a
   // base::ObserverList of observers.
   class MediaRoutesQuery {
@@ -264,7 +266,7 @@
     base::flat_map<mojom::MediaRouteProviderId, std::vector<MediaRoute>>
         providers_to_routes_;
 
-    base::ObserverList<MediaRoutesObserver>::Unchecked observers_;
+    base::ObserverList<MediaRoutesObserver> observers_;
   };
 
   // See note in OnDesktopPickerDone().
@@ -308,7 +310,8 @@
 
   // Notifies |observer| of any existing cached routes, if it is still
   // registered.
-  void NotifyOfExistingRoutesIfRegistered(MediaRoutesObserver* observer) const;
+  void NotifyOfExistingRoutes(
+      base::WeakPtr<MediaRoutesObserver> observer) const;
 
   // mojom::MediaRouter implementation.
   void OnIssue(const IssueInfo& issue) override;
@@ -424,7 +427,7 @@
       sinks_queries_;
 
   // Holds observers for media route updates and a map of providers to route
-  // ids..
+  // ids.
   MediaRoutesQuery routes_query_;
 
   using PresentationConnectionMessageObserverList =
@@ -445,8 +448,6 @@
   // Collects logs from the Media Router and the native Media Route Providers.
   // TODO(crbug.com/1077138): Limit logging before Media Router usage.
   LoggerImpl logger_;
-
-  base::WeakPtrFactory<MediaRouterMojoImpl> weak_factory_{this};
 };
 
 }  // namespace media_router
diff --git a/chrome/browser/media/router/mojo/media_router_mojo_impl_unittest.cc b/chrome/browser/media/router/mojo/media_router_mojo_impl_unittest.cc
index 2a71c01b..8464b78 100644
--- a/chrome/browser/media/router/mojo/media_router_mojo_impl_unittest.cc
+++ b/chrome/browser/media/router/mojo/media_router_mojo_impl_unittest.cc
@@ -36,7 +36,6 @@
 #include "components/media_router/common/media_source.h"
 #include "components/media_router/common/test/test_helper.h"
 #include "components/version_info/version_info.h"
-#include "content/public/test/browser_task_environment.h"
 #include "mojo/public/cpp/bindings/pending_receiver.h"
 #include "mojo/public/cpp/bindings/remote.h"
 #include "testing/gmock/include/gmock/gmock.h"
@@ -619,33 +618,38 @@
 TEST_F(MediaRouterMojoImplTest, RegisterAndUnregisterMediaRoutesObserver) {
   MockMediaRouter mock_router;
 
-  MediaRoutesObserver* observer_captured;
-  EXPECT_CALL(mock_router, RegisterMediaRoutesObserver(_))
-      .Times(3)
-      .WillRepeatedly(SaveArg<0>(&observer_captured));
-  MockMediaRoutesObserver routes_observer(&mock_router);
-  EXPECT_EQ(observer_captured, &routes_observer);
+  auto routes_observer =
+      std::make_unique<MockMediaRoutesObserver>(&mock_router);
+  EXPECT_TRUE(
+      mock_router.routes_observers().HasObserver(routes_observer.get()));
 
-  MockMediaRoutesObserver extra_routes_observer(&mock_router);
-  EXPECT_EQ(observer_captured, &extra_routes_observer);
+  auto extra_routes_observer =
+      std::make_unique<MockMediaRoutesObserver>(&mock_router);
+  EXPECT_TRUE(
+      mock_router.routes_observers().HasObserver(extra_routes_observer.get()));
 
-  MockMediaRoutesObserver different_routes_observer(&mock_router);
-  EXPECT_EQ(observer_captured, &different_routes_observer);
+  routes_observer.reset();
+  extra_routes_observer.reset();
+  EXPECT_TRUE(mock_router.routes_observers().empty());
+}
 
-  EXPECT_CALL(mock_router, UnregisterMediaRoutesObserver(&routes_observer));
-  EXPECT_CALL(mock_router,
-              UnregisterMediaRoutesObserver(&extra_routes_observer));
-  EXPECT_CALL(mock_router,
-              UnregisterMediaRoutesObserver(&different_routes_observer));
-  UnregisterMediaRoutesObserver(&routes_observer);
-  UnregisterMediaRoutesObserver(&extra_routes_observer);
-  UnregisterMediaRoutesObserver(&different_routes_observer);
+TEST_F(MediaRouterMojoImplTest, UnregisterBeforeNotificationDoesntCrash) {
+  auto routes_observer = std::make_unique<MediaRoutesObserver>(router());
+  auto routes_observer_two = std::make_unique<MediaRoutesObserver>(router());
+
+  // Resetting the observer immediately should cause it to be invalidated before
+  // the callback for NotifyOfExistingRoutesIfRegistered is called.
+  routes_observer_two.reset();
+
+  // Make sure the Notify task executes.
+  base::RunLoop().RunUntilIdle();
 }
 
 TEST_F(MediaRouterMojoImplTest, RegisteredObserversGetMediaRouteUpdates) {
-  StrictMock<MockMediaRoutesObserver> routes_observer(router());
-  StrictMock<MockMediaRoutesObserver> extra_routes_observer(router());
-  StrictMock<MockMediaRoutesObserver> different_routes_observer(router());
+  auto routes_observer =
+      std::make_unique<StrictMock<MockMediaRoutesObserver>>(router());
+  auto extra_routes_observer =
+      std::make_unique<StrictMock<MockMediaRoutesObserver>>(router());
 
   MediaSource media_source(kSource);
   std::vector<MediaRoute> expected_routes{
@@ -656,16 +660,14 @@
   incognito_expected_route.set_off_the_record(true);
   expected_routes.push_back(incognito_expected_route);
 
-  EXPECT_CALL(routes_observer, OnRoutesUpdated(expected_routes)).Times(1);
-  EXPECT_CALL(extra_routes_observer, OnRoutesUpdated(expected_routes)).Times(1);
-  EXPECT_CALL(different_routes_observer, OnRoutesUpdated(expected_routes))
+  EXPECT_CALL(*routes_observer, OnRoutesUpdated(expected_routes)).Times(1);
+  EXPECT_CALL(*extra_routes_observer, OnRoutesUpdated(expected_routes))
       .Times(1);
 
   UpdateRoutes(mojom::MediaRouteProviderId::CAST, expected_routes);
 
-  UnregisterMediaRoutesObserver(&routes_observer);
-  UnregisterMediaRoutesObserver(&extra_routes_observer);
-  UnregisterMediaRoutesObserver(&different_routes_observer);
+  routes_observer.reset();
+  extra_routes_observer.reset();
 
   // No route observers should be notified.
   UpdateRoutes(mojom::MediaRouteProviderId::CAST, expected_routes);
diff --git a/chrome/browser/media/router/test/media_router_mojo_test.cc b/chrome/browser/media/router/test/media_router_mojo_test.cc
index 914a264..230c92b 100644
--- a/chrome/browser/media/router/test/media_router_mojo_test.cc
+++ b/chrome/browser/media/router/test/media_router_mojo_test.cc
@@ -141,8 +141,9 @@
 void MediaRouterMojoTest::ProvideTestRoute(
     mojom::MediaRouteProviderId provider_id,
     const MediaRoute::Id& route_id) {
-  if (!routes_observer_)
+  if (!routes_observer_) {
     routes_observer_ = std::make_unique<MediaRoutesObserver>(router());
+  }
   MediaRoute route = CreateMediaRoute();
   route.set_media_route_id(route_id);
   router()->OnRoutesUpdated(provider_id, {route});
diff --git a/chrome/browser/metrics/structured/BUILD.gn b/chrome/browser/metrics/structured/BUILD.gn
index 6e9c44e0..df9cd4f 100644
--- a/chrome/browser/metrics/structured/BUILD.gn
+++ b/chrome/browser/metrics/structured/BUILD.gn
@@ -28,9 +28,14 @@
     sources += [
       "ash_structured_metrics_recorder.cc",
       "ash_structured_metrics_recorder.h",
+      "structured_metrics_user_session_observer.cc",
+      "structured_metrics_user_session_observer.h",
     ]
 
-    deps += [ "//chrome/browser/ash/crosapi" ]
+    deps += [
+      "//chrome/browser/ash/crosapi",
+      "//components/user_manager:user_manager",
+    ]
   }
 
   if (is_chromeos_lacros) {
diff --git a/chrome/browser/metrics/structured/ash_structured_metrics_recorder.cc b/chrome/browser/metrics/structured/ash_structured_metrics_recorder.cc
index b8fe8aa2..e2e6745a 100644
--- a/chrome/browser/metrics/structured/ash_structured_metrics_recorder.cc
+++ b/chrome/browser/metrics/structured/ash_structured_metrics_recorder.cc
@@ -3,6 +3,7 @@
 // found in the LICENSE file.
 
 #include "chrome/browser/metrics/structured/ash_structured_metrics_recorder.h"
+#include <memory>
 
 #include "base/threading/sequenced_task_runner_handle.h"
 #include "chrome/browser/ash/crosapi/crosapi_ash.h"
@@ -11,6 +12,7 @@
 #include "components/metrics/structured/event.h"
 #include "components/metrics/structured/histogram_util.h"
 #include "components/metrics/structured/recorder.h"
+#include "components/metrics/structured/structured_metrics_features.h"
 
 namespace metrics {
 namespace structured {
@@ -22,13 +24,21 @@
   DCHECK(!is_initialized_);
 
   // If already initialized, do nothing.
-  if (is_initialized_)
+  if (is_initialized_) {
     return;
+  }
 
   // Crosapi may not be initialized, in which case a pipe cannot be setup.
   if (crosapi::CrosapiManager::IsInitialized()) {
     crosapi::CrosapiManager::Get()->crosapi_ash()->BindStructuredMetricsService(
         remote_.BindNewPipeAndPassReceiver());
+
+    if (base::FeatureList::IsEnabled(kEventSequenceLogging)) {
+      auto* user_manager = user_manager::UserManager::Get();
+      DCHECK(user_manager);
+      user_session_observer_ =
+          std::make_unique<StructuredMetricsUserSessionObserver>(user_manager);
+    }
     is_initialized_ = true;
   } else {
     VLOG(2) << "Initialize() called before CrosApi is initialized.";
diff --git a/chrome/browser/metrics/structured/ash_structured_metrics_recorder.h b/chrome/browser/metrics/structured/ash_structured_metrics_recorder.h
index 411bcca..360c633 100644
--- a/chrome/browser/metrics/structured/ash_structured_metrics_recorder.h
+++ b/chrome/browser/metrics/structured/ash_structured_metrics_recorder.h
@@ -5,6 +5,7 @@
 #ifndef CHROME_BROWSER_METRICS_STRUCTURED_ASH_STRUCTURED_METRICS_RECORDER_H_
 #define CHROME_BROWSER_METRICS_STRUCTURED_ASH_STRUCTURED_METRICS_RECORDER_H_
 
+#include "chrome/browser/metrics/structured/structured_metrics_user_session_observer.h"
 #include "chromeos/crosapi/mojom/structured_metrics_service.mojom.h"
 #include "components/metrics/structured/event.h"
 #include "components/metrics/structured/structured_metrics_client.h"
@@ -38,7 +39,7 @@
 
  private:
   mojo::Remote<crosapi::mojom::StructuredMetricsService> remote_;
-
+  std::unique_ptr<StructuredMetricsUserSessionObserver> user_session_observer_;
   bool is_initialized_ = false;
 };
 
diff --git a/chrome/browser/metrics/structured/structured_metrics_user_session_observer.cc b/chrome/browser/metrics/structured/structured_metrics_user_session_observer.cc
new file mode 100644
index 0000000..f76afbdb
--- /dev/null
+++ b/chrome/browser/metrics/structured/structured_metrics_user_session_observer.cc
@@ -0,0 +1,27 @@
+// Copyright 2022 The Chromium Authors
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include "chrome/browser/metrics/structured/structured_metrics_user_session_observer.h"
+#include "components/metrics/structured/structured_events.h"
+
+namespace metrics::structured {
+
+StructuredMetricsUserSessionObserver::StructuredMetricsUserSessionObserver(
+    user_manager::UserManager* user_manager)
+    : user_manager_(user_manager) {
+  user_manager_->AddSessionStateObserver(this);
+}
+
+StructuredMetricsUserSessionObserver::~StructuredMetricsUserSessionObserver() {
+  user_manager_->RemoveSessionStateObserver(this);
+}
+
+void StructuredMetricsUserSessionObserver::ActiveUserChanged(
+    user_manager::User* active_user) {
+  if (active_user->is_active()) {
+    events::v2::cr_os_events::UserLogin().Record();
+  }
+}
+
+}  // namespace metrics::structured
diff --git a/chrome/browser/metrics/structured/structured_metrics_user_session_observer.h b/chrome/browser/metrics/structured/structured_metrics_user_session_observer.h
new file mode 100644
index 0000000..f68abcc
--- /dev/null
+++ b/chrome/browser/metrics/structured/structured_metrics_user_session_observer.h
@@ -0,0 +1,31 @@
+// Copyright 2022 The Chromium Authors
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#ifndef CHROME_BROWSER_METRICS_STRUCTURED_STRUCTURED_METRICS_USER_SESSION_OBSERVER_H_
+#define CHROME_BROWSER_METRICS_STRUCTURED_STRUCTURED_METRICS_USER_SESSION_OBSERVER_H_
+
+#include "components/user_manager/user_manager.h"
+
+namespace metrics::structured {
+
+// An UserSession observer to detect a login for an event to be recorded
+// when such a user action takes place.
+class StructuredMetricsUserSessionObserver
+    : public user_manager::UserManager::UserSessionStateObserver {
+ public:
+  explicit StructuredMetricsUserSessionObserver(
+      user_manager::UserManager* user_manager);
+
+  ~StructuredMetricsUserSessionObserver() override;
+
+  // user_manager::UserSessionStateObserver:
+  void ActiveUserChanged(user_manager::User* active_user) override;
+
+ private:
+  user_manager::UserManager* user_manager_;
+};
+
+}  // namespace metrics::structured
+
+#endif  // CHROME_BROWSER_METRICS_STRUCTURED_STRUCTURED_METRICS_USER_SESSION_OBSERVER_H_
diff --git a/chrome/browser/policy/configuration_policy_handler_list_factory.cc b/chrome/browser/policy/configuration_policy_handler_list_factory.cc
index 869ba08..046ecfe 100644
--- a/chrome/browser/policy/configuration_policy_handler_list_factory.cc
+++ b/chrome/browser/policy/configuration_policy_handler_list_factory.cc
@@ -20,6 +20,7 @@
 #include "build/chromeos_buildflags.h"
 #include "chrome/browser/browsing_data/browsing_data_lifetime_policy_handler.h"
 #include "chrome/browser/enterprise/connectors/device_trust/prefs.h"
+#include "chrome/browser/enterprise/idle/idle_timeout_policy_handler.h"
 #include "chrome/browser/first_party_sets/first_party_sets_overrides_policy_handler.h"
 #include "chrome/browser/net/disk_cache_dir_policy_handler.h"
 #include "chrome/browser/net/explicitly_allowed_network_ports_policy_handler.h"
@@ -161,6 +162,7 @@
 #include "chrome/browser/ash/plugin_vm/plugin_vm_pref_names.h"
 #include "chrome/browser/ash/policy/handlers/configuration_policy_handler_ash.h"
 #include "chrome/browser/ash/policy/handlers/lacros_availability_policy_handler.h"
+#include "chrome/browser/ash/policy/handlers/lacros_selection_policy_handler.h"
 #include "chrome/browser/nearby_sharing/common/nearby_share_prefs.h"
 #include "chrome/browser/policy/default_geolocation_policy_handler.h"
 #include "chrome/browser/policy/os_color_mode_policy_handler.h"
@@ -1826,22 +1828,6 @@
 }
 #endif  // BUILDFLAG(ENABLE_EXTENSIONS)
 
-#if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_WIN) || BUILDFLAG(IS_MAC)
-void GetIdleTimeoutActionsMap(
-    std::vector<std::unique_ptr<StringMappingListPolicyHandler::MappingEntry>>*
-        result) {
-  // Mapping from IdleTimeoutActions action names to ActionType.
-  for (size_t i = 0; i < enterprise_idle::kActionTypeMapSize; i++) {
-    const enterprise_idle::ActionTypeMapEntry& entry =
-        enterprise_idle::kActionTypeMap[i];
-    result->push_back(
-        std::make_unique<StringMappingListPolicyHandler::MappingEntry>(
-            entry.name, std::make_unique<base::Value>(
-                            static_cast<int>(entry.action_type))));
-  }
-}
-#endif  // BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_WIN) || BUILDFLAG(IS_MAC)
-
 // Future policies are not supported on Stable and Beta by default.
 bool AreFuturePoliciesSupported() {
   // Enable future policies for branded browser tests.
@@ -2032,11 +2018,11 @@
       SimpleSchemaValidatingPolicyHandler::MANDATORY_ALLOWED));
 
 #if BUILDFLAG(IS_WIN) || BUILDFLAG(IS_MAC) || BUILDFLAG(IS_LINUX)
-  handlers->AddHandler(std::make_unique<IntRangePolicyHandler>(
-      key::kIdleTimeout, prefs::kIdleTimeout, 5, INT_MAX, /*clamp=*/true));
-  handlers->AddHandler(std::make_unique<StringMappingListPolicyHandler>(
-      key::kIdleTimeoutActions, prefs::kIdleTimeoutActions,
-      base::BindRepeating(&GetIdleTimeoutActionsMap)));
+  handlers->AddHandler(
+      std::make_unique<enterprise_idle::IdleTimeoutPolicyHandler>());
+  handlers->AddHandler(
+      std::make_unique<enterprise_idle::IdleTimeoutActionsPolicyHandler>(
+          chrome_schema));
 #endif  // BUILDFLAG(IS_WIN) || BUILDFLAG(IS_MAC) || BUILDFLAG(IS_LINUX)
 
   handlers->AddHandler(std::make_unique<RestoreOnStartupPolicyHandler>());
@@ -2408,6 +2394,7 @@
   handlers->AddHandler(std::make_unique<BooleanDisablingPolicyHandler>(
       key::kNearbyShareAllowed, prefs::kNearbySharingEnabledPrefName));
   handlers->AddHandler(std::make_unique<LacrosAvailabilityPolicyHandler>());
+  handlers->AddHandler(std::make_unique<LacrosSelectionPolicyHandler>());
   handlers->AddHandler(std::make_unique<BooleanDisablingPolicyHandler>(
       key::kFastPairEnabled, ash::prefs::kFastPairEnabled));
   handlers->AddHandler(std::make_unique<arc::ArcPolicyHandler>());
diff --git a/chrome/browser/profiles/profile_impl.cc b/chrome/browser/profiles/profile_impl.cc
index 0e17f238..a2a12efa 100644
--- a/chrome/browser/profiles/profile_impl.cc
+++ b/chrome/browser/profiles/profile_impl.cc
@@ -657,6 +657,7 @@
         policy::PolicyNamespace(policy::POLICY_DOMAIN_CHROME, std::string()));
     crosapi::browser_util::CacheLacrosAvailability(map);
     crosapi::browser_util::CacheLacrosDataBackwardMigrationMode(map);
+    crosapi::browser_util::CacheLacrosSelection(map);
   }
 #endif
 }
@@ -1183,6 +1184,7 @@
           policy::PolicyNamespace(policy::POLICY_DOMAIN_CHROME, std::string()));
       crosapi::browser_util::CacheLacrosAvailability(map);
       crosapi::browser_util::CacheLacrosDataBackwardMigrationMode(map);
+      crosapi::browser_util::CacheLacrosSelection(map);
     }
 
     ash::UserSessionManager::GetInstance()->RespectLocalePreferenceWrapper(
diff --git a/chrome/browser/reading_list/OWNERS b/chrome/browser/reading_list/OWNERS
index b4a76a2b..adf8b80d 100644
--- a/chrome/browser/reading_list/OWNERS
+++ b/chrome/browser/reading_list/OWNERS
@@ -1,6 +1,6 @@
-dtrainor@chromium.org
-shaktisahu@chromium.org
+wylieb@chromium.org
 
 # Backup reviewers:
 twellington@chromium.org
-wylieb@chromium.org
+dtrainor@chromium.org
+shaktisahu@chromium.org
\ No newline at end of file
diff --git a/chrome/browser/resources/chromeos/accessibility/chromevox/background/output/output.js b/chrome/browser/resources/chromeos/accessibility/chromevox/background/output/output.js
index dcaf21cf..52a083f3 100644
--- a/chrome/browser/resources/chromeos/accessibility/chromevox/background/output/output.js
+++ b/chrome/browser/resources/chromeos/accessibility/chromevox/background/output/output.js
@@ -672,38 +672,6 @@
   }
 
   /** @override */
-  formatCellIndexText_(data, token, options) {
-    const buff = data.outputBuffer;
-    const node = data.node;
-    const formatLog = data.outputFormatLogger;
-
-    if (node.htmlAttributes['aria-coltext']) {
-      let value = node.htmlAttributes['aria-coltext'];
-      let row = node;
-      while (row && row.role !== RoleType.ROW) {
-        row = row.parent;
-      }
-      if (!row || !row.htmlAttributes['aria-rowtext']) {
-        return;
-      }
-      value += row.htmlAttributes['aria-rowtext'];
-      this.append_(buff, value, options);
-      formatLog.writeTokenWithValue(token, value);
-    } else {
-      formatLog.write(token);
-      this.format_({
-        node,
-        outputFormat: ` @cell_summary($if($tableCellAriaRowIndex,
-                  $tableCellAriaRowIndex, $tableCellRowIndex),
-                $if($tableCellAriaColumnIndex, $tableCellAriaColumnIndex,
-                  $tableCellColumnIndex))`,
-        outputBuffer: buff,
-        outputFormatLogger: formatLog,
-      });
-    }
-  }
-
-  /** @override */
   formatNode_(data, token, tree, options) {
     const buff = data.outputBuffer;
     const node = data.node;
diff --git a/chrome/browser/resources/chromeos/accessibility/chromevox/background/output/output_formatter.js b/chrome/browser/resources/chromeos/accessibility/chromevox/background/output/output_formatter.js
index cfa54678..9031378d 100644
--- a/chrome/browser/resources/chromeos/accessibility/chromevox/background/output/output_formatter.js
+++ b/chrome/browser/resources/chromeos/accessibility/chromevox/background/output/output_formatter.js
@@ -97,7 +97,7 @@
         token === 'tableCellRowIndex' || token === 'tableCellColumnIndex') {
       this.formatTableCellIndex_(this.params_, token, options);
     } else if (token === 'cellIndexText') {
-      this.output_.formatCellIndexText_(this.params_, token, options);
+      this.formatCellIndexText_(this.params_, token, options);
     } else if (token === 'node') {
       this.output_.formatNode_(this.params_, token, tree, options);
     } else if (token === 'nameOrTextContent' || token === 'textContent') {
@@ -184,6 +184,43 @@
   /**
    * @param {!outputTypes.OutputFormattingData} data
    * @param {string} token
+   * @param {!{annotation: Array<*>, isUnique: (boolean|undefined)}} options
+   * @private
+   */
+  formatCellIndexText_(data, token, options) {
+    const buff = data.outputBuffer;
+    const node = data.node;
+    const formatLog = data.outputFormatLogger;
+
+    if (node.htmlAttributes['aria-coltext']) {
+      let value = node.htmlAttributes['aria-coltext'];
+      let row = node;
+      while (row && row.role !== RoleType.ROW) {
+        row = row.parent;
+      }
+      if (!row || !row.htmlAttributes['aria-rowtext']) {
+        return;
+      }
+      value += row.htmlAttributes['aria-rowtext'];
+      this.output_.append_(buff, value, options);
+      formatLog.writeTokenWithValue(token, value);
+    } else {
+      formatLog.write(token);
+      this.output_.format_({
+        node,
+        outputFormat: ` @cell_summary($if($tableCellAriaRowIndex,
+                  $tableCellAriaRowIndex, $tableCellRowIndex),
+                $if($tableCellAriaColumnIndex, $tableCellAriaColumnIndex,
+                  $tableCellColumnIndex))`,
+        outputBuffer: buff,
+        outputFormatLogger: formatLog,
+      });
+    }
+  }
+
+  /**
+   * @param {!outputTypes.OutputFormattingData} data
+   * @param {string} token
    * @private
    */
   formatChecked_(data, token) {
diff --git a/chrome/browser/resources/chromeos/accessibility/chromevox/background/output/output_interface.js b/chrome/browser/resources/chromeos/accessibility/chromevox/background/output/output_interface.js
index be41ead..7bae0e6 100644
--- a/chrome/browser/resources/chromeos/accessibility/chromevox/background/output/output_interface.js
+++ b/chrome/browser/resources/chromeos/accessibility/chromevox/background/output/output_interface.js
@@ -65,13 +65,6 @@
   /**
    * @param {!outputTypes.OutputFormattingData} data
    * @param {string} token
-   * @param {!{annotation: Array<*>, isUnique: (boolean|undefined)}} options
-   */
-  formatCellIndexText_(data, token, options) {}
-
-  /**
-   * @param {!outputTypes.OutputFormattingData} data
-   * @param {string} token
    * @param {!OutputFormatTree} tree
    * @param {!{annotation: Array<*>, isUnique: (boolean|undefined)}} options
    */
diff --git a/chrome/browser/resources/extensions/service.ts b/chrome/browser/resources/extensions/service.ts
index 9fbe8e71..d6329613 100644
--- a/chrome/browser/resources/extensions/service.ts
+++ b/chrome/browser/resources/extensions/service.ts
@@ -138,11 +138,10 @@
           populateError: true,
         },
         extraOptions);
-
     return chrome.developerPrivate.loadUnpacked(options)
         .then(loadError => {
           if (loadError) {
-            throw new Error(loadError.error);
+            throw loadError;
           }
           // The load was successful if there's no loadError.
           return true;
diff --git a/chrome/browser/resources/management/management_ui.ts b/chrome/browser/resources/management/management_ui.ts
index 767e7135..cc01578e 100644
--- a/chrome/browser/resources/management/management_ui.ts
+++ b/chrome/browser/resources/management/management_ui.ts
@@ -379,8 +379,8 @@
       }
       // </if>
       // <if expr="not chromeos_ash">
-      this.managementNoticeHtml_ =
-          sanitizeInnerHtml(data.browserManagementNotice);
+      this.managementNoticeHtml_ = sanitizeInnerHtml(
+          data.browserManagementNotice, {attrs: ['aria-label']});
       // </if>
     });
   }
diff --git a/chrome/browser/resources/settings/chromeos/internet_page/BUILD.gn b/chrome/browser/resources/settings/chromeos/internet_page/BUILD.gn
index 6ed68ea..acec84f 100644
--- a/chrome/browser/resources/settings/chromeos/internet_page/BUILD.gn
+++ b/chrome/browser/resources/settings/chromeos/internet_page/BUILD.gn
@@ -20,7 +20,6 @@
     ":internet_known_networks_page",
     ":internet_page",
     ":internet_page_browser_proxy",
-    ":internet_shared_css",
     ":internet_subpage",
     ":network_always_on_vpn",
     ":network_proxy_section",
@@ -60,7 +59,6 @@
 
 js_library("internet_config") {
   deps = [
-    ":internet_shared_css",
     "//ash/webui/common/resources:i18n_behavior",
     "//ash/webui/common/resources:util",
     "//ash/webui/common/resources/network:network_config",
@@ -99,7 +97,6 @@
 js_library("internet_detail_page") {
   deps = [
     ":internet_page_browser_proxy",
-    ":internet_shared_css",
     ":network_proxy_section",
     "//ash/webui/common/resources:assert",
     "//ash/webui/common/resources:i18n_behavior",
@@ -140,7 +137,6 @@
 
 js_library("internet_known_networks_page") {
   deps = [
-    ":internet_shared_css",
     "//ash/webui/common/resources:assert",
     "//ash/webui/common/resources:i18n_behavior",
     "//ash/webui/common/resources/network:cr_policy_network_behavior_mojo",
@@ -195,9 +191,6 @@
 js_library("internet_page_browser_proxy") {
 }
 
-js_library("internet_shared_css") {
-}
-
 js_library("internet_subpage") {
   deps = [
     ":internet_page_browser_proxy",
@@ -233,7 +226,6 @@
 
 js_library("network_always_on_vpn") {
   deps = [
-    ":internet_shared_css",
     "//ash/webui/common/resources:assert",
     "//ash/webui/common/resources:i18n_behavior",
     "//ash/webui/common/resources/network:onc_mojo",
@@ -243,7 +235,6 @@
 
 js_library("network_proxy_section") {
   deps = [
-    ":internet_shared_css",
     "//ash/webui/common/resources:i18n_behavior",
     "//ash/webui/common/resources/network:cr_policy_network_behavior_mojo",
     "//ash/webui/common/resources/network:cr_policy_network_indicator_mojo",
@@ -302,7 +293,6 @@
     "internet_detail_page.js",
     "internet_known_networks_page.js",
     "internet_page.js",
-    "internet_shared_css.js",
     "internet_subpage.js",
     "network_always_on_vpn.js",
     "network_proxy_section.js",
diff --git a/chrome/browser/resources/settings/chromeos/internet_page/apn_subpage.js b/chrome/browser/resources/settings/chromeos/internet_page/apn_subpage.js
index 12e1ad2..947cd72 100644
--- a/chrome/browser/resources/settings/chromeos/internet_page/apn_subpage.js
+++ b/chrome/browser/resources/settings/chromeos/internet_page/apn_subpage.js
@@ -7,7 +7,7 @@
  * Settings subpage for managing APNs.
  */
 
-import './internet_shared_css.js';
+import './internet_shared.css.js';
 import 'chrome://resources/ash/common/network/apn_list.js';
 
 import {assert} from 'chrome://resources/ash/common/assert.js';
@@ -232,4 +232,4 @@
   }
 }
 
-customElements.define(ApnSubpageElement.is, ApnSubpageElement);
\ No newline at end of file
+customElements.define(ApnSubpageElement.is, ApnSubpageElement);
diff --git a/chrome/browser/resources/settings/chromeos/internet_page/hotspot_subpage.js b/chrome/browser/resources/settings/chromeos/internet_page/hotspot_subpage.js
index 973d495..e01b0a0 100644
--- a/chrome/browser/resources/settings/chromeos/internet_page/hotspot_subpage.js
+++ b/chrome/browser/resources/settings/chromeos/internet_page/hotspot_subpage.js
@@ -7,7 +7,7 @@
  * Settings subpage for managing Hotspot.
  */
 
-import './internet_shared_css.js';
+import './internet_shared.css.js';
 
 import {html, PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js';
 
@@ -23,4 +23,4 @@
 }
 
 customElements.define(
-    SettingsHotspotSubpageElement.is, SettingsHotspotSubpageElement);
\ No newline at end of file
+    SettingsHotspotSubpageElement.is, SettingsHotspotSubpageElement);
diff --git a/chrome/browser/resources/settings/chromeos/internet_page/internet_config.js b/chrome/browser/resources/settings/chromeos/internet_page/internet_config.js
index 571376c..23c39dd 100644
--- a/chrome/browser/resources/settings/chromeos/internet_page/internet_config.js
+++ b/chrome/browser/resources/settings/chromeos/internet_page/internet_config.js
@@ -10,11 +10,11 @@
 import 'chrome://resources/cr_elements/cr_button/cr_button.js';
 import 'chrome://resources/cr_elements/cr_dialog/cr_dialog.js';
 import 'chrome://resources/polymer/v3_0/iron-flex-layout/iron-flex-layout-classes.js';
-import './internet_shared_css.js';
+import './internet_shared.css.js';
 
-import {OncMojo} from 'chrome://resources/ash/common/network/onc_mojo.js';
 import {I18nBehavior, I18nBehaviorInterface} from 'chrome://resources/ash/common/i18n_behavior.js';
 import {loadTimeData} from 'chrome://resources/ash/common/load_time_data.m.js';
+import {OncMojo} from 'chrome://resources/ash/common/network/onc_mojo.js';
 import {HTMLEscape} from 'chrome://resources/ash/common/util.js';
 import {NetworkType} from 'chrome://resources/mojo/chromeos/services/network_config/public/mojom/network_types.mojom-webui.js';
 import {html, mixinBehaviors, PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js';
diff --git a/chrome/browser/resources/settings/chromeos/internet_page/internet_detail_page.js b/chrome/browser/resources/settings/chromeos/internet_page/internet_detail_page.js
index 5b2d084..93c2fcd 100644
--- a/chrome/browser/resources/settings/chromeos/internet_page/internet_detail_page.js
+++ b/chrome/browser/resources/settings/chromeos/internet_page/internet_detail_page.js
@@ -30,20 +30,20 @@
 import '../../controls/settings_toggle_button.js';
 import '../../prefs/prefs.js';
 import './cellular_roaming_toggle_button.js';
-import './internet_shared_css.js';
+import './internet_shared.css.js';
 import './network_proxy_section.js';
 import './settings_traffic_counters.js';
 import './tether_connection_dialog.js';
 
+import {assert} from 'chrome://resources/ash/common/assert.js';
 import {I18nBehavior, I18nBehaviorInterface} from 'chrome://resources/ash/common/i18n_behavior.js';
+import {loadTimeData} from 'chrome://resources/ash/common/load_time_data.m.js';
 import {isActiveSim, processDeviceState} from 'chrome://resources/ash/common/network/cellular_utils.js';
 import {CrPolicyNetworkBehaviorMojo, CrPolicyNetworkBehaviorMojoInterface} from 'chrome://resources/ash/common/network/cr_policy_network_behavior_mojo.js';
 import {MojoInterfaceProvider, MojoInterfaceProviderImpl} from 'chrome://resources/ash/common/network/mojo_interface_provider.js';
 import {NetworkListenerBehavior, NetworkListenerBehaviorInterface} from 'chrome://resources/ash/common/network/network_listener_behavior.js';
 import {OncMojo} from 'chrome://resources/ash/common/network/onc_mojo.js';
 import {WebUIListenerBehavior, WebUIListenerBehaviorInterface} from 'chrome://resources/ash/common/web_ui_listener_behavior.js';
-import {assert} from 'chrome://resources/ash/common/assert.js';
-import {loadTimeData} from 'chrome://resources/ash/common/load_time_data.m.js';
 import {ActivationStateType, ApnProperties, ConfigProperties, CrosNetworkConfigRemote, FilterType, GlobalPolicy, HiddenSsidMode, IPConfigProperties, ManagedProperties, NetworkStateProperties, NO_LIMIT, ProxySettings, SecurityType, VpnType} from 'chrome://resources/mojo/chromeos/services/network_config/public/mojom/cros_network_config.mojom-webui.js';
 import {ConnectionStateType, DeviceStateType, IPConfigType, NetworkType, OncSource, PolicySource, PortalState} from 'chrome://resources/mojo/chromeos/services/network_config/public/mojom/network_types.mojom-webui.js';
 import {afterNextRender, flush, html, mixinBehaviors, PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js';
diff --git a/chrome/browser/resources/settings/chromeos/internet_page/internet_known_networks_page.js b/chrome/browser/resources/settings/chromeos/internet_page/internet_known_networks_page.js
index ad6184d0..51a6510 100644
--- a/chrome/browser/resources/settings/chromeos/internet_page/internet_known_networks_page.js
+++ b/chrome/browser/resources/settings/chromeos/internet_page/internet_known_networks_page.js
@@ -12,24 +12,24 @@
 import 'chrome://resources/cr_elements/cr_link_row/cr_link_row.js';
 import 'chrome://resources/cr_elements/icons.html.js';
 import '../../settings_shared.css.js';
-import './internet_shared_css.js';
+import './internet_shared.css.js';
 
+import {assert} from 'chrome://resources/ash/common/assert.js';
+import {I18nBehavior, I18nBehaviorInterface} from 'chrome://resources/ash/common/i18n_behavior.js';
 import {CrPolicyNetworkBehaviorMojo, CrPolicyNetworkBehaviorMojoInterface} from 'chrome://resources/ash/common/network/cr_policy_network_behavior_mojo.js';
 import {MojoInterfaceProvider, MojoInterfaceProviderImpl} from 'chrome://resources/ash/common/network/mojo_interface_provider.js';
 import {NetworkListenerBehavior, NetworkListenerBehaviorInterface} from 'chrome://resources/ash/common/network/network_listener_behavior.js';
 import {OncMojo} from 'chrome://resources/ash/common/network/onc_mojo.js';
-import {I18nBehavior, I18nBehaviorInterface} from 'chrome://resources/ash/common/i18n_behavior.js';
-import {assert} from 'chrome://resources/ash/common/assert.js';
 import {ConfigProperties, CrosNetworkConfigRemote, FilterType, NO_LIMIT} from 'chrome://resources/mojo/chromeos/services/network_config/public/mojom/cros_network_config.mojom-webui.js';
 import {NetworkType} from 'chrome://resources/mojo/chromeos/services/network_config/public/mojom/network_types.mojom-webui.js';
 import {html, mixinBehaviors, PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js';
 
 import {Setting} from '../../mojom-webui/setting.mojom-webui.js';
-import {Route} from '../router.js';
 import {DeepLinkingBehavior, DeepLinkingBehaviorInterface} from '../deep_linking_behavior.js';
 import {recordSettingChange} from '../metrics_recorder.js';
 import {routes} from '../os_route.js';
 import {RouteObserverBehavior, RouteObserverBehaviorInterface} from '../route_observer_behavior.js';
+import {Route} from '../router.js';
 
 /**
  * @constructor
diff --git a/chrome/browser/resources/settings/chromeos/internet_page/internet_shared.css b/chrome/browser/resources/settings/chromeos/internet_page/internet_shared.css
new file mode 100644
index 0000000..e6948c0
--- /dev/null
+++ b/chrome/browser/resources/settings/chromeos/internet_page/internet_shared.css
@@ -0,0 +1,31 @@
+/* Copyright 2022 The Chromium Authors
+ * Use of this source code is governed by a BSD-style license that can be
+ * found in the LICENSE file.
+ */
+
+/* #css_wrapper_metadata_start
+ * #type=style
+ * #import=chrome://resources/cr_elements/cr_shared_vars.css.js
+ * #include=settings-shared
+ * #css_wrapper_metadata_end */
+
+network-icon {
+  padding-inline-end: var(--cr-section-padding);
+}
+
+iron-icon.policy {
+  margin-inline-end: var(--cr-controlled-by-spacing);
+}
+
+.indented {
+  margin-inline-start: var(--cr-section-padding);
+}
+
+.stretch {
+  align-items: stretch;
+}
+
+.title {
+  font-size: 107.69%;  /* 14px / 13px */
+  font-weight: 500;
+}
diff --git a/chrome/browser/resources/settings/chromeos/internet_page/internet_shared_css.html b/chrome/browser/resources/settings/chromeos/internet_page/internet_shared_css.html
deleted file mode 100644
index f56ed93..0000000
--- a/chrome/browser/resources/settings/chromeos/internet_page/internet_shared_css.html
+++ /dev/null
@@ -1,25 +0,0 @@
-
-<template>
-  <style include="settings-shared">
-    network-icon {
-      padding-inline-end: var(--cr-section-padding);
-    }
-
-    iron-icon.policy {
-      margin-inline-end: var(--cr-controlled-by-spacing);
-    }
-
-    .indented {
-      margin-inline-start: var(--cr-section-padding);
-    }
-
-    .stretch {
-      align-items: stretch;
-    }
-
-    .title {
-      font-size: 107.69%;  /* 14px / 13px */
-      font-weight: 500;
-    }
-  </style>
-</template>
diff --git a/chrome/browser/resources/settings/chromeos/internet_page/internet_shared_css.js b/chrome/browser/resources/settings/chromeos/internet_page/internet_shared_css.js
deleted file mode 100644
index bbfeab4c..0000000
--- a/chrome/browser/resources/settings/chromeos/internet_page/internet_shared_css.js
+++ /dev/null
@@ -1,11 +0,0 @@
-// Copyright 2022 The Chromium Authors
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-import 'chrome://resources/cr_elements/cr_shared_vars.css.js';
-import '../../settings_shared.css.js';
-const template = document.createElement('template');
-template.innerHTML = `
-<dom-module id="internet-shared" assetpath="chrome://resources/">{__html_template__}</dom-module>
-`;
-document.body.appendChild(template.content.cloneNode(true));
diff --git a/chrome/browser/resources/settings/chromeos/internet_page/network_proxy_section.js b/chrome/browser/resources/settings/chromeos/internet_page/network_proxy_section.js
index e8d97ce..a9cec8b 100644
--- a/chrome/browser/resources/settings/chromeos/internet_page/network_proxy_section.js
+++ b/chrome/browser/resources/settings/chromeos/internet_page/network_proxy_section.js
@@ -16,13 +16,13 @@
 import 'chrome://resources/polymer/v3_0/iron-flex-layout/iron-flex-layout-classes.js';
 import '../../controls/extension_controlled_indicator.js';
 import '../../settings_vars.css.js';
-import './internet_shared_css.js';
+import './internet_shared.css.js';
 import '../../controls/settings_toggle_button.js';
 import 'chrome://resources/cr_elements/cr_toggle/cr_toggle.js';
 
+import {I18nBehavior, I18nBehaviorInterface} from 'chrome://resources/ash/common/i18n_behavior.js';
 import {CrPolicyNetworkBehaviorMojo, CrPolicyNetworkBehaviorMojoInterface} from 'chrome://resources/ash/common/network/cr_policy_network_behavior_mojo.js';
 import {OncMojo} from 'chrome://resources/ash/common/network/onc_mojo.js';
-import {I18nBehavior, I18nBehaviorInterface} from 'chrome://resources/ash/common/i18n_behavior.js';
 import {ManagedProperties, ManagedString} from 'chrome://resources/mojo/chromeos/services/network_config/public/mojom/cros_network_config.mojom-webui.js';
 import {OncSource} from 'chrome://resources/mojo/chromeos/services/network_config/public/mojom/network_types.mojom-webui.js';
 import {html, mixinBehaviors, PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js';
diff --git a/chrome/browser/resources/settings/chromeos/internet_page/settings_traffic_counters.ts b/chrome/browser/resources/settings/chromeos/internet_page/settings_traffic_counters.ts
index 7091634..0940206 100644
--- a/chrome/browser/resources/settings/chromeos/internet_page/settings_traffic_counters.ts
+++ b/chrome/browser/resources/settings/chromeos/internet_page/settings_traffic_counters.ts
@@ -7,7 +7,7 @@
  * Settings UI.
  */
 
-import './internet_shared_css.js';
+import './internet_shared.css.js';
 import 'chrome://resources/ash/common/network/network_shared.css.js';
 import 'chrome://resources/cr_elements/cr_button/cr_button.js';
 import 'chrome://resources/cr_elements/cr_toggle/cr_toggle.js';
diff --git a/chrome/browser/resources/settings/chromeos/os_a11y_page/text_to_speech_page.ts b/chrome/browser/resources/settings/chromeos/os_a11y_page/text_to_speech_page.ts
index c87d29b..095ec0f 100644
--- a/chrome/browser/resources/settings/chromeos/os_a11y_page/text_to_speech_page.ts
+++ b/chrome/browser/resources/settings/chromeos/os_a11y_page/text_to_speech_page.ts
@@ -18,6 +18,7 @@
 import {mixinBehaviors, PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js';
 
 import {Setting} from '../../mojom-webui/setting.mojom-webui.js';
+import {PrefsMixin, PrefsMixinInterface} from '../../prefs/prefs_mixin.js';
 import {DeepLinkingBehavior, DeepLinkingBehaviorInterface} from '../deep_linking_behavior.js';
 import {DevicePageBrowserProxy, DevicePageBrowserProxyImpl} from '../device_page/device_page_browser_proxy.js';
 import {routes} from '../os_route.js';
@@ -32,10 +33,11 @@
         [
           DeepLinkingBehavior,
         ],
-        RouteOriginMixin(WebUiListenerMixin(I18nMixin(PolymerElement)))) as {
+        RouteOriginMixin(
+            PrefsMixin(WebUiListenerMixin(I18nMixin(PolymerElement))))) as {
       new (): PolymerElement & I18nMixinInterface &
-          WebUiListenerMixinInterface & RouteOriginMixinInterface &
-          DeepLinkingBehaviorInterface,
+          WebUiListenerMixinInterface & PrefsMixinInterface &
+          RouteOriginMixinInterface & DeepLinkingBehaviorInterface,
     };
 
 class SettingsTextToSpeechPageElement extends
@@ -51,14 +53,6 @@
   static get properties() {
     return {
       /**
-       * Preferences state.
-       */
-      prefs: {
-        type: Object,
-        notify: true,
-      },
-
-      /**
        * |hasKeyboard_| starts undefined so observer doesn't trigger until it
        * has been populated.
        */
@@ -85,7 +79,6 @@
     };
   }
 
-  prefs: {[key: string]: any};
   private deviceBrowserProxy_: DevicePageBrowserProxy;
   private hasKeyboard_: boolean;
   private route_: Route;
diff --git a/chrome/browser/resources/settings/chromeos/os_settings.gni b/chrome/browser/resources/settings/chromeos/os_settings.gni
index ffd35abc..278f7bd 100644
--- a/chrome/browser/resources/settings/chromeos/os_settings.gni
+++ b/chrome/browser/resources/settings/chromeos/os_settings.gni
@@ -204,6 +204,7 @@
 
 # Files that are passed as input to css_to_wrapper().
 css_files = [
+  "chromeos/internet_page/internet_shared.css",
   "chromeos/os_apps_page/app_management_page/app_management_cros_shared_style.css",
   "chromeos/os_apps_page/app_management_page/app_management_cros_shared_vars.css",
   "chromeos/os_languages_page/shared_style.css",
@@ -353,7 +354,6 @@
   "chromeos/internet_page/internet_detail_page.js",
   "chromeos/internet_page/internet_known_networks_page.js",
   "chromeos/internet_page/internet_page.js",
-  "chromeos/internet_page/internet_shared_css.js",
   "chromeos/internet_page/internet_subpage.js",
   "chromeos/internet_page/network_always_on_vpn.js",
   "chromeos/internet_page/network_proxy_section.js",
diff --git a/chrome/browser/resources/side_panel/customize_chrome/color.html b/chrome/browser/resources/side_panel/customize_chrome/color.html
index be49563..bfb971f 100644
--- a/chrome/browser/resources/side_panel/customize_chrome/color.html
+++ b/chrome/browser/resources/side_panel/customize_chrome/color.html
@@ -1,32 +1,51 @@
 <style>
   :host {
+    align-items: center;
     cursor: pointer;
+    display: flex;
+    justify-content: center;
   }
 
   :host,
   svg {
-    border-radius: 25px;
-    display: block;
     height: 50px;
     width: 50px;
   }
 
+  :host([checked]) svg {
+    height: 46px;
+    width: 46px;
+  }
+
+  customize-chrome-check-mark {
+    --customize-chrome-check-mark-wrapper-end: -1px;
+  }
+
+  svg {
+    border-radius: 25px;
+  }
+
   #background {
     fill: var(--customize-chrome-color-background-color);
   }
 
+  :host([checked]) #background {
+    r: 25px;
+  }
+
   #foreground {
     fill: var(--customize-chrome-color-foreground-color);
   }
 </style>
-<!-- TODO(crbug.com/1384229): Allow clicking. -->
-<svg viewBox="0 0 50 50" xmlns="http://www.w3.org/2000/svg"
-    xmlns:xlink="http://www.w3.org/1999/xlink">
-  <circle id="foreground" cx="25" cy="25" r="25">
-  </circle>
-  <clipPath id="bottomHalf">
-    <rect x="1" y="25" width="48" height="24"></rect>
-  </clipPath>
-  <circle id="background" cx="25" cy="25" r="24"
-      clip-path="url(#bottomHalf)"></circle>
-</svg>
+<customize-chrome-check-mark-wrapper checked="[[checked]]">
+  <svg viewBox="0 0 50 50" xmlns="http://www.w3.org/2000/svg"
+      xmlns:xlink="http://www.w3.org/1999/xlink">
+    <circle id="foreground" cx="25" cy="25" r="25">
+    </circle>
+    <clipPath id="bottomHalf">
+      <rect x="1" y="25" width="50" height="25"></rect>
+    </clipPath>
+    <circle id="background" cx="25" cy="25" r="24"
+        clip-path="url(#bottomHalf)"></circle>
+  </svg>
+</customize-chrome-check-mark-wrapper>
diff --git a/chrome/browser/resources/side_panel/customize_chrome/color.ts b/chrome/browser/resources/side_panel/customize_chrome/color.ts
index 83312e8..5b3d943 100644
--- a/chrome/browser/resources/side_panel/customize_chrome/color.ts
+++ b/chrome/browser/resources/side_panel/customize_chrome/color.ts
@@ -2,6 +2,8 @@
 // Use of this source code is governed by a BSD-style license that can be
 // found in the LICENSE file.
 
+import './check_mark_wrapper.js';
+
 import {skColorToRgba} from 'chrome://resources/js/color_utils.js';
 import {SkColor} from 'chrome://resources/mojo/skia/public/mojom/skcolor.mojom-webui.js';
 import {PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js';
@@ -36,11 +38,16 @@
         value: 0,
         observer: 'onColorChange_',
       },
+      checked: {
+        type: Boolean,
+        reflectToAttribute: true,
+      },
     };
   }
 
   public backgroundColor: SkColor;
   public foregroundColor: SkColor;
+  public checked: boolean;
 
   private onColorChange_() {
     this.updateStyles({
diff --git a/chrome/browser/resources/side_panel/customize_chrome/colors.html b/chrome/browser/resources/side_panel/customize_chrome/colors.html
index 6138b869..f88e9dc9 100644
--- a/chrome/browser/resources/side_panel/customize_chrome/colors.html
+++ b/chrome/browser/resources/side_panel/customize_chrome/colors.html
@@ -5,33 +5,84 @@
     justify-content: center;
   }
 
-  customize-chrome-check-mark {
-    --customize-chrome-check-mark-wrapper-end: -1px;
+  #customColorContainer {
+    position: relative;
+  }
+
+  /* colorPicker is placed on top of the theme icon to work around
+     https://crbug.com/1162053. */
+  #colorPicker {
+    border: 0;
+    height: 50px;
+    left: 0;
+    margin: 0;
+    opacity: 0;
+    padding: 0;
+    pointer-events: none;
+    position: absolute;
+    top: 0;
+    width: 50px;
+  }
+
+  #colorPickerIcon {
+    -webkit-mask-image: url(chrome://resources/cr_components/customize_themes/colorize.svg);
+    -webkit-mask-repeat: no-repeat;
+    -webkit-mask-size: 100%;
+    background-color: var(--google-grey-700);
+    height: 20px;
+    left: calc(50% - 10px);
+    pointer-events: none;
+    position: absolute;
+    top: calc(50% - 10px);
+    width: 20px;
   }
 </style>
-<!-- TODO(crbug.com/1384229): Make grid adaptive. -->
-<cr-grid columns="4">
-  <!-- TODO(crbug.com/1384229): Add extracted color. -->
-  <customize-chrome-check-mark-wrapper checked>
-    <customize-chrome-color
-        id="defaultColor"
-        background-color="[[defaultColor_.background]]"
-        foreground-color="[[defaultColor_.foreground]]"
-        title="$i18n{defaultColorName}"
-        on-click="onDefaultColorClick_">
-    </customize-chrome-color>
-  </customize-chrome-check-mark-wrapper>
+<!-- TODO(crbug.com/1395210): Make grid adaptive. -->
+<cr-grid columns="4" role="radiogroup"
+    aria-label="$i18n{colorsContainerLabel}">
+  <!-- TODO(crbug.com/1401018): Add extracted color. -->
+  <customize-chrome-color
+      id="defaultColor"
+      background-color="[[defaultColor_.background]]"
+      foreground-color="[[defaultColor_.foreground]]"
+      title="$i18n{defaultColorName}"
+      aria-label="$i18n{defaultColorName}"
+      role="radio"
+      checked="[[isDefaultColorSelected_]]"
+      aria-checked$="[[isDefaultColorSelected_]]"
+      tabindex$="[[tabIndex_(isDefaultColorSelected_)]]"
+      on-click="onDefaultColorClick_">
+  </customize-chrome-color>
   <template id="chromeColors" is="dom-repeat" items="[[colors_]]">
-    <!-- TODO(crbug.com/1384229): Only set checked if selected. -->
-    <customize-chrome-check-mark-wrapper checked>
-      <customize-chrome-color
-          class="chrome-color"
-          background-color="[[item.background]]"
-          foreground-color="[[item.foreground]]"
-          title="[[item.name]]"
-          on-click="onChromeColorClick_">
-      </customize-chrome-color>
-    </customize-chrome-check-mark-wrapper>
+    <customize-chrome-color
+        class="chrome-color"
+        background-color="[[item.background]]"
+        foreground-color="[[item.foreground]]"
+        title="[[item.name]]"
+        aria-label$="[[item.name]]"
+        role="radio"
+        checked="[[isChromeColorSelected_(item.foreground, theme_, colors_)]]"
+        aria-checked$=
+            "[[isChromeColorSelected_(item.foreground, theme_, colors_)]]"
+        tabindex$="[[chromeColorTabIndex_(item.foreground, theme_, colors_)]]"
+        on-click="onChromeColorClick_">
+    </customize-chrome-color>
   </template>
-  <!-- TODO(crbug.com/1384229): Add color picker. -->
+  <div id="customColorContainer"
+      title="$i18n{colorPickerLabel}"
+      aria-label="$i18n{colorPickerLabel}"
+      role="radio"
+      aria-checked$="[[isCustomColorSelected_]]"
+      tabindex$="[[tabIndex_(isCustomColorSelected_)]]"
+      on-click="onCustomColorClick_">
+    <customize-chrome-color
+        id="customColor"
+        background-color="[[customColor_.background]]"
+        foreground-color="[[customColor_.foreground]]"
+        checked="[[isCustomColorSelected_]]">
+    </customize-chrome-color>
+    <div id="colorPickerIcon"></div>
+    <input id="colorPicker" type="color" tabindex="-1" aria-hidden="true"
+        on-change="onCustomColorChange_">
+  </div>
 </cr-grid>
diff --git a/chrome/browser/resources/side_panel/customize_chrome/colors.ts b/chrome/browser/resources/side_panel/customize_chrome/colors.ts
index 926fd80..ac571877 100644
--- a/chrome/browser/resources/side_panel/customize_chrome/colors.ts
+++ b/chrome/browser/resources/side_panel/customize_chrome/colors.ts
@@ -4,8 +4,8 @@
 
 import 'chrome://resources/cr_elements/cr_grid/cr_grid.js';
 import './color.js';
-import './check_mark_wrapper.js';
 
+import {hexColorToSkColor, skColorToRgba} from 'chrome://resources/js/color_utils.js';
 import {SkColor} from 'chrome://resources/mojo/skia/public/mojom/skcolor.mojom-webui.js';
 import {DomRepeat, PolymerElement} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js';
 
@@ -29,10 +29,20 @@
   foreground: {value: 0xff202124},
 };
 
+function isChromeColorSelected(
+    theme: Theme|undefined, colors: ChromeColor[]|undefined, color: SkColor) {
+  return !!theme && !!colors && !!theme.foregroundColor &&
+      theme.foregroundColor.value === color.value;
+}
+
 export interface ColorsElement {
   $: {
     defaultColor: ColorElement,
     chromeColors: DomRepeat,
+    customColorContainer: HTMLElement,
+    customColor: ColorElement,
+    colorPicker: HTMLInputElement,
+    colorPickerIcon: HTMLElement,
   };
 }
 
@@ -53,11 +63,34 @@
       },
       colors_: Array,
       theme_: Object,
+      isDefaultColorSelected_: {
+        type: Object,
+        computed: 'computeIsDefaultColorSelected_(theme_)',
+      },
+      isCustomColorSelected_: {
+        type: Object,
+        computed: 'computeIsCustomColorSelected_(theme_, color_)',
+      },
+      customColor_: {
+        type: Object,
+        value: {
+          background: {value: 0xffffffff},
+          foreground: {value: 0xfff1f3f4},
+        },
+      },
     };
   }
 
+  static get observers() {
+    return [
+      'updateCustomColor_(colors_, theme_, isCustomColorSelected_)',
+    ];
+  }
+
   private colors_: ChromeColor[];
   private theme_: Theme;
+  private isCustomColorSelected_: boolean;
+  private customColor_: Color;
   private setThemeListenerId_: number|null = null;
 
   constructor() {
@@ -89,6 +122,29 @@
                                         LIGHT_DEFAULT_COLOR;
   }
 
+  private computeIsDefaultColorSelected_(): boolean {
+    return this.theme_ && !this.theme_.foregroundColor;
+  }
+
+  private computeIsCustomColorSelected_(): boolean {
+    return !!this.colors_ && !!this.theme_ && !!this.theme_.foregroundColor &&
+        !this.colors_.find(
+            (color: ChromeColor) =>
+                color.foreground.value === this.theme_.foregroundColor!.value);
+  }
+
+  private isChromeColorSelected_(color: SkColor): boolean {
+    return isChromeColorSelected(this.theme_, this.colors_, color);
+  }
+
+  private tabIndex_(selected: boolean): string {
+    return selected ? '0' : '-1';
+  }
+
+  private chromeColorTabIndex_(color: SkColor): string {
+    return isChromeColorSelected(this.theme_, this.colors_, color) ? '0' : '-1';
+  }
+
   private onDefaultColorClick_() {
     CustomizeChromeApiProxy.getInstance().handler.setDefaultColor();
   }
@@ -97,6 +153,30 @@
     CustomizeChromeApiProxy.getInstance().handler.setForegroundColor(
         this.$.chromeColors.itemForElement(e.target as HTMLElement).foreground);
   }
+
+  private onCustomColorClick_() {
+    this.$.colorPicker.focus();
+    this.$.colorPicker.click();
+  }
+
+  private onCustomColorChange_(e: Event) {
+    CustomizeChromeApiProxy.getInstance().handler.setForegroundColor(
+        hexColorToSkColor((e.target as HTMLInputElement).value));
+  }
+
+  private updateCustomColor_() {
+    // We only change the custom color when theme updates to a new custom color
+    // so that the picked color persists while clicking on other color circles.
+    if (!this.isCustomColorSelected_) {
+      return;
+    }
+    this.customColor_ = {
+      background: this.theme_.backgroundColor,
+      foreground: this.theme_.foregroundColor!,
+    };
+    this.$.colorPickerIcon.style.setProperty(
+        'background-color', skColorToRgba(this.theme_.colorPickerIconColor));
+  }
 }
 
 declare global {
diff --git a/chrome/browser/resources/side_panel/customize_chrome/icons/BUILD.gn b/chrome/browser/resources/side_panel/customize_chrome/icons/BUILD.gn
index c6f0ebfd..60f2b9c1 100644
--- a/chrome/browser/resources/side_panel/customize_chrome/icons/BUILD.gn
+++ b/chrome/browser/resources/side_panel/customize_chrome/icons/BUILD.gn
@@ -9,7 +9,10 @@
 generate_grd("build_grdp") {
   grd_prefix = "side_panel_customize_chrome"
   out_grd = "$target_gen_dir/resources.grdp"
-  input_files = [ "mini_new_tab_page.svg" ]
+  input_files = [
+    "mini_new_tab_page.svg",
+    "uploaded_image.svg",
+  ]
   input_files_base_dir = rebase_path(".", "//")
   resource_path_prefix = "icons"
 }
diff --git a/chrome/browser/resources/side_panel/customize_chrome/icons/uploaded_image.svg b/chrome/browser/resources/side_panel/customize_chrome/icons/uploaded_image.svg
new file mode 100644
index 0000000..a7b06bae
--- /dev/null
+++ b/chrome/browser/resources/side_panel/customize_chrome/icons/uploaded_image.svg
@@ -0,0 +1 @@
+<svg width="288" height="162" viewBox="0 0 288 162" fill="none" xmlns="http://www.w3.org/2000/svg"><rect x="-4" width="295" height="166" rx="20" fill="#F1F3F4"/><path fill-rule="evenodd" clip-rule="evenodd" d="M152.333 71h-18.666A2.675 2.675 0 0 0 131 73.667v18.666C131 93.8 132.2 95 133.667 95h18.666C153.8 95 155 93.8 155 92.333V73.667C155 72.2 153.8 71 152.333 71Zm0 21.333h-18.666V73.667h18.666v18.666Zm-10.666-4.373 4-4.96L151 89.667h-16l4-5.334 2.667 3.627Z" fill="#5F6368"/></svg>
\ No newline at end of file
diff --git a/chrome/browser/resources/side_panel/customize_chrome/theme_snapshot.html b/chrome/browser/resources/side_panel/customize_chrome/theme_snapshot.html
index 804b196..d1846f0a 100644
--- a/chrome/browser/resources/side_panel/customize_chrome/theme_snapshot.html
+++ b/chrome/browser/resources/side_panel/customize_chrome/theme_snapshot.html
@@ -1,13 +1,18 @@
 <style>
-  #themeSnapshot {
+  .theme-snapshot {
     align-items: center;
     display: flex;
     flex-wrap: wrap;
     justify-content: center;
+    margin-bottom: 8px;
     margin-inline-start: 16px;
     width: 288px;
   }
 
+  .theme-snapshot img {
+    object-fit: fill;
+  }
+
   .image {
     border: 1px solid rgba(0, 0, 0, .14);
     border-radius: 20px;
@@ -16,11 +21,7 @@
     width: inherit;
   }
 
-  #themeSnapshot img {
-    object-fit: fill;
-  }
-
-  #classicChromeImageContainer {
+  #classicChrome {
     align-items: center;
     background-color: var(--customize-chrome-color-background-color);
     display: flex;
@@ -33,22 +34,25 @@
   }
 
   #themeTitle {
-    margin-bottom: 8px;
     overflow: hidden;
     white-space: nowrap;
   }
 </style>
-<div id="themeSnapshot">
-  <template is="dom-if" if="[[showThemeSnapshot_]]">
+<iron-pages selected="[[themeType_]]" attr-for-selected="theme-type">
+  <div class="theme-snapshot" theme-type="customTheme">
     <img class="image" is="cr-auto-img"
         auto-src="[[theme_.backgroundImage.url.url]]" draggable="false">
     </img>
     <label id="themeTitle">[[theme_.backgroundImage.title]]</label>
-  </template>
-  <template is="dom-if" if="[[!showThemeSnapshot_]]">
-    <div class="image" id="classicChromeImageContainer">
+  </div>
+  <div class="theme-snapshot" theme-type="classicChrome">
+    <div class="image" id="classicChrome">
       <img id="miniNewTabPage" src="icons/mini_new_tab_page.svg"></img>
     </div>
     <label id="themeTitle">$i18n{classicChrome}</label>
-  </template>
-</div>
+  </div>
+  <div class="theme-snapshot" theme-type="uploadedImage">
+    <img class="image" src="icons/uploaded_image.svg"></img>
+    <label id="themeTitle">$i18n{uploadedImage}</label>
+  </div>
+</iron-pages>
diff --git a/chrome/browser/resources/side_panel/customize_chrome/theme_snapshot.ts b/chrome/browser/resources/side_panel/customize_chrome/theme_snapshot.ts
index 22c030ae..692bff4 100644
--- a/chrome/browser/resources/side_panel/customize_chrome/theme_snapshot.ts
+++ b/chrome/browser/resources/side_panel/customize_chrome/theme_snapshot.ts
@@ -3,6 +3,7 @@
 // found in the LICENSE file.
 
 import 'chrome://resources/cr_elements/cr_auto_img/cr_auto_img.js';
+import 'chrome://resources/polymer/v3_0/iron-pages/iron-pages.js';
 
 import {assert} from 'chrome://resources/js/assert_ts.js';
 import {skColorToRgba} from 'chrome://resources/js/color_utils.js';
@@ -12,7 +13,13 @@
 import {CustomizeChromeApiProxy} from './customize_chrome_api_proxy.js';
 import {getTemplate} from './theme_snapshot.html.js';
 
-/** Element used to display a snapshot of a NTP custom background. */
+export enum CustomizeThemeType {
+  CLASSIC_CHROME = 'classicChrome',
+  CUSTOM_THEME = 'customTheme',
+  UPLOADED_IMAGE = 'uploadedImage',
+}
+
+/** Element used to display a snapshot of the NTP. */
 export class ThemeSnapshotElement extends PolymerElement {
   static get is() {
     return 'customize-chrome-theme-snapshot';
@@ -26,16 +33,16 @@
     return {
       theme_: Object,
 
-      showThemeSnapshot_: {
-        type: Boolean,
-        computed: 'computeShowThemeSnapshot_(theme_)',
+      themeType_: {
+        type: String,
+        computed: 'computeThemeType_(theme_)',
       },
     };
   }
 
-  private theme_: Theme;
+  private theme_: Theme|undefined = undefined;
   private setThemeListenerId_: number|null = null;
-  private showThemeSnapshot_: boolean;
+  private themeType_: CustomizeThemeType|null = null;
 
   private callbackRouter_: CustomizeChromePageCallbackRouter;
   private pageHandler_: CustomizeChromePageHandlerInterface;
@@ -51,10 +58,12 @@
     this.setThemeListenerId_ =
         this.callbackRouter_.setTheme.addListener((theme: Theme) => {
           this.theme_ = theme;
-          this.updateStyles({
-            '--customize-chrome-color-background-color':
-                skColorToRgba(this.theme_.backgroundColor),
-          });
+          if (this.theme_) {
+            this.updateStyles({
+              '--customize-chrome-color-background-color':
+                  skColorToRgba(this.theme_.backgroundColor),
+            });
+          }
         });
     this.pageHandler_.updateTheme();
   }
@@ -66,8 +75,17 @@
     this.callbackRouter_.removeListener(this.setThemeListenerId_);
   }
 
-  private computeShowThemeSnapshot_(): boolean {
-    return !!this.theme_.backgroundImage;
+  private computeThemeType_(): CustomizeThemeType|null {
+    if (!this.theme_) {
+      return null;
+    }
+    if (!this.theme_.backgroundImage) {
+      return CustomizeThemeType.CLASSIC_CHROME;
+    } else if (!this.theme_.backgroundImage.isUploadedImage) {
+      return CustomizeThemeType.CUSTOM_THEME;
+    } else {
+      return CustomizeThemeType.UPLOADED_IMAGE;
+    }
   }
 }
 
diff --git a/chrome/browser/search/background/ntp_background_data.h b/chrome/browser/search/background/ntp_background_data.h
index 1426bbe..4cf64b8 100644
--- a/chrome/browser/search/background/ntp_background_data.h
+++ b/chrome/browser/search/background/ntp_background_data.h
@@ -119,6 +119,9 @@
   // Url of the custom background selected by the user.
   GURL custom_background_url;
 
+  // Whether the image is a local resource.
+  bool is_uploaded_image;
+
   // First attribution string for custom background.
   std::string custom_background_attribution_line_1;
 
diff --git a/chrome/browser/search/background/ntp_custom_background_service.cc b/chrome/browser/search/background/ntp_custom_background_service.cc
index fcdc2745..2c5b6b030 100644
--- a/chrome/browser/search/background/ntp_custom_background_service.cc
+++ b/chrome/browser/search/background/ntp_custom_background_service.cc
@@ -398,6 +398,7 @@
     std::string local_string(chrome::kChromeUIUntrustedNewTabPageBackgroundUrl);
     GURL timestamped_url(local_string + "?ts=" + time_string);
     custom_background->custom_background_url = timestamped_url;
+    custom_background->is_uploaded_image = true;
     custom_background->custom_background_attribution_line_1 = std::string();
     custom_background->custom_background_attribution_line_2 = std::string();
     custom_background->custom_background_attribution_action_url = GURL();
@@ -429,6 +430,7 @@
     const base::Value* color =
         background_info.Find(kNtpCustomBackgroundMainColor);
     custom_background->custom_background_url = custom_background_url;
+    custom_background->is_uploaded_image = false;
     custom_background->collection_id = collection_id;
 
     if (attribution_line_1) {
diff --git a/chrome/browser/search/background/ntp_custom_background_service_unittest.cc b/chrome/browser/search/background/ntp_custom_background_service_unittest.cc
index c4c4bd5..3348082 100644
--- a/chrome/browser/search/background/ntp_custom_background_service_unittest.cc
+++ b/chrome/browser/search/background/ntp_custom_background_service_unittest.cc
@@ -100,6 +100,7 @@
 
   auto custom_background = custom_background_service_->GetCustomBackground();
   EXPECT_EQ(kUrl, custom_background->custom_background_url);
+  EXPECT_EQ(false, custom_background->is_uploaded_image);
   EXPECT_TRUE(custom_background_service_->IsCustomBackgroundSet());
 }
 
@@ -138,6 +139,7 @@
 
   auto custom_background = custom_background_service_->GetCustomBackground();
   EXPECT_EQ(kUrl, custom_background->custom_background_url);
+  EXPECT_EQ(false, custom_background->is_uploaded_image);
   EXPECT_EQ(kAttributionLine1,
             custom_background->custom_background_attribution_line_1);
   EXPECT_EQ(kAttributionLine2,
@@ -168,6 +170,8 @@
   EXPECT_EQ(true, profile_.GetTestingPrefService()->GetBoolean(
                       prefs::kNtpCustomBackgroundLocalToDevice));
   EXPECT_TRUE(custom_background_service_->IsCustomBackgroundSet());
+  auto custom_background = custom_background_service_->GetCustomBackground();
+  EXPECT_EQ(true, custom_background->is_uploaded_image);
 }
 
 TEST_F(NtpCustomBackgroundServiceTest,
@@ -194,6 +198,8 @@
   EXPECT_EQ(false, profile_.GetTestingPrefService()->GetBoolean(
                        prefs::kNtpCustomBackgroundLocalToDevice));
   ASSERT_TRUE(custom_background_service_->IsCustomBackgroundSet());
+  auto custom_background = custom_background_service_->GetCustomBackground();
+  EXPECT_EQ(false, custom_background->is_uploaded_image);
 }
 
 TEST_F(NtpCustomBackgroundServiceTest, UpdatingPrefUpdatesNtpTheme) {
@@ -218,6 +224,7 @@
 
   custom_background = custom_background_service_->GetCustomBackground();
   EXPECT_EQ(kUrlBar, custom_background->custom_background_url);
+  EXPECT_EQ(false, custom_background->is_uploaded_image);
   EXPECT_EQ(false,
             pref_service->GetBoolean(prefs::kNtpCustomBackgroundLocalToDevice));
   EXPECT_TRUE(custom_background_service_->IsCustomBackgroundSet());
@@ -247,6 +254,7 @@
   EXPECT_TRUE(
       pref_service->GetBoolean(prefs::kNtpCustomBackgroundLocalToDevice));
   EXPECT_TRUE(custom_background_service_->IsCustomBackgroundSet());
+  EXPECT_EQ(true, custom_background->is_uploaded_image);
 }
 
 TEST_F(NtpCustomBackgroundServiceTest, SyncPrefOverridesAndRemovesLocalImage) {
@@ -278,6 +286,7 @@
 
   auto custom_background = custom_background_service_->GetCustomBackground();
   EXPECT_EQ(kUrl, custom_background->custom_background_url);
+  EXPECT_EQ(false, custom_background->is_uploaded_image);
   EXPECT_FALSE(
       pref_service->GetBoolean(prefs::kNtpCustomBackgroundLocalToDevice));
   EXPECT_FALSE(base::PathExists(path));
@@ -360,6 +369,7 @@
   EXPECT_TRUE(
       pref_service->GetBoolean(prefs::kNtpCustomBackgroundLocalToDevice));
   EXPECT_TRUE(custom_background_service_->IsCustomBackgroundSet());
+  EXPECT_EQ(true, custom_background->is_uploaded_image);
   EXPECT_EQ("", custom_background->custom_background_attribution_line_1);
   EXPECT_EQ("", custom_background->custom_background_attribution_line_2);
   EXPECT_EQ(GURL(),
diff --git a/chrome/browser/share/share_features.cc b/chrome/browser/share/share_features.cc
index 0b6d30dd..48fe6465 100644
--- a/chrome/browser/share/share_features.cc
+++ b/chrome/browser/share/share_features.cc
@@ -34,7 +34,7 @@
 #if !BUILDFLAG(IS_ANDROID)
 BASE_FEATURE(kDesktopSharePreview,
              "DesktopSharePreview",
-             base::FEATURE_DISABLED_BY_DEFAULT);
+             base::FEATURE_ENABLED_BY_DEFAULT);
 extern const char kDesktopSharePreviewVariant16[] = "16";
 extern const char kDesktopSharePreviewVariant40[] = "40";
 extern const char kDesktopSharePreviewVariant72[] = "72";
diff --git a/chrome/browser/speech/speech_recognition_browsertest.cc b/chrome/browser/speech/speech_recognition_browsertest.cc
index be381c0f..f5e95dd5 100644
--- a/chrome/browser/speech/speech_recognition_browsertest.cc
+++ b/chrome/browser/speech/speech_recognition_browsertest.cc
@@ -60,7 +60,7 @@
  public:
   explicit SpeechWebContentsObserver(WebContents* web_contents)
       : WebContentsObserver(web_contents),
-        render_view_host_changed_(false),
+        render_frame_host_changed_(false),
         web_contents_destroyed_(false) {}
 
   SpeechWebContentsObserver(const SpeechWebContentsObserver&) = delete;
@@ -70,23 +70,23 @@
   ~SpeechWebContentsObserver() override {}
 
   // content::WebContentsObserver overrides.
-  void RenderViewHostChanged(content::RenderViewHost* old_host,
-                             content::RenderViewHost* new_host) override {
-    render_view_host_changed_ = true;
+  void RenderFrameHostChanged(content::RenderFrameHost* old_host,
+                              content::RenderFrameHost* new_host) override {
+    render_frame_host_changed_ = true;
   }
   void WebContentsDestroyed() override { web_contents_destroyed_ = true; }
 
   bool web_contents_destroyed() { return web_contents_destroyed_; }
-  bool render_view_host_changed() { return render_view_host_changed_; }
+  bool render_frame_host_changed() { return render_frame_host_changed_; }
 
  private:
-  bool render_view_host_changed_;
+  bool render_frame_host_changed_;
   bool web_contents_destroyed_;
 };
 
 // Tests that ChromeSpeechRecognitionManagerDelegate works properly
-// when a WebContents goes away (WCO::WebContentsDestroyed) or the RVH
-// changes within a WebContents (WCO::RenderViewHostChanged).
+// when a WebContents goes away (WCO::WebContentsDestroyed) or the RFH
+// changes within a WebContents (WCO::RenderFrameHostChanged).
 IN_PROC_BROWSER_TEST_F(ChromeSpeechRecognitionTest, BasicTearDown) {
   ASSERT_TRUE(embedded_test_server()->Start());
   net::EmbeddedTestServer https_server(net::EmbeddedTestServer::TYPE_HTTPS);
@@ -128,11 +128,11 @@
     EXPECT_EQ(kExpectedTranscript, output);
   }
 
-  // Navigating to an https page will force RVH change within
-  // |web_contents|, results in WCO::RenderViewHostChanged().
+  // Navigating to an https page will force RFH change within
+  // |web_contents|, results in WCO::RenderFrameHostChanged().
   ASSERT_TRUE(ui_test_utils::NavigateToURL(browser(), https_url));
 
-  EXPECT_TRUE(speech_contents_observer.render_view_host_changed());
+  EXPECT_TRUE(speech_contents_observer.render_frame_host_changed());
 
   {
     content::TitleWatcher title_watcher(web_contents, success_title);
diff --git a/chrome/browser/supervised_user/android/java/src/org/chromium/chrome/browser/supervised_user/WebsiteParentApprovalMetrics.java b/chrome/browser/supervised_user/android/java/src/org/chromium/chrome/browser/supervised_user/WebsiteParentApprovalMetrics.java
index cc2356d1..25acd717 100644
--- a/chrome/browser/supervised_user/android/java/src/org/chromium/chrome/browser/supervised_user/WebsiteParentApprovalMetrics.java
+++ b/chrome/browser/supervised_user/android/java/src/org/chromium/chrome/browser/supervised_user/WebsiteParentApprovalMetrics.java
@@ -11,10 +11,14 @@
 class WebsiteParentApprovalMetrics {
     // Histogram name
     static final String WEB_APPOVAL_OUTCOME_NAME = "FamilyLinkUser.LocalWebApprovalOutcome";
+    static final String WEB_APPOVAL_PACP_ERROR_CODE =
+            "Android.FamilyLinkUser.LocalWebApprovalParentAuthenticationError";
 
     // These values are persisted to logs. Entries should not be renumbered and
     // numeric values should never be reused.
     // The values need to be in sync with FamilyLinkUserLocalWebApprovalOutcome in enums.xml.
+    // TODO (b/261403529): Remove the unused values once the solution that records directly
+    // GMS error codes is integrated in Clank.
     @IntDef({FamilyLinkUserLocalWebApprovalOutcome.APPROVED_BY_PARENT,
             FamilyLinkUserLocalWebApprovalOutcome.DENIED_BY_PARENT,
             FamilyLinkUserLocalWebApprovalOutcome.PARENT_APPROVAL_CANCELLED,
@@ -42,4 +46,8 @@
         RecordHistogram.recordEnumeratedHistogram(
                 WEB_APPOVAL_OUTCOME_NAME, outcome, FamilyLinkUserLocalWebApprovalOutcome.COUNT);
     }
+
+    public static void recordParentAuthenticationErrorCode(int errorCode) {
+        RecordHistogram.recordSparseHistogram(WEB_APPOVAL_PACP_ERROR_CODE, errorCode);
+    }
 }
\ No newline at end of file
diff --git a/chrome/browser/supervised_user/android/javatests/src/org/chromium/chrome/browser/supervised_user/WebsiteParentApprovalMetricsUnitTest.java b/chrome/browser/supervised_user/android/javatests/src/org/chromium/chrome/browser/supervised_user/WebsiteParentApprovalMetricsUnitTest.java
index c8c1dea..c40bfa8 100644
--- a/chrome/browser/supervised_user/android/javatests/src/org/chromium/chrome/browser/supervised_user/WebsiteParentApprovalMetricsUnitTest.java
+++ b/chrome/browser/supervised_user/android/javatests/src/org/chromium/chrome/browser/supervised_user/WebsiteParentApprovalMetricsUnitTest.java
@@ -61,4 +61,25 @@
         count = mHistogramTestRule.getHistogramTotalCount(histogramName);
         Assert.assertEquals(3, count);
     }
+
+    @Test
+    @SmallTest
+    public void recordParentAuthenticationErrorMetrics() {
+        final String histogramName =
+                "Android.FamilyLinkUser.LocalWebApprovalParentAuthenticationError";
+        final int negativeErrorCode = -1;
+        final int lowValueCode = 10; // Example: value of CommonStatusCode.DEVELOPER_ERROR.
+
+        WebsiteParentApprovalMetrics.recordParentAuthenticationErrorCode(negativeErrorCode);
+        int count = mHistogramTestRule.getHistogramValueCount(histogramName, negativeErrorCode);
+        Assert.assertEquals(1, count);
+
+        WebsiteParentApprovalMetrics.recordParentAuthenticationErrorCode(lowValueCode);
+        WebsiteParentApprovalMetrics.recordParentAuthenticationErrorCode(lowValueCode);
+        count = mHistogramTestRule.getHistogramValueCount(histogramName, lowValueCode);
+        Assert.assertEquals(2, count);
+
+        count = mHistogramTestRule.getHistogramTotalCount(histogramName);
+        Assert.assertEquals(3, count);
+    }
 }
\ No newline at end of file
diff --git a/chrome/browser/ui/android/omnibox/BUILD.gn b/chrome/browser/ui/android/omnibox/BUILD.gn
index be8ec10..83de8451 100644
--- a/chrome/browser/ui/android/omnibox/BUILD.gn
+++ b/chrome/browser/ui/android/omnibox/BUILD.gn
@@ -130,7 +130,6 @@
     "java/src/org/chromium/chrome/browser/omnibox/suggestions/tail/TailSuggestionViewProperties.java",
     "java/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchConsentBottomSheet.java",
     "java/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchConsentController.java",
-    "java/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchConsentModal.java",
     "java/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchConsentUi.java",
     "java/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchService.java",
     "java/src/org/chromium/chrome/browser/omnibox/voice/VoiceRecognitionHandler.java",
@@ -307,7 +306,6 @@
     "java/res/drawable-xxxhdpi/btn_suggestion_refine.png",
     "java/res/drawable-xxxhdpi/ic_history_googblue_24dp.png",
     "java/res/drawable-xxxhdpi/ic_suggestion_magnifier.png",
-    "java/res/drawable/assistant_consent_illustration.xml",
     "java/res/drawable/ic_book_round.xml",
     "java/res/drawable/ic_colorful_mic.xml",
     "java/res/drawable/ic_content_copy_black.xml",
@@ -324,7 +322,6 @@
     "java/res/drawable/trending_up_black_24dp.xml",
     "java/res/layout-sw600dp/location_bar.xml",
     "java/res/layout/assistant_voice_search_consent_ui.xml",
-    "java/res/layout/assistant_voice_search_modal_consent_ui.xml",
     "java/res/layout/assistant_voice_search_simplified_consent_ui.xml",
     "java/res/layout/location_bar.xml",
     "java/res/layout/location_bar_base.xml",
diff --git a/chrome/browser/ui/android/omnibox/java/res/drawable/assistant_consent_illustration.xml b/chrome/browser/ui/android/omnibox/java/res/drawable/assistant_consent_illustration.xml
deleted file mode 100644
index 7a1597f..0000000
--- a/chrome/browser/ui/android/omnibox/java/res/drawable/assistant_consent_illustration.xml
+++ /dev/null
@@ -1,120 +0,0 @@
-<?xml version="1.0" encoding="utf-8"?>
-<!--
-Copyright 2022 The Chromium Authors
-Use of this source code is governed by a BSD-style license that can be
-found in the LICENSE file.
--->
-<!--VectorRaster is ignored because: The image should be shown rarely enough to
-    prioritize binary size over inflation time.-->
-<vector
-    xmlns:android="http://schemas.android.com/apk/res/android"
-    xmlns:tools="http://schemas.android.com/tools"
-    android:width="256dp"
-    android:height="181dp"
-    android:viewportWidth="256"
-    android:viewportHeight="181"
-    tools:ignore="VectorRaster">
-  <group>
-    <clip-path
-        android:pathData="M-36,0h332v181h-332z"/>
-    <path
-        android:pathData="m161,19.5h-62c-4.142,0 -7.5,3.358 -7.5,7.5v122c0,4.142 3.358,7.5 7.5,7.5h62c4.142,0 7.5,-3.358 7.5,-7.5L168.5,27c0,-4.142 -3.358,-7.5 -7.5,-7.5z"
-        android:strokeWidth="5"
-        android:fillColor="#ffffff"
-        android:strokeColor="#9aa0a6"/>
-    <path
-        android:pathData="m145,30h-42c-2.761,0 -5,2.239 -5,5 0,2.761 2.239,5 5,5h42c2.761,0 5,-2.239 5,-5 0,-2.761 -2.239,-5 -5,-5z"
-        android:fillColor="#9aa0a6"/>
-    <path
-        android:pathData="m152,35c0,2.761 2.239,5 5,5 2.761,0 5,-2.239 5,-5 0,-2.761 -2.239,-5 -5,-5 -2.761,0 -5,2.239 -5,5z"
-        android:fillColor="#9aa0a6"/>
-    <path
-        android:pathData="m146.502,83.128c1.476,0 2.673,-1.164 2.673,-2.601 0,-1.436 -1.197,-2.601 -2.673,-2.601 -1.475,0 -2.672,1.164 -2.672,2.601 0,1.436 1.197,2.601 2.672,2.601z"
-        android:fillColor="#33a853"/>
-    <path
-        android:pathData="m138.49,91.788c2.949,0 5.34,-2.327 5.34,-5.197 0,-2.87 -2.391,-5.197 -5.34,-5.197 -2.95,0 -5.34,2.327 -5.34,5.197 0,2.87 2.39,5.197 5.34,5.197z"
-        android:fillColor="#e55145"/>
-    <path
-        android:pathData="m138.49,105.645c3.442,0 6.232,-2.715 6.232,-6.064 0,-3.349 -2.79,-6.064 -6.232,-6.064 -3.442,0 -6.233,2.715 -6.233,6.064 0,3.349 2.791,6.064 6.233,6.064z"
-        android:fillColor="#fcc934"/>
-    <path
-        android:pathData="m120.685,91.788c5.901,0 10.684,-4.654 10.684,-10.394C131.369,75.654 126.586,71 120.685,71 114.784,71 110,75.654 110,81.394c0,5.74 4.784,10.394 10.685,10.394z"
-        android:fillColor="#4284f4"/>
-    <path
-        android:pathData="M82.484,132.567L55.478,132.567C45.825,132.567 38,140.422 38,150.111v0.005c0,9.69 7.825,17.545 17.478,17.545h27.006c9.653,0 17.478,-7.855 17.478,-17.545v-0.005c0,-9.689 -7.825,-17.544 -17.478,-17.544z"
-        android:fillColor="#fde293"/>
-    <path
-        android:pathData="M68.063,135.249 L82.058,118v17.249z"
-        android:fillColor="#fde293"/>
-    <path
-        android:pathData="m56.973,143.863 l0.682,13.839c0.003,0.148 0.047,0.293 0.129,0.418 0.082,0.125 0.197,0.225 0.333,0.289 0.136,0.064 0.288,0.089 0.438,0.073 0.15,-0.016 0.292,-0.073 0.411,-0.164l11.888,-8.648c0.121,-0.083 0.216,-0.196 0.276,-0.329 0.059,-0.132 0.082,-0.277 0.064,-0.421 -0.017,-0.144 -0.074,-0.28 -0.164,-0.394 -0.09,-0.115 -0.209,-0.203 -0.346,-0.256l-12.582,-5.199c-0.128,-0.054 -0.268,-0.075 -0.406,-0.06 -0.138,0.015 -0.27,0.064 -0.384,0.144 -0.113,0.079 -0.204,0.186 -0.263,0.31 -0.059,0.124 -0.085,0.261 -0.076,0.398z"
-        android:fillColor="#e29a64"/>
-    <path
-        android:pathData="m77.87,142.637 l-0.101,-0.007c-0.506,-0.041 -0.949,0.339 -0.989,0.847l-1.163,14.811c-0.04,0.508 0.338,0.952 0.844,0.992l0.1,0.008c0.507,0.04 0.949,-0.339 0.989,-0.847l1.164,-14.811c0.04,-0.508 -0.338,-0.952 -0.844,-0.993z"
-        android:fillColor="#e29a64"/>
-    <path
-        android:pathData="m82.141,142.974 l-0.101,-0.008c-0.506,-0.04 -0.949,0.339 -0.989,0.847l-1.164,14.811c-0.04,0.508 0.338,0.952 0.845,0.992l0.1,0.008c0.506,0.04 0.949,-0.339 0.989,-0.847l1.164,-14.811c0.04,-0.508 -0.338,-0.952 -0.844,-0.992z"
-        android:fillColor="#e29a64"/>
-    <path
-        android:pathData="M228.147,105L198.86,105c-13.786,0 -24.962,11.218 -24.962,25.056v0.006c0,13.838 11.176,25.056 24.962,25.056h29.287c13.786,0 24.962,-11.218 24.962,-25.056v-0.006C253.109,116.218 241.933,105 228.147,105Z"
-        android:fillColor="#4285f4"/>
-    <path
-        android:pathData="M193.684,109.785 L161,106.713l23.15,22.589z"
-        android:fillColor="#4285f4"/>
-    <path
-        android:pathData="m191,122h13"
-        android:strokeWidth="3"
-        android:fillColor="#00000000"
-        android:strokeColor="#ffffff"/>
-    <path
-        android:pathData="m209,122h28"
-        android:strokeWidth="3"
-        android:fillColor="#00000000"
-        android:strokeColor="#ffffff"/>
-    <path
-        android:pathData="m188,137h15"
-        android:strokeWidth="3"
-        android:fillColor="#00000000"
-        android:strokeColor="#ffffff"/>
-    <path
-        android:pathData="m210,137h8"
-        android:strokeWidth="3"
-        android:fillColor="#00000000"
-        android:strokeColor="#ffffff"/>
-    <path
-        android:pathData="m224,137h16"
-        android:strokeWidth="3"
-        android:fillColor="#00000000"
-        android:strokeColor="#ffffff"/>
-    <path
-        android:pathData="m161.876,66.669 l31.476,-41.265 8.888,18.441z"
-        android:fillColor="#aecbfa"/>
-    <path
-        android:pathData="m217.116,48.732c9.833,-9.87 9.767,-25.938 -0.147,-35.89 -9.913,-9.951 -25.921,-10.017 -35.754,-0.147 -9.832,9.87 -9.767,25.938 0.147,35.89 9.914,9.951 25.922,10.017 35.754,0.147z"
-        android:fillColor="#aecbfa"/>
-    <path
-        android:pathData="m190.806,43.51c-0.099,0.001 -0.197,-0.027 -0.282,-0.078 -0.086,-0.052 -0.155,-0.126 -0.201,-0.215 -0.046,-0.089 -0.066,-0.188 -0.059,-0.288 0.006,-0.1 0.041,-0.196 0.098,-0.277l6.457,-9.278c0.039,-0.062 0.09,-0.115 0.15,-0.157 0.061,-0.041 0.129,-0.07 0.2,-0.085 0.072,-0.014 0.146,-0.014 0.217,0.001 0.072,0.015 0.14,0.044 0.2,0.086 0.06,0.042 0.111,0.096 0.149,0.158 0.039,0.062 0.065,0.132 0.076,0.204 0.011,0.072 0.008,0.147 -0.01,0.218 -0.018,0.071 -0.05,0.138 -0.094,0.196l-6.457,9.298c-0.051,0.069 -0.119,0.124 -0.196,0.162 -0.077,0.038 -0.162,0.056 -0.248,0.055z"
-        android:strokeWidth="0.5"
-        android:fillColor="#2e4469"
-        android:strokeColor="#2e4469"/>
-    <path
-        android:strokeWidth="1"
-        android:pathData="m201.495,34.841c4.463,0 8.082,-3.632 8.082,-8.112 0,-4.48 -3.619,-8.112 -8.082,-8.112 -4.463,0 -8.082,3.632 -8.082,8.112 0,4.48 3.619,8.112 8.082,8.112z"
-        android:fillColor="#00000000"
-        android:strokeColor="#000000"/>
-    <path
-        android:pathData="m201.495,35.385c-1.706,0 -3.373,-0.508 -4.791,-1.459 -1.418,-0.951 -2.523,-2.303 -3.176,-3.885 -0.653,-1.582 -0.823,-3.322 -0.491,-5.001 0.333,-1.679 1.154,-3.221 2.36,-4.432 1.206,-1.211 2.743,-2.035 4.416,-2.369 1.673,-0.334 3.406,-0.163 4.982,0.493 1.576,0.655 2.923,1.765 3.87,3.188 0.948,1.423 1.454,3.097 1.454,4.809 -0.004,2.295 -0.913,4.495 -2.53,6.117 -1.616,1.623 -3.808,2.536 -6.094,2.539zM201.495,19.16c-1.491,0 -2.949,0.444 -4.189,1.276 -1.24,0.832 -2.206,2.014 -2.777,3.397 -0.571,1.383 -0.72,2.905 -0.429,4.373 0.291,1.468 1.009,2.817 2.064,3.875 1.054,1.059 2.397,1.779 3.86,2.071 1.463,0.292 2.979,0.142 4.356,-0.431 1.378,-0.573 2.556,-1.543 3.384,-2.788 0.829,-1.245 1.271,-2.708 1.271,-4.205 -0.004,-2.005 -0.8,-3.927 -2.214,-5.344 -1.413,-1.417 -3.328,-2.214 -5.326,-2.216z"
-        android:strokeWidth="0.5"
-        android:fillColor="#2e4469"
-        android:strokeColor="#2e4469"/>
-    <path
-        android:pathData="m226.5,72c3.038,0 5.5,-2.462 5.5,-5.5 0,-3.038 -2.462,-5.5 -5.5,-5.5 -3.038,0 -5.5,2.462 -5.5,5.5 0,3.038 2.462,5.5 5.5,5.5z"
-        android:fillColor="#dadce0"/>
-    <path
-        android:pathData="m179.404,165.212 l-5.037,2.908c-0.905,0.522 -1.215,1.679 -0.693,2.584l2.908,5.037c0.523,0.905 1.68,1.215 2.585,0.692l5.036,-2.908c0.905,-0.522 1.215,-1.679 0.693,-2.584l-2.908,-5.037c-0.523,-0.905 -1.68,-1.215 -2.584,-0.692z"
-        android:fillColor="#dadce0"/>
-    <path
-        android:pathData="m52.757,67.708 l9.765,3.234c0.185,0.062 0.384,0.075 0.577,0.038 0.192,-0.037 0.371,-0.122 0.519,-0.248 0.148,-0.125 0.259,-0.286 0.322,-0.467 0.064,-0.18 0.077,-0.373 0.038,-0.56l-1.999,-9.83c-0.038,-0.187 -0.126,-0.361 -0.255,-0.504 -0.129,-0.143 -0.295,-0.251 -0.48,-0.313 -0.186,-0.062 -0.385,-0.074 -0.577,-0.037 -0.192,0.037 -0.371,0.123 -0.519,0.248l-7.766,6.596c-0.148,0.125 -0.259,0.286 -0.323,0.466 -0.064,0.18 -0.077,0.374 -0.039,0.561 0.038,0.187 0.126,0.361 0.255,0.504 0.129,0.143 0.295,0.251 0.481,0.312z"
-        android:fillColor="#dadce0"/>
-  </group>
-</vector>
diff --git a/chrome/browser/ui/android/omnibox/java/res/layout/assistant_voice_search_modal_consent_ui.xml b/chrome/browser/ui/android/omnibox/java/res/layout/assistant_voice_search_modal_consent_ui.xml
deleted file mode 100644
index 16c47da..0000000
--- a/chrome/browser/ui/android/omnibox/java/res/layout/assistant_voice_search_modal_consent_ui.xml
+++ /dev/null
@@ -1,53 +0,0 @@
-<?xml version="1.0" encoding="utf-8"?>
-<!--
-Copyright 2022 The Chromium Authors
-Use of this source code is governed by a BSD-style license that can be
-found in the LICENSE file.
--->
-<org.chromium.components.browser_ui.widget.FadingEdgeScrollView
-    xmlns:android="http://schemas.android.com/apk/res/android"
-    xmlns:app="http://schemas.android.com/apk/res-auto"
-    xmlns:tools="http://schemas.android.com/tools"
-    android:id="@+id/avs_consent_ui"
-    android:layout_width="match_parent"
-    android:layout_height="wrap_content"
-    android:importantForAccessibility="no">
-    <LinearLayout
-        android:layout_width="match_parent"
-        android:layout_height="wrap_content"
-        android:orientation="vertical"
-        android:gravity="center_vertical"
-        android:paddingHorizontal="20dp"
-        android:paddingVertical="10dp"
-        android:importantForAccessibility="no">
-        <ImageView
-            android:layout_width="match_parent"
-            android:layout_height="122dp"
-            android:layout_marginTop="20dp"
-            android:importantForAccessibility="no"
-            android:src="@drawable/assistant_consent_illustration"/>
-
-        <TextView
-            android:layout_width="match_parent"
-            android:layout_height="wrap_content"
-            android:layout_marginTop="20dp"
-            style="@style/TextAppearance.Headline.Primary"
-            android:text="@string/avs_consent_ui_simplified_title" />
-
-        <TextView
-            android:layout_width="match_parent"
-            android:layout_height="wrap_content"
-            android:layout_marginTop="20dp"
-            style="@style/TextAppearance.TextSmall.Secondary"
-            android:text="@string/avs_consent_ui_simplified_body" />
-
-        <TextView
-            android:id="@+id/avs_consent_ui_learn_more"
-            android:layout_width="match_parent"
-            android:layout_height="wrap_content"
-            android:layout_marginTop="2dp"
-            style="@style/TextAppearance.TextSmall.Link"
-            android:text="@string/learn_more" />
-
-    </LinearLayout>
-</org.chromium.components.browser_ui.widget.FadingEdgeScrollView>
diff --git a/chrome/browser/ui/android/omnibox/java/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchConsentController.java b/chrome/browser/ui/android/omnibox/java/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchConsentController.java
index 83d93aa..c07cc9d7 100644
--- a/chrome/browser/ui/android/omnibox/java/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchConsentController.java
+++ b/chrome/browser/ui/android/omnibox/java/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchConsentController.java
@@ -18,7 +18,6 @@
 import org.chromium.chrome.browser.preferences.SharedPreferencesManager;
 import org.chromium.components.browser_ui.bottomsheet.BottomSheetController;
 import org.chromium.ui.base.WindowAndroid;
-import org.chromium.ui.modaldialog.ModalDialogManager;
 
 import java.lang.annotation.Retention;
 import java.lang.annotation.RetentionPolicy;
@@ -40,9 +39,6 @@
      * @param bottomSheetController The {@link BottomSheetController} used to show the bottom sheet
      *                              UI. This can be null when starting the consent flow from
      *                              SearchActivity.
-     * @param modalDialogManager The {@link ModalDialogManager} used to show the modal dialog UI.
-     *                           This can be null when starting the consent flow from
-     *                           SearchActivity.
      * @param completionCallback A callback to be invoked if the user is continuing with the
      *                           requested voice search.
      */
@@ -50,38 +46,24 @@
             @NonNull SharedPreferencesManager sharedPreferencesManager,
             @NonNull Runnable launchAssistantSettingsAction,
             @Nullable BottomSheetController bottomSheetController,
-            @Nullable ModalDialogManager modalDialogManager,
             @NonNull Callback<Boolean> completionCallback) {
         AssistantVoiceSearchConsentUi consentUi;
         assert (!ChromeFeatureList.isEnabled(
                 ChromeFeatureList.ASSISTANT_NON_PERSONALIZED_VOICE_SEARCH));
 
-        if (ChromeFeatureList.isEnabled(ChromeFeatureList.ASSISTANT_CONSENT_MODAL)) {
-            // If the modal manager isn't available, bail out of the consent flow and fallback to
-            // system-ui. Consent will be retried the next time.
-            if (modalDialogManager == null) {
-                PostTask.postTask(TaskTraits.USER_VISIBLE,
-                        () -> { completionCallback.onResult(/* useAssistant= */ false); });
-                return;
-            }
-
-            consentUi = new AssistantVoiceSearchConsentModal(
-                    windowAndroid.getContext().get(), modalDialogManager);
-        } else {
-            // When attempting voice search through the search widget, the bottom sheet isn't
-            // available. When this happens, bail out of the consent flow and fallback to system-ui.
-            // Consent will be retried the next time.
-            if (bottomSheetController == null) {
-                PostTask.postTask(TaskTraits.USER_VISIBLE,
-                        () -> { completionCallback.onResult(/* useAssistant= */ false); });
-                return;
-            }
-
-            // Use the bottom sheet consent by default.
-            consentUi = new AssistantVoiceSearchConsentBottomSheet(
-                    windowAndroid.getContext().get(), bottomSheetController);
+        // When attempting voice search through the search widget, the bottom sheet isn't
+        // available. When this happens, bail out of the consent flow and fallback to system-ui.
+        // Consent will be retried the next time.
+        if (bottomSheetController == null) {
+            PostTask.postTask(TaskTraits.USER_VISIBLE,
+                    () -> { completionCallback.onResult(/* useAssistant= */ false); });
+            return;
         }
 
+        // Use the bottom sheet consent by default.
+        consentUi = new AssistantVoiceSearchConsentBottomSheet(
+                windowAndroid.getContext().get(), bottomSheetController);
+
         AssistantVoiceSearchConsentController consent =
                 new AssistantVoiceSearchConsentController(windowAndroid, sharedPreferencesManager,
                         completionCallback, launchAssistantSettingsAction, consentUi);
diff --git a/chrome/browser/ui/android/omnibox/java/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchConsentModal.java b/chrome/browser/ui/android/omnibox/java/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchConsentModal.java
deleted file mode 100644
index 2411a16..0000000
--- a/chrome/browser/ui/android/omnibox/java/src/org/chromium/chrome/browser/omnibox/voice/AssistantVoiceSearchConsentModal.java
+++ /dev/null
@@ -1,93 +0,0 @@
-// Copyright 2022 The Chromium Authors
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-package org.chromium.chrome.browser.omnibox.voice;
-
-import android.content.Context;
-import android.content.res.Resources;
-import android.view.LayoutInflater;
-import android.view.View;
-
-import androidx.annotation.NonNull;
-
-import org.chromium.chrome.browser.omnibox.R;
-import org.chromium.ui.modaldialog.DialogDismissalCause;
-import org.chromium.ui.modaldialog.ModalDialogManager;
-import org.chromium.ui.modaldialog.ModalDialogProperties;
-import org.chromium.ui.modaldialog.SimpleModalDialogController;
-import org.chromium.ui.modelutil.PropertyModel;
-
-/**
- * The modal dialog implementation of the ConsentUI shown to users upon first using Assistant Voice
- * Search.
- */
-class AssistantVoiceSearchConsentModal implements AssistantVoiceSearchConsentUi {
-    private View mContentView;
-    private AssistantVoiceSearchConsentUi.Observer mObserver;
-
-    private ModalDialogManager mModalDialogManager;
-    private PropertyModel mConsentModal;
-
-    public AssistantVoiceSearchConsentModal(
-            @NonNull Context context, @NonNull ModalDialogManager modalDialogManager) {
-        mModalDialogManager = modalDialogManager;
-
-        mContentView = LayoutInflater.from(context).inflate(
-                R.layout.assistant_voice_search_modal_consent_ui, /* root= */ null);
-
-        View learnMore = mContentView.findViewById(R.id.avs_consent_ui_learn_more);
-        learnMore.setOnClickListener((v) -> mObserver.onLearnMoreClicked());
-
-        Resources resources = context.getResources();
-        mConsentModal =
-                new PropertyModel.Builder(ModalDialogProperties.ALL_KEYS)
-                        .with(ModalDialogProperties.CONTROLLER,
-                                new SimpleModalDialogController(
-                                        mModalDialogManager, this::onDismissConsentModal))
-                        .with(ModalDialogProperties.POSITIVE_BUTTON_TEXT, resources,
-                                R.string.avs_consent_ui_simplified_accept)
-                        .with(ModalDialogProperties.NEGATIVE_BUTTON_TEXT, resources,
-                                R.string.avs_consent_ui_simplified_deny)
-                        .with(ModalDialogProperties.BUTTON_STYLES,
-                                ModalDialogProperties.ButtonStyles.PRIMARY_FILLED_NEGATIVE_OUTLINE)
-                        .with(ModalDialogProperties.CANCEL_ON_TOUCH_OUTSIDE, true)
-                        .with(ModalDialogProperties.CUSTOM_VIEW, mContentView)
-                        .build();
-    }
-
-    // AssistantVoiceSearchConsentUi implementation.
-
-    @Override
-    public void show(AssistantVoiceSearchConsentUi.Observer observer) {
-        assert mObserver == null;
-
-        mObserver = observer;
-        mModalDialogManager.showDialog(mConsentModal, ModalDialogManager.ModalDialogType.APP);
-    }
-
-    @Override
-    public void dismiss() {
-        mObserver = null;
-        mModalDialogManager.dismissDialog(mConsentModal, DialogDismissalCause.ACTION_ON_CONTENT);
-    }
-
-    private void onDismissConsentModal(@DialogDismissalCause int dismissalCause) {
-        if (mObserver == null) {
-            // The observer was already notified of a result, so the UI was dismissed by the
-            // controller.
-            return;
-        }
-
-        if (dismissalCause == DialogDismissalCause.POSITIVE_BUTTON_CLICKED) {
-            mObserver.onConsentAccepted();
-        } else if (dismissalCause == DialogDismissalCause.NEGATIVE_BUTTON_CLICKED) {
-            mObserver.onConsentRejected();
-        } else if (dismissalCause == DialogDismissalCause.NAVIGATE_BACK_OR_TOUCH_OUTSIDE) {
-            mObserver.onConsentCanceled();
-        } else {
-            mObserver.onNonUserCancel();
-        }
-        mObserver = null;
-    }
-}
diff --git a/chrome/browser/ui/android/omnibox/java/src/org/chromium/chrome/browser/omnibox/voice/VoiceRecognitionHandler.java b/chrome/browser/ui/android/omnibox/java/src/org/chromium/chrome/browser/omnibox/voice/VoiceRecognitionHandler.java
index 6054406..56f299f 100644
--- a/chrome/browser/ui/android/omnibox/java/src/org/chromium/chrome/browser/omnibox/voice/VoiceRecognitionHandler.java
+++ b/chrome/browser/ui/android/omnibox/java/src/org/chromium/chrome/browser/omnibox/voice/VoiceRecognitionHandler.java
@@ -863,8 +863,7 @@
             mDelegate.clearOmniboxFocus();
             AssistantVoiceSearchConsentController.show(windowAndroid,
                     SharedPreferencesManager.getInstance(), mLaunchAssistanceSettingsAction,
-                    BottomSheetControllerProvider.from(windowAndroid),
-                    windowAndroid.getModalDialogManager(), (useAssistant) -> {
+                    BottomSheetControllerProvider.from(windowAndroid), (useAssistant) -> {
                         // Notify the service about the consent completion.
                         assistantVoiceSearchService.onAssistantConsentDialogComplete(useAssistant);
 
diff --git a/chrome/browser/ui/android/quickactionsearchwidget/java/src/org/chromium/chrome/browser/ui/quickactionsearchwidget/QuickActionSearchWidgetProviderDelegate.java b/chrome/browser/ui/android/quickactionsearchwidget/java/src/org/chromium/chrome/browser/ui/quickactionsearchwidget/QuickActionSearchWidgetProviderDelegate.java
index bb51425..8b496f47 100644
--- a/chrome/browser/ui/android/quickactionsearchwidget/java/src/org/chromium/chrome/browser/ui/quickactionsearchwidget/QuickActionSearchWidgetProviderDelegate.java
+++ b/chrome/browser/ui/android/quickactionsearchwidget/java/src/org/chromium/chrome/browser/ui/quickactionsearchwidget/QuickActionSearchWidgetProviderDelegate.java
@@ -363,29 +363,21 @@
      *
      * @param context Current context.
      * @param prefs Structure describing current preferences and feature availability.
-     * @param portraitModeWidthDp Width of the widget area in portrait mode.
-     * @param portraitModeHeightDp Height of the widget area in portrait mode.
-     * @param landscapeModeWidthDp Width of the widget area in landscape mode.
-     * @param landscapeModeHeightDp Height of the widget area in landscape mode.
+     * @param areaWidthDp Width of the widget area.
+     * @param areaHeightDp Height of the widget area.
      * @return RemoteViews to be installed on the Dino widget.
      */
     public @NonNull RemoteViews createDinoWidgetRemoteViews(@NonNull Context context,
-            @NonNull SearchActivityPreferences prefs, int portraitModeWidthDp,
-            int portraitModeHeightDp, int landscapeModeWidthDp, int landscapeModeHeightDp) {
-        RemoteViews landscapeViews =
-                createWidgetRemoteViews(context, R.layout.quick_action_search_widget_dino_layout);
-        RemoteViews portraitViews =
+            @NonNull SearchActivityPreferences prefs, int areaWidthDp, int areaHeightDp) {
+        RemoteViews views =
                 createWidgetRemoteViews(context, R.layout.quick_action_search_widget_dino_layout);
 
         // Dino widget is specific; we want to scale up a lot of dimensions based on the actual size
         // of the widget area. This makes layout a lot more responsive but also a lot more
         // complicated since we have to compute everything manually.
-        Resources res = context.getApplicationContext().getResources();
         resizeDinoWidgetToFillTargetCellArea(
-                res, landscapeViews, landscapeModeWidthDp, landscapeModeHeightDp);
-        resizeDinoWidgetToFillTargetCellArea(
-                res, portraitViews, portraitModeWidthDp, portraitModeHeightDp);
-        return new RemoteViews(landscapeViews, portraitViews);
+                context.getApplicationContext().getResources(), views, areaWidthDp, areaHeightDp);
+        return views;
     }
 
     /**
@@ -396,26 +388,16 @@
      *
      * @param context Current context.
      * @param prefs Structure describing current preferences and feature availability.
-     * @param portraitModeWidthDp Width of the widget area in portrait mode.
-     * @param portraitModeHeightDp Height of the widget area in portrait mode.
-     * @param landscapeModeWidthDp Width of the widget area in landscape mode.
-     * @param landscapeModeHeightDp Height of the widget area in landscape mode.
-     * @return RemoteViews to be installed on the Search widget.
+     * @param areaWidthDp Width of the widget area.
+     * @param areaHeightDp Height of the widget area.
+     * @return RemoteViews to be installed on the Search widget for the passed variant.
      */
     public @NonNull RemoteViews createSearchWidgetRemoteViews(@NonNull Context context,
-            @NonNull SearchActivityPreferences prefs, int portraitModeWidthDp,
-            int portraitModeHeightDp, int landscapeModeWidthDp, int landscapeModeHeightDp) {
-        WidgetVariant landscapeVariant = getSearchWidgetVariantForHeight(landscapeModeHeightDp);
-        RemoteViews landscapeViews = createWidgetRemoteViews(context, landscapeVariant.layout);
-        applyRemoteViewsButtonVisibilityToFitWidth(
-                landscapeViews, prefs, landscapeVariant, landscapeModeWidthDp);
-
-        WidgetVariant portraitVariant = getSearchWidgetVariantForHeight(portraitModeHeightDp);
-        RemoteViews portraitViews = createWidgetRemoteViews(context, portraitVariant.layout);
-        applyRemoteViewsButtonVisibilityToFitWidth(
-                portraitViews, prefs, portraitVariant, portraitModeWidthDp);
-
-        return new RemoteViews(landscapeViews, portraitViews);
+            @NonNull SearchActivityPreferences prefs, int areaWidthDp, int areaHeightDp) {
+        WidgetVariant variant = getSearchWidgetVariantForHeight(areaHeightDp);
+        RemoteViews views = createWidgetRemoteViews(context, variant.layout);
+        applyRemoteViewsButtonVisibilityToFitWidth(views, prefs, variant, areaWidthDp);
+        return views;
     }
 
     /**
diff --git a/chrome/browser/ui/android/quickactionsearchwidget/java/src/org/chromium/chrome/browser/ui/quickactionsearchwidget/QuickActionSearchWidgetProviderDelegateTest.java b/chrome/browser/ui/android/quickactionsearchwidget/java/src/org/chromium/chrome/browser/ui/quickactionsearchwidget/QuickActionSearchWidgetProviderDelegateTest.java
index 377fb0e..381f7e5 100644
--- a/chrome/browser/ui/android/quickactionsearchwidget/java/src/org/chromium/chrome/browser/ui/quickactionsearchwidget/QuickActionSearchWidgetProviderDelegateTest.java
+++ b/chrome/browser/ui/android/quickactionsearchwidget/java/src/org/chromium/chrome/browser/ui/quickactionsearchwidget/QuickActionSearchWidgetProviderDelegateTest.java
@@ -179,18 +179,13 @@
         float density = res.getDisplayMetrics().density;
 
         mWidgetView = mDelegate
-                              .createSearchWidgetRemoteViews(mContext, prefs,
-                                      // Simulate optimally sized rectangular screen:
-                                      // Portrait mode dimensions:
-                                      mDefaultWidgetWidthDp, mMediumWidgetMinHeightDp,
-                                      // Landscape mode dimensions:
-                                      mDefaultWidgetWidthDp, mMediumWidgetMinHeightDp)
+                              .createSearchWidgetRemoteViews(mContext, prefs, mDefaultWidgetWidthDp,
+                                      mMediumWidgetMinHeightDp)
                               .apply(mContext, null);
-        mDinoWidgetView =
-                mDelegate
-                        .createDinoWidgetRemoteViews(mContext, prefs, mDinoWidgetEdgeSizeDp,
-                                mDinoWidgetEdgeSizeDp, mDinoWidgetEdgeSizeDp, mDinoWidgetEdgeSizeDp)
-                        .apply(mContext, null);
+        mDinoWidgetView = mDelegate
+                                  .createDinoWidgetRemoteViews(mContext, prefs,
+                                          mDinoWidgetEdgeSizeDp, mDinoWidgetEdgeSizeDp)
+                                  .apply(mContext, null);
     }
 
     /**
diff --git a/chrome/browser/ui/android/strings/android_chrome_strings.grd b/chrome/browser/ui/android/strings/android_chrome_strings.grd
index 97d6b46..fef7816 100644
--- a/chrome/browser/ui/android/strings/android_chrome_strings.grd
+++ b/chrome/browser/ui/android/strings/android_chrome_strings.grd
@@ -3558,7 +3558,7 @@
         Restore tab group of <ph name="TAB_COUNT">%1$s<ex>5</ex></ph> tabs as a new background tab group.
       </message>
       <message name="IDS_WEB_FEED_AWARENESS" desc="Educating the user about the web feed.">
-        See sites you follow here
+        See content you follow here
       </message>
 
       <!-- Explore sites strings -->
diff --git a/chrome/browser/ui/android/strings/android_chrome_strings_grd/IDS_WEB_FEED_AWARENESS.png.sha1 b/chrome/browser/ui/android/strings/android_chrome_strings_grd/IDS_WEB_FEED_AWARENESS.png.sha1
index d9a5695..52cc55de 100644
--- a/chrome/browser/ui/android/strings/android_chrome_strings_grd/IDS_WEB_FEED_AWARENESS.png.sha1
+++ b/chrome/browser/ui/android/strings/android_chrome_strings_grd/IDS_WEB_FEED_AWARENESS.png.sha1
@@ -1 +1 @@
-ad4be6652099b56efeedc53b02a1fee65ea4f81d
\ No newline at end of file
+839ef508d898ad0600a9995d2b24a4f10287f37b
\ No newline at end of file
diff --git a/chrome/browser/ui/ash/shelf/chrome_shelf_controller_unittest.cc b/chrome/browser/ui/ash/shelf/chrome_shelf_controller_unittest.cc
index a35409c..74b470e 100644
--- a/chrome/browser/ui/ash/shelf/chrome_shelf_controller_unittest.cc
+++ b/chrome/browser/ui/ash/shelf/chrome_shelf_controller_unittest.cc
@@ -537,11 +537,6 @@
     provider->Start();
 
     system_web_app_manager->ScheduleStart();
-
-    base::RunLoop run_loop;
-    provider->on_external_managers_synchronized().Post(FROM_HERE,
-                                                       run_loop.QuitClosure());
-    run_loop.Run();
   }
 
   // Note that this resets previously installed SWAs.
diff --git a/chrome/browser/ui/ash/system_tray_tray_cast_browsertest_media_router_chromeos.cc b/chrome/browser/ui/ash/system_tray_tray_cast_browsertest_media_router_chromeos.cc
index 78729c7..8c82cfe 100644
--- a/chrome/browser/ui/ash/system_tray_tray_cast_browsertest_media_router_chromeos.cc
+++ b/chrome/browser/ui/ash/system_tray_tray_cast_browsertest_media_router_chromeos.cc
@@ -131,8 +131,7 @@
   }
 
   media_router::MediaRoutesObserver* media_routes_observer() const {
-    DCHECK(media_routes_observer_);
-    return media_routes_observer_;
+    return &(*media_router_->routes_observers().begin());
   }
 
  private:
@@ -149,9 +148,6 @@
     ON_CALL(*media_router_, RegisterMediaSinksObserver(_))
         .WillByDefault(Invoke(
             this, &SystemTrayTrayCastMediaRouterChromeOSTest::CaptureSink));
-    ON_CALL(*media_router_, RegisterMediaRoutesObserver(_))
-        .WillByDefault(Invoke(
-            this, &SystemTrayTrayCastMediaRouterChromeOSTest::CaptureRoutes));
     CastConfigControllerMediaRouter::SetMediaRouterForTest(media_router_.get());
     InProcessBrowserTest::PreRunTestOnMainThread();
   }
@@ -171,13 +167,8 @@
     return true;
   }
 
-  void CaptureRoutes(media_router::MediaRoutesObserver* media_routes_observer) {
-    media_routes_observer_ = media_routes_observer;
-  }
-
   std::unique_ptr<media_router::MockMediaRouter> media_router_;
   media_router::MediaSinksObserver* media_sinks_observer_ = nullptr;
-  media_router::MediaRoutesObserver* media_routes_observer_ = nullptr;
   std::unique_ptr<ash::SystemTrayTestApi> tray_test_api_;
 };
 
diff --git a/chrome/browser/ui/views/bookmarks/bookmark_bar_view.cc b/chrome/browser/ui/views/bookmarks/bookmark_bar_view.cc
index 7e90858..6c2e7da1 100644
--- a/chrome/browser/ui/views/bookmarks/bookmark_bar_view.cc
+++ b/chrome/browser/ui/views/bookmarks/bookmark_bar_view.cc
@@ -318,9 +318,7 @@
   }
 
   std::u16string GetTooltipText(const gfx::Point& p) const override {
-    return label()->GetPreferredSize().width() > label()->size().width()
-               ? GetAccessibleName()
-               : std::u16string();
+    return GetAccessibleName();
   }
 
   bool OnMousePressed(const ui::MouseEvent& event) override {
diff --git a/chrome/browser/ui/views/bookmarks/saved_tab_groups/saved_tab_group_bar.cc b/chrome/browser/ui/views/bookmarks/saved_tab_groups/saved_tab_group_bar.cc
index 6c91bc8..cfba815 100644
--- a/chrome/browser/ui/views/bookmarks/saved_tab_groups/saved_tab_group_bar.cc
+++ b/chrome/browser/ui/views/bookmarks/saved_tab_groups/saved_tab_group_bar.cc
@@ -81,8 +81,10 @@
   SavedTabGroupRemoved(removed_group->saved_guid());
 }
 
-void SavedTabGroupBar::SavedTabGroupUpdatedLocally(const base::GUID& guid) {
-  SavedTabGroupUpdated(guid);
+void SavedTabGroupBar::SavedTabGroupUpdatedLocally(
+    const base::GUID& group_guid,
+    const absl::optional<base::GUID>& tab_guid) {
+  SavedTabGroupUpdated(group_guid);
 }
 
 void SavedTabGroupBar::SavedTabGroupReorderedLocally() {
@@ -104,8 +106,10 @@
   SavedTabGroupRemoved(removed_group->saved_guid());
 }
 
-void SavedTabGroupBar::SavedTabGroupUpdatedFromSync(const base::GUID& guid) {
-  SavedTabGroupUpdated(guid);
+void SavedTabGroupBar::SavedTabGroupUpdatedFromSync(
+    const base::GUID& group_guid,
+    const absl::optional<base::GUID>& tab_guid) {
+  SavedTabGroupUpdated(group_guid);
 }
 
 int SavedTabGroupBar::CalculatePreferredWidthRestrictedBy(int max_x) {
diff --git a/chrome/browser/ui/views/bookmarks/saved_tab_groups/saved_tab_group_bar.h b/chrome/browser/ui/views/bookmarks/saved_tab_groups/saved_tab_group_bar.h
index c7eb73c..db9100c 100644
--- a/chrome/browser/ui/views/bookmarks/saved_tab_groups/saved_tab_group_bar.h
+++ b/chrome/browser/ui/views/bookmarks/saved_tab_groups/saved_tab_group_bar.h
@@ -11,6 +11,7 @@
 #include "components/saved_tab_groups/saved_tab_group_model.h"
 #include "components/saved_tab_groups/saved_tab_group_model_observer.h"
 #include "content/public/browser/page.h"
+#include "third_party/abseil-cpp/absl/types/optional.h"
 #include "ui/views/accessible_pane_view.h"
 
 class Browser;
@@ -47,12 +48,16 @@
   // SavedTabGroupModelObserver
   void SavedTabGroupAddedLocally(const base::GUID& guid) override;
   void SavedTabGroupRemovedLocally(const SavedTabGroup* removed_group) override;
-  void SavedTabGroupUpdatedLocally(const base::GUID& guid) override;
+  void SavedTabGroupUpdatedLocally(
+      const base::GUID& group_guid,
+      const absl::optional<base::GUID>& tab_guid = absl::nullopt) override;
   void SavedTabGroupReorderedLocally() override;
   void SavedTabGroupAddedFromSync(const base::GUID& guid) override;
   void SavedTabGroupRemovedFromSync(
       const SavedTabGroup* removed_group) override;
-  void SavedTabGroupUpdatedFromSync(const base::GUID& guid) override;
+  void SavedTabGroupUpdatedFromSync(
+      const base::GUID& group_guid,
+      const absl::optional<base::GUID>& tab_guid = absl::nullopt) override;
 
   // Calculates what the visible width would be when a restriction on width is
   // placed on the bar.
diff --git a/chrome/browser/ui/views/download/bubble/download_bubble_row_view.cc b/chrome/browser/ui/views/download/bubble/download_bubble_row_view.cc
index a0f7a9a5..e7ab662 100644
--- a/chrome/browser/ui/views/download/bubble/download_bubble_row_view.cc
+++ b/chrome/browser/ui/views/download/bubble/download_bubble_row_view.cc
@@ -136,6 +136,17 @@
     GetViewAccessibility().AnnounceText(alert_text);
   }
 
+  // When in progress, announce the progress immediately and start the timer for
+  // further updates.
+  if (state == download::DownloadItem::IN_PROGRESS &&
+      (initial_setup || state_ != state)) {
+    AnnounceInProgressAlert();
+    accessible_alert_in_progress_timer_.Reset();
+  } else if (!initial_setup && state_ == download::DownloadItem::IN_PROGRESS &&
+             state != state_) {
+    accessible_alert_in_progress_timer_.Stop();
+  }
+
   mode_ = mode;
   state_ = state;
   is_paused_ = is_paused;
@@ -285,7 +296,12 @@
       navigation_handler_(navigation_handler),
       browser_(browser),
       inkdrop_container_(
-          AddChildView(std::make_unique<views::InkDropContainerView>())) {
+          AddChildView(std::make_unique<views::InkDropContainerView>())),
+      accessible_alert_in_progress_timer_(
+          FROM_HERE,
+          base::Minutes(3),
+          base::BindRepeating(&DownloadBubbleRowView::AnnounceInProgressAlert,
+                              base::Unretained(this))) {
   model_->SetDelegate(this);
   SetBorder(views::CreateEmptyBorder(GetLayoutInsets(DOWNLOAD_ROW)));
 
@@ -569,8 +585,9 @@
 void DownloadBubbleRowView::OnMainButtonPressed() {
   if (ui_info_.has_subpage) {
     DownloadItemWarningData::AddWarningActionEvent(
-        model_->GetDownloadItem(), DownloadItemWarningData::BUBBLE_MAINPAGE,
-        DownloadItemWarningData::OPEN_SUBPAGE);
+        model_->GetDownloadItem(),
+        DownloadItemWarningData::WarningSurface::BUBBLE_MAINPAGE,
+        DownloadItemWarningData::WarningAction::OPEN_SUBPAGE);
     navigation_handler_->OpenSecurityDialog(this);
   } else {
     RecordDownloadOpenButtonPressed(model_->IsDone());
@@ -677,8 +694,9 @@
 void DownloadBubbleRowView::RecordDownloadDisplayed() {
   if (model_->IsDangerous()) {
     DownloadItemWarningData::AddWarningActionEvent(
-        model_->GetDownloadItem(), DownloadItemWarningData::BUBBLE_MAINPAGE,
-        DownloadItemWarningData::SHOWN);
+        model_->GetDownloadItem(),
+        DownloadItemWarningData::WarningSurface::BUBBLE_MAINPAGE,
+        DownloadItemWarningData::WarningAction::SHOWN);
   }
   if (!model_->GetEphemeralWarningUiShownTime().has_value() &&
       model_->IsEphemeralWarning()) {
@@ -821,5 +839,10 @@
                      base::RepeatingClosure());
 }
 
+void DownloadBubbleRowView::AnnounceInProgressAlert() {
+  GetViewAccessibility().AnnounceText(
+      model_->GetInProgressAccessibleAlertText());
+}
+
 BEGIN_METADATA(DownloadBubbleRowView, views::View)
 END_METADATA
diff --git a/chrome/browser/ui/views/download/bubble/download_bubble_row_view.h b/chrome/browser/ui/views/download/bubble/download_bubble_row_view.h
index c137677..c9b94589 100644
--- a/chrome/browser/ui/views/download/bubble/download_bubble_row_view.h
+++ b/chrome/browser/ui/views/download/bubble/download_bubble_row_view.h
@@ -127,6 +127,8 @@
   void OnDiscardButtonPressed();
   void OnMainButtonPressed();
 
+  void AnnounceInProgressAlert();
+
   // The icon for the file. We get platform-specific icons from IconLoader.
   raw_ptr<views::ImageView> icon_ = nullptr;
   raw_ptr<views::ImageView> subpage_icon_ = nullptr;
@@ -210,6 +212,9 @@
   // avoid inaccurate repeated logging.
   bool has_download_completion_been_logged_ = false;
 
+  // A timer for accessible alerts of progress updates
+  base::RepeatingTimer accessible_alert_in_progress_timer_;
+
   base::WeakPtrFactory<DownloadBubbleRowView> weak_factory_{this};
 };
 
diff --git a/chrome/browser/ui/views/download/bubble/download_bubble_security_view.cc b/chrome/browser/ui/views/download/bubble/download_bubble_security_view.cc
index a97ace7..ab7c9a0 100644
--- a/chrome/browser/ui/views/download/bubble/download_bubble_security_view.cc
+++ b/chrome/browser/ui/views/download/bubble/download_bubble_security_view.cc
@@ -99,7 +99,8 @@
 void DownloadBubbleSecurityView::BackButtonPressed() {
   DownloadItemWarningData::AddWarningActionEvent(
       download_row_view_->model()->GetDownloadItem(),
-      DownloadItemWarningData::BUBBLE_SUBPAGE, DownloadItemWarningData::BACK);
+      DownloadItemWarningData::WarningSurface::BUBBLE_SUBPAGE,
+      DownloadItemWarningData::WarningAction::BACK);
   navigation_handler_->OpenPrimaryDialog();
   base::UmaHistogramEnumeration(
       kSubpageActionHistogram, DownloadBubbleSubpageAction::kPressedBackButton);
@@ -114,7 +115,8 @@
 void DownloadBubbleSecurityView::CloseBubble() {
   DownloadItemWarningData::AddWarningActionEvent(
       download_row_view_->model()->GetDownloadItem(),
-      DownloadItemWarningData::BUBBLE_SUBPAGE, DownloadItemWarningData::CLOSE);
+      DownloadItemWarningData::WarningSurface::BUBBLE_SUBPAGE,
+      DownloadItemWarningData::WarningAction::CLOSE);
   // CloseDialog will delete the object. Do not access any members below.
   navigation_handler_->CloseDialog(
       views::Widget::ClosedReason::kCloseButtonClicked);
diff --git a/chrome/browser/ui/views/download/download_item_view.cc b/chrome/browser/ui/views/download/download_item_view.cc
index 75a714d2..1aece888 100644
--- a/chrome/browser/ui/views/download/download_item_view.cc
+++ b/chrome/browser/ui/views/download/download_item_view.cc
@@ -948,33 +948,7 @@
   if (model_->GetOpenWhenComplete() || has_warning_label(mode_))
     return accessible_name_;
 
-  // Prefer to announce the time remaining, if known.
-  base::TimeDelta remaining;
-  if (model_->TimeRemaining(&remaining)) {
-    // If complete, skip this round: a completion status update is coming soon.
-    if (remaining.is_zero())
-      return std::u16string();
-
-    const std::u16string remaining_string =
-        ui::TimeFormat::Simple(ui::TimeFormat::FORMAT_REMAINING,
-                               ui::TimeFormat::LENGTH_SHORT, remaining);
-    return l10n_util::GetStringFUTF16(
-        IDS_DOWNLOAD_STATUS_TIME_REMAINING_ACCESSIBLE_ALERT, remaining_string);
-  }
-
-  // Time remaining is unknown, try to announce percent remaining.
-  if (model_->PercentComplete() > 0) {
-    DCHECK_LE(model_->PercentComplete(), 100);
-    return l10n_util::GetStringFUTF16Int(
-        IDS_DOWNLOAD_STATUS_PERCENT_COMPLETE_ACCESSIBLE_ALERT,
-        100 - model_->PercentComplete());
-  }
-
-  // Percent remaining is also unknown, announce bytes to download.
-  return l10n_util::GetStringFUTF16(
-      IDS_DOWNLOAD_STATUS_IN_PROGRESS_ACCESSIBLE_ALERT,
-      ui::FormatBytes(model_->GetTotalBytes()),
-      model_->GetFileNameToReportUser().LossyDisplayName());
+  return model_->GetInProgressAccessibleAlertText();
 }
 
 void DownloadItemView::AnnounceAccessibleAlert() {
diff --git a/chrome/browser/ui/views/global_media_controls/media_dialog_view_interactive_browsertest.cc b/chrome/browser/ui/views/global_media_controls/media_dialog_view_interactive_browsertest.cc
index ddd9301..fba0a13 100644
--- a/chrome/browser/ui/views/global_media_controls/media_dialog_view_interactive_browsertest.cc
+++ b/chrome/browser/ui/views/global_media_controls/media_dialog_view_interactive_browsertest.cc
@@ -316,24 +316,13 @@
     return logger_.get();
   }
 
-  void RegisterMediaRoutesObserver(
-      media_router::MediaRoutesObserver* observer) override {
-    routes_observers_.push_back(observer);
-  }
-
-  void UnregisterMediaRoutesObserver(
-      media_router::MediaRoutesObserver* observer) override {
-    base::Erase(routes_observers_, observer);
-  }
-
   void NotifyMediaRoutesChanged(
       const std::vector<media_router::MediaRoute>& routes) {
-    for (auto* observer : routes_observers_)
-      observer->OnRoutesUpdated(routes);
+    for (auto& observer : routes_observers_)
+      observer.OnRoutesUpdated(routes);
   }
 
  private:
-  std::vector<media_router::MediaRoutesObserver*> routes_observers_;
   std::unique_ptr<media_router::LoggerImpl> logger_;
 };
 
diff --git a/chrome/browser/ui/views/page_info/safety_tip_page_info_bubble_view_browsertest.cc b/chrome/browser/ui/views/page_info/safety_tip_page_info_bubble_view_browsertest.cc
index f2ba822..4eec2f5c 100644
--- a/chrome/browser/ui/views/page_info/safety_tip_page_info_bubble_view_browsertest.cc
+++ b/chrome/browser/ui/views/page_info/safety_tip_page_info_bubble_view_browsertest.cc
@@ -269,6 +269,9 @@
 
     test_ukm_recorder_ = std::make_unique<ukm::TestAutoSetUkmRecorder>();
     SetUpLookalikeTestParams();
+    // Check that the test top domain list contains google.
+    ASSERT_TRUE(IsTopDomain(GetDomainInfo("google.com")));
+
     InProcessBrowserTest::SetUpOnMainThread();
   }
 
@@ -567,6 +570,10 @@
   NavigateToURL(browser(), kNavigatedUrl, WindowOpenDisposition::CURRENT_TAB);
   EXPECT_FALSE(IsUIShowing());
   ASSERT_NO_FATAL_FAILURE(CheckPageInfoDoesNotShowSafetyTipInfo(browser()));
+
+  // TODO(crbug.com/1401102): Only one UKM should have been recorded, but
+  // allowlisted domain also records one.
+  CheckRecordedHeuristicsUkmCount(2);
 }
 
 // Ensure sites allowed by enterprise policy don't get blocked.
@@ -584,11 +591,15 @@
     EXPECT_FALSE(IsUIShowing());
     ASSERT_NO_FATAL_FAILURE(CheckPageInfoDoesNotShowSafetyTipInfo(browser()));
   }
+
+  // TODO(crbug.com/1401102): This shouldn't record a UKM.
+  CheckRecordedHeuristicsUkmCount(2);
 }
 
 // After the user clicks 'leave site', the user should end up on a safe domain.
 IN_PROC_BROWSER_TEST_F(SafetyTipPageInfoBubbleViewBrowserTest,
                        LeaveSiteLeavesSite) {
+  ASSERT_TRUE(IsTopDomain(GetDomainInfo("google.sk")));
   // This domain is a lookalike of a top domain not in the top 500.
   const GURL kNavigatedUrl = GetURL("googlé.sk");
   SetEngagementScore(browser(), kNavigatedUrl, kLowEngagement);
@@ -753,6 +764,7 @@
                      browser()->tab_strip_model()->active_index() + 1);
   EXPECT_FALSE(IsUIShowing());
   ASSERT_NO_FATAL_FAILURE(CheckPageInfoDoesNotShowSafetyTipInfo(browser()));
+  CheckRecordedHeuristicsUkmCount(0);
 }
 
 // Tests that Safety Tips do NOT trigger on lookalike domains that trigger an
@@ -763,12 +775,14 @@
   SetEngagementScore(browser(), kNavigatedUrl, kLowEngagement);
   NavigateToURL(browser(), kNavigatedUrl, WindowOpenDisposition::CURRENT_TAB);
   EXPECT_FALSE(IsUIShowing());
+  CheckRecordedHeuristicsUkmCount(0);
 }
 
 // Tests that Safety Tips trigger on lookalike domains that don't qualify for an
-// interstitial, but do not impact Page Info.
+// interstitial and Page Info shows Safety Tip information.
 IN_PROC_BROWSER_TEST_F(SafetyTipPageInfoBubbleViewBrowserTest,
                        TriggersOnLookalike) {
+  ASSERT_TRUE(IsTopDomain(GetDomainInfo("google.sk")));
   // This domain is a lookalike of a top domain not in the top 500.
   const GURL kNavigatedUrl = GetURL("googlé.sk");
   SetEngagementScore(browser(), kNavigatedUrl, kLowEngagement);
@@ -778,12 +792,14 @@
   ASSERT_NO_FATAL_FAILURE(CheckPageInfoShowsSafetyTipInfo(
       browser(), security_state::SafetyTipStatus::kLookalike,
       GURL("https://google.sk")));
+  CheckRecordedHeuristicsUkmCount(0);
 }
 
 // Tests that Safety Tips don't trigger on lookalike domains that are explicitly
 // allowed by the allowlist.
 IN_PROC_BROWSER_TEST_F(SafetyTipPageInfoBubbleViewBrowserTest,
                        NoTriggersOnLookalikeAllowlist) {
+  ASSERT_TRUE(IsTopDomain(GetDomainInfo("google.sk")));
   // This domain is a lookalike of a top domain not in the top 500.
   const GURL kNavigatedUrl = GetURL("googlé.sk");
 
@@ -802,6 +818,10 @@
 
   EXPECT_FALSE(IsUIShowing());
   ASSERT_NO_FATAL_FAILURE(CheckPageInfoDoesNotShowSafetyTipInfo(browser()));
+
+  // TODO(crbug.com/1401102): Only one UKM should have been recorded, but
+  // allowlisted domain also records one.
+  CheckRecordedHeuristicsUkmCount(2);
 }
 
 // Tests that Safety Tips don't trigger on lookalike domains that are explicitly
@@ -833,6 +853,10 @@
   SetEngagementScore(browser(), kTargetUrl, kHighEngagement);
   NavigateToURL(browser(), kNavigatedUrl, WindowOpenDisposition::CURRENT_TAB);
   EXPECT_TRUE(IsUIShowing());
+  CheckRecordedHeuristicsUkmCount(0);
+
+  CloseWarningLeaveSite(browser());
+  CheckRecordedHeuristicsUkmCount(1);
 }
 
 // Tests that Safety Tips don't trigger when using a scoped allowlist.
@@ -847,6 +871,9 @@
   NavigateToURL(browser(), kNavigatedUrl, WindowOpenDisposition::CURRENT_TAB);
   EXPECT_FALSE(IsUIShowing());
   ASSERT_NO_FATAL_FAILURE(CheckPageInfoDoesNotShowSafetyTipInfo(browser()));
+
+  // TODO(crbug.com/1401102): This shouldn't record metrics.
+  CheckRecordedHeuristicsUkmCount(1);
 }
 
 // Tests that Safety Tips trigger when the URL is on the allowlist, but is
@@ -861,6 +888,10 @@
   SetEngagementScore(browser(), kTargetUrl, kHighEngagement);
   NavigateToURL(browser(), kNavigatedUrl, WindowOpenDisposition::CURRENT_TAB);
   EXPECT_TRUE(IsUIShowing());
+  CheckRecordedHeuristicsUkmCount(0);
+
+  CloseWarningLeaveSite(browser());
+  CheckRecordedHeuristicsUkmCount(1);
 }
 
 // Tests that Character Swap is enabled for lookalikes matching engaged sites.
@@ -872,6 +903,10 @@
   SetEngagementScore(browser(), kTargetUrl, kHighEngagement);
   NavigateToURL(browser(), kNavigatedUrl, WindowOpenDisposition::CURRENT_TAB);
   EXPECT_TRUE(IsUIShowing());
+  CheckRecordedHeuristicsUkmCount(0);
+
+  CloseWarningLeaveSite(browser());
+  CheckRecordedHeuristicsUkmCount(1);
 }
 
 // Same as TriggersOnCharacterSwap_SiteEngagement, but this time
@@ -886,11 +921,17 @@
   SetEngagementScore(browser(), kTargetUrl, kHighEngagement);
   NavigateToURL(browser(), kNavigatedUrl, WindowOpenDisposition::CURRENT_TAB);
   EXPECT_TRUE(IsUIShowing());
+  CheckRecordedHeuristicsUkmCount(0);
+
+  CloseWarningLeaveSite(browser());
+  CheckRecordedHeuristicsUkmCount(1);
 }
 
 // Tests that Character Swap is enabled for lookalikes matching top sites.
 IN_PROC_BROWSER_TEST_F(SafetyTipPageInfoBubbleViewBrowserTest,
                        TriggersOnCharacterSwap_TopSite) {
+  base::HistogramTester histograms;
+
   const GURL kNavigatedUrl = GetURL("goolge.com");
   const GURL kTargetUrl = GetURL("google.com");
   // Both the lookalike and the target have low engagement.
@@ -898,18 +939,28 @@
   SetEngagementScore(browser(), kTargetUrl, kLowEngagement);
   NavigateToURL(browser(), kNavigatedUrl, WindowOpenDisposition::CURRENT_TAB);
   EXPECT_TRUE(IsUIShowing());
+  CheckRecordedHeuristicsUkmCount(0);
+
+  CloseWarningLeaveSite(browser());
+  CheckRecordedHeuristicsUkmCount(1);
 }
 
 // Tests that a hostname on a safe TLD can spoof another hostname without a
 // lookalike warning.
 IN_PROC_BROWSER_TEST_F(SafetyTipPageInfoBubbleViewBrowserTest,
                        TriggersOnCharacterSwapSafeTLD_CanSpoof) {
+  base::HistogramTester histograms;
+
   const GURL kNavigatedUrl = GetURL("digitla.gov");
   const GURL kTargetUrl = GetURL("digital.gov");
   SetEngagementScore(browser(), kNavigatedUrl, kLowEngagement);
   SetEngagementScore(browser(), kTargetUrl, kHighEngagement);
   NavigateToURL(browser(), kNavigatedUrl, WindowOpenDisposition::CURRENT_TAB);
   EXPECT_FALSE(IsUIShowing());
+
+  histograms.ExpectTotalCount(lookalikes::kHistogramName, 0);
+  // TODO(crbug.com/1401102): This shouldn't record metrics.
+  CheckRecordedHeuristicsUkmCount(1);
 }
 
 // Navigate to a domain within a character swap of 1 to a top domain,
@@ -918,6 +969,8 @@
 IN_PROC_BROWSER_TEST_F(
     SafetyTipPageInfoBubbleViewBrowserTest,
     DoesntTriggerOnCharacterSwap_TopSiteWithDifferentRegistry) {
+  ASSERT_TRUE(IsTopDomain(GetDomainInfo("google.rs")));
+
   base::HistogramTester histograms;
   // google.sr is within one character swap of google.rs which is a top domain.
   const GURL kNavigatedUrl = GetURL("google.sr");
@@ -925,11 +978,11 @@
   // considered for lookalike suggestions.
   SetEngagementScore(browser(), kNavigatedUrl, kLowEngagement);
 
-  // TestInterstitialNotShown(browser(), kNavigatedUrl);
-  // histograms.ExpectTotalCount(lookalikes::kHistogramName, 0);
-  // CheckNoUkm();
   NavigateToURL(browser(), kNavigatedUrl, WindowOpenDisposition::CURRENT_TAB);
   EXPECT_FALSE(IsUIShowing());
+
+  histograms.ExpectTotalCount(lookalikes::kHistogramName, 0);
+  CheckRecordedHeuristicsUkmCount(0);
 }
 
 // Tests that Safety Tips trigger on lookalike domains with tail embedding when
@@ -1009,6 +1062,7 @@
   // the safety tip.
   histogram_tester.ExpectTotalCount(
       GetInteractionHistogram("SafetyTip_Lookalike"), 0);
+
   CloseWarningLeaveSite(browser());
   histogram_tester.ExpectUniqueSample(
       GetInteractionHistogram("SafetyTip_Lookalike"),
diff --git a/chrome/browser/ui/views/toolbar/toolbar_view.cc b/chrome/browser/ui/views/toolbar/toolbar_view.cc
index ee4e67a..dbe08461 100644
--- a/chrome/browser/ui/views/toolbar/toolbar_view.cc
+++ b/chrome/browser/ui/views/toolbar/toolbar_view.cc
@@ -791,13 +791,13 @@
     case AppMenuIconController::Severity::NONE:
       return GetColorProvider()->GetColor(kColorToolbarButtonIcon);
     case AppMenuIconController::Severity::LOW:
-      color_id = ui::kColorAlertLowSeverity;
+      color_id = kColorAppMenuHighlightSeverityLow;
       break;
     case AppMenuIconController::Severity::MEDIUM:
-      color_id = ui::kColorAlertMediumSeverity;
+      color_id = kColorAppMenuHighlightSeverityMedium;
       break;
     case AppMenuIconController::Severity::HIGH:
-      color_id = ui::kColorAlertHighSeverity;
+      color_id = kColorAppMenuHighlightSeverityHigh;
       break;
   }
   return GetColorProvider()->GetColor(color_id);
diff --git a/chrome/browser/ui/views/web_apps/web_app_integration_test_driver.cc b/chrome/browser/ui/views/web_apps/web_app_integration_test_driver.cc
index 4b5068c..8963382 100644
--- a/chrome/browser/ui/views/web_apps/web_app_integration_test_driver.cc
+++ b/chrome/browser/ui/views/web_apps/web_app_integration_test_driver.cc
@@ -1189,7 +1189,9 @@
     FilesOptions files_options,
     AllowDenyOptions allow_deny,
     AskAgainOptions ask_again) {
-  BeforeStateChangeAction(__FUNCTION__);
+  if (!BeforeStateChangeAction(__FUNCTION__)) {
+    return;
+  }
   AppId app_id = GetAppIdBySiteMode(site);
   views::NamedWidgetShownWaiter waiter(views::test::AnyWidgetTestPasskey{},
                                        "FileHandlerLaunchDialogView");
@@ -1229,7 +1231,9 @@
 void WebAppIntegrationTestDriver::LaunchFileExpectNoDialog(
     Site site,
     FilesOptions files_options) {
-  BeforeStateChangeAction(__FUNCTION__);
+  if (!BeforeStateChangeAction(__FUNCTION__)) {
+    return;
+  }
   AppId app_id = GetAppIdBySiteMode(site);
   base::RunLoop run_loop;
 #if BUILDFLAG(IS_MAC)
@@ -1787,7 +1791,9 @@
 }
 
 void WebAppIntegrationTestDriver::SwitchIncognitoProfile() {
-  BeforeStateChangeAction(__FUNCTION__);
+  if (!BeforeStateChangeAction(__FUNCTION__)) {
+    return;
+  }
   content::WebContentsAddedObserver nav_observer;
   CHECK(chrome::ExecuteCommand(browser(), IDC_NEW_INCOGNITO_WINDOW));
   ASSERT_EQ(1U, BrowserList::GetIncognitoBrowserCount());
@@ -2069,7 +2075,9 @@
 }
 
 void WebAppIntegrationTestDriver::CheckAppInListIconCorrect(Site site) {
-  BeforeStateCheckAction(__FUNCTION__);
+  if (!BeforeStateCheckAction(__FUNCTION__)) {
+    return;
+  }
   GURL icon_url =
       apps::AppIconSource::GetIconURL(active_app_id_, icon_size::k128);
   SkBitmap icon_bitmap;
@@ -2307,13 +2315,17 @@
 }
 
 void WebAppIntegrationTestDriver::CheckCreateShortcutNotShown() {
-  BeforeStateCheckAction(__FUNCTION__);
+  if (!BeforeStateCheckAction(__FUNCTION__)) {
+    return;
+  }
   EXPECT_EQ(GetAppMenuCommandState(IDC_CREATE_SHORTCUT, browser()), kDisabled);
   AfterStateCheckAction();
 }
 
 void WebAppIntegrationTestDriver::CheckCreateShortcutShown() {
-  BeforeStateCheckAction(__FUNCTION__);
+  if (!BeforeStateCheckAction(__FUNCTION__)) {
+    return;
+  }
   EXPECT_EQ(GetAppMenuCommandState(IDC_CREATE_SHORTCUT, browser()), kEnabled);
   AfterStateCheckAction();
 }
diff --git a/chrome/browser/ui/views/web_apps/web_app_integration_test_driver.h b/chrome/browser/ui/views/web_apps/web_app_integration_test_driver.h
index 169b9e6..1f7759c 100644
--- a/chrome/browser/ui/views/web_apps/web_app_integration_test_driver.h
+++ b/chrome/browser/ui/views/web_apps/web_app_integration_test_driver.h
@@ -337,12 +337,12 @@
  private:
   // Must be called at the beginning of every state change action function.
   // Returns if the test should continue.
-  bool BeforeStateChangeAction(const char* function);
+  [[nodiscard]] bool BeforeStateChangeAction(const char* function);
   // Must be called at the end of every state change action function.
   void AfterStateChangeAction();
   // Must be called at the beginning of every state check action function.
   // Returns if the test should continue.
-  bool BeforeStateCheckAction(const char* function);
+  [[nodiscard]] bool BeforeStateCheckAction(const char* function);
   // Must be called at the end of every state check action function.
   void AfterStateCheckAction();
 
diff --git a/chrome/browser/ui/views/webauthn/sheet_view_factory.cc b/chrome/browser/ui/views/webauthn/sheet_view_factory.cc
index 48241ec..2e31c04 100644
--- a/chrome/browser/ui/views/webauthn/sheet_view_factory.cc
+++ b/chrome/browser/ui/views/webauthn/sheet_view_factory.cc
@@ -136,6 +136,10 @@
           std::make_unique<AuthenticatorNoAvailableTransportsErrorModel>(
               dialog_model));
       break;
+    case Step::kErrorNoPasskeys:
+      sheet_view = std::make_unique<AuthenticatorRequestSheetView>(
+          std::make_unique<AuthenticatorNoPasskeysErrorModel>(dialog_model));
+      break;
     case Step::kTimedOut:
       sheet_view = std::make_unique<AuthenticatorRequestSheetView>(
           std::make_unique<AuthenticatorTimeoutErrorModel>(dialog_model));
diff --git a/chrome/browser/ui/webauthn/sheet_models.cc b/chrome/browser/ui/webauthn/sheet_models.cc
index 953b0af..03961d46 100644
--- a/chrome/browser/ui/webauthn/sheet_models.cc
+++ b/chrome/browser/ui/webauthn/sheet_models.cc
@@ -337,6 +337,36 @@
       IDS_WEBAUTHN_ERROR_NO_TRANSPORTS_DESCRIPTION);
 }
 
+// AuthenticatorNoPasskeysErrorModel ------------------------------------------
+
+bool AuthenticatorNoPasskeysErrorModel::IsBackButtonVisible() const {
+  return false;
+}
+
+std::u16string AuthenticatorNoPasskeysErrorModel::GetCancelButtonLabel() const {
+  return l10n_util::GetStringUTF16(IDS_CLOSE);
+}
+
+const gfx::VectorIcon& AuthenticatorNoPasskeysErrorModel::GetStepIllustration(
+    ImageColorScheme color_scheme) const {
+  if (base::FeatureList::IsEnabled(
+          device::kWebAuthnNewDiscoverableCredentialsUi)) {
+    return color_scheme == ImageColorScheme::kDark ? kPasskeyErrorDarkIcon
+                                                   : kPasskeyErrorIcon;
+  }
+  return color_scheme == ImageColorScheme::kDark ? kWebauthnErrorDarkIcon
+                                                 : kWebauthnErrorIcon;
+}
+
+std::u16string AuthenticatorNoPasskeysErrorModel::GetStepTitle() const {
+  return l10n_util::GetStringUTF16(IDS_WEBAUTHN_ERROR_NO_PASSKEYS_TITLE);
+}
+
+std::u16string AuthenticatorNoPasskeysErrorModel::GetStepDescription() const {
+  return l10n_util::GetStringFUTF16(IDS_WEBAUTHN_ERROR_NO_PASSKEYS_DESCRIPTION,
+                                    GetRelyingPartyIdString(dialog_model()));
+}
+
 // AuthenticatorNotRegisteredErrorModel ---------------------------------------
 
 bool AuthenticatorNotRegisteredErrorModel::IsBackButtonVisible() const {
diff --git a/chrome/browser/ui/webauthn/sheet_models.h b/chrome/browser/ui/webauthn/sheet_models.h
index a458e7f..8373c89 100644
--- a/chrome/browser/ui/webauthn/sheet_models.h
+++ b/chrome/browser/ui/webauthn/sheet_models.h
@@ -152,6 +152,20 @@
   std::u16string GetStepDescription() const override;
 };
 
+class AuthenticatorNoPasskeysErrorModel : public AuthenticatorSheetModelBase {
+ public:
+  using AuthenticatorSheetModelBase::AuthenticatorSheetModelBase;
+
+ private:
+  // AuthenticatorSheetModelBase:
+  bool IsBackButtonVisible() const override;
+  std::u16string GetCancelButtonLabel() const override;
+  const gfx::VectorIcon& GetStepIllustration(
+      ImageColorScheme color_scheme) const override;
+  std::u16string GetStepTitle() const override;
+  std::u16string GetStepDescription() const override;
+};
+
 class AuthenticatorNotRegisteredErrorModel
     : public AuthenticatorSheetModelBase {
  public:
diff --git a/chrome/browser/ui/webui/app_home/app_home_page_handler_browsertest.cc b/chrome/browser/ui/webui/app_home/app_home_page_handler_browsertest.cc
index e52dad0..01283fb9 100644
--- a/chrome/browser/ui/webui/app_home/app_home_page_handler_browsertest.cc
+++ b/chrome/browser/ui/webui/app_home/app_home_page_handler_browsertest.cc
@@ -212,7 +212,7 @@
     scoped_refptr<extensions::Extension> extension =
         extensions::Extension::Create(
             base::FilePath(), extensions::mojom::ManifestLocation::kUnpacked,
-            manifest, 0, &error);
+            manifest.GetDict(), 0, &error);
 
     extension_service()->AddExtension(extension.get());
     return extension;
diff --git a/chrome/browser/ui/webui/ash/lock_screen_reauth/lock_screen_network_ui.cc b/chrome/browser/ui/webui/ash/lock_screen_reauth/lock_screen_network_ui.cc
index 5c404fd..a6afd58 100644
--- a/chrome/browser/ui/webui/ash/lock_screen_reauth/lock_screen_network_ui.cc
+++ b/chrome/browser/ui/webui/ash/lock_screen_reauth/lock_screen_network_ui.cc
@@ -72,7 +72,7 @@
   content::WebUIDataSource* html =
       content::WebUIDataSource::Create(chrome::kChromeUILockScreenNetworkHost);
 
-  // TODO(crbug.com/1098690): Trusted Type Polymer.
+  // TODO(crbug.com/1400799): Enable TrustedTypes
   html->DisableTrustedTypesCSP();
 
   html->AddLocalizedStrings(localized_strings);
diff --git a/chrome/browser/ui/webui/ash/lock_screen_reauth/lock_screen_start_reauth_ui.cc b/chrome/browser/ui/webui/ash/lock_screen_reauth/lock_screen_start_reauth_ui.cc
index 0f750dcd..15f75ca0 100644
--- a/chrome/browser/ui/webui/ash/lock_screen_reauth/lock_screen_start_reauth_ui.cc
+++ b/chrome/browser/ui/webui/ash/lock_screen_reauth/lock_screen_start_reauth_ui.cc
@@ -49,7 +49,7 @@
   web_ui->AddMessageHandler(std::move(main_handler));
   web_ui->AddMessageHandler(std::make_unique<MetricsHandler>());
 
-  // TODO(crbug.com/1098690): Trusted Type Polymer
+  // TODO(crbug.com/1400799): Enable TrustedTypes.
   source->DisableTrustedTypesCSP();
 
   source->EnableReplaceI18nInJS();
diff --git a/chrome/browser/ui/webui/chrome_url_data_manager_browsertest.cc b/chrome/browser/ui/webui/chrome_url_data_manager_browsertest.cc
index 9c930c56..e912fbd 100644
--- a/chrome/browser/ui/webui/chrome_url_data_manager_browsertest.cc
+++ b/chrome/browser/ui/webui/chrome_url_data_manager_browsertest.cc
@@ -189,15 +189,7 @@
 
     ASSERT_TRUE(embedded_test_server()->Start());
     ASSERT_TRUE(ui_test_utils::NavigateToURL(browser(), GURL(url)));
-
-    if (url == "chrome://network-error" || url == "chrome://dino") {
-      // We don't ASSERT_TRUE here because some WebUI pages are
-      // PAGE_TYPE_ERROR by design.
-      content::WaitForLoadStop(content);
-    } else {
-      ASSERT_TRUE(content::WaitForLoadStop(content));
-    }
-
+    ASSERT_TRUE(content::WaitForLoadStop(content));
     EXPECT_TRUE(console_observer.messages().empty());
   }
 
@@ -206,14 +198,7 @@
         browser()->tab_strip_model()->GetActiveWebContents();
     ASSERT_TRUE(embedded_test_server()->Start());
     ASSERT_TRUE(ui_test_utils::NavigateToURL(browser(), GURL(url)));
-
-    if (url == "chrome://network-error" || url == "chrome://dino") {
-      // We don't ASSERT_TRUE here because some WebUI pages are
-      // PAGE_TYPE_ERROR by design.
-      content::WaitForLoadStop(content);
-    } else {
-      ASSERT_TRUE(content::WaitForLoadStop(content));
-    }
+    ASSERT_TRUE(content::WaitForLoadStop(content));
 
     const char kIsTrustedTypesEnabled[] =
         "(function isTrustedTypesEnabled() {"
@@ -260,17 +245,11 @@
 // Non-exhaustive list of chrome:// URLs to test for
 //  1) TrustedTypes violations (see NoTrustedTypesViolation test).
 //  2) Presence of TrustedTypes checks (see TrustedTypesEnabled test).
-// This list was derived from chrome://about. :)
-//
-// TODO(crbug.com/1098690): For all commented out URLs below, uncomment after
-// enabling TrustedTypes, unless they are commented out for another reason
-// (like flaky failures).
 static constexpr const char* const kChromeUrls[] = {
     "chrome://accessibility",
     "chrome://app-service-internals",
     "chrome://attribution-internals",
     "chrome://autofill-internals",
-    // "chrome://blob-internals",
     "chrome://bookmarks",
     "chrome://bookmarks-side-panel.top-chrome",
     "chrome://chrome-urls",
@@ -281,10 +260,7 @@
     "chrome://crashes",
     "chrome://credits",
     "chrome://customize-chrome-side-panel.top-chrome",
-    // TODO(crbug/1396866): Enable when bug is fixed.
-    // "chrome://customize-chrome-side-panel.top-chrome",
     "chrome://device-log",
-    // "chrome://dino",
     // TODO(crbug.com/1113446): Test failure due to excessive output.
     // "chrome://discards",
     "chrome://download-internals",
@@ -316,7 +292,6 @@
     // "chrome://memory-internals",
     "chrome://net-export",
     "chrome://net-internals",
-    // "chrome://network-error",
     "chrome://network-errors",
     "chrome://new-tab-page",
     "chrome://new-tab-page-third-party",
@@ -354,7 +329,6 @@
     // "chrome://tab-strip",
     "chrome://terms",
     "chrome://topics-internals",
-    // "chrome://tracing",
     "chrome://translate-internals",
     "chrome://ukm",
     "chrome://usb-internals",
@@ -381,38 +355,23 @@
 #endif
 
 #if BUILDFLAG(IS_CHROMEOS_ASH)
-    // "chrome://account-manager-error",
-    // "chrome://account-migration-welcome",
-    // "chrome://add-supervision",
+    // TODO(crbug.com/1400799): Add CrOS-only WebUI URLs here as TrustedTypes
+    // are deployed to more WebUIs.
+
     // TODO(crbug.com/1102129): DCHECK failure in
     // ArcGraphicsTracingHandler::ArcGraphicsTracingHandler.
     // "chrome://arc-graphics-tracing",
-    // "chrome://arc-overview-tracing",
-    // "chrome://assistant-optin",
-    // "chrome://bluetooth-pairing",
-    // "chrome://certificate-manager",
-    // "chrome://crostini-credits",
-    // "chrome://crostini-installer",
     "chrome://cryptohome",
     "chrome://drive-internals",
     "chrome://family-link-user-internals",
     "chrome://help-app",
-    // "chrome://internet-config-dialog",
-    // "chrome://internet-detail-dialog",
     "chrome://linux-proxy-config",
-    // "chrome://multidevice-setup",
-    // "chrome://network",
-    // "chrome://os-credits",
-    // "chrome://os-settings",
+    "chrome://multidevice-internals",
+    "chrome://nearby-internals",
     "chrome://power",
     "chrome://projector",
     "chrome://proximity-auth/proximity_auth.html",
-    // "chrome://set-time",
     "chrome://slow",
-    // "chrome://smb-credentials-dialog",
-    // "chrome://smb-share-dialog",
-    // "chrome://sys-internals",
-    // "chrome-untrusted://terminal",
 #endif
 #if !BUILDFLAG(IS_CHROMEOS)
     "chrome://apps",
diff --git a/chrome/browser/ui/webui/downloads/downloads_dom_handler.cc b/chrome/browser/ui/webui/downloads/downloads_dom_handler.cc
index 7cd951a..0d131dc 100644
--- a/chrome/browser/ui/webui/downloads/downloads_dom_handler.cc
+++ b/chrome/browser/ui/webui/downloads/downloads_dom_handler.cc
@@ -179,8 +179,8 @@
   download::DownloadItem* file = GetDownloadByStringId(id);
   if (file) {
     DownloadItemWarningData::AddWarningActionEvent(
-        file, DownloadItemWarningData::DOWNLOADS_PAGE,
-        DownloadItemWarningData::KEEP);
+        file, DownloadItemWarningData::WarningSurface::DOWNLOADS_PAGE,
+        DownloadItemWarningData::WarningAction::KEEP);
     ShowDangerPrompt(file);
   }
 }
@@ -192,8 +192,8 @@
     // The warning action event needs to be added before Safe Browsing report is
     // sent, because this event should be included in the report.
     DownloadItemWarningData::AddWarningActionEvent(
-        download, DownloadItemWarningData::DOWNLOADS_PAGE,
-        DownloadItemWarningData::DISCARD);
+        download, DownloadItemWarningData::WarningSurface::DOWNLOADS_PAGE,
+        DownloadItemWarningData::WarningAction::DISCARD);
     // If this download is no longer dangerous, is already canceled or
     // completed, don't send any report.
     // Only sends dangerous download discard report if :
diff --git a/chrome/browser/ui/webui/management/management_ui_browsertest.cc b/chrome/browser/ui/webui/management/management_ui_browsertest.cc
index f109a589..be8dbab 100644
--- a/chrome/browser/ui/webui/management/management_ui_browsertest.cc
+++ b/chrome/browser/ui/webui/management/management_ui_browsertest.cc
@@ -3,6 +3,7 @@
 // found in the LICENSE file.
 
 #include "base/json/json_reader.h"
+#include "base/strings/escape.h"
 #include "base/strings/utf_string_conversions.h"
 #include "build/chromeos_buildflags.h"
 #include "chrome/browser/policy/chrome_browser_policy_connector.h"
@@ -84,7 +85,9 @@
       {"browserManagementNotice",
        l10n_util::GetStringFUTF16(
            IDS_MANAGEMENT_NOT_MANAGED_NOTICE,
-           base::UTF8ToUTF16(chrome::kManagedUiLearnMoreUrl))},
+           base::UTF8ToUTF16(chrome::kManagedUiLearnMoreUrl),
+           base::EscapeForHTML(l10n_util::GetStringUTF16(
+               IDS_MANAGEMENT_LEARN_MORE_ACCCESSIBILITY_TEXT)))},
       {"extensionReportingTitle",
        l10n_util::GetStringUTF16(IDS_MANAGEMENT_EXTENSIONS_INSTALLED)},
       {"pageSubtitle",
@@ -117,7 +120,9 @@
       {"browserManagementNotice",
        l10n_util::GetStringFUTF16(
            IDS_MANAGEMENT_BROWSER_NOTICE,
-           base::UTF8ToUTF16(chrome::kManagedUiLearnMoreUrl))},
+           base::UTF8ToUTF16(chrome::kManagedUiLearnMoreUrl),
+           base::EscapeForHTML(l10n_util::GetStringUTF16(
+               IDS_MANAGEMENT_LEARN_MORE_ACCCESSIBILITY_TEXT)))},
       {"extensionReportingTitle",
        l10n_util::GetStringUTF16(IDS_MANAGEMENT_EXTENSIONS_INSTALLED)},
       {"pageSubtitle", l10n_util::GetStringUTF16(IDS_MANAGEMENT_SUBTITLE)},
diff --git a/chrome/browser/ui/webui/management/management_ui_handler.cc b/chrome/browser/ui/webui/management/management_ui_handler.cc
index f611708..f8ed735 100644
--- a/chrome/browser/ui/webui/management/management_ui_handler.cc
+++ b/chrome/browser/ui/webui/management/management_ui_handler.cc
@@ -18,6 +18,7 @@
 #include "base/feature_list.h"
 #include "base/metrics/histogram_functions.h"
 #include "base/metrics/user_metrics.h"
+#include "base/strings/escape.h"
 #include "base/strings/string_util.h"
 #include "base/strings/utf_string_conversions.h"
 #include "base/values.h"
@@ -841,7 +842,9 @@
                l10n_util::GetStringFUTF16(
                    managed_() ? IDS_MANAGEMENT_BROWSER_NOTICE
                               : IDS_MANAGEMENT_NOT_MANAGED_NOTICE,
-                   base::UTF8ToUTF16(chrome::kManagedUiLearnMoreUrl)));
+                   base::UTF8ToUTF16(chrome::kManagedUiLearnMoreUrl),
+                   base::EscapeForHTML(l10n_util::GetStringUTF16(
+                       IDS_MANAGEMENT_LEARN_MORE_ACCCESSIBILITY_TEXT))));
 #endif
 
   if (enterprise_manager.empty()) {
diff --git a/chrome/browser/ui/webui/management/management_ui_handler_unittest.cc b/chrome/browser/ui/webui/management/management_ui_handler_unittest.cc
index 2c45f3b..378817b 100644
--- a/chrome/browser/ui/webui/management/management_ui_handler_unittest.cc
+++ b/chrome/browser/ui/webui/management/management_ui_handler_unittest.cc
@@ -13,6 +13,7 @@
 #include "base/json/json_reader.h"
 #include "base/memory/raw_ptr.h"
 #include "base/memory/scoped_refptr.h"
+#include "base/strings/escape.h"
 #include "base/strings/utf_string_conversions.h"
 #include "base/task/single_thread_task_runner.h"
 #include "build/chromeos_buildflags.h"
@@ -714,7 +715,9 @@
   EXPECT_EQ(GetBrowserManagementNotice(),
             l10n_util::GetStringFUTF16(
                 IDS_MANAGEMENT_NOT_MANAGED_NOTICE,
-                base::UTF8ToUTF16(chrome::kManagedUiLearnMoreUrl)));
+                base::UTF8ToUTF16(chrome::kManagedUiLearnMoreUrl),
+                base::EscapeForHTML(l10n_util::GetStringUTF16(
+                    IDS_MANAGEMENT_LEARN_MORE_ACCCESSIBILITY_TEXT))));
   EXPECT_EQ(GetPageSubtitle(),
             l10n_util::GetStringUTF16(IDS_MANAGEMENT_NOT_MANAGED_SUBTITLE));
 }
@@ -732,7 +735,9 @@
   EXPECT_EQ(GetBrowserManagementNotice(),
             l10n_util::GetStringFUTF16(
                 IDS_MANAGEMENT_BROWSER_NOTICE,
-                base::UTF8ToUTF16(chrome::kManagedUiLearnMoreUrl)));
+                base::UTF8ToUTF16(chrome::kManagedUiLearnMoreUrl),
+                base::EscapeForHTML(l10n_util::GetStringUTF16(
+                    IDS_MANAGEMENT_LEARN_MORE_ACCCESSIBILITY_TEXT))));
   EXPECT_EQ(GetPageSubtitle(),
             l10n_util::GetStringUTF16(IDS_MANAGEMENT_SUBTITLE));
   EXPECT_TRUE(GetManaged());
@@ -752,7 +757,9 @@
   EXPECT_EQ(GetBrowserManagementNotice(),
             l10n_util::GetStringFUTF16(
                 IDS_MANAGEMENT_BROWSER_NOTICE,
-                base::UTF8ToUTF16(chrome::kManagedUiLearnMoreUrl)));
+                base::UTF8ToUTF16(chrome::kManagedUiLearnMoreUrl),
+                base::EscapeForHTML(l10n_util::GetStringUTF16(
+                    IDS_MANAGEMENT_LEARN_MORE_ACCCESSIBILITY_TEXT))));
   EXPECT_EQ(GetPageSubtitle(),
             l10n_util::GetStringUTF16(IDS_MANAGEMENT_SUBTITLE));
   EXPECT_TRUE(GetManaged());
@@ -777,7 +784,9 @@
   EXPECT_EQ(GetBrowserManagementNotice(),
             l10n_util::GetStringFUTF16(
                 IDS_MANAGEMENT_NOT_MANAGED_NOTICE,
-                base::UTF8ToUTF16(chrome::kManagedUiLearnMoreUrl)));
+                base::UTF8ToUTF16(chrome::kManagedUiLearnMoreUrl),
+                base::EscapeForHTML(l10n_util::GetStringUTF16(
+                    IDS_MANAGEMENT_LEARN_MORE_ACCCESSIBILITY_TEXT))));
   EXPECT_EQ(GetPageSubtitle(),
             l10n_util::GetStringUTF16(IDS_MANAGEMENT_NOT_MANAGED_SUBTITLE));
   EXPECT_FALSE(GetManaged());
@@ -797,7 +806,9 @@
   EXPECT_EQ(GetBrowserManagementNotice(),
             l10n_util::GetStringFUTF16(
                 IDS_MANAGEMENT_NOT_MANAGED_NOTICE,
-                base::UTF8ToUTF16(chrome::kManagedUiLearnMoreUrl)));
+                base::UTF8ToUTF16(chrome::kManagedUiLearnMoreUrl),
+                base::EscapeForHTML(l10n_util::GetStringUTF16(
+                    IDS_MANAGEMENT_LEARN_MORE_ACCCESSIBILITY_TEXT))));
   EXPECT_EQ(GetPageSubtitle(),
             l10n_util::GetStringUTF16(IDS_MANAGEMENT_NOT_MANAGED_SUBTITLE));
   EXPECT_FALSE(GetManaged());
@@ -821,7 +832,9 @@
   EXPECT_EQ(GetBrowserManagementNotice(),
             l10n_util::GetStringFUTF16(
                 IDS_MANAGEMENT_BROWSER_NOTICE,
-                base::UTF8ToUTF16(chrome::kManagedUiLearnMoreUrl)));
+                base::UTF8ToUTF16(chrome::kManagedUiLearnMoreUrl),
+                base::EscapeForHTML(l10n_util::GetStringUTF16(
+                    IDS_MANAGEMENT_LEARN_MORE_ACCCESSIBILITY_TEXT))));
   EXPECT_EQ(GetPageSubtitle(),
             l10n_util::GetStringFUTF16(IDS_MANAGEMENT_SUBTITLE_MANAGED_BY,
                                        base::UTF8ToUTF16(domain)));
diff --git a/chrome/browser/ui/webui/side_panel/customize_chrome/customize_chrome.mojom b/chrome/browser/ui/webui/side_panel/customize_chrome/customize_chrome.mojom
index f28a707..183df02 100644
--- a/chrome/browser/ui/webui/side_panel/customize_chrome/customize_chrome.mojom
+++ b/chrome/browser/ui/webui/side_panel/customize_chrome/customize_chrome.mojom
@@ -11,6 +11,8 @@
 struct BackgroundImage {
   // URL to the background image. Can point to untrusted content.
   url.mojom.Url url;
+  // Whether the image is a local resource.
+  bool is_uploaded_image;
   // Title of the background image.
   string title;
 };
@@ -25,6 +27,8 @@
   skia.mojom.SkColor background_color;
   // The current theme foreground color. If not set, we use the default theme.
   skia.mojom.SkColor? foreground_color;
+  // The color of the color picker icon.
+  skia.mojom.SkColor color_picker_icon_color;
 };
 
 struct ChromeColor {
diff --git a/chrome/browser/ui/webui/side_panel/customize_chrome/customize_chrome_page_handler.cc b/chrome/browser/ui/webui/side_panel/customize_chrome/customize_chrome_page_handler.cc
index 1733c816..9a9b5107 100644
--- a/chrome/browser/ui/webui/side_panel/customize_chrome/customize_chrome_page_handler.cc
+++ b/chrome/browser/ui/webui/side_panel/customize_chrome/customize_chrome_page_handler.cc
@@ -103,6 +103,7 @@
   auto background_image = side_panel::mojom::BackgroundImage::New();
   if (custom_background.has_value()) {
     background_image->url = custom_background->custom_background_url;
+    background_image->is_uploaded_image = custom_background->is_uploaded_image;
     background_image->title =
         custom_background->custom_background_attribution_line_1;
   } else {
@@ -116,6 +117,8 @@
     theme->foreground_color =
         web_contents_->GetColorProvider().GetColor(ui::kColorFrameActive);
   }
+  theme->color_picker_icon_color =
+      web_contents_->GetColorProvider().GetColor(kColorNewTabPageText);
   auto* native_theme = ui::NativeTheme::GetInstanceForNativeUi();
   CHECK(native_theme);
   theme->system_dark_mode = native_theme->ShouldUseDarkColors();
diff --git a/chrome/browser/ui/webui/side_panel/customize_chrome/customize_chrome_page_handler_unittest.cc b/chrome/browser/ui/webui/side_panel/customize_chrome/customize_chrome_page_handler_unittest.cc
index 7a2b33e..df9063e 100644
--- a/chrome/browser/ui/webui/side_panel/customize_chrome/customize_chrome_page_handler_unittest.cc
+++ b/chrome/browser/ui/webui/side_panel/customize_chrome/customize_chrome_page_handler_unittest.cc
@@ -270,6 +270,7 @@
   CustomBackground custom_background;
   custom_background.custom_background_url = GURL("https://foo.com/img.png");
   custom_background.custom_background_attribution_line_1 = "foo line";
+  custom_background.is_uploaded_image = false;
   ON_CALL(mock_ntp_custom_background_service_, GetCustomBackground())
       .WillByDefault(testing::Return(absl::make_optional(custom_background)));
   ON_CALL(mock_theme_service(), UsingDefaultTheme())
@@ -284,6 +285,7 @@
   ASSERT_TRUE(theme);
   ASSERT_TRUE(theme->background_image);
   EXPECT_EQ("https://foo.com/img.png", theme->background_image->url);
+  ASSERT_FALSE(theme->background_image->is_uploaded_image);
   EXPECT_EQ("foo line", theme->background_image->title);
   EXPECT_TRUE(theme->system_dark_mode);
   EXPECT_EQ(
@@ -291,6 +293,34 @@
       theme->background_color);
   EXPECT_EQ(web_contents().GetColorProvider().GetColor(ui::kColorFrameActive),
             theme->foreground_color);
+  EXPECT_EQ(web_contents().GetColorProvider().GetColor(kColorNewTabPageText),
+            theme->color_picker_icon_color);
+}
+
+TEST_P(CustomizeChromePageHandlerSetThemeTest, SetUploadedImage) {
+  side_panel::mojom::ThemePtr theme;
+  EXPECT_CALL(mock_page_, SetTheme)
+      .Times(1)
+      .WillOnce(testing::Invoke([&theme](side_panel::mojom::ThemePtr arg) {
+        theme = std::move(arg);
+      }));
+  CustomBackground custom_background;
+  custom_background.custom_background_url = GURL("https://foo.com/img.png");
+  custom_background.is_uploaded_image = true;
+  ON_CALL(mock_ntp_custom_background_service_, GetCustomBackground())
+      .WillByDefault(testing::Return(absl::make_optional(custom_background)));
+  ON_CALL(mock_theme_service(), UsingDefaultTheme())
+      .WillByDefault(testing::Return(false));
+  ON_CALL(mock_theme_service(), UsingSystemTheme())
+      .WillByDefault(testing::Return(false));
+
+  UpdateTheme();
+  mock_page_.FlushForTesting();
+
+  ASSERT_TRUE(theme);
+  ASSERT_TRUE(theme->background_image);
+  EXPECT_EQ("https://foo.com/img.png", theme->background_image->url);
+  ASSERT_TRUE(theme->background_image->is_uploaded_image);
 }
 
 INSTANTIATE_TEST_SUITE_P(All,
diff --git a/chrome/browser/ui/webui/side_panel/customize_chrome/customize_chrome_ui.cc b/chrome/browser/ui/webui/side_panel/customize_chrome/customize_chrome_ui.cc
index 9d2b08b5..160e331c 100644
--- a/chrome/browser/ui/webui/side_panel/customize_chrome/customize_chrome_ui.cc
+++ b/chrome/browser/ui/webui/side_panel/customize_chrome/customize_chrome_ui.cc
@@ -30,6 +30,8 @@
 
   static constexpr webui::LocalizedString kLocalizedStrings[] = {
       {"classicChrome", IDS_NTP_CUSTOMIZE_NO_BACKGROUND_LABEL},
+      {"colorsContainerLabel", IDS_NTP_THEMES_CONTAINER_LABEL},
+      {"colorPickerLabel", IDS_NTP_CUSTOMIZE_COLOR_PICKER_LABEL},
       {"customizeThisPage", IDS_NTP_CUSTOM_BG_CUSTOMIZE_NTP_LABEL},
       {"appearanceHeader", IDS_NTP_CUSTOMIZE_APPEARANCE_LABEL},
       {"defaultColorName", IDS_NTP_CUSTOMIZE_DEFAULT_LABEL},
@@ -40,6 +42,7 @@
       {"shortcutsSuggested", IDS_NTP_CUSTOMIZE_MOST_VISITED_DESC},
       {"showShortcutsToggle", IDS_NTP_CUSTOMIZE_SHOW_SHORTCUTS_LABEL},
       {"title", IDS_SIDE_PANEL_CUSTOMIZE_CHROME_TITLE},
+      {"uploadedImage", IDS_NTP_CUSTOMIZE_UPLOADED_IMAGE_LABEL},
   };
   source->AddLocalizedStrings(kLocalizedStrings);
 
diff --git a/chrome/browser/ui/webui/web_app_internals/web_app_internals_source.cc b/chrome/browser/ui/webui/web_app_internals/web_app_internals_source.cc
index ceb3b878..da667a5 100644
--- a/chrome/browser/ui/webui/web_app_internals/web_app_internals_source.cc
+++ b/chrome/browser/ui/webui/web_app_internals/web_app_internals_source.cc
@@ -335,7 +335,6 @@
 #if BUILDFLAG(IS_MAC)
   root.Append(BuildAppShimRegistryLocalStorageJson());
 #endif
-  root.Append(BuildInstallProcessErrorLogJson(*provider));
   base::ThreadPool::PostTaskAndReplyWithResult(
       FROM_HERE, {base::TaskPriority::USER_VISIBLE, base::MayBlock()},
       base::BindOnce(&BuildWebAppDiskStateJson,
diff --git a/chrome/browser/web_applications/extensions/web_app_policy_manager_unittest.cc b/chrome/browser/web_applications/extensions/web_app_policy_manager_unittest.cc
index fe92068..54e98d5 100644
--- a/chrome/browser/web_applications/extensions/web_app_policy_manager_unittest.cc
+++ b/chrome/browser/web_applications/extensions/web_app_policy_manager_unittest.cc
@@ -537,6 +537,8 @@
   if (ShouldSkipPWASpecificTest())
     return;
 
+  base::RunLoop().RunUntilIdle();
+
   const auto& install_requests =
       externally_managed_app_manager().install_requests();
   EXPECT_TRUE(install_requests.empty());
@@ -918,7 +920,6 @@
 TEST_P(WebAppPolicyManagerTest, UninstallAppInstalledInPreviousSession) {
   if (ShouldSkipPWASpecificTest())
     return;
-
   // Simulate two policy apps and a regular app that were installed in the
   // previous session.
   SimulatePreviouslyInstalledApp(GURL(kWindowedUrl),
@@ -952,6 +953,7 @@
 TEST_P(WebAppPolicyManagerTest, UninstallAppInstalledInCurrentSession) {
   if (ShouldSkipPWASpecificTest())
     return;
+  base::RunLoop().RunUntilIdle();
 
   // Add two sites, one that opens in a window and one that opens in a tab.
   base::Value::List first_list;
@@ -1103,7 +1105,7 @@
 TEST_P(WebAppPolicyManagerTest, SayRefreshTwoTimesQuickly) {
   if (ShouldSkipPWASpecificTest())
     return;
-
+  base::RunLoop().RunUntilIdle();
   // Add an app.
   {
     base::Value::List list;
@@ -1118,17 +1120,7 @@
     profile()->GetPrefs()->SetList(prefs::kWebAppInstallForceList,
                                    std::move(list));
   }
-
-  // `OnAppsSynchronized` should be triggered twice.
-  base::RunLoop loop;
-  policy_manager().SetOnAppsSynchronizedCompletedCallbackForTesting(
-      loop.QuitClosure());
-  loop.Run();
-
-  base::RunLoop loop2;
-  policy_manager().SetOnAppsSynchronizedCompletedCallbackForTesting(
-      loop2.QuitClosure());
-  loop2.Run();
+  base::RunLoop().RunUntilIdle();
 
   // Both apps should have been installed.
   std::vector<ExternalInstallOptions> expected_options_list;
@@ -1193,6 +1185,8 @@
 
 #if BUILDFLAG(IS_CHROMEOS_ASH)
 TEST_P(WebAppPolicyManagerTest, DisableSystemWebApps) {
+  base::RunLoop().RunUntilIdle();
+
   auto disabled_apps = policy_manager().GetDisabledSystemWebApps();
   EXPECT_TRUE(disabled_apps.empty());
 
diff --git a/chrome/browser/web_applications/externally_managed_app_manager.cc b/chrome/browser/web_applications/externally_managed_app_manager.cc
index f382c6b..5c2844c 100644
--- a/chrome/browser/web_applications/externally_managed_app_manager.cc
+++ b/chrome/browser/web_applications/externally_managed_app_manager.cc
@@ -15,8 +15,6 @@
 #include "base/stl_util.h"
 #include "base/task/single_thread_task_runner.h"
 #include "build/chromeos_buildflags.h"
-#include "chrome/browser/web_applications/locks/full_system_lock.h"
-#include "chrome/browser/web_applications/web_app_command_scheduler.h"
 #include "chrome/browser/web_applications/web_app_install_utils.h"
 #include "chrome/browser/web_applications/web_app_registrar.h"
 #include "components/webapps/browser/install_result_code.h"
@@ -90,6 +88,7 @@
     std::vector<ExternalInstallOptions> desired_apps_install_options,
     ExternalInstallSource install_source,
     SynchronizeCallback callback) {
+  DCHECK(registrar_);
   DCHECK(base::ranges::all_of(
       desired_apps_install_options,
       [&install_source](const ExternalInstallOptions& install_options) {
@@ -98,29 +97,14 @@
   // Only one concurrent SynchronizeInstalledApps() expected per
   // ExternalInstallSource.
   DCHECK(!base::Contains(synchronize_requests_, install_source));
-  command_scheduler_->ScheduleCallbackWithLock<FullSystemLock>(
-      "ExternallyManagedAppManager::SynchronizeInstalledApps",
-      std::make_unique<FullSystemLockDescription>(),
-      base::BindOnce(
-          &ExternallyManagedAppManager::SynchronizeInstalledAppsOnLockAcquired,
-          weak_ptr_factory_.GetWeakPtr(),
-          std::move(desired_apps_install_options), std::move(install_source),
-          std::move(callback)));
-}
 
-void ExternallyManagedAppManager::SynchronizeInstalledAppsOnLockAcquired(
-    std::vector<ExternalInstallOptions> desired_apps_install_options,
-    ExternalInstallSource install_source,
-    SynchronizeCallback callback,
-    FullSystemLock& lock) {
   std::vector<GURL> installed_urls;
   for (const auto& apps_it :
-       lock.registrar().GetExternallyInstalledApps(install_source)) {
+       registrar_->GetExternallyInstalledApps(install_source)) {
     // TODO(crbug.com/1339965): Remove this check once we cleanup
     // ExternallyInstalledWebAppPrefs on external app uninstall.
     bool has_same_external_source =
-        lock.registrar()
-            .GetAppById(apps_it.first)
+        registrar_->GetAppById(apps_it.first)
             ->GetSources()
             .test(ConvertExternalInstallSourceToSource(install_source));
     if (has_same_external_source) {
diff --git a/chrome/browser/web_applications/externally_managed_app_manager.h b/chrome/browser/web_applications/externally_managed_app_manager.h
index 8e15a41..1b7eb9d0 100644
--- a/chrome/browser/web_applications/externally_managed_app_manager.h
+++ b/chrome/browser/web_applications/externally_managed_app_manager.h
@@ -24,7 +24,6 @@
 
 namespace web_app {
 
-class FullSystemLock;
 class WebAppRegistrar;
 class WebAppInstallFinalizer;
 class WebAppCommandScheduler;
@@ -191,12 +190,6 @@
     std::map<GURL, bool> uninstall_results;
   };
 
-  void SynchronizeInstalledAppsOnLockAcquired(
-      std::vector<ExternalInstallOptions> desired_apps_install_options,
-      ExternalInstallSource install_source,
-      SynchronizeCallback callback,
-      FullSystemLock& lock);
-
   void InstallForSynchronizeCallback(
       ExternalInstallSource source,
       const GURL& install_url,
diff --git a/chrome/browser/web_applications/externally_managed_app_manager_impl.cc b/chrome/browser/web_applications/externally_managed_app_manager_impl.cc
index 2228de8..73a6064 100644
--- a/chrome/browser/web_applications/externally_managed_app_manager_impl.cc
+++ b/chrome/browser/web_applications/externally_managed_app_manager_impl.cc
@@ -12,12 +12,9 @@
 #include "base/bind.h"
 #include "base/callback_helpers.h"
 #include "base/feature_list.h"
-#include "base/functional/bind.h"
 #include "base/task/single_thread_task_runner.h"
 #include "chrome/browser/profiles/profile.h"
-#include "chrome/browser/web_applications/externally_managed_app_manager.h"
 #include "chrome/browser/web_applications/externally_managed_app_registration_task.h"
-#include "chrome/browser/web_applications/locks/full_system_lock.h"
 #include "chrome/browser/web_applications/web_app.h"
 #include "chrome/browser/web_applications/web_app_command_scheduler.h"
 #include "chrome/browser/web_applications/web_app_install_finalizer.h"
@@ -161,19 +158,6 @@
 void ExternallyManagedAppManagerImpl::MaybeStartNext() {
   if (current_install_ || is_in_shutdown_)
     return;
-  command_scheduler()->ScheduleCallbackWithLock<FullSystemLock>(
-      "ExternallyManagedAppManagerImpl::MaybeStartNext",
-      std::make_unique<FullSystemLockDescription>(),
-      base::BindOnce(
-          &ExternallyManagedAppManagerImpl::MaybeStartNextOnLockAcquired,
-          weak_ptr_factory_.GetWeakPtr()));
-}
-
-void ExternallyManagedAppManagerImpl::MaybeStartNextOnLockAcquired(
-    FullSystemLock& lock) {
-  if (current_install_ || is_in_shutdown_) {
-    return;
-  }
 
   while (!pending_installs_.empty()) {
     std::unique_ptr<TaskAndCallback> front =
@@ -189,7 +173,7 @@
     }
 
     absl::optional<AppId> app_id =
-        lock.registrar().LookupExternalAppId(install_options.install_url);
+        registrar()->LookupExternalAppId(install_options.install_url);
 
     // If the URL is not in web_app registrar,
     // then no external source has installed it.
@@ -198,10 +182,10 @@
       return;
     }
 
-    if (lock.registrar().IsInstalled(app_id.value())) {
+    if (registrar()->IsInstalled(app_id.value())) {
       if (install_options.wait_for_windows_closed &&
-          lock.ui_manager().GetNumWindowsForApp(app_id.value()) != 0) {
-        lock.ui_manager().NotifyOnAllAppWindowsClosed(
+          ui_manager()->GetNumWindowsForApp(app_id.value()) != 0) {
+        ui_manager()->NotifyOnAllAppWindowsClosed(
             app_id.value(),
             base::BindOnce(&ExternallyManagedAppManagerImpl::Install,
                            weak_ptr_factory_.GetWeakPtr(), install_options,
@@ -212,9 +196,9 @@
       // If the app is already installed, only reinstall it if the app is a
       // placeholder app and the client asked for it to be reinstalled.
       if (install_options.reinstall_placeholder &&
-          lock.registrar().IsPlaceholderApp(
-              app_id.value(), ConvertExternalInstallSourceToSource(
-                                  install_options.install_source))) {
+          registrar()->IsPlaceholderApp(app_id.value(),
+                                        ConvertExternalInstallSourceToSource(
+                                            install_options.install_source))) {
         StartInstallationTask(std::move(front));
         return;
       }
@@ -227,7 +211,7 @@
         return;
       } else {
         // Add install source before returning the result.
-        ScopedRegistryUpdate update(&lock.sync_bridge());
+        ScopedRegistryUpdate update(sync_bridge());
         WebApp* app_to_update = update->UpdateApp(app_id.value());
         app_to_update->AddSource(ConvertExternalInstallSourceToSource(
             install_options.install_source));
diff --git a/chrome/browser/web_applications/externally_managed_app_manager_impl.h b/chrome/browser/web_applications/externally_managed_app_manager_impl.h
index 794125722..9fe18dd 100644
--- a/chrome/browser/web_applications/externally_managed_app_manager_impl.h
+++ b/chrome/browser/web_applications/externally_managed_app_manager_impl.h
@@ -27,7 +27,6 @@
 
 namespace web_app {
 
-class FullSystemLock;
 class ExternallyManagedAppRegistrationTaskBase;
 
 // Installs, uninstalls, and updates any External Web Apps. This class should
@@ -75,7 +74,6 @@
   void PostMaybeStartNext();
 
   void MaybeStartNext();
-  void MaybeStartNextOnLockAcquired(FullSystemLock& lock);
 
   void StartInstallationTask(std::unique_ptr<TaskAndCallback> task);
 
diff --git a/chrome/browser/web_applications/externally_managed_app_manager_impl_unittest.cc b/chrome/browser/web_applications/externally_managed_app_manager_impl_unittest.cc
index 1ca55fcf..596ca26 100644
--- a/chrome/browser/web_applications/externally_managed_app_manager_impl_unittest.cc
+++ b/chrome/browser/web_applications/externally_managed_app_manager_impl_unittest.cc
@@ -466,7 +466,7 @@
   }
 
   void TearDown() override {
-    provider().Shutdown();
+    command_scheduler().Shutdown();
     WebAppTest::TearDown();
   }
 
@@ -510,6 +510,7 @@
               barrier_closure.Run();
             }));
     run_loop.Run();
+
     return results;
   }
 
diff --git a/chrome/browser/web_applications/isolated_web_apps/OWNERS b/chrome/browser/web_applications/isolated_web_apps/OWNERS
new file mode 100644
index 0000000..5c9ed56
--- /dev/null
+++ b/chrome/browser/web_applications/isolated_web_apps/OWNERS
@@ -0,0 +1,2 @@
+cmfcmf@chromium.org
+rmcelrath@chromium.org
diff --git a/chrome/browser/web_applications/policy/web_app_policy_manager.cc b/chrome/browser/web_applications/policy/web_app_policy_manager.cc
index 4893400..83ccc9a 100644
--- a/chrome/browser/web_applications/policy/web_app_policy_manager.cc
+++ b/chrome/browser/web_applications/policy/web_app_policy_manager.cc
@@ -118,11 +118,7 @@
   system_web_apps_delegate_map_ = system_web_apps_delegate_map;
 }
 
-void WebAppPolicyManager::Start(base::OnceClosure initialization_complete) {
-  DCHECK(initialization_complete_.is_null());
-
-  initialization_complete_ = std::move(initialization_complete);
-
+void WebAppPolicyManager::Start(base::OnceClosure on_done) {
   // When Lacros is enabled, don't run PWA-specific logic in Ash.
   // TODO(crbug.com/1251491): Consider factoring out logic that should only run
   // in Ash into a separate class. This way, when running in Ash, we won't need
@@ -135,7 +131,8 @@
       ->PostTask(FROM_HERE,
                  base::BindOnce(
                      &WebAppPolicyManager::InitChangeRegistrarAndRefreshPolicy,
-                     weak_ptr_factory_.GetWeakPtr(), enable_pwa_support));
+                     weak_ptr_factory_.GetWeakPtr(), enable_pwa_support)
+                     .Then(std::move(on_done)));
 }
 
 void WebAppPolicyManager::ReinstallPlaceholderAppIfNecessary(const GURL& url) {
@@ -207,10 +204,6 @@
             weak_ptr_factory_.GetWeakPtr()));
     RefreshPolicyInstalledIsolatedWebApps();
 #endif
-  } else {
-    if (initialization_complete_) {
-      std::move(initialization_complete_).Run();
-    }
   }
   ObserveDisabledSystemFeaturesPolicy();
 }
@@ -545,7 +538,7 @@
 
 void WebAppPolicyManager::SetOnAppsSynchronizedCompletedCallbackForTesting(
     base::OnceClosure callback) {
-  on_apps_synchronized_for_testing_ = std::move(callback);
+  on_apps_synchronized_ = std::move(callback);
 }
 
 void WebAppPolicyManager::SetRefreshPolicySettingsCompletedCallbackForTesting(
@@ -638,13 +631,8 @@
                                   url_and_result.second.code);
   }
 
-  if (on_apps_synchronized_for_testing_) {
-    std::move(on_apps_synchronized_for_testing_).Run();
-  }
-
-  if (initialization_complete_) {
-    std::move(initialization_complete_).Run();
-  }
+  if (on_apps_synchronized_)
+    std::move(on_apps_synchronized_).Run();
 }
 
 WebAppPolicyManager::WebAppSetting::WebAppSetting() {
diff --git a/chrome/browser/web_applications/policy/web_app_policy_manager.h b/chrome/browser/web_applications/policy/web_app_policy_manager.h
index 8bf8375..6ea29f7 100644
--- a/chrome/browser/web_applications/policy/web_app_policy_manager.h
+++ b/chrome/browser/web_applications/policy/web_app_policy_manager.h
@@ -65,8 +65,6 @@
   void SetSystemWebAppDelegateMap(
       const ash::SystemWebAppDelegateMap* system_web_apps_delegate_map);
 
-  // `initialization_complete` waits for the first `SynchronizeInstalledApps` to
-  // finish if it's triggered on `Start`.
   void Start(base::OnceClosure initialization_complete);
 
   void ReinstallPlaceholderAppIfNecessary(const GURL& url);
@@ -192,7 +190,7 @@
 
   // Testing callbacks
   base::OnceClosure refresh_policy_settings_completed_;
-  base::OnceClosure on_apps_synchronized_for_testing_;
+  base::OnceClosure on_apps_synchronized_;
 
   bool is_refreshing_ = false;
   bool needs_refresh_ = false;
@@ -202,9 +200,6 @@
   std::unique_ptr<WebAppSetting> default_settings_;
 
   ExternallyInstalledWebAppPrefs externally_installed_app_prefs_;
-
-  base::OnceClosure initialization_complete_;
-
 #if BUILDFLAG(IS_CHROMEOS)
   std::unique_ptr<IsolatedWebAppPolicyManager> iwa_policy_manager_;
 #endif
diff --git a/chrome/browser/webauthn/authenticator_request_dialog_model.cc b/chrome/browser/webauthn/authenticator_request_dialog_model.cc
index 651b7ab..9cfccebe 100644
--- a/chrome/browser/webauthn/authenticator_request_dialog_model.cc
+++ b/chrome/browser/webauthn/authenticator_request_dialog_model.cc
@@ -245,7 +245,12 @@
     SetCurrentStep(*pending_step_);
     pending_step_.reset();
   } else if (mechanisms_.empty()) {
-    SetCurrentStep(Step::kErrorNoAvailableTransports);
+    if (base::FeatureList::IsEnabled(device::kWebAuthnNoPasskeysError) &&
+        transport_availability_.transport_list_did_include_internal) {
+      SetCurrentStep(Step::kErrorNoPasskeys);
+    } else {
+      SetCurrentStep(Step::kErrorNoAvailableTransports);
+    }
   } else if (mechanisms_.size() == 1) {
     mechanisms_[0].callback.Run();
   } else if (priority_mechanism_it != mechanisms_.end()) {
diff --git a/chrome/browser/webauthn/authenticator_request_dialog_model.h b/chrome/browser/webauthn/authenticator_request_dialog_model.h
index 2bcd8fdd..3a5986da4 100644
--- a/chrome/browser/webauthn/authenticator_request_dialog_model.h
+++ b/chrome/browser/webauthn/authenticator_request_dialog_model.h
@@ -73,6 +73,7 @@
     // The request errored out before completing. Error will only be sent
     // after user interaction.
     kErrorNoAvailableTransports,
+    kErrorNoPasskeys,
     kErrorInternalUnrecognized,
 
     // The request is already complete, but the error dialog should wait
diff --git a/chrome/build/linux.pgo.txt b/chrome/build/linux.pgo.txt
index 8d4b0b0..2dec125 100644
--- a/chrome/build/linux.pgo.txt
+++ b/chrome/build/linux.pgo.txt
@@ -1 +1 @@
-chrome-linux-main-1671019116-6b9cae0d3e8c73277c9f214478366172a511d31a.profdata
+chrome-linux-main-1671040780-7354adbba216d8d22545e6cbff01983a0ef786cc.profdata
diff --git a/chrome/build/mac-arm.pgo.txt b/chrome/build/mac-arm.pgo.txt
index 3178b67b..846237e3 100644
--- a/chrome/build/mac-arm.pgo.txt
+++ b/chrome/build/mac-arm.pgo.txt
@@ -1 +1 @@
-chrome-mac-arm-main-1670997253-a4bdf1b52f9c04eb5ecbbf26461b8564531d27cc.profdata
+chrome-mac-arm-main-1671040780-ab587c46ae159837cb1901b47db9f541feb1ec6f.profdata
diff --git a/chrome/build/win32.pgo.txt b/chrome/build/win32.pgo.txt
index d401ef2..7dd319db 100644
--- a/chrome/build/win32.pgo.txt
+++ b/chrome/build/win32.pgo.txt
@@ -1 +1 @@
-chrome-win32-main-1671008076-19a9c8909deb1a4f7619d99c4ef88613e515e068.profdata
+chrome-win32-main-1671029899-f511e3be18f418e934aad5640a3a4197391f4ec1.profdata
diff --git a/chrome/build/win64.pgo.txt b/chrome/build/win64.pgo.txt
index 0d62531af..5106b24 100644
--- a/chrome/build/win64.pgo.txt
+++ b/chrome/build/win64.pgo.txt
@@ -1 +1 @@
-chrome-win64-main-1671029899-ba84905e9939d7394e959eb36f6891b3bd5d0d8c.profdata
+chrome-win64-main-1671040780-fa17ea6b4718dc5c5320f88590d164f43c2aa427.profdata
diff --git a/chrome/common/pref_names.cc b/chrome/common/pref_names.cc
index 314d831b..f3e79a7 100644
--- a/chrome/common/pref_names.cc
+++ b/chrome/common/pref_names.cc
@@ -3478,13 +3478,21 @@
 
 #if BUILDFLAG(IS_CHROMEOS_ASH)
 // Enum pref indicating how to launch the Lacros browser. It is managed by
-// LacrosAvailability policy can have one of the following values:
+// LacrosAvailability policy and can have one of the following values:
 // 0: User choice (default value).
 // 1: Lacros is disallowed.
 // 2: Lacros is enabled but not the pimary browser.
 // 3: Lacros is enabled as the primary browser.
 // 4: Lacros is the only available browser.
 const char kLacrosLaunchSwitch[] = "lacros_launch_switch";
+
+// Enum pref indicating which Lacros browser to launch: rootfs or stateful. It
+// is managed by LacrosSelection policy and can have one of the following
+// values:
+// 0: User choice (default value).
+// 1: Always load rootfs Lacros.
+// 2: Always load stateful Lacros.
+const char kLacrosSelection[] = "lacros_selection";
 #endif
 
 #if BUILDFLAG(IS_CHROMEOS_ASH)
diff --git a/chrome/common/pref_names.h b/chrome/common/pref_names.h
index e3b12b2..6230f1a 100644
--- a/chrome/common/pref_names.h
+++ b/chrome/common/pref_names.h
@@ -1208,6 +1208,7 @@
 
 #if BUILDFLAG(IS_CHROMEOS_ASH)
 extern const char kLacrosLaunchSwitch[];
+extern const char kLacrosSelection[];
 #endif
 
 #if BUILDFLAG(IS_CHROMEOS_ASH)
diff --git a/chrome/test/BUILD.gn b/chrome/test/BUILD.gn
index 6a1f4147..ca11bd4 100644
--- a/chrome/test/BUILD.gn
+++ b/chrome/test/BUILD.gn
@@ -8618,6 +8618,7 @@
       "../browser/enterprise/idle/action_runner_unittest.cc",
       "../browser/enterprise/idle/action_unittest.cc",
       "../browser/enterprise/idle/browser_closer_unittest.cc",
+      "../browser/enterprise/idle/idle_timeout_policy_handler_unittest.cc",
       "../browser/enterprise/remote_commands/rotate_attestation_credential_job_unittest.cc",
       "../browser/enterprise/signals/user_delegate_impl_unittest.cc",
       "../browser/net/disk_cache_dir_policy_handler_unittest.cc",
diff --git a/chrome/test/data/policy/policy_test_cases.json b/chrome/test/data/policy/policy_test_cases.json
index 99e44fae7..033b3fb 100644
--- a/chrome/test/data/policy/policy_test_cases.json
+++ b/chrome/test/data/policy/policy_test_cases.json
@@ -18779,6 +18779,56 @@
       }
     ]
   },
+  "LacrosSelection": {
+    "os": [
+      "chromeos_ash"
+    ],
+    "can_be_recommended": false,
+    "policy_pref_mapping_tests": [
+      {
+        "policies": {},
+        "prefs": {
+          "lacros_selection": {
+            "location": "local_state",
+            "value": 1
+          }
+        }
+      },
+      {
+        "policies": {
+          "LacrosSelection": "user_choice"
+        },
+        "prefs": {
+          "lacros_selection": {
+            "location": "local_state",
+            "value": 0
+          }
+        }
+      },
+      {
+        "policies": {
+          "LacrosSelection": "rootfs"
+        },
+        "prefs": {
+          "lacros_selection": {
+            "location": "local_state",
+            "value": 1
+          }
+        }
+      },
+      {
+        "policies": {
+          "LacrosSelection": "stateful"
+        },
+        "prefs": {
+          "lacros_selection": {
+            "location": "local_state",
+            "value": 2
+          }
+        }
+      }
+    ]
+  },
   "LacrosDataBackwardMigrationMode": {
     "os": [
       "chromeos_ash"
@@ -20603,18 +20653,18 @@
         },
         "prefs": {
           "idle_timeout": {
-            "value": 10
+            "value": "600000000"
           }
         }
       },
       {
         "policies": {
-          "IdleTimeout": 1,
+          "IdleTimeout": 0,
           "IdleTimeoutActions": []
         },
         "prefs": {
           "idle_timeout": {
-            "value": 5
+            "value": "60000000"
           }
         }
       }
diff --git a/chrome/test/data/webui/BUILD.gn b/chrome/test/data/webui/BUILD.gn
index 5dabfeb2..f941d99 100644
--- a/chrome/test/data/webui/BUILD.gn
+++ b/chrome/test/data/webui/BUILD.gn
@@ -169,12 +169,12 @@
         "chromeos/emoji_picker/emoji_picker_browsertest.js",
         "chromeos/gaia_action_buttons/gaia_action_buttons_browsertest.js",
         "chromeos/internet_detail_dialog_browsertest.js",
+        "chromeos/set_time_dialog_browsertest.js",
+        "chromeos/sys_internals/sys_internals_browsertest.js",
         "cr_components/chromeos/cr_components_chromeos_v3_browsertest.js",
         "js/i18n_process_test.js",
-        "set_time_dialog_browsertest.js",
         "settings/chromeos/a11y/v3_os_a11y_browsertest.js",
         "settings/chromeos/os_settings_v3_browsertest.js",
-        "sys_internals/sys_internals_browsertest.js",
       ]
     } else {
       sources += [ "signin/signin_browsertest.js" ]
@@ -461,16 +461,16 @@
 
   if (is_chromeos_ash) {
     input_files += [
+      "chromeos/arc_account_picker/test_util.js",
+      "chromeos/ash_common/i18n_behavior_test.js",
       "chromeos/chai_assert.js",
+      "chromeos/fake_network_config_mojom.js",
       "chromeos/mock_controller.js",
       "chromeos/mock_controller.m.js",
+      "chromeos/set_time_dialog_test.js",
       "chromeos/test_browser_proxy.js",
       "chromeos/test_store.js",
       "chromeos/test_util.js",
-      "chromeos/arc_account_picker/test_util.js",
-      "chromeos/ash_common/i18n_behavior_test.js",
-      "chromeos/fake_network_config_mojom.js",
-      "set_time_dialog_test.js",
     ]
   }
 
diff --git a/chrome/test/data/webui/set_time_dialog_browsertest.js b/chrome/test/data/webui/chromeos/set_time_dialog_browsertest.js
similarity index 85%
rename from chrome/test/data/webui/set_time_dialog_browsertest.js
rename to chrome/test/data/webui/chromeos/set_time_dialog_browsertest.js
index 0767e75f..2f466b9 100644
--- a/chrome/test/data/webui/set_time_dialog_browsertest.js
+++ b/chrome/test/data/webui/chromeos/set_time_dialog_browsertest.js
@@ -10,7 +10,7 @@
 var SetTimeDialogBrowserTest = class extends PolymerTest {
   /** @override */
   get browsePreload() {
-    return 'chrome://set-time/test_loader.html?module=set_time_dialog_test.js';
+    return 'chrome://set-time/test_loader.html?module=chromeos/set_time_dialog_test.js';
   }
 };
 
diff --git a/chrome/test/data/webui/set_time_dialog_test.js b/chrome/test/data/webui/chromeos/set_time_dialog_test.js
similarity index 99%
rename from chrome/test/data/webui/set_time_dialog_test.js
rename to chrome/test/data/webui/chromeos/set_time_dialog_test.js
index 4f19c3e9..6b6ede7 100644
--- a/chrome/test/data/webui/set_time_dialog_test.js
+++ b/chrome/test/data/webui/chromeos/set_time_dialog_test.js
@@ -7,7 +7,6 @@
 import {loadTimeData} from 'chrome://resources/js/load_time_data.js';
 import {flush} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js';
 import {SetTimeBrowserProxyImpl} from 'chrome://set-time/set_time_browser_proxy.js';
-
 import {TestBrowserProxy} from 'chrome://webui-test/test_browser_proxy.js';
 
 suite('SetTimeDialog', function() {
diff --git a/chrome/test/data/webui/sys_internals/all_tests.js b/chrome/test/data/webui/chromeos/sys_internals/all_tests.js
similarity index 100%
rename from chrome/test/data/webui/sys_internals/all_tests.js
rename to chrome/test/data/webui/chromeos/sys_internals/all_tests.js
diff --git a/chrome/test/data/webui/sys_internals/api_test.js b/chrome/test/data/webui/chromeos/sys_internals/api_test.js
similarity index 100%
rename from chrome/test/data/webui/sys_internals/api_test.js
rename to chrome/test/data/webui/chromeos/sys_internals/api_test.js
diff --git a/chrome/test/data/webui/sys_internals/line_chart/data_series_test.js b/chrome/test/data/webui/chromeos/sys_internals/line_chart/data_series_test.js
similarity index 100%
rename from chrome/test/data/webui/sys_internals/line_chart/data_series_test.js
rename to chrome/test/data/webui/chromeos/sys_internals/line_chart/data_series_test.js
diff --git a/chrome/test/data/webui/sys_internals/line_chart/line_chart_test.js b/chrome/test/data/webui/chromeos/sys_internals/line_chart/line_chart_test.js
similarity index 100%
rename from chrome/test/data/webui/sys_internals/line_chart/line_chart_test.js
rename to chrome/test/data/webui/chromeos/sys_internals/line_chart/line_chart_test.js
diff --git a/chrome/test/data/webui/sys_internals/line_chart/menu_test.js b/chrome/test/data/webui/chromeos/sys_internals/line_chart/menu_test.js
similarity index 100%
rename from chrome/test/data/webui/sys_internals/line_chart/menu_test.js
rename to chrome/test/data/webui/chromeos/sys_internals/line_chart/menu_test.js
diff --git a/chrome/test/data/webui/sys_internals/line_chart/scrollbar_test.js b/chrome/test/data/webui/chromeos/sys_internals/line_chart/scrollbar_test.js
similarity index 100%
rename from chrome/test/data/webui/sys_internals/line_chart/scrollbar_test.js
rename to chrome/test/data/webui/chromeos/sys_internals/line_chart/scrollbar_test.js
diff --git a/chrome/test/data/webui/sys_internals/line_chart/sub_chart_test.js b/chrome/test/data/webui/chromeos/sys_internals/line_chart/sub_chart_test.js
similarity index 100%
rename from chrome/test/data/webui/sys_internals/line_chart/sub_chart_test.js
rename to chrome/test/data/webui/chromeos/sys_internals/line_chart/sub_chart_test.js
diff --git a/chrome/test/data/webui/sys_internals/line_chart/unit_label_test.js b/chrome/test/data/webui/chromeos/sys_internals/line_chart/unit_label_test.js
similarity index 100%
rename from chrome/test/data/webui/sys_internals/line_chart/unit_label_test.js
rename to chrome/test/data/webui/chromeos/sys_internals/line_chart/unit_label_test.js
diff --git a/chrome/test/data/webui/sys_internals/page_drawer_test.js b/chrome/test/data/webui/chromeos/sys_internals/page_drawer_test.js
similarity index 100%
rename from chrome/test/data/webui/sys_internals/page_drawer_test.js
rename to chrome/test/data/webui/chromeos/sys_internals/page_drawer_test.js
diff --git a/chrome/test/data/webui/sys_internals/page_infopage_test.js b/chrome/test/data/webui/chromeos/sys_internals/page_infopage_test.js
similarity index 100%
rename from chrome/test/data/webui/sys_internals/page_infopage_test.js
rename to chrome/test/data/webui/chromeos/sys_internals/page_infopage_test.js
diff --git a/chrome/test/data/webui/sys_internals/page_switch_test.js b/chrome/test/data/webui/chromeos/sys_internals/page_switch_test.js
similarity index 100%
rename from chrome/test/data/webui/sys_internals/page_switch_test.js
rename to chrome/test/data/webui/chromeos/sys_internals/page_switch_test.js
diff --git a/chrome/test/data/webui/sys_internals/page_unit_test.js b/chrome/test/data/webui/chromeos/sys_internals/page_unit_test.js
similarity index 100%
rename from chrome/test/data/webui/sys_internals/page_unit_test.js
rename to chrome/test/data/webui/chromeos/sys_internals/page_unit_test.js
diff --git a/chrome/test/data/webui/sys_internals/sys_internals_browsertest.js b/chrome/test/data/webui/chromeos/sys_internals/sys_internals_browsertest.js
similarity index 94%
rename from chrome/test/data/webui/sys_internals/sys_internals_browsertest.js
rename to chrome/test/data/webui/chromeos/sys_internals/sys_internals_browsertest.js
index f05096f..60d011f0 100644
--- a/chrome/test/data/webui/sys_internals/sys_internals_browsertest.js
+++ b/chrome/test/data/webui/chromeos/sys_internals/sys_internals_browsertest.js
@@ -18,7 +18,7 @@
   __proto__: testing.Test.prototype,
 
   browsePreload:
-      'chrome://sys-internals/index.html?module=sys_internals/all_tests.js&host=test',
+      'chrome://sys-internals/index.html?module=chromeos/sys_internals/all_tests.js&host=test',
 
   isAsync: true,
 
diff --git a/chrome/test/data/webui/sys_internals/test_util.js b/chrome/test/data/webui/chromeos/sys_internals/test_util.js
similarity index 100%
rename from chrome/test/data/webui/sys_internals/test_util.js
rename to chrome/test/data/webui/chromeos/sys_internals/test_util.js
diff --git a/chrome/test/data/webui/settings/chromeos/text_to_speech_page_tests.js b/chrome/test/data/webui/settings/chromeos/text_to_speech_page_tests.js
index 782fe7b..15a07159 100644
--- a/chrome/test/data/webui/settings/chromeos/text_to_speech_page_tests.js
+++ b/chrome/test/data/webui/settings/chromeos/text_to_speech_page_tests.js
@@ -4,32 +4,24 @@
 
 import 'chrome://os-settings/chromeos/lazy_load.js';
 
-import {Router, routes} from 'chrome://os-settings/chromeos/os_settings.js';
+import {CrSettingsPrefs, Router, routes} from 'chrome://os-settings/chromeos/os_settings.js';
 import {flush} from 'chrome://resources/polymer/v3_0/polymer/polymer_bundled.min.js';
+import {assertEquals, assertFalse, assertTrue} from 'chrome://webui-test/chai_assert.js';
 import {waitAfterNextRender, waitBeforeNextRender} from 'chrome://webui-test/polymer_test_util.js';
 import {eventToPromise, isVisible} from 'chrome://webui-test/test_util.js';
 
-import {assertEquals, assertFalse, assertTrue} from 'chrome://webui-test/chai_assert.js';
-
 suite('TextToSpeechPageTests', function() {
   let page = null;
 
-  function initPage(opt_prefs) {
-    page = document.createElement('settings-text-to-speech-page');
-    page.prefs = opt_prefs || getDefaultPrefs();
-    document.body.appendChild(page);
-  }
+  async function initPage() {
+    const prefElement = document.createElement('settings-prefs');
+    document.body.appendChild(prefElement);
 
-  function getDefaultPrefs() {
-    return {
-      'settings': {
-        'accessibility': {
-          key: 'settings.accessibility',
-          type: chrome.settingsPrivate.PrefType.BOOLEAN,
-          value: false,
-        },
-      },
-    };
+    await CrSettingsPrefs.initialized;
+    page = document.createElement('settings-text-to-speech-page');
+    page.prefs = prefElement.prefs;
+    document.body.appendChild(page);
+    flush();
   }
 
   setup(function() {
@@ -50,8 +42,7 @@
         `should focus ${selector} button when returning from ${
             route.path} subpage`,
         async () => {
-          initPage();
-          flush();
+          await initPage();
           const router = Router.getInstance();
 
           const subpageButton = page.shadowRoot.querySelector(selector);
@@ -75,13 +66,12 @@
         });
   });
 
-  test('only allowed subpages are available in kiosk mode', function() {
+  test('only allowed subpages are available in kiosk mode', async function() {
     loadTimeData.overrideValues({
       isKioskModeActive: true,
       showTabletModeShelfNavigationButtonsSettings: true,
     });
-    initPage();
-    flush();
+    await initPage();
 
     const allowed_subpages = [
       'chromeVoxSubpageButton',
diff --git a/chrome/test/data/webui/side_panel/customize_chrome/color_test.ts b/chrome/test/data/webui/side_panel/customize_chrome/color_test.ts
index 5990693f..ebb12f6 100644
--- a/chrome/test/data/webui/side_panel/customize_chrome/color_test.ts
+++ b/chrome/test/data/webui/side_panel/customize_chrome/color_test.ts
@@ -5,6 +5,7 @@
 import 'chrome://webui-test/mojo_webui_test_support.js';
 
 import {ColorElement} from 'chrome://customize-chrome-side-panel.top-chrome/color.js';
+import {assertFalse, assertTrue} from 'chrome://webui-test/chai_assert.js';
 
 import {assertStyle} from './test_support.js';
 
@@ -24,4 +25,30 @@
     assertStyle(colorElement.$.background, 'fill', 'rgb(255, 0, 0)');
     assertStyle(colorElement.$.foreground, 'fill', 'rgb(0, 255, 0)');
   });
+
+  test('color can be checked', () => {
+    colorElement.checked = true;
+
+    const wrapper = colorElement.shadowRoot!.querySelector(
+        'customize-chrome-check-mark-wrapper')!;
+    assertTrue(wrapper.checked);
+    const svg = colorElement.shadowRoot!.querySelector('svg')!;
+    assertStyle(svg, 'width', '46px');
+    assertStyle(svg, 'height', '46px');
+    const background = colorElement.shadowRoot!.querySelector('#background')!;
+    assertStyle(background, 'r', '25px');
+  });
+
+  test('color can be unchecked', () => {
+    colorElement.checked = false;
+
+    const wrapper = colorElement.shadowRoot!.querySelector(
+        'customize-chrome-check-mark-wrapper')!;
+    assertFalse(wrapper.checked);
+    const svg = colorElement.shadowRoot!.querySelector('svg')!;
+    assertStyle(svg, 'width', '50px');
+    assertStyle(svg, 'height', '50px');
+    const background = colorElement.shadowRoot!.querySelector('#background')!;
+    assertStyle(background, 'r', '24px');
+  });
 });
diff --git a/chrome/test/data/webui/side_panel/customize_chrome/colors_test.ts b/chrome/test/data/webui/side_panel/customize_chrome/colors_test.ts
index 16a1964..06dda2220 100644
--- a/chrome/test/data/webui/side_panel/customize_chrome/colors_test.ts
+++ b/chrome/test/data/webui/side_panel/customize_chrome/colors_test.ts
@@ -9,11 +9,11 @@
 import {ChromeColor, CustomizeChromePageCallbackRouter, CustomizeChromePageHandlerRemote, CustomizeChromePageRemote, Theme} from 'chrome://customize-chrome-side-panel.top-chrome/customize_chrome.mojom-webui.js';
 import {CustomizeChromeApiProxy} from 'chrome://customize-chrome-side-panel.top-chrome/customize_chrome_api_proxy.js';
 import {PromiseResolver} from 'chrome://resources/js/promise_resolver.js';
-import {assertDeepEquals, assertEquals} from 'chrome://webui-test/chai_assert.js';
+import {assertDeepEquals, assertEquals, assertGE, assertTrue} from 'chrome://webui-test/chai_assert.js';
 import {waitAfterNextRender} from 'chrome://webui-test/polymer_test_util.js';
 import {TestBrowserProxy} from 'chrome://webui-test/test_browser_proxy.js';
 
-import {createTheme, installMock} from './test_support.js';
+import {assertStyle, capture, createTheme, installMock} from './test_support.js';
 
 suite('ColorsTest', () => {
   let colorsElement: ColorsElement;
@@ -100,4 +100,114 @@
     assertEquals(1, handler.getCallCount('setForegroundColor'));
     assertEquals(2, handler.getArgs('setForegroundColor')[0].value);
   });
+
+  test('opens color picker', () => {
+    const focus = capture(colorsElement.$.colorPicker, 'focus');
+    const click = capture(colorsElement.$.colorPicker, 'click');
+
+    colorsElement.$.customColor.click();
+
+    assertTrue(focus.received);
+    assertTrue(click.received);
+  });
+
+  test('sets custom color', () => {
+    colorsElement.$.colorPicker.value = '#ff0000';
+    colorsElement.$.colorPicker.dispatchEvent(new Event('change'));
+
+    const args = handler.getArgs('setForegroundColor');
+    assertGE(1, args.length);
+    assertEquals(0xffff0000, args.at(-1).value);
+  });
+
+  test('updates custom color for theme', async () => {
+    const colors = {
+      colors: [
+        {id: 1, name: 'foo', background: {value: 1}, foreground: {value: 2}},
+      ],
+    };
+    chromeColorsResolver.resolve(colors);
+
+    // Set a custom color theme.
+    const customColortheme = createTheme();
+    customColortheme.backgroundColor = {value: 0xffff0000};
+    customColortheme.foregroundColor = {value: 0xff00ff00};
+    customColortheme.colorPickerIconColor = {value: 0xff0000ff};
+    callbackRouter.setTheme(customColortheme);
+    await callbackRouter.$.flushForTesting();
+
+    // Custom color circle should be updated.
+    assertEquals(0xffff0000, colorsElement.$.customColor.backgroundColor.value);
+    assertEquals(0xff00ff00, colorsElement.$.customColor.foregroundColor.value);
+    assertStyle(
+        colorsElement.$.colorPickerIcon, 'background-color', 'rgb(0, 0, 255)');
+
+    // Set a theme that is not a custom color theme.
+    const otherTheme = createTheme();
+    otherTheme.backgroundColor = {value: 0xffffffff};
+    otherTheme.foregroundColor = undefined;  // Makes a default theme.
+    otherTheme.colorPickerIconColor = {value: 0xffffffff};
+    callbackRouter.setTheme(otherTheme);
+    await callbackRouter.$.flushForTesting();
+
+    // Custom color circle should be not be updated.
+    assertEquals(0xffff0000, colorsElement.$.customColor.backgroundColor.value);
+    assertEquals(0xff00ff00, colorsElement.$.customColor.foregroundColor.value);
+    assertStyle(
+        colorsElement.$.colorPickerIcon, 'background-color', 'rgb(0, 0, 255)');
+  });
+
+  test('checks selected color', async () => {
+    const colors = {
+      colors: [
+        {id: 1, name: 'foo', background: {value: 1}, foreground: {value: 2}},
+        {id: 2, name: 'bar', background: {value: 3}, foreground: {value: 4}},
+      ],
+    };
+    chromeColorsResolver.resolve(colors);
+    const theme = createTheme();
+
+    // Set default color.
+    theme.foregroundColor = undefined;
+    callbackRouter.setTheme(theme);
+    await callbackRouter.$.flushForTesting();
+
+    // Check default color selected.
+    let checkedColors = colorsElement.shadowRoot!.querySelectorAll('[checked]');
+    assertEquals(1, checkedColors.length);
+    assertEquals(colorsElement.$.defaultColor, checkedColors[0]);
+    let indexedColors =
+        colorsElement.shadowRoot!.querySelectorAll('[tabindex="0"]');
+    assertEquals(1, indexedColors.length);
+    assertEquals(colorsElement.$.defaultColor, indexedColors[0]);
+
+    // Set Chrome color.
+    theme.foregroundColor = {value: 2};
+    callbackRouter.setTheme(theme);
+    await callbackRouter.$.flushForTesting();
+
+    // Check Chrome color selected.
+    checkedColors = colorsElement.shadowRoot!.querySelectorAll('[checked]');
+    assertEquals(1, checkedColors.length);
+    assertEquals('chrome-color', checkedColors[0]!.className);
+    assertEquals(2, (checkedColors[0]! as ColorElement).foregroundColor.value);
+    indexedColors =
+        colorsElement.shadowRoot!.querySelectorAll('[tabindex="0"]');
+    assertEquals(1, indexedColors.length);
+    assertEquals('chrome-color', indexedColors[0]!.className);
+
+    // Set custom color.
+    theme.foregroundColor = {value: 5};
+    callbackRouter.setTheme(theme);
+    await callbackRouter.$.flushForTesting();
+
+    // Check custom color selected.
+    checkedColors = colorsElement.shadowRoot!.querySelectorAll('[checked]');
+    assertEquals(1, checkedColors.length);
+    assertEquals(colorsElement.$.customColor, checkedColors[0]);
+    indexedColors =
+        colorsElement.shadowRoot!.querySelectorAll('[tabindex="0"]');
+    assertEquals(1, indexedColors.length);
+    assertEquals(colorsElement.$.customColorContainer, indexedColors[0]);
+  });
 });
diff --git a/chrome/test/data/webui/side_panel/customize_chrome/test_support.ts b/chrome/test/data/webui/side_panel/customize_chrome/test_support.ts
index dd47307..3107b072 100644
--- a/chrome/test/data/webui/side_panel/customize_chrome/test_support.ts
+++ b/chrome/test/data/webui/side_panel/customize_chrome/test_support.ts
@@ -41,6 +41,7 @@
 export function createBackgroundImage(url: string): BackgroundImage {
   return {
     url: {url},
+    isUploadedImage: false,
     title: '',
   };
 }
@@ -51,5 +52,13 @@
     systemDarkMode,
     backgroundColor: {value: 0xffff0000},
     foregroundColor: undefined,
+    colorPickerIconColor: {value: 0xffff0000},
   };
 }
+
+export function capture(
+    target: HTMLElement, event: string): {received: boolean} {
+  const capture = {received: false};
+  target.addEventListener(event, () => capture.received = true);
+  return capture;
+}
diff --git a/chrome/test/data/webui/side_panel/customize_chrome/theme_snapshot_test.ts b/chrome/test/data/webui/side_panel/customize_chrome/theme_snapshot_test.ts
index 3fcfa8b..3ff7543 100644
--- a/chrome/test/data/webui/side_panel/customize_chrome/theme_snapshot_test.ts
+++ b/chrome/test/data/webui/side_panel/customize_chrome/theme_snapshot_test.ts
@@ -8,7 +8,7 @@
 import {CustomizeChromePageCallbackRouter, CustomizeChromePageHandlerRemote, CustomizeChromePageRemote} from 'chrome://customize-chrome-side-panel.top-chrome/customize_chrome.mojom-webui.js';
 import {CustomizeChromeApiProxy} from 'chrome://customize-chrome-side-panel.top-chrome/customize_chrome_api_proxy.js';
 import {ThemeSnapshotElement} from 'chrome://customize-chrome-side-panel.top-chrome/theme_snapshot.js';
-import {assertEquals} from 'chrome://webui-test/chai_assert.js';
+import {assertEquals, assertTrue} from 'chrome://webui-test/chai_assert.js';
 import {TestBrowserProxy} from 'chrome://webui-test/test_browser_proxy.js';
 
 import {$$, assertStyle, createBackgroundImage, createTheme, installMock} from './test_support.js';
@@ -52,13 +52,16 @@
 
     // Assert.
     assertEquals(1, handler.getCallCount('updateTheme'));
+    const selector =
+        themeSnapshotElement.shadowRoot!.querySelector('iron-pages');
+    assertTrue(!!selector);
+    assertEquals('customTheme', selector.selected);
     assertEquals(
         'chrome://theme/foo',
-        $$<HTMLImageElement>(
-            themeSnapshotElement, '#themeSnapshot .image')!.src);
+        $$<HTMLImageElement>(themeSnapshotElement, '.theme-snapshot img')!.src);
   });
 
-  test('having no theme set updates preview background color', async () => {
+  test('not setting a theme updates preview background color', async () => {
     // Arrange.
     createThemeSnapshotElement();
     const theme = createTheme();
@@ -70,8 +73,31 @@
 
     // Assert.
     assertEquals(1, handler.getCallCount('updateTheme'));
+    const selector =
+        themeSnapshotElement.shadowRoot!.querySelector('iron-pages');
+    assertTrue(!!selector);
+    assertEquals('classicChrome', selector.selected);
     assertStyle(
-        $$(themeSnapshotElement, '#themeSnapshot .image')!, 'background-color',
-        'rgb(20, 83, 154)');
+        $$(themeSnapshotElement, '.theme-snapshot #classicChrome')!,
+        'background-color', 'rgb(20, 83, 154)');
+  });
+
+  test('uploading a background updates theme snapshot', async () => {
+    // Arrange.
+    createThemeSnapshotElement();
+    const theme = createTheme();
+    theme.backgroundImage = createBackgroundImage('chrome://theme/foo');
+    theme.backgroundImage.isUploadedImage = true;
+
+    // Act.
+    callbackRouterRemote.setTheme(theme);
+    await callbackRouterRemote.$.flushForTesting();
+
+    // Assert.
+    assertEquals(1, handler.getCallCount('updateTheme'));
+    const selector =
+        themeSnapshotElement.shadowRoot!.querySelector('iron-pages');
+    assertTrue(!!selector);
+    assertEquals('uploadedImage', selector.selected);
   });
 });
diff --git a/chrome/test/media_router/access_code_cast/access_code_cast_integration_browsertest.cc b/chrome/test/media_router/access_code_cast/access_code_cast_integration_browsertest.cc
index 257e75b..a29f59e2 100644
--- a/chrome/test/media_router/access_code_cast/access_code_cast_integration_browsertest.cc
+++ b/chrome/test/media_router/access_code_cast/access_code_cast_integration_browsertest.cc
@@ -143,22 +143,6 @@
         }
       });
 
-  ON_CALL(*media_router_, RegisterMediaRoutesObserver(_))
-      .WillByDefault([this](MediaRoutesObserver* observer) {
-        media_routes_observers_.push_back(observer);
-        return true;
-      });
-
-  // Remove route observers as appropriate (destructing handlers will cause
-  // this to occur).
-  ON_CALL(*media_router_, UnregisterMediaRoutesObserver(_))
-      .WillByDefault([this](MediaRoutesObserver* observer) {
-        auto it = base::ranges::find(media_routes_observers_, observer);
-        if (it != media_routes_observers_.end()) {
-          media_routes_observers_.erase(it);
-        }
-      });
-
   // Handler so MockMediaRouter will respond to requests to create a route.
   // Will construct a RouteRequestResult based on the set result code and
   // then call the handler's callback, which should call the page's callback.
@@ -547,8 +531,9 @@
 
 void AccessCodeCastIntegrationBrowserTest::UpdateRoutes(
     const std::vector<MediaRoute>& routes) {
-  for (MediaRoutesObserver* routes_observer : media_routes_observers_) {
-    routes_observer->OnRoutesUpdated(routes);
+  for (MediaRoutesObserver& routes_observer :
+       media_router_->routes_observers()) {
+    routes_observer.OnRoutesUpdated(routes);
   }
 }
 
diff --git a/chrome/test/media_router/media_router_integration_browsertest.cc b/chrome/test/media_router/media_router_integration_browsertest.cc
index cc299e54..a5f50c3c 100644
--- a/chrome/test/media_router/media_router_integration_browsertest.cc
+++ b/chrome/test/media_router/media_router_integration_browsertest.cc
@@ -220,7 +220,7 @@
   // TerminateRoute, or add pass callback to the TestProvider to run when all
   // routes are gone.
   base::RunLoop run_loop;
-  NoRoutesObserver no_routes_observer(
+  auto no_routes_observer = std::make_unique<NoRoutesObserver>(
       MediaRouterFactory::GetApiForBrowserContext(
           web_contents->GetBrowserContext()),
       run_loop.QuitClosure());
diff --git a/chromecast/browser/android/apk/src/org/chromium/chromecast/shell/CastContentWindowAndroid.java b/chromecast/browser/android/apk/src/org/chromium/chromecast/shell/CastContentWindowAndroid.java
index 809ee0e..e0bf8cff7 100644
--- a/chromecast/browser/android/apk/src/org/chromium/chromecast/shell/CastContentWindowAndroid.java
+++ b/chromecast/browser/android/apk/src/org/chromium/chromecast/shell/CastContentWindowAndroid.java
@@ -5,6 +5,8 @@
 package org.chromium.chromecast.shell;
 
 import android.content.Context;
+import android.hardware.display.DisplayManager;
+import android.view.Display;
 
 import org.chromium.base.ContextUtils;
 import org.chromium.base.Log;
@@ -40,12 +42,29 @@
     @CalledByNative
     private static CastContentWindowAndroid create(long nativeCastContentWindowAndroid,
             boolean enableTouchInput, boolean isRemoteControlMode, boolean turnOnScreen,
-            boolean keepScreenOn, String sessionId) {
+            boolean keepScreenOn, String sessionId, String displayId) {
         return new CastContentWindowAndroid(nativeCastContentWindowAndroid,
-                ContextUtils.getApplicationContext(), enableTouchInput, isRemoteControlMode,
+                getContextWithDisplay(displayId), enableTouchInput, isRemoteControlMode,
                 turnOnScreen, keepScreenOn, sessionId);
     }
 
+    private static Context getContextWithDisplay(String displayId) {
+        Context context = ContextUtils.getApplicationContext();
+        try {
+            int id = Integer.parseInt(displayId);
+            DisplayManager displayManager = context.getSystemService(DisplayManager.class);
+            Display display = displayManager.getDisplay(id);
+            if (display != null) {
+                return context.createDisplayContext(display);
+            }
+        } catch (NumberFormatException e) {
+        }
+        Log.i(TAG,
+                "Display with the given cast display id is not available, "
+                        + "use the default display to create the web view.");
+        return context;
+    }
+
     private CastContentWindowAndroid(long nativeCastContentWindowAndroid, final Context context,
             boolean enableTouchInput, boolean isRemoteControlMode, boolean turnOnScreen,
             boolean keepScreenOn, String sessionId) {
diff --git a/chromecast/browser/android/apk/src/org/chromium/chromecast/shell/CastWebContentsComponent.java b/chromecast/browser/android/apk/src/org/chromium/chromecast/shell/CastWebContentsComponent.java
index 836cf997..16d5abd 100644
--- a/chromecast/browser/android/apk/src/org/chromium/chromecast/shell/CastWebContentsComponent.java
+++ b/chromecast/browser/android/apk/src/org/chromium/chromecast/shell/CastWebContentsComponent.java
@@ -10,14 +10,17 @@
 import android.content.IntentFilter;
 import android.content.ServiceConnection;
 import android.net.Uri;
+import android.os.Bundle;
 import android.os.IBinder;
 import android.os.PatternMatcher;
 
 import androidx.annotation.VisibleForTesting;
 
+import org.chromium.base.ApiCompatibilityUtils;
 import org.chromium.base.Log;
 import org.chromium.chromecast.base.Controller;
 import org.chromium.content_public.browser.WebContents;
+import org.chromium.ui.display.DisplayAndroidManager;
 
 /**
  * A layer of indirection between CastContentWindowAndroid and CastWebContents(Activity|Service).
@@ -100,9 +103,11 @@
             boolean isRemoteControlMode, boolean turnOnScreen) {
         Intent intent = CastWebContentsIntentUtils.requestStartCastActivity(context, webContents,
                 enableTouch, isRemoteControlMode, turnOnScreen, mKeepScreenOn, mSessionId);
-        if (DEBUG) Log.d(TAG, "start activity by intent: " + intent);
+        int displayId = DisplayAndroidManager.getDefaultDisplayForContext(context).getDisplayId();
+        if (DEBUG) Log.d(TAG, "start activity by intent: " + intent + " on display: " + displayId);
         sResumeIntent.set(intent);
-        context.startActivity(intent);
+        Bundle bundle = ApiCompatibilityUtils.createLaunchDisplayIdActivityOptions(displayId);
+        context.startActivity(intent, bundle);
     }
 
     private void sendStopWebContentEvent() {
diff --git a/chromecast/browser/android/cast_content_window_android.cc b/chromecast/browser/android/cast_content_window_android.cc
index 7315c85..29efb38 100644
--- a/chromecast/browser/android/cast_content_window_android.cc
+++ b/chromecast/browser/android/cast_content_window_android.cc
@@ -24,11 +24,13 @@
     bool is_remote_control_mode,
     bool turn_on_screen,
     bool keep_screen_on,
-    const std::string& session_id) {
+    const std::string& session_id,
+    const std::string& display_id) {
   JNIEnv* env = base::android::AttachCurrentThread();
   return Java_CastContentWindowAndroid_create(
       env, native_window, enable_touch_input, is_remote_control_mode,
-      turn_on_screen, keep_screen_on, ConvertUTF8ToJavaString(env, session_id));
+      turn_on_screen, keep_screen_on, ConvertUTF8ToJavaString(env, session_id),
+      ConvertUTF8ToJavaString(env, display_id));
 }
 
 }  // namespace
@@ -42,7 +44,8 @@
                                     params_->is_remote_control_mode,
                                     params_->turn_on_screen,
                                     params_->keep_screen_on,
-                                    params_->session_id)) {}
+                                    params_->session_id,
+                                    params_->display_id)) {}
 
 CastContentWindowAndroid::~CastContentWindowAndroid() {
   JNIEnv* env = base::android::AttachCurrentThread();
diff --git a/chromecast/browser/android/junit/src/org/chromium/chromecast/shell/CastWebContentsComponentTest.java b/chromecast/browser/android/junit/src/org/chromium/chromecast/shell/CastWebContentsComponentTest.java
index 0d6bb44..bdc6e69 100644
--- a/chromecast/browser/android/junit/src/org/chromium/chromecast/shell/CastWebContentsComponentTest.java
+++ b/chromecast/browser/android/junit/src/org/chromium/chromecast/shell/CastWebContentsComponentTest.java
@@ -10,13 +10,18 @@
 import static org.mockito.Mockito.never;
 import static org.mockito.Mockito.times;
 import static org.mockito.Mockito.verify;
+import static org.mockito.Mockito.when;
 
 import android.app.Activity;
 import android.content.BroadcastReceiver;
 import android.content.Context;
+import android.content.ContextWrapper;
 import android.content.Intent;
 import android.content.IntentFilter;
 import android.content.ServiceConnection;
+import android.os.Build.VERSION_CODES;
+import android.os.Bundle;
+import android.view.Display;
 
 import androidx.localbroadcastmanager.content.LocalBroadcastManager;
 import androidx.test.core.app.ApplicationProvider;
@@ -51,7 +56,11 @@
 
     private static final String SESSION_ID = "123456789";
 
+    private static final int DISPLAY_ID = 1;
+    private static final String ACTIVITY_OPTIONS_DISPLAY_ID = "android.activity.launchDisplayId";
+
     private @Mock WebContents mWebContents;
+    private @Mock Display mDisplay;
     private Activity mActivity;
     private ShadowActivity mShadowActivity;
     private StartParams mStartParams;
@@ -62,6 +71,7 @@
     @Before
     public void setUp() {
         MockitoAnnotations.initMocks(this);
+        when(mDisplay.getDisplayId()).thenReturn(DISPLAY_ID);
         mActivity = Mockito.spy(Robolectric.buildActivity(Activity.class).setup().get());
         mShadowActivity = Shadows.shadowOf(mActivity);
         mStartParams = new StartParams(mActivity, mWebContents, APP_ID);
@@ -82,6 +92,29 @@
     }
 
     @Test
+    @Config(minSdk = VERSION_CODES.R)
+    public void testStartStartsWebContentsActivityWithDisplayId() {
+        Assume.assumeFalse(BuildConfig.DISPLAY_WEB_CONTENTS_IN_SERVICE);
+
+        ContextWrapper context =
+                Mockito.spy(new ContextWrapper(ContextUtils.getApplicationContext()) {
+                    @Override
+                    public Display getDisplay() {
+                        return mDisplay;
+                    }
+                });
+        StartParams startParams = new StartParams(context, mWebContents, APP_ID);
+
+        CastWebContentsComponent component =
+                new CastWebContentsComponent(SESSION_ID, null, null, false, false, true, false);
+        component.start(startParams, false);
+
+        ArgumentCaptor<Bundle> bundle = ArgumentCaptor.forClass(Bundle.class);
+        verify(context).startActivity(any(Intent.class), bundle.capture());
+        Assert.assertEquals(bundle.getValue().getInt(ACTIVITY_OPTIONS_DISPLAY_ID), DISPLAY_ID);
+    }
+
+    @Test
     public void testStartStartsWebContentsService() {
         Assume.assumeFalse(BuildConfig.DISPLAY_WEB_CONTENTS_IN_SERVICE);
 
diff --git a/chromecast/browser/mojom/cast_web_service.mojom b/chromecast/browser/mojom/cast_web_service.mojom
index 7f002d14..90f4a64 100644
--- a/chromecast/browser/mojom/cast_web_service.mojom
+++ b/chromecast/browser/mojom/cast_web_service.mojom
@@ -55,6 +55,9 @@
   // Cast Application/Activity session ID.
   string session_id = "";
 
+  // Cast display ID to create the web view (if available).
+  string display_id = "";
+
   // Cast Receiver SDK version of the application (if available).
   string sdk_version = "";
 
diff --git a/chromeos/ash/components/cryptohome/auth_factor_conversions.cc b/chromeos/ash/components/cryptohome/auth_factor_conversions.cc
index 2add076..543543a 100644
--- a/chromeos/ash/components/cryptohome/auth_factor_conversions.cc
+++ b/chromeos/ash/components/cryptohome/auth_factor_conversions.cc
@@ -10,6 +10,7 @@
 #include "base/check_op.h"
 #include "base/logging.h"
 #include "base/notreached.h"
+#include "base/strings/string_number_conversions.h"
 #include "chromeos/ash/components/dbus/cryptohome/auth_factor.pb.h"
 
 namespace cryptohome {
@@ -152,7 +153,9 @@
         proto_input->set_recovery_response(recovery_auth.recovery_data);
       } else {
         const auto& recovery_creation = auth_input.GetRecoveryCreationInput();
-        proto_input->set_mediator_pub_key(recovery_creation.pub_key);
+        DCHECK(
+            base::HexStringToString(recovery_creation.pub_key,
+                                    proto_input->mutable_mediator_pub_key()));
         proto_input->set_user_gaia_id(recovery_creation.user_gaia_id);
         proto_input->set_device_user_id(recovery_creation.device_user_id);
       }
diff --git a/chromeos/crosapi/mojom/BUILD.gn b/chromeos/crosapi/mojom/BUILD.gn
index 14ff114..12d04d1 100644
--- a/chromeos/crosapi/mojom/BUILD.gn
+++ b/chromeos/crosapi/mojom/BUILD.gn
@@ -106,6 +106,7 @@
     "web_page_info.mojom",
   ]
   disable_variants = true
+  cpp_only = true
 
   public_deps = [
     "//chromeos/components/remote_apps/mojom",
diff --git a/components/autofill/content/renderer/form_autofill_util.cc b/components/autofill/content/renderer/form_autofill_util.cc
index d1e7399..023bc02 100644
--- a/components/autofill/content/renderer/form_autofill_util.cc
+++ b/components/autofill/content/renderer/form_autofill_util.cc
@@ -18,6 +18,7 @@
 #include "base/containers/contains.h"
 #include "base/containers/flat_map.h"
 #include "base/containers/flat_set.h"
+#include "base/feature_list.h"
 #include "base/i18n/case_conversion.h"
 #include "base/metrics/field_trial.h"
 #include "base/metrics/histogram_functions.h"
@@ -499,6 +500,31 @@
   return std::u16string();
 }
 
+// Detects a label declared after the `element`, which is visually positioned
+// above the element (usually using CCS). Such labels often act as a
+// placeholders. E.g.
+// <div>
+//  <input>
+//  <span>Placeholder</span>
+// </div>
+std::u16string InferLabelFromOverlayingSuccessor(
+    const WebFormControlElement& element) {
+  if (!base::FeatureList::IsEnabled(
+          features::kAutofillSupportPoorMansPlaceholder)) {
+    return std::u16string();
+  }
+
+  WebNode next = element.NextSibling();
+  while (!next.IsNull() && !next.IsElementNode())
+    next = next.NextSibling();
+  if (!next.IsNull()) {
+    gfx::Rect bounds = next.To<WebElement>().BoundsInWidget();
+    if (!bounds.IsEmpty() && element.BoundsInWidget().Contains(bounds))
+      return FindChildText(next);
+  }
+  return std::u16string();
+}
+
 // Helper for |InferLabelForElement()| that infers a label, from
 // the value attribute when it is present and user has not typed in (if
 // element's value attribute is same as the element's value).
@@ -800,6 +826,13 @@
     return true;
   }
 
+  inferred_label = InferLabelFromOverlayingSuccessor(element);
+  if (IsLabelValid(inferred_label)) {
+    label_source = FormFieldData::LabelSource::kOverlayingLabel;
+    label = std::move(inferred_label);
+    return true;
+  }
+
   // If we didn't find a placeholder, check for aria-label text.
   inferred_label = GetAriaLabel(element.GetDocument(), element);
   if (IsLabelValid(inferred_label)) {
diff --git a/components/autofill/content/renderer/form_autofill_util_browsertest.cc b/components/autofill/content/renderer/form_autofill_util_browsertest.cc
index 26791bb..19952796 100644
--- a/components/autofill/content/renderer/form_autofill_util_browsertest.cc
+++ b/components/autofill/content/renderer/form_autofill_util_browsertest.cc
@@ -59,6 +59,22 @@
   const char16_t* expected_label;
 };
 
+// An <input> with a label placed on top of it (usually used as a placeholder
+// replacement).
+const char* kPoorMansPlaceholder = R"(
+  <style>
+    .fixed_position_and_size {
+      position: fixed;
+      top: 0;
+      left: 0;
+      width: 100px;
+      height: 20px;
+    }
+  </style>
+  <input id='target' class=fixed_position_and_size>
+  <span class=fixed_position_and_size>label</span>
+)";
+
 void VerifyButtonTitleCache(const WebFormElement& form_target,
                             const ButtonTitleList& expected_button_titles,
                             const ButtonTitlesCache& actual_cache) {
@@ -172,6 +188,9 @@
 }
 
 TEST_F(FormAutofillUtilsTest, InferLabelForElementTest) {
+  base::test::ScopedFeatureList features;
+  features.InitAndEnableFeature(features::kAutofillSupportPoorMansPlaceholder);
+
   static const AutofillFieldUtilCase test_cases[] = {
       {"DIV table test 1", R"(
        <div>
@@ -215,6 +234,7 @@
        u""},
       {"Infer from next sibling",
        "<input id='target' type='checkbox'>hello <b>world</b>", u"hello world"},
+      {"Poor man's placeholder", kPoorMansPlaceholder, u"label"},
   };
   for (auto test_case : test_cases) {
     SCOPED_TRACE(test_case.description);
@@ -233,6 +253,9 @@
 }
 
 TEST_F(FormAutofillUtilsTest, InferLabelSourceTest) {
+  base::test::ScopedFeatureList features;
+  features.InitAndEnableFeature(features::kAutofillSupportPoorMansPlaceholder);
+
   struct AutofillFieldLabelSourceCase {
     const char* html;
     const FormFieldData::LabelSource label_source;
@@ -263,7 +286,7 @@
        FormFieldData::LabelSource::kTdTag},
       {"<dl><dt>label</dt><dd><input id='target'></dd></dl>",
        FormFieldData::LabelSource::kDdTag},
-  };
+      {kPoorMansPlaceholder, FormFieldData::LabelSource::kOverlayingLabel}};
 
   for (auto test_case : test_cases) {
     SCOPED_TRACE(testing::Message() << test_case.label_source);
diff --git a/components/autofill/core/common/autofill_features.cc b/components/autofill/core/common/autofill_features.cc
index 24da4705..af6d86f7 100644
--- a/components/autofill/core/common/autofill_features.cc
+++ b/components/autofill/core/common/autofill_features.cc
@@ -488,6 +488,13 @@
              "AutofillSkipComparingInferredLabels",
              base::FEATURE_DISABLED_BY_DEFAULT);
 
+// Enables support for artificial placeholders, implemented by placing text on
+// top of the input field using CSS.
+// TODO(crbug.com/1396374): Remove when launched.
+BASE_FEATURE(kAutofillSupportPoorMansPlaceholder,
+             "AutofillSupportPoorMansPlaceholder",
+             base::FEATURE_DISABLED_BY_DEFAULT);
+
 // Controls whether Autofill should search prefixes of all words/tokens when
 // filtering profiles, or only on prefixes of the whole string.
 BASE_FEATURE(kAutofillTokenPrefixMatching,
diff --git a/components/autofill/core/common/autofill_features.h b/components/autofill/core/common/autofill_features.h
index db665334..9c093072 100644
--- a/components/autofill/core/common/autofill_features.h
+++ b/components/autofill/core/common/autofill_features.h
@@ -148,6 +148,8 @@
 COMPONENT_EXPORT(AUTOFILL) BASE_DECLARE_FEATURE(kAutofillTokenPrefixMatching);
 COMPONENT_EXPORT(AUTOFILL) BASE_DECLARE_FEATURE(kAutofillUploadThrottling);
 COMPONENT_EXPORT(AUTOFILL)
+BASE_DECLARE_FEATURE(kAutofillSupportPoorMansPlaceholder);
+COMPONENT_EXPORT(AUTOFILL)
 BASE_DECLARE_FEATURE(kAutofillUseAlternativeStateNameMap);
 COMPONENT_EXPORT(AUTOFILL)
 BASE_DECLARE_FEATURE(kAutofillUseImprovedLabelDisambiguation);
diff --git a/components/autofill/core/common/mojom/autofill_types.mojom b/components/autofill/core/common/mojom/autofill_types.mojom
index 46a45154..7e4368f 100644
--- a/components/autofill/core/common/mojom/autofill_types.mojom
+++ b/components/autofill/core/common/mojom/autofill_types.mojom
@@ -284,18 +284,19 @@
 
   // From which source the label is inferred.
   enum LabelSource {
-    kUnknown,  // The source is unknown.
-    kLabelTag,
-    kPTag,
-    kDivTable,
-    kTdTag,
-    kDdTag,
-    kLiTag,
-    kPlaceHolder,
-    kAriaLabel,
-    kCombined,  // Combined with various elements.
-    kValue,     // label is the value of element.
-    kFor,       // Derived from the "for" attribute of a label element.
+    kUnknown,          // No label.
+    kLabelTag,         // <label> sibling/ancestor.
+    kPTag,             // <p> sibling.
+    kDivTable,         // <div> ancestor.
+    kTdTag,            // <td> sibling.
+    kDdTag,            // <dd> sibling.
+    kLiTag,            // <li> ancestor.
+    kPlaceHolder,      // placeholder attribute.
+    kAriaLabel,        // aria-label attribute.
+    kCombined,         // Text node sibling (includes <b>, etc).
+    kValue,            // value attribute.
+    kFor,              // <label> for-attribute
+    kOverlayingLabel,  // Succeeding DOM node overlaying the input.
   };
 
   mojo_base.mojom.String16 label;
diff --git a/components/browser_ui/widget/android/BUILD.gn b/components/browser_ui/widget/android/BUILD.gn
index 9bc7f19..631b786 100644
--- a/components/browser_ui/widget/android/BUILD.gn
+++ b/components/browser_ui/widget/android/BUILD.gn
@@ -148,14 +148,6 @@
     "java/res/anim/chip_out.xml",
     "java/res/anim/image_grid_enter.xml",
     "java/res/anim/image_tile_enter.xml",
-    "java/res/anim/menu_enter.xml",
-    "java/res/anim/menu_enter_from_bottom.xml",
-    "java/res/anim/menu_enter_from_bottom_left.xml",
-    "java/res/anim/menu_enter_from_top_left.xml",
-    "java/res/anim/menu_exit.xml",
-    "java/res/anim/menu_exit_from_bottom.xml",
-    "java/res/anim/menu_exit_from_bottom_left.xml",
-    "java/res/anim/menu_exit_from_top_left.xml",
     "java/res/anim/textbubble_in.xml",
     "java/res/anim/textbubble_out.xml",
     "java/res/color/chip_background_color.xml",
@@ -270,11 +262,9 @@
     "java/res/layout/tile_view_modern.xml",
     "java/res/layout/tile_view_modern_condensed.xml",
     "java/res/layout/title_and_description_layout.xml",
-    "java/res/values-ldrtl/values.xml",
     "java/res/values-night/colors.xml",
     "java/res/values-night/dimens.xml",
     "java/res/values-night/drawables.xml",
-    "java/res/values-sw600dp/dimens.xml",
     "java/res/values-v23/drawables.xml",
     "java/res/values/attrs.xml",
     "java/res/values/colors.xml",
diff --git a/components/browser_ui/widget/android/java/res/values-ldrtl/values.xml b/components/browser_ui/widget/android/java/res/values-ldrtl/values.xml
deleted file mode 100644
index 7c7ab04..0000000
--- a/components/browser_ui/widget/android/java/res/values-ldrtl/values.xml
+++ /dev/null
@@ -1,11 +0,0 @@
-<?xml version="1.0" encoding="utf-8"?>
-<!--
-Copyright 2019 The Chromium Authors
-Use of this source code is governed by a BSD-style license that can be
-found in the LICENSE file.
--->
-<resources>
-    <!-- Menu Animation Constants -->
-    <item type="fraction" format="fraction" name="menu_animation_pivot_x">5%</item>
-    <item type="fraction" format="fraction" name="menu_start_animation_pivot_x">95%</item>
-</resources>
diff --git a/components/browser_ui/widget/android/java/res/values-sw600dp/dimens.xml b/components/browser_ui/widget/android/java/res/values-sw600dp/dimens.xml
deleted file mode 100644
index b469f2f1..0000000
--- a/components/browser_ui/widget/android/java/res/values-sw600dp/dimens.xml
+++ /dev/null
@@ -1,13 +0,0 @@
-<?xml version="1.0" encoding="utf-8"?>
-<!--
-Copyright 2019 The Chromium Authors
-Use of this source code is governed by a BSD-style license that can be
-found in the LICENSE file.
--->
-
-<resources>
-    <!-- Menu Dimensions -->
-    <!-- Necessary to align the menu icon with the actual button. -->
-    <dimen name="menu_negative_software_vertical_offset">6dp</dimen>
-</resources>
-
diff --git a/components/browser_ui/widget/android/java/res/values/dimens.xml b/components/browser_ui/widget/android/java/res/values/dimens.xml
index c8672d3b..7ab6fd8 100644
--- a/components/browser_ui/widget/android/java/res/values/dimens.xml
+++ b/components/browser_ui/widget/android/java/res/values/dimens.xml
@@ -42,7 +42,6 @@
     <dimen name="list_menu_width">180dp</dimen>
 
     <!-- Custom Menu dimensions -->
-    <dimen name="menu_negative_software_vertical_offset">0dp</dimen>
     <dimen name="menu_divider_padding">8dp</dimen>
     <dimen name="menu_padding_start">16dp</dimen>
 
diff --git a/components/browser_ui/widget/android/java/res/values/styles.xml b/components/browser_ui/widget/android/java/res/values/styles.xml
index c11fd0f..00c37d2 100644
--- a/components/browser_ui/widget/android/java/res/values/styles.xml
+++ b/components/browser_ui/widget/android/java/res/values/styles.xml
@@ -61,22 +61,10 @@
         <item name="android:paddingEnd">@dimen/default_list_row_padding</item>
     </style>
 
-    <style name="EndIconMenuAnim">
-        <item name="android:windowEnterAnimation">@anim/menu_enter</item>
-        <item name="android:windowExitAnimation">@anim/menu_exit</item>
-    </style>
-    <style name="EndIconMenuAnimBottom">
-        <item name="android:windowEnterAnimation">@anim/menu_enter_from_bottom</item>
-        <item name="android:windowExitAnimation">@anim/menu_exit_from_bottom</item>
-    </style>
-    <style name="StartIconMenuAnim">
-        <item name="android:windowEnterAnimation">@anim/menu_enter_from_top_left</item>
-        <item name="android:windowExitAnimation">@anim/menu_exit_from_top_left</item>
-    </style>
-    <style name="StartIconMenuAnimBottom">
-        <item name="android:windowEnterAnimation">@anim/menu_enter_from_bottom_left</item>
-        <item name="android:windowExitAnimation">@anim/menu_exit_from_bottom_left</item>
-    </style>
+    <style name="EndIconMenuAnim" parent="AnchoredPopupAnimEndTop" />
+    <style name="EndIconMenuAnimBottom" parent="AnchoredPopupAnimEndBottom" />
+    <style name="StartIconMenuAnim" parent="AnchoredPopupAnimStartTop" />
+    <style name="StartIconMenuAnimBottom" parent="AnchoredPopupAnimStartBottom" />
 
     <!-- Modern List Item -->
     <style name="ListItemContainer">
diff --git a/components/browser_ui/widget/android/java/res/values/values.xml b/components/browser_ui/widget/android/java/res/values/values.xml
index e80066d..ad03559 100644
--- a/components/browser_ui/widget/android/java/res/values/values.xml
+++ b/components/browser_ui/widget/android/java/res/values/values.xml
@@ -6,10 +6,6 @@
 -->
 
 <resources>
-    <!-- Menu Animation Constants -->
-    <item type="fraction" format="fraction" name="menu_animation_pivot_x">95%</item>
-    <item type="fraction" format="fraction" name="menu_start_animation_pivot_x">5%</item>
-
     <!-- Drag-Reorderable List Item -->
     <!-- Used for 90% = 230/255 alpha -->
     <integer name="list_item_dragged_alpha">230</integer>
diff --git a/components/browser_ui/widget/android/java/src/org/chromium/components/browser_ui/widget/ContextMenuDialog.java b/components/browser_ui/widget/android/java/src/org/chromium/components/browser_ui/widget/ContextMenuDialog.java
index 9ffbf96..e8cea36 100644
--- a/components/browser_ui/widget/android/java/src/org/chromium/components/browser_ui/widget/ContextMenuDialog.java
+++ b/components/browser_ui/widget/android/java/src/org/chromium/components/browser_ui/widget/ContextMenuDialog.java
@@ -188,6 +188,7 @@
                     mPopupWindow.setSmartAnchorWithMaxWidth(true);
                     mPopupWindow.setVerticalOverlapAnchor(true);
                     mPopupWindow.setOutsideTouchable(false);
+                    mPopupWindow.setAnimateFromAnchor(true);
                     // Set popup focusable so the screen reader can announce the popup properly.
                     if (mAccessibilityUtil != null
                             && mAccessibilityUtil.isTouchExplorationEnabled()) {
diff --git a/components/history_clusters/core/config.cc b/components/history_clusters/core/config.cc
index 94ab0f1..c7c61df 100644
--- a/components/history_clusters/core/config.cc
+++ b/components/history_clusters/core/config.cc
@@ -107,6 +107,9 @@
             "JourneysPersistClustersInHistoryDbPeriodMinutes",
             persist_clusters_in_history_db_period_minutes);
 
+    persist_on_query = base::GetFieldTrialParamByFeatureAsBool(
+        internal::kPersistedClusters, "persist_on_query", persist_on_query);
+
     max_persisted_clusters_to_fetch = base::GetFieldTrialParamByFeatureAsInt(
         internal::kPersistedClusters, "max_persisted_clusters_to_fetch",
         max_persisted_clusters_to_fetch);
diff --git a/components/history_clusters/core/config.h b/components/history_clusters/core/config.h
index 42c0800..69edc4e 100644
--- a/components/history_clusters/core/config.h
+++ b/components/history_clusters/core/config.h
@@ -107,6 +107,16 @@
   // every hour.
   int persist_clusters_in_history_db_period_minutes = 60;
 
+  // No effect if `persist_clusters_in_history_db` is disabled. If disabled,
+  // persistence occurs on a timer (see the above 2 params). If enabled, will
+  // instead occur on query like refreshing the keyword cache does. This may
+  // help bound the number of persistence requests. If enabled, will continue to
+  // also be capped to at most 1 request per
+  // `persist_clusters_in_history_db_period_minutes`, but
+  // `persist_clusters_in_history_db_after_startup_delay_minutes` will be
+  // unused.
+  bool persist_on_query = false;
+
   // Hard cap on max clusters to fetch after exhausting unclustered visits and
   // fetching persisted clusters for the get most recent flow. Doesn't affect
   // the update flow, which uses day boundaries as well as
diff --git a/components/history_clusters/core/history_clusters_service.cc b/components/history_clusters/core/history_clusters_service.cc
index ba1407e2..e079b85e 100644
--- a/components/history_clusters/core/history_clusters_service.cc
+++ b/components/history_clusters/core/history_clusters_service.cc
@@ -184,8 +184,13 @@
 }
 
 void HistoryClustersService::RepeatedlyUpdateClusters() {
-  if (!GetConfig().persist_clusters_in_history_db)
+  // If `persist_on_query` is enabled, clusters are updated on query and not on
+  // a timer.
+  if (!GetConfig().persist_clusters_in_history_db ||
+      GetConfig().persist_on_query) {
     return;
+  }
+
   // Update clusters, both periodically and once after startup because:
   // 1) To avoid having very stale (up to 90 days) clusters for the initial
   //    period after startup.
@@ -205,8 +210,38 @@
 
 void HistoryClustersService::UpdateClusters() {
   DCHECK(history_service_);
+
+  // TODO(manukh): This logic (if task not done, if time since < period) is
+  //  repeated for both keyword caches and here. Should probably share it in
+  //  some kind of base task that all 3 inherit from.
+
   if (update_clusters_task_ && !update_clusters_task_->Done())
     return;
+
+  // Make sure clusters aren't updated too frequently. If `persist_on_query` is
+  // false, this is already ensured by `update_clusters_period_timer_`. If
+  // update_clusters_task_ is null, this is the 1st request which shouldn't be
+  // delayed.
+  if (GetConfig().persist_on_query &&
+      update_clusters_timer_.Elapsed() <=
+          base::Minutes(
+              GetConfig().persist_clusters_in_history_db_period_minutes) &&
+      update_clusters_task_) {
+    return;
+  }
+
+  // Using custom histogram as this occurs too infrequently to be captured by
+  // the built in histograms. `persist_clusters_in_history_db_period_minutes`
+  // ranges from 1 to 12 hours while the built in timing histograms go up to 1
+  // hr.
+  base::UmaHistogramCustomTimes(
+      "History.Clusters.UpdateClusters.TimeBetweenTasks",
+      update_clusters_timer_.Elapsed(), base::Minutes(60), base::Hours(48),
+      100);
+
+  // Reset the timer.
+  update_clusters_timer_ = {};
+
   update_clusters_task_ =
       std::make_unique<HistoryClustersServiceTaskUpdateClusters>(
           weak_ptr_factory_.GetWeakPtr(), incomplete_visit_context_annotations_,
@@ -223,6 +258,8 @@
     return absl::nullopt;
 
   StartKeywordCacheRefresh();
+  if (GetConfig().persist_on_query)
+    UpdateClusters();
 
   // Early exit for single-character queries, even if it's an exact match.
   // We still want to allow for two-character exact matches like "uk".
@@ -254,6 +291,8 @@
     return false;
 
   StartKeywordCacheRefresh();
+  if (GetConfig().persist_on_query)
+    UpdateClusters();
 
   return short_url_keywords_cache_.find(url_keyword) !=
              short_url_keywords_cache_.end() ||
diff --git a/components/history_clusters/core/history_clusters_service.h b/components/history_clusters/core/history_clusters_service.h
index 96ab6d32..e40926e 100644
--- a/components/history_clusters/core/history_clusters_service.h
+++ b/components/history_clusters/core/history_clusters_service.h
@@ -210,6 +210,11 @@
   friend class HistoryClustersServiceTestApi;
 
   // Starts a keyword cache refresh, if necessary.
+  // TODO(manukh): `StartKeywordCacheRefresh()` and
+  //  `PopulateClusterKeywordCache()` should be encapsulated into their own task
+  //  to avoid cluttering `HistoryClusterService` with their callbacks. Similar
+  //  to the `HistoryClustersServiceTaskGetMostRecentClusters` and
+  //  `HistoryClustersServiceTaskUpdateClusters` tasks.
   void StartKeywordCacheRefresh();
 
   // This is a callback used for the `QueryClusters()` call from
@@ -281,6 +286,10 @@
   // `RepeatedlyUpdateClusters()`'s comment.
   base::RepeatingTimer update_clusters_period_timer_;
 
+  // The time of the last `UpdateClusters()` call. Used for logging and to limit
+  // requests when `persist_on_query` is enabled.
+  base::ElapsedTimer update_clusters_timer_;
+
   // A list of observers for this service.
   base::ObserverList<Observer> observers_;
 
diff --git a/components/keyed_service/content/browser_context_keyed_service_factory.cc b/components/keyed_service/content/browser_context_keyed_service_factory.cc
index 417ff7d..b2d52be 100644
--- a/components/keyed_service/content/browser_context_keyed_service_factory.cc
+++ b/components/keyed_service/content/browser_context_keyed_service_factory.cc
@@ -87,12 +87,24 @@
 }
 
 std::unique_ptr<KeyedService>
+BrowserContextKeyedServiceFactory::BuildServiceInstanceForBrowserContext(
+    content::BrowserContext* context) const {
+  // TODO(tsepez): fully deprecate the form below.
+  return base::WrapUnique(BuildServiceInstanceFor(context));
+}
+
+KeyedService* BrowserContextKeyedServiceFactory::BuildServiceInstanceFor(
+    content::BrowserContext* context) const {
+  // Stub to prevent converted sub-classes from needing to implement this form.
+  NOTREACHED();
+  return nullptr;
+}
+
+std::unique_ptr<KeyedService>
 BrowserContextKeyedServiceFactory::BuildServiceInstanceFor(
     void* context) const {
-  // TODO(isherman): The wrapped BuildServiceInstanceFor() should return a
-  // scoped_ptr as well.
-  return base::WrapUnique(
-      BuildServiceInstanceFor(static_cast<content::BrowserContext*>(context)));
+  return BuildServiceInstanceForBrowserContext(
+      static_cast<content::BrowserContext*>(context));
 }
 
 bool BrowserContextKeyedServiceFactory::IsOffTheRecord(void* context) const {
diff --git a/components/keyed_service/content/browser_context_keyed_service_factory.h b/components/keyed_service/content/browser_context_keyed_service_factory.h
index f546244..dda3f0c 100644
--- a/components/keyed_service/content/browser_context_keyed_service_factory.h
+++ b/components/keyed_service/content/browser_context_keyed_service_factory.h
@@ -141,8 +141,15 @@
   //
   // This should not return nullptr; instead, return nullptr from
   // `GetBrowserContextToUse()`.
+  //
+  // Sub-classes implement one of these two forms:
+  virtual std::unique_ptr<KeyedService> BuildServiceInstanceForBrowserContext(
+      content::BrowserContext* context) const;
+
+  // DEPRECATED: allows incremental conversion to the unique_ptr<> form
+  // above. New code should not be using this form.
   virtual KeyedService* BuildServiceInstanceFor(
-      content::BrowserContext* context) const = 0;
+      content::BrowserContext* context) const;
 
   // A helper object actually listens for notifications about BrowserContext
   // destruction, calculates the order in which things are destroyed and then
diff --git a/components/management_strings.grdp b/components/management_strings.grdp
index 7ede678..153625b6 100644
--- a/components/management_strings.grdp
+++ b/components/management_strings.grdp
@@ -61,20 +61,23 @@
     <if expr="not chromeos_ash">
       <if expr="_google_chrome">
         <message name="IDS_MANAGEMENT_BROWSER_NOTICE" desc="Message shown when the (Google-branded) Chrome browser is managed, it indicates what the administrator can do on the browser.">
-        Your administrator can change your browser setup remotely. Activity on this device may also be managed outside of Chrome. <ph name="BEGIN_LINK">&lt;a target="_blank" href="$1"&gt;</ph>Learn more<ph name="END_LINK">&lt;/a&gt;</ph>
+        Your administrator can change your browser setup remotely. Activity on this device may also be managed outside of Chrome. <ph name="BEGIN_LINK">&lt;a target="_blank" href="$1" aria-label="$2"&gt;</ph>Learn more<ph name="END_LINK">&lt;/a&gt;</ph>
         </message>
         <message name="IDS_MANAGEMENT_NOT_MANAGED_NOTICE" desc="Message indicating that the (Google-branded) Chrome browser is not managed">
-        This browser is not managed by a company or other organization. Activity on this device may be managed outside of Chrome. <ph name="BEGIN_LINK">&lt;a target="_blank" href="$1"&gt;</ph>Learn more<ph name="END_LINK">&lt;/a&gt;</ph>
+        This browser is not managed by a company or other organization. Activity on this device may be managed outside of Chrome. <ph name="BEGIN_LINK">&lt;a target="_blank" href="$1" aria-label="$2"&gt;</ph>Learn more<ph name="END_LINK">&lt;/a&gt;</ph>
         </message>
       </if>
       <if expr="not _google_chrome">
         <message name="IDS_MANAGEMENT_BROWSER_NOTICE" desc="Message shown when the (non-Google-branded) Chromium browser is managed, it indicates what the administrator can do on the browser.">
-        Your administrator can change your browser setup remotely. Activity on this device may also be managed outside of Chromium. <ph name="BEGIN_LINK">&lt;a target="_blank" href="$1"&gt;</ph>Learn more<ph name="END_LINK">&lt;/a&gt;</ph>
+        Your administrator can change your browser setup remotely. Activity on this device may also be managed outside of Chromium. <ph name="BEGIN_LINK">&lt;a target="_blank" href="$1" aria-label="$2"&gt;</ph>Learn more<ph name="END_LINK">&lt;/a&gt;</ph>
         </message>
         <message name="IDS_MANAGEMENT_NOT_MANAGED_NOTICE" desc="Message indicating that the (non-Google-branded) Chromium browser is not managed">
-        This browser is not managed by a company or other organization. Activity on this device may be managed outside of Chromium. <ph name="BEGIN_LINK">&lt;a target="_blank" href="$1"&gt;</ph>Learn more<ph name="END_LINK">&lt;/a&gt;</ph>
+        This browser is not managed by a company or other organization. Activity on this device may be managed outside of Chromium. <ph name="BEGIN_LINK">&lt;a target="_blank" href="$1" aria-label="$2"&gt;</ph>Learn more<ph name="END_LINK">&lt;/a&gt;</ph>
         </message>
       </if>
+      <message name="IDS_MANAGEMENT_LEARN_MORE_ACCCESSIBILITY_TEXT" desc="Accessibility text for a 'Learn more' link that links to a help article about how the browser is managed.">
+        Learn more about how your browser is managed
+      </message>
     </if>
   </if>
 
diff --git a/components/management_strings_grdp/IDS_MANAGEMENT_BROWSER_NOTICE.png.sha1 b/components/management_strings_grdp/IDS_MANAGEMENT_BROWSER_NOTICE.png.sha1
index 2ea1997..16e39345 100644
--- a/components/management_strings_grdp/IDS_MANAGEMENT_BROWSER_NOTICE.png.sha1
+++ b/components/management_strings_grdp/IDS_MANAGEMENT_BROWSER_NOTICE.png.sha1
@@ -1 +1 @@
-ad0a9ae496e2d8a74660cd371bcc9acc79b475a2
\ No newline at end of file
+4a860cd9622b611c3b76432b7d4b84ed1eeb0460
\ No newline at end of file
diff --git a/components/management_strings_grdp/IDS_MANAGEMENT_LEARN_MORE_ACCCESSIBILITY_TEXT.png.sha1 b/components/management_strings_grdp/IDS_MANAGEMENT_LEARN_MORE_ACCCESSIBILITY_TEXT.png.sha1
new file mode 100644
index 0000000..4ecf846
--- /dev/null
+++ b/components/management_strings_grdp/IDS_MANAGEMENT_LEARN_MORE_ACCCESSIBILITY_TEXT.png.sha1
@@ -0,0 +1 @@
+6734674614d77faf8b2ac866ca1da0c22d06a495
\ No newline at end of file
diff --git a/components/management_strings_grdp/IDS_MANAGEMENT_NOT_MANAGED_NOTICE.png.sha1 b/components/management_strings_grdp/IDS_MANAGEMENT_NOT_MANAGED_NOTICE.png.sha1
index 20ed25a..84b7d02 100644
--- a/components/management_strings_grdp/IDS_MANAGEMENT_NOT_MANAGED_NOTICE.png.sha1
+++ b/components/management_strings_grdp/IDS_MANAGEMENT_NOT_MANAGED_NOTICE.png.sha1
@@ -1 +1 @@
-1a56c5e54aeb312486f7f3b4564cc6f64525aac3
\ No newline at end of file
+17731ccf8561565d284cbd30a16f7d9a88a5c8f5
\ No newline at end of file
diff --git a/components/media_router/browser/android/media_router_android.h b/components/media_router/browser/android/media_router_android.h
index c53880a..69186a6 100644
--- a/components/media_router/browser/android/media_router_android.h
+++ b/components/media_router/browser/android/media_router_android.h
@@ -168,7 +168,7 @@
                          std::unique_ptr<MediaSinksObserverList>>;
   MediaSinkObservers sinks_observers_;
 
-  base::ObserverList<MediaRoutesObserver>::Unchecked routes_observers_;
+  base::ObserverList<MediaRoutesObserver> routes_observers_;
 
   struct MediaRouteRequest {
     MediaRouteRequest(const MediaSource& source,
diff --git a/components/media_router/browser/media_routes_observer.cc b/components/media_router/browser/media_routes_observer.cc
index b94975c..0b68da1 100644
--- a/components/media_router/browser/media_routes_observer.cc
+++ b/components/media_router/browser/media_routes_observer.cc
@@ -6,16 +6,20 @@
 
 #include "base/check.h"
 #include "components/media_router/browser/media_router.h"
+#include "content/public/browser/browser_thread.h"
 
 namespace media_router {
 
 MediaRoutesObserver::MediaRoutesObserver(MediaRouter* router)
     : router_(router) {
+  DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
   DCHECK(router_);
   router_->RegisterMediaRoutesObserver(this);
 }
 
 MediaRoutesObserver::~MediaRoutesObserver() {
+  DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
+  CHECK(IsInObserverList());
   router_->UnregisterMediaRoutesObserver(this);
 }
 
diff --git a/components/media_router/browser/media_routes_observer.h b/components/media_router/browser/media_routes_observer.h
index c7fb0cc..251fbb8 100644
--- a/components/media_router/browser/media_routes_observer.h
+++ b/components/media_router/browser/media_routes_observer.h
@@ -8,6 +8,8 @@
 #include <vector>
 
 #include "base/memory/raw_ptr.h"
+#include "base/memory/weak_ptr.h"
+#include "base/observer_list_types.h"
 #include "components/media_router/common/media_route.h"
 
 namespace media_router {
@@ -20,14 +22,17 @@
 // |OnRoutesUpdated| that match the route IDs contained in the
 // |joinable_route_ids| can be connected joined by the source.  If no
 // |source_id| is supplied, then the idea of joinable routes no longer applies.
-class MediaRoutesObserver {
+class MediaRoutesObserver : public base::CheckedObserver,
+                            public base::SupportsWeakPtr<MediaRoutesObserver> {
  public:
   explicit MediaRoutesObserver(MediaRouter* router);
 
   MediaRoutesObserver(const MediaRoutesObserver&) = delete;
   MediaRoutesObserver& operator=(const MediaRoutesObserver&) = delete;
 
-  virtual ~MediaRoutesObserver();
+  // NOTE: must be destroyed on the Browser UI thread to avoid threading issues
+  // with access.
+  ~MediaRoutesObserver() override;
 
   // Invoked when the list of routes and their associated sinks have been
   // updated.
diff --git a/components/media_router/browser/test/mock_media_router.h b/components/media_router/browser/test/mock_media_router.h
index 6235c7b..ff2f589 100644
--- a/components/media_router/browser/test/mock_media_router.h
+++ b/components/media_router/browser/test/mock_media_router.h
@@ -10,9 +10,11 @@
 #include <string>
 #include <vector>
 
+#include "base/observer_list.h"
 #include "build/build_config.h"
 #include "components/media_router/browser/logger_impl.h"
 #include "components/media_router/browser/media_router_base.h"
+#include "components/media_router/browser/media_routes_observer.h"
 #include "components/media_router/common/media_route.h"
 #include "components/media_router/common/media_sink.h"
 #include "components/media_router/common/media_source.h"
@@ -127,16 +129,25 @@
   MOCK_METHOD1(RegisterMediaSinksObserver, bool(MediaSinksObserver* observer));
   MOCK_METHOD1(UnregisterMediaSinksObserver,
                void(MediaSinksObserver* observer));
-  MOCK_METHOD1(RegisterMediaRoutesObserver,
-               void(MediaRoutesObserver* observer));
-  MOCK_METHOD1(UnregisterMediaRoutesObserver,
-               void(MediaRoutesObserver* observer));
+  void RegisterMediaRoutesObserver(MediaRoutesObserver* observer) override {
+    routes_observers_.AddObserver(observer);
+  }
+  void UnregisterMediaRoutesObserver(MediaRoutesObserver* observer) override {
+    routes_observers_.RemoveObserver(observer);
+  }
   MOCK_METHOD1(RegisterPresentationConnectionMessageObserver,
                void(PresentationConnectionMessageObserver* observer));
   MOCK_METHOD1(UnregisterPresentationConnectionMessageObserver,
                void(PresentationConnectionMessageObserver* observer));
   MOCK_METHOD0(GetMediaSinkServiceStatus, std::string());
 
+  base::ObserverList<MediaRoutesObserver>& routes_observers() {
+    return routes_observers_;
+  }
+
+ protected:
+  base::ObserverList<MediaRoutesObserver> routes_observers_;
+
  private:
 #if !BUILDFLAG(IS_ANDROID)
   IssueManager issue_manager_;
diff --git a/components/messages/android/internal/java/res/layout/message_banner_view.xml b/components/messages/android/internal/java/res/layout/message_banner_view.xml
index ef8b723d..a2824757 100644
--- a/components/messages/android/internal/java/res/layout/message_banner_view.xml
+++ b/components/messages/android/internal/java/res/layout/message_banner_view.xml
@@ -71,7 +71,7 @@
         android:visibility="gone"
         android:contentDescription="@null"
         android:background="?actionBarItemBackground"
-        android:layout_width="@dimen/message_banner_button_width"
+        android:layout_width="@dimen/min_touch_target_size"
         android:layout_height="@dimen/min_touch_target_size" />
 
     <ImageView
diff --git a/components/messages/android/internal/java/res/values/dimens.xml b/components/messages/android/internal/java/res/values/dimens.xml
index 337256a..9cd54a1 100644
--- a/components/messages/android/internal/java/res/values/dimens.xml
+++ b/components/messages/android/internal/java/res/values/dimens.xml
@@ -14,8 +14,6 @@
     <dimen name="message_banner_elevation">6dp</dimen>
     <!-- For messages below the front messages, used in stacking animation. -->
     <dimen name="message_banner_back_elevation">5dp</dimen>
-    <!-- Min width to ensure the min touchable size. -->
-    <dimen name="message_banner_button_width">44dp</dimen>
     <dimen name="message_primary_button_max_width">100dp</dimen>
     <dimen name="message_icon_margin">12dp</dimen>
     <dimen name="message_icon_size">24dp</dimen>
diff --git a/components/omnibox/browser/omnibox_prefs.cc b/components/omnibox/browser/omnibox_prefs.cc
index 9c93f02..efe11e3 100644
--- a/components/omnibox/browser/omnibox_prefs.cc
+++ b/components/omnibox/browser/omnibox_prefs.cc
@@ -83,8 +83,8 @@
     SuggestionGroupVisibility visibility) {
   DCHECK(prefs);
 
-  DictionaryPrefUpdate update(prefs, kSuggestionGroupVisibility);
-  update->SetIntKey(base::NumberToString(suggestion_group_id), visibility);
+  ScopedDictPrefUpdate update(prefs, kSuggestionGroupVisibility);
+  update->Set(base::NumberToString(suggestion_group_id), visibility);
 
   base::SparseHistogram::FactoryGet(
       visibility == SuggestionGroupVisibility::SHOWN
diff --git a/components/payments/content/BUILD.gn b/components/payments/content/BUILD.gn
index 2b3eeae..0846734 100644
--- a/components/payments/content/BUILD.gn
+++ b/components/payments/content/BUILD.gn
@@ -88,7 +88,7 @@
   ]
 
   if (is_chromeos_ash) {
-    sources += [ "android_app_communication_chrome_os.cc" ]
+    sources += [ "android_app_communication_ash.cc" ]
 
     deps += [
       "//ash/components/arc",
diff --git a/components/payments/content/android_app_communication_chrome_os.cc b/components/payments/content/android_app_communication_ash.cc
similarity index 96%
rename from components/payments/content/android_app_communication_chrome_os.cc
rename to components/payments/content/android_app_communication_ash.cc
index 3a8d4da..2710288d 100644
--- a/components/payments/content/android_app_communication_chrome_os.cc
+++ b/components/payments/content/android_app_communication_ash.cc
@@ -155,8 +155,9 @@
   // at this time.
   auto supported_method_iterator =
       stringified_method_data.find(methods::kGooglePlayBilling);
-  if (supported_method_iterator == stringified_method_data.end())
+  if (supported_method_iterator == stringified_method_data.end()) {
     return nullptr;
+  }
 
   // Chrome OS TWA supports only one set of payment method specific data.
   if (supported_method_iterator->second.size() > 1) {
@@ -193,20 +194,19 @@
 }
 
 // Invokes the TWA Android app in Android subsystem on Chrome OS.
-class AndroidAppCommunicationChromeOS : public AndroidAppCommunication {
+class AndroidAppCommunicationAsh : public AndroidAppCommunication {
  public:
-  explicit AndroidAppCommunicationChromeOS(content::BrowserContext* context)
+  explicit AndroidAppCommunicationAsh(content::BrowserContext* context)
       : AndroidAppCommunication(context),
         get_app_service_(base::BindRepeating(
             &arc::ArcPaymentAppBridge::GetForBrowserContext)) {}
 
-  ~AndroidAppCommunicationChromeOS() override = default;
+  ~AndroidAppCommunicationAsh() override = default;
 
   // Disallow copy and assign.
-  AndroidAppCommunicationChromeOS(
-      const AndroidAppCommunicationChromeOS& other) = delete;
-  AndroidAppCommunicationChromeOS& operator=(
-      const AndroidAppCommunicationChromeOS& other) = delete;
+  AndroidAppCommunicationAsh(const AndroidAppCommunicationAsh& other) = delete;
+  AndroidAppCommunicationAsh& operator=(
+      const AndroidAppCommunicationAsh& other) = delete;
 
   // AndroidAppCommunication implementation:
   void GetAppDescriptions(const std::string& twa_package_name,
@@ -371,7 +371,7 @@
 // static
 std::unique_ptr<AndroidAppCommunication> AndroidAppCommunication::Create(
     content::BrowserContext* context) {
-  return std::make_unique<AndroidAppCommunicationChromeOS>(context);
+  return std::make_unique<AndroidAppCommunicationAsh>(context);
 }
 
 }  // namespace payments
diff --git a/components/policy/resources/templates/policies.yaml b/components/policy/resources/templates/policies.yaml
index 68770da1..ca758916 100644
--- a/components/policy/resources/templates/policies.yaml
+++ b/components/policy/resources/templates/policies.yaml
@@ -1043,6 +1043,7 @@
   1042: ShowCastSessionsStartedByOtherDevices
   1043: CloudAPAuthEnabled
   1044: UsbDetectorNotificationEnabled
+  1045: LacrosSelection
 atomic_groups:
   1: Homepage
   2: RemoteAccess
diff --git a/components/policy/resources/templates/policy_definitions/Miscellaneous/LacrosSelection.yaml b/components/policy/resources/templates/policy_definitions/Miscellaneous/LacrosSelection.yaml
new file mode 100644
index 0000000..0850325
--- /dev/null
+++ b/components/policy/resources/templates/policy_definitions/Miscellaneous/LacrosSelection.yaml
@@ -0,0 +1,66 @@
+caption: Select <ph name="LACROS_NAME">Lacros</ph> browser binary
+default: user_choice
+default_for_enterprise_users: rootfs
+desc: |-
+  This setting configures which <ph name="LACROS_NAME">Lacros</ph> browser to use.
+
+  If the policy is set to <ph name="LACROS_SELECTION_USER_CHOICE_VALUE">user_choice</ph>,
+  the user can decide which <ph name="LACROS_NAME">Lacros</ph> browser to load: binary from
+  <ph name="LACROS_ROOTFS_NAME">rootfs</ph> or <ph name="LACROS_STATEFUL_NAME">stateful</ph> partition.
+  If the user has not set any preference, the binary with the newest version will be chosen.
+
+  If the policy is set to <ph name="LACROS_SELECTION_ROOTFS_VALUE">rootfs</ph>,
+  always load <ph name="LACROS_ROOTFS_NAME">rootfs</ph> binary of
+  <ph name="LACROS_NAME">Lacros</ph> browser.
+
+  If the policy is set to <ph name="LACROS_SELECTION_STATEFUL_VALUE">stateful</ph>,
+  always load <ph name="LACROS_STATEFUL_NAME">stateful</ph> binary of
+  <ph name="LACROS_NAME">Lacros</ph> browser.
+  Using this value may cause unexpected behavior if the <ph name="LACROS_STATEFUL_NAME">stateful</ph>
+  browser version becomes older than the <ph name="PRODUCT_OS_NAME">$2<ex>Google ChromeOS</ex></ph> version.
+  This case is not supported and is not guaranteed to work correctly.
+
+  If the policy is unset, the default is <ph name="LACROS_SELECTION_ROOTFS_VALUE">rootfs</ph> for enterprise-managed users and
+  <ph name="LACROS_SELECTION_USER_CHOICE_VALUE">user_choice</ph> for non-managed users.
+
+  Note that changing the policy's value may cause
+  <ph name="LACROS_NAME">Lacros</ph> browser's data loss if the browser's
+  version it changes to is older than the current one. For example, if the
+  policy changes from <ph name="LACROS_SELECTION_STATEFUL_VALUE">stateful</ph>
+  to <ph name="LACROS_SELECTION_ROOTFS_VALUE">rootfs</ph>, and the first one was
+  updated. Or if <ph name="PRODUCT_OS_NAME">$2<ex>Google ChromeOS</ex></ph> was
+  updated together with <ph name="LACROS_ROOTFS_NAME">rootfs</ph>
+  <ph name="LACROS_NAME">Lacros</ph> browser, and
+  <ph name="LACROS_STATEFUL_NAME">stateful</ph> has not been updated yet.
+  In such scenarios the correct data migration is not guaranteed.
+
+  Using <ph name="LACROS_SELECTION_USER_CHOICE_VALUE">user_choice</ph> or <ph name="LACROS_SELECTION_ROOTFS_VALUE">rootfs</ph>
+  is a safe option. Switching from <ph name="LACROS_SELECTION_ROOTFS_VALUE">rootfs</ph> to <ph name="LACROS_SELECTION_USER_CHOICE_VALUE">user_choice</ph>
+  is safe as well.
+
+example_value: stateful
+features:
+  dynamic_refresh: false
+  per_profile: false
+items:
+- caption: Allow users to select <ph name="LACROS_NAME">Lacros</ph> browser binary
+  name: user_choice
+  value: user_choice
+- caption: Always load <ph name="LACROS_ROOTFS_NAME">rootfs</ph> <ph name="LACROS_NAME">Lacros</ph> browser
+  name: rootfs
+  value: rootfs
+- caption: Always load <ph name="LACROS_STATEFUL_NAME">stateful</ph> <ph name="LACROS_NAME">Lacros</ph> browser
+  name: stateful
+  value: stateful
+owners:
+- asumaneev@google.com
+schema:
+  enum:
+  - user_choice
+  - rootfs
+  - stateful
+  type: string
+supported_on:
+- chrome_os:110-
+tags: []
+type: string-enum
diff --git a/components/policy_strings.grdp b/components/policy_strings.grdp
index d189d8e..f4c2dc75 100644
--- a/components/policy_strings.grdp
+++ b/components/policy_strings.grdp
@@ -326,6 +326,9 @@
   <message name="IDS_POLICY_DEPENDENCY_ERROR" desc="The text displayed in the status column when a policy is ignored as another policies is not set properly.">
     Ignored because <ph name="POLICY_NAME">$1<ex>CloudReportingEnabled</ex></ph> is not set to <ph name="VALUE">$2<ex>Enabled</ex></ph>.
   </message>
+  <message name="IDS_POLICY_DEPENDENCY_ERROR_ANY_VALUE" desc="The text displayed in the status column when a policy is ignored as another policies is not set at all.">
+    Ignored because <ph name="POLICY_NAME">$1<ex>CloudReportingEnabled</ex></ph> is not set.
+  </message>
   <message name="IDS_POLICY_USER_IS_NOT_AFFILIATED_ERROR" desc="The text displayed in the status column when a user cloud policy is ignored due to user is not affiliated to the machine management.">
     Ignored because user is not affiliated to the machine management or the machine is not managed.
   </message>
diff --git a/components/policy_strings_grdp/IDS_POLICY_DEPENDENCY_ERROR_ANY_VALUE.png.sha1 b/components/policy_strings_grdp/IDS_POLICY_DEPENDENCY_ERROR_ANY_VALUE.png.sha1
new file mode 100644
index 0000000..face0ce
--- /dev/null
+++ b/components/policy_strings_grdp/IDS_POLICY_DEPENDENCY_ERROR_ANY_VALUE.png.sha1
@@ -0,0 +1 @@
+6e7315f3cbbe1df993996f5128e7b2e3c01d2b71
\ No newline at end of file
diff --git a/components/remote_cocoa/app_shim/immersive_mode_controller.h b/components/remote_cocoa/app_shim/immersive_mode_controller.h
index 940775a1..299777e 100644
--- a/components/remote_cocoa/app_shim/immersive_mode_controller.h
+++ b/components/remote_cocoa/app_shim/immersive_mode_controller.h
@@ -59,7 +59,8 @@
   void RevealUnlock();
   int reveal_lock_count() { return reveal_lock_count_; }
 
-  NSWindow* overlay_widget() { return overlay_widget_; }
+  NSWindow* browser_window() { return browser_window_; }
+  NSWindow* overlay_window() { return overlay_window_; }
 
  private:
   // Pin or unpin the titlebar.
@@ -73,8 +74,8 @@
 
   bool enabled_ = false;
 
-  NSWindow* const browser_widget_;
-  NSWindow* const overlay_widget_;
+  NSWindow* const browser_window_;
+  NSWindow* const overlay_window_;
 
   // A controller for top chrome.
   base::scoped_nsobject<ImmersiveModeTitlebarViewController>
@@ -127,13 +128,12 @@
 //
 // This class will keep the position of the overlay window in sync with its
 // original content (top chrome).
-REMOTE_COCOA_APP_SHIM_EXPORT @interface ImmersiveModeTitlebarObserver
-    : NSObject {
-  NSWindow* _overlay_window;
-  NSView* _overlay_view;
-}
-- (instancetype)initWithOverlayWindow:(NSWindow*)overlay_window
-                          overlayView:(NSView*)overlay_view;
+REMOTE_COCOA_APP_SHIM_EXPORT @interface ImmersiveModeTitlebarObserver : NSObject
+
+- (instancetype)initWithController:
+                    (base::WeakPtr<remote_cocoa::ImmersiveModeController>)
+                        controller
+                       overlayView:(NSView*)overlay_view;
 @end
 
 #endif  // COMPONENTS_REMOTE_COCOA_APP_SHIM_IMMERSIVE_MODE_CONTROLLER_H_
diff --git a/components/remote_cocoa/app_shim/immersive_mode_controller.mm b/components/remote_cocoa/app_shim/immersive_mode_controller.mm
index d9cf7a8..2f17d51 100644
--- a/components/remote_cocoa/app_shim/immersive_mode_controller.mm
+++ b/components/remote_cocoa/app_shim/immersive_mode_controller.mm
@@ -4,6 +4,7 @@
 
 #include "components/remote_cocoa/app_shim/immersive_mode_controller.h"
 
+#include "base/auto_reset.h"
 #include "base/check.h"
 #include "base/mac/foundation_util.h"
 #include "base/mac/scoped_block.h"
@@ -38,13 +39,22 @@
 
 }  // namespace
 
+@interface ImmersiveModeTitlebarObserver () {
+  base::WeakPtr<remote_cocoa::ImmersiveModeController> _controller;
+  NSView* _overlay_view;
+  BOOL _barrier;
+}
+@end
+
 @implementation ImmersiveModeTitlebarObserver
 
-- (instancetype)initWithOverlayWindow:(NSWindow*)overlay_window
-                          overlayView:(NSView*)overlay_view {
+- (instancetype)initWithController:
+                    (base::WeakPtr<remote_cocoa::ImmersiveModeController>)
+                        controller
+                       overlayView:(NSView*)overlay_view {
   self = [super init];
   if (self) {
-    _overlay_window = overlay_window;
+    _controller = std::move(controller);
     _overlay_view = overlay_view;
   }
   return self;
@@ -80,9 +90,29 @@
   if (_overlay_view.visibleRect.size.height !=
       _overlay_view.frame.size.height) {
     point_on_screen.y = -_overlay_view.frame.size.height;
+  } else {
+    // If there are sub-windows and the titlebar is fully visible (a y origin of
+    // 0), pin the titlebar. This will prevent the titlebar from autohiding and
+    // causing the sub-windows from moving up when the mouse leaves top chrome.
+    NSRect frame = [change[@"new"] rectValue];
+    if (!_barrier && frame.origin.y == 0 &&
+        _controller->titlebar_lock_count() > 0) {
+      // Add a barrier to prevent re-entry, which is a byproduct of
+      // TitlebarLock() and TitlebarUnlock().
+      base::AutoReset<BOOL> set_barrier(&_barrier, YES);
+      // This lock / unlock scheme is to force the titlebar to be pinned in
+      // place, which can only be done when the titlebar is fully visible.
+      // Existing sub-windows hold a lock, however since the titlebar isn't
+      // fully revealed until this point the existing locks don't actually pin
+      // the titlebar. The existing locks are still important for knowing when
+      // to unpin the titlebar. When all outstanding locks are released the
+      // titlebar be unpinned.
+      _controller->TitlebarLock();
+      _controller->TitlebarUnlock();
+    }
   }
 
-  [_overlay_window setFrameOrigin:point_on_screen];
+  [_controller->overlay_window() setFrameOrigin:point_on_screen];
 }
 
 @end
@@ -284,8 +314,8 @@
 ImmersiveModeController::ImmersiveModeController(NSWindow* browser_widget,
                                                  NSWindow* overlay_widget,
                                                  base::OnceClosure callback)
-    : browser_widget_(browser_widget),
-      overlay_widget_(overlay_widget),
+    : browser_window_(browser_widget),
+      overlay_window_(overlay_widget),
       weak_ptr_factory_(this) {
   immersive_mode_window_observer_.reset([[ImmersiveModeWindowObserver alloc]
       initWithController:weak_ptr_factory_.GetWeakPtr()]);
@@ -295,20 +325,20 @@
   // bit. We do not want a separator. Pre-macOS 11 there is no titlebar
   // separator.
   if (@available(macOS 11.0, *)) {
-    browser_widget_.titlebarSeparatorStyle = NSTitlebarSeparatorStyleNone;
+    browser_window_.titlebarSeparatorStyle = NSTitlebarSeparatorStyleNone;
   }
 
   // Create a new NSTitlebarAccessoryViewController that will host the
   // overlay_view_.
   immersive_mode_titlebar_view_controller_.reset(
       [[ImmersiveModeTitlebarViewController alloc]
-           initWithOverlayWindow:overlay_widget_
+           initWithOverlayWindow:overlay_window_
           viewWillAppearCallback:std::move(callback)]);
 
   // Create a NSWindow delegate that will be used to map the AppKit created
   // NSWindow to the overlay view widget's NSWindow.
   immersive_mode_mapper_.reset([[ImmersiveModeMapper alloc] init]);
-  immersive_mode_mapper_.get().originalHostingWindow = overlay_widget_;
+  immersive_mode_mapper_.get().originalHostingWindow = overlay_window_;
   immersive_mode_titlebar_view_controller_.get().view =
       [[ImmersiveModeView alloc]
           initWithImmersiveModeDelegate:immersive_mode_mapper_.get()];
@@ -317,7 +347,7 @@
   // view will be re-parented into the AppKit created NSWindow.
   BridgedContentView* overlay_content_view =
       base::mac::ObjCCastStrict<BridgedContentView>(
-          overlay_widget_.contentView);
+          overlay_window_.contentView);
   [overlay_content_view retain];
   [overlay_content_view removeFromSuperview];
 
@@ -326,15 +356,15 @@
   // window.
   ImmersiveModeTitlebarObserver* titlebar_observer =
       [[[ImmersiveModeTitlebarObserver alloc]
-          initWithOverlayWindow:overlay_widget_
-                    overlayView:overlay_content_view] autorelease];
+          initWithController:weak_ptr_factory_.GetWeakPtr()
+                 overlayView:overlay_content_view] autorelease];
   [immersive_mode_titlebar_view_controller_
       setTitlebarObserver:titlebar_observer];
 
   // The original content view (top chrome) has been moved to the AppKit
   // created NSWindow. Create a new content view but reuse the original bridge
   // so that mouse drags are handled.
-  overlay_widget_.contentView =
+  overlay_window_.contentView =
       [[[BridgedContentView alloc] initWithBridge:overlay_content_view.bridge
                                            bounds:gfx::Rect()] autorelease];
 
@@ -350,6 +380,9 @@
       [[NSTitlebarAccessoryViewController alloc] init]);
   thin_titlebar_view_controller_.get().view =
       [[[NSView alloc] init] autorelease];
+  thin_titlebar_view_controller_.get().view.wantsLayer = YES;
+  thin_titlebar_view_controller_.get().view.layer.backgroundColor =
+      NSColor.blackColor.CGColor;
   thin_titlebar_view_controller_.get().layoutAttribute =
       NSLayoutAttributeBottom;
   thin_titlebar_view_controller_.get().fullScreenMinHeight =
@@ -358,7 +391,7 @@
   // Move sub-widgets from the browser widget to the overlay widget so that
   // they are rendered above the toolbar.
   ObserveOverlayChildWindows();
-  ReparentChildWindows(browser_widget_, overlay_widget_);
+  ReparentChildWindows(browser_window_, overlay_window_);
 }
 
 ImmersiveModeController::~ImmersiveModeController() {
@@ -370,25 +403,25 @@
   NSView* overlay_content_view =
       immersive_mode_titlebar_view_controller_.get().view.subviews.firstObject;
   [overlay_content_view removeFromSuperview];
-  overlay_widget_.contentView = overlay_content_view;
+  overlay_window_.contentView = overlay_content_view;
   [immersive_mode_titlebar_view_controller_ removeFromParentViewController];
   [immersive_mode_titlebar_view_controller_.get().view release];
   immersive_mode_titlebar_view_controller_.reset();
-  browser_widget_.styleMask |= NSWindowStyleMaskFullSizeContentView;
+  browser_window_.styleMask |= NSWindowStyleMaskFullSizeContentView;
   if (@available(macOS 11.0, *)) {
-    browser_widget_.titlebarSeparatorStyle = NSTitlebarSeparatorStyleAutomatic;
+    browser_window_.titlebarSeparatorStyle = NSTitlebarSeparatorStyleAutomatic;
   }
 
   // Move sub-widgets back to the browser widget.
-  ReparentChildWindows(overlay_widget_, browser_widget_);
+  ReparentChildWindows(overlay_window_, browser_window_);
 }
 
 void ImmersiveModeController::Enable() {
   DCHECK(!enabled_);
   enabled_ = true;
-  [browser_widget_ addTitlebarAccessoryViewController:
+  [browser_window_ addTitlebarAccessoryViewController:
                        immersive_mode_titlebar_view_controller_];
-  [browser_widget_
+  [browser_window_
       addTitlebarAccessoryViewController:thin_titlebar_view_controller_];
   NSRect frame = thin_titlebar_view_controller_.get().view.frame;
   frame.size.height = kThinControllerHeight;
@@ -422,7 +455,7 @@
       immersive_mode_titlebar_view_controller_.get().fullScreenMinHeight =
           immersive_mode_titlebar_view_controller_.get().view.frame.size.height;
       thin_titlebar_view_controller_.get().hidden = YES;
-      browser_widget_.styleMask &= ~NSWindowStyleMaskFullSizeContentView;
+      browser_window_.styleMask &= ~NSWindowStyleMaskFullSizeContentView;
 
       // Toggling the controller will allow the content view to resize below Top
       // Chrome.
@@ -438,7 +471,7 @@
       thin_titlebar_view_controller_.get().hidden = NO;
 
       immersive_mode_titlebar_view_controller_.get().fullScreenMinHeight = 0;
-      browser_widget_.styleMask |= NSWindowStyleMaskFullSizeContentView;
+      browser_window_.styleMask |= NSWindowStyleMaskFullSizeContentView;
       break;
     case mojom::ToolbarVisibilityStyle::kNone:
       thin_titlebar_view_controller_.get().hidden = YES;
@@ -457,8 +490,15 @@
 // will reveal and auto-hide itself based on mouse movement (controlled by
 // AppKit).
 void ImmersiveModeController::SetTitlebarPinned(bool pinned) {
-  // Remove the current, if any, clear controller from the window.
-  [clear_titlebar_view_controller_.get() removeFromParentViewController];
+  // Remove current, if any, clear controllers from the window. For some reason
+  // -removeFromParentViewController does not always remove the controller.
+  // Attempt to remove the current and any stale controllers.
+  for (NSTitlebarAccessoryViewController* c in browser_window_
+           .titlebarAccessoryViewControllers) {
+    if ([c isKindOfClass:[ClearTitlebarViewController class]]) {
+      [c removeFromParentViewController];
+    }
+  }
 
   if (!pinned) {
     clear_titlebar_view_controller_.reset();
@@ -466,13 +506,13 @@
   }
 
   clear_titlebar_view_controller_.reset([[ClearTitlebarViewController alloc]
-      initWithHeight:browser_widget_.contentView.frame.size.height -
+      initWithHeight:browser_window_.contentView.frame.size.height -
                      kThinControllerHeight]);
   clear_titlebar_view_controller_.get().view =
       [[[NSView alloc] init] autorelease];
   clear_titlebar_view_controller_.get().layoutAttribute =
       NSLayoutAttributeBottom;
-  [browser_widget_
+  [browser_window_
       addTitlebarAccessoryViewController:clear_titlebar_view_controller_];
 }
 
@@ -486,7 +526,7 @@
                 context:nullptr];
   };
   NativeWidgetMacNSWindow* overlay_window =
-      base::mac::ObjCCastStrict<NativeWidgetMacNSWindow>(overlay_widget_);
+      base::mac::ObjCCastStrict<NativeWidgetMacNSWindow>(overlay_window_);
   overlay_window.childWindowAddedHandler = ^(NSWindow* child) {
     // Ignore non-visible children.
     if (!child.visible) {
diff --git a/components/remote_cocoa/app_shim/native_widget_ns_window_bridge.mm b/components/remote_cocoa/app_shim/native_widget_ns_window_bridge.mm
index 5085d15..fd71a76d 100644
--- a/components/remote_cocoa/app_shim/native_widget_ns_window_bridge.mm
+++ b/components/remote_cocoa/app_shim/native_widget_ns_window_bridge.mm
@@ -1301,7 +1301,7 @@
   // break cross-display fullscreen transitions by losing focus of the
   // transitioning window (crbug.com/1338659) or changing the z-order of
   // windows on the previous space. Making the window key here seems to
-  // alleviate those apparent defects (crbug.com/1392542). 
+  // alleviate those apparent defects (crbug.com/1392542).
   if (is_key_window)
     [window_ makeKeyAndOrderFront:nil];
 }
@@ -1591,7 +1591,7 @@
       if (child_window.parentWindow == window)
         continue;
       if (immersive_mode_controller_ &&
-          immersive_mode_controller_->overlay_widget() == child_window) {
+          immersive_mode_controller_->overlay_window() == child_window) {
         continue;
       }
       [window addChildWindow:child_window ordered:NSWindowAbove];
diff --git a/components/reporting/storage/storage_queue.cc b/components/reporting/storage/storage_queue.cc
index 9b0550b7..8f0a9d9 100644
--- a/components/reporting/storage/storage_queue.cc
+++ b/components/reporting/storage/storage_queue.cc
@@ -1529,6 +1529,14 @@
     ScopedReservation scoped_reservation(
         wrapped_record.ByteSizeLong(),
         storage_queue_->options().memory_resource());
+    // Inject "memory unavailable" failure, if requested.
+    if (storage_queue_->test_injection_handler_ &&
+        !storage_queue_->test_injection_handler_
+             .Run(test::StorageQueueOperationKind::kWrappedRecordLowMemory,
+                  storage_queue_->next_sequencing_id_)
+             .ok()) {
+      scoped_reservation.Reduce(0);
+    }
     if (!scoped_reservation.reserved()) {
       Schedule(&ReadContext::Response, base::Unretained(this),
                Status(error::RESOURCE_EXHAUSTED,
@@ -1604,6 +1612,14 @@
     ScopedReservation scoped_reservation(
         encrypted_record_result.ValueOrDie().ByteSizeLong(),
         storage_queue_->options().memory_resource());
+    // Inject "memory unavailable" failure, if requested.
+    if (storage_queue_->test_injection_handler_ &&
+        !storage_queue_->test_injection_handler_
+             .Run(test::StorageQueueOperationKind::kEncryptedRecordLowMemory,
+                  storage_queue_->next_sequencing_id_)
+             .ok()) {
+      scoped_reservation.Reduce(0);
+    }
     if (!scoped_reservation.reserved()) {
       Schedule(&ReadContext::Response, base::Unretained(this),
                Status(error::RESOURCE_EXHAUSTED,
@@ -1789,7 +1805,12 @@
 }
 
 Status StorageQueue::ReserveNewRecordDiskSpace(const size_t total_size) {
-  if (!options_.disk_space_resource()->Reserve(total_size)) {
+  if ((test_injection_handler_ &&  // Test only: Simulate failure if requested
+       !test_injection_handler_
+            .Run(test::StorageQueueOperationKind::kWriteLowDiskSpace,
+                 next_sequencing_id_)
+            .ok()) ||
+      !options_.disk_space_resource()->Reserve(total_size)) {
     const uint64_t space_used = options_.disk_space_resource()->GetUsed();
     const uint64_t space_total = options_.disk_space_resource()->GetTotal();
     return Status(
@@ -2080,6 +2101,7 @@
     scoped_refptr<ResourceManager> memory_resource,
     scoped_refptr<ResourceManager> disk_space_resource,
     scoped_refptr<RefCountedClosureList> completion_closure_list) {
+  // Reserve specified disk space for the file.
   if (!disk_space_resource->Reserve(size)) {
     LOG(WARNING) << "Disk space exceeded adding file "
                  << filename.MaybeAsASCII();
@@ -2088,6 +2110,7 @@
         base::StrCat({"Not enough disk space available to include file=",
                       filename.MaybeAsASCII()}));
   }
+
   // Cannot use base::MakeRefCounted, since the constructor is private.
   return scoped_refptr<StorageQueue::SingleFile>(
       new SingleFile(filename, size, memory_resource, disk_space_resource,
diff --git a/components/reporting/storage/storage_queue.h b/components/reporting/storage/storage_queue.h
index a2c93c06..39edd8b 100644
--- a/components/reporting/storage/storage_queue.h
+++ b/components/reporting/storage/storage_queue.h
@@ -47,6 +47,9 @@
   kReadBlock,
   kWriteBlock,
   kWriteMetadata,
+  kWrappedRecordLowMemory,
+  kEncryptedRecordLowMemory,
+  kWriteLowDiskSpace,
 };
 
 }  // namespace test
diff --git a/components/reporting/storage/storage_queue_unittest.cc b/components/reporting/storage/storage_queue_unittest.cc
index dbaa07e6..635ccae 100644
--- a/components/reporting/storage/storage_queue_unittest.cc
+++ b/components/reporting/storage/storage_queue_unittest.cc
@@ -2173,12 +2173,17 @@
 TEST_P(StorageQueueTest, WriteRecordWithInsufficientDiskSpace) {
   CreateTestStorageQueueOrDie(BuildStorageQueueOptionsOnlyManual());
 
-  // Update total disk space and reset after running the write operation so it
-  // does not affect other tests
-  const auto original_disk_space = options_.disk_space_resource()->GetTotal();
-  options_.disk_space_resource()->Test_SetTotal(0);
+  // Inject simulated failures.
+  auto inject = InjectFailures();
+  EXPECT_CALL(
+      *inject,
+      Call(Eq(test::StorageQueueOperationKind::kWriteLowDiskSpace), Eq(0)))
+      .WillRepeatedly(WithArg<1>(Invoke([](int64_t seq_id) {
+        return Status(error::INTERNAL,
+                      base::StrCat({"Simulated data write low disk space, seq=",
+                                    base::NumberToString(seq_id)}));
+      })));
   Status write_result = WriteString(kData[0]);
-  options_.disk_space_resource()->Test_SetTotal(original_disk_space);
   EXPECT_FALSE(write_result.ok());
   EXPECT_EQ(write_result.error_code(), error::RESOURCE_EXHAUSTED);
 }
@@ -2196,6 +2201,64 @@
   EXPECT_EQ(write_result.error_code(), error::RESOURCE_EXHAUSTED);
 }
 
+TEST_P(StorageQueueTest, WrappedRecordWithInsufficientMemoryWithFailure) {
+  CreateTestStorageQueueOrDie(BuildStorageQueueOptionsOnlyManual());
+
+  // Inject "low memory" error multiple times, then retire and return success.
+  auto inject = InjectFailures();
+  EXPECT_CALL(
+      *inject,
+      Call(Eq(test::StorageQueueOperationKind::kWrappedRecordLowMemory), Eq(0)))
+      .WillRepeatedly(WithArg<1>(Invoke([](int64_t seq_id) {
+        return Status(error::RESOURCE_EXHAUSTED,
+                      base::StrCat({"Not enough memory for WrappedRecord, seq=",
+                                    base::NumberToString(seq_id)}));
+      })))
+      .RetiresOnSaturation();
+  Record record;
+  record.set_data(std::string(kData[0]));
+  record.set_destination(UPLOAD_EVENTS);
+  if (!dm_token_.empty()) {
+    record.set_dm_token(dm_token_);
+  }
+  test::TestEvent<Status> write_event;
+  LOG(ERROR) << "Write data='" << record.data() << "'";
+  storage_queue_->Write(std::move(record), write_event.cb());
+  Status write_result = write_event.result();
+  EXPECT_FALSE(write_result.ok());
+  EXPECT_EQ(write_result.error_code(), error::RESOURCE_EXHAUSTED);
+}
+
+TEST_P(StorageQueueTest, EncryptedRecordWithInsufficientMemoryWithFailure) {
+  CreateTestStorageQueueOrDie(BuildStorageQueueOptionsOnlyManual());
+
+  // Inject "low memory" error multiple times, then retire and return success.
+  auto inject = InjectFailures();
+  EXPECT_CALL(
+      *inject,
+      Call(Eq(test::StorageQueueOperationKind::kEncryptedRecordLowMemory),
+           Eq(0)))
+      .WillRepeatedly(WithArg<1>(Invoke([](int64_t seq_id) {
+        return Status(
+            error::RESOURCE_EXHAUSTED,
+            base::StrCat({"Not enough memory for EncryptedRecord, seq=",
+                          base::NumberToString(seq_id)}));
+      })))
+      .RetiresOnSaturation();
+  Record record;
+  record.set_data(std::string(kData[0]));
+  record.set_destination(UPLOAD_EVENTS);
+  if (!dm_token_.empty()) {
+    record.set_dm_token(dm_token_);
+  }
+  test::TestEvent<Status> write_event;
+  LOG(ERROR) << "Write data='" << record.data() << "'";
+  storage_queue_->Write(std::move(record), write_event.cb());
+  Status write_result = write_event.result();
+  EXPECT_FALSE(write_result.ok());
+  EXPECT_EQ(write_result.error_code(), error::RESOURCE_EXHAUSTED);
+}
+
 TEST_P(StorageQueueTest, WriteRecordWithReservedSpace) {
   CreateTestStorageQueueOrDie(BuildStorageQueueOptionsOnlyManual());
 
diff --git a/components/saved_tab_groups/saved_tab_group_model.cc b/components/saved_tab_groups/saved_tab_group_model.cc
index 7c29cd8..e0ee868 100644
--- a/components/saved_tab_groups/saved_tab_group_model.cc
+++ b/components/saved_tab_groups/saved_tab_group_model.cc
@@ -226,11 +226,12 @@
   if (!Contains(group_id))
     return;
 
+  const base::GUID tab_id = tab.saved_tab_guid();
   absl::optional<int> group_index = GetIndexOf(group_id);
   saved_tab_groups_[group_index.value()].AddTab(index, tab);
 
   for (auto& observer : observers_)
-    observer.SavedTabGroupUpdatedLocally(group_id);
+    observer.SavedTabGroupUpdatedLocally(group_id, tab_id);
 }
 
 void SavedTabGroupModel::RemoveTabFromGroup(const base::GUID& group_id,
@@ -252,7 +253,7 @@
   // TODO(dljames): Update to use SavedTabGroupRemoveLocally and update the API
   // to pass a group_id and an optional tab_id.
   for (auto& observer : observers_)
-    observer.SavedTabGroupUpdatedLocally(group_id);
+    observer.SavedTabGroupUpdatedLocally(group_id, tab_id);
 }
 
 void SavedTabGroupModel::ReplaceTabInGroupAt(const base::GUID& group_id,
@@ -261,11 +262,14 @@
   if (!Contains(group_id))
     return;
 
+  const base::GUID guid = new_tab.saved_tab_guid();
   absl::optional<int> index = GetIndexOf(group_id);
   saved_tab_groups_[index.value()].ReplaceTabAt(tab_id, new_tab);
 
-  for (auto& observer : observers_)
-    observer.SavedTabGroupUpdatedLocally(group_id);
+  for (auto& observer : observers_) {
+    observer.SavedTabGroupUpdatedLocally(group_id, tab_id);
+    observer.SavedTabGroupUpdatedLocally(group_id, guid);
+  }
 }
 
 void SavedTabGroupModel::MoveTabInGroupTo(const base::GUID& group_id,
@@ -278,7 +282,7 @@
   saved_tab_groups_[index.value()].MoveTab(tab_id, new_index);
 
   for (auto& observer : observers_)
-    observer.SavedTabGroupUpdatedLocally(group_id);
+    observer.SavedTabGroupUpdatedLocally(group_id, tab_id);
 }
 
 std::unique_ptr<sync_pb::SavedTabGroupSpecifics> SavedTabGroupModel::MergeGroup(
@@ -315,7 +319,7 @@
   tab->MergeTab(std::move(sync_specific));
 
   for (auto& observer : observers_)
-    observer.SavedTabGroupUpdatedFromSync(tab->saved_group_guid());
+    observer.SavedTabGroupUpdatedFromSync(group_id, tab->saved_group_guid());
 
   return tab->ToSpecifics();
 }
diff --git a/components/saved_tab_groups/saved_tab_group_model_observer.h b/components/saved_tab_groups/saved_tab_group_model_observer.h
index f1eee43..4b4a9aed 100644
--- a/components/saved_tab_groups/saved_tab_group_model_observer.h
+++ b/components/saved_tab_groups/saved_tab_group_model_observer.h
@@ -9,6 +9,7 @@
 #include "components/saved_tab_groups/saved_tab_group_model.h"
 #include "components/saved_tab_groups/saved_tab_group_tab.h"
 #include "components/tab_groups/tab_group_id.h"
+#include "third_party/abseil-cpp/absl/types/optional.h"
 
 // Serves to notify any SavedTabGroupModel listeners that a change has occurred
 // supply the SavedTabGroup that was changed.
@@ -25,10 +26,13 @@
   virtual void SavedTabGroupRemovedLocally(const SavedTabGroup* removed_group) {
   }
 
-  // Called when the title, tabs, or color change.
-  // TODO(dljames): Update parameters to take 2 guids. 1 for the group and an
-  // optional guid for the tab. Do the same for the sync version.
-  virtual void SavedTabGroupUpdatedLocally(const base::GUID& guid) {}
+  // Called when the title, tabs, or color change. `group_guid` denotes the
+  // group that is currently being updated. `tab_guid` denotes if a tab in this
+  // group was changed (added, removed, updated). Otherwise, only the group is
+  // being changed.
+  virtual void SavedTabGroupUpdatedLocally(
+      const base::GUID& group_guid,
+      const absl::optional<base::GUID>& tab_guid = absl::nullopt) {}
 
   // Called when the order of saved tab groups in the bookmark bar are changed.
   // TODO(crbug/1372052): Figure out if we can maintain ordering of groups and
@@ -43,7 +47,13 @@
   virtual void SavedTabGroupRemovedFromSync(
       const SavedTabGroup* removed_group) {}
 
-  virtual void SavedTabGroupUpdatedFromSync(const base::GUID& guid) {}
+  // Called when the title, tabs, or color change. `group_guid` denotes the
+  // group that is currently being updated. `tab_guid` denotes if a tab in this
+  // group was changed (added, removed, updated). Specifically, this is called
+  // when addressing merge conflicts for duplicate groups and tabs.
+  virtual void SavedTabGroupUpdatedFromSync(
+      const base::GUID& group_guid,
+      const absl::optional<base::GUID>& tab_guid = absl::nullopt) {}
 
  protected:
   SavedTabGroupModelObserver() = default;
diff --git a/components/saved_tab_groups/saved_tab_group_model_unittest.cc b/components/saved_tab_groups/saved_tab_group_model_unittest.cc
index a38d946..8b28dc4 100644
--- a/components/saved_tab_groups/saved_tab_group_model_unittest.cc
+++ b/components/saved_tab_groups/saved_tab_group_model_unittest.cc
@@ -129,9 +129,12 @@
     retrieved_guid_ = removed_group->saved_guid();
   }
 
-  void SavedTabGroupUpdatedLocally(const base::GUID& guid) override {
-    retrieved_group_.emplace_back(*saved_tab_group_model_->Get(guid));
-    retrieved_index_ = saved_tab_group_model_->GetIndexOf(guid).value_or(-1);
+  void SavedTabGroupUpdatedLocally(
+      const base::GUID& group_guid,
+      const absl::optional<base::GUID>& tab_guid = absl::nullopt) override {
+    retrieved_group_.emplace_back(*saved_tab_group_model_->Get(group_guid));
+    retrieved_index_ =
+        saved_tab_group_model_->GetIndexOf(group_guid).value_or(-1);
   }
 
   void SavedTabGroupAddedFromSync(const base::GUID& guid) override {
@@ -144,9 +147,12 @@
     retrieved_guid_ = removed_group->saved_guid();
   }
 
-  void SavedTabGroupUpdatedFromSync(const base::GUID& guid) override {
-    retrieved_group_.emplace_back(*saved_tab_group_model_->Get(guid));
-    retrieved_index_ = saved_tab_group_model_->GetIndexOf(guid).value_or(-1);
+  void SavedTabGroupUpdatedFromSync(
+      const base::GUID& group_guid,
+      const absl::optional<base::GUID>& tab_guid = absl::nullopt) override {
+    retrieved_group_.emplace_back(*saved_tab_group_model_->Get(group_guid));
+    retrieved_index_ =
+        saved_tab_group_model_->GetIndexOf(group_guid).value_or(-1);
   }
 
   void SavedTabGroupReorderedLocally() override { reordered_called_ = true; }
@@ -1002,4 +1008,4 @@
             saved_tab_group_model_->GetGroupContainingTab(GenerateNextGUID()));
   EXPECT_EQ(nullptr,
             saved_tab_group_model_->GetGroupContainingTab(base::Token()));
-}
\ No newline at end of file
+}
diff --git a/components/saved_tab_groups/saved_tab_group_sync_bridge.cc b/components/saved_tab_groups/saved_tab_group_sync_bridge.cc
index f64eba4..4e0447a 100644
--- a/components/saved_tab_groups/saved_tab_group_sync_bridge.cc
+++ b/components/saved_tab_groups/saved_tab_group_sync_bridge.cc
@@ -231,15 +231,25 @@
 }
 
 void SavedTabGroupSyncBridge::SavedTabGroupUpdatedLocally(
-    const base::GUID& guid) {
+    const base::GUID& group_guid,
+    const absl::optional<base::GUID>& tab_guid) {
   std::unique_ptr<syncer::ModelTypeStore::WriteBatch> write_batch =
       store_->CreateWriteBatch();
 
-  const SavedTabGroup* group = model_->Get(guid);
+  const SavedTabGroup* group = model_->Get(group_guid);
   DCHECK(group);
-  UpsertEntitySpecific(group->ToSpecifics(), write_batch.get());
-  for (const SavedTabGroupTab& tab : group->saved_tabs())
-    UpsertEntitySpecific(tab.ToSpecifics(), write_batch.get());
+
+  if (tab_guid.has_value()) {
+    if (!group->ContainsTab(tab_guid.value())) {
+      RemoveEntitySpecific(tab_guid.value(), write_batch.get());
+    } else {
+      int tab_index = group->GetIndexOfTab(tab_guid.value()).value();
+      UpsertEntitySpecific(group->saved_tabs()[tab_index].ToSpecifics(),
+                           write_batch.get());
+    }
+  } else {
+    UpsertEntitySpecific(group->ToSpecifics(), write_batch.get());
+  }
 
   store_->CommitWriteBatch(
       std::move(write_batch),
diff --git a/components/saved_tab_groups/saved_tab_group_sync_bridge.h b/components/saved_tab_groups/saved_tab_group_sync_bridge.h
index dc6e6ce..0e882054 100644
--- a/components/saved_tab_groups/saved_tab_group_sync_bridge.h
+++ b/components/saved_tab_groups/saved_tab_group_sync_bridge.h
@@ -16,6 +16,7 @@
 #include "components/sync/model/model_type_change_processor.h"
 #include "components/sync/model/model_type_store.h"
 #include "components/sync/model/model_type_sync_bridge.h"
+#include "third_party/abseil-cpp/absl/types/optional.h"
 
 class SavedTabGroupModel;
 
@@ -59,7 +60,9 @@
   // SavedTabGroupModelObserver
   void SavedTabGroupAddedLocally(const base::GUID& guid) override;
   void SavedTabGroupRemovedLocally(const SavedTabGroup* removed_group) override;
-  void SavedTabGroupUpdatedLocally(const base::GUID& guid) override;
+  void SavedTabGroupUpdatedLocally(
+      const base::GUID& group_guid,
+      const absl::optional<base::GUID>& tab_guid = absl::nullopt) override;
   void SavedTabGroupReorderedLocally() override;
 
  private:
diff --git a/components/saved_tab_groups/saved_tab_group_sync_bridge_unittest.cc b/components/saved_tab_groups/saved_tab_group_sync_bridge_unittest.cc
index f60b8a8..88ff2ee 100644
--- a/components/saved_tab_groups/saved_tab_group_sync_bridge_unittest.cc
+++ b/components/saved_tab_groups/saved_tab_group_sync_bridge_unittest.cc
@@ -614,14 +614,10 @@
   base::GUID tab_2_guid = tab_2.saved_tab_guid();
   saved_tab_group_model_.Add(std::move(group));
 
-  EXPECT_CALL(processor_, Put(tab_1_guid.AsLowercaseString(), _, _));
-  EXPECT_CALL(processor_, Put(tab_2_guid.AsLowercaseString(), _, _));
   EXPECT_CALL(processor_, Put(group_guid.AsLowercaseString(), _, _));
+  EXPECT_CALL(processor_, Put(tab_1_guid.AsLowercaseString(), _, _)).Times(0);
+  EXPECT_CALL(processor_, Put(tab_2_guid.AsLowercaseString(), _, _)).Times(0);
 
-  // TODO(dljames): For now, updating a groups visual data also notifies
-  // observers of the tabs in the group. This is done for simplicity. Once the
-  // observer function changes are made this test will fail and we can remove
-  // the Put() calls for the tabs.
   tab_groups::TabGroupVisualData visual_data(
       u"New Title", tab_groups::TabGroupColorId::kYellow);
   saved_tab_group_model_.UpdateVisualData(group_guid, &visual_data);
@@ -646,10 +642,10 @@
   base::GUID tab_3_guid = tab_3.saved_tab_guid();
   saved_tab_group_model_.Add(std::move(group));
 
-  EXPECT_CALL(processor_, Put(tab_1_guid.AsLowercaseString(), _, _));
-  EXPECT_CALL(processor_, Put(tab_2_guid.AsLowercaseString(), _, _));
   EXPECT_CALL(processor_, Put(tab_3_guid.AsLowercaseString(), _, _));
-  EXPECT_CALL(processor_, Put(group_guid.AsLowercaseString(), _, _));
+  EXPECT_CALL(processor_, Put(tab_1_guid.AsLowercaseString(), _, _)).Times(0);
+  EXPECT_CALL(processor_, Put(tab_2_guid.AsLowercaseString(), _, _)).Times(0);
+  EXPECT_CALL(processor_, Put(group_guid.AsLowercaseString(), _, _)).Times(0);
 
   // TODO(dljames): Because `tab_3` was added to the middle of the group, only
   // `tab_2` will have its position updated. Once tab ordering is implemented,
@@ -659,8 +655,6 @@
 }
 
 // Verify that locally removed tabs remove the correct tabs from the processor.
-// TODO(dljames): Update the test with delete calls once the observer API is
-// updated.
 TEST_F(SavedTabGroupSyncBridgeTest, RemoveTabLocally) {
   EXPECT_TRUE(saved_tab_group_model_.saved_tab_groups().empty());
 
@@ -676,9 +670,9 @@
   base::GUID tab_2_guid = tab_2.saved_tab_guid();
   saved_tab_group_model_.Add(std::move(group));
 
-  EXPECT_CALL(processor_, Put(tab_1_guid.AsLowercaseString(), _, _)).Times(0);
-  EXPECT_CALL(processor_, Put(tab_2_guid.AsLowercaseString(), _, _));
-  EXPECT_CALL(processor_, Put(group_guid.AsLowercaseString(), _, _));
+  EXPECT_CALL(processor_, Delete(tab_1_guid.AsLowercaseString(), _));
+  EXPECT_CALL(processor_, Put(tab_2_guid.AsLowercaseString(), _, _)).Times(0);
+  EXPECT_CALL(processor_, Put(group_guid.AsLowercaseString(), _, _)).Times(0);
 
   saved_tab_group_model_.RemoveTabFromGroup(group_guid, tab_1_guid);
 }
@@ -702,12 +696,10 @@
   base::GUID tab_3_guid = tab_3.saved_tab_guid();
   saved_tab_group_model_.Add(std::move(group));
 
-  // TODO(dljames): Ensure Delete() is called on tab_1 once the observer api
-  // changes are made.
-  EXPECT_CALL(processor_, Put(tab_1_guid.AsLowercaseString(), _, _)).Times(0);
-  EXPECT_CALL(processor_, Put(tab_2_guid.AsLowercaseString(), _, _));
+  EXPECT_CALL(processor_, Delete(tab_1_guid.AsLowercaseString(), _));
   EXPECT_CALL(processor_, Put(tab_3_guid.AsLowercaseString(), _, _));
-  EXPECT_CALL(processor_, Put(group_guid.AsLowercaseString(), _, _));
+  EXPECT_CALL(processor_, Put(tab_2_guid.AsLowercaseString(), _, _)).Times(0);
+  EXPECT_CALL(processor_, Put(group_guid.AsLowercaseString(), _, _)).Times(0);
 
   saved_tab_group_model_.ReplaceTabInGroupAt(group_guid, tab_1_guid, tab_3);
 }
diff --git a/components/url_formatter/spoof_checks/top_domains/top_domain_state_generator.cc b/components/url_formatter/spoof_checks/top_domains/top_domain_state_generator.cc
index 2e170f1..b3d8574 100644
--- a/components/url_formatter/spoof_checks/top_domains/top_domain_state_generator.cc
+++ b/components/url_formatter/spoof_checks/top_domains/top_domain_state_generator.cc
@@ -108,7 +108,9 @@
   // most space efficient Huffman table for the given inputs. This table is used
   // for the second run.
 
+  // Tables must outlive `trie_entries` and `raw_trie_entries`.
   HuffmanRepresentationTable approximate_table = ApproximateHuffman(entries);
+  HuffmanRepresentationTable optimal_table;
   HuffmanBuilder huffman_builder;
 
   // Create trie entries for the first pass.
@@ -126,7 +128,7 @@
   if (!writer.WriteEntries(raw_trie_entries, &root_position))
     return std::string();
 
-  HuffmanRepresentationTable optimal_table = huffman_builder.ToTable();
+  optimal_table = huffman_builder.ToTable();
   TrieWriter new_writer(optimal_table, &huffman_builder);
 
   // Create trie entries using the optimal table for the second pass.
diff --git a/components/viz/service/display_embedder/output_presenter_gl.cc b/components/viz/service/display_embedder/output_presenter_gl.cc
index ae3b11d..c8f5020 100644
--- a/components/viz/service/display_embedder/output_presenter_gl.cc
+++ b/components/viz/service/display_embedder/output_presenter_gl.cc
@@ -18,7 +18,6 @@
 #include "components/viz/service/display_embedder/skia_output_surface_dependency.h"
 #include "gpu/command_buffer/common/shared_image_usage.h"
 #include "gpu/command_buffer/service/shared_context_state.h"
-#include "gpu/ipc/common/gpu_surface_lookup.h"
 #include "ui/display/types/display_snapshot.h"
 #include "ui/gfx/buffer_format_util.h"
 #include "ui/gfx/geometry/rect_conversions.h"
@@ -29,6 +28,7 @@
 #include "ui/gl/gl_surface.h"
 
 #if BUILDFLAG(IS_ANDROID)
+#include "gpu/ipc/common/gpu_surface_lookup.h"
 #include "ui/gl/android/scoped_a_native_window.h"
 #include "ui/gl/android/scoped_java_surface.h"
 #include "ui/gl/gl_surface_egl_surface_control.h"
diff --git a/components/viz/service/display_embedder/skia_output_device_dcomp.cc b/components/viz/service/display_embedder/skia_output_device_dcomp.cc
index 2efce022..d528235 100644
--- a/components/viz/service/display_embedder/skia_output_device_dcomp.cc
+++ b/components/viz/service/display_embedder/skia_output_device_dcomp.cc
@@ -90,7 +90,7 @@
     gpu::MailboxManager* mailbox_manager,
     gpu::SharedImageRepresentationFactory* shared_image_representation_factory,
     gpu::SharedContextState* context_state,
-    scoped_refptr<gl::GLSurface> gl_surface,
+    gl::GLSurface* gl_surface,
     scoped_refptr<gpu::gles2::FeatureInfo> feature_info,
     gpu::MemoryTracker* memory_tracker,
     DidSwapBufferCompleteCallback did_swap_buffer_complete_callback)
@@ -99,89 +99,60 @@
                        std::move(did_swap_buffer_complete_callback)),
       mailbox_manager_(mailbox_manager),
       shared_image_representation_factory_(shared_image_representation_factory),
-      context_state_(context_state),
-      gl_surface_(std::move(gl_surface)),
-      supports_async_swap_(gl_surface_->SupportsAsyncSwap()) {
+      context_state_(context_state) {
+  DCHECK(gl_surface->SupportsPostSubBuffer());
+  DCHECK(!gl_surface->SupportsAsyncSwap());
+  DCHECK(!feature_info->workarounds()
+              .disable_post_sub_buffers_for_onscreen_surfaces);
+  DCHECK(gl_surface->SupportsDCLayers());
+  DCHECK_EQ(gl_surface->GetOrigin(), gfx::SurfaceOrigin::kTopLeft);
+  DCHECK(gl_surface->SupportsGpuVSync());
+  DCHECK(!gl_surface->SupportsCommitOverlayPlanes());
+
   capabilities_.uses_default_gl_framebuffer = true;
-  capabilities_.output_surface_origin = gl_surface_->GetOrigin();
-  capabilities_.supports_post_sub_buffer = gl_surface_->SupportsPostSubBuffer();
-  if (gl_surface_->SupportsDCLayers()) {
-    // DWM handles preserving the contents of the backbuffer in Present1, so we
-    // don't need to have SkiaOutputSurface handle it.
-    capabilities_.preserve_buffer_content = false;
-    capabilities_.number_of_buffers =
-        gl::DirectCompositionRootSurfaceBufferCount();
-    capabilities_.supports_delegated_ink = gl_surface_->SupportsDelegatedInk();
-  }
-  if (feature_info->workarounds()
-          .disable_post_sub_buffers_for_onscreen_surfaces) {
-    capabilities_.supports_post_sub_buffer = false;
-  }
+  capabilities_.output_surface_origin = gfx::SurfaceOrigin::kTopLeft;
+  capabilities_.supports_post_sub_buffer = true;
+  // DWM handles preserving the contents of the backbuffer in Present1, so we
+  // don't need to have SkiaOutputSurface handle it.
+  capabilities_.preserve_buffer_content = false;
+  capabilities_.number_of_buffers =
+      gl::DirectCompositionRootSurfaceBufferCount();
+  capabilities_.supports_delegated_ink = gl_surface->SupportsDelegatedInk();
   if (feature_info->workarounds().supports_two_yuv_hardware_overlays) {
     capabilities_.supports_two_yuv_hardware_overlays = true;
   }
   capabilities_.pending_swap_params.max_pending_swaps =
-      gl_surface_->GetBufferCount() - 1;
-  capabilities_.supports_commit_overlay_planes =
-      gl_surface_->SupportsCommitOverlayPlanes();
-  capabilities_.supports_gpu_vsync = gl_surface_->SupportsGpuVSync();
-  capabilities_.supports_dc_layers = gl_surface_->SupportsDCLayers();
+      gl_surface->GetBufferCount() - 1;
+  capabilities_.supports_commit_overlay_planes = false;
+  capabilities_.supports_gpu_vsync = true;
+  capabilities_.supports_dc_layers = true;
 
   DCHECK(context_state_);
-  DCHECK(gl_surface_);
+  DCHECK(gl_surface);
 
-  if (gl_surface_->SupportsSwapTimestamps()) {
-    gl_surface_->SetEnableSwapTimestamps();
+  if (gl_surface->SupportsSwapTimestamps()) {
+    gl_surface->SetEnableSwapTimestamps();
 
     // Changes to swap timestamp queries are only picked up when making current.
     context_state_->ReleaseCurrent(nullptr);
-    context_state_->MakeCurrent(gl_surface_.get());
+    context_state_->MakeCurrent(gl_surface);
   }
 
   DCHECK(context_state_->gr_context());
   DCHECK(context_state_->context());
 
-  GrDirectContext* gr_context = context_state_->gr_context();
-  gl::CurrentGL* current_gl = context_state_->context()->GetCurrentGL();
-
-  // Get alpha bits from the default frame buffer.
-  int alpha_bits = 0;
-  glBindFramebufferEXT(GL_FRAMEBUFFER, 0);
-  gr_context->resetContext(kRenderTarget_GrGLBackendState);
-  const auto* version = current_gl->Version.get();
-  if (version->is_desktop_core_profile) {
-    glGetFramebufferAttachmentParameterivEXT(
-        GL_FRAMEBUFFER, GL_BACK_LEFT, GL_FRAMEBUFFER_ATTACHMENT_ALPHA_SIZE,
-        &alpha_bits);
-  } else {
-    glGetIntegerv(GL_ALPHA_BITS, &alpha_bits);
-  }
-  CHECK_GL_ERROR();
-
-  auto color_type = kRGBA_8888_SkColorType;
-
-  if (alpha_bits == 0) {
-    color_type = gl_surface_->GetFormat().GetBufferSize() == 16
-                     ? kRGB_565_SkColorType
-                     : kRGB_888x_SkColorType;
-    // Skia disables RGBx on some GPUs, fallback to RGBA if it's the
-    // case. This doesn't change framebuffer itself, as we already allocated it,
-    // but will change any temporary buffer Skia needs to allocate.
-    if (!context_state_->gr_context()
-             ->defaultBackendFormat(color_type, GrRenderable::kYes)
-             .isValid()) {
-      color_type = kRGBA_8888_SkColorType;
-    }
-  }
   // SRGB
+  constexpr SkColorType kSrgbColorType = kRGBA_8888_SkColorType;
+  // TODO(tangm): switch to kRGB_888x_SkColorType
+  constexpr SkColorType kSrgbColorTypeOpaque = kRGBA_8888_SkColorType;
   capabilities_.sk_color_types[static_cast<int>(gfx::BufferFormat::RGBA_8888)] =
-      color_type;
+      kSrgbColorType;
   capabilities_.sk_color_types[static_cast<int>(gfx::BufferFormat::RGBX_8888)] =
-      color_type;
+      kSrgbColorTypeOpaque;
   capabilities_.sk_color_types[static_cast<int>(gfx::BufferFormat::BGRA_8888)] =
-      color_type;
+      kSrgbColorType;
   capabilities_.sk_color_types[static_cast<int>(gfx::BufferFormat::BGRX_8888)] =
-      color_type;
+      kSrgbColorTypeOpaque;
   // HDR10
   capabilities_
       .sk_color_types[static_cast<int>(gfx::BufferFormat::RGBA_1010102)] =
@@ -191,12 +162,127 @@
       kRGBA_F16_SkColorType;
 }
 
-SkiaOutputDeviceDComp::~SkiaOutputDeviceDComp() {
+SkiaOutputDeviceDComp::~SkiaOutputDeviceDComp() = default;
+
+void SkiaOutputDeviceDComp::SwapBuffers(BufferPresentedCallback feedback,
+                                        OutputSurfaceFrame frame) {
+  NOTREACHED();
+}
+
+void SkiaOutputDeviceDComp::PostSubBuffer(const gfx::Rect& rect,
+                                          BufferPresentedCallback feedback,
+                                          OutputSurfaceFrame frame) {
+  StartSwapBuffers({});
+
+  gfx::SwapResult result =
+      DoPostSubBuffer(rect, std::move(feedback), frame.data);
+
+  // Remove entries from |overlays_| for textures that weren't scheduled as an
+  // overlay this frame.
+  if (!overlays_.empty()) {
+    base::EraseIf(overlays_, [this](auto& entry) {
+      const gpu::Mailbox& mailbox = entry.first;
+      return !scheduled_overlay_mailboxes_.contains(mailbox);
+    });
+    scheduled_overlay_mailboxes_.clear();
+    // End access for the remaining overlays that were scheduled this frame.
+    for (auto& kv : overlays_)
+      kv.second.EndOverlayAccess();
+  }
+
+  FinishSwapBuffers(gfx::SwapCompletionResult(result), GetRootSurfaceSize(),
+                    std::move(frame));
+}
+
+void SkiaOutputDeviceDComp::ScheduleOverlays(
+    SkiaOutputSurface::OverlayList overlays) {
+  for (auto& dc_layer : overlays) {
+    auto params = std::make_unique<ui::DCRendererLayerParams>();
+    // Get GLImages for DC layer textures.
+    bool success = true;
+    for (size_t i = 0; i < DCLayerOverlay::kNumResources; ++i) {
+      const gpu::Mailbox& mailbox = dc_layer.mailbox[i];
+      if (i > 0 && mailbox.IsZero())
+        break;
+
+      auto* read_access = BeginOverlayAccess(mailbox);
+      if (!read_access) {
+        success = false;
+        break;
+      }
+
+      if (auto dcomp_surface_proxy = read_access->GetDCOMPSurfaceProxy()) {
+        params->dcomp_surface_proxy = std::move(dcomp_surface_proxy);
+      } else if (auto* image = read_access->gl_image()) {
+        image->SetColorSpace(dc_layer.color_space);
+        params->images[i] = std::move(image);
+      } else {
+        success = false;
+        break;
+      }
+
+      scheduled_overlay_mailboxes_.insert(mailbox);
+    }
+
+    if (!success) {
+      DLOG(ERROR) << "Failed to get GLImage for DC layer.";
+      continue;
+    }
+
+    params->z_order = dc_layer.z_order;
+    params->content_rect = dc_layer.content_rect;
+    params->quad_rect = dc_layer.quad_rect;
+    DCHECK(dc_layer.transform.IsFlat());
+    params->transform = dc_layer.transform;
+    params->clip_rect = dc_layer.clip_rect;
+    params->protected_video_type = dc_layer.protected_video_type;
+    params->hdr_metadata = dc_layer.hdr_metadata;
+    params->is_video_fullscreen_letterboxing =
+        dc_layer.is_video_fullscreen_letterboxing;
+
+    // Schedule DC layer overlay to be presented at next SwapBuffers().
+    if (!ScheduleDCLayer(std::move(params)))
+      DLOG(ERROR) << "ScheduleDCLayer failed";
+  }
+}
+
+gpu::OverlayImageRepresentation::ScopedReadAccess*
+SkiaOutputDeviceDComp::BeginOverlayAccess(const gpu::Mailbox& mailbox) {
+  auto it = overlays_.find(mailbox);
+  if (it != overlays_.end())
+    return it->second.BeginOverlayAccess();
+
+  auto overlay = shared_image_representation_factory_->ProduceOverlay(mailbox);
+  if (!overlay)
+    return nullptr;
+
+  std::tie(it, std::ignore) = overlays_.emplace(mailbox, std::move(overlay));
+  return it->second.BeginOverlayAccess();
+}
+
+SkiaOutputDeviceDCompGLSurface::SkiaOutputDeviceDCompGLSurface(
+    gpu::MailboxManager* mailbox_manager,
+    gpu::SharedImageRepresentationFactory* shared_image_representation_factory,
+    gpu::SharedContextState* context_state,
+    scoped_refptr<gl::GLSurface> gl_surface,
+    scoped_refptr<gpu::gles2::FeatureInfo> feature_info,
+    gpu::MemoryTracker* memory_tracker,
+    DidSwapBufferCompleteCallback did_swap_buffer_complete_callback)
+    : SkiaOutputDeviceDComp(mailbox_manager,
+                            shared_image_representation_factory,
+                            context_state,
+                            gl_surface.get(),
+                            std::move(feature_info),
+                            memory_tracker,
+                            std::move(did_swap_buffer_complete_callback)),
+      gl_surface_(std::move(gl_surface)) {}
+
+SkiaOutputDeviceDCompGLSurface::~SkiaOutputDeviceDCompGLSurface() {
   // gl_surface_ will be destructed soon.
   memory_type_tracker_->TrackMemFree(backbuffer_estimated_size_);
 }
 
-bool SkiaOutputDeviceDComp::Reshape(
+bool SkiaOutputDeviceDCompGLSurface::Reshape(
     const SkSurfaceCharacterization& characterization,
     const gfx::ColorSpace& color_space,
     float device_scale_factor,
@@ -277,198 +363,42 @@
   return !!sk_surface_;
 }
 
-void SkiaOutputDeviceDComp::SwapBuffers(BufferPresentedCallback feedback,
-                                        OutputSurfaceFrame frame) {
-  StartSwapBuffers({});
-
-  gfx::Size surface_size =
-      gfx::Size(sk_surface_->width(), sk_surface_->height());
-
-  auto data = frame.data;
-  if (supports_async_swap_) {
-    auto callback = base::BindOnce(
-        &SkiaOutputDeviceDComp::DoFinishSwapBuffersAsync,
-        weak_ptr_factory_.GetWeakPtr(), surface_size, std::move(frame));
-    gl_surface_->SwapBuffersAsync(std::move(callback), std::move(feedback),
-                                  data);
-  } else {
-    gfx::SwapResult result =
-        gl_surface_->SwapBuffers(std::move(feedback), data);
-    DoFinishSwapBuffers(surface_size, std::move(frame),
-                        gfx::SwapCompletionResult(result));
-  }
-}
-
-void SkiaOutputDeviceDComp::PostSubBuffer(const gfx::Rect& rect,
-                                          BufferPresentedCallback feedback,
-                                          OutputSurfaceFrame frame) {
-  StartSwapBuffers({});
-
-  gfx::Size surface_size =
-      gfx::Size(sk_surface_->width(), sk_surface_->height());
-
-  auto data = frame.data;
-  if (supports_async_swap_) {
-    auto callback = base::BindOnce(
-        &SkiaOutputDeviceDComp::DoFinishSwapBuffersAsync,
-        weak_ptr_factory_.GetWeakPtr(), surface_size, std::move(frame));
-    gl_surface_->PostSubBufferAsync(rect.x(), rect.y(), rect.width(),
-                                    rect.height(), std::move(callback),
-                                    std::move(feedback), data);
-  } else {
-    gfx::SwapResult result =
-        gl_surface_->PostSubBuffer(rect.x(), rect.y(), rect.width(),
-                                   rect.height(), std::move(feedback), data);
-    DoFinishSwapBuffers(surface_size, std::move(frame),
-                        gfx::SwapCompletionResult(result));
-  }
-}
-
-void SkiaOutputDeviceDComp::CommitOverlayPlanes(
-    BufferPresentedCallback feedback,
-    OutputSurfaceFrame frame) {
-  StartSwapBuffers({});
-
-  gfx::Size surface_size =
-      gfx::Size(sk_surface_->width(), sk_surface_->height());
-
-  auto data = frame.data;
-  if (supports_async_swap_) {
-    auto callback = base::BindOnce(
-        &SkiaOutputDeviceDComp::DoFinishSwapBuffersAsync,
-        weak_ptr_factory_.GetWeakPtr(), surface_size, std::move(frame));
-    gl_surface_->CommitOverlayPlanesAsync(std::move(callback),
-                                          std::move(feedback), data);
-  } else {
-    gfx::SwapResult result =
-        gl_surface_->CommitOverlayPlanes(std::move(feedback), data);
-    DoFinishSwapBuffers(surface_size, std::move(frame),
-                        gfx::SwapCompletionResult(result));
-  }
-}
-
-void SkiaOutputDeviceDComp::DoFinishSwapBuffersAsync(
-    const gfx::Size& size,
-    OutputSurfaceFrame frame,
-    gfx::SwapCompletionResult result) {
-  DCHECK(result.release_fence.is_null());
-  FinishSwapBuffers(std::move(result), size, std::move(frame));
-}
-
-void SkiaOutputDeviceDComp::DoFinishSwapBuffers(
-    const gfx::Size& size,
-    OutputSurfaceFrame frame,
-    gfx::SwapCompletionResult result) {
-  DCHECK(result.release_fence.is_null());
-
-  // Remove entries from |overlays_| for textures that weren't scheduled as an
-  // overlay this frame.
-  if (!overlays_.empty()) {
-    base::EraseIf(overlays_, [this](auto& entry) {
-      const gpu::Mailbox& mailbox = entry.first;
-      return !scheduled_overlay_mailboxes_.contains(mailbox);
-    });
-    scheduled_overlay_mailboxes_.clear();
-    // End access for the remaining overlays that were scheduled this frame.
-    for (auto& kv : overlays_)
-      kv.second.EndOverlayAccess();
-  }
-
-  FinishSwapBuffers(std::move(result), size, std::move(frame));
-}
-
-bool SkiaOutputDeviceDComp::SetDrawRectangle(const gfx::Rect& draw_rectangle) {
+bool SkiaOutputDeviceDCompGLSurface::SetDrawRectangle(
+    const gfx::Rect& draw_rectangle) {
   return gl_surface_->SetDrawRectangle(draw_rectangle);
 }
 
-void SkiaOutputDeviceDComp::SetGpuVSyncEnabled(bool enabled) {
-  gl_surface_->SetGpuVSyncEnabled(enabled);
-}
-
-void SkiaOutputDeviceDComp::SetEnableDCLayers(bool enable) {
+void SkiaOutputDeviceDCompGLSurface::SetEnableDCLayers(bool enable) {
   gl_surface_->SetEnableDCLayers(enable);
 }
 
-void SkiaOutputDeviceDComp::ScheduleOverlays(
-    SkiaOutputSurface::OverlayList overlays) {
-  for (auto& dc_layer : overlays) {
-    auto params = std::make_unique<ui::DCRendererLayerParams>();
-    // Get GLImages for DC layer textures.
-    bool success = true;
-    for (size_t i = 0; i < DCLayerOverlay::kNumResources; ++i) {
-      const gpu::Mailbox& mailbox = dc_layer.mailbox[i];
-      if (i > 0 && mailbox.IsZero())
-        break;
-
-      auto* read_access = BeginOverlayAccess(mailbox);
-      if (!read_access) {
-        success = false;
-        break;
-      }
-
-      if (auto dcomp_surface_proxy = read_access->GetDCOMPSurfaceProxy()) {
-        params->dcomp_surface_proxy = std::move(dcomp_surface_proxy);
-      } else if (auto* image = read_access->gl_image()) {
-        image->SetColorSpace(dc_layer.color_space);
-        params->images[i] = std::move(image);
-      } else {
-        success = false;
-        break;
-      }
-
-      scheduled_overlay_mailboxes_.insert(mailbox);
-    }
-
-    if (!success) {
-      DLOG(ERROR) << "Failed to get GLImage for DC layer.";
-      continue;
-    }
-
-    params->z_order = dc_layer.z_order;
-    params->content_rect = dc_layer.content_rect;
-    params->quad_rect = dc_layer.quad_rect;
-    DCHECK(dc_layer.transform.IsFlat());
-    params->transform = dc_layer.transform;
-    params->clip_rect = dc_layer.clip_rect;
-    params->protected_video_type = dc_layer.protected_video_type;
-    params->hdr_metadata = dc_layer.hdr_metadata;
-    params->is_video_fullscreen_letterboxing =
-        dc_layer.is_video_fullscreen_letterboxing;
-
-    // Schedule DC layer overlay to be presented at next SwapBuffers().
-    if (!gl_surface_->ScheduleDCLayer(std::move(params)))
-      DLOG(ERROR) << "ScheduleDCLayer failed";
-  }
+void SkiaOutputDeviceDCompGLSurface::SetGpuVSyncEnabled(bool enabled) {
+  gl_surface_->SetGpuVSyncEnabled(enabled);
 }
 
-void SkiaOutputDeviceDComp::EnsureBackbuffer() {
-  gl_surface_->SetBackbufferAllocation(true);
-}
-
-void SkiaOutputDeviceDComp::DiscardBackbuffer() {
-  gl_surface_->SetBackbufferAllocation(false);
-}
-
-SkSurface* SkiaOutputDeviceDComp::BeginPaint(
+SkSurface* SkiaOutputDeviceDCompGLSurface::BeginPaint(
     std::vector<GrBackendSemaphore>* end_semaphores) {
   DCHECK(sk_surface_);
   return sk_surface_.get();
 }
 
-void SkiaOutputDeviceDComp::EndPaint() {}
+void SkiaOutputDeviceDCompGLSurface::EndPaint() {}
 
-gpu::OverlayImageRepresentation::ScopedReadAccess*
-SkiaOutputDeviceDComp::BeginOverlayAccess(const gpu::Mailbox& mailbox) {
-  auto it = overlays_.find(mailbox);
-  if (it != overlays_.end())
-    return it->second.BeginOverlayAccess();
+bool SkiaOutputDeviceDCompGLSurface::ScheduleDCLayer(
+    std::unique_ptr<ui::DCRendererLayerParams> params) {
+  return gl_surface_->ScheduleDCLayer(std::move(params));
+}
 
-  auto overlay = shared_image_representation_factory_->ProduceOverlay(mailbox);
-  if (!overlay)
-    return nullptr;
+gfx::Size SkiaOutputDeviceDCompGLSurface::GetRootSurfaceSize() const {
+  return gl_surface_->GetSize();
+}
 
-  std::tie(it, std::ignore) = overlays_.emplace(mailbox, std::move(overlay));
-  return it->second.BeginOverlayAccess();
+gfx::SwapResult SkiaOutputDeviceDCompGLSurface::DoPostSubBuffer(
+    const gfx::Rect& rect,
+    BufferPresentedCallback feedback,
+    gl::FrameData data) {
+  return gl_surface_->PostSubBuffer(rect.x(), rect.y(), rect.width(),
+                                    rect.height(), std::move(feedback), data);
 }
 
 }  // namespace viz
diff --git a/components/viz/service/display_embedder/skia_output_device_dcomp.h b/components/viz/service/display_embedder/skia_output_device_dcomp.h
index 3718f500..5aa6fbaf 100644
--- a/components/viz/service/display_embedder/skia_output_device_dcomp.h
+++ b/components/viz/service/display_embedder/skia_output_device_dcomp.h
@@ -32,64 +32,49 @@
 
 namespace viz {
 
-class SkiaOutputDeviceDComp final : public SkiaOutputDevice {
+// Base class for DComp-backed OutputDevices.
+class SkiaOutputDeviceDComp : public SkiaOutputDevice {
  public:
-  SkiaOutputDeviceDComp(
-      gpu::MailboxManager* mailbox_manager,
-      gpu::SharedImageRepresentationFactory*
-          shared_image_representation_factory,
-      gpu::SharedContextState* context_state,
-      scoped_refptr<gl::GLSurface> gl_surface,
-      scoped_refptr<gpu::gles2::FeatureInfo> feature_info,
-      gpu::MemoryTracker* memory_tracker,
-      DidSwapBufferCompleteCallback did_swap_buffer_complete_callback);
-
   SkiaOutputDeviceDComp(const SkiaOutputDeviceDComp&) = delete;
   SkiaOutputDeviceDComp& operator=(const SkiaOutputDeviceDComp&) = delete;
 
   ~SkiaOutputDeviceDComp() override;
 
   // SkiaOutputDevice implementation:
-  bool Reshape(const SkSurfaceCharacterization& characterization,
-               const gfx::ColorSpace& color_space,
-               float device_scale_factor,
-               gfx::OverlayTransform transform) override;
   void SwapBuffers(BufferPresentedCallback feedback,
                    OutputSurfaceFrame frame) override;
   void PostSubBuffer(const gfx::Rect& rect,
                      BufferPresentedCallback feedback,
                      OutputSurfaceFrame frame) override;
-  void CommitOverlayPlanes(BufferPresentedCallback feedback,
-                           OutputSurfaceFrame frame) override;
-  bool SetDrawRectangle(const gfx::Rect& draw_rectangle) override;
-  void SetGpuVSyncEnabled(bool enabled) override;
-  void SetEnableDCLayers(bool enable) override;
   void ScheduleOverlays(SkiaOutputSurface::OverlayList overlays) override;
-  void EnsureBackbuffer() override;
-  void DiscardBackbuffer() override;
-  SkSurface* BeginPaint(
-      std::vector<GrBackendSemaphore>* end_semaphores) override;
-  void EndPaint() override;
 
- private:
+ protected:
+  SkiaOutputDeviceDComp(
+      gpu::MailboxManager* mailbox_manager,
+      gpu::SharedImageRepresentationFactory*
+          shared_image_representation_factory,
+      gpu::SharedContextState* context_state,
+      gl::GLSurface* gl_surface,
+      scoped_refptr<gpu::gles2::FeatureInfo> feature_info,
+      gpu::MemoryTracker* memory_tracker,
+      DidSwapBufferCompleteCallback did_swap_buffer_complete_callback);
+
   class OverlayData;
 
-  // Use instead of calling FinishSwapBuffers() directly. On Windows this cleans
-  // up old entries in |overlays_|.
-  void DoFinishSwapBuffers(const gfx::Size& size,
-                           OutputSurfaceFrame frame,
-                           gfx::SwapCompletionResult result);
-  // Used as callback for SwapBuffersAsync and PostSubBufferAsync to finish
-  // operation
-  void DoFinishSwapBuffersAsync(const gfx::Size& size,
-                                OutputSurfaceFrame frame,
-                                gfx::SwapCompletionResult result);
-
   gpu::OverlayImageRepresentation::ScopedReadAccess* BeginOverlayAccess(
       const gpu::Mailbox& mailbox);
 
   void CreateSkSurface();
 
+  virtual bool ScheduleDCLayer(
+      std::unique_ptr<ui::DCRendererLayerParams> params) = 0;
+
+  virtual gfx::Size GetRootSurfaceSize() const = 0;
+
+  virtual gfx::SwapResult DoPostSubBuffer(const gfx::Rect& rect,
+                                          BufferPresentedCallback feedback,
+                                          gl::FrameData data) = 0;
+
   // Mailboxes of overlays scheduled in the current frame.
   base::flat_set<gpu::Mailbox> scheduled_overlay_mailboxes_;
 
@@ -102,21 +87,57 @@
       shared_image_representation_factory_;
 
   const raw_ptr<gpu::SharedContextState> context_state_;
-  scoped_refptr<gl::GLSurface> gl_surface_;
-  const bool supports_async_swap_;
-
-  uint64_t backbuffer_estimated_size_ = 0;
-
-  gfx::Size size_;
-  SkColorType color_type_;
-  gfx::ColorSpace color_space_;
-  GrGLFramebufferInfo framebuffer_info_ = {};
-  int sample_count_ = 1;
-  sk_sp<SkSurface> sk_surface_;
 
   base::WeakPtrFactory<SkiaOutputDeviceDComp> weak_ptr_factory_{this};
 };
 
+// A DComp-backed OutputDevice whose root surface is wrapped in a GLSurface.
+class VIZ_SERVICE_EXPORT SkiaOutputDeviceDCompGLSurface final
+    : public SkiaOutputDeviceDComp {
+ public:
+  SkiaOutputDeviceDCompGLSurface(
+      gpu::MailboxManager* mailbox_manager,
+      gpu::SharedImageRepresentationFactory*
+          shared_image_representation_factory,
+      gpu::SharedContextState* context_state,
+      scoped_refptr<gl::GLSurface> gl_surface,
+      scoped_refptr<gpu::gles2::FeatureInfo> feature_info,
+      gpu::MemoryTracker* memory_tracker,
+      DidSwapBufferCompleteCallback did_swap_buffer_complete_callback);
+
+  ~SkiaOutputDeviceDCompGLSurface() override;
+
+  // SkiaOutputDevice implementation:
+  bool Reshape(const SkSurfaceCharacterization& characterization,
+               const gfx::ColorSpace& color_space,
+               float device_scale_factor,
+               gfx::OverlayTransform transform) override;
+  bool SetDrawRectangle(const gfx::Rect& draw_rectangle) override;
+  void SetEnableDCLayers(bool enable) override;
+  void SetGpuVSyncEnabled(bool enabled) override;
+  SkSurface* BeginPaint(
+      std::vector<GrBackendSemaphore>* end_semaphores) override;
+  void EndPaint() override;
+
+ protected:
+  bool ScheduleDCLayer(
+      std::unique_ptr<ui::DCRendererLayerParams> params) override;
+  gfx::Size GetRootSurfaceSize() const override;
+  gfx::SwapResult DoPostSubBuffer(const gfx::Rect& rect,
+                                  BufferPresentedCallback feedback,
+                                  gl::FrameData data) override;
+
+ private:
+  scoped_refptr<gl::GLSurface> gl_surface_;
+
+  gfx::Size size_;
+  gfx::ColorSpace color_space_;
+  GrGLFramebufferInfo framebuffer_info_ = {};
+  sk_sp<SkSurface> sk_surface_;
+
+  uint64_t backbuffer_estimated_size_ = 0;
+};
+
 }  // namespace viz
 
 #endif  // COMPONENTS_VIZ_SERVICE_DISPLAY_EMBEDDER_SKIA_OUTPUT_DEVICE_DCOMP_H_
diff --git a/components/viz/service/display_embedder/skia_output_device_vulkan.cc b/components/viz/service/display_embedder/skia_output_device_vulkan.cc
index 11529228..d5da933 100644
--- a/components/viz/service/display_embedder/skia_output_device_vulkan.cc
+++ b/components/viz/service/display_embedder/skia_output_device_vulkan.cc
@@ -14,7 +14,6 @@
 #include "components/viz/common/gpu/vulkan_context_provider.h"
 #include "gpu/command_buffer/service/memory_tracking.h"
 #include "gpu/config/gpu_finch_features.h"
-#include "gpu/ipc/common/gpu_surface_lookup.h"
 #include "gpu/vulkan/vulkan_fence_helper.h"
 #include "gpu/vulkan/vulkan_function_pointers.h"
 #include "gpu/vulkan/vulkan_implementation.h"
@@ -28,6 +27,7 @@
 
 #if BUILDFLAG(IS_ANDROID)
 #include <android/native_window_jni.h>
+#include "gpu/ipc/common/gpu_surface_lookup.h"
 #include "ui/gl/android/scoped_a_native_window.h"
 #include "ui/gl/android/scoped_java_surface.h"
 #endif
diff --git a/components/viz/service/display_embedder/skia_output_surface_impl_on_gpu.cc b/components/viz/service/display_embedder/skia_output_surface_impl_on_gpu.cc
index 59b7970..394aec6 100644
--- a/components/viz/service/display_embedder/skia_output_surface_impl_on_gpu.cc
+++ b/components/viz/service/display_embedder/skia_output_surface_impl_on_gpu.cc
@@ -1750,7 +1750,7 @@
         } else {
 #if BUILDFLAG(IS_WIN)
           if (gl_surface_->SupportsDCLayers()) {
-            output_device_ = std::make_unique<SkiaOutputDeviceDComp>(
+            output_device_ = std::make_unique<SkiaOutputDeviceDCompGLSurface>(
                 dependency_->GetMailboxManager(),
                 shared_image_representation_factory_.get(),
                 context_state_.get(), gl_surface_, feature_info_,
diff --git a/components/webxr/android/arcore_java_utils.cc b/components/webxr/android/arcore_java_utils.cc
index 028a871..7d41876 100644
--- a/components/webxr/android/arcore_java_utils.cc
+++ b/components/webxr/android/arcore_java_utils.cc
@@ -13,6 +13,7 @@
 #include "device/vr/android/arcore/arcore_shim.h"
 #include "gpu/ipc/common/gpu_surface_tracker.h"
 #include "ui/android/window_android.h"
+#include "ui/gl/android/scoped_java_surface.h"
 
 using base::android::AttachCurrentThread;
 using base::android::ScopedJavaLocalRef;
@@ -80,7 +81,8 @@
   gpu::SurfaceHandle surface_handle =
       gpu::GpuSurfaceTracker::Get()->AddSurfaceForNativeWidget(
           gpu::GpuSurfaceTracker::SurfaceRecord(
-              window, surface, /*can_be_used_with_surface_control=*/false));
+              gl::ScopedJavaSurface(surface, /*auto_release=*/false),
+              /*can_be_used_with_surface_control=*/false));
   ui::WindowAndroid* root_window =
       ui::WindowAndroid::FromJavaWindowAndroid(java_root_window);
   display::Display::Rotation display_rotation =
diff --git a/components/webxr/mailbox_to_surface_bridge_impl.cc b/components/webxr/mailbox_to_surface_bridge_impl.cc
index eed2e4b..c54cfae3 100644
--- a/components/webxr/mailbox_to_surface_bridge_impl.cc
+++ b/components/webxr/mailbox_to_surface_bridge_impl.cc
@@ -220,10 +220,9 @@
   gpu::GpuSurfaceTracker* tracker = gpu::GpuSurfaceTracker::Get();
   ANativeWindow_acquire(window);
   // Skip ANativeWindow_setBuffersGeometry, the default size appears to work.
-  surface_ = std::make_unique<gl::ScopedJavaSurface>(surface_texture);
   surface_handle_ =
       tracker->AddSurfaceForNativeWidget(gpu::GpuSurfaceTracker::SurfaceRecord(
-          window, surface_->j_surface(),
+          gl::ScopedJavaSurface(surface_texture),
           false /* can_be_used_with_surface_control */));
   // Unregistering happens in the destructor.
   ANativeWindow_release(window);
diff --git a/components/webxr/mailbox_to_surface_bridge_impl.h b/components/webxr/mailbox_to_surface_bridge_impl.h
index 2eb12c30..fc996c05d 100644
--- a/components/webxr/mailbox_to_surface_bridge_impl.h
+++ b/components/webxr/mailbox_to_surface_bridge_impl.h
@@ -88,7 +88,6 @@
   void DrawQuad(unsigned int textureHandle, const gfx::Transform& uv_transform);
 
   scoped_refptr<viz::ContextProvider> context_provider_;
-  std::unique_ptr<gl::ScopedJavaSurface> surface_;
   raw_ptr<gpu::gles2::GLES2Interface> gl_ = nullptr;
   raw_ptr<gpu::ContextSupport> context_support_ = nullptr;
   int surface_handle_ = gpu::kNullSurfaceHandle;
diff --git a/content/app/android/content_child_process_service_delegate.cc b/content/app/android/content_child_process_service_delegate.cc
index fbdd977..7acb6a65d 100644
--- a/content/app/android/content_child_process_service_delegate.cc
+++ b/content/app/android/content_child_process_service_delegate.cc
@@ -76,7 +76,8 @@
       return gl::ScopedJavaSurface();
 
     gl::ScopedJavaSurface surface(
-        content::JNI_SurfaceWrapper_getSurface(env, surface_wrapper));
+        content::JNI_SurfaceWrapper_takeSurface(env, surface_wrapper),
+        /*auto_release=*/true);
     DCHECK(!surface.j_surface().is_null());
 
     *can_be_used_with_surface_control =
diff --git a/content/browser/android/dialog_overlay_impl.cc b/content/browser/android/dialog_overlay_impl.cc
index 379bf3c..9e5a1b7 100644
--- a/content/browser/android/dialog_overlay_impl.cc
+++ b/content/browser/android/dialog_overlay_impl.cc
@@ -327,8 +327,8 @@
   DCHECK_CURRENTLY_ON(BrowserThread::UI);
   return gpu::GpuSurfaceTracker::Get()->AddSurfaceForNativeWidget(
       gpu::GpuSurfaceTracker::SurfaceRecord(
-          gfx::kNullAcceleratedWidget, surface,
-          false /* can_be_used_with_surface_control */));
+          gl::ScopedJavaSurface(surface, /*auto_release=*/false),
+          /*can_be_used_with_surface_control=*/false));
 }
 
 static void JNI_DialogOverlayImpl_UnregisterSurface(
diff --git a/content/browser/android/gpu_process_callback.cc b/content/browser/android/gpu_process_callback.cc
index aabd7bc..bfafcc1d 100644
--- a/content/browser/android/gpu_process_callback.cc
+++ b/content/browser/android/gpu_process_callback.cc
@@ -30,7 +30,7 @@
   base::android::ScopedJavaGlobalRef<jobject> jsurface;
   jsurface.Reset(env, surface);
   ScopedSurfaceRequestManager::GetInstance()->FulfillScopedSurfaceRequest(
-      requestToken, gl::ScopedJavaSurface(jsurface));
+      requestToken, gl::ScopedJavaSurface(jsurface, /*auto_release=*/true));
 }
 
 base::android::ScopedJavaLocalRef<jobject>
diff --git a/content/browser/attribution_reporting/attribution_internals.mojom b/content/browser/attribution_reporting/attribution_internals.mojom
index dd9fb0c..8d908b6 100644
--- a/content/browser/attribution_reporting/attribution_internals.mojom
+++ b/content/browser/attribution_reporting/attribution_internals.mojom
@@ -89,8 +89,9 @@
   DebugReportStatus status;
 };
 
-struct DebugKey {
-  uint64 value;
+union SourceDebugKey {
+  uint64 debug_key;
+  uint64 cleared_debug_key;
 };
 
 // Struct representing a stored attribution source that will be displayed by WebUI.
@@ -106,7 +107,7 @@
   double aggregatable_report_window_time;
   attribution_reporting.mojom.SourceType source_type;
   int64 priority;
-  DebugKey? debug_key;
+  SourceDebugKey? debug_key;
   array<uint64> dedup_keys;
   map<string, array<string>> filter_data;
   // The value is a hex-encoded unsigned 128-bit integer.
@@ -130,6 +131,10 @@
   Attributability attributability;
 };
 
+struct TriggerDebugKey {
+  uint64 value;
+};
+
 struct WebUITrigger {
   double trigger_time;
   url.mojom.Origin destination_origin;
@@ -137,6 +142,7 @@
   // String instead of mojo_base.mojom.DictionaryValue because the value is
   // simply displayed in the UI, never inspected.
   string registration_json;
+  TriggerDebugKey? cleared_debug_key;
 
   enum Status {
     // Shared statuses:
@@ -176,21 +182,6 @@
   attribution_reporting.mojom.SourceRegistrationError error;
 };
 
-// Struct containing details for a debug key that was removed from a source or
-// trigger due to a missing cookie to be displayed as a log by WebUI
-struct ClearedDebugKey {
-  double time;
-  url.mojom.Origin reporting_origin;
-  DebugKey cleared_debug_key;
-
-  enum Type {
-    kTrigger,
-    kSource,
-  };
-
-  Type cleared_from;
-};
-
 // Observer for events relevant to the attribution internals WebUI.
 interface Observer {
   // Called when the sources in storage changed, indicating that the observer
@@ -219,9 +210,6 @@
 
   // Called when the source header registration JSON parser fails.
   OnFailedSourceRegistration(FailedSourceRegistration failure);
-
-  // Called when a debug key was removed from a source or trigger due to a missing cookie.
-  OnDebugKeyCleared(ClearedDebugKey notification);
 };
 
 // Mojo interface for the attribution internals WebUI to communicate with the
diff --git a/content/browser/attribution_reporting/attribution_internals_browsertest.cc b/content/browser/attribution_reporting/attribution_internals_browsertest.cc
index abe25398..d95e016 100644
--- a/content/browser/attribution_reporting/attribution_internals_browsertest.cc
+++ b/content/browser/attribution_reporting/attribution_internals_browsertest.cc
@@ -299,7 +299,8 @@
 
   manager()->NotifySourceHandled(
       SourceBuilder(now + base::Hours(5)).Build(),
-      StorableSource::Result::kInsufficientSourceCapacity);
+      StorableSource::Result::kInsufficientSourceCapacity,
+      /*cleared_debug_key=*/987);
 
   manager()->NotifySourceHandled(
       SourceBuilder(now + base::Hours(6)).Build(),
@@ -327,6 +328,7 @@
           table.children[1].children[13]?.innerText === "1300 / 65536" &&
           table.children[0].children[14]?.innerText === "19" &&
           table.children[1].children[14]?.innerText === "" &&
+          table.children[4].children[14]?.innerText === 'Cleared (was 987)' &&
           table.children[0].children[15]?.innerText === "" &&
           table.children[1].children[15]?.innerText === "13, 17" &&
           table.children[0].children[16]?.innerText === "" &&
@@ -392,89 +394,6 @@
 }
 
 IN_PROC_BROWSER_TEST_F(AttributionInternalsWebUiBrowserTest,
-                       ClearedDebugKeyFromSource_LogShown) {
-  ASSERT_TRUE(NavigateToURL(shell(), GURL(kAttributionInternalsUrl)));
-
-  static constexpr char wait_script[] = R"(
-    const table = document.querySelector('#logTable')
-        .shadowRoot.querySelector('tbody');
-
-    const description = '<a href="https://github.com/WICG/attribution-report' +
-                    'ing-api/blob/main/EVENT.md#attribution-success-debugging-' +
-                    'reports" target="_blank">Cleared Debug Key</a>';
-    const metadata = '<dl><dt>Cleared Debug Key</dt><dd>1234</dd>' +
-                     '<dt>From</dt><dd>Source</dd>'+
-                     '<dt>Reporting Origin</dt><dd>https://report.test</dd></dl>';
-
-    let obs = new MutationObserver((_, obs) => {
-      if (table.children.length === 1 &&
-          table.children[0].children[1]?.innerHTML === description &&
-          table.children[0].children[2]?.innerHTML === metadata
-      ) {
-        obs.disconnect();
-        document.title = $1;
-      }
-    });
-    obs.observe(table, {childList: true, subtree: true, characterData: true});)";
-
-  ASSERT_TRUE(ExecJsInWebUI(JsReplace(wait_script, kCompleteTitle)));
-
-  TitleWatcher title_watcher(shell()->web_contents(), kCompleteTitle);
-
-  manager()->NotifySourceHandled(SourceBuilder().Build(),
-                                 StorableSource::Result::kSuccess,
-                                 /*cleared_debug_key=*/1234);
-
-  EXPECT_EQ(kCompleteTitle, title_watcher.WaitAndGetTitle());
-}
-
-IN_PROC_BROWSER_TEST_F(AttributionInternalsWebUiBrowserTest,
-                       ClearedDebugKeyFromTrigger_LogShown) {
-  ASSERT_TRUE(NavigateToURL(shell(), GURL(kAttributionInternalsUrl)));
-
-  static constexpr char wait_script[] = R"(
-    const table = document.querySelector('#logTable')
-        .shadowRoot.querySelector('tbody');
-
-    const description = '<a href="https://github.com/WICG/attribution-report' +
-                    'ing-api/blob/main/EVENT.md#attribution-success-debugging-' +
-                    'reports" target="_blank">Cleared Debug Key</a>';
-    const metadata = '<dl><dt>Cleared Debug Key</dt><dd>1234</dd>' +
-                     '<dt>From</dt><dd>Trigger</dd>'+
-                     '<dt>Reporting Origin</dt><dd>https://report.test</dd></dl>';
-
-    let obs = new MutationObserver((_, obs) => {
-      if (table.children.length === 1 &&
-          table.children[0].children[1]?.innerHTML === description &&
-          table.children[0].children[2]?.innerHTML === metadata
-      ) {
-        obs.disconnect();
-        document.title = $1;
-      }
-    });
-    obs.observe(table, {childList: true, subtree: true, characterData: true});)";
-
-  ASSERT_TRUE(ExecJsInWebUI(JsReplace(wait_script, kCompleteTitle)));
-
-  TitleWatcher title_watcher(shell()->web_contents(), kCompleteTitle);
-
-  manager()->NotifyTriggerHandled(
-      DefaultTrigger(),
-      CreateReportResult(
-          /*trigger_time=*/base::Time::Now(),
-          /*event_level_status=*/AttributionTrigger::EventLevelResult::kSuccess,
-          /*aggregatable_status=*/
-          AttributionTrigger::AggregatableResult::kSuccess,
-          /*replaced_event_level_report=*/absl::nullopt,
-          /*new_event_level_report=*/IrreleventEventLevelReport(),
-          /*new_aggregatable_report=*/IrreleventAggregatableReport(),
-          /*source=*/SourceBuilder().BuildStored()),
-      /*cleared_debug_key=*/1234);
-
-  EXPECT_EQ(kCompleteTitle, title_watcher.WaitAndGetTitle());
-}
-
-IN_PROC_BROWSER_TEST_F(AttributionInternalsWebUiBrowserTest,
                        WebUIShownWithNoReports_NoReportsDisplayed) {
   ASSERT_TRUE(NavigateToURL(shell(), GURL(kAttributionInternalsUrl)));
 
@@ -1091,12 +1010,14 @@
       let table = document.querySelector('#triggerTable')
           .shadowRoot.querySelector('tbody');
       let obs = new MutationObserver((_, obs) => {
-        if (table.children.length === 1 &&
+        if (table.children.length === 2 &&
             table.children[0].children[1]?.innerText === "Success: Report stored" &&
             table.children[0].children[2]?.innerText === "Success: Report stored" &&
             table.children[0].children[3]?.innerText === "https://d.test" &&
             table.children[0].children[4]?.innerText === "https://r.test" &&
-            table.children[0].children[5]?.innerText.includes('{')) {
+            table.children[0].children[5]?.innerText.includes('{') &&
+            table.children[0].children[6]?.innerText === '' &&
+            table.children[1].children[6]?.innerText === '123') {
           obs.disconnect();
           document.title = $1;
         }
@@ -1106,7 +1027,8 @@
 
   auto notify_trigger_handled =
       [&](AttributionTrigger::EventLevelResult event_status,
-          AttributionTrigger::AggregatableResult aggregatable_status) {
+          AttributionTrigger::AggregatableResult aggregatable_status,
+          absl::optional<uint64_t> cleared_debug_key = absl::nullopt) {
         static int offset_hours = 0;
         manager()->NotifyTriggerHandled(
             trigger,
@@ -1116,12 +1038,17 @@
                 /*replaced_event_level_report=*/absl::nullopt,
                 /*new_event_level_report=*/IrreleventEventLevelReport(),
                 /*new_aggregatable_report=*/IrreleventAggregatableReport(),
-                /*source=*/SourceBuilder().BuildStored()));
+                /*source=*/SourceBuilder().BuildStored()),
+            cleared_debug_key);
       };
 
   notify_trigger_handled(AttributionTrigger::EventLevelResult::kSuccess,
                          AttributionTrigger::AggregatableResult::kSuccess);
 
+  notify_trigger_handled(AttributionTrigger::EventLevelResult::kSuccess,
+                         AttributionTrigger::AggregatableResult::kSuccess,
+                         /*cleared_debug_key=*/123);
+
   // TODO(apaseltiner): Add tests for other statuses.
 
   TitleWatcher title_watcher(shell()->web_contents(), kCompleteTitle);
diff --git a/content/browser/attribution_reporting/attribution_internals_handler_impl.cc b/content/browser/attribution_reporting/attribution_internals_handler_impl.cc
index 1896281..f0b6bf0 100644
--- a/content/browser/attribution_reporting/attribution_internals_handler_impl.cc
+++ b/content/browser/attribution_reporting/attribution_internals_handler_impl.cc
@@ -54,34 +54,38 @@
 using Attributability =
     ::attribution_internals::mojom::WebUISource::Attributability;
 
-using SourceOrTrigger = ::attribution_internals::mojom::ClearedDebugKey::Type;
 using Empty = ::attribution_internals::mojom::Empty;
 using ReportStatus = ::attribution_internals::mojom::ReportStatus;
 using ReportStatusPtr = ::attribution_internals::mojom::ReportStatusPtr;
 
 using ::attribution_internals::mojom::WebUIDebugReport;
 
-attribution_internals::mojom::DebugKeyPtr WebUIDebugKey(
-    absl::optional<uint64_t> debug_key) {
-  return debug_key ? attribution_internals::mojom::DebugKey::New(*debug_key)
-                   : nullptr;
-}
-
 attribution_internals::mojom::WebUISourcePtr WebUISource(
     const CommonSourceInfo& source,
     Attributability attributability,
     const std::vector<uint64_t>& dedup_keys,
     int64_t aggregatable_budget_consumed,
     const std::vector<uint64_t>& aggregatable_dedup_keys,
-    bool debug_reporting_enabled) {
+    bool debug_reporting_enabled,
+    absl::optional<uint64_t> cleared_debug_key) {
   DCHECK_GE(aggregatable_budget_consumed, 0);
+
+  attribution_internals::mojom::SourceDebugKeyPtr debug_key =
+      cleared_debug_key
+          ? attribution_internals::mojom::SourceDebugKey::NewClearedDebugKey(
+                *cleared_debug_key)
+          : (source.debug_key()
+                 ? attribution_internals::mojom::SourceDebugKey::NewDebugKey(
+                       *source.debug_key())
+                 : nullptr);
+
   return attribution_internals::mojom::WebUISource::New(
       source.source_event_id(), source.source_origin(),
       source.DestinationSite().Serialize(), source.reporting_origin(),
       source.source_time().ToJsTime(), source.expiry_time().ToJsTime(),
       source.event_report_window_time().ToJsTime(),
       source.aggregatable_report_window_time().ToJsTime(), source.source_type(),
-      source.priority(), WebUIDebugKey(source.debug_key()), dedup_keys,
+      source.priority(), std::move(debug_key), dedup_keys,
       source.filter_data().filter_values(),
       base::MakeFlatMap<std::string, std::string>(
           source.aggregation_keys().keys(), {},
@@ -125,7 +129,8 @@
     web_ui_sources.push_back(WebUISource(
         source.common_info(), attributability, source.dedup_keys(),
         source.aggregatable_budget_consumed(), source.aggregatable_dedup_keys(),
-        /*debug_reporting_enabled=*/false));
+        /*debug_reporting_enabled=*/false,
+        /*cleared_debug_key=*/absl::nullopt));
   }
 
   std::move(web_ui_callback).Run(std::move(web_ui_sources));
@@ -298,18 +303,6 @@
     absl::optional<uint64_t> cleared_debug_key,
     StorableSource::Result result) {
   Attributability attributability;
-  if (cleared_debug_key.has_value()) {
-    auto web_ui_log = attribution_internals::mojom::ClearedDebugKey::New();
-    web_ui_log->cleared_debug_key = WebUIDebugKey(cleared_debug_key.value());
-    web_ui_log->time = source.common_info().source_time().ToJsTime();
-    web_ui_log->reporting_origin = source.common_info().reporting_origin();
-    web_ui_log->cleared_from = SourceOrTrigger::kSource;
-
-    for (auto& observer : observers_) {
-      observer->OnDebugKeyCleared(web_ui_log.Clone());
-    }
-  }
-
   switch (result) {
     case StorableSource::Result::kSuccess:
     // TODO(linnan): Consider displaying source noised in internals UI.
@@ -335,7 +328,8 @@
   auto web_ui_source =
       WebUISource(source.common_info(), attributability, /*dedup_keys=*/{},
                   /*aggregatable_budget_consumed=*/0,
-                  /*aggregatable_dedup_keys=*/{}, source.debug_reporting());
+                  /*aggregatable_dedup_keys=*/{}, source.debug_reporting(),
+                  cleared_debug_key);
 
   for (auto& observer : observers_) {
     observer->OnSourceRejected(web_ui_source.Clone());
@@ -496,17 +490,6 @@
   const attribution_reporting::TriggerRegistration& registration =
       trigger.registration();
 
-  if (cleared_debug_key.has_value()) {
-    auto web_ui_log = attribution_internals::mojom::ClearedDebugKey::New();
-    web_ui_log->cleared_debug_key = WebUIDebugKey(cleared_debug_key.value());
-    web_ui_log->time = result.trigger_time().ToJsTime();
-    web_ui_log->reporting_origin = trigger.reporting_origin();
-    web_ui_log->cleared_from = SourceOrTrigger::kTrigger;
-
-    for (auto& observer : observers_) {
-      observer->OnDebugKeyCleared(web_ui_log.Clone());
-    }
-  }
   auto web_ui_trigger = attribution_internals::mojom::WebUITrigger::New();
   web_ui_trigger->trigger_time = result.trigger_time().ToJsTime();
   web_ui_trigger->destination_origin = trigger.destination_origin();
@@ -514,6 +497,10 @@
   web_ui_trigger->registration_json =
       SerializeAttributionJson(registration.ToJson(),
                                /*pretty_print=*/true);
+  web_ui_trigger->cleared_debug_key =
+      cleared_debug_key ? attribution_internals::mojom::TriggerDebugKey::New(
+                              *cleared_debug_key)
+                        : nullptr;
   web_ui_trigger->event_level_status =
       GetWebUITriggerStatus(result.event_level_status());
   web_ui_trigger->aggregatable_status =
diff --git a/content/browser/back_forward_cache_no_store_browsertest.cc b/content/browser/back_forward_cache_no_store_browsertest.cc
index 3262bdcc..194674c 100644
--- a/content/browser/back_forward_cache_no_store_browsertest.cc
+++ b/content/browser/back_forward_cache_no_store_browsertest.cc
@@ -750,12 +750,12 @@
 namespace {
 // Causes a fetch using the "Authorization" header to start and complete in the
 // target frame.
-void UseAuthorizationHeaderFetch(const ToRenderFrameHost& execution_target,
-                                 const GURL& url) {
+void UseAuthorizationHeader(const ToRenderFrameHost& execution_target,
+                            GURL url) {
   ASSERT_EQ(42, EvalJs(execution_target, JsReplace(
                                              R"(
       fetch($1, {headers: {Authorization: 'foo'}})
-          .then(p => {
+          .then(p   => {
               // Ensure that we drain the pipe to avoid blocking on network
               // activity.
               p.text();
@@ -765,23 +765,6 @@
                                              url)));
 }
 
-// Causes an XHR using the "Authorization" header to start and complete in the
-// target frame.
-void UseAuthorizationHeaderXhr(const ToRenderFrameHost& execution_target,
-                               const GURL& url) {
-  ASSERT_EQ(42, EvalJs(execution_target, JsReplace(
-                                             R"(
-      const xhr = new XMLHttpRequest();
-      xhr.open('GET', $1);
-      xhr.setRequestHeader('Authorization', 'foo');
-      xhr.send();
-      new Promise(resolve => {
-        xhr.onload = () => {resolve(42)};
-      });
-      )",
-                                             url)));
-}
-
 // Creates an iframe in the target frame with this url. It waits until the frame
 // has loaded.
 void CreateIframe(const ToRenderFrameHost& execution_target, GURL url) {
@@ -798,49 +781,9 @@
 }
 }  // namespace
 
-enum class RequestType {
-  kFetch,
-  kXhr,
-};
-
-class BackForwardCacheAuthorizationHeaderBrowserTest
-    : public BackForwardCacheBrowserTest,
-      public ::testing::WithParamInterface<RequestType> {
- public:
-  // Provides meaningful param names instead of /0 and /1.
-  static std::string DescribeParams(
-      const ::testing::TestParamInfo<ParamType>& info) {
-    switch (info.param) {
-      case RequestType::kFetch:
-        return "Fetch";
-      case RequestType::kXhr:
-        return "XHR";
-    }
-  }
-
- protected:
-  // Make a request using the appropriate method.
-  void UseAuthorizationHeader(const ToRenderFrameHost& execution_target,
-                              GURL url) {
-    switch (GetParam()) {
-      case RequestType::kFetch:
-        UseAuthorizationHeaderFetch(execution_target, url);
-        break;
-      case RequestType::kXhr:
-        UseAuthorizationHeaderXhr(execution_target, url);
-        break;
-    }
-  }
-};
-INSTANTIATE_TEST_SUITE_P(
-    All,
-    BackForwardCacheAuthorizationHeaderBrowserTest,
-    ::testing::Values(RequestType::kFetch, RequestType::kXhr),
-    &BackForwardCacheAuthorizationHeaderBrowserTest::DescribeParams);
-
-// Test that a page without CCNS that makes a request with the "Authorization"
+// Test that a page without CCNS that makes a fetch with the "Authorization"
 // header does not log the header.
-IN_PROC_BROWSER_TEST_P(BackForwardCacheAuthorizationHeaderBrowserTest,
+IN_PROC_BROWSER_TEST_F(BackForwardCacheBrowserTest,
                        AuthorizationHeaderNotLogged) {
   ASSERT_TRUE(embedded_test_server()->Start());
 
@@ -851,7 +794,7 @@
   ASSERT_TRUE(NavigateToURL(shell(), url_a));
   RenderFrameHostImplWrapper rfh_a(current_frame_host());
 
-  // Make a request with the "Authorization" header in the main frame.
+  // Do a fetch with the "Authorization" header in the main frame.
   UseAuthorizationHeader(shell(), url_a);
 
   // Navigate away.
@@ -865,9 +808,9 @@
   ExpectRestored(FROM_HERE);
 }
 
-// Test that a page with CCNS that makes a request with the "Authorization"
-// header logs the header.
-IN_PROC_BROWSER_TEST_P(BackForwardCacheAuthorizationHeaderBrowserTest,
+// Test that a page with CCNS that makes a fetch with the "Authorization" header
+// logs the header.
+IN_PROC_BROWSER_TEST_F(BackForwardCacheBrowserTest,
                        AuthorizationHeaderLoggedMainFrame) {
   ASSERT_TRUE(embedded_test_server()->Start());
 
@@ -880,7 +823,7 @@
   ASSERT_TRUE(NavigateToURL(shell(), url_a_no_store));
   RenderFrameHostImplWrapper rfh_a(current_frame_host());
 
-  // Make a request with the "Authorization" header in the main frame.
+  // Do a fetch with the "Authorization" header in the main frame.
   UseAuthorizationHeader(shell(), url_a_2);
 
   // Navigate away.
@@ -897,10 +840,9 @@
                     {}, {}, {}, FROM_HERE);
 }
 
-// Test that a page with CCNS that makes a request with the "Authorization"
-// header in a same-as-root-origin subframe of a cross-origin subframe logs the
-// header.
-IN_PROC_BROWSER_TEST_P(BackForwardCacheAuthorizationHeaderBrowserTest,
+// Test that a page with CCNS that makes a fetch with the "Authorization" header
+// in a same-as-root-origin subframe of a cross-origin subframe logs the header.
+IN_PROC_BROWSER_TEST_F(BackForwardCacheBrowserTest,
                        AuthorizationHeaderSameOriginSubFrameLogged) {
   ASSERT_TRUE(embedded_test_server()->Start());
 
@@ -914,8 +856,7 @@
   RenderFrameHostImplWrapper rfh_a(current_frame_host());
 
   // Create a cross-origin iframe with same-as-root-origin iframe inside that
-  // and make a request with the "Authorization" header in that grand-child
-  // iframe.
+  // and do a fetch with the "Authorization" header in that grand-child iframe.
   CreateIframe(rfh_a.get(), url_b);
   CreateIframe(DescendantRenderFrameHostImplAt(rfh_a.get(), {0}), url_a_2);
 
@@ -936,10 +877,10 @@
                     {}, {}, {}, FROM_HERE);
 }
 
-// Test that a page with CCNS that makes a request with the "Authorization"
-// header in a same-origin subframe logs the header in the correct place in the
-// tree of reasons.
-IN_PROC_BROWSER_TEST_P(BackForwardCacheAuthorizationHeaderBrowserTest,
+// Test that a page with CCNS that makes a fetch with the "Authorization" header
+// in a same-origin subframe logs the header in the correct place in the tree of
+// reasons.
+IN_PROC_BROWSER_TEST_F(BackForwardCacheBrowserTest,
                        AuthorizationHeaderSubFrameTree) {
   ASSERT_TRUE(embedded_test_server()->Start());
 
@@ -952,7 +893,7 @@
   ASSERT_TRUE(NavigateToURL(shell(), url_a_no_store));
   RenderFrameHostImplWrapper rfh_a(current_frame_host());
 
-  // Create a same-origin iframe make a request with the "Authorization" header.
+  // Create a same-origin iframe do a fetch with the "Authorization" header.
   CreateIframe(rfh_a.get(), url_a_2);
 
   UseAuthorizationHeader(DescendantRenderFrameHostImplAt(rfh_a.get(), {0}),
@@ -989,9 +930,9 @@
                       {subframe_result})));
 }
 
-// Test that a page with CCNS that makes a request with the "Authorization"
-// header in a cross-origin subframe does not log the header.
-IN_PROC_BROWSER_TEST_P(BackForwardCacheAuthorizationHeaderBrowserTest,
+// Test that a page with CCNS that makes a fetch with the "Authorization" header
+// in a cross-origin subframe does not log the header.
+IN_PROC_BROWSER_TEST_F(BackForwardCacheBrowserTest,
                        AuthorizationHeaderCrossOriginSubFrameNotLogged) {
   ASSERT_TRUE(embedded_test_server()->Start());
 
@@ -1004,8 +945,8 @@
   ASSERT_TRUE(NavigateToURL(shell(), url_a_no_store));
   RenderFrameHostImplWrapper rfh_a(current_frame_host());
 
-  // Create an same-origin iframe and make a request with the "Authorization"
-  // header in that iframe.
+  // Create an same-origin iframe and do a fetch with the "Authorization" header
+  // in that iframe.
   CreateIframe(rfh_a.get(), url_b);
 
   UseAuthorizationHeader(DescendantRenderFrameHostImplAt(rfh_a.get(), {0}),
@@ -1069,9 +1010,7 @@
 }
 
 // Test that a page with CCNS that makes a fetch with the "Authorization" header
-// is blocked even when CCNS pages are allowed to be restored. This only tests
-// fetch, the blocking mechanism is the same for all kinds of requests, so if it
-// works for one it will work for all.
+// is blocked even when CCNS pages are allowed to be restored.
 IN_PROC_BROWSER_TEST_F(
     BackForwardCacheBrowserTestRestoreCacheControlNoStoreUnlessCookieChange,
     AuthorizationHeaderBlocks) {
@@ -1086,8 +1025,8 @@
   ASSERT_TRUE(NavigateToURL(shell(), url_a_no_store));
   RenderFrameHostImplWrapper rfh_a(current_frame_host());
 
-  // Make a request with the "Authorization" header in the main frame.
-  UseAuthorizationHeaderFetch(shell(), url_a_2);
+  // Do a fetch with the "Authorization" header in the main frame.
+  UseAuthorizationHeader(shell(), url_a_2);
 
   // Navigate away.
   ASSERT_TRUE(NavigateToURL(shell(), url_b));
diff --git a/content/browser/child_process_security_policy_unittest.cc b/content/browser/child_process_security_policy_unittest.cc
index 25d1eaf..12805d2 100644
--- a/content/browser/child_process_security_policy_unittest.cc
+++ b/content/browser/child_process_security_policy_unittest.cc
@@ -3221,6 +3221,9 @@
                        ProcessLock::FromSiteInfo(bar_instance->GetSiteInfo()));
       },
       "Cannot lock an already used process to .*bar\\.com");
+
+  // We need to remove it otherwise other tests may fail.
+  p->Remove(kRendererID);
 }
 
 }  // namespace content
diff --git a/content/browser/file_system/file_system_manager_impl.cc b/content/browser/file_system/file_system_manager_impl.cc
index cf6639ba..21cd61d 100644
--- a/content/browser/file_system/file_system_manager_impl.cc
+++ b/content/browser/file_system/file_system_manager_impl.cc
@@ -993,6 +993,28 @@
     RegisterBlobCallback callback) {
   storage::FileSystemURL crack_url =
       context_->CrackURL(url, receivers_.current_context());
+
+  content::GetUIThreadTaskRunner({})->PostTaskAndReplyWithResult(
+      FROM_HERE,
+      // security_policy_ is a singleton so refcounting is unnecessary
+      base::BindOnce(&ChildProcessSecurityPolicyImpl::CanReadFileSystemFile,
+                     base::Unretained(security_policy_), process_id_,
+                     crack_url),
+      base::BindOnce(&FileSystemManagerImpl::ContinueRegisterBlob,
+                     weak_factory_.GetWeakPtr(), content_type, url, length,
+                     expected_modification_time, std::move(callback),
+                     crack_url));
+}
+
+void FileSystemManagerImpl::ContinueRegisterBlob(
+    const std::string& content_type,
+    const GURL& url,
+    uint64_t length,
+    absl::optional<base::Time> expected_modification_time,
+    RegisterBlobCallback callback,
+    storage::FileSystemURL crack_url,
+    bool security_check_success) {
+  DCHECK_CURRENTLY_ON(BrowserThread::IO);
   std::string uuid = base::GenerateGUID();
   mojo::PendingRemote<blink::mojom::Blob> blob_remote;
   mojo::PendingReceiver<blink::mojom::Blob> blob_receiver =
@@ -1000,7 +1022,7 @@
 
   if (crack_url.is_valid() &&
       context_->GetFileSystemBackend(crack_url.type()) &&
-      security_policy_->CanReadFileSystemFile(process_id_, crack_url)) {
+      security_check_success) {
     blob_storage_context_->CreateFileSystemBlob(
         context_, std::move(blob_receiver), crack_url, uuid, content_type,
         length, expected_modification_time.value_or(base::Time()));
diff --git a/content/browser/file_system/file_system_manager_impl.h b/content/browser/file_system/file_system_manager_impl.h
index 4e84e97..d89a708 100644
--- a/content/browser/file_system/file_system_manager_impl.h
+++ b/content/browser/file_system/file_system_manager_impl.h
@@ -209,6 +209,15 @@
   void ContinueCreateSnapshotFile(const storage::FileSystemURL& url,
                                   CreateSnapshotFileCallback callback,
                                   bool security_check_success);
+  void ContinueRegisterBlob(
+      const std::string& content_type,
+      const GURL& url,
+      uint64_t length,
+      absl::optional<base::Time> expected_modification_time,
+      RegisterBlobCallback callback,
+      storage::FileSystemURL crack_url,
+      bool security_check_success);
+
   void Cancel(
       OperationID op_id,
       blink::mojom::FileSystemCancellableOperation::CancelCallback callback);
diff --git a/content/browser/gpu/gpu_process_host.cc b/content/browser/gpu/gpu_process_host.cc
index 63f23b3..de753e5 100644
--- a/content/browser/gpu/gpu_process_host.cc
+++ b/content/browser/gpu/gpu_process_host.cc
@@ -111,10 +111,6 @@
 #include "content/common/zygote/zygote_handle_impl_linux.h"
 #endif
 
-#if BUILDFLAG(IS_MAC) || BUILDFLAG(IS_ANDROID)
-#include "gpu/ipc/common/gpu_surface_tracker.h"
-#endif
-
 #if BUILDFLAG(IS_MAC)
 #include "content/browser/gpu/ca_transaction_gpu_coordinator.h"
 #endif
diff --git a/content/browser/loader/web_transport_browsertest.cc b/content/browser/loader/web_transport_browsertest.cc
index 72730c9a..a198023 100644
--- a/content/browser/loader/web_transport_browsertest.cc
+++ b/content/browser/loader/web_transport_browsertest.cc
@@ -108,5 +108,44 @@
   ASSERT_TRUE(WaitForTitle(u"PASS", {u"FAIL"}));
 }
 
+// A test that aims to reproduce https://crbug.com/1369030 -- note that since
+// the bug in question is a race condition, this test will probably be flaky if
+// this is actually broken.
+IN_PROC_BROWSER_TEST_F(WebTransportBrowserTest, EchoLargeBidirectionalStreams) {
+  ASSERT_TRUE(embedded_test_server()->Start());
+  ASSERT_TRUE(
+      NavigateToURL(shell(), embedded_test_server()->GetURL("/title2.html")));
+
+  ASSERT_TRUE(WaitForTitle(u"Title Of Awesomeness"));
+
+  ASSERT_TRUE(
+      ExecJs(shell(), base::StringPrintf(R"JS(
+    async function run() {
+      const transport = new WebTransport('https://localhost:%d/echo');
+      await transport.ready;
+
+      const numBytes = 1024 * 1024;
+      const numStreams = 5;
+      for (let i = 0; i < numStreams; i++) {
+        const stream = await transport.createBidirectionalStream();
+        const writer = stream.writable.getWriter();
+        await writer.write(new Uint8Array(numBytes));
+        await writer.close();
+        let response = await (new Response(stream.readable).arrayBuffer());
+        if (response.byteLength != numBytes) {
+          throw Error('Size mismatch, received size: '
+                         + response.byteLength.toString());
+        }
+      }
+    }
+
+    run().then(() => { document.title = 'PASS'; },
+               (e) => { console.log(e); document.title = 'FAIL'; });
+)JS",
+                                         server_.server_address().port())));
+
+  ASSERT_TRUE(WaitForTitle(u"PASS", {u"FAIL"}));
+}
+
 }  // namespace
 }  // namespace content
diff --git a/content/browser/media/cdm_registry_impl.cc b/content/browser/media/cdm_registry_impl.cc
index 3dc2047..d577857d 100644
--- a/content/browser/media/cdm_registry_impl.cc
+++ b/content/browser/media/cdm_registry_impl.cc
@@ -188,29 +188,6 @@
                               {media::CdmSessionType::kTemporary});
 }
 
-// Software secure capability can be obtained synchronously in all supported
-// cases. If needed, this can be easily converted to an asynchronous call.
-absl::optional<media::CdmCapability> GetSoftwareSecureCapability(
-    const CdmRegistryImpl& cdm_registry_impl,
-    const std::string& key_system) {
-  auto cdm_info = cdm_registry_impl.GetCdmInfo(
-      key_system, CdmInfo::Robustness::kSoftwareSecure);
-  if (!cdm_info) {
-    ReportSoftwareSecureCdmAvailableUMA(key_system, false);
-    return absl::nullopt;
-  }
-
-  ReportSoftwareSecureCdmAvailableUMA(key_system, true);
-
-  if (!cdm_info->capability) {
-    DVLOG(1) << "Lazy initialization of SoftwareSecure CdmCapability not "
-                "supported!";
-    return absl::nullopt;
-  }
-
-  return cdm_info->capability;
-}
-
 #if BUILDFLAG(IS_WIN)
 bool IsMediaFoundationHardwareSecurityDisabledByGpuFeature() {
   auto* gpu_data_manager = GpuDataManagerImpl::GetInstance();
@@ -226,71 +203,6 @@
 }
 #endif  // BUILDFLAG(IS_WIN)
 
-// Trying to get hardware secure capability synchronously. If lazy
-// initialization is needed, set `lazy_initialize` to true.
-std::tuple<absl::optional<media::CdmCapability>, CdmInfo::Status>
-GetHardwareSecureCapability(const CdmRegistryImpl& cdm_registry_impl,
-                            const std::string& key_system) {
-  using Status = CdmInfo::Status;
-
-#if BUILDFLAG(IS_CHROMEOS_LACROS)
-  if (!base::CommandLine::ForCurrentProcess()->HasSwitch(
-          switches::kLacrosUseChromeosProtectedMedia)) {
-    return {absl::nullopt, Status::kHardwareSecureDecryptionDisabled};
-  }
-#elif !BUILDFLAG(USE_CHROMEOS_PROTECTED_MEDIA)
-  if (!media::IsHardwareSecureDecryptionEnabled()) {
-    DVLOG(1) << "Hardware secure decryption disabled";
-    return {absl::nullopt, Status::kHardwareSecureDecryptionDisabled};
-  }
-#endif  // !BUILDFLAG(USE_CHROMEOS_PROTECTED_MEDIA)
-
-  // Secure codecs override takes precedence over other checks.
-  auto overridden_capability =
-      GetHardwareSecureCapabilityOverriddenFromCommandLine();
-  if (overridden_capability) {
-    DVLOG(1) << "Hardware secure codecs overridden from command line";
-    return {overridden_capability, Status::kCommandLineOverridden};
-  }
-
-  // Hardware secure video codecs need hardware video decoder support.
-  // TODO(xhwang): Make sure this check is as close as possible to the check
-  // in the render process. For example, also check check GPU features like
-  // GPU_FEATURE_TYPE_ACCELERATED_VIDEO_DECODE.
-  auto* command_line = base::CommandLine::ForCurrentProcess();
-  if (command_line &&
-      command_line->HasSwitch(switches::kDisableAcceleratedVideoDecode)) {
-    DVLOG(1) << "Hardware security not supported because accelerated video "
-                "decode disabled";
-    return {absl::nullopt, Status::kAcceleratedVideoDecodeDisabled};
-  }
-
-#if BUILDFLAG(IS_WIN)
-  if (IsMediaFoundationHardwareSecurityDisabledByGpuFeature()) {
-    DVLOG(1) << "Hardware security not supported: GPU workarounds";
-    return {absl::nullopt, Status::kGpuFeatureDisabled};
-  }
-
-  if (IsGpuHardwareCompositionDisabled()) {
-    DVLOG(1) << "Hardware security not supported: GPU composition disabled";
-    return {absl::nullopt, Status::kGpuCompositionDisabled};
-  }
-#endif  // BUILDFLAG(IS_WIN)
-
-  auto cdm_info = cdm_registry_impl.GetCdmInfo(
-      key_system, CdmInfo::Robustness::kHardwareSecure);
-  if (!cdm_info) {
-    DVLOG(1) << "No Hardware secure decryption CDM registered";
-    return {absl::nullopt, Status::kEnabled};
-  }
-
-  DCHECK(!(cdm_info->status == CdmInfo::Status::kUninitialized &&
-           cdm_info->capability))
-      << "Capability should not have value if uninitialized.";
-
-  return {cdm_info->capability, cdm_info->status};
-}
-
 }  // namespace
 
 // static
@@ -326,7 +238,7 @@
 
 void CdmRegistryImpl::RegisterCdm(const CdmInfo& info) {
   DVLOG(1) << __func__ << ": key_system=" << info.key_system
-           << ", robustness=" << static_cast<int>(info.robustness)
+           << ", robustness=" << info.robustness
            << ", status=" << static_cast<int>(info.status);
   DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
 
@@ -399,7 +311,8 @@
 std::unique_ptr<CdmInfo> CdmRegistryImpl::GetCdmInfo(
     const std::string& key_system,
     CdmInfo::Robustness robustness) const {
-  DVLOG(2) << __func__;
+  DVLOG(2) << __func__ << ": key_system=" << key_system
+           << ", robustness=" << robustness;
   DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
 
   for (const auto& cdm : cdms_) {
@@ -417,7 +330,7 @@
 
   key_system_capabilities_update_callbacks_.AddUnsafe(cb);
 
-  if (!pending_lazy_initialize_key_systems_.empty()) {
+  if (!pending_lazy_initializations_.empty()) {
     // Lazy initializing some key systems. All callbacks will be notified when
     // that's finished.
     return;
@@ -431,6 +344,84 @@
   FinalizeKeySystemCapabilities();
 }
 
+std::pair<absl::optional<media::CdmCapability>, CdmInfo::Status>
+CdmRegistryImpl::GetCapability(const std::string& key_system,
+                               CdmInfo::Robustness robustness) {
+  DVLOG(2) << __func__ << ": key_system=" << key_system
+           << ", robustness=" << robustness;
+  using Status = CdmInfo::Status;
+
+  if (robustness == CdmInfo::Robustness::kHardwareSecure) {
+#if BUILDFLAG(IS_CHROMEOS_LACROS)
+    if (!base::CommandLine::ForCurrentProcess()->HasSwitch(
+            switches::kLacrosUseChromeosProtectedMedia)) {
+      return {absl::nullopt, Status::kHardwareSecureDecryptionDisabled};
+    }
+#elif !BUILDFLAG(USE_CHROMEOS_PROTECTED_MEDIA)
+    if (!media::IsHardwareSecureDecryptionEnabled()) {
+      DVLOG(1) << "Hardware secure decryption disabled";
+      return {absl::nullopt, Status::kHardwareSecureDecryptionDisabled};
+    }
+#endif  // !BUILDFLAG(USE_CHROMEOS_PROTECTED_MEDIA)
+
+    // Secure codecs override takes precedence over other checks.
+    auto overridden_capability =
+        GetHardwareSecureCapabilityOverriddenFromCommandLine();
+    if (overridden_capability) {
+      DVLOG(1) << "Hardware secure codecs overridden from command line";
+      return {overridden_capability, Status::kCommandLineOverridden};
+    }
+
+    // Hardware secure video codecs need hardware video decoder support.
+    // TODO(xhwang): Make sure this check is as close as possible to the check
+    // in the render process. For example, also check check GPU features like
+    // GPU_FEATURE_TYPE_ACCELERATED_VIDEO_DECODE.
+    auto* command_line = base::CommandLine::ForCurrentProcess();
+    if (command_line &&
+        command_line->HasSwitch(switches::kDisableAcceleratedVideoDecode)) {
+      DVLOG(1) << "Hardware security not supported because accelerated video "
+                  "decode disabled";
+      return {absl::nullopt, Status::kAcceleratedVideoDecodeDisabled};
+    }
+
+#if BUILDFLAG(IS_WIN)
+    if (IsMediaFoundationHardwareSecurityDisabledByGpuFeature()) {
+      DVLOG(1) << "Hardware security not supported: GPU workarounds";
+      return {absl::nullopt, Status::kGpuFeatureDisabled};
+    }
+
+    if (IsGpuHardwareCompositionDisabled()) {
+      DVLOG(1) << "Hardware security not supported: GPU composition disabled";
+      return {absl::nullopt, Status::kGpuCompositionDisabled};
+    }
+#endif  // BUILDFLAG(IS_WIN)
+  }
+
+  auto cdm_info = GetCdmInfo(key_system, robustness);
+  if (!cdm_info) {
+    DVLOG(1) << "No " << robustness << " decryption CDM registered for "
+             << key_system;
+    return {absl::nullopt, Status::kEnabled};
+  }
+
+  DCHECK(!(cdm_info->status == Status::kUninitialized && cdm_info->capability))
+      << "Capability for " << robustness << " " << key_system
+      << " should not have value if uninitialized.";
+
+  return {cdm_info->capability, cdm_info->status};
+}
+
+std::pair<absl::optional<media::CdmCapability>, CdmInfo::Status>
+CdmRegistryImpl::GetFinalCapability(const std::string& key_system,
+                                    CdmInfo::Robustness robustness) {
+  DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
+
+  const auto [capability, status] = GetCapability(key_system, robustness);
+  DCHECK(status != CdmInfo::Status::kUninitialized);
+
+  return {IsEnabled(status) ? capability : absl::nullopt, status};
+}
+
 void CdmRegistryImpl::FinalizeKeySystemCapabilities() {
   DVLOG(2) << __func__;
   DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
@@ -438,7 +429,7 @@
 
   // Abort existing pending LazyInitializeHardwareSecureCapability() operations
   // to avoid updating the observer twice.
-  pending_lazy_initialize_key_systems_.clear();
+  pending_lazy_initializations_.clear();
   weak_ptr_factory_.InvalidateWeakPtrs();
 
   // Get the set of supported key systems in case two CDMs are registered with
@@ -446,87 +437,100 @@
   // while iterating through it.
   std::set<std::string> supported_key_systems = GetSupportedKeySystems();
 
-  // Finalize hardware secure capabilities for all key systems. (Assumes
-  // software secure capabilities are always already finalized.)
+  // Finalize software secure capabilities for all key systems.
   for (const auto& key_system : supported_key_systems) {
-    auto cdm_info =
-        GetCdmInfo(key_system, CdmInfo::Robustness::kHardwareSecure);
-    if (!cdm_info) {
-      DVLOG(1) << "No Hardware secure CDM registered";
-      continue;
+    for (const auto robustness : {CdmInfo::Robustness::kSoftwareSecure,
+                                  CdmInfo::Robustness::kHardwareSecure}) {
+      AttemptToFinalizeKeySystemCapability(key_system, robustness);
     }
-
-    if (cdm_info->status != CdmInfo::Status::kUninitialized) {
-      DVLOG(1) << "Hardware secure capability already finalized";
-      continue;
-    }
-
-    absl::optional<media::CdmCapability> hw_secure_capability;
-    CdmInfo::Status status;
-    std::tie(hw_secure_capability, status) =
-        GetHardwareSecureCapability(*this, key_system);
-    if (status != CdmInfo::Status::kUninitialized) {
-      FinalizeHardwareSecureCapability(key_system, hw_secure_capability,
-                                       status);
-      continue;
-    }
-
-    // Needs lazy initialize. Use BindToCurrentLoop() to force a post.
-    pending_lazy_initialize_key_systems_.insert(key_system);
-    LazyInitializeHardwareSecureCapability(
-        key_system, media::BindToCurrentLoop(base::BindOnce(
-                        &CdmRegistryImpl::OnHardwareSecureCapabilityInitialized,
-                        weak_ptr_factory_.GetWeakPtr(), key_system)));
   }
 
-  // If not empty, we'll handle it in OnHardwareSecureCapabilityInitialized().
-  if (pending_lazy_initialize_key_systems_.empty())
+  // If not empty, we'll handle it in OnCapabilityInitialized().
+  if (pending_lazy_initializations_.empty())
     UpdateAndNotifyKeySystemCapabilities();
 }
 
+void CdmRegistryImpl::AttemptToFinalizeKeySystemCapability(
+    const std::string& key_system,
+    CdmInfo::Robustness robustness) {
+  auto cdm_info = GetCdmInfo(key_system, robustness);
+  if (!cdm_info) {
+    DVLOG(1) << "No " << robustness << " CDM registered for " << key_system;
+    return;
+  }
+
+  if (cdm_info->status != CdmInfo::Status::kUninitialized) {
+    DVLOG(1) << robustness << " capability already finalized for "
+             << key_system;
+    return;
+  }
+
+  const auto [capability, status] = GetCapability(key_system, robustness);
+  if (status != CdmInfo::Status::kUninitialized) {
+    FinalizeCapability(key_system, robustness, capability, status);
+    return;
+  }
+
+  // Needs lazy initialize. Use BindToCurrentLoop() to force a post.
+  pending_lazy_initializations_.insert({key_system, robustness});
+  LazyInitializeCapability(
+      key_system, robustness,
+      media::BindToCurrentLoop(base::BindOnce(
+          &CdmRegistryImpl::OnCapabilityInitialized,
+          weak_ptr_factory_.GetWeakPtr(), key_system, robustness)));
+}
+
 // TODO(xhwang): Find a way to register this as callbacks so we don't have to
 // hardcode platform-specific logic here.
-// TODO(jrummell): Support Android query.
-void CdmRegistryImpl::LazyInitializeHardwareSecureCapability(
+// TODO(crbug.com/853336): Support Android query.
+void CdmRegistryImpl::LazyInitializeCapability(
     const std::string& key_system,
+    CdmInfo::Robustness robustness,
     CdmCapabilityCB cdm_capability_cb) {
   DVLOG(2) << __func__;
   DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
 
-  if (hw_secure_capability_cb_for_testing_) {
-    hw_secure_capability_cb_for_testing_.Run(key_system,
-                                             std::move(cdm_capability_cb));
+  if (capability_cb_for_testing_) {
+    capability_cb_for_testing_.Run(key_system, robustness,
+                                   std::move(cdm_capability_cb));
     return;
   }
 
 #if BUILDFLAG(IS_WIN)
-  auto cdm_info = GetCdmInfo(key_system, CdmInfo::Robustness::kHardwareSecure);
-  DCHECK(cdm_info && !cdm_info->capability);
-  GetMediaFoundationServiceHardwareSecureCdmCapability(
-      key_system, cdm_info->path, std::move(cdm_capability_cb));
-#else
-  std::move(cdm_capability_cb).Run(absl::nullopt);
+  if (robustness == CdmInfo::Robustness::kHardwareSecure) {
+    auto cdm_info =
+        GetCdmInfo(key_system, CdmInfo::Robustness::kHardwareSecure);
+    DCHECK(cdm_info && !cdm_info->capability);
+    GetMediaFoundationServiceHardwareSecureCdmCapability(
+        key_system, cdm_info->path, std::move(cdm_capability_cb));
+    return;
+  }
 #endif  // BUILDFLAG(IS_WIN)
+
+  std::move(cdm_capability_cb).Run(absl::nullopt);
 }
 
-void CdmRegistryImpl::OnHardwareSecureCapabilityInitialized(
+void CdmRegistryImpl::OnCapabilityInitialized(
     const std::string& key_system,
+    const CdmInfo::Robustness robustness,
     absl::optional<media::CdmCapability> cdm_capability) {
   DVLOG(1) << __func__ << ": key_system=" << key_system
+           << ", robustness=" << robustness
            << ", cdm_capability=" << (cdm_capability ? "yes" : "no");
   DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_);
-  DCHECK(pending_lazy_initialize_key_systems_.count(key_system));
+  DCHECK(pending_lazy_initializations_.count({key_system, robustness}));
 
-  FinalizeHardwareSecureCapability(key_system, std::move(cdm_capability),
-                                   CdmInfo::Status::kEnabled);
+  FinalizeCapability(key_system, robustness, std::move(cdm_capability),
+                     CdmInfo::Status::kEnabled);
 
-  pending_lazy_initialize_key_systems_.erase(key_system);
-  if (pending_lazy_initialize_key_systems_.empty())
+  pending_lazy_initializations_.erase({key_system, robustness});
+  if (pending_lazy_initializations_.empty())
     UpdateAndNotifyKeySystemCapabilities();
 }
 
-void CdmRegistryImpl::FinalizeHardwareSecureCapability(
+void CdmRegistryImpl::FinalizeCapability(
     const std::string& key_system,
+    const CdmInfo::Robustness robustness,
     absl::optional<media::CdmCapability> cdm_capability,
     CdmInfo::Status status) {
   DVLOG(2) << __func__;
@@ -535,19 +539,20 @@
 
   auto itr = cdms_.begin();
   for (; itr != cdms_.end(); itr++) {
-    if (itr->robustness == CdmInfo::Robustness::kHardwareSecure &&
-        MatchKeySystem(*itr, key_system)) {
+    if (itr->robustness == robustness && MatchKeySystem(*itr, key_system)) {
       break;
     }
   }
 
   if (itr == cdms_.end()) {
-    DLOG(ERROR) << __func__ << ": Cannot find CdmInfo to finalize";
+    DLOG(ERROR) << __func__ << ": Cannot find CdmInfo to finalize for "
+                << key_system << " with robustness " << robustness;
     return;
   }
 
   if (itr->status != CdmInfo::Status::kUninitialized) {
-    DLOG(ERROR) << __func__ << ": CdmCapability already finalized";
+    DLOG(ERROR) << __func__ << ": CdmCapability already finalized for "
+                << key_system << " with robustness " << robustness;
     return;
   }
 
@@ -588,22 +593,21 @@
 
   std::set<std::string> supported_key_systems = GetSupportedKeySystems();
   for (const auto& key_system : supported_key_systems) {
+    CdmInfo::Status status;
     media::mojom::KeySystemCapability capability;
 
-    // Software secure capability
-    capability.sw_secure_capability =
-        GetSoftwareSecureCapability(*this, key_system);
+    // Software secure capability.
+    std::tie(capability.sw_secure_capability, status) =
+        GetFinalCapability(key_system, CdmInfo::Robustness::kSoftwareSecure);
+    ReportSoftwareSecureCdmAvailableUMA(
+        key_system, capability.sw_secure_capability != absl::nullopt);
 
-    // Hardware secure capability
-    absl::optional<media::CdmCapability> hw_secure_capability;
-    CdmInfo::Status status;
-    std::tie(hw_secure_capability, status) =
-        GetHardwareSecureCapability(*this, key_system);
-    DCHECK(status != CdmInfo::Status::kUninitialized);
+    // Hardware secure capability.
+    std::tie(capability.hw_secure_capability, status) =
+        GetFinalCapability(key_system, CdmInfo::Robustness::kHardwareSecure);
     ReportHardwareSecureCapabilityStatusUMA(
-        key_system, status, base::OptionalToPtr(hw_secure_capability));
-    capability.hw_secure_capability =
-        IsEnabled(status) ? hw_secure_capability : absl::nullopt;
+        key_system, status,
+        base::OptionalToPtr(capability.hw_secure_capability));
 
     if (capability.sw_secure_capability || capability.hw_secure_capability)
       key_system_capabilities[key_system] = std::move(capability);
@@ -612,9 +616,8 @@
   return key_system_capabilities;
 }
 
-void CdmRegistryImpl::SetHardwareSecureCapabilityCBForTesting(
-    HardwareSecureCapabilityCB cb) {
-  hw_secure_capability_cb_for_testing_ = std::move(cb);
+void CdmRegistryImpl::SetCapabilityCBForTesting(CapabilityCB cb) {
+  capability_cb_for_testing_ = std::move(cb);
 }
 
 }  // namespace content
diff --git a/content/browser/media/cdm_registry_impl.h b/content/browser/media/cdm_registry_impl.h
index e4d6d4c..6616e7d9 100644
--- a/content/browser/media/cdm_registry_impl.h
+++ b/content/browser/media/cdm_registry_impl.h
@@ -72,18 +72,45 @@
   CdmRegistryImpl();
   ~CdmRegistryImpl() override;
 
+  // Get the capability for `key_system` with robustness `robustness`
+  // synchronously. If lazy initialization is needed, return
+  // Status::kUninitialized.
+  std::pair<absl::optional<media::CdmCapability>, CdmInfo::Status>
+  GetCapability(const std::string& key_system, CdmInfo::Robustness robustness);
+
+  // Get the capability for `key_system` with robustness `robustness`
+  // synchronously. All initialization should have been completed.
+  std::pair<absl::optional<media::CdmCapability>, CdmInfo::Status>
+  GetFinalCapability(const std::string& key_system,
+                     CdmInfo::Robustness robustness);
+
   // Finalizes KeySystemCapabilities. May lazy initialize CDM capabilities
   // asynchronously if needed.
   void FinalizeKeySystemCapabilities();
 
+  // Attempt to finalize KeySystemCapability for `key_system` with robustness
+  // `robustness`. May lazy initialize it asynchronously if needed.
+  void AttemptToFinalizeKeySystemCapability(const std::string& key_system,
+                                            CdmInfo::Robustness robustness);
+
+  // Callback for when a key system is initialized if lazy initialization
+  // required.
   using CdmCapabilityCB =
       base::OnceCallback<void(absl::optional<media::CdmCapability>)>;
-  void LazyInitializeHardwareSecureCapability(
-      const std::string& key_system,
-      CdmCapabilityCB cdm_capability_cb);
 
-  void OnHardwareSecureCapabilityInitialized(
+  // Lazily initialize `key_system` with robustness `robustness`, calling
+  // `cdm_capability_cb`. Callback may be called synchronously
+  // or asynchronously.
+  void LazyInitializeCapability(const std::string& key_system,
+                                CdmInfo::Robustness robustness,
+                                CdmCapabilityCB cdm_capability_cb);
+
+  // Called when initialization of `key_system` with robustness `robustness`
+  // is complete. `cdm_capability` will be absl::nullopt if the key system
+  // with specified robustness isn't supported.
+  void OnCapabilityInitialized(
       const std::string& key_system,
+      const CdmInfo::Robustness robustness,
       absl::optional<media::CdmCapability> cdm_capability);
 
   // Finalizes the CdmInfo corresponding to `key_system` and `robustness` if its
@@ -91,21 +118,28 @@
   // exist, or if the CdmInfo's CdmCapability is not null. The CdmInfo will be
   // removed if `cdm_capability` is null, since the CDM does not support any
   // capability.
-  void FinalizeHardwareSecureCapability(
-      const std::string& key_system,
-      absl::optional<media::CdmCapability> cdm_capability,
-      CdmInfo::Status status);
+  void FinalizeCapability(const std::string& key_system,
+                          const CdmInfo::Robustness robustness,
+                          absl::optional<media::CdmCapability> cdm_capability,
+                          CdmInfo::Status status);
 
+  // When capabilities for all registered key systems have been determined,
+  // notify all observers with the updated values. No notification is done
+  // if the capabilities have not changed.
   void UpdateAndNotifyKeySystemCapabilities();
 
+  // Returns the set of all registered key systems.
   std::set<std::string> GetSupportedKeySystems() const;
 
+  // Returns the capabailities for all registered key systems.
   KeySystemCapabilities GetKeySystemCapabilities();
 
-  // Sets a callback to query for hardware secure capability for testing.
-  using HardwareSecureCapabilityCB =
-      base::RepeatingCallback<void(const std::string&, CdmCapabilityCB)>;
-  void SetHardwareSecureCapabilityCBForTesting(HardwareSecureCapabilityCB cb);
+  // Sets callbacks to query for secure capability for testing.
+  using CapabilityCB =
+      base::RepeatingCallback<void(const std::string&,
+                                   const CdmInfo::Robustness robustness,
+                                   CdmCapabilityCB)>;
+  void SetCapabilityCBForTesting(CapabilityCB cb);
 
   std::vector<CdmInfo> cdms_ GUARDED_BY_CONTEXT(sequence_checker_);
 
@@ -116,11 +150,12 @@
   // Cached current KeySystemCapabilities value.
   absl::optional<KeySystemCapabilities> key_system_capabilities_;
 
-  // Key systems pending CdmCapability lazy initialization.
-  std::set<std::string> pending_lazy_initialize_key_systems_;
+  // Key system and robustness pairs pending CdmCapability lazy initialization.
+  std::set<std::pair<std::string, CdmInfo::Robustness>>
+      pending_lazy_initializations_;
 
-  // A callback for testing to avoid hardware dependency.
-  HardwareSecureCapabilityCB hw_secure_capability_cb_for_testing_;
+  // Callback for testing to avoid device dependency.
+  CapabilityCB capability_cb_for_testing_;
 
   SEQUENCE_CHECKER(sequence_checker_);
 
diff --git a/content/browser/media/cdm_registry_impl_unittest.cc b/content/browser/media/cdm_registry_impl_unittest.cc
index 59a4ad2..b375511b 100644
--- a/content/browser/media/cdm_registry_impl_unittest.cc
+++ b/content/browser/media/cdm_registry_impl_unittest.cc
@@ -111,14 +111,14 @@
     gpu_data_manager->UpdateGpuInfo(gpu_info, absl::nullopt);
 #endif  // BUILDFLAG(IS_WIN)
 
-    cdm_registry_.SetHardwareSecureCapabilityCBForTesting(
-        hw_secure_capability_cb_.Get());
+    cdm_registry_.SetCapabilityCBForTesting(capability_cb_.Get());
   }
 
   void OnKeySystemCapabilitiesUpdated(
       int observer_id,
       base::OnceClosure done_cb,
       KeySystemCapabilities key_system_capabilities) {
+    DVLOG(1) << __func__;
     results_[observer_id].push_back(std::move(key_system_capabilities));
     std::move(done_cb).Run();
   }
@@ -159,7 +159,17 @@
                      base::FilePath::FromUTF8Unsafe(kTestPath)));
   }
 
-  void RegisterForLazyInitialization() {
+  void RegisterForLazySoftwareSecureInitialization() {
+    // Register a CdmInfo without CdmCapability to allow lazy initialization.
+    Register(CdmInfo(kTestKeySystem, CdmInfo::Robustness::kSoftwareSecure,
+                     absl::nullopt, kTestCdmType));
+    auto cdm_info = cdm_registry_.GetCdmInfo(
+        kTestKeySystem, CdmInfo::Robustness::kSoftwareSecure);
+    ASSERT_TRUE(cdm_info);
+    ASSERT_FALSE(cdm_info->capability);
+  }
+
+  void RegisterForLazyHardwareSecureInitialization() {
     // Register a CdmInfo without CdmCapability to allow lazy initialization.
     Register(CdmInfo(kTestKeySystem, CdmInfo::Robustness::kHardwareSecure,
                      absl::nullopt, kTestCdmType));
@@ -187,6 +197,7 @@
   }
 
   void GetKeySystemCapabilities() {
+    DVLOG(1) << __func__;
     base::RunLoop run_loop;
     cdm_registry_.ObserveKeySystemCapabilities(base::BindRepeating(
         &CdmRegistryImplTest::OnKeySystemCapabilitiesUpdated,
@@ -229,10 +240,9 @@
   BrowserTaskEnvironment task_environment_;
 
   CdmRegistryImpl cdm_registry_;
-  base::MockCallback<CdmRegistryImpl::HardwareSecureCapabilityCB>
-      hw_secure_capability_cb_;
+  base::MockCallback<CdmRegistryImpl::CapabilityCB> capability_cb_;
 
-  // Map of "obserber ID" to the list of updated KeySystemCapabilities.
+  // Map of "observer ID" to the list of updated KeySystemCapabilities.
   std::map<int, std::vector<KeySystemCapabilities>> results_;
 };
 
@@ -397,12 +407,34 @@
   ASSERT_EQ(support.hw_secure_capability.value(), GetTestCdmCapability());
 }
 
-TEST_F(CdmRegistryImplTest, KeySystemCapabilities_LazyInitialize_Supported) {
-  RegisterForLazyInitialization();
+TEST_F(CdmRegistryImplTest,
+       KeySystemCapabilities_LazySoftwareSecureInitialize_Supported) {
+  RegisterForLazySoftwareSecureInitialization();
   SelectHardwareSecureDecryption(true);
 
-  EXPECT_CALL(hw_secure_capability_cb_, Run(kTestKeySystem, _))
-      .WillOnce(RunOnceCallback<1>(GetTestCdmCapability()));
+  EXPECT_CALL(capability_cb_,
+              Run(kTestKeySystem, Robustness::kSoftwareSecure, _))
+      .WillOnce(RunOnceCallback<2>(GetTestCdmCapability()));
+  GetKeySystemCapabilities();
+
+  ASSERT_TRUE(results_.count(kObserver1));
+  ASSERT_EQ(results_[kObserver1].size(), 1u);
+  auto& key_system_capabilities = results_[kObserver1][0];
+  ASSERT_EQ(key_system_capabilities.size(), 1u);
+  ASSERT_TRUE(key_system_capabilities.count(kTestKeySystem));
+  const auto& support = key_system_capabilities[kTestKeySystem];
+  ASSERT_EQ(support.sw_secure_capability.value(), GetTestCdmCapability());
+  ASSERT_FALSE(support.hw_secure_capability);
+}
+
+TEST_F(CdmRegistryImplTest,
+       KeySystemCapabilities_LazyHardwareSecureInitialize_Supported) {
+  RegisterForLazyHardwareSecureInitialization();
+  SelectHardwareSecureDecryption(true);
+
+  EXPECT_CALL(capability_cb_,
+              Run(kTestKeySystem, Robustness::kHardwareSecure, _))
+      .WillOnce(RunOnceCallback<2>(GetTestCdmCapability()));
   GetKeySystemCapabilities();
 
   ASSERT_TRUE(results_.count(kObserver1));
@@ -415,12 +447,35 @@
   ASSERT_EQ(support.hw_secure_capability.value(), GetTestCdmCapability());
 }
 
-TEST_F(CdmRegistryImplTest, KeySystemCapabilities_LazyInitialize_NotSupported) {
-  RegisterForLazyInitialization();
+TEST_F(CdmRegistryImplTest,
+       KeySystemCapabilities_LazySoftwareSecureInitialize_NotSupported) {
+  RegisterForLazySoftwareSecureInitialization();
   SelectHardwareSecureDecryption(true);
 
-  EXPECT_CALL(hw_secure_capability_cb_, Run(kTestKeySystem, _))
-      .WillOnce(RunOnceCallback<1>(absl::nullopt));
+  EXPECT_CALL(capability_cb_,
+              Run(kTestKeySystem, Robustness::kSoftwareSecure, _))
+      .WillOnce(RunOnceCallback<2>(absl::nullopt));
+  GetKeySystemCapabilities();
+
+  ASSERT_TRUE(results_.count(kObserver1));
+  ASSERT_EQ(results_[kObserver1].size(), 1u);
+  auto& key_system_capabilities = results_[kObserver1][0];
+  ASSERT_TRUE(key_system_capabilities.empty());
+
+  auto cdm_info = cdm_registry_.GetCdmInfo(
+      kTestKeySystem, CdmInfo::Robustness::kSoftwareSecure);
+  ASSERT_EQ(cdm_info->status, CdmInfo::Status::kEnabled);
+  ASSERT_FALSE(cdm_info->capability);
+}
+
+TEST_F(CdmRegistryImplTest,
+       KeySystemCapabilities_LazyHardwareSecureInitialize_NotSupported) {
+  RegisterForLazyHardwareSecureInitialization();
+  SelectHardwareSecureDecryption(true);
+
+  EXPECT_CALL(capability_cb_,
+              Run(kTestKeySystem, Robustness::kHardwareSecure, _))
+      .WillOnce(RunOnceCallback<2>(absl::nullopt));
   GetKeySystemCapabilities();
 
   ASSERT_TRUE(results_.count(kObserver1));
@@ -435,7 +490,7 @@
 }
 
 TEST_F(CdmRegistryImplTest, KeySystemCapabilities_HardwareSecureDisabled) {
-  RegisterForLazyInitialization();
+  RegisterForLazyHardwareSecureInitialization();
   SelectHardwareSecureDecryption(false);
   GetKeySystemCapabilities();
 
@@ -452,12 +507,16 @@
 }
 
 TEST_F(CdmRegistryImplTest, KeySystemCapabilities_SoftwareAndHardwareSecure) {
-  Register(GetTestCdmInfo());
-  RegisterForLazyInitialization();
+  RegisterForLazySoftwareSecureInitialization();
+  RegisterForLazyHardwareSecureInitialization();
   SelectHardwareSecureDecryption(true);
 
-  EXPECT_CALL(hw_secure_capability_cb_, Run(kTestKeySystem, _))
-      .WillOnce(RunOnceCallback<1>(GetTestCdmCapability()));
+  EXPECT_CALL(capability_cb_,
+              Run(kTestKeySystem, Robustness::kSoftwareSecure, _))
+      .WillOnce(RunOnceCallback<2>(GetTestCdmCapability()));
+  EXPECT_CALL(capability_cb_,
+              Run(kTestKeySystem, Robustness::kHardwareSecure, _))
+      .WillOnce(RunOnceCallback<2>(GetOtherCdmCapability()));
   GetKeySystemCapabilities();
 
   ASSERT_TRUE(results_.count(kObserver1));
@@ -467,7 +526,7 @@
   ASSERT_TRUE(key_system_capabilities.count(kTestKeySystem));
   const auto& support = key_system_capabilities[kTestKeySystem];
   ASSERT_EQ(support.sw_secure_capability.value(), GetTestCdmCapability());
-  ASSERT_EQ(support.hw_secure_capability.value(), GetTestCdmCapability());
+  ASSERT_EQ(support.hw_secure_capability.value(), GetOtherCdmCapability());
 }
 
 TEST_F(CdmRegistryImplTest, KeySystemCapabilities_MultipleObservers) {
@@ -495,14 +554,49 @@
   ASSERT_EQ(key_system_capabilities, results_[kObserver2][0]);
 }
 
-TEST_F(CdmRegistryImplTest,
-       KeySystemCapabilities_MultipleObservers_PendingLazyInitialize) {
+TEST_F(
+    CdmRegistryImplTest,
+    KeySystemCapabilities_MultipleObservers_PendingLazySoftwareSecureInitialize) {
+  RegisterForLazySoftwareSecureInitialization();
+  SelectHardwareSecureDecryption(false);
+
+  EXPECT_CALL(capability_cb_,
+              Run(kTestKeySystem, Robustness::kSoftwareSecure, _))
+      .WillOnce(RunOnceCallback<2>(GetTestCdmCapability()));
+
+  base::RunLoop run_loop;
+  cdm_registry_.ObserveKeySystemCapabilities(base::BindRepeating(
+      &CdmRegistryImplTest::OnKeySystemCapabilitiesUpdated,
+      base::Unretained(this), kObserver1, base::DoNothing()));
+  cdm_registry_.ObserveKeySystemCapabilities(base::BindRepeating(
+      &CdmRegistryImplTest::OnKeySystemCapabilitiesUpdated,
+      base::Unretained(this), kObserver2, run_loop.QuitClosure()));
+  run_loop.Run();
+
+  ASSERT_TRUE(results_.count(kObserver1));
+  ASSERT_EQ(results_[kObserver1].size(), 1u);
+  auto& key_system_capabilities = results_[kObserver1][0];
+  ASSERT_EQ(key_system_capabilities.size(), 1u);
+  ASSERT_TRUE(key_system_capabilities.count(kTestKeySystem));
+  const auto& support = key_system_capabilities[kTestKeySystem];
+  ASSERT_EQ(support.sw_secure_capability.value(), GetTestCdmCapability());
+  ASSERT_FALSE(support.hw_secure_capability);
+
+  ASSERT_TRUE(results_.count(kObserver2));
+  ASSERT_EQ(results_[kObserver2].size(), 1u);
+  ASSERT_EQ(key_system_capabilities, results_[kObserver2][0]);
+}
+
+TEST_F(
+    CdmRegistryImplTest,
+    KeySystemCapabilities_MultipleObservers_PendingLazyHardwareSecureInitialize) {
   Register(GetTestCdmInfo());
-  RegisterForLazyInitialization();
+  RegisterForLazyHardwareSecureInitialization();
   SelectHardwareSecureDecryption(true);
 
-  EXPECT_CALL(hw_secure_capability_cb_, Run(kTestKeySystem, _))
-      .WillOnce(RunOnceCallback<1>(GetTestCdmCapability()));
+  EXPECT_CALL(capability_cb_,
+              Run(kTestKeySystem, Robustness::kHardwareSecure, _))
+      .WillOnce(RunOnceCallback<2>(GetTestCdmCapability()));
 
   base::RunLoop run_loop;
   cdm_registry_.ObserveKeySystemCapabilities(base::BindRepeating(
@@ -527,14 +621,16 @@
   ASSERT_EQ(key_system_capabilities, results_[kObserver2][0]);
 }
 
-TEST_F(CdmRegistryImplTest,
-       KeySystemCapabilities_MultipleObservers_AfterLazyInitialize) {
+TEST_F(
+    CdmRegistryImplTest,
+    KeySystemCapabilities_MultipleObservers_AfterLazyHardwareSecureInitialize) {
   Register(GetTestCdmInfo());
-  RegisterForLazyInitialization();
+  RegisterForLazyHardwareSecureInitialization();
   SelectHardwareSecureDecryption(true);
 
-  EXPECT_CALL(hw_secure_capability_cb_, Run(kTestKeySystem, _))
-      .WillOnce(RunOnceCallback<1>(GetTestCdmCapability()));
+  EXPECT_CALL(capability_cb_,
+              Run(kTestKeySystem, Robustness::kHardwareSecure, _))
+      .WillOnce(RunOnceCallback<2>(GetTestCdmCapability()));
 
   {
     base::RunLoop run_loop;
@@ -589,9 +685,10 @@
 
   {
     base::RunLoop run_loop;
-    EXPECT_CALL(hw_secure_capability_cb_, Run(kTestKeySystem, _))
-        .WillOnce(RunOnceCallback<1>(GetOtherCdmCapability()));
-    RegisterForLazyInitialization();
+    EXPECT_CALL(capability_cb_,
+                Run(kTestKeySystem, Robustness::kHardwareSecure, _))
+        .WillOnce(RunOnceCallback<2>(GetOtherCdmCapability()));
+    RegisterForLazyHardwareSecureInitialization();
     run_loop.RunUntilIdle();
   }
 
@@ -613,11 +710,13 @@
   // Save the callbacks so we can control when and how they are fired.
   base::OnceCallback<void(absl::optional<media::CdmCapability>)> callback_1,
       callback_2, callback_3;
-  EXPECT_CALL(hw_secure_capability_cb_, Run(kTestKeySystem, _))
-      .WillOnce(MoveArg<1>(&callback_1))
-      .WillOnce(MoveArg<1>(&callback_2));
-  EXPECT_CALL(hw_secure_capability_cb_, Run(kOtherKeySystem, _))
-      .WillOnce(MoveArg<1>(&callback_3));
+  EXPECT_CALL(capability_cb_,
+              Run(kTestKeySystem, Robustness::kHardwareSecure, _))
+      .WillOnce(MoveArg<2>(&callback_1))
+      .WillOnce(MoveArg<2>(&callback_2));
+  EXPECT_CALL(capability_cb_,
+              Run(kOtherKeySystem, Robustness::kHardwareSecure, _))
+      .WillOnce(MoveArg<2>(&callback_3));
 
   // Register CdmInfo for lazy initialization.
   {
@@ -705,7 +804,7 @@
   gpu_info.overlay_info.direct_composition = false;
   GpuDataManagerImpl::GetInstance()->UpdateGpuInfo(gpu_info, absl::nullopt);
 
-  RegisterForLazyInitialization();
+  RegisterForLazyHardwareSecureInitialization();
   SelectHardwareSecureDecryption(true);
   GetKeySystemCapabilities();
 
diff --git a/content/browser/media/media_internals_cdm_helper.cc b/content/browser/media/media_internals_cdm_helper.cc
index e1c6dd4..9b7e9b2 100644
--- a/content/browser/media/media_internals_cdm_helper.cc
+++ b/content/browser/media/media_internals_cdm_helper.cc
@@ -19,15 +19,6 @@
 
 namespace {
 
-std::string GetCdmInfoRobustnessName(CdmInfo::Robustness robustness) {
-  switch (robustness) {
-    case CdmInfo::Robustness::kHardwareSecure:
-      return "Hardware Secure";
-    case CdmInfo::Robustness::kSoftwareSecure:
-      return "Software Secure";
-  }
-}
-
 std::string GetCdmInfoCapabilityStatusName(CdmInfo::Status status) {
   switch (status) {
     case CdmInfo::Status::kUninitialized:
diff --git a/content/browser/preloading/prefetch/prefetch_document_manager.cc b/content/browser/preloading/prefetch/prefetch_document_manager.cc
index 15e248e..5dbbafd9 100644
--- a/content/browser/preloading/prefetch/prefetch_document_manager.cc
+++ b/content/browser/preloading/prefetch/prefetch_document_manager.cc
@@ -81,7 +81,8 @@
   // No-Vary-Search equivalence. If there is not then stop.
   auto prefetch_iter = all_prefetches_.find(navigation_handle->GetURL());
   if (prefetch_iter == all_prefetches_.end() || !prefetch_iter->second) {
-    if (!base::FeatureList::IsEnabled(
+    if (!no_vary_search_support_enabled_ ||
+        !base::FeatureList::IsEnabled(
             network::features::kPrefetchNoVarySearch)) {
       return;
     }
@@ -299,7 +300,8 @@
 }
 
 void PrefetchDocumentManager::OnPrefetchedHeadReceived(const GURL& url) {
-  if (!base::FeatureList::IsEnabled(network::features::kPrefetchNoVarySearch)) {
+  if (!no_vary_search_support_enabled_ ||
+      !base::FeatureList::IsEnabled(network::features::kPrefetchNoVarySearch)) {
     return;
   }
   // Find the PrefetchContainer associated with |url|.
@@ -317,6 +319,10 @@
   referring_page_metrics_.prefetch_successful_count++;
 }
 
+void PrefetchDocumentManager::EnableNoVarySearchSupport() {
+  no_vary_search_support_enabled_ = true;
+}
+
 DOCUMENT_USER_DATA_KEY_IMPL(PrefetchDocumentManager);
 
 }  // namespace content
diff --git a/content/browser/preloading/prefetch/prefetch_document_manager.h b/content/browser/preloading/prefetch/prefetch_document_manager.h
index aa0a13f..21a4ac8d 100644
--- a/content/browser/preloading/prefetch/prefetch_document_manager.h
+++ b/content/browser/preloading/prefetch/prefetch_document_manager.h
@@ -96,6 +96,8 @@
   // Helper function to get the |NoVarySearchHelper| associated with |this|.
   const NoVarySearchHelper& GetNoVarySearchHelper() const;
 
+  void EnableNoVarySearchSupport();
+
   static void SetPrefetchServiceForTesting(PrefetchService* prefetch_service);
 
  private:
@@ -129,6 +131,8 @@
   // Used through the getter GetNoVarySearchHelper
   NoVarySearchHelper no_vary_search_helper_;
 
+  bool no_vary_search_support_enabled_ = false;
+
   base::WeakPtrFactory<PrefetchDocumentManager> weak_method_factory_{this};
 
   DOCUMENT_USER_DATA_KEY_DECL();
diff --git a/content/browser/preloading/prefetch/prefetch_document_manager_unittest.cc b/content/browser/preloading/prefetch/prefetch_document_manager_unittest.cc
index 718b3fbd..0ed7ae5b 100644
--- a/content/browser/preloading/prefetch/prefetch_document_manager_unittest.cc
+++ b/content/browser/preloading/prefetch/prefetch_document_manager_unittest.cc
@@ -129,6 +129,7 @@
   auto* prefetch_document_manager =
       PrefetchDocumentManager::GetOrCreateForCurrentDocument(
           &GetPrimaryMainFrame());
+  prefetch_document_manager->EnableNoVarySearchSupport();
   {
     // Create list of SpeculationCandidatePtrs.
     std::vector<blink::mojom::SpeculationCandidatePtr> candidates;
@@ -245,6 +246,9 @@
 }
 
 TEST_F(PrefetchDocumentManagerTest, ProcessSpeculationCandidates) {
+  base::test::ScopedFeatureList scoped_feature_list;
+  scoped_feature_list.InitAndDisableFeature(
+      network::features::kPrefetchNoVarySearch);
   // Create list of SpeculationCandidatePtrs.
   std::vector<blink::mojom::SpeculationCandidatePtr> candidates;
 
diff --git a/content/browser/preloading/prefetch/prefetch_service_unittest.cc b/content/browser/preloading/prefetch/prefetch_service_unittest.cc
index 6ae34239..11d5c942 100644
--- a/content/browser/preloading/prefetch/prefetch_service_unittest.cc
+++ b/content/browser/preloading/prefetch/prefetch_service_unittest.cc
@@ -261,10 +261,11 @@
   }
 
   virtual void InitScopedFeatureList() {
-    scoped_feature_list_.InitAndEnableFeatureWithParameters(
-        content::features::kPrefetchUseContentRefactor,
-        {{"ineligible_decoy_request_probability", "0"},
-         {"prefetch_container_lifetime_s", "-1"}});
+    scoped_feature_list_.InitWithFeaturesAndParameters(
+        {{content::features::kPrefetchUseContentRefactor,
+          {{"ineligible_decoy_request_probability", "0"},
+           {"prefetch_container_lifetime_s", "-1"}}}},
+        {network::features::kPrefetchNoVarySearch});
   }
 
   void MakePrefetchService(std::unique_ptr<MockPrefetchServiceDelegate>
@@ -280,9 +281,12 @@
 
   // Creates a prefetch request for |url| on the current main frame.
   void MakePrefetchOnMainFrame(const GURL& url,
-                               const PrefetchType& prefetch_type) {
+                               const PrefetchType& prefetch_type,
+                               bool enable_no_vary_search_header = false) {
     PrefetchDocumentManager* prefetch_document_manager =
         PrefetchDocumentManager::GetOrCreateForCurrentDocument(main_rfh());
+    if (enable_no_vary_search_header)
+      prefetch_document_manager->EnableNoVarySearchSupport();
     prefetch_document_manager->PrefetchUrl(url, prefetch_type,
                                            blink::mojom::Referrer(), nullptr);
   }
@@ -587,8 +591,9 @@
   // Enable feature, which means that we should be able to create a
   // PrefetchService instance.
   base::test::ScopedFeatureList scoped_feature_list;
-  scoped_feature_list.InitAndEnableFeature(
-      content::features::kPrefetchUseContentRefactor);
+  scoped_feature_list.InitWithFeatures(
+      {content::features::kPrefetchUseContentRefactor},
+      {network::features::kPrefetchNoVarySearch});
 
   EXPECT_TRUE(PrefetchService::CreateIfPossible(browser_context()));
 }
@@ -597,8 +602,9 @@
   // Disable feature, which means that we shouldn't be able to create a
   // PrefetchService instance.
   base::test::ScopedFeatureList scoped_feature_list;
-  scoped_feature_list.InitAndDisableFeature(
-      content::features::kPrefetchUseContentRefactor);
+  scoped_feature_list.InitWithFeatures(
+      {}, {content::features::kPrefetchUseContentRefactor,
+           network::features::kPrefetchNoVarySearch});
 
   EXPECT_FALSE(PrefetchService::CreateIfPossible(browser_context()));
 }
@@ -791,11 +797,12 @@
 class PrefetchServiceAllowAllDomainsTest : public PrefetchServiceTest {
  public:
   void InitScopedFeatureList() override {
-    scoped_feature_list_.InitAndEnableFeatureWithParameters(
-        content::features::kPrefetchUseContentRefactor,
-        {{"ineligible_decoy_request_probability", "0"},
-         {"prefetch_container_lifetime_s", "-1"},
-         {"allow_all_domains", "true"}});
+    scoped_feature_list_.InitWithFeaturesAndParameters(
+        {{content::features::kPrefetchUseContentRefactor,
+          {{"ineligible_decoy_request_probability", "0"},
+           {"prefetch_container_lifetime_s", "-1"},
+           {"allow_all_domains", "true"}}}},
+        {network::features::kPrefetchNoVarySearch});
   }
 };
 
@@ -874,11 +881,12 @@
     : public PrefetchServiceTest {
  public:
   void InitScopedFeatureList() override {
-    scoped_feature_list_.InitAndEnableFeatureWithParameters(
-        content::features::kPrefetchUseContentRefactor,
-        {{"ineligible_decoy_request_probability", "0"},
-         {"prefetch_container_lifetime_s", "-1"},
-         {"allow_all_domains_for_extended_preloading", "true"}});
+    scoped_feature_list_.InitWithFeaturesAndParameters(
+        {{content::features::kPrefetchUseContentRefactor,
+          {{"ineligible_decoy_request_probability", "0"},
+           {"prefetch_container_lifetime_s", "-1"},
+           {"allow_all_domains_for_extended_preloading", "true"}}}},
+        {network::features::kPrefetchNoVarySearch});
   }
 };
 
@@ -2263,11 +2271,12 @@
 class PrefetchServiceLimitedPrefetchesTest : public PrefetchServiceTest {
  public:
   void InitScopedFeatureList() override {
-    scoped_feature_list_.InitAndEnableFeatureWithParameters(
-        content::features::kPrefetchUseContentRefactor,
-        {{"ineligible_decoy_request_probability", "0"},
-         {"prefetch_container_lifetime_s", "-1"},
-         {"max_srp_prefetches", "2"}});
+    scoped_feature_list_.InitWithFeaturesAndParameters(
+        {{content::features::kPrefetchUseContentRefactor,
+          {{"ineligible_decoy_request_probability", "0"},
+           {"prefetch_container_lifetime_s", "-1"},
+           {"max_srp_prefetches", "2"}}}},
+        {network::features::kPrefetchNoVarySearch});
   }
 };
 
@@ -2425,11 +2434,12 @@
 class PrefetchServiceWithHTMLOnlyTest : public PrefetchServiceTest {
  public:
   void InitScopedFeatureList() override {
-    scoped_feature_list_.InitAndEnableFeatureWithParameters(
-        content::features::kPrefetchUseContentRefactor,
-        {{"ineligible_decoy_request_probability", "0"},
-         {"prefetch_container_lifetime_s", "-1"},
-         {"html_only", "true"}});
+    scoped_feature_list_.InitWithFeaturesAndParameters(
+        {{content::features::kPrefetchUseContentRefactor,
+          {{"ineligible_decoy_request_probability", "0"},
+           {"prefetch_container_lifetime_s", "-1"},
+           {"html_only", "true"}}}},
+        {network::features::kPrefetchNoVarySearch});
   }
 };
 
@@ -2497,10 +2507,11 @@
 class PrefetchServiceAlwaysMakeDecoyRequestTest : public PrefetchServiceTest {
  public:
   void InitScopedFeatureList() override {
-    scoped_feature_list_.InitAndEnableFeatureWithParameters(
-        content::features::kPrefetchUseContentRefactor,
-        {{"ineligible_decoy_request_probability", "1"},
-         {"prefetch_container_lifetime_s", "-1"}});
+    scoped_feature_list_.InitWithFeaturesAndParameters(
+        {{content::features::kPrefetchUseContentRefactor,
+          {{"ineligible_decoy_request_probability", "1"},
+           {"prefetch_container_lifetime_s", "-1"}}}},
+        {network::features::kPrefetchNoVarySearch});
   }
 };
 
@@ -2971,7 +2982,8 @@
 
   MakePrefetchOnMainFrame(GURL("https://example.com/?a=1"),
                           PrefetchType(/*use_isolated_network_context=*/true,
-                                       /*use_prefetch_proxy=*/true));
+                                       /*use_prefetch_proxy=*/true),
+                          /*enable_no_vary_search_header*/ true);
   base::RunLoop().RunUntilIdle();
 
   VerifyCommonRequestState(GURL("https://example.com/?a=1"),
diff --git a/content/browser/preloading/speculation_rules/speculation_host_impl.cc b/content/browser/preloading/speculation_rules/speculation_host_impl.cc
index c42b2be..0bbc49d 100644
--- a/content/browser/preloading/speculation_rules/speculation_host_impl.cc
+++ b/content/browser/preloading/speculation_rules/speculation_host_impl.cc
@@ -6,6 +6,7 @@
 
 #include <functional>
 
+#include "content/browser/preloading/prefetch/prefetch_document_manager.h"
 #include "content/browser/preloading/preloading_decider.h"
 #include "third_party/blink/public/common/features.h"
 
@@ -72,4 +73,12 @@
   preloading_decider->UpdateSpeculationCandidates(candidates);
 }
 
+void SpeculationHostImpl::EnableNoVarySearchSupport() {
+  auto* prefetch_document_manager =
+      PrefetchDocumentManager::GetOrCreateForCurrentDocument(
+          &render_frame_host());
+  DCHECK(prefetch_document_manager);
+  prefetch_document_manager->EnableNoVarySearchSupport();
+}
+
 }  // namespace content
diff --git a/content/browser/preloading/speculation_rules/speculation_host_impl.h b/content/browser/preloading/speculation_rules/speculation_host_impl.h
index 96acce3..d95679c 100644
--- a/content/browser/preloading/speculation_rules/speculation_host_impl.h
+++ b/content/browser/preloading/speculation_rules/speculation_host_impl.h
@@ -37,6 +37,7 @@
 
   void UpdateSpeculationCandidates(
       std::vector<blink::mojom::SpeculationCandidatePtr> candidates) override;
+  void EnableNoVarySearchSupport() override;
 };
 
 }  // namespace content
diff --git a/content/browser/renderer_host/back_forward_cache_impl.cc b/content/browser/renderer_host/back_forward_cache_impl.cc
index b6d7893..1d0859f2 100644
--- a/content/browser/renderer_host/back_forward_cache_impl.cc
+++ b/content/browser/renderer_host/back_forward_cache_impl.cc
@@ -654,8 +654,8 @@
     const RenderFrameHostImpl& rfh,
     const BackForwardCacheCanStoreDocumentResult& flattened_result,
     const perfetto::StaticString& caller) {
-  LOG(ERROR) << caller.value << ": " << rfh.GetLastCommittedURL() << " : "
-             << flattened_result.ToString();
+  VLOG(1) << caller.value << ": " << rfh.GetLastCommittedURL() << " : "
+          << flattened_result.ToString();
   TRACE_EVENT("navigation", caller,
               ChromeTrackEvent::kBackForwardCacheCanStoreDocumentResult,
               flattened_result);
diff --git a/content/browser/renderer_host/compositor_impl_android.cc b/content/browser/renderer_host/compositor_impl_android.cc
index 96b3bf25..b48bb7c 100644
--- a/content/browser/renderer_host/compositor_impl_android.cc
+++ b/content/browser/renderer_host/compositor_impl_android.cc
@@ -433,36 +433,26 @@
 
 void CompositorImpl::SetSurface(const base::android::JavaRef<jobject>& surface,
                                 bool can_be_used_with_surface_control) {
-  JNIEnv* env = base::android::AttachCurrentThread();
   gpu::GpuSurfaceTracker* tracker = gpu::GpuSurfaceTracker::Get();
 
   if (window_) {
     // Shut down GL context before unregistering surface.
     SetVisible(false);
     tracker->RemoveSurface(surface_handle_);
-    ANativeWindow_release(window_);
     window_ = nullptr;
     surface_handle_ = gpu::kNullSurfaceHandle;
   }
 
-  ANativeWindow* window = nullptr;
-  if (surface) {
-    // Note: This ensures that any local references used by
-    // ANativeWindow_fromSurface are released immediately. This is needed as a
-    // workaround for https://code.google.com/p/android/issues/detail?id=68174
-    base::android::ScopedJavaLocalFrame scoped_local_reference_frame(env);
-    window = ANativeWindow_fromSurface(env, surface.obj());
-  }
+  gl::ScopedJavaSurface scoped_surface(surface, /*auto_release=*/false);
+  gl::ScopedANativeWindow window(scoped_surface);
 
   if (window) {
-    window_ = window;
-    ANativeWindow_acquire(window);
+    window_ = std::move(window);
     // Register first, SetVisible() might create a LayerTreeFrameSink.
     surface_handle_ = tracker->AddSurfaceForNativeWidget(
         gpu::GpuSurfaceTracker::SurfaceRecord(
-            window, surface, can_be_used_with_surface_control));
+            std::move(scoped_surface), can_be_used_with_surface_control));
     SetVisible(true);
-    ANativeWindow_release(window);
   }
 }
 
diff --git a/content/browser/renderer_host/compositor_impl_android.h b/content/browser/renderer_host/compositor_impl_android.h
index 6992231..40d5e6f 100644
--- a/content/browser/renderer_host/compositor_impl_android.h
+++ b/content/browser/renderer_host/compositor_impl_android.h
@@ -40,8 +40,7 @@
 #include "ui/android/window_android_compositor.h"
 #include "ui/compositor/compositor_lock.h"
 #include "ui/display/display_observer.h"
-
-struct ANativeWindow;
+#include "ui/gl/android/scoped_a_native_window.h"
 
 namespace cc {
 class AnimationHost;
@@ -263,7 +262,7 @@
   gfx::Size size_;
   bool requires_alpha_channel_ = false;
 
-  raw_ptr<ANativeWindow> window_;
+  gl::ScopedANativeWindow window_;
   gpu::SurfaceHandle surface_handle_;
   std::unique_ptr<ScopedCachedBackBuffer> cached_back_buffer_;
 
diff --git a/content/browser/resources/attribution_reporting/attribution_internals.ts b/content/browser/resources/attribution_reporting/attribution_internals.ts
index 9ddde4a..e3e86d5 100644
--- a/content/browser/resources/attribution_reporting/attribution_internals.ts
+++ b/content/browser/resources/attribution_reporting/attribution_internals.ts
@@ -9,7 +9,7 @@
 import {getTrustedHTML} from 'chrome://resources/js/static_types.js';
 import {Origin} from 'chrome://resources/mojo/url/mojom/origin.mojom-webui.js';
 
-import {ClearedDebugKey, ClearedDebugKey_Type, FailedSourceRegistration, Handler as AttributionInternalsHandler, HandlerRemote as AttributionInternalsHandlerRemote, ObserverInterface, ObserverReceiver, ReportID, WebUIDebugReport, WebUIReport, WebUISource, WebUISource_Attributability, WebUITrigger, WebUITrigger_Status} from './attribution_internals.mojom-webui.js';
+import {FailedSourceRegistration, Handler as AttributionInternalsHandler, HandlerRemote as AttributionInternalsHandlerRemote, ObserverInterface, ObserverReceiver, ReportID, WebUIDebugReport, WebUIReport, WebUISource, WebUISource_Attributability, WebUITrigger, WebUITrigger_Status} from './attribution_internals.mojom-webui.js';
 import {AttributionInternalsTableElement} from './attribution_internals_table.js';
 import {ReportType, SourceType} from './attribution_reporting.mojom-webui.js';
 import {SourceRegistrationError} from './source_registration_error.mojom-webui.js';
@@ -258,7 +258,15 @@
     this.filterData = JSON.stringify(mojo.filterData, null, ' ');
     this.aggregationKeys =
         JSON.stringify(mojo.aggregationKeys, bigintReplacer, ' ');
-    this.debugKey = mojo.debugKey ? mojo.debugKey.value.toString() : '';
+
+    if (mojo.debugKey?.debugKey !== undefined) {
+      this.debugKey = mojo.debugKey.debugKey.toString();
+    } else if (mojo.debugKey?.clearedDebugKey !== undefined) {
+      this.debugKey = `Cleared (was ${mojo.debugKey.clearedDebugKey})`;
+    } else {
+      this.debugKey = '';
+    }
+
     this.dedupKeys = mojo.dedupKeys.join(', ');
     this.aggregatableBudgetConsumed = mojo.aggregatableBudgetConsumed;
     this.aggregatableDedupKeys = mojo.aggregatableDedupKeys.join(', ');
@@ -343,6 +351,7 @@
   destinationOrigin: string;
   reportingOrigin: string;
   registrationJson: string;
+  clearedDebugKey: string;
   eventLevelStatus: string;
   aggregatableStatus: string;
 
@@ -351,6 +360,8 @@
     this.destinationOrigin = originToText(mojo.destinationOrigin);
     this.reportingOrigin = originToText(mojo.reportingOrigin);
     this.registrationJson = mojo.registrationJson;
+    this.clearedDebugKey =
+        mojo.clearedDebugKey ? `${mojo.clearedDebugKey.value}` : '';
     this.eventLevelStatus = triggerStatusToText(mojo.eventLevelStatus);
     this.aggregatableStatus = triggerStatusToText(mojo.aggregatableStatus);
   }
@@ -373,6 +384,8 @@
               'Reporting Origin', (e) => e.reportingOrigin),
           new CodeColumn<Trigger>(
               'Registration JSON', (e) => e.registrationJson),
+          new ValueColumn<Trigger, string>(
+              'Cleared Debug Key', (e) => e.clearedDebugKey),
         ],
         0,  // Sort by trigger time by default.
         'No triggers.',
@@ -717,49 +730,6 @@
   abstract renderMetadata(td: HTMLElement): void;
 }
 
-const CLEARED_DEBUG_KEY_COLS: Array<Column<ClearedDebugKeyLog>> = [
-  new ValueColumn<ClearedDebugKeyLog, string>(
-      'Cleared Debug Key', e => e.clearedDebugKey),
-  new ValueColumn<ClearedDebugKeyLog, string>('From', e => e.clearedFrom),
-  new ValueColumn<ClearedDebugKeyLog, string>(
-      'Reporting Origin', e => e.reportingOrigin),
-];
-
-class ClearedDebugKeyLog extends Log {
-  readonly clearedFrom: string;
-  readonly clearedDebugKey: string;
-
-  constructor(mojo: ClearedDebugKey) {
-    super(mojo);
-
-    this.clearedDebugKey = `${mojo.clearedDebugKey.value}`;
-
-    switch (mojo.clearedFrom) {
-      case (ClearedDebugKey_Type.kSource):
-        this.clearedFrom = 'Source';
-        break;
-      case (ClearedDebugKey_Type.kTrigger):
-        this.clearedFrom = 'Trigger';
-        break;
-      default:
-        this.clearedFrom = 'Unknown type';
-        break;
-    }
-  }
-
-  renderDescription(td: HTMLElement): void {
-    renderA(
-        td,
-        'Cleared Debug Key',
-        'https://github.com/WICG/attribution-reporting-api/blob/main/EVENT.md#attribution-success-debugging-reports',
-    );
-  }
-
-  renderMetadata(td: HTMLElement) {
-    renderDL(td, this, CLEARED_DEBUG_KEY_COLS);
-  }
-}
-
 const FAILED_SOURCE_REGISTRATION_COLS:
     Array<Column<FailedSourceRegistrationLog>> = [
       new ValueColumn<FailedSourceRegistrationLog, string>(
@@ -1151,11 +1121,6 @@
     assert(logTableModel);
     logTableModel.addLog(new FailedSourceRegistrationLog(mojo));
   }
-
-  onDebugKeyCleared(mojo: ClearedDebugKey) {
-    assert(logTableModel);
-    logTableModel.addLog(new ClearedDebugKeyLog(mojo));
-  }
 }
 
 function installUnreadIndicator(model: TableModel<any>, tab: HTMLElement|null) {
diff --git a/content/browser/shared_storage/shared_storage_document_service_impl.cc b/content/browser/shared_storage/shared_storage_document_service_impl.cc
index 9af47aa8..9eb65f2e 100644
--- a/content/browser/shared_storage/shared_storage_document_service_impl.cc
+++ b/content/browser/shared_storage/shared_storage_document_service_impl.cc
@@ -344,8 +344,8 @@
     return true;
 
   return GetContentClient()->browser()->IsSharedStorageAllowed(
-      &render_frame_host(), main_frame_origin_,
-      render_frame_host().GetLastCommittedOrigin());
+      render_frame_host().GetBrowserContext(), &render_frame_host(),
+      main_frame_origin_, render_frame_host().GetLastCommittedOrigin());
 }
 
 std::string SharedStorageDocumentServiceImpl::SerializeLastCommittedOrigin()
diff --git a/content/browser/shared_storage/shared_storage_worklet_host.cc b/content/browser/shared_storage/shared_storage_worklet_host.cc
index ed6958d..ecce4f40 100644
--- a/content/browser/shared_storage/shared_storage_worklet_host.cc
+++ b/content/browser/shared_storage/shared_storage_worklet_host.cc
@@ -864,9 +864,10 @@
 }
 
 bool SharedStorageWorkletHost::IsSharedStorageAllowed() {
+  RenderFrameHost* rfh =
+      document_service_ ? &(document_service_->render_frame_host()) : nullptr;
   return GetContentClient()->browser()->IsSharedStorageAllowed(
-      &(document_service_->render_frame_host()), main_frame_origin_,
-      shared_storage_origin_);
+      browser_context_, rfh, main_frame_origin_, shared_storage_origin_);
 }
 
 }  // namespace content
diff --git a/content/common/android/surface_wrapper.cc b/content/common/android/surface_wrapper.cc
index fed61b3..ba00294 100644
--- a/content/common/android/surface_wrapper.cc
+++ b/content/common/android/surface_wrapper.cc
@@ -21,10 +21,10 @@
   return Java_SurfaceWrapper_canBeUsedWithSurfaceControl(env, obj);
 }
 
-base::android::ScopedJavaLocalRef<jobject> JNI_SurfaceWrapper_getSurface(
+base::android::ScopedJavaLocalRef<jobject> JNI_SurfaceWrapper_takeSurface(
     JNIEnv* env,
     const base::android::JavaRef<jobject>& obj) {
-  return Java_SurfaceWrapper_getSurface(env, obj);
+  return Java_SurfaceWrapper_takeSurface(env, obj);
 }
 
 }  // namespace content.
diff --git a/content/common/android/surface_wrapper.h b/content/common/android/surface_wrapper.h
index 0dde535..c6223dcc 100644
--- a/content/common/android/surface_wrapper.h
+++ b/content/common/android/surface_wrapper.h
@@ -18,7 +18,7 @@
     JNIEnv* env,
     const base::android::JavaRef<jobject>& obj);
 
-base::android::ScopedJavaLocalRef<jobject> JNI_SurfaceWrapper_getSurface(
+base::android::ScopedJavaLocalRef<jobject> JNI_SurfaceWrapper_takeSurface(
     JNIEnv* env,
     const base::android::JavaRef<jobject>& obj);
 
diff --git a/content/public/android/java/src/org/chromium/content/common/SurfaceWrapper.java b/content/public/android/java/src/org/chromium/content/common/SurfaceWrapper.java
index 44818a0..ed896228 100644
--- a/content/public/android/java/src/org/chromium/content/common/SurfaceWrapper.java
+++ b/content/public/android/java/src/org/chromium/content/common/SurfaceWrapper.java
@@ -18,21 +18,23 @@
 @JNINamespace("content")
 @MainDex
 public class SurfaceWrapper implements Parcelable {
-    private final Surface mSurface;
+    private Surface mSurface;
     private final boolean mCanBeUsedWithSurfaceControl;
 
-    public SurfaceWrapper(Surface surface, boolean canBeUsedWithSurfaceControl) {
+    private SurfaceWrapper(Surface surface, boolean canBeUsedWithSurfaceControl) {
         mSurface = surface;
         mCanBeUsedWithSurfaceControl = canBeUsedWithSurfaceControl;
     }
 
     @CalledByNative
-    public Surface getSurface() {
-        return mSurface;
+    private Surface takeSurface() {
+        Surface surface = mSurface;
+        mSurface = null;
+        return surface;
     }
 
     @CalledByNative
-    public boolean canBeUsedWithSurfaceControl() {
+    private boolean canBeUsedWithSurfaceControl() {
         return mCanBeUsedWithSurfaceControl;
     }
 
diff --git a/content/public/browser/content_browser_client.cc b/content/public/browser/content_browser_client.cc
index 110e353..b075b996 100644
--- a/content/public/browser/content_browser_client.cc
+++ b/content/public/browser/content_browser_client.cc
@@ -511,6 +511,7 @@
 }
 
 bool ContentBrowserClient::IsSharedStorageAllowed(
+    content::BrowserContext* browser_context,
     content::RenderFrameHost* rfh,
     const url::Origin& top_frame_origin,
     const url::Origin& accessing_origin) {
diff --git a/content/public/browser/content_browser_client.h b/content/public/browser/content_browser_client.h
index 99f1c92..450b57f 100644
--- a/content/public/browser/content_browser_client.h
+++ b/content/public/browser/content_browser_client.h
@@ -882,7 +882,10 @@
 
   // Allows the embedder to control if Shared Storage API operations can happen
   // in a given context.
-  virtual bool IsSharedStorageAllowed(content::RenderFrameHost* rfh,
+  //
+  // Note that `rfh` can be nullptr.
+  virtual bool IsSharedStorageAllowed(content::BrowserContext* browser_context,
+                                      content::RenderFrameHost* rfh,
                                       const url::Origin& top_frame_origin,
                                       const url::Origin& accessing_origin);
 
diff --git a/content/public/common/BUILD.gn b/content/public/common/BUILD.gn
index 80c8aa2..292a43ab 100644
--- a/content/public/common/BUILD.gn
+++ b/content/public/common/BUILD.gn
@@ -153,6 +153,7 @@
 
   sources = [
     "bindings_policy.h",
+    "cdm_info.cc",
     "cdm_info.h",
     "child_process_host.h",
     "child_process_host_delegate.h",
diff --git a/content/public/common/cdm_info.cc b/content/public/common/cdm_info.cc
new file mode 100644
index 0000000..36e73b2
--- /dev/null
+++ b/content/public/common/cdm_info.cc
@@ -0,0 +1,23 @@
+// Copyright 2022 The Chromium Authors
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include <iostream>
+
+#include "cdm_info.h"
+
+#include "base/notreached.h"
+
+namespace content {
+
+std::string GetCdmInfoRobustnessName(CdmInfo::Robustness robustness) {
+  switch (robustness) {
+    case CdmInfo::Robustness::kHardwareSecure:
+      return "Hardware Secure";
+    case CdmInfo::Robustness::kSoftwareSecure:
+      return "Software Secure";
+  }
+  NOTREACHED();
+}
+
+}  // namespace content
\ No newline at end of file
diff --git a/content/public/common/cdm_info.h b/content/public/common/cdm_info.h
index 80a08df..2733702b 100644
--- a/content/public/common/cdm_info.h
+++ b/content/public/common/cdm_info.h
@@ -5,6 +5,7 @@
 #ifndef CONTENT_PUBLIC_COMMON_CDM_INFO_H_
 #define CONTENT_PUBLIC_COMMON_CDM_INFO_H_
 
+#include <iosfwd>
 #include <string>
 #include <vector>
 
@@ -104,6 +105,14 @@
   base::FilePath path;
 };
 
+CONTENT_EXPORT std::string GetCdmInfoRobustnessName(
+    CdmInfo::Robustness robustness);
+
+inline std::ostream& operator<<(std::ostream& os,
+                                CdmInfo::Robustness robustness) {
+  return os << GetCdmInfoRobustnessName(robustness);
+}
+
 }  // namespace content
 
 #endif  // CONTENT_PUBLIC_COMMON_CDM_INFO_H_
diff --git a/device/fido/features.cc b/device/fido/features.cc
index 69d89d34..361f2dc 100644
--- a/device/fido/features.cc
+++ b/device/fido/features.cc
@@ -52,4 +52,8 @@
              "DisableWebAuthnWithBrokenCerts",
              base::FEATURE_ENABLED_BY_DEFAULT);
 
+BASE_FEATURE(kWebAuthnNoPasskeysError,
+             "WebAuthenticationNoPasskeysError",
+             base::FEATURE_ENABLED_BY_DEFAULT);
+
 }  // namespace device
diff --git a/device/fido/features.h b/device/fido/features.h
index c8a846c1..7ba3470 100644
--- a/device/fido/features.h
+++ b/device/fido/features.h
@@ -53,6 +53,10 @@
 COMPONENT_EXPORT(DEVICE_FIDO)
 BASE_DECLARE_FEATURE(kDisableWebAuthnWithBrokenCerts);
 
+// Enable a special-case dialog for when there are no internal credentials.
+COMPONENT_EXPORT(DEVICE_FIDO)
+BASE_DECLARE_FEATURE(kWebAuthnNoPasskeysError);
+
 }  // namespace device
 
 #endif  // DEVICE_FIDO_FEATURES_H_
diff --git a/device/fido/fido_request_handler_base.h b/device/fido/fido_request_handler_base.h
index 66566100..b6f57a5 100644
--- a/device/fido/fido_request_handler_base.h
+++ b/device/fido/fido_request_handler_base.h
@@ -123,6 +123,18 @@
     // authenticator and thus have privacy implications.
     ResidentKeyRequirement resident_key_requirement =
         ResidentKeyRequirement::kDiscouraged;
+
+    // transport_list_did_include_internal is set to true during a getAssertion
+    // request if at least one element of the allowList included the "internal"
+    // transport, or didn't have any transports.
+    //
+    // An embedder may use this to show a more precise UI when no transports
+    // are available. If the lack of transports is because the allowList only
+    // contained NFC-based credentials, and there's no NFC support, then that
+    // might be meaningfully different from the case where the allowList
+    // contained credentials that could have been on the local device but
+    // weren't.
+    bool transport_list_did_include_internal = false;
   };
 
   class COMPONENT_EXPORT(DEVICE_FIDO) Observer {
diff --git a/device/fido/get_assertion_request_handler.cc b/device/fido/get_assertion_request_handler.cc
index 393f8d6..7780b113 100644
--- a/device/fido/get_assertion_request_handler.cc
+++ b/device/fido/get_assertion_request_handler.cc
@@ -10,6 +10,7 @@
 #include <vector>
 
 #include "base/bind.h"
+#include "base/containers/contains.h"
 #include "base/containers/cxx20_erase.h"
 #include "base/metrics/histogram_functions.h"
 #include "base/ranges/algorithm.h"
@@ -322,6 +323,13 @@
       request_.allow_list.empty();
   transport_availability_info().is_off_the_record_context =
       request_.is_off_the_record_context;
+  transport_availability_info().transport_list_did_include_internal =
+      std::any_of(request_.allow_list.begin(), request_.allow_list.end(),
+                  [](const PublicKeyCredentialDescriptor& cred) {
+                    return cred.transports.empty() ||
+                           base::Contains(cred.transports,
+                                          FidoTransportProtocol::kInternal);
+                  });
 
   if (request_.allow_list.empty()) {
     // Resident credential requests always involve user verification.
diff --git a/docs/adding_to_third_party.md b/docs/adding_to_third_party.md
index 98531ca1..1ee92840 100644
--- a/docs/adding_to_third_party.md
+++ b/docs/adding_to_third_party.md
@@ -27,9 +27,29 @@
 * Motivation of your project
 * Design docs
 * Additional checkout size
+   * If the increase is significant (e.g., 20+ MB), can we consider limiting the
+   files to be checked in?
 * Build time increase
-* Binary size increase on Android ([official](https://www.chromium.org/developers/gn-build-configuration)  builds)
+   * If the increase is significant (e.g., 30+ seconds), can we consider making
+   this an optional build target?
+* Binary size increase on Android ([official](https://www.chromium.org/developers/gn-build-configuration) builds)
+   * Any 16 KB increase on Android is flagged on the build bots and
+   justification is needed.
 * Binary size increase on Windows
+* Is this library maintained on all platforms that we will use it on?
+   * If not, will the Chrome org be expected to maintain this for some or all
+   platforms?
+* Does it have any performance / memory implications (esp. on Android)? Was the
+library designed with intended use on Android?
+* Do we really need the library? Is there any alternative such as an existing
+library already in Chromium? If introducing a library with similar functionality
+as existing, will it be easy for another developer to understand which should be
+used where? Will you commit to consolidating uses in Chromium and remove the
+alternative libraries?
+* For desktop (Win/Mac/Linux/ChromeOS), does the dependency introduce closed
+source components (e.g., binaries, WASM binaries, obfuscated code)? If yes,
+please reach out to Chrome ATLs.
+
 
 Googlers can access [go/chrome-atls](https://goto.google.com/chrome-atls) and review
 existing topics in g/chrome-atls, and can also come to office hours to ask
diff --git a/docs/speed/perf_lab_platforms.md b/docs/speed/perf_lab_platforms.md
index 908a401..f49c865 100644
--- a/docs/speed/perf_lab_platforms.md
+++ b/docs/speed/perf_lab_platforms.md
@@ -8,7 +8,7 @@
 
  * [android-go-perf](https://ci.chromium.org/p/chrome/builders/ci/android-go-perf): Android O (gobo).
  * [android-go-perf-pgo](https://ci.chromium.org/p/chrome/builders/ci/android-go-perf-pgo): Android O (gobo).
- * [android-go-wembley-perf](https://ci.chromium.org/p/chrome/builders/ci/android-go-wembley-perf): Android M | MASTER.
+ * [android-go-wembley-perf](https://ci.chromium.org/p/chrome/builders/ci/android-go-wembley-perf): Android 9131443.
  * [android-go_webview-perf](https://ci.chromium.org/p/chrome/builders/ci/android-go_webview-perf): Android OPM1.171019.021 (gobo).
  * [android-pixel2-perf](https://ci.chromium.org/p/chrome/builders/ci/android-pixel2-perf): Android OPM1.171019.021.
  * [android-pixel2-perf-calibration](https://ci.chromium.org/p/chrome/builders/ci/android-pixel2-perf-calibration): Android OPM1.171019.021.
diff --git a/extensions/browser/api/management/management_api.cc b/extensions/browser/api/management/management_api.cc
index b4427c47..9a4ea8c 100644
--- a/extensions/browser/api/management/management_api.cc
+++ b/extensions/browser/api/management/management_api.cc
@@ -360,8 +360,7 @@
   std::string error;
   scoped_refptr<Extension> extension =
       Extension::Create(base::FilePath(), ManifestLocation::kInvalidLocation,
-                        base::DictAdapterForMigration(*parsed_manifest),
-                        Extension::NO_FLAGS, &error);
+                        *parsed_manifest, Extension::NO_FLAGS, &error);
   // TODO(lazyboy): Do we need to use |error|?
   if (!extension) {
     Respond(Error(keys::kExtensionCreateError));
diff --git a/extensions/browser/image_loader_unittest.cc b/extensions/browser/image_loader_unittest.cc
index 30b1c04..e2177ce 100644
--- a/extensions/browser/image_loader_unittest.cc
+++ b/extensions/browser/image_loader_unittest.cc
@@ -79,19 +79,20 @@
     std::string error;
     JSONFileValueDeserializer deserializer(
         extension_dir.AppendASCII("manifest.json"));
-    std::unique_ptr<base::DictionaryValue> valid_value =
-        base::DictionaryValue::From(
-            deserializer.Deserialize(&error_code, &error));
+    std::unique_ptr<base::Value> valid_value =
+        deserializer.Deserialize(&error_code, &error);
     EXPECT_EQ(0, error_code) << error;
     if (error_code != 0)
       return nullptr;
 
     EXPECT_TRUE(valid_value.get());
-    if (!valid_value)
+    EXPECT_TRUE(valid_value->is_dict());
+    if (!valid_value || !valid_value->is_dict()) {
       return nullptr;
+    }
 
-    return Extension::Create(
-        extension_dir, location, *valid_value, Extension::NO_FLAGS, &error);
+    return Extension::Create(extension_dir, location, valid_value->GetDict(),
+                             Extension::NO_FLAGS, &error);
   }
 
   gfx::Image image_;
diff --git a/extensions/browser/sandboxed_unpacker.cc b/extensions/browser/sandboxed_unpacker.cc
index 182ba5bd..e478f9f 100644
--- a/extensions/browser/sandboxed_unpacker.cc
+++ b/extensions/browser/sandboxed_unpacker.cc
@@ -538,8 +538,7 @@
 
   std::string error_msg;
   scoped_refptr<Extension> extension(
-      Extension::Create(extension_root_, location_,
-                        base::Value::AsDictionaryValue(manifest.value()),
+      Extension::Create(extension_root_, location_, manifest->GetDict(),
                         creation_flags_, extension_id_, &error_msg));
   if (!extension) {
     ReportUnpackExtensionFailed(error_msg);
diff --git a/extensions/common/extension.cc b/extensions/common/extension.cc
index 9cdcae6..df8bc5b 100644
--- a/extensions/common/extension.cc
+++ b/extensions/common/extension.cc
@@ -140,7 +140,7 @@
 
 // Computes the |extension_id| from the given parameters. On success, returns
 // true. On failure, populates |error| and returns false.
-bool ComputeExtensionID(const base::DictAdapterForMigration& manifest,
+bool ComputeExtensionID(const base::Value::Dict& manifest,
                         const base::FilePath& path,
                         int creation_flags,
                         std::u16string* error,
@@ -221,12 +221,11 @@
 }
 
 // static
-scoped_refptr<Extension> Extension::Create(
-    const base::FilePath& path,
-    ManifestLocation location,
-    const base::DictAdapterForMigration& value,
-    int flags,
-    std::string* utf8_error) {
+scoped_refptr<Extension> Extension::Create(const base::FilePath& path,
+                                           ManifestLocation location,
+                                           const base::Value::Dict& value,
+                                           int flags,
+                                           std::string* utf8_error) {
   return Extension::Create(path,
                            location,
                            value,
@@ -237,13 +236,12 @@
 
 // TODO(sungguk): Continue removing std::string errors and replacing
 // with std::u16string. See http://crbug.com/71980.
-scoped_refptr<Extension> Extension::Create(
-    const base::FilePath& path,
-    ManifestLocation location,
-    const base::DictAdapterForMigration& value,
-    int flags,
-    const std::string& explicit_id,
-    std::string* utf8_error) {
+scoped_refptr<Extension> Extension::Create(const base::FilePath& path,
+                                           ManifestLocation location,
+                                           const base::Value::Dict& value,
+                                           int flags,
+                                           const std::string& explicit_id,
+                                           std::string* utf8_error) {
   base::ElapsedTimer timer;
   DCHECK(utf8_error);
   std::u16string error;
diff --git a/extensions/common/extension.h b/extensions/common/extension.h
index 0862216d..91698320 100644
--- a/extensions/common/extension.h
+++ b/extensions/common/extension.h
@@ -32,10 +32,6 @@
 #error "Extensions must be enabled"
 #endif
 
-namespace base {
-class DictAdapterForMigration;
-}
-
 namespace extensions {
 class HashedExtensionId;
 class PermissionsData;
@@ -156,22 +152,20 @@
   Extension(const Extension&) = delete;
   Extension& operator=(const Extension&) = delete;
 
-  static scoped_refptr<Extension> Create(
-      const base::FilePath& path,
-      mojom::ManifestLocation location,
-      const base::DictAdapterForMigration& value,
-      int flags,
-      std::string* error);
+  static scoped_refptr<Extension> Create(const base::FilePath& path,
+                                         mojom::ManifestLocation location,
+                                         const base::Value::Dict& value,
+                                         int flags,
+                                         std::string* error);
 
   // In a few special circumstances, we want to create an Extension and give it
   // an explicit id. Most consumers should just use the other Create() method.
-  static scoped_refptr<Extension> Create(
-      const base::FilePath& path,
-      mojom::ManifestLocation location,
-      const base::DictAdapterForMigration& value,
-      int flags,
-      const ExtensionId& explicit_id,
-      std::string* error);
+  static scoped_refptr<Extension> Create(const base::FilePath& path,
+                                         mojom::ManifestLocation location,
+                                         const base::Value::Dict& value,
+                                         int flags,
+                                         const ExtensionId& explicit_id,
+                                         std::string* error);
 
   // Valid schemes for web extent URLPatterns.
   static const int kValidWebExtentSchemes;
@@ -307,9 +301,7 @@
   const std::vector<InstallWarning>& install_warnings() const {
     return install_warnings_;
   }
-  const extensions::Manifest* manifest() const {
-    return manifest_.get();
-  }
+  const extensions::Manifest* manifest() const { return manifest_.get(); }
   bool wants_file_access() const { return wants_file_access_; }
   // TODO(rdevlin.cronin): This is needed for ContentScriptsHandler, and should
   // be moved out as part of crbug.com/159265. This should not be used anywhere
@@ -344,7 +336,7 @@
   bool is_extension() const;            // Regular browser extension, not an app
   bool is_shared_module() const;        // Shared module
   bool is_theme() const;                // Theme
-  bool is_login_screen_extension() const;  // Extension on login screen.
+  bool is_login_screen_extension() const;     // Extension on login screen.
   bool is_chromeos_system_extension() const;  // ChromeOS System Extension.
 
   // True if this is a platform app, hosted app, or legacy packaged app.
@@ -493,7 +485,7 @@
   base::GUID guid_;
 };
 
-typedef std::vector<scoped_refptr<const Extension> > ExtensionList;
+typedef std::vector<scoped_refptr<const Extension>> ExtensionList;
 
 // Handy struct to pass core extension info around.
 struct ExtensionInfo {
diff --git a/extensions/common/extension_builder.cc b/extensions/common/extension_builder.cc
index 9207b4d5..bc9b698 100644
--- a/extensions/common/extension_builder.cc
+++ b/extensions/common/extension_builder.cc
@@ -144,11 +144,17 @@
     id_ = crx_file::id_util::GenerateId(manifest_data_->name);
 
   std::string error;
+
+  // This allows `*manifest_value` to be passed as a reference instead of
+  // needing to be cloned.
+  absl::optional<base::Value::Dict> manifest_data_value;
+  if (manifest_data_) {
+    manifest_data_value = manifest_data_->GetValue();
+  }
   scoped_refptr<const Extension> extension = Extension::Create(
       path_, location_,
-      manifest_data_ ? base::DictAdapterForMigration(manifest_data_->GetValue())
-                     : *manifest_value_,
-      flags_, id_, &error);
+      manifest_data_value ? *manifest_data_value : *manifest_value_, flags_,
+      id_, &error);
 
   CHECK(error.empty()) << error;
   CHECK(extension);
diff --git a/extensions/common/features/simple_feature_unittest.cc b/extensions/common/features/simple_feature_unittest.cc
index 8894dbb..520a8f2 100644
--- a/extensions/common/features/simple_feature_unittest.cc
+++ b/extensions/common/features/simple_feature_unittest.cc
@@ -338,11 +338,11 @@
   feature.set_min_manifest_version(21);
   feature.set_max_manifest_version(25);
 
-  base::DictionaryValue manifest;
-  manifest.SetStringKey("name", "test");
-  manifest.SetStringKey("version", "1");
-  manifest.SetIntKey("manifest_version", 21);
-  manifest.SetStringPath("app.launch.local_path", "foo.html");
+  base::Value::Dict manifest;
+  manifest.Set("name", "test");
+  manifest.Set("version", "1");
+  manifest.Set("manifest_version", 21);
+  manifest.SetByDottedPath("app.launch.local_path", "foo.html");
 
   std::string error;
   scoped_refptr<const Extension> extension(
@@ -456,11 +456,11 @@
 }
 
 TEST_F(SimpleFeatureTest, SessionType) {
-  base::DictionaryValue manifest;
-  manifest.SetStringKey("name", "test");
-  manifest.SetStringKey("version", "1");
-  manifest.SetIntKey("manifest_version", 2);
-  manifest.SetStringPath("app.launch.local_path", "foo.html");
+  base::Value::Dict manifest;
+  manifest.Set("name", "test");
+  manifest.Set("version", "1");
+  manifest.Set("manifest_version", 2);
+  manifest.SetByDottedPath("app.launch.local_path", "foo.html");
 
   std::string error;
   scoped_refptr<const Extension> extension(
diff --git a/extensions/common/manifest_handler_unittest.cc b/extensions/common/manifest_handler_unittest.cc
index 47661eab..ba85545 100644
--- a/extensions/common/manifest_handler_unittest.cc
+++ b/extensions/common/manifest_handler_unittest.cc
@@ -213,18 +213,17 @@
   ScopedTestingManifestHandlerRegistry scoped_registry;
   // Can't use ExtensionBuilder, because this extension will fail to
   // be parsed.
-  std::unique_ptr<base::DictionaryValue> manifest_a(
-      DictionaryBuilder()
-          .Set("name", "no name")
-          .Set("version", "0")
-          .Set("manifest_version", 2)
-          .Set("a", 1)
-          .Build());
+  base::Value::Dict manifest_a(DictionaryBuilder()
+                                   .Set("name", "no name")
+                                   .Set("version", "0")
+                                   .Set("manifest_version", 2)
+                                   .Set("a", 1)
+                                   .BuildDict());
 
   // Succeeds when "a" is not recognized.
   std::string error;
   scoped_refptr<Extension> extension = Extension::Create(
-      base::FilePath(), mojom::ManifestLocation::kInvalidLocation, *manifest_a,
+      base::FilePath(), mojom::ManifestLocation::kInvalidLocation, manifest_a,
       Extension::NO_FLAGS, &error);
   EXPECT_TRUE(extension.get());
 
@@ -237,7 +236,7 @@
 
   extension = Extension::Create(base::FilePath(),
                                 mojom::ManifestLocation::kInvalidLocation,
-                                *manifest_a, Extension::NO_FLAGS, &error);
+                                manifest_a, Extension::NO_FLAGS, &error);
   EXPECT_FALSE(extension.get());
   EXPECT_EQ("A", error);
 }
diff --git a/extensions/common/manifest_test.cc b/extensions/common/manifest_test.cc
index dd15eb4..74a8527 100644
--- a/extensions/common/manifest_test.cc
+++ b/extensions/common/manifest_test.cc
@@ -126,8 +126,7 @@
   const base::Value& value = manifest.GetManifest(test_data_dir, error);
   if (value.is_none())
     return nullptr;
-  return Extension::Create(test_data_dir.DirName(), location,
-                           base::DictAdapterForMigration(value.GetDict()),
+  return Extension::Create(test_data_dir.DirName(), location, value.GetDict(),
                            flags, GetTestExtensionID(), error);
 }
 
diff --git a/extensions/common/mojom/permission_set_mojom_traits_unittest.cc b/extensions/common/mojom/permission_set_mojom_traits_unittest.cc
index 2ae2422..a75a2e4 100644
--- a/extensions/common/mojom/permission_set_mojom_traits_unittest.cc
+++ b/extensions/common/mojom/permission_set_mojom_traits_unittest.cc
@@ -47,8 +47,7 @@
     value.Append("tcp-connect:*.example.com:80");
     value.Append("udp-bind::8080");
     value.Append("udp-send-to::8888");
-    ASSERT_TRUE(
-        input->FromValue(&base::Value::AsListValue(value), nullptr, nullptr));
+    ASSERT_TRUE(input->FromValue(&value, nullptr, nullptr));
   }
 
   std::unique_ptr<APIPermission> output = nullptr;
@@ -67,8 +66,7 @@
     value.Append("tcp-connect:*.example.com:80");
     value.Append("udp-bind::8080");
     value.Append("udp-send-to::8888");
-    ASSERT_TRUE(permission->FromValue(&base::Value::AsListValue(value), nullptr,
-                                      nullptr));
+    ASSERT_TRUE(permission->FromValue(&value, nullptr, nullptr));
   }
 
   APIPermissionSet input;
@@ -135,8 +133,7 @@
     value.Append("tcp-connect:*.example.com:80");
     value.Append("udp-bind::8080");
     value.Append("udp-send-to::8888");
-    ASSERT_TRUE(permission->FromValue(&base::Value::AsListValue(value), nullptr,
-                                      nullptr));
+    ASSERT_TRUE(permission->FromValue(&value, nullptr, nullptr));
   }
 
   APIPermissionSet apis;
diff --git a/gpu/command_buffer/service/image_reader_gl_owner.cc b/gpu/command_buffer/service/image_reader_gl_owner.cc
index 4b6fb13..2497c096 100644
--- a/gpu/command_buffer/service/image_reader_gl_owner.cc
+++ b/gpu/command_buffer/service/image_reader_gl_owner.cc
@@ -226,14 +226,14 @@
   // If we've already lost the texture, then do nothing.
   if (!image_reader_) {
     DLOG(ERROR) << "Already lost texture / image reader";
-    return gl::ScopedJavaSurface::AcquireExternalSurface(nullptr);
+    return nullptr;
   }
 
   // Get the android native window from the image reader.
   ANativeWindow* window = nullptr;
   if (loader_->AImageReader_getWindow(image_reader_, &window) != AMEDIA_OK) {
     DLOG(ERROR) << "unable to get a window from image reader.";
-    return gl::ScopedJavaSurface::AcquireExternalSurface(nullptr);
+    return nullptr;
   }
 
   // Get the java surface object from the Android native window.
@@ -243,7 +243,7 @@
   DCHECK(j_surface);
 
   // Get the scoped java surface that will call release() on destruction.
-  return gl::ScopedJavaSurface(j_surface);
+  return gl::ScopedJavaSurface(j_surface, /*auto_release=*/true);
 }
 
 void ImageReaderGLOwner::UpdateTexImage() {
diff --git a/gpu/command_buffer/service/shared_image/compound_image_backing.cc b/gpu/command_buffer/service/shared_image/compound_image_backing.cc
index 79550ab9..9e94f8bb 100644
--- a/gpu/command_buffer/service/shared_image/compound_image_backing.cc
+++ b/gpu/command_buffer/service/shared_image/compound_image_backing.cc
@@ -309,39 +309,33 @@
     bool allow_shm_overlays,
     const Mailbox& mailbox,
     gfx::GpuMemoryBufferHandle handle,
-    gfx::BufferFormat buffer_format,
+    gfx::BufferFormat format,
     gfx::BufferPlane plane,
     const gfx::Size& size,
     const gfx::ColorSpace& color_space,
     GrSurfaceOrigin surface_origin,
     SkAlphaType alpha_type,
     uint32_t usage) {
-  DCHECK(IsValidSharedMemoryBufferFormat(size, buffer_format, plane));
-
-  const gfx::Size plane_size = GetPlaneSize(plane, size);
-  const viz::ResourceFormat plane_format =
-      viz::GetResourceFormat(GetPlaneBufferFormat(plane, buffer_format));
-
-  const size_t plane_index = GetPlaneIndex(plane, buffer_format);
-  handle.offset +=
-      gfx::BufferOffsetForBufferFormat(size, buffer_format, plane_index);
+  DCHECK(IsValidSharedMemoryBufferFormat(size, format, plane));
 
   SharedMemoryRegionWrapper shm_wrapper;
-  if (!shm_wrapper.Initialize(handle, plane_size, plane_format)) {
+  if (!shm_wrapper.Initialize(handle, size, format, plane)) {
     DLOG(ERROR) << "Failed to create SharedMemoryRegionWrapper";
     return nullptr;
   }
 
-  auto si_format = viz::SharedImageFormat::SinglePlane(plane_format);
+  const gfx::Size plane_size = GetPlaneSize(plane, size);
+  const auto plane_format = viz::SharedImageFormat::SinglePlane(
+      viz::GetResourceFormat(GetPlaneBufferFormat(plane, format)));
 
   auto shm_backing = std::make_unique<SharedMemoryImageBacking>(
-      mailbox, si_format, plane_size, color_space, surface_origin, alpha_type,
-      SHARED_IMAGE_USAGE_CPU_WRITE, std::move(shm_wrapper));
+      mailbox, plane_format, plane_size, color_space, surface_origin,
+      alpha_type, SHARED_IMAGE_USAGE_CPU_WRITE, std::move(shm_wrapper));
   shm_backing->SetNotRefCounted();
 
   return base::WrapUnique(new CompoundImageBacking(
-      mailbox, si_format, plane_size, color_space, surface_origin, alpha_type,
-      usage, allow_shm_overlays, std::move(shm_backing),
+      mailbox, plane_format, plane_size, color_space, surface_origin,
+      alpha_type, usage, allow_shm_overlays, std::move(shm_backing),
       gpu_backing_factory->GetWeakPtr()));
 }
 
diff --git a/gpu/command_buffer/service/shared_image/egl_image_backing.cc b/gpu/command_buffer/service/shared_image/egl_image_backing.cc
index 2fd05b2f..ca0e2bc 100644
--- a/gpu/command_buffer/service/shared_image/egl_image_backing.cc
+++ b/gpu/command_buffer/service/shared_image/egl_image_backing.cc
@@ -485,12 +485,6 @@
   return base::MakeRefCounted<TextureHolder>(std::move(texture));
 }
 
-void EGLImageBacking::SetEndReadFence(
-    scoped_refptr<gl::SharedGLFenceEGL> shared_egl_fence) {
-  AutoLock auto_lock(this);
-  read_fences_[gl::g_current_gl_context] = std::move(shared_egl_fence);
-}
-
 void EGLImageBacking::MarkForDestruction() {
   AutoLock auto_lock(this);
   DCHECK(!have_context() || created_on_context_ == gl::g_current_gl_context);
diff --git a/gpu/command_buffer/service/shared_image/egl_image_backing.h b/gpu/command_buffer/service/shared_image/egl_image_backing.h
index 3ddf2da9..86e8d60e 100644
--- a/gpu/command_buffer/service/shared_image/egl_image_backing.h
+++ b/gpu/command_buffer/service/shared_image/egl_image_backing.h
@@ -90,8 +90,6 @@
   scoped_refptr<TextureHolder> GenEGLImageSibling(
       base::span<const uint8_t> pixel_data);
 
-  void SetEndReadFence(scoped_refptr<gl::SharedGLFenceEGL> shared_egl_fence);
-
   const GLCommonImageBackingFactory::FormatInfo format_info_;
   scoped_refptr<TextureHolder> source_texture_holder_;
   raw_ptr<gl::GLApi> created_on_context_;
diff --git a/gpu/command_buffer/service/shared_image/gl_texture_image_backing.cc b/gpu/command_buffer/service/shared_image/gl_texture_image_backing.cc
index b8dff58..340359e 100644
--- a/gpu/command_buffer/service/shared_image/gl_texture_image_backing.cc
+++ b/gpu/command_buffer/service/shared_image/gl_texture_image_backing.cc
@@ -469,10 +469,8 @@
   SharedContextState* shared_context_state = factory()->GetSharedContextState();
   ui::ScopedMakeCurrent smc(shared_context_state->context(),
                             shared_context_state->surface());
-  auto image_np = base::MakeRefCounted<gl::GLImageNativePixmap>(
-      size(), ToBufferFormat(format()));
-  image_np->InitializeFromTexture(GetGLServiceId());
-  image_egl_ = image_np;
+  image_egl_ = gl::GLImageNativePixmap::CreateFromTexture(
+      size(), ToBufferFormat(format()), GetGLServiceId());
   if (passthrough_texture_) {
     passthrough_texture_->SetLevelImage(passthrough_texture_->target(), 0,
                                         image_egl_.get());
diff --git a/gpu/command_buffer/service/shared_image/shared_image_factory.cc b/gpu/command_buffer/service/shared_image/shared_image_factory.cc
index aa2b9eb02..0c5e3f8 100644
--- a/gpu/command_buffer/service/shared_image/shared_image_factory.cc
+++ b/gpu/command_buffer/service/shared_image/shared_image_factory.cc
@@ -449,10 +449,12 @@
 
   std::unique_ptr<SharedImageBacking> backing;
   if (use_compound) {
+    // Only allow shmem overlays for NV12 on Windows.
 #if BUILDFLAG(IS_WIN)
-    constexpr bool allow_shm_overlays = true;
+    const bool allow_shm_overlays =
+        format == gfx::BufferFormat::YUV_420_BIPLANAR;
 #else
-    constexpr bool allow_shm_overlays = false;
+    const bool allow_shm_overlays = false;
 #endif
     backing = CompoundImageBacking::CreateSharedMemory(
         factory, allow_shm_overlays, mailbox, std::move(handle), format, plane,
diff --git a/gpu/command_buffer/service/shared_image/shared_memory_image_backing_factory.cc b/gpu/command_buffer/service/shared_image/shared_memory_image_backing_factory.cc
index 1d88dd89..261fdc0 100644
--- a/gpu/command_buffer/service/shared_image/shared_memory_image_backing_factory.cc
+++ b/gpu/command_buffer/service/shared_image/shared_memory_image_backing_factory.cc
@@ -60,12 +60,11 @@
     SkAlphaType alpha_type,
     uint32_t usage) {
   DCHECK(handle.type == gfx::SHARED_MEMORY_BUFFER);
-  viz::ResourceFormat format = viz::GetResourceFormat(buffer_format);
   SharedMemoryRegionWrapper shm_wrapper;
-  if (!shm_wrapper.Initialize(handle, size, format)) {
+  if (!shm_wrapper.Initialize(handle, size, buffer_format, plane)) {
     return nullptr;
   }
-
+  const auto format = viz::GetResourceFormat(buffer_format);
   auto backing = std::make_unique<SharedMemoryImageBacking>(
       mailbox, viz::SharedImageFormat::SinglePlane(format), size, color_space,
       surface_origin, alpha_type, usage, std::move(shm_wrapper));
diff --git a/gpu/command_buffer/service/shared_memory_region_wrapper.cc b/gpu/command_buffer/service/shared_memory_region_wrapper.cc
index 11c93282..91fa8ce 100644
--- a/gpu/command_buffer/service/shared_memory_region_wrapper.cc
+++ b/gpu/command_buffer/service/shared_memory_region_wrapper.cc
@@ -7,8 +7,8 @@
 #include "base/logging.h"
 #include "base/numerics/checked_math.h"
 #include "base/system/sys_info.h"
-#include "components/viz/common/resources/resource_format_utils.h"
-#include "components/viz/common/resources/resource_sizes.h"
+#include "gpu/command_buffer/common/gpu_memory_buffer_support.h"
+#include "ui/gfx/buffer_format_util.h"
 #include "ui/gfx/gpu_memory_buffer.h"
 
 namespace gpu {
@@ -16,39 +16,25 @@
 
 // Validate that |stride| will work for pixels with |size| and |format|.
 bool ValidateStride(const gfx::Size size,
-                    viz::ResourceFormat format,
+                    gfx::BufferFormat format,
                     uint32_t stride) {
   if (!base::IsValueInRangeForNumericType<size_t>(stride))
     return false;
 
-  uint32_t min_width_in_bytes = 0;
-  if (!viz::ResourceSizes::MaybeWidthInBytes(size.width(), format,
-                                             &min_width_in_bytes)) {
+  // Use plane index 0 since we can't handle different plane strides anyway.
+  size_t alignment = gfx::RowByteAlignmentForBufferFormat(format, /*plane=*/0);
+  if (stride % alignment != 0)
+    return false;
+
+  size_t min_width_in_bytes = 0;
+  if (!gfx::RowSizeForBufferFormatChecked(size.width(), format, /*plane=*/0,
+                                          &min_width_in_bytes)) {
     return false;
   }
 
   if (stride < min_width_in_bytes)
     return false;
 
-  // Check that stride is a multiple of pixel byte size.
-  int bits_per_pixel = viz::BitsPerPixel(format);
-  switch (bits_per_pixel) {
-    case 64:
-    case 32:
-    case 16:
-      if (stride % (bits_per_pixel / 8) != 0)
-        return false;
-      break;
-    case 8:
-    case 4:
-      break;
-    default:
-      // YVU420, YUV_420_BIPLANAR, and YUVA_420_TRIPLANAR format aren't
-      // supported.
-      NOTREACHED();
-      return false;
-  }
-
   return true;
 }
 
@@ -64,7 +50,8 @@
 bool SharedMemoryRegionWrapper::Initialize(
     const gfx::GpuMemoryBufferHandle& handle,
     const gfx::Size& size,
-    viz::ResourceFormat format) {
+    gfx::BufferFormat format,
+    gfx::BufferPlane plane) {
   DCHECK(!mapping_.IsValid());
 
   if (!handle.region.IsValid()) {
@@ -77,30 +64,49 @@
     return false;
   }
 
-  // Minimize the amount of address space we use but make sure offset is a
-  // multiple of page size as required by MapAt().
-  size_t allocation_granularity = base::SysInfo::VMAllocationGranularity();
-  size_t memory_offset = handle.offset % allocation_granularity;
-  size_t map_offset =
-      allocation_granularity * (handle.offset / allocation_granularity);
-
-  base::CheckedNumeric<size_t> checked_size = handle.stride;
-  checked_size *= size.height();
-  checked_size += memory_offset;
-  if (!checked_size.IsValid()) {
+  size_t buffer_size;
+  if (!gfx::BufferSizeForBufferFormatChecked(size, format, &buffer_size)) {
     DLOG(ERROR) << "Invalid GMB size.";
     return false;
   }
 
-  mapping_ = handle.region.MapAt(static_cast<off_t>(map_offset),
-                                 checked_size.ValueOrDie());
+  // Minimize the amount of address space we use but make sure offset is a
+  // multiple of page size as required by MapAt().
+  // TODO(sunnyps): This doesn't seem to be a requirement of MapAt() anymore.
+  const size_t allocation_granularity =
+      base::SysInfo::VMAllocationGranularity();
+  const size_t memory_offset = handle.offset % allocation_granularity;
+  const size_t map_offset =
+      allocation_granularity * (handle.offset / allocation_granularity);
 
+  base::CheckedNumeric<size_t> checked_size = buffer_size;
+  checked_size += memory_offset;
+  if (!checked_size.IsValid()) {
+    DLOG(ERROR) << "Invalid shared memory region map size.";
+    return false;
+  }
+
+  const size_t map_size = checked_size.ValueOrDie();
+  mapping_ = handle.region.MapAt(static_cast<off_t>(map_offset), map_size);
   if (!mapping_.IsValid()) {
     DLOG(ERROR) << "Failed to map shared memory.";
     return false;
   }
 
-  offset_ = memory_offset;
+  // Add plane offset separately so that we map the entire buffer even if we're
+  // accessing an individual plane - this helps with shared memory overlays on
+  // Windows by allowing access via the Y plane shared image only.
+  const size_t plane_index = GetPlaneIndex(plane, format);
+  const size_t plane_offset =
+      gfx::BufferOffsetForBufferFormat(size, format, plane_index);
+#if DCHECK_IS_ON()
+  const gfx::Size plane_size = GetPlaneSize(plane, size);
+  const size_t plane_size_bytes =
+      gfx::PlaneSizeForBufferFormat(plane_size, format, plane_index);
+  DCHECK_LE(memory_offset + plane_offset + plane_size_bytes, map_size);
+#endif
+
+  offset_ = memory_offset + plane_offset;
   stride_ = handle.stride;
 
   return true;
diff --git a/gpu/command_buffer/service/shared_memory_region_wrapper.h b/gpu/command_buffer/service/shared_memory_region_wrapper.h
index 19ee638..d8ace07 100644
--- a/gpu/command_buffer/service/shared_memory_region_wrapper.h
+++ b/gpu/command_buffer/service/shared_memory_region_wrapper.h
@@ -8,7 +8,7 @@
 #include "base/containers/span.h"
 #include "base/memory/shared_memory_mapping.h"
 #include "base/unguessable_token.h"
-#include "components/viz/common/resources/resource_format.h"
+#include "ui/gfx/buffer_types.h"
 #include "ui/gfx/geometry/size.h"
 
 namespace gfx {
@@ -31,7 +31,8 @@
   // until destruction.
   bool Initialize(const gfx::GpuMemoryBufferHandle& handle,
                   const gfx::Size& size,
-                  viz::ResourceFormat format);
+                  gfx::BufferFormat format,
+                  gfx::BufferPlane plane);
 
   bool IsValid() const;
   uint8_t* GetMemory() const;
diff --git a/gpu/command_buffer/tests/gl_test_utils.cc b/gpu/command_buffer/tests/gl_test_utils.cc
index a3108c9c..686c482 100644
--- a/gpu/command_buffer/tests/gl_test_utils.cc
+++ b/gpu/command_buffer/tests/gl_test_utils.cc
@@ -484,9 +484,9 @@
   EXPECT_NE(0u, tex_service_id);
 
   // Create an EGLImage from the real texture id.
-  auto image = base::MakeRefCounted<gl::GLImageNativePixmap>(size, format);
-  bool result = image->InitializeFromTexture(tex_service_id);
-  DCHECK(result);
+  auto image =
+      gl::GLImageNativePixmap::CreateFromTexture(size, format, tex_service_id);
+  DCHECK(image);
 
   // The test will own the EGLImage no need to keep a reference on the GL
   // texture after returning from this function. This is covered by the
diff --git a/gpu/ipc/common/gpu_surface_lookup.h b/gpu/ipc/common/gpu_surface_lookup.h
index f7b3a4e..ac1cddc 100644
--- a/gpu/ipc/common/gpu_surface_lookup.h
+++ b/gpu/ipc/common/gpu_surface_lookup.h
@@ -5,14 +5,10 @@
 #ifndef GPU_IPC_COMMON_GPU_SURFACE_LOOKUP_H_
 #define GPU_IPC_COMMON_GPU_SURFACE_LOOKUP_H_
 
-#include "build/build_config.h"
 #include "gpu/gpu_export.h"
 #include "gpu/ipc/common/surface_handle.h"
 #include "ui/gfx/native_widget_types.h"
-
-#if BUILDFLAG(IS_ANDROID)
 #include "ui/gl/android/scoped_java_surface.h"
-#endif
 
 namespace gpu {
 
@@ -30,11 +26,9 @@
   static GpuSurfaceLookup* GetInstance();
   static void InitInstance(GpuSurfaceLookup* lookup);
 
-#if BUILDFLAG(IS_ANDROID)
   virtual gl::ScopedJavaSurface AcquireJavaSurface(
       int surface_id,
       bool* can_be_used_with_surface_control) = 0;
-#endif
 };
 
 }  // namespace gpu
diff --git a/gpu/ipc/common/gpu_surface_tracker.cc b/gpu/ipc/common/gpu_surface_tracker.cc
index 84921d2..0d28aec 100644
--- a/gpu/ipc/common/gpu_surface_tracker.cc
+++ b/gpu/ipc/common/gpu_surface_tracker.cc
@@ -6,31 +6,15 @@
 
 #include "base/check.h"
 #include "build/build_config.h"
-
-#if BUILDFLAG(IS_ANDROID)
-#include <android/native_window_jni.h>
 #include "ui/gl/android/scoped_java_surface.h"
-#endif  // BUILDFLAG(IS_ANDROID)
 
 namespace gpu {
 
-#if BUILDFLAG(IS_ANDROID)
 GpuSurfaceTracker::SurfaceRecord::SurfaceRecord(
-    gfx::AcceleratedWidget widget,
-    const base::android::JavaRef<jobject>& j_surface,
+    gl::ScopedJavaSurface surface,
     bool can_be_used_with_surface_control)
-    : widget(widget),
-      can_be_used_with_surface_control(can_be_used_with_surface_control) {
-  // TODO(liberato): It would be nice to assert |surface|, but we
-  // can't.  in_process_context_factory.cc (for tests) actually calls us without
-  // a Surface from java.  Presumably, nobody uses it.  crbug.com/712717 .
-  if (j_surface)
-    surface = gl::ScopedJavaSurface::AcquireExternalSurface(j_surface);
-}
-#else   // BUILDFLAG(IS_ANDROID)
-GpuSurfaceTracker::SurfaceRecord::SurfaceRecord(gfx::AcceleratedWidget widget)
-    : widget(widget) {}
-#endif  // !BUILDFLAG(IS_ANDROID)
+    : surface(std::move(surface)),
+      can_be_used_with_surface_control(can_be_used_with_surface_control) {}
 
 GpuSurfaceTracker::SurfaceRecord::SurfaceRecord(SurfaceRecord&&) = default;
 
@@ -66,7 +50,6 @@
   surface_map_.erase(surface_handle);
 }
 
-#if BUILDFLAG(IS_ANDROID)
 gl::ScopedJavaSurface GpuSurfaceTracker::AcquireJavaSurface(
     gpu::SurfaceHandle surface_handle,
     bool* can_be_used_with_surface_control) {
@@ -80,9 +63,8 @@
 
   *can_be_used_with_surface_control =
       it->second.can_be_used_with_surface_control;
-  return gl::ScopedJavaSurface::AcquireExternalSurface(j_surface.j_surface());
+  return j_surface.CopyRetainOwnership();
 }
-#endif
 
 std::size_t GpuSurfaceTracker::GetSurfaceCount() {
   base::AutoLock lock(surface_map_lock_);
diff --git a/gpu/ipc/common/gpu_surface_tracker.h b/gpu/ipc/common/gpu_surface_tracker.h
index 3b245fa..9281127 100644
--- a/gpu/ipc/common/gpu_surface_tracker.h
+++ b/gpu/ipc/common/gpu_surface_tracker.h
@@ -9,60 +9,41 @@
 
 #include <map>
 
+#include "base/android/scoped_java_ref.h"
 #include "base/memory/singleton.h"
 #include "base/synchronization/lock.h"
-#include "build/build_config.h"
 #include "gpu/gpu_export.h"
 #include "gpu/ipc/common/gpu_surface_lookup.h"
 #include "gpu/ipc/common/surface_handle.h"
-#include "ui/gfx/native_widget_types.h"
-
-#if BUILDFLAG(IS_ANDROID)
-#include "base/android/scoped_java_ref.h"
-#endif
+#include "ui/gl/android/scoped_java_surface.h"
 
 namespace gpu {
 
-// This class is used on Android and Mac, and is responsible for tracking native
+// This class is used on Android, and is responsible for tracking native
 // window surfaces exposed to the GPU process. Every surface gets registered to
 // this class, and gets a handle.  The handle can be passed to
 // CommandBufferProxyImpl::Create or to
 // GpuMemoryBufferManager::CreateGpuMemoryBuffer.
 // On Android, the handle is used in the GPU process to get a reference to the
-// ANativeWindow, using GpuSurfaceLookup (implemented by
-// ChildProcessSurfaceManager).  We require that an Android Surface is provided
-// with the ANativeWindow, so one must provide an explicit GpuSurfaceTracker::
-// SurfaceRecord when adding it.
-// On Mac, the handle just passes through the GPU process, and is sent back via
-// GpuCommandBufferMsg_SwapBuffersCompleted to reference the surface.
+// ScopedJavaSurface, using GpuSurfaceLookup (implemented by
+// ChildProcessSurfaceManager).
 // This class is thread safe.
 class GPU_EXPORT GpuSurfaceTracker : public gpu::GpuSurfaceLookup {
  public:
   struct SurfaceRecord {
-#if BUILDFLAG(IS_ANDROID)
-    SurfaceRecord(gfx::AcceleratedWidget widget,
-                  const base::android::JavaRef<jobject>& j_surface,
+    SurfaceRecord(gl::ScopedJavaSurface surface,
                   bool can_be_used_with_surface_control);
-#else   // BUILDFLAG(IS_ANDROID)
-    explicit SurfaceRecord(gfx::AcceleratedWidget widget);
-#endif  // !BUILDFLAG(IS_ANDROID)
 
     SurfaceRecord(SurfaceRecord&&);
     SurfaceRecord(const SurfaceRecord&) = delete;
 
-    // TODO(crbug.com/1399516): Make this non-android.
-    gfx::AcceleratedWidget widget;
-#if BUILDFLAG(IS_ANDROID)
     gl::ScopedJavaSurface surface;
     bool can_be_used_with_surface_control;
-#endif
   };
 
-#if BUILDFLAG(IS_ANDROID)
   gl::ScopedJavaSurface AcquireJavaSurface(
       gpu::SurfaceHandle surface_handle,
       bool* can_be_used_with_surface_control) override;
-#endif
 
   // Gets the global instance of the surface tracker.
   static GpuSurfaceTracker* Get() { return GetInstance(); }
diff --git a/gpu/ipc/service/gpu_memory_buffer_factory_native_pixmap.cc b/gpu/ipc/service/gpu_memory_buffer_factory_native_pixmap.cc
index 84dd4215..a581c3de 100644
--- a/gpu/ipc/service/gpu_memory_buffer_factory_native_pixmap.cc
+++ b/gpu/ipc/service/gpu_memory_buffer_factory_native_pixmap.cc
@@ -116,8 +116,8 @@
                << gfx::BufferUsageToString(usage);
     return nullptr;
   }
-  auto image = base::MakeRefCounted<gl::GLImageNativePixmap>(size, format);
-  if (!image->Initialize(std::move(pixmap))) {
+  auto image = gl::GLImageNativePixmap::Create(size, format, std::move(pixmap));
+  if (!image) {
     LOG(ERROR) << "Failed to create GLImage " << size.ToString() << ", "
                << gfx::BufferFormatToString(format) << ", usage "
                << gfx::BufferUsageToString(usage);
diff --git a/headless/BUILD.gn b/headless/BUILD.gn
index c227453..2225f2e 100644
--- a/headless/BUILD.gn
+++ b/headless/BUILD.gn
@@ -311,7 +311,6 @@
     "public/headless_browser.cc",
     "public/headless_browser.h",
     "public/headless_export.h",
-    "public/internal/headless_devtools_client_impl.h",
     "public/internal/message_dispatcher.h",
     "public/internal/value_conversions.h",
     "public/util/error_reporter.cc",
@@ -386,7 +385,6 @@
     "lib/browser/headless_devtools.h",
     "lib/browser/headless_devtools_agent_host_client.cc",
     "lib/browser/headless_devtools_agent_host_client.h",
-    "lib/browser/headless_devtools_client_impl.cc",
     "lib/browser/headless_devtools_manager_delegate.cc",
     "lib/browser/headless_devtools_manager_delegate.h",
     "lib/browser/headless_permission_manager.cc",
@@ -715,7 +713,6 @@
 
   deps = [
     ":headless_browsertests",
-    ":headless_example",
     ":headless_unittests",
   ]
 }
@@ -1062,22 +1059,3 @@
   ]
   output = "$target_gen_dir/public/version.h"
 }
-
-executable("headless_example") {
-  sources = [ "app/headless_example.cc" ]
-  defines = []
-
-  deps = [
-    ":headless_shell_lib",
-    "//content",
-    "//sandbox",
-    "//skia",  # we need this to override font render hinting in headless build
-    "//ui/gfx/geometry",
-  ]
-
-  if (is_win) {
-    deps += [ "//content/public/app" ]
-  }
-
-  configs += [ ":headless_defines_config" ]
-}
diff --git a/headless/app/headless_example.cc b/headless/app/headless_example.cc
deleted file mode 100644
index 3a9ea309..0000000
--- a/headless/app/headless_example.cc
+++ /dev/null
@@ -1,202 +0,0 @@
-// Copyright 2017 The Chromium Authors
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-// A small example application showing the use of the C++ Headless Chrome
-// library. It navigates to a web site given on the command line, waits for it
-// to load and prints out the DOM.
-//
-// Tip: start reading from the main() function below.
-
-#include "base/bind.h"
-#include "base/command_line.h"
-#include "base/memory/raw_ptr.h"
-#include "base/memory/weak_ptr.h"
-#include "build/build_config.h"
-#include "headless/public/devtools/domains/page.h"
-#include "headless/public/devtools/domains/runtime.h"
-#include "headless/public/headless_browser.h"
-#include "headless/public/headless_devtools_client.h"
-#include "headless/public/headless_devtools_target.h"
-#include "headless/public/headless_web_contents.h"
-#include "ui/gfx/geometry/size.h"
-
-#if BUILDFLAG(IS_WIN)
-#include "base/strings/utf_string_conversions.h"
-#include "content/public/app/sandbox_helper_win.h"
-#include "sandbox/win/src/sandbox_types.h"
-#endif
-
-// This class contains the main application logic, i.e., waiting for a page to
-// load and printing its DOM. Note that browser initialization happens outside
-// this class.
-class HeadlessExample : public headless::HeadlessWebContents::Observer,
-                        public headless::page::Observer {
- public:
-  HeadlessExample(headless::HeadlessBrowser* browser,
-                  headless::HeadlessWebContents* web_contents);
-  ~HeadlessExample() override;
-
-  // headless::HeadlessWebContents::Observer implementation:
-  void DevToolsTargetReady() override;
-
-  // headless::page::Observer implementation:
-  void OnLoadEventFired(
-      const headless::page::LoadEventFiredParams& params) override;
-
-  // Tip: Observe headless::inspector::ExperimentalObserver::OnTargetCrashed to
-  // be notified of renderer crashes.
-
-  void OnDomFetched(std::unique_ptr<headless::runtime::EvaluateResult> result);
-
- private:
-  // The headless browser instance. Owned by the headless library. See main().
-  raw_ptr<headless::HeadlessBrowser> browser_;
-  // Our tab. Owned by |browser_|.
-  raw_ptr<headless::HeadlessWebContents> web_contents_;
-  // The DevTools client used to control the tab.
-  std::unique_ptr<headless::HeadlessDevToolsClient> devtools_client_;
-  // A helper for creating weak pointers to this class.
-  base::WeakPtrFactory<HeadlessExample> weak_factory_{this};
-};
-
-namespace {
-HeadlessExample* g_example;
-}
-
-HeadlessExample::HeadlessExample(headless::HeadlessBrowser* browser,
-                                 headless::HeadlessWebContents* web_contents)
-    : browser_(browser),
-      web_contents_(web_contents),
-      devtools_client_(headless::HeadlessDevToolsClient::Create()) {
-  web_contents_->AddObserver(this);
-}
-
-HeadlessExample::~HeadlessExample() {
-  // Note that we shut down the browser last, because it owns objects such as
-  // the web contents which can no longer be accessed after the browser is gone.
-  devtools_client_->GetPage()->RemoveObserver(this);
-  web_contents_->GetDevToolsTarget()->DetachClient(devtools_client_.get());
-  web_contents_->RemoveObserver(this);
-  browser_->Shutdown();
-}
-
-// This method is called when the tab is ready for DevTools inspection.
-void HeadlessExample::DevToolsTargetReady() {
-  // Attach our DevTools client to the tab so that we can send commands to it
-  // and observe events.
-  web_contents_->GetDevToolsTarget()->AttachClient(devtools_client_.get());
-
-  // Start observing events from DevTools's page domain. This lets us get
-  // notified when the page has finished loading. Note that it is possible
-  // the page has already finished loading by now. See
-  // HeadlessShell::DevToolTargetReady for how to handle that case correctly.
-  devtools_client_->GetPage()->AddObserver(this);
-  devtools_client_->GetPage()->Enable();
-}
-
-void HeadlessExample::OnLoadEventFired(
-    const headless::page::LoadEventFiredParams& params) {
-  // The page has now finished loading. Let's grab a snapshot of the DOM by
-  // evaluating the innerHTML property on the document element.
-  devtools_client_->GetRuntime()->Evaluate(
-      "(document.doctype ? new "
-      "XMLSerializer().serializeToString(document.doctype) + '\\n' : '') + "
-      "document.documentElement.outerHTML",
-      base::BindOnce(&HeadlessExample::OnDomFetched,
-                     weak_factory_.GetWeakPtr()));
-}
-
-void HeadlessExample::OnDomFetched(
-    std::unique_ptr<headless::runtime::EvaluateResult> result) {
-  // Make sure the evaluation succeeded before reading the result.
-  if (result->HasExceptionDetails()) {
-    LOG(ERROR) << "Failed to serialize document: "
-               << result->GetExceptionDetails()->GetText();
-  } else {
-    printf("%s\n", result->GetResult()->GetValue()->GetString().c_str());
-  }
-
-  // Shut down the browser (see ~HeadlessExample).
-  delete g_example;
-  g_example = nullptr;
-}
-
-// This function is called by the headless library after the browser has been
-// initialized. It runs on the UI thread.
-void OnHeadlessBrowserStarted(headless::HeadlessBrowser* browser) {
-  // In order to open tabs, we first need a browser context. It corresponds to a
-  // user profile and contains things like the user's cookies, local storage,
-  // cache, etc.
-  headless::HeadlessBrowserContext::Builder context_builder =
-      browser->CreateBrowserContextBuilder();
-
-  // Here we can set options for the browser context. As an example we enable
-  // incognito mode, which makes sure profile data is not written to disk.
-  context_builder.SetIncognitoMode(true);
-
-  // Construct the context and set it as the default. The default browser
-  // context is used by the Target.createTarget() DevTools command when no other
-  // context is given.
-  headless::HeadlessBrowserContext* browser_context = context_builder.Build();
-  browser->SetDefaultBrowserContext(browser_context);
-
-  // Get the URL from the command line.
-  base::CommandLine::StringVector args =
-      base::CommandLine::ForCurrentProcess()->GetArgs();
-  if (args.empty()) {
-    LOG(ERROR) << "No URL to load";
-    browser->Shutdown();
-    return;
-  }
-#if BUILDFLAG(IS_WIN)
-  GURL url(base::WideToUTF16(args[0]));
-#else
-  GURL url(args[0]);
-#endif
-
-  // Open a tab (i.e., HeadlessWebContents) in the newly created browser
-  // context.
-  headless::HeadlessWebContents::Builder tab_builder(
-      browser_context->CreateWebContentsBuilder());
-
-  // We can set options for the opened tab here. In this example we are just
-  // setting the initial URL to navigate to.
-  tab_builder.SetInitialURL(url);
-
-  // Create an instance of the example app, which will wait for the page to load
-  // and print its DOM.
-  headless::HeadlessWebContents* web_contents = tab_builder.Build();
-  g_example = new HeadlessExample(browser, web_contents);
-}
-
-int main(int argc, const char** argv) {
-#if !BUILDFLAG(IS_WIN)
-  // This function must be the first thing we call to make sure child processes
-  // such as the renderer are started properly. The headless library starts
-  // child processes by forking and exec'ing the main application.
-  headless::RunChildProcessIfNeeded(argc, argv);
-#endif
-
-  // Create a headless browser instance. There can be one of these per process
-  // and it can only be initialized once.
-  headless::HeadlessBrowser::Options::Builder builder(argc, argv);
-
-#if BUILDFLAG(IS_WIN)
-  // In windows, you must initialize and set the sandbox, or pass it along
-  // if it has already been initialized.
-  sandbox::SandboxInterfaceInfo sandbox_info = {nullptr};
-  content::InitializeSandboxInfo(&sandbox_info);
-  builder.SetSandboxInfo(&sandbox_info);
-#endif
-  // Here you can customize browser options. As an example we set the window
-  // size.
-  builder.SetWindowSize(gfx::Size(800, 600));
-
-  // Pass control to the headless library. It will bring up the browser and
-  // invoke the given callback on the browser UI thread. Note: if you need to
-  // pass more parameters to the callback, you can add them to the Bind() call
-  // below.
-  return headless::HeadlessBrowserMain(
-      builder.Build(), base::BindOnce(&OnHeadlessBrowserStarted));
-}
diff --git a/headless/lib/browser/headless_devtools_client_impl.cc b/headless/lib/browser/headless_devtools_client_impl.cc
deleted file mode 100644
index b1902338a..0000000
--- a/headless/lib/browser/headless_devtools_client_impl.cc
+++ /dev/null
@@ -1,524 +0,0 @@
-// Copyright 2016 The Chromium Authors
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#include "headless/public/internal/headless_devtools_client_impl.h"
-
-#include <memory>
-
-#include "base/bind.h"
-#include "base/json/json_reader.h"
-#include "base/json/json_writer.h"
-#include "base/logging.h"
-#include "base/memory/ptr_util.h"
-#include "base/values.h"
-#include "content/public/browser/browser_task_traits.h"
-#include "content/public/browser/browser_thread.h"
-#include "headless/public/headless_devtools_target.h"
-
-namespace headless {
-
-namespace {
-int g_next_message_id = 0;
-int g_next_raw_message_id = 1;
-}  // namespace
-
-// static
-std::unique_ptr<HeadlessDevToolsClient> HeadlessDevToolsClient::Create() {
-  auto result = std::make_unique<HeadlessDevToolsClientImpl>();
-  result->InitBrowserMainThread();
-  return result;
-}
-
-// static
-std::unique_ptr<HeadlessDevToolsClient>
-HeadlessDevToolsClient::CreateWithExternalHost(ExternalHost* external_host) {
-  auto result = std::make_unique<HeadlessDevToolsClientImpl>();
-  result->AttachToExternalHost(external_host);
-  return result;
-}
-
-HeadlessDevToolsClientImpl::HeadlessDevToolsClientImpl()
-    : accessibility_domain_(this),
-      animation_domain_(this),
-      browser_domain_(this),
-      cache_storage_domain_(this),
-      console_domain_(this),
-      css_domain_(this),
-      database_domain_(this),
-      debugger_domain_(this),
-      device_orientation_domain_(this),
-      dom_domain_(this),
-      dom_debugger_domain_(this),
-      dom_snapshot_domain_(this),
-      dom_storage_domain_(this),
-      emulation_domain_(this),
-      fetch_domain_(this),
-      headless_experimental_domain_(this),
-      heap_profiler_domain_(this),
-      indexeddb_domain_(this),
-      input_domain_(this),
-      inspector_domain_(this),
-      io_domain_(this),
-      layer_tree_domain_(this),
-      log_domain_(this),
-      memory_domain_(this),
-      network_domain_(this),
-      page_domain_(this),
-      performance_domain_(this),
-      profiler_domain_(this),
-      runtime_domain_(this),
-      security_domain_(this),
-      service_worker_domain_(this),
-      target_domain_(this),
-      tracing_domain_(this) {}
-
-HeadlessDevToolsClientImpl::~HeadlessDevToolsClientImpl() {
-  if (parent_client_)
-    parent_client_->sessions_.erase(session_id_);
-}
-
-void HeadlessDevToolsClientImpl::AttachToExternalHost(
-    ExternalHost* external_host) {
-  DCHECK(!channel_ && !external_host_);
-  external_host_ = external_host;
-}
-
-void HeadlessDevToolsClientImpl::InitBrowserMainThread() {
-  browser_main_thread_ = content::GetUIThreadTaskRunner({});
-}
-
-void HeadlessDevToolsClientImpl::ChannelClosed() {
-  pending_messages_.clear();
-  channel_ = nullptr;
-}
-
-void HeadlessDevToolsClientImpl::AttachToChannel(
-    std::unique_ptr<HeadlessDevToolsChannel> channel) {
-  DCHECK(!channel_ && !external_host_);
-  channel_ = std::move(channel);
-  channel_->SetClient(this);
-}
-
-void HeadlessDevToolsClientImpl::DetachFromChannel() {
-  pending_messages_.clear();
-  channel_ = nullptr;
-}
-
-void HeadlessDevToolsClientImpl::SetRawProtocolListener(
-    RawProtocolListener* raw_protocol_listener) {
-  raw_protocol_listener_ = raw_protocol_listener;
-}
-
-std::unique_ptr<HeadlessDevToolsClient>
-HeadlessDevToolsClientImpl::CreateSession(const std::string& session_id) {
-  std::unique_ptr<HeadlessDevToolsClientImpl> client =
-      std::make_unique<HeadlessDevToolsClientImpl>();
-  client->parent_client_ = this;
-  client->session_id_ = session_id;
-  client->browser_main_thread_ = browser_main_thread_;
-  sessions_[session_id] = client.get();
-  return client;
-}
-
-int HeadlessDevToolsClientImpl::GetNextRawDevToolsMessageId() {
-  int id = g_next_raw_message_id;
-  g_next_raw_message_id += 2;
-  return id;
-}
-
-void HeadlessDevToolsClientImpl::SendRawDevToolsMessage(
-    const std::string& json_message) {
-  std::unique_ptr<base::Value> message =
-      base::JSONReader::ReadDeprecated(json_message);
-  if (!message || !message->is_dict()) {
-    LOG(ERROR) << "Malformed raw message";
-    return;
-  }
-  if (!session_id_.empty())
-    message->GetDict().Set("sessionId", session_id_);
-  SendProtocolMessage(message->GetDict());
-}
-
-void HeadlessDevToolsClientImpl::DispatchMessageFromExternalHost(
-    base::span<const uint8_t> json_message) {
-  DCHECK(external_host_);
-  ReceiveProtocolMessage(json_message);
-}
-
-void HeadlessDevToolsClientImpl::ReceiveProtocolMessage(
-    base::span<const uint8_t> json_message) {
-  base::StringPiece message_str(
-      reinterpret_cast<const char*>(json_message.data()), json_message.size());
-  // LOG(ERROR) << "[RECV] " << message_str;
-  std::unique_ptr<base::Value> message =
-      base::JSONReader::ReadDeprecated(message_str, base::JSON_PARSE_RFC);
-  if (!message || !message->is_dict()) {
-    NOTREACHED() << "Badly formed reply " << message_str;
-    return;
-  }
-  std::unique_ptr<base::DictionaryValue> message_dict =
-      base::DictionaryValue::From(std::move(message));
-
-  const std::string* session_id = message_dict->FindStringKey("sessionId");
-  if (session_id) {
-    auto it = sessions_.find(*session_id);
-    if (it != sessions_.end()) {
-      it->second->ReceiveProtocolMessage(json_message, std::move(message_dict));
-      return;
-    }
-  }
-  ReceiveProtocolMessage(json_message, std::move(message_dict));
-}
-
-void HeadlessDevToolsClientImpl::ReceiveProtocolMessage(
-    base::span<const uint8_t> json_message,
-    std::unique_ptr<base::DictionaryValue> message) {
-  base::StringPiece message_str(
-      reinterpret_cast<const char*>(json_message.data()), json_message.size());
-  const base::DictionaryValue* message_dict;
-  if (!message || !message->GetAsDictionary(&message_dict)) {
-    NOTREACHED() << "Badly formed reply " << message_str;
-    return;
-  }
-
-  if (raw_protocol_listener_ &&
-      raw_protocol_listener_->OnProtocolMessage(json_message, *message_dict)) {
-    return;
-  }
-
-  bool success = false;
-  if (message_dict->FindKey("id"))
-    success = DispatchMessageReply(std::move(message), *message_dict);
-  else
-    success = DispatchEvent(std::move(message), *message_dict);
-  if (!success)
-    DLOG(ERROR) << "Unhandled protocol message: " << message_str;
-}
-
-bool HeadlessDevToolsClientImpl::DispatchMessageReply(
-    std::unique_ptr<base::Value> owning_message,
-    const base::DictionaryValue& message_dict) {
-  const base::Value* id_value = message_dict.FindKey("id");
-  if (!id_value) {
-    NOTREACHED() << "ID must be specified.";
-    return false;
-  }
-  auto it = pending_messages_.find(id_value->GetInt());
-  if (it == pending_messages_.end()) {
-    NOTREACHED() << "Unexpected reply";
-    return false;
-  }
-  Callback callback = std::move(it->second);
-  pending_messages_.erase(it);
-  if (!callback.callback_with_result.is_null()) {
-    const base::DictionaryValue* result_dict;
-    if (message_dict.GetDictionary("result", &result_dict)) {
-      if (browser_main_thread_) {
-        browser_main_thread_->PostTask(
-            FROM_HERE,
-            base::BindOnce(
-                &HeadlessDevToolsClientImpl::DispatchMessageReplyWithResultTask,
-                weak_ptr_factory_.GetWeakPtr(), std::move(owning_message),
-                std::move(callback.callback_with_result), result_dict));
-      } else {
-        std::move(callback.callback_with_result).Run(*result_dict);
-      }
-    } else if (message_dict.GetDictionary("error", &result_dict)) {
-      auto null_value = std::make_unique<base::Value>();
-      base::Value* null_value_ptr = null_value.get();
-      DLOG(ERROR) << "Error in method call result: " << *result_dict;
-      if (browser_main_thread_) {
-        browser_main_thread_->PostTask(
-            FROM_HERE,
-            base::BindOnce(
-                &HeadlessDevToolsClientImpl::DispatchMessageReplyWithResultTask,
-                weak_ptr_factory_.GetWeakPtr(), std::move(null_value),
-                std::move(callback.callback_with_result), null_value_ptr));
-      } else {
-        std::move(callback.callback_with_result).Run(*null_value);
-      }
-    } else {
-      NOTREACHED() << "Reply has neither result nor error";
-      return false;
-    }
-  } else if (!callback.callback.is_null()) {
-    if (browser_main_thread_) {
-      browser_main_thread_->PostTask(
-          FROM_HERE,
-          base::BindOnce(
-              [](base::WeakPtr<HeadlessDevToolsClientImpl> self,
-                 base::OnceClosure callback) {
-                if (self)
-                  std::move(callback).Run();
-              },
-              weak_ptr_factory_.GetWeakPtr(), std::move(callback.callback)));
-    } else {
-      std::move(callback.callback).Run();
-    }
-  }
-  return true;
-}
-
-void HeadlessDevToolsClientImpl::DispatchMessageReplyWithResultTask(
-    std::unique_ptr<base::Value> owning_message,
-    base::OnceCallback<void(const base::Value&)> callback,
-    const base::Value* result_dict) {
-  std::move(callback).Run(*result_dict);
-}
-
-bool HeadlessDevToolsClientImpl::DispatchEvent(
-    std::unique_ptr<base::Value> owning_message,
-    const base::DictionaryValue& message_dict) {
-  const base::Value* method_value = message_dict.FindKey("method");
-  if (!method_value)
-    return false;
-  const std::string& method = method_value->GetString();
-  if (method == "Inspector.targetCrashed")
-    renderer_crashed_ = true;
-  EventHandlerMap::const_iterator it = event_handlers_.find(method);
-  if (it == event_handlers_.end()) {
-    if (method != "Inspector.targetCrashed")
-      NOTREACHED() << "Unknown event: " << method;
-    return false;
-  }
-  if (!it->second.is_null()) {
-    const base::DictionaryValue* result_dict;
-    if (!message_dict.GetDictionary("params", &result_dict)) {
-      NOTREACHED() << "Badly formed event parameters";
-      return false;
-    }
-    if (browser_main_thread_) {
-      // DevTools assumes event handling is async so we must post a task here or
-      // we risk breaking things.
-      browser_main_thread_->PostTask(
-          FROM_HERE,
-          base::BindOnce(&HeadlessDevToolsClientImpl::DispatchEventTask,
-                         weak_ptr_factory_.GetWeakPtr(),
-                         std::move(owning_message), &it->second, result_dict));
-    } else {
-      DispatchEventTask(std::move(owning_message), &it->second, result_dict);
-    }
-  }
-  return true;
-}
-
-void HeadlessDevToolsClientImpl::DispatchEventTask(
-    std::unique_ptr<base::Value> owning_message,
-    const EventHandler* event_handler,
-    const base::DictionaryValue* result_dict) {
-  event_handler->Run(*result_dict);
-}
-
-accessibility::Domain* HeadlessDevToolsClientImpl::GetAccessibility() {
-  return &accessibility_domain_;
-}
-
-animation::Domain* HeadlessDevToolsClientImpl::GetAnimation() {
-  return &animation_domain_;
-}
-
-browser::Domain* HeadlessDevToolsClientImpl::GetBrowser() {
-  return &browser_domain_;
-}
-
-cache_storage::Domain* HeadlessDevToolsClientImpl::GetCacheStorage() {
-  return &cache_storage_domain_;
-}
-
-console::Domain* HeadlessDevToolsClientImpl::GetConsole() {
-  return &console_domain_;
-}
-
-css::Domain* HeadlessDevToolsClientImpl::GetCSS() {
-  return &css_domain_;
-}
-
-database::Domain* HeadlessDevToolsClientImpl::GetDatabase() {
-  return &database_domain_;
-}
-
-debugger::Domain* HeadlessDevToolsClientImpl::GetDebugger() {
-  return &debugger_domain_;
-}
-
-device_orientation::Domain* HeadlessDevToolsClientImpl::GetDeviceOrientation() {
-  return &device_orientation_domain_;
-}
-
-dom::Domain* HeadlessDevToolsClientImpl::GetDOM() {
-  return &dom_domain_;
-}
-
-dom_debugger::Domain* HeadlessDevToolsClientImpl::GetDOMDebugger() {
-  return &dom_debugger_domain_;
-}
-
-dom_snapshot::Domain* HeadlessDevToolsClientImpl::GetDOMSnapshot() {
-  return &dom_snapshot_domain_;
-}
-
-dom_storage::Domain* HeadlessDevToolsClientImpl::GetDOMStorage() {
-  return &dom_storage_domain_;
-}
-
-emulation::Domain* HeadlessDevToolsClientImpl::GetEmulation() {
-  return &emulation_domain_;
-}
-
-fetch::Domain* HeadlessDevToolsClientImpl::GetFetch() {
-  return &fetch_domain_;
-}
-
-headless_experimental::Domain*
-HeadlessDevToolsClientImpl::GetHeadlessExperimental() {
-  return &headless_experimental_domain_;
-}
-
-heap_profiler::Domain* HeadlessDevToolsClientImpl::GetHeapProfiler() {
-  return &heap_profiler_domain_;
-}
-
-indexeddb::Domain* HeadlessDevToolsClientImpl::GetIndexedDB() {
-  return &indexeddb_domain_;
-}
-
-input::Domain* HeadlessDevToolsClientImpl::GetInput() {
-  return &input_domain_;
-}
-
-inspector::Domain* HeadlessDevToolsClientImpl::GetInspector() {
-  return &inspector_domain_;
-}
-
-io::Domain* HeadlessDevToolsClientImpl::GetIO() {
-  return &io_domain_;
-}
-
-layer_tree::Domain* HeadlessDevToolsClientImpl::GetLayerTree() {
-  return &layer_tree_domain_;
-}
-
-log::Domain* HeadlessDevToolsClientImpl::GetLog() {
-  return &log_domain_;
-}
-
-memory::Domain* HeadlessDevToolsClientImpl::GetMemory() {
-  return &memory_domain_;
-}
-
-network::Domain* HeadlessDevToolsClientImpl::GetNetwork() {
-  return &network_domain_;
-}
-
-page::Domain* HeadlessDevToolsClientImpl::GetPage() {
-  return &page_domain_;
-}
-
-performance::Domain* HeadlessDevToolsClientImpl::GetPerformance() {
-  return &performance_domain_;
-}
-
-profiler::Domain* HeadlessDevToolsClientImpl::GetProfiler() {
-  return &profiler_domain_;
-}
-
-runtime::Domain* HeadlessDevToolsClientImpl::GetRuntime() {
-  return &runtime_domain_;
-}
-
-security::Domain* HeadlessDevToolsClientImpl::GetSecurity() {
-  return &security_domain_;
-}
-
-service_worker::Domain* HeadlessDevToolsClientImpl::GetServiceWorker() {
-  return &service_worker_domain_;
-}
-
-target::Domain* HeadlessDevToolsClientImpl::GetTarget() {
-  return &target_domain_;
-}
-
-tracing::Domain* HeadlessDevToolsClientImpl::GetTracing() {
-  return &tracing_domain_;
-}
-
-template <typename CallbackType>
-void HeadlessDevToolsClientImpl::FinalizeAndSendMessage(
-    base::Value::Dict message,
-    CallbackType callback) {
-  if (renderer_crashed_)
-    return;
-  int id = g_next_message_id;
-  g_next_message_id += 2;  // We only send even numbered messages.
-  message.Set("id", id);
-  if (!session_id_.empty())
-    message.Set("sessionId", session_id_);
-  pending_messages_[id] = Callback(std::move(callback));
-  SendProtocolMessage(message);
-}
-
-void HeadlessDevToolsClientImpl::SendProtocolMessage(
-    const base::Value::Dict& message) {
-  if (parent_client_) {
-    parent_client_->SendProtocolMessage(message);
-    return;
-  }
-
-  std::string json_message;
-  base::JSONWriter::Write(message, &json_message);
-  // LOG(ERROR) << "[SEND] " << json_message;
-  auto bytes_message = base::as_bytes(base::make_span(json_message));
-  if (channel_)
-    channel_->SendProtocolMessage(bytes_message);
-  else
-    external_host_->SendProtocolMessage(bytes_message);
-}
-
-template <typename CallbackType>
-void HeadlessDevToolsClientImpl::SendMessageWithParams(const char* method,
-                                                       base::Value params,
-                                                       CallbackType callback) {
-  base::Value::Dict message;
-  message.Set("method", method);
-  message.Set("params", std::move(params));
-  FinalizeAndSendMessage(std::move(message), std::move(callback));
-}
-
-void HeadlessDevToolsClientImpl::SendMessage(
-    const char* method,
-    base::Value params,
-    base::OnceCallback<void(const base::Value&)> callback) {
-  SendMessageWithParams(method, std::move(params), std::move(callback));
-}
-
-void HeadlessDevToolsClientImpl::SendMessage(const char* method,
-                                             base::Value params,
-                                             base::OnceClosure callback) {
-  SendMessageWithParams(method, std::move(params), std::move(callback));
-}
-
-void HeadlessDevToolsClientImpl::RegisterEventHandler(
-    const char* method,
-    base::RepeatingCallback<void(const base::Value&)> callback) {
-  DCHECK(event_handlers_.find(method) == event_handlers_.end());
-  event_handlers_[method] = std::move(callback);
-}
-
-HeadlessDevToolsClientImpl::Callback::Callback() = default;
-
-HeadlessDevToolsClientImpl::Callback::Callback(Callback&& other) = default;
-
-HeadlessDevToolsClientImpl::Callback::Callback(base::OnceClosure callback)
-    : callback(std::move(callback)) {}
-
-HeadlessDevToolsClientImpl::Callback::Callback(
-    base::OnceCallback<void(const base::Value&)> callback)
-    : callback_with_result(std::move(callback)) {}
-
-HeadlessDevToolsClientImpl::Callback::~Callback() = default;
-
-HeadlessDevToolsClientImpl::Callback&
-HeadlessDevToolsClientImpl::Callback::operator=(Callback&& other) = default;
-
-}  // namespace headless
diff --git a/headless/public/internal/headless_devtools_client_impl.h b/headless/public/internal/headless_devtools_client_impl.h
deleted file mode 100644
index 56bfd36..0000000
--- a/headless/public/internal/headless_devtools_client_impl.h
+++ /dev/null
@@ -1,235 +0,0 @@
-// Copyright 2016 The Chromium Authors
-// Use of this source code is governed by a BSD-style license that can be
-// found in the LICENSE file.
-
-#ifndef HEADLESS_PUBLIC_INTERNAL_HEADLESS_DEVTOOLS_CLIENT_IMPL_H_
-#define HEADLESS_PUBLIC_INTERNAL_HEADLESS_DEVTOOLS_CLIENT_IMPL_H_
-
-#include <unordered_map>
-
-#include "base/containers/flat_map.h"
-#include "base/containers/span.h"
-#include "base/memory/raw_ptr.h"
-#include "base/memory/weak_ptr.h"
-#include "base/task/sequenced_task_runner.h"
-#include "headless/public/devtools/domains/accessibility.h"
-#include "headless/public/devtools/domains/animation.h"
-#include "headless/public/devtools/domains/browser.h"
-#include "headless/public/devtools/domains/cache_storage.h"
-#include "headless/public/devtools/domains/console.h"
-#include "headless/public/devtools/domains/css.h"
-#include "headless/public/devtools/domains/database.h"
-#include "headless/public/devtools/domains/debugger.h"
-#include "headless/public/devtools/domains/device_orientation.h"
-#include "headless/public/devtools/domains/dom.h"
-#include "headless/public/devtools/domains/dom_debugger.h"
-#include "headless/public/devtools/domains/dom_snapshot.h"
-#include "headless/public/devtools/domains/dom_storage.h"
-#include "headless/public/devtools/domains/emulation.h"
-#include "headless/public/devtools/domains/fetch.h"
-#include "headless/public/devtools/domains/headless_experimental.h"
-#include "headless/public/devtools/domains/heap_profiler.h"
-#include "headless/public/devtools/domains/indexeddb.h"
-#include "headless/public/devtools/domains/input.h"
-#include "headless/public/devtools/domains/inspector.h"
-#include "headless/public/devtools/domains/io.h"
-#include "headless/public/devtools/domains/layer_tree.h"
-#include "headless/public/devtools/domains/log.h"
-#include "headless/public/devtools/domains/memory.h"
-#include "headless/public/devtools/domains/network.h"
-#include "headless/public/devtools/domains/page.h"
-#include "headless/public/devtools/domains/performance.h"
-#include "headless/public/devtools/domains/profiler.h"
-#include "headless/public/devtools/domains/runtime.h"
-#include "headless/public/devtools/domains/security.h"
-#include "headless/public/devtools/domains/service_worker.h"
-#include "headless/public/devtools/domains/target.h"
-#include "headless/public/devtools/domains/tracing.h"
-#include "headless/public/headless_devtools_client.h"
-#include "headless/public/headless_export.h"
-#include "headless/public/internal/message_dispatcher.h"
-
-namespace base {
-class DictionaryValue;
-}
-
-namespace headless {
-
-class HEADLESS_EXPORT HeadlessDevToolsClientImpl
-    : public HeadlessDevToolsClient,
-      public HeadlessDevToolsChannel::Client,
-      public internal::MessageDispatcher {
- public:
-  HeadlessDevToolsClientImpl();
-
-  HeadlessDevToolsClientImpl(const HeadlessDevToolsClientImpl&) = delete;
-  HeadlessDevToolsClientImpl& operator=(const HeadlessDevToolsClientImpl&) =
-      delete;
-
-  ~HeadlessDevToolsClientImpl() override;
-
-  // HeadlessDevToolsClient implementation:
-  accessibility::Domain* GetAccessibility() override;
-  animation::Domain* GetAnimation() override;
-  browser::Domain* GetBrowser() override;
-  cache_storage::Domain* GetCacheStorage() override;
-  console::Domain* GetConsole() override;
-  css::Domain* GetCSS() override;
-  database::Domain* GetDatabase() override;
-  debugger::Domain* GetDebugger() override;
-  device_orientation::Domain* GetDeviceOrientation() override;
-  dom::Domain* GetDOM() override;
-  dom_debugger::Domain* GetDOMDebugger() override;
-  dom_snapshot::Domain* GetDOMSnapshot() override;
-  dom_storage::Domain* GetDOMStorage() override;
-  emulation::Domain* GetEmulation() override;
-  fetch::Domain* GetFetch() override;
-  headless_experimental::Domain* GetHeadlessExperimental() override;
-  heap_profiler::Domain* GetHeapProfiler() override;
-  indexeddb::Domain* GetIndexedDB() override;
-  input::Domain* GetInput() override;
-  inspector::Domain* GetInspector() override;
-  io::Domain* GetIO() override;
-  layer_tree::Domain* GetLayerTree() override;
-  log::Domain* GetLog() override;
-  memory::Domain* GetMemory() override;
-  network::Domain* GetNetwork() override;
-  page::Domain* GetPage() override;
-  performance::Domain* GetPerformance() override;
-  profiler::Domain* GetProfiler() override;
-  runtime::Domain* GetRuntime() override;
-  security::Domain* GetSecurity() override;
-  service_worker::Domain* GetServiceWorker() override;
-  target::Domain* GetTarget() override;
-  tracing::Domain* GetTracing() override;
-  void SetRawProtocolListener(
-      RawProtocolListener* raw_protocol_listener) override;
-  std::unique_ptr<HeadlessDevToolsClient> CreateSession(
-      const std::string& session_id) override;
-  int GetNextRawDevToolsMessageId() override;
-  void SendRawDevToolsMessage(const std::string& json_message) override;
-  void DispatchMessageFromExternalHost(
-      base::span<const uint8_t> json_message) override;
-  void AttachToChannel(
-      std::unique_ptr<HeadlessDevToolsChannel> channel) override;
-  void DetachFromChannel() override;
-
-  // HeadlessDevToolsChannel::Client implementation.
-  void ReceiveProtocolMessage(base::span<const uint8_t> message) override;
-  void ChannelClosed() override;
-
-  // internal::MessageDispatcher implementation:
-  void SendMessage(
-      const char* method,
-      base::Value params,
-      base::OnceCallback<void(const base::Value&)> callback) override;
-  void SendMessage(const char* method,
-                   base::Value params,
-                   base::OnceClosure callback) override;
-  void RegisterEventHandler(
-      const char* method,
-      base::RepeatingCallback<void(const base::Value&)> callback) override;
-
-  // TODO(dgozman): remove with ExternalHost.
-  void AttachToExternalHost(ExternalHost* external_host);
-  void InitBrowserMainThread();
-
-  void SetTaskRunnerForTests(
-      scoped_refptr<base::SequencedTaskRunner> task_runner) {
-    browser_main_thread_ = task_runner;
-  }
-
- private:
-  // Contains a callback with or without a result parameter depending on the
-  // message type.
-  struct Callback {
-    Callback();
-    Callback(Callback&& other);
-    explicit Callback(base::OnceClosure callback);
-    explicit Callback(base::OnceCallback<void(const base::Value&)> callback);
-    ~Callback();
-
-    Callback& operator=(Callback&& other);
-
-    base::OnceClosure callback;
-    base::OnceCallback<void(const base::Value&)> callback_with_result;
-  };
-
-  template <typename CallbackType>
-  void FinalizeAndSendMessage(base::Value::Dict message, CallbackType callback);
-
-  template <typename CallbackType>
-  void SendMessageWithParams(const char* method,
-                             base::Value params,
-                             CallbackType callback);
-
-  bool DispatchMessageReply(std::unique_ptr<base::Value> owning_message,
-                            const base::DictionaryValue& message_dict);
-  void DispatchMessageReplyWithResultTask(
-      std::unique_ptr<base::Value> owning_message,
-      base::OnceCallback<void(const base::Value&)> callback,
-      const base::Value* result_dict);
-  using EventHandler = base::RepeatingCallback<void(const base::Value&)>;
-  using EventHandlerMap = std::unordered_map<std::string, EventHandler>;
-
-  bool DispatchEvent(std::unique_ptr<base::Value> owning_message,
-                     const base::DictionaryValue& message_dict);
-  void DispatchEventTask(std::unique_ptr<base::Value> owning_message,
-                         const EventHandler* event_handler,
-                         const base::DictionaryValue* result_dict);
-
-  void ReceiveProtocolMessage(base::span<const uint8_t> json_message,
-                              std::unique_ptr<base::DictionaryValue> message);
-  void SendProtocolMessage(const base::Value::Dict& message);
-
-  std::unique_ptr<HeadlessDevToolsChannel> channel_;
-  raw_ptr<ExternalHost> external_host_ = nullptr;
-  raw_ptr<RawProtocolListener> raw_protocol_listener_ = nullptr;
-
-  std::unordered_map<int, Callback> pending_messages_;
-  EventHandlerMap event_handlers_;
-  std::string session_id_;
-  raw_ptr<HeadlessDevToolsClientImpl> parent_client_ = nullptr;
-  base::flat_map<std::string, HeadlessDevToolsClientImpl*> sessions_;
-  bool renderer_crashed_ = false;
-
-  accessibility::ExperimentalDomain accessibility_domain_;
-  animation::ExperimentalDomain animation_domain_;
-  browser::ExperimentalDomain browser_domain_;
-  cache_storage::ExperimentalDomain cache_storage_domain_;
-  console::ExperimentalDomain console_domain_;
-  css::ExperimentalDomain css_domain_;
-  database::ExperimentalDomain database_domain_;
-  debugger::ExperimentalDomain debugger_domain_;
-  device_orientation::ExperimentalDomain device_orientation_domain_;
-  dom::ExperimentalDomain dom_domain_;
-  dom_debugger::ExperimentalDomain dom_debugger_domain_;
-  dom_snapshot::ExperimentalDomain dom_snapshot_domain_;
-  dom_storage::ExperimentalDomain dom_storage_domain_;
-  emulation::ExperimentalDomain emulation_domain_;
-  fetch::ExperimentalDomain fetch_domain_;
-  headless_experimental::ExperimentalDomain headless_experimental_domain_;
-  heap_profiler::ExperimentalDomain heap_profiler_domain_;
-  indexeddb::ExperimentalDomain indexeddb_domain_;
-  input::ExperimentalDomain input_domain_;
-  inspector::ExperimentalDomain inspector_domain_;
-  io::ExperimentalDomain io_domain_;
-  layer_tree::ExperimentalDomain layer_tree_domain_;
-  log::ExperimentalDomain log_domain_;
-  memory::ExperimentalDomain memory_domain_;
-  network::ExperimentalDomain network_domain_;
-  page::ExperimentalDomain page_domain_;
-  performance::ExperimentalDomain performance_domain_;
-  profiler::ExperimentalDomain profiler_domain_;
-  runtime::ExperimentalDomain runtime_domain_;
-  security::ExperimentalDomain security_domain_;
-  service_worker::ExperimentalDomain service_worker_domain_;
-  target::ExperimentalDomain target_domain_;
-  tracing::ExperimentalDomain tracing_domain_;
-  scoped_refptr<base::SequencedTaskRunner> browser_main_thread_;
-  base::WeakPtrFactory<HeadlessDevToolsClientImpl> weak_ptr_factory_{this};
-};
-
-}  // namespace headless
-
-#endif  // HEADLESS_PUBLIC_INTERNAL_HEADLESS_DEVTOOLS_CLIENT_IMPL_H_
diff --git a/infra/config/generated/builders/reclient/Mac Builder reclient test untrusted/properties.json b/infra/config/generated/builders/reclient/Mac Builder reclient test untrusted/properties.json
index 99652c55..67b06f7 100644
--- a/infra/config/generated/builders/reclient/Mac Builder reclient test untrusted/properties.json
+++ b/infra/config/generated/builders/reclient/Mac Builder reclient test untrusted/properties.json
@@ -46,14 +46,15 @@
   "$build/reclient": {
     "bootstrap_env": {
       "GLOG_vmodule": "bridge*=2",
-      "GOMA_COMPILER_PROXY_ENABLE_CRASH_DUMP": "false",
+      "GOMA_COMPILER_PROXY_ENABLE_CRASH_DUMP": "true",
       "RBE_deps_cache_mode": "reproxy",
       "RBE_experimental_goma_deps_cache": "True",
       "RBE_ip_reset_min_delay": "-1s"
     },
     "instance": "rbe-chromium-untrusted-test",
     "metrics_project": "chromium-reclient-metrics",
-    "profiler_service": "reclient-mac"
+    "profiler_service": "reclient-mac",
+    "scandeps_server": true
   },
   "$recipe_engine/resultdb/test_presentation": {
     "column_keys": [],
diff --git a/infra/config/generated/builders/reclient/Mac Builder reclient test/properties.json b/infra/config/generated/builders/reclient/Mac Builder reclient test/properties.json
index 3bb4e9d..40dc381 100644
--- a/infra/config/generated/builders/reclient/Mac Builder reclient test/properties.json
+++ b/infra/config/generated/builders/reclient/Mac Builder reclient test/properties.json
@@ -46,14 +46,15 @@
   "$build/reclient": {
     "bootstrap_env": {
       "GLOG_vmodule": "bridge*=2",
-      "GOMA_COMPILER_PROXY_ENABLE_CRASH_DUMP": "false",
+      "GOMA_COMPILER_PROXY_ENABLE_CRASH_DUMP": "true",
       "RBE_deps_cache_mode": "reproxy",
       "RBE_experimental_goma_deps_cache": "True",
       "RBE_ip_reset_min_delay": "-1s"
     },
     "instance": "rbe-chromium-trusted-test",
     "metrics_project": "chromium-reclient-metrics",
-    "profiler_service": "reclient-mac"
+    "profiler_service": "reclient-mac",
+    "scandeps_server": true
   },
   "$recipe_engine/resultdb/test_presentation": {
     "column_keys": [],
diff --git a/infra/config/generated/builders/reclient/ios-simulator reclient test untrusted/properties.json b/infra/config/generated/builders/reclient/ios-simulator reclient test untrusted/properties.json
index 905b8a1f..f79200e5 100644
--- a/infra/config/generated/builders/reclient/ios-simulator reclient test untrusted/properties.json
+++ b/infra/config/generated/builders/reclient/ios-simulator reclient test untrusted/properties.json
@@ -46,13 +46,14 @@
   "$build/reclient": {
     "bootstrap_env": {
       "GLOG_vmodule": "bridge*=2",
-      "GOMA_COMPILER_PROXY_ENABLE_CRASH_DUMP": "false",
+      "GOMA_COMPILER_PROXY_ENABLE_CRASH_DUMP": "true",
       "RBE_deps_cache_mode": "reproxy",
       "RBE_experimental_goma_deps_cache": "True",
       "RBE_ip_reset_min_delay": "-1s"
     },
     "instance": "rbe-chromium-untrusted-test",
-    "metrics_project": "chromium-reclient-metrics"
+    "metrics_project": "chromium-reclient-metrics",
+    "scandeps_server": true
   },
   "$recipe_engine/resultdb/test_presentation": {
     "column_keys": [],
diff --git a/infra/config/generated/builders/reclient/ios-simulator reclient test/properties.json b/infra/config/generated/builders/reclient/ios-simulator reclient test/properties.json
index 1ec5bf8..6fccaa3b 100644
--- a/infra/config/generated/builders/reclient/ios-simulator reclient test/properties.json
+++ b/infra/config/generated/builders/reclient/ios-simulator reclient test/properties.json
@@ -46,13 +46,14 @@
   "$build/reclient": {
     "bootstrap_env": {
       "GLOG_vmodule": "bridge*=2",
-      "GOMA_COMPILER_PROXY_ENABLE_CRASH_DUMP": "false",
+      "GOMA_COMPILER_PROXY_ENABLE_CRASH_DUMP": "true",
       "RBE_deps_cache_mode": "reproxy",
       "RBE_experimental_goma_deps_cache": "True",
       "RBE_ip_reset_min_delay": "-1s"
     },
     "instance": "rbe-chromium-trusted-test",
-    "metrics_project": "chromium-reclient-metrics"
+    "metrics_project": "chromium-reclient-metrics",
+    "scandeps_server": true
   },
   "$recipe_engine/resultdb/test_presentation": {
     "column_keys": [],
diff --git a/infra/config/generated/builders/reclient/mac-arm64-rel reclient test untrusted/properties.json b/infra/config/generated/builders/reclient/mac-arm64-rel reclient test untrusted/properties.json
index 6cc4d63..6a3f71d 100644
--- a/infra/config/generated/builders/reclient/mac-arm64-rel reclient test untrusted/properties.json
+++ b/infra/config/generated/builders/reclient/mac-arm64-rel reclient test untrusted/properties.json
@@ -45,13 +45,14 @@
   "$build/reclient": {
     "bootstrap_env": {
       "GLOG_vmodule": "bridge*=2",
-      "GOMA_COMPILER_PROXY_ENABLE_CRASH_DUMP": "false",
+      "GOMA_COMPILER_PROXY_ENABLE_CRASH_DUMP": "true",
       "RBE_deps_cache_mode": "reproxy",
       "RBE_experimental_goma_deps_cache": "True",
       "RBE_ip_reset_min_delay": "-1s"
     },
     "instance": "rbe-chromium-untrusted-test",
-    "metrics_project": "chromium-reclient-metrics"
+    "metrics_project": "chromium-reclient-metrics",
+    "scandeps_server": true
   },
   "$recipe_engine/resultdb/test_presentation": {
     "column_keys": [],
diff --git a/infra/config/generated/builders/reclient/mac-arm64-rel reclient test/properties.json b/infra/config/generated/builders/reclient/mac-arm64-rel reclient test/properties.json
index 111024b..4ca7f05c 100644
--- a/infra/config/generated/builders/reclient/mac-arm64-rel reclient test/properties.json
+++ b/infra/config/generated/builders/reclient/mac-arm64-rel reclient test/properties.json
@@ -45,13 +45,14 @@
   "$build/reclient": {
     "bootstrap_env": {
       "GLOG_vmodule": "bridge*=2",
-      "GOMA_COMPILER_PROXY_ENABLE_CRASH_DUMP": "false",
+      "GOMA_COMPILER_PROXY_ENABLE_CRASH_DUMP": "true",
       "RBE_deps_cache_mode": "reproxy",
       "RBE_experimental_goma_deps_cache": "True",
       "RBE_ip_reset_min_delay": "-1s"
     },
     "instance": "rbe-chromium-trusted-test",
-    "metrics_project": "chromium-reclient-metrics"
+    "metrics_project": "chromium-reclient-metrics",
+    "scandeps_server": true
   },
   "$recipe_engine/resultdb/test_presentation": {
     "column_keys": [],
diff --git a/infra/config/lib/builders.star b/infra/config/lib/builders.star
index cfdc647..52b61ed 100644
--- a/infra/config/lib/builders.star
+++ b/infra/config/lib/builders.star
@@ -289,7 +289,7 @@
 
 _VALID_REPROXY_ENV_PREFIX_LIST = ["RBE_", "GLOG_", "GOMA_"]
 
-def _reclient_property(*, instance, service, jobs, rewrapper_env, profiler_service, publish_trace, cache_silo, ensure_verified, bootstrap_env):
+def _reclient_property(*, instance, service, jobs, rewrapper_env, profiler_service, publish_trace, cache_silo, ensure_verified, bootstrap_env, scandeps_server):
     reclient = {}
     instance = defaults.get_value("reclient_instance", instance)
     if not instance:
@@ -317,6 +317,9 @@
                      ", ".join(_VALID_REPROXY_ENV_PREFIX_LIST) +
                      "), got '%s'" % k)
         reclient["bootstrap_env"] = bootstrap_env
+    scandeps_server = defaults.get_value("reclient_scandeps_server", scandeps_server)
+    if scandeps_server:
+        reclient["scandeps_server"] = scandeps_server
     profiler_service = defaults.get_value("reclient_profiler_service", profiler_service)
     if profiler_service:
         reclient["profiler_service"] = profiler_service
@@ -400,6 +403,7 @@
     reclient_bootstrap_env = None,
     reclient_profiler_service = None,
     reclient_publish_trace = None,
+    reclient_scandeps_server = False,
     reclient_cache_silo = None,
     reclient_ensure_verified = None,
 
@@ -465,6 +469,7 @@
         reclient_bootstrap_env = args.DEFAULT,
         reclient_profiler_service = args.DEFAULT,
         reclient_publish_trace = args.DEFAULT,
+        reclient_scandeps_server = args.DEFAULT,
         reclient_cache_silo = None,
         reclient_ensure_verified = None,
         omit_python2 = args.DEFAULT,
@@ -642,6 +647,7 @@
             not set.
         reclient_publish_trace: If True, it publish trace by rpl2cloudtrace. Has
             no effect if reclient_instance is not set.
+        reclient_scandeps_server: If true, reproxy should start its own scandeps_server
         reclient_cache_silo: A string indicating a cache siling key to use for
             remote caching. Has no effect if reclient_instance is not set.
         reclient_ensure_verified: If True, it verifies build artifacts. Has no
@@ -797,6 +803,7 @@
         bootstrap_env = reclient_bootstrap_env,
         profiler_service = reclient_profiler_service,
         publish_trace = reclient_publish_trace,
+        scandeps_server = reclient_scandeps_server,
         cache_silo = reclient_cache_silo,
         ensure_verified = reclient_ensure_verified,
     )
diff --git a/infra/config/subprojects/reclient/reclient.star b/infra/config/subprojects/reclient/reclient.star
index 5a95839..546abd8 100644
--- a/infra/config/subprojects/reclient/reclient.star
+++ b/infra/config/subprojects/reclient/reclient.star
@@ -60,6 +60,7 @@
             "chromium-cq-staging-builder@chops-service-accounts.iam.gserviceaccount.com"
         ),
         reclient_version = "staging",
+        enable_crash_dump = "false",
         **kwargs):
     trusted_instance = reclient_instance % "trusted"
     unstrusted_instance = reclient_instance % "untrusted"
@@ -71,7 +72,7 @@
         "RBE_ip_reset_min_delay": "-1s",
         "RBE_deps_cache_mode": "reproxy",
         # TODO(b/258210757) remove once long term breakpad plans are dertermined
-        "GOMA_COMPILER_PROXY_ENABLE_CRASH_DUMP": "false",
+        "GOMA_COMPILER_PROXY_ENABLE_CRASH_DUMP": enable_crash_dump,
     })
     return [
         ci.builder(
@@ -193,6 +194,7 @@
             build_gs_bucket = "chromium-fyi-archive",
         ),
     ),
+    enable_crash_dump = "true",
     builderless = True,
     cores = None,
     os = os.MAC_DEFAULT,
@@ -202,6 +204,7 @@
         "GLOG_vmodule": "bridge*=2",
     },
     reclient_profiler_service = "reclient-mac",
+    reclient_scandeps_server = True,
 )
 
 fyi_reclient_staging_builder(
@@ -308,12 +311,14 @@
     builderless = True,
     cores = None,
     os = os.MAC_DEFAULT,
+    enable_crash_dump = "true",
     xcode = xcode.x14main,
     console_view_category = "ios",
     priority = 35,
     reclient_bootstrap_env = {
         "GLOG_vmodule": "bridge*=2",
     },
+    reclient_scandeps_server = True,
 )
 
 fyi_reclient_staging_builder(
@@ -386,10 +391,12 @@
     cores = None,
     os = os.MAC_DEFAULT,
     console_view_category = "mac",
+    enable_crash_dump = "true",
     priority = 35,
     reclient_bootstrap_env = {
         "GLOG_vmodule": "bridge*=2",
     },
+    reclient_scandeps_server = True,
 )
 
 ci.builder(
diff --git a/ios/chrome/app/BUILD.gn b/ios/chrome/app/BUILD.gn
index f430897..25f9cfe 100644
--- a/ios/chrome/app/BUILD.gn
+++ b/ios/chrome/app/BUILD.gn
@@ -498,6 +498,7 @@
     "//ios/chrome/browser/screenshot",
     "//ios/chrome/browser/search_engines",
     "//ios/chrome/browser/search_engines:extension_search_engine_data_updater",
+    "//ios/chrome/browser/search_engines:search_engines_util",
     "//ios/chrome/browser/sessions:scene_util",
     "//ios/chrome/browser/sessions:session_service",
     "//ios/chrome/browser/share_extension",
diff --git a/ios/chrome/browser/providers/push_notification/chromium_push_notification.mm b/ios/chrome/browser/providers/push_notification/chromium_push_notification.mm
index 6428d39..86299fa3 100644
--- a/ios/chrome/browser/providers/push_notification/chromium_push_notification.mm
+++ b/ios/chrome/browser/providers/push_notification/chromium_push_notification.mm
@@ -24,6 +24,12 @@
   void RegisterDevice(PushNotificationConfiguration* config,
                       void (^completion_handler)(NSError* error)) final;
   void UnregisterDevice(void (^completion_handler)(NSError* error)) final;
+  bool DeviceTokenIsSet() const final;
+
+ protected:
+  // PushNotificationService implementation.
+  void SetAccountsToDevice(NSArray<NSString*>* account_ids,
+                           CompletionHandler completion_handler) final;
 };
 
 void ChromiumPushNotificationService::RegisterDevice(
@@ -58,6 +64,27 @@
       }));
 }
 
+bool ChromiumPushNotificationService::DeviceTokenIsSet() const {
+  return false;
+}
+
+void ChromiumPushNotificationService::SetAccountsToDevice(
+    NSArray<NSString*>* account_ids,
+    void (^completion_handler)(NSError* error)) {
+  // Chromium does not initialize the device's connection to the push
+  // notification server. As a result, the `completion_handler` is called with
+  // a NSFeatureUnsupportedError.
+
+  base::SequencedTaskRunnerHandle::Get()->PostTask(
+      FROM_HERE, base::BindOnce(^() {
+        NSError* error =
+            [NSError errorWithDomain:kChromiumPushNotificationErrorDomain
+                                code:NSFeatureUnsupportedError
+                            userInfo:nil];
+        completion_handler(error);
+      }));
+}
+
 }  // namespace
 
 std::unique_ptr<PushNotificationService> CreatePushNotificationService() {
diff --git a/ios/chrome/browser/push_notification/BUILD.gn b/ios/chrome/browser/push_notification/BUILD.gn
index bc7b2aa8..20637ac 100644
--- a/ios/chrome/browser/push_notification/BUILD.gn
+++ b/ios/chrome/browser/push_notification/BUILD.gn
@@ -5,6 +5,8 @@
 source_set("push_notification_service") {
   configs += [ "//build/config/compiler:enable_arc" ]
   sources = [
+    "push_notification_account_context_manager.h",
+    "push_notification_account_context_manager.mm",
     "push_notification_client_manager.h",
     "push_notification_client_manager.mm",
     "push_notification_configuration.h",
@@ -37,3 +39,29 @@
   ]
   deps = [ "//base" ]
 }
+
+source_set("unit_tests") {
+  configs += [ "//build/config/compiler:enable_arc" ]
+  testonly = true
+  sources = [ "push_notification_client_manager_unittest.mm" ]
+  deps = [
+    ":push_notification_client",
+    ":push_notification_service",
+    ":test_support",
+    "//testing/gtest",
+  ]
+}
+
+source_set("test_support") {
+  configs += [ "//build/config/compiler:enable_arc" ]
+  sources = [
+    "test_push_notification_client.h",
+    "test_push_notification_client.mm",
+  ]
+
+  deps = [
+    ":push_notification_client",
+    ":push_notification_service",
+    "//base",
+  ]
+}
diff --git a/ios/chrome/browser/push_notification/push_notification_account_context_manager.h b/ios/chrome/browser/push_notification/push_notification_account_context_manager.h
new file mode 100644
index 0000000..397386e
--- /dev/null
+++ b/ios/chrome/browser/push_notification/push_notification_account_context_manager.h
@@ -0,0 +1,60 @@
+// Copyright 2022 The Chromium Authors
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#ifndef IOS_CHROME_BROWSER_PUSH_NOTIFICATION_PUSH_NOTIFICATION_ACCOUNT_CONTEXT_MANAGER_H_
+#define IOS_CHROME_BROWSER_PUSH_NOTIFICATION_PUSH_NOTIFICATION_ACCOUNT_CONTEXT_MANAGER_H_
+
+#import <Foundation/Foundation.h>
+
+#import "ios/chrome/browser/browser_state/chrome_browser_state_manager.h"
+
+class BrowserStateInfoCache;
+
+namespace ios {
+class ChromeBrowserStateManager;
+
+}
+
+// This class is intended to store the permissions for each push notification
+// enabled feature for a given account and the number of times the account is
+// signed in across BrowserStates.
+@interface PushNotificationAccountContext : NSObject
+// A dictionary that maps the string value of a push notification client id to
+// the perf service value for that push notification enable feature.
+@property(nonatomic, copy) NSDictionary<NSString*, NSNumber*>* preferenceMap;
+// A counter that stores the number of times a given account is used across
+// BrowserStates.
+@property(nonatomic, readonly) NSUInteger occurrencesAcrossBrowserStates;
+@end
+
+// The purpose of this class is to manage the mapping between GaiaIDs and its
+// context data related to push notifications.
+@interface PushNotificationAccountContextManager : NSObject
+
+// The designated initializer. `BrowserStateInfoCache` must not be nil.
+- (instancetype)initWithChromeBrowserStateManager:
+    (ios::ChromeBrowserStateManager*)manager NS_DESIGNATED_INITIALIZER;
+
+- (instancetype)init NS_UNAVAILABLE;
+
+// Adds the account to the manager if the account is not signed into the device
+// in any BrowserState. This function returns a BOOL value indicating whether
+// the account was added to the manager.
+- (BOOL)addAccount:(NSString*)gaiaID;
+
+// Removes the account from the manager if the account is not signed into the
+// device in any BrowserState. This function returns a BOOL value indicating
+// whether the account was removed from the maanger.
+- (BOOL)removeAccount:(NSString*)gaiaID;
+
+// A dictionary that maps a user's GAIA ID to an object containing the account's
+// preferences for all push notification enabled features and an number
+// representing the number of times the account is signed in across
+// BrowserStates.
+@property(nonatomic, readonly)
+    NSDictionary<NSString*, PushNotificationAccountContext*>* contextMap;
+
+@end
+
+#endif  // IOS_CHROME_BROWSER_PUSH_NOTIFICATION_PUSH_NOTIFICATION_ACCOUNT_CONTEXT_MANAGER_H_
diff --git a/ios/chrome/browser/push_notification/push_notification_account_context_manager.mm b/ios/chrome/browser/push_notification/push_notification_account_context_manager.mm
new file mode 100644
index 0000000..4c28430d
--- /dev/null
+++ b/ios/chrome/browser/push_notification/push_notification_account_context_manager.mm
@@ -0,0 +1,145 @@
+// Copyright 2022 The Chromium Authors
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#import "ios/chrome/browser/push_notification/push_notification_account_context_manager.h"
+
+#import "base/strings/sys_string_conversions.h"
+#import "components/prefs/pref_service.h"
+#import "ios/chrome/browser/browser_state/browser_state_info_cache.h"
+#import "ios/chrome/browser/browser_state/chrome_browser_state.h"
+#import "ios/chrome/browser/browser_state/chrome_browser_state_manager.h"
+#import "ios/chrome/browser/prefs/pref_names.h"
+
+#if !defined(__has_feature) || !__has_feature(objc_arc)
+#error "This file requires ARC support."
+#endif
+
+@interface PushNotificationAccountContext ()
+
+@property(nonatomic, assign) NSUInteger occurrencesAcrossBrowserStates;
+
+@end
+
+@implementation PushNotificationAccountContext
+@end
+
+@implementation PushNotificationAccountContextManager {
+  // Used to retrieve BrowserStates located at a given path.
+  ios::ChromeBrowserStateManager* _chromeBrowserStateManager;
+  // Maps a GaiaID to the given account's AccountContext object.
+  NSMutableDictionary<NSString*, PushNotificationAccountContext*>* _contextMap;
+}
+
+- (instancetype)initWithChromeBrowserStateManager:
+    (ios::ChromeBrowserStateManager*)manager {
+  self = [super init];
+
+  if (self) {
+    _contextMap = [[NSMutableDictionary alloc] init];
+    _chromeBrowserStateManager = manager;
+    BrowserStateInfoCache* infoCache = manager->GetBrowserStateInfoCache();
+
+    const size_t numberOfBrowserStates = infoCache->GetNumberOfBrowserStates();
+    for (size_t i = 0; i < numberOfBrowserStates; i++) {
+      NSString* gaiaID =
+          base::SysUTF8ToNSString(infoCache->GetGAIAIdOfBrowserStateAtIndex(i));
+      base::FilePath path = infoCache->GetPathOfBrowserStateAtIndex(i);
+      ChromeBrowserState* chromeBrowserState =
+          _chromeBrowserStateManager->GetBrowserState(path);
+      [self addAccount:gaiaID withBrowserState:chromeBrowserState];
+    }
+  }
+
+  return self;
+}
+
+- (BOOL)addAccount:(NSString*)gaiaID {
+  ChromeBrowserState* chromeBrowserState = [self chromeBrowserStateFrom:gaiaID];
+  if (!chromeBrowserState)
+    return NO;
+
+  return [self addAccount:gaiaID
+         withBrowserState:[self chromeBrowserStateFrom:gaiaID]];
+}
+
+- (BOOL)removeAccount:(NSString*)gaiaID {
+  PushNotificationAccountContext* context = _contextMap[gaiaID];
+  DCHECK(context);
+
+  context.occurrencesAcrossBrowserStates--;
+  if (context.occurrencesAcrossBrowserStates > 0)
+    return NO;
+
+  [_contextMap removeObjectForKey:gaiaID];
+  return YES;
+}
+
+#pragma mark - Properties
+
+- (NSDictionary<NSString*, PushNotificationAccountContext*>*)contextMap {
+  return _contextMap;
+}
+
+#pragma mark - Private
+
+// Create a new AccountContext object for the given gaiaID, if the gaia id does
+// not already exist in the dictionary, and maps the gaiaID to the new context
+// object.
+- (BOOL)addAccount:(NSString*)gaiaID
+    withBrowserState:(ChromeBrowserState*)browserState {
+  DCHECK(browserState);
+
+  if (!gaiaID.length)
+    return NO;
+
+  PushNotificationAccountContext* context = _contextMap[gaiaID];
+  if (context) {
+    context.occurrencesAcrossBrowserStates++;
+    return NO;
+  }
+
+  PrefService* prefService = browserState->GetPrefs();
+  NSMutableDictionary<NSString*, NSNumber*>* preferenceMap =
+      [[NSMutableDictionary alloc] init];
+  const base::Value::Dict& permissions =
+      prefService->GetDict(prefs::kFeaturePushNotificationPermissions);
+
+  for (const auto pair : permissions) {
+    preferenceMap[base::SysUTF8ToNSString(pair.first)] =
+        [NSNumber numberWithBool:pair.second.GetBool()];
+  }
+
+  context = [[PushNotificationAccountContext alloc] init];
+  context.preferenceMap = preferenceMap;
+  context.occurrencesAcrossBrowserStates = 1;
+  _contextMap[gaiaID] = context;
+
+  return YES;
+}
+
+// Returns the ChromeBrowserState that has the given gaiaID set as the primary
+// account. TODO(crbug.com/1400732) Implement policy that computes correct
+// permission set. This function naively chooses the first ChromeBrowserState
+// that is associated with the given gaiaID. In a multi-profile environment
+// where the given gaiaID is signed into multiple profiles, it is possible that
+// the push notification enabled features' permissions may be incorrectly
+// applied.
+- (ChromeBrowserState*)chromeBrowserStateFrom:(NSString*)gaiaID {
+  BrowserStateInfoCache* infoCache =
+      _chromeBrowserStateManager->GetBrowserStateInfoCache();
+  const size_t numberOfBrowserStates = infoCache->GetNumberOfBrowserStates();
+  for (size_t i = 0; i < numberOfBrowserStates; i++) {
+    NSString* browserStateGaiaID =
+        base::SysUTF8ToNSString(infoCache->GetGAIAIdOfBrowserStateAtIndex(i));
+
+    if (gaiaID == browserStateGaiaID) {
+      base::FilePath path = infoCache->GetPathOfBrowserStateAtIndex(i);
+      return _chromeBrowserStateManager->GetBrowserState(path);
+    }
+  }
+
+  return nil;
+}
+
+@end
diff --git a/ios/chrome/browser/push_notification/push_notification_client_id.h b/ios/chrome/browser/push_notification/push_notification_client_id.h
index 249dfd6..e85417b 100644
--- a/ios/chrome/browser/push_notification/push_notification_client_id.h
+++ b/ios/chrome/browser/push_notification/push_notification_client_id.h
@@ -16,7 +16,7 @@
 enum class PushNotificationClientId {
   // TODO(crbug.com/1353801): Once Chrome has a push notification enabled
   // feature, add that feature's identifier to this class.
-  kCommerce = 1,
+  kCommerce = 1
 };
 
 #endif  // IOS_CHROME_BROWSER_PUSH_NOTIFICATION_PUSH_NOTIFICATION_CLIENT_ID_H_
diff --git a/ios/chrome/browser/push_notification/push_notification_client_manager.h b/ios/chrome/browser/push_notification/push_notification_client_manager.h
index aebd6e8..8547bf2 100644
--- a/ios/chrome/browser/push_notification/push_notification_client_manager.h
+++ b/ios/chrome/browser/push_notification/push_notification_client_manager.h
@@ -31,6 +31,14 @@
   void AddPushNotificationClient(
       std::unique_ptr<PushNotificationClient> client);
 
+  // This function removes the mapping between a PushNotificationClientId and a
+  // PushNotificationClient from the manager.
+  void RemovePushNotificationClient(PushNotificationClientId client_id);
+
+  // This function returns a list of the PushNotificationClients stored by the
+  // manager.
+  std::vector<const PushNotificationClient*> GetPushNotificationClients();
+
   // This function is called when the user interacts with the delivered
   // notification. This function identifies and delegates the interacted with
   // notification to the appropriate PushNotificationClient.
@@ -58,11 +66,10 @@
   static std::vector<PushNotificationClientId> GetClients();
 
  private:
-  // A list of features that support push notifications.
-  std::unordered_map<PushNotificationClientId,
-                     std::unique_ptr<PushNotificationClient>>
-      clients_ = std::unordered_map<PushNotificationClientId,
-                                    std::unique_ptr<PushNotificationClient>>();
+  using ClientMap = std::unordered_map<PushNotificationClientId,
+                                       std::unique_ptr<PushNotificationClient>>;
+  // A map of client ids to the features that support push notifications.
+  ClientMap clients_;
 };
 
-#endif  // IOS_CHROME_BROWSER_PUSH_NOTIFICATION_PUSH_NOTIFICATION_CLIENT_MANAGER_H_
\ No newline at end of file
+#endif  // IOS_CHROME_BROWSER_PUSH_NOTIFICATION_PUSH_NOTIFICATION_CLIENT_MANAGER_H_
diff --git a/ios/chrome/browser/push_notification/push_notification_client_manager.mm b/ios/chrome/browser/push_notification/push_notification_client_manager.mm
index 46c15df..38cad88 100644
--- a/ios/chrome/browser/push_notification/push_notification_client_manager.mm
+++ b/ios/chrome/browser/push_notification/push_notification_client_manager.mm
@@ -15,11 +15,6 @@
 #error "This file requires ARC support."
 #endif
 
-namespace {
-NSString* kClientIdPushNotificationDictionaryKey =
-    @"push_notification_client_id";
-}  // namespace
-
 PushNotificationClientManager::PushNotificationClientManager() {
   if (IsPriceNotificationsEnabled()) {
     AddPushNotificationClient(
@@ -33,34 +28,44 @@
   clients_.insert(std::make_pair(client->GetClientId(), std::move(client)));
 }
 
+void PushNotificationClientManager::RemovePushNotificationClient(
+    PushNotificationClientId client_id) {
+  clients_.erase(client_id);
+}
+
+std::vector<const PushNotificationClient*>
+PushNotificationClientManager::GetPushNotificationClients() {
+  std::vector<const PushNotificationClient*> manager_clients;
+
+  for (auto& client : clients_) {
+    manager_clients.push_back(std::move(client.second.get()));
+  }
+
+  return manager_clients;
+}
+
 void PushNotificationClientManager::HandleNotificationInteraction(
     UNNotificationResponse* notification_response) {
-  NSDictionary* user_info =
-      notification_response.notification.request.content.userInfo;
-
-  PushNotificationClientId client_id =
-      static_cast<PushNotificationClientId>([[user_info
-          objectForKey:kClientIdPushNotificationDictionaryKey] integerValue]);
-
-  auto it = clients_.find(client_id);
-  if (it != clients_.end()) {
-    it->second->HandleNotificationInteraction(notification_response);
+  for (auto& client : clients_) {
+    client.second->HandleNotificationInteraction(notification_response);
   }
 }
 
 UIBackgroundFetchResult
 PushNotificationClientManager::HandleNotificationReception(
     NSDictionary<NSString*, id>* user_info) {
-  PushNotificationClientId client_id =
-      static_cast<PushNotificationClientId>([[user_info
-          objectForKey:kClientIdPushNotificationDictionaryKey] integerValue]);
-
-  auto it = clients_.find(client_id);
-  if (it != clients_.end()) {
-    return it->second->HandleNotificationReception(user_info);
+  UIBackgroundFetchResult result = UIBackgroundFetchResultNoData;
+  for (auto& client : clients_) {
+    UIBackgroundFetchResult client_result =
+        client.second->HandleNotificationReception(user_info);
+    if (client_result == UIBackgroundFetchResultNewData) {
+      return UIBackgroundFetchResultNewData;
+    } else if (client_result == UIBackgroundFetchResultFailed) {
+      result = client_result;
+    }
   }
 
-  return UIBackgroundFetchResultNoData;
+  return result;
 }
 
 void PushNotificationClientManager::RegisterActionableNotifications() {
diff --git a/ios/chrome/browser/push_notification/push_notification_client_manager_unittest.mm b/ios/chrome/browser/push_notification/push_notification_client_manager_unittest.mm
new file mode 100644
index 0000000..183629a
--- /dev/null
+++ b/ios/chrome/browser/push_notification/push_notification_client_manager_unittest.mm
@@ -0,0 +1,123 @@
+// Copyright 2022 The Chromium Authors
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#import <UserNotifications/UserNotifications.h>
+
+#import "ios/chrome/browser/push_notification/push_notification_client.h"
+#import "ios/chrome/browser/push_notification/push_notification_client_manager.h"
+#import "ios/chrome/browser/push_notification/test_push_notification_client.h"
+#import "testing/platform_test.h"
+
+#if !defined(__has_feature) || !__has_feature(objc_arc)
+#error "This file requires ARC support."
+#endif
+
+namespace {
+
+void GenerateClients(std::unique_ptr<PushNotificationClientManager>& manager,
+                     size_t number_of_clients) {
+  for (size_t i = 0; i < number_of_clients; i++) {
+    std::unique_ptr<PushNotificationClient> client =
+        std::make_unique<TestPushNotificationClient>(i + 1);
+    manager->AddPushNotificationClient(std::move(client));
+  }
+}
+
+TestPushNotificationClient* GetClient(
+    std::unique_ptr<PushNotificationClientManager>& manager,
+    size_t index) {
+  return const_cast<TestPushNotificationClient*>(
+      static_cast<const TestPushNotificationClient*>(
+          manager->GetPushNotificationClients()[index]));
+}
+
+}  // namespace
+
+class PushNotificationClientManagerTest : public PlatformTest {
+ protected:
+  PushNotificationClientManagerTest() {
+    size_t number_of_clients = manager->GetPushNotificationClients().size();
+
+    for (size_t i = 0; i < number_of_clients; i++) {
+      manager->RemovePushNotificationClient(
+          GetClient(manager, i)->GetClientId());
+    }
+  }
+  std::unique_ptr<PushNotificationClientManager> manager =
+      std::make_unique<PushNotificationClientManager>();
+};
+
+TEST_F(PushNotificationClientManagerTest, AddClient) {
+  const unsigned long number_of_clients = 1;
+  GenerateClients(manager, number_of_clients);
+
+  ASSERT_EQ(number_of_clients, manager->GetPushNotificationClients().size());
+}
+
+TEST_F(PushNotificationClientManagerTest, AddMultipleClients) {
+  const unsigned long number_of_clients = 5;
+  GenerateClients(manager, number_of_clients);
+
+  ASSERT_EQ(number_of_clients, manager->GetPushNotificationClients().size());
+}
+
+TEST_F(PushNotificationClientManagerTest, HandleNotificationReception) {
+  GenerateClients(manager, 1);
+  ASSERT_EQ(UIBackgroundFetchResultNoData,
+            manager->HandleNotificationReception(nil));
+}
+
+TEST_F(PushNotificationClientManagerTest,
+       HandleNotificationReceptionWithNewData) {
+  const unsigned long number_of_clients = 5;
+  GenerateClients(manager, number_of_clients);
+
+  TestPushNotificationClient* client = GetClient(manager, 0);
+  client->SetBackgroundFetchResult(UIBackgroundFetchResultNewData);
+  ASSERT_EQ(UIBackgroundFetchResultNewData,
+            manager->HandleNotificationReception(nil));
+}
+
+TEST_F(PushNotificationClientManagerTest,
+       HandleNotificationReceptionWithFailure) {
+  const unsigned long number_of_clients = 5;
+  GenerateClients(manager, number_of_clients);
+
+  TestPushNotificationClient* client = GetClient(manager, 0);
+  client->SetBackgroundFetchResult(UIBackgroundFetchResultFailed);
+  ASSERT_EQ(UIBackgroundFetchResultFailed,
+            manager->HandleNotificationReception(nil));
+}
+
+TEST_F(PushNotificationClientManagerTest,
+       HandleNotificationReceptionWithNewDataAndFailure) {
+  const unsigned long number_of_clients = 5;
+  GenerateClients(manager, number_of_clients);
+
+  GetClient(manager, 0)
+      ->SetBackgroundFetchResult(UIBackgroundFetchResultNewData);
+  GetClient(manager, 1)
+      ->SetBackgroundFetchResult(UIBackgroundFetchResultFailed);
+  ASSERT_EQ(UIBackgroundFetchResultNewData,
+            manager->HandleNotificationReception(nil));
+}
+
+TEST_F(PushNotificationClientManagerTest, HandleNotificationInteraction) {
+  const unsigned long number_of_clients = 1;
+  GenerateClients(manager, number_of_clients);
+
+  manager->HandleNotificationInteraction(nil);
+  ASSERT_TRUE(GetClient(manager, 0)->HasNotificationReceivedInteraction());
+}
+
+TEST_F(PushNotificationClientManagerTest,
+       HandleNotificationInteractionWithMultipleClients) {
+  const unsigned long number_of_clients = 5;
+  GenerateClients(manager, number_of_clients);
+
+  manager->HandleNotificationInteraction(nil);
+  for (size_t i = 0; i < manager->GetPushNotificationClients().size(); i++) {
+    ASSERT_TRUE(GetClient(manager, i)->HasNotificationReceivedInteraction());
+  }
+}
diff --git a/ios/chrome/browser/push_notification/push_notification_configuration.h b/ios/chrome/browser/push_notification/push_notification_configuration.h
index aa38d51..ed3f3503 100644
--- a/ios/chrome/browser/push_notification/push_notification_configuration.h
+++ b/ios/chrome/browser/push_notification/push_notification_configuration.h
@@ -7,6 +7,7 @@
 
 #import <Foundation/Foundation.h>
 
+@class PushNotificationAccountContextManager;
 @protocol SingleSignOnService;
 
 using GaiaIdToPushNotificationPreferenceMap =
@@ -28,10 +29,18 @@
 // SingleSignOnService used by PushNotificationService.
 @property(nonatomic, strong) id<SingleSignOnService> ssoService;
 
-// A dictionary that maps a user's GAIA ID to its preferences for all push
-// notification enabled features.
+// DEPRECATED. Please use the `contextManager.contextMap` instead. A dictionary
+// that maps a user's GAIA ID to its preferences for all push notification
+// enabled features.
 @property(nonatomic, copy) GaiaIdToPushNotificationPreferenceMap* preferenceMap;
 
+// A dictionary that maps a user's GAIA ID to an object containing the account's
+// preferences for all push notification enabled features and an number
+// representing the number of times the account is signed in across
+// BrowserStates.
+@property(nonatomic, copy)
+    PushNotificationAccountContextManager* contextManager;
+
 @end
 
-#endif  // IOS_CHROME_BROWSER_PUSH_NOTIFICATION_PUSH_NOTIFICATION_CONFIGURATION_H_
\ No newline at end of file
+#endif  // IOS_CHROME_BROWSER_PUSH_NOTIFICATION_PUSH_NOTIFICATION_CONFIGURATION_H_
diff --git a/ios/chrome/browser/push_notification/push_notification_service.h b/ios/chrome/browser/push_notification/push_notification_service.h
index c5aaa0e8..09e64dc 100644
--- a/ios/chrome/browser/push_notification/push_notification_service.h
+++ b/ios/chrome/browser/push_notification/push_notification_service.h
@@ -12,6 +12,10 @@
 namespace user_prefs {
 class PrefRegistrySyncable;
 }  // namespace user_prefs
+namespace ios {
+class ChromeBrowserStateManager;
+}
+@class PushNotificationAccountContextManager;
 class PushNotificationClientManager;
 
 // Service responsible for establishing connection and interacting
@@ -35,6 +39,21 @@
   // the operation successfully or unsuccessfully completes.
   virtual void UnregisterDevice(CompletionHandler completion_handler) = 0;
 
+  // Returns whether the device has retrieved and stored its APNS device token.
+  virtual bool DeviceTokenIsSet() const;
+
+  // Registers the new account to the push notification server. In a multi
+  // BrowserState environment, the PushNotificationService tracks the signed in
+  // account across BrowserStates.
+  void RegisterAccount(NSString* account_id,
+                       CompletionHandler completion_handler);
+
+  // Unregisters the account from the push notification server. In a multi
+  // BrowserState environment, the account will not be signed out until it's
+  // signed out across BrowserStates.
+  void UnregisterAccount(NSString* account_id,
+                         CompletionHandler completion_handler);
+
   // Updates the current user's push notification preferences with the push
   // notification server.
   void UpdateFeaturePushNotificationPreferences(
@@ -51,10 +70,30 @@
   // Returns PushNotificationService's PushNotificationClientManager.
   PushNotificationClientManager* GetPushNotificationClientManager();
 
+ protected:
+  PushNotificationService(ios::ChromeBrowserStateManager* manager);
+  // Registers the device with the push notification server. By supplying a list
+  // of the GAIA IDs currently logged into Chrome on the device and the device's
+  // APNS token, the server associates the GAIA IDs to the device, which allows
+  // the server to begin sending push notifications to that device. When this
+  // method is called, the server overwrites the accountIDs that are associated
+  // with the device token with the given array of accountIDs. Thus, to
+  // unregister an account from receiving push notifications on the device, this
+  // method should be called with an array of accountIDs that omits the account
+  // that is intended to be unregistered.
+  virtual void SetAccountsToDevice(NSArray<NSString*>* account_ids,
+                                   CompletionHandler completion_handler) {}
+
  private:
   // The PushNotificationClientManager manages all interactions between the
   // system and push notification enabled features.
   std::unique_ptr<PushNotificationClientManager> client_manager_;
+
+  // Stores a mapping of each account's GAIA ID signed into the device to its
+  // context object. This object contains the account's pref service values
+  // pertaining to push notification supported features and the number of times
+  // the given account is signed in across multiple browser states.
+  __strong PushNotificationAccountContextManager* context_manager_;
 };
 
 #endif  // IOS_CHROME_BROWSER_PUSH_NOTIFICATION_PUSH_NOTIFICATION_SERVICE_H_
diff --git a/ios/chrome/browser/push_notification/push_notification_service.mm b/ios/chrome/browser/push_notification/push_notification_service.mm
index 272baf8..1cc868d 100644
--- a/ios/chrome/browser/push_notification/push_notification_service.mm
+++ b/ios/chrome/browser/push_notification/push_notification_service.mm
@@ -7,7 +7,9 @@
 #import "base/strings/string_number_conversions.h"
 #import "base/values.h"
 #import "components/pref_registry/pref_registry_syncable.h"
+#import "ios/chrome/browser/application_context/application_context.h"
 #import "ios/chrome/browser/prefs/pref_names.h"
+#import "ios/chrome/browser/push_notification/push_notification_account_context_manager.h"
 #import "ios/chrome/browser/push_notification/push_notification_client_id.h"
 #import "ios/chrome/browser/push_notification/push_notification_client_manager.h"
 
@@ -16,7 +18,15 @@
 #endif
 
 PushNotificationService::PushNotificationService()
-    : client_manager_(std::make_unique<PushNotificationClientManager>()) {}
+    : PushNotificationService(
+          GetApplicationContext()->GetChromeBrowserStateManager()) {}
+
+PushNotificationService::PushNotificationService(
+    ios::ChromeBrowserStateManager* manager)
+    : client_manager_(std::make_unique<PushNotificationClientManager>()) {
+  context_manager_ = [[PushNotificationAccountContextManager alloc]
+      initWithChromeBrowserStateManager:manager];
+}
 
 PushNotificationService::~PushNotificationService() = default;
 
@@ -25,6 +35,28 @@
   return client_manager_.get();
 }
 
+bool PushNotificationService::DeviceTokenIsSet() const {
+  return false;
+}
+
+void PushNotificationService::RegisterAccount(
+    NSString* account_id,
+    CompletionHandler completion_handler) {
+  if ([context_manager_ addAccount:account_id]) {
+    SetAccountsToDevice(context_manager_.contextMap.allKeys,
+                        completion_handler);
+  }
+}
+
+void PushNotificationService::UnregisterAccount(
+    NSString* account_id,
+    CompletionHandler completion_handler) {
+  if ([context_manager_ removeAccount:account_id]) {
+    SetAccountsToDevice(context_manager_.contextMap.allKeys,
+                        completion_handler);
+  }
+}
+
 void PushNotificationService::RegisterBrowserStatePrefs(
     user_prefs::PrefRegistrySyncable* registry) {
   base::Value::Dict feature_push_notification_permission = base::Value::Dict();
diff --git a/ios/chrome/browser/push_notification/test_push_notification_client.h b/ios/chrome/browser/push_notification/test_push_notification_client.h
new file mode 100644
index 0000000..2903b4c
--- /dev/null
+++ b/ios/chrome/browser/push_notification/test_push_notification_client.h
@@ -0,0 +1,31 @@
+// Copyright 2022 The Chromium Authors
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#ifndef IOS_CHROME_BROWSER_PUSH_NOTIFICATION_TEST_PUSH_NOTIFICATION_CLIENT_H_
+#define IOS_CHROME_BROWSER_PUSH_NOTIFICATION_TEST_PUSH_NOTIFICATION_CLIENT_H_
+
+#import "ios/chrome/browser/push_notification/push_notification_client.h"
+
+class TestPushNotificationClient : public PushNotificationClient {
+ public:
+  TestPushNotificationClient(size_t client_id);
+  ~TestPushNotificationClient() override;
+
+  // Override PushNotificationClient::
+  void HandleNotificationInteraction(
+      UNNotificationResponse* notification_response) override;
+  UIBackgroundFetchResult HandleNotificationReception(
+      NSDictionary<NSString*, id>* notification) override;
+  NSArray<UNNotificationCategory*>* RegisterActionableNotifications() override;
+
+  // Indicates whether the client has been
+  bool HasNotificationReceivedInteraction();
+  // Sets the client's UIBackgroundFetchResult to given FetchResult.
+  void SetBackgroundFetchResult(UIBackgroundFetchResult result);
+
+ private:
+  UIBackgroundFetchResult fetch_result_ = UIBackgroundFetchResultNoData;
+  bool has_notification_received_interaction_ = false;
+};
+#endif  // IOS_CHROME_BROWSER_PUSH_NOTIFICATION_TEST_PUSH_NOTIFICATION_CLIENT_H_
diff --git a/ios/chrome/browser/push_notification/test_push_notification_client.mm b/ios/chrome/browser/push_notification/test_push_notification_client.mm
new file mode 100644
index 0000000..b5992725
--- /dev/null
+++ b/ios/chrome/browser/push_notification/test_push_notification_client.mm
@@ -0,0 +1,41 @@
+// Copyright 2022 The Chromium Authors
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#import "ios/chrome/browser/push_notification/test_push_notification_client.h"
+
+#import "ios/chrome/browser/push_notification/push_notification_client_id.h"
+
+#if !defined(__has_feature) || !__has_feature(objc_arc)
+#error "This file requires ARC support."
+#endif
+
+TestPushNotificationClient::TestPushNotificationClient(size_t client_id)
+    : PushNotificationClient(static_cast<PushNotificationClientId>(client_id)) {
+}
+TestPushNotificationClient::~TestPushNotificationClient() = default;
+
+void TestPushNotificationClient::HandleNotificationInteraction(
+    UNNotificationResponse* notification) {
+  has_notification_received_interaction_ = true;
+}
+
+UIBackgroundFetchResult TestPushNotificationClient::HandleNotificationReception(
+    NSDictionary<NSString*, id>* notification) {
+  return fetch_result_;
+}
+
+NSArray<UNNotificationCategory*>*
+TestPushNotificationClient::RegisterActionableNotifications() {
+  // Add actional notifications as new notification types are added.
+  return @[];
+}
+
+bool TestPushNotificationClient::HasNotificationReceivedInteraction() {
+  return has_notification_received_interaction_;
+}
+
+void TestPushNotificationClient::SetBackgroundFetchResult(
+    UIBackgroundFetchResult result) {
+  fetch_result_ = result;
+}
diff --git a/ios/chrome/browser/search_engines/BUILD.gn b/ios/chrome/browser/search_engines/BUILD.gn
index 940e83c..58a78c6 100644
--- a/ios/chrome/browser/search_engines/BUILD.gn
+++ b/ios/chrome/browser/search_engines/BUILD.gn
@@ -16,8 +16,6 @@
     "search_engine_tab_helper.mm",
     "search_engine_tab_helper_factory.h",
     "search_engine_tab_helper_factory.mm",
-    "search_engines_util.cc",
-    "search_engines_util.h",
     "template_url_fetcher_factory.cc",
     "template_url_fetcher_factory.h",
     "template_url_service_client_impl.cc",
@@ -69,7 +67,7 @@
     "extension_search_engine_data_updater.mm",
   ]
   deps = [
-    ":search_engines",
+    ":search_engines_util",
     "//base",
     "//components/search_engines",
     "//ios/chrome/browser/widget_kit:features",
@@ -123,6 +121,20 @@
   configs += [ "//build/config/compiler:enable_arc" ]
 }
 
+source_set("search_engines_util") {
+  sources = [
+    "search_engines_util.cc",
+    "search_engines_util.h",
+  ]
+  deps = [
+    "//base",
+    "//components/country_codes",
+    "//components/prefs",
+    "//components/search_engines",
+  ]
+  configs += [ "//build/config/compiler:enable_arc" ]
+}
+
 optimize_js("search_engine_js") {
   primary_script = "resources/search_engine.js"
   sources = [ "resources/search_engine.js" ]
diff --git a/ios/chrome/browser/ui/context_menu/BUILD.gn b/ios/chrome/browser/ui/context_menu/BUILD.gn
index ec710165..a73b5459 100644
--- a/ios/chrome/browser/ui/context_menu/BUILD.gn
+++ b/ios/chrome/browser/ui/context_menu/BUILD.gn
@@ -18,6 +18,7 @@
     "//ios/chrome/browser/policy:policy_util",
     "//ios/chrome/browser/prefs:pref_names",
     "//ios/chrome/browser/search_engines",
+    "//ios/chrome/browser/search_engines:search_engines_util",
     "//ios/chrome/browser/ui:feature_flags",
     "//ios/chrome/browser/ui/alert_coordinator",
     "//ios/chrome/browser/ui/commands",
diff --git a/ios/chrome/browser/ui/infobars/banners/infobar_banner_consumer.h b/ios/chrome/browser/ui/infobars/banners/infobar_banner_consumer.h
index f79f7a1bc..5ba4abea 100644
--- a/ios/chrome/browser/ui/infobars/banners/infobar_banner_consumer.h
+++ b/ios/chrome/browser/ui/infobars/banners/infobar_banner_consumer.h
@@ -40,6 +40,11 @@
 // The subtitle displayed by this InfobarBanner.
 - (void)setSubtitleText:(NSString*)subtitleText;
 
+// Applies a custom style (e.g. bold, italic) to a range of the subtitle
+// displayed by the InfobarBanner.
+- (void)addCustomStyle:(UIFontDescriptorSymbolicTraits)symbolicTraits
+       toSubtitleRange:(NSRange)range;
+
 // If YES, restricts the number of lines in subtitle to 1.
 - (void)setRestrictSubtitleTextToSingleLine:
     (BOOL)restrictSubtitleTextToSingleLine;
diff --git a/ios/chrome/browser/ui/infobars/banners/infobar_banner_view_controller.mm b/ios/chrome/browser/ui/infobars/banners/infobar_banner_view_controller.mm
index 71e2c24..0ba77823 100644
--- a/ios/chrome/browser/ui/infobars/banners/infobar_banner_view_controller.mm
+++ b/ios/chrome/browser/ui/infobars/banners/infobar_banner_view_controller.mm
@@ -58,6 +58,21 @@
 constexpr base::TimeDelta kLongPressTimeDuration = base::Milliseconds(400);
 }  // namespace
 
+#pragma mark - StyledRange
+
+// Used to track ranges of a string that should receive custom styling.
+@interface StyledRange : NSObject
+
+@property(nonatomic) UIFontDescriptorSymbolicTraits symbolicTraits;
+@property(nonatomic) NSRange range;
+
+@end
+
+@implementation StyledRange
+@end
+
+#pragma mark - InfobarBannerViewController
+
 @interface InfobarBannerViewController ()
 
 // Properties backing the InfobarBannerConsumer protocol.
@@ -67,6 +82,7 @@
 @property(nonatomic, assign) BOOL presentsModal;
 @property(nonatomic, copy) NSString* titleText;
 @property(nonatomic, copy) NSString* subtitleText;
+@property(nonatomic, copy) NSMutableArray<StyledRange*>* subtitleStyledRanges;
 @property(nonatomic, assign) BOOL useIconBackgroundTint;
 @property(nonatomic, strong) UIColor* iconImageTintColor;
 @property(nonatomic, strong) UIColor* iconBackgroundColor;
@@ -117,6 +133,7 @@
     _presentsModal = presentsModal;
     _useIconBackgroundTint = YES;
     _restrictSubtitleTextToSingleLine = NO;
+    _subtitleStyledRanges = [[NSMutableArray alloc] init];
   }
   return self;
 }
@@ -203,11 +220,8 @@
                                       forAxis:UILayoutConstraintAxisVertical];
 
   self.subTitleLabel = [[UILabel alloc] init];
-  self.subTitleLabel.text = self.subtitleText;
-  self.subTitleLabel.font =
-      [UIFont preferredFontForTextStyle:UIFontTextStyleFootnote];
+  self.subTitleLabel.attributedText = [self subtitleAttributedText];
   self.subTitleLabel.adjustsFontForContentSizeCategory = YES;
-  self.subTitleLabel.textColor = [UIColor colorNamed:kTextSecondaryColor];
   if (_restrictSubtitleTextToSingleLine) {
     self.subTitleLabel.numberOfLines = 1;
   } else {
@@ -405,7 +419,7 @@
 
 - (void)setSubtitleText:(NSString*)subtitleText {
   _subtitleText = subtitleText;
-  self.subTitleLabel.text = _subtitleText;
+  self.subTitleLabel.attributedText = [self subtitleAttributedText];
   self.subTitleLabel.hidden = !self.subtitleText.length;
 }
 
@@ -435,6 +449,14 @@
   _iconBackgroundColor = iconBackgroundColor;
 }
 
+- (void)addCustomStyle:(UIFontDescriptorSymbolicTraits)symbolicTraits
+       toSubtitleRange:(NSRange)range {
+  StyledRange* styledRange = [[StyledRange alloc] init];
+  styledRange.symbolicTraits = symbolicTraits;
+  styledRange.range = range;
+  [_subtitleStyledRanges addObject:styledRange];
+}
+
 - (void)setRestrictSubtitleTextToSingleLine:
     (BOOL)restrictSubtitleTextToSingleLine {
   _restrictSubtitleTextToSingleLine = restrictSubtitleTextToSingleLine;
@@ -643,4 +665,33 @@
   return self.titleText;
 }
 
+- (NSMutableAttributedString*)subtitleAttributedText {
+  if (!self.self.subtitleText) {
+    return nil;
+  }
+
+  UIFontDescriptor* defaultDescriptor = [UIFontDescriptor
+      preferredFontDescriptorWithTextStyle:UIFontTextStyleFootnote];
+  // Passing 0 defers the size responsibility to the descriptor.
+  UIFont* defaultFont = [UIFont fontWithDescriptor:defaultDescriptor size:0.0];
+  NSMutableAttributedString* attributedText = [[NSMutableAttributedString alloc]
+      initWithString:self.subtitleText
+          attributes:@{
+            NSFontAttributeName : defaultFont,
+            NSForegroundColorAttributeName :
+                [UIColor colorNamed:kTextSecondaryColor]
+          }];
+
+  for (StyledRange* styledRange in self.subtitleStyledRanges) {
+    UIFontDescriptor* customDescriptor = [defaultDescriptor
+        fontDescriptorWithSymbolicTraits:styledRange.symbolicTraits];
+    // Passing 0 defers the size responsibility to the descriptor.
+    UIFont* customFont = [UIFont fontWithDescriptor:customDescriptor size:0.0];
+    [attributedText addAttribute:NSFontAttributeName
+                           value:customFont
+                           range:styledRange.range];
+  }
+  return attributedText;
+}
+
 @end
diff --git a/ios/chrome/browser/ui/infobars/banners/infobar_banner_view_controller_unittest.mm b/ios/chrome/browser/ui/infobars/banners/infobar_banner_view_controller_unittest.mm
index d00b2d3e..d0b33aa 100644
--- a/ios/chrome/browser/ui/infobars/banners/infobar_banner_view_controller_unittest.mm
+++ b/ios/chrome/browser/ui/infobars/banners/infobar_banner_view_controller_unittest.mm
@@ -56,9 +56,9 @@
   // Add view_controller_ to the UI Hierarchy to make sure views are created and
   // retained correctly.
   [scoped_key_window_.Get() setRootViewController:view_controller_];
-  ASSERT_EQ(view_controller_.titleLabel.text, kTitle);
-  ASSERT_EQ(view_controller_.subTitleLabel.text, kSubtitleText);
-  ASSERT_EQ(view_controller_.infobarButton.titleLabel.text, kButtonText);
+  ASSERT_NSEQ(view_controller_.titleLabel.text, kTitle);
+  ASSERT_NSEQ(view_controller_.subTitleLabel.text, kSubtitleText);
+  ASSERT_NSEQ(view_controller_.infobarButton.titleLabel.text, kButtonText);
 }
 
 TEST_F(InfobarBannerViewControllerTest, TestSubtitleLabelHidden) {
@@ -66,6 +66,38 @@
   // Add view_controller_ to the UI Hierarchy to make sure views are created and
   // retained correctly.
   [scoped_key_window_.Get() setRootViewController:view_controller_];
-  ASSERT_EQ(view_controller_.titleLabel.text, @"title");
+  ASSERT_NSEQ(view_controller_.titleLabel.text, @"title");
   ASSERT_TRUE(view_controller_.subTitleLabel.hidden);
 }
+
+TEST_F(InfobarBannerViewControllerTest, TestAddCustomStyleToSubtitleRange) {
+  NSString* const kSubtitleText = @"BoldItalic";
+  NSRange expectedBoldRange = [kSubtitleText rangeOfString:@"Bold"];
+  NSRange expectedItalicRange = [kSubtitleText rangeOfString:@"Italic"];
+  [view_controller_ setSubtitleText:kSubtitleText];
+
+  [view_controller_ addCustomStyle:UIFontDescriptorTraitBold
+                   toSubtitleRange:expectedBoldRange];
+  [view_controller_ addCustomStyle:UIFontDescriptorTraitItalic
+                   toSubtitleRange:expectedItalicRange];
+  [scoped_key_window_.Get() setRootViewController:view_controller_];
+
+  NSRange maxSearchRange = NSMakeRange(0, [kSubtitleText length]);
+  NSRange range1;
+  UIFont* font1 = [view_controller_.subTitleLabel.attributedText
+                  attribute:NSFontAttributeName
+                    atIndex:0
+      longestEffectiveRange:&range1
+                    inRange:maxSearchRange];
+  EXPECT_TRUE(NSEqualRanges(range1, expectedBoldRange));
+  EXPECT_EQ([[font1 fontDescriptor] symbolicTraits], UIFontDescriptorTraitBold);
+  NSRange range2;
+  UIFont* font2 = [view_controller_.subTitleLabel.attributedText
+                  attribute:NSFontAttributeName
+                    atIndex:expectedItalicRange.location
+      longestEffectiveRange:&range2
+                    inRange:maxSearchRange];
+  EXPECT_TRUE(NSEqualRanges(range2, expectedItalicRange));
+  EXPECT_EQ([[font2 fontDescriptor] symbolicTraits],
+            UIFontDescriptorTraitItalic);
+}
diff --git a/ios/chrome/browser/ui/infobars/banners/test/fake_infobar_banner_consumer.mm b/ios/chrome/browser/ui/infobars/banners/test/fake_infobar_banner_consumer.mm
index 5bb7469..f9480c6 100644
--- a/ios/chrome/browser/ui/infobars/banners/test/fake_infobar_banner_consumer.mm
+++ b/ios/chrome/browser/ui/infobars/banners/test/fake_infobar_banner_consumer.mm
@@ -9,4 +9,9 @@
 #endif
 
 @implementation FakeInfobarBannerConsumer
+
+- (void)addCustomStyle:(UIFontDescriptorSymbolicTraits)symbolicTraits
+       toSubtitleRange:(NSRange)range {
+}
+
 @end
diff --git a/ios/chrome/browser/ui/location_bar/BUILD.gn b/ios/chrome/browser/ui/location_bar/BUILD.gn
index 8195192f..7bbb4011 100644
--- a/ios/chrome/browser/ui/location_bar/BUILD.gn
+++ b/ios/chrome/browser/ui/location_bar/BUILD.gn
@@ -51,6 +51,7 @@
     "//ios/chrome/browser/overlays",
     "//ios/chrome/browser/overlays/public/web_content_area",
     "//ios/chrome/browser/search_engines",
+    "//ios/chrome/browser/search_engines:search_engines_util",
     "//ios/chrome/browser/ssl",
     "//ios/chrome/browser/ui:feature_flags",
     "//ios/chrome/browser/ui/badges",
diff --git a/ios/chrome/browser/ui/omnibox/BUILD.gn b/ios/chrome/browser/ui/omnibox/BUILD.gn
index b901aa74..601f88a 100644
--- a/ios/chrome/browser/ui/omnibox/BUILD.gn
+++ b/ios/chrome/browser/ui/omnibox/BUILD.gn
@@ -167,6 +167,7 @@
     "//ios/chrome/browser/net",
     "//ios/chrome/browser/prerender",
     "//ios/chrome/browser/search_engines",
+    "//ios/chrome/browser/search_engines:search_engines_util",
     "//ios/chrome/browser/sessions",
     "//ios/chrome/browser/ui:feature_flags",
     "//ios/chrome/browser/ui/commands",
diff --git a/ios/chrome/browser/ui/omnibox/keyboard_assist/BUILD.gn b/ios/chrome/browser/ui/omnibox/keyboard_assist/BUILD.gn
index 3f616231..614f3714 100644
--- a/ios/chrome/browser/ui/omnibox/keyboard_assist/BUILD.gn
+++ b/ios/chrome/browser/ui/omnibox/keyboard_assist/BUILD.gn
@@ -27,7 +27,8 @@
     "//components/search_engines",
     "//ios/chrome/app/strings",
     "//ios/chrome/browser/flags:system_flags",
-    "//ios/chrome/browser/search_engines:search_engines",
+    "//ios/chrome/browser/search_engines",
+    "//ios/chrome/browser/search_engines:search_engines_util",
     "//ios/chrome/browser/ui:feature_flags",
     "//ios/chrome/browser/ui/commands",
     "//ios/chrome/browser/ui/lens:lens_entrypoint",
diff --git a/ios/chrome/browser/ui/popup_menu/BUILD.gn b/ios/chrome/browser/ui/popup_menu/BUILD.gn
index 3f848db..e98ebf368 100644
--- a/ios/chrome/browser/ui/popup_menu/BUILD.gn
+++ b/ios/chrome/browser/ui/popup_menu/BUILD.gn
@@ -78,6 +78,7 @@
     "//ios/chrome/browser/policy:policy_util",
     "//ios/chrome/browser/reading_list",
     "//ios/chrome/browser/search_engines",
+    "//ios/chrome/browser/search_engines:search_engines_util",
     "//ios/chrome/browser/translate",
     "//ios/chrome/browser/ui:feature_flags",
     "//ios/chrome/browser/ui/activity_services",
diff --git a/ios/chrome/browser/ui/toolbar/BUILD.gn b/ios/chrome/browser/ui/toolbar/BUILD.gn
index 60f434c..403aa3b 100644
--- a/ios/chrome/browser/ui/toolbar/BUILD.gn
+++ b/ios/chrome/browser/ui/toolbar/BUILD.gn
@@ -45,6 +45,7 @@
     "//ios/chrome/browser/prerender",
     "//ios/chrome/browser/reading_list",
     "//ios/chrome/browser/search_engines",
+    "//ios/chrome/browser/search_engines:search_engines_util",
     "//ios/chrome/browser/ui:feature_flags",
     "//ios/chrome/browser/ui/bookmarks:core",
     "//ios/chrome/browser/ui/broadcaster",
diff --git a/ios/chrome/browser/web_state_list/web_state_list_metrics_browser_agent.mm b/ios/chrome/browser/web_state_list/web_state_list_metrics_browser_agent.mm
index 24709ae..bd21205 100644
--- a/ios/chrome/browser/web_state_list/web_state_list_metrics_browser_agent.mm
+++ b/ios/chrome/browser/web_state_list/web_state_list_metrics_browser_agent.mm
@@ -4,6 +4,7 @@
 
 #import "ios/chrome/browser/web_state_list/web_state_list_metrics_browser_agent.h"
 
+#import "base/metrics/histogram_functions.h"
 #import "base/metrics/histogram_macros.h"
 #import "base/metrics/user_metrics.h"
 #import "base/metrics/user_metrics_action.h"
@@ -85,6 +86,12 @@
     bool user_action) {
   if (metric_collection_paused_)
     return;
+
+  base::TimeDelta age_at_deletion =
+      base::Time::Now() - web_state->GetCreationTime();
+  base::UmaHistogramCustomTimes("Tab.AgeAtDeletion", age_at_deletion,
+                                base::Minutes(1), base::Days(24), 50);
+
   if (user_action)
     base::RecordAction(base::UserMetricsAction("MobileTabClosed"));
 }
diff --git a/ios/chrome/test/BUILD.gn b/ios/chrome/test/BUILD.gn
index 169f407..b62c4445 100644
--- a/ios/chrome/test/BUILD.gn
+++ b/ios/chrome/test/BUILD.gn
@@ -239,6 +239,7 @@
     "//ios/chrome/browser/policy:unit_tests",
     "//ios/chrome/browser/prerender:unit_tests",
     "//ios/chrome/browser/promos_manager:unit_tests",
+    "//ios/chrome/browser/push_notification:unit_tests",
     "//ios/chrome/browser/reading_list:unit_tests",
     "//ios/chrome/browser/safe_browsing:unit_tests",
     "//ios/chrome/browser/safe_browsing/tailored_security:unit_tests",
diff --git a/ios/chrome/test/earl_grey/BUILD.gn b/ios/chrome/test/earl_grey/BUILD.gn
index 747291c..150177c 100644
--- a/ios/chrome/test/earl_grey/BUILD.gn
+++ b/ios/chrome/test/earl_grey/BUILD.gn
@@ -75,6 +75,7 @@
     "//components/password_manager/core/common",
     "//components/prefs",
     "//components/safe_browsing/core/common",
+    "//components/search_engines",
     "//components/strings",
     "//components/sync/base",
     "//components/translate/core/browser",
@@ -97,6 +98,8 @@
     "//ios/chrome/browser/passwords",
     "//ios/chrome/browser/passwords:eg_app_support+eg2",
     "//ios/chrome/browser/policy:eg_app_support+eg2",
+    "//ios/chrome/browser/search_engines:search_engines_util",
+    "//ios/chrome/browser/search_engines:template_url_service_factory",
     "//ios/chrome/browser/signin:fake_system_identity",
     "//ios/chrome/browser/sync",
     "//ios/chrome/browser/translate:eg_app_support+eg2",
diff --git a/ios/chrome/test/earl_grey/DEPS b/ios/chrome/test/earl_grey/DEPS
index 1f3f327..a564f78 100644
--- a/ios/chrome/test/earl_grey/DEPS
+++ b/ios/chrome/test/earl_grey/DEPS
@@ -2,6 +2,7 @@
   # To compile base::Feature under EG2
   "chrome_earl_grey_app_interface\.mm":[
    "+components/autofill/core/common/autofill_features.h",
+   "+components/search_engines/template_url_service.h",
    "+components/password_manager/core/common/password_manager_features.h",
    "+components/payments/core/features.h",
    "+components/ukm/ios/features.h",
diff --git a/ios/chrome/test/earl_grey/chrome_earl_grey_app_interface.mm b/ios/chrome/test/earl_grey/chrome_earl_grey_app_interface.mm
index 5105203c..9cc11e7 100644
--- a/ios/chrome/test/earl_grey/chrome_earl_grey_app_interface.mm
+++ b/ios/chrome/test/earl_grey/chrome_earl_grey_app_interface.mm
@@ -21,6 +21,7 @@
 #import "components/metrics/demographics/demographic_metrics_provider.h"
 #import "components/password_manager/core/common/password_manager_features.h"
 #import "components/prefs/pref_service.h"
+#import "components/search_engines/template_url_service.h"
 #import "components/sync/base/pref_names.h"
 #import "components/unified_consent/unified_consent_service.h"
 #import "components/variations/variations_associated_data.h"
@@ -31,6 +32,8 @@
 #import "ios/chrome/browser/browser_state/chrome_browser_state.h"
 #import "ios/chrome/browser/content_settings/host_content_settings_map_factory.h"
 #import "ios/chrome/browser/ntp/features.h"
+#import "ios/chrome/browser/search_engines/search_engines_util.h"
+#import "ios/chrome/browser/search_engines/template_url_service_factory.h"
 #import "ios/chrome/browser/signin/fake_system_identity.h"
 #import "ios/chrome/browser/sync/sync_service_factory.h"
 #import "ios/chrome/browser/ui/commands/application_commands.h"
@@ -1104,9 +1107,13 @@
 }
 
 + (BOOL)isUseLensToSearchForImageEnabled {
+  TemplateURLService* service =
+      ios::TemplateURLServiceFactory::GetForBrowserState(
+          chrome_test_util::GetOriginalBrowserState());
   return base::FeatureList::IsEnabled(kUseLensToSearchForImage) &&
          ios::provider::IsLensSupported() &&
-         ui::GetDeviceFormFactor() != ui::DEVICE_FORM_FACTOR_TABLET;
+         ui::GetDeviceFormFactor() != ui::DEVICE_FORM_FACTOR_TABLET &&
+         search_engines::SupportsSearchImageWithLens(service);
 }
 
 + (BOOL)isThumbstripEnabledForWindowWithNumber:(int)windowNumber {
diff --git a/ios/chrome/test/providers/push_notification/test_push_notification.mm b/ios/chrome/test/providers/push_notification/test_push_notification.mm
index 628ccc7..41661499 100644
--- a/ios/chrome/test/providers/push_notification/test_push_notification.mm
+++ b/ios/chrome/test/providers/push_notification/test_push_notification.mm
@@ -27,6 +27,12 @@
   void RegisterDevice(PushNotificationConfiguration* config,
                       void (^completion_handler)(NSError* error)) final;
   void UnregisterDevice(void (^completion_handler)(NSError* error)) final;
+  bool DeviceTokenIsSet() const final;
+
+ protected:
+  // PushNotificationService implementation.
+  void SetAccountsToDevice(NSArray<NSString*>* account_ids,
+                           CompletionHandler completion_handler) final;
 };
 
 void TestPushNotificationService::RegisterDevice(
@@ -59,8 +65,28 @@
       }));
 }
 
+bool TestPushNotificationService::DeviceTokenIsSet() const {
+  return false;
+}
+
+void TestPushNotificationService::SetAccountsToDevice(
+    NSArray<NSString*>* account_ids,
+    void (^completion_handler)(NSError* error)) {
+  // Test implementation does nothing. As a result, the `completion_handler` is
+  // called with a NSFeatureUnsupportedError.
+
+  base::SequencedTaskRunnerHandle::Get()->PostTask(
+      FROM_HERE, base::BindOnce(^() {
+        NSError* error =
+            [NSError errorWithDomain:kTestPushNotificationErrorDomain
+                                code:NSFeatureUnsupportedError
+                            userInfo:nil];
+        completion_handler(error);
+      }));
+}
+
 std::unique_ptr<PushNotificationService> CreatePushNotificationService() {
   return std::make_unique<TestPushNotificationService>();
 }
 }  // namespace provider
-}  // namespace ios
\ No newline at end of file
+}  // namespace ios
diff --git a/ios/chrome/test/wpt/cwt_webdriver_app_interface.mm b/ios/chrome/test/wpt/cwt_webdriver_app_interface.mm
index fa433e2..3a7b6c4 100644
--- a/ios/chrome/test/wpt/cwt_webdriver_app_interface.mm
+++ b/ios/chrome/test/wpt/cwt_webdriver_app_interface.mm
@@ -192,56 +192,38 @@
 + (NSString*)executeAsyncJavaScriptFunction:(NSString*)function
                                       inTab:(NSString*)tabID
                                     timeout:(base::TimeDelta)timeout {
-  const std::string kMessageResultKey("result");
-
-  // Use a distinct messageID value for each invocation of this method to
-  // distinguish stale messages (from previous script invocations that timed
-  // out) from a message for the current script.
-  static NSUInteger messageID = 0;
-  std::string command = base::StringPrintf("CWTWebDriver%lu", messageID++);
-
-  // Construct a completion handler that takes a single argument and sends a
-  // message with this argument.
-  std::string scriptCompletionHandler =
-      base::StringPrintf("function(value) {"
-                         "__gCrWeb.message.invokeOnHost({command: "
-                         "'%s.result', %s: value}); }",
-                         command.c_str(), kMessageResultKey.c_str());
-
-  // Construct a script that calls the given `function` with
-  // `scriptCompletionHandler` as an argument.
-  std::string scriptFunctionWithCompletionHandler = base::StringPrintf(
-      "(%s).call(null, %s)", base::SysNSStringToUTF8(function).c_str(),
-      scriptCompletionHandler.c_str());
-
-  __block absl::optional<base::Value> messageValue;
-  const web::WebState::ScriptCommandCallback callback =
-      base::BindRepeating(^(const base::Value& value, const GURL&,
-                            /*interacted*/ bool,
-                            /*sender_frame*/ web::WebFrame*) {
-        const base::Value* result = value.FindKey(kMessageResultKey);
-
-        // `result` will be null when the computed result in JavaScript is
-        // `undefined`. This happens, for example, when injecting a script that
-        // performs some action (like setting the document's title) but doesn't
-        // return any value.
-        if (result)
-          messageValue = result->Clone();
-        else
-          messageValue = base::Value();
-      });
-
   __block BOOL webStateFound = NO;
-  __block base::CallbackListSubscription subscription;
+  __block absl::optional<base::Value> messageValue;
   DispatchSyncOnMainThread(^{
     web::WebState* webState = GetWebStateWithId(tabID);
     if (!webState)
       return;
+    web::WebFrame* mainFrame = web::GetMainFrame(webState);
+    if (!mainFrame) {
+      return;
+    }
     webStateFound = YES;
-    subscription = webState->AddScriptCommandCallback(callback, command);
-    web::WebFrame* main_frame = web::GetMainFrame(webState);
-    main_frame->ExecuteJavaScript(
-        base::UTF8ToUTF16(scriptFunctionWithCompletionHandler));
+
+    NSString* script =
+        [NSString stringWithFormat:@"var result;"
+                                   @"(%@).call(null, (r) => { result = r; } );"
+                                   @"result;",
+                                   function];
+
+    mainFrame->ExecuteJavaScript(base::SysNSStringToUTF16(script),
+                                 base::BindOnce(^(const base::Value* result) {
+                                   // `result` will be null when the computed
+                                   // result in JavaScript is `undefined`. This
+                                   // happens, for example, when injecting a
+                                   // script that performs some action (like
+                                   // setting the document's title) but doesn't
+                                   // return any value.
+                                   if (result) {
+                                     messageValue = result->Clone();
+                                   } else {
+                                     messageValue = base::Value();
+                                   }
+                                 }));
   });
 
   if (!webStateFound)
diff --git a/media/gpu/chromeos/gl_image_processor_backend.cc b/media/gpu/chromeos/gl_image_processor_backend.cc
index 05f3d42..347c6ff 100644
--- a/media/gpu/chromeos/gl_image_processor_backend.cc
+++ b/media/gpu/chromeos/gl_image_processor_backend.cc
@@ -83,9 +83,10 @@
   DCHECK(native_pixmap->AreDmaBufFdsValid());
 
   // Import the NativePixmap into GL.
-  auto image = base::MakeRefCounted<gl::GLImageNativePixmap>(
-      video_frame->coded_size(), gfx::BufferFormat::YUV_420_BIPLANAR);
-  if (!image->Initialize(std::move(native_pixmap))) {
+  auto image = gl::GLImageNativePixmap::Create(
+      video_frame->coded_size(), gfx::BufferFormat::YUV_420_BIPLANAR,
+      std::move(native_pixmap));
+  if (!image) {
     LOG(ERROR) << "Could not initialize the GL image";
     return nullptr;
   }
diff --git a/media/gpu/test/video_frame_validator.cc b/media/gpu/test/video_frame_validator.cc
index c1ccb518..521ea3a 100644
--- a/media/gpu/test/video_frame_validator.cc
+++ b/media/gpu/test/video_frame_validator.cc
@@ -20,6 +20,7 @@
 #include "media/gpu/buildflags.h"
 #include "media/gpu/macros.h"
 #include "media/gpu/test/image_quality_metrics.h"
+#include "media/gpu/test/video_frame_helpers.h"
 #include "media/gpu/test/video_test_helpers.h"
 #include "media/gpu/video_frame_mapper.h"
 #include "media/gpu/video_frame_mapper_factory.h"
@@ -31,8 +32,10 @@
 namespace test {
 
 VideoFrameValidator::VideoFrameValidator(
-    std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor)
+    std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor,
+    CropHelper crop_helper)
     : corrupt_frame_processor_(std::move(corrupt_frame_processor)),
+      crop_helper_(crop_helper),
       num_frames_validating_(0),
       frame_validator_thread_("FrameValidatorThread"),
       frame_validator_cv_(&frame_validator_lock_) {
@@ -179,6 +182,9 @@
 
   ASSERT_TRUE(frame->IsMappable());
 
+  if (ShouldCrop())
+    frame = CloneAndCropFrame(std::move(frame));
+
   auto mismatched_info = Validate(frame, frame_index);
 
   base::AutoLock auto_lock(frame_validator_lock_);
@@ -195,6 +201,36 @@
   frame_validator_cv_.Signal();
 }
 
+scoped_refptr<VideoFrame> VideoFrameValidator::CloneAndCropFrame(
+    scoped_refptr<const VideoFrame> frame) const {
+  const auto crop = crop_helper_.Run(*frame);
+  const auto& visible = frame->visible_rect();
+  auto cloned_frame = CloneVideoFrame(frame.get(), frame->layout());
+  // Ensures that the crop is within the previous visible rectangle.
+  if (!visible.Contains(crop)) {
+    LOG(ERROR) << "Crop " << crop.ToString()
+               << "must be contained by the visible area of the input frame: "
+               << visible.ToString() << ".";
+    return cloned_frame;
+  }
+  return VideoFrame::WrapVideoFrame(cloned_frame, frame->format(), crop,
+                                    crop.size());
+}
+
+gfx::Rect BottomRowCrop(int row_height, const VideoFrame& frame) {
+  const auto& visible = frame.visible_rect();
+  // Performs bounds checking on the row_height to ensure that the returned
+  // crop is contained by the current visible area.
+  if (visible.size().height() < row_height) {
+    LOG(ERROR) << "Could not get a row of height " << row_height
+               << " from a visible area of height " << visible.size().height();
+    return visible;
+  }
+
+  return gfx::Rect{visible.x(), visible.bottom() - row_height, visible.width(),
+                   row_height};
+}
+
 struct MD5VideoFrameValidator::MD5MismatchedFrameInfo
     : public VideoFrameValidator::MismatchedFrameInfo {
   MD5MismatchedFrameInfo(size_t frame_index,
@@ -218,10 +254,11 @@
 std::unique_ptr<MD5VideoFrameValidator> MD5VideoFrameValidator::Create(
     const std::vector<std::string>& expected_frame_checksums,
     VideoPixelFormat validation_format,
-    std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor) {
-  auto video_frame_validator = base::WrapUnique(
-      new MD5VideoFrameValidator(expected_frame_checksums, validation_format,
-                                 std::move(corrupt_frame_processor)));
+    std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor,
+    CropHelper crop_helper) {
+  auto video_frame_validator = base::WrapUnique(new MD5VideoFrameValidator(
+      expected_frame_checksums, validation_format,
+      std::move(corrupt_frame_processor), std::move(crop_helper)));
   if (!video_frame_validator->Initialize()) {
     LOG(ERROR) << "Failed to initialize MD5VideoFrameValidator.";
     return nullptr;
@@ -233,8 +270,10 @@
 MD5VideoFrameValidator::MD5VideoFrameValidator(
     const std::vector<std::string>& expected_frame_checksums,
     VideoPixelFormat validation_format,
-    std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor)
-    : VideoFrameValidator(std::move(corrupt_frame_processor)),
+    std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor,
+    CropHelper crop_helper)
+    : VideoFrameValidator(std::move(corrupt_frame_processor),
+                          std::move(crop_helper)),
       expected_frame_checksums_(expected_frame_checksums),
       validation_format_(validation_format) {}
 
@@ -325,9 +364,11 @@
 std::unique_ptr<RawVideoFrameValidator> RawVideoFrameValidator::Create(
     const GetModelFrameCB& get_model_frame_cb,
     std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor,
-    uint8_t tolerance) {
+    uint8_t tolerance,
+    CropHelper crop_helper) {
   auto video_frame_validator = base::WrapUnique(new RawVideoFrameValidator(
-      get_model_frame_cb, std::move(corrupt_frame_processor), tolerance));
+      get_model_frame_cb, std::move(corrupt_frame_processor), tolerance,
+      std::move(crop_helper)));
   if (!video_frame_validator->Initialize()) {
     LOG(ERROR) << "Failed to initialize RawVideoFrameValidator.";
     return nullptr;
@@ -339,8 +380,10 @@
 RawVideoFrameValidator::RawVideoFrameValidator(
     const GetModelFrameCB& get_model_frame_cb,
     std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor,
-    uint8_t tolerance)
-    : VideoFrameValidator(std::move(corrupt_frame_processor)),
+    uint8_t tolerance,
+    CropHelper crop_helper)
+    : VideoFrameValidator(std::move(corrupt_frame_processor),
+                          std::move(crop_helper)),
       get_model_frame_cb_(get_model_frame_cb),
       tolerance_(tolerance) {}
 
@@ -352,6 +395,10 @@
   DCHECK_CALLED_ON_VALID_SEQUENCE(validator_thread_sequence_checker_);
   auto model_frame = get_model_frame_cb_.Run(frame_index);
   CHECK(model_frame);
+
+  if (ShouldCrop())
+    model_frame = CloneAndCropFrame(std::move(model_frame));
+
   size_t diff_cnt =
       CompareFramesWithErrorDiff(*frame, *model_frame, tolerance_);
   if (diff_cnt > 0)
@@ -376,15 +423,15 @@
     const GetModelFrameCB& get_model_frame_cb,
     std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor,
     ValidationMode validation_mode,
-    double tolerance) {
+    double tolerance,
+    CropHelper crop_helper) {
   auto video_frame_validator = base::WrapUnique(new PSNRVideoFrameValidator(
       get_model_frame_cb, std::move(corrupt_frame_processor), validation_mode,
-      tolerance));
+      tolerance, std::move(crop_helper)));
   if (!video_frame_validator->Initialize()) {
     LOG(ERROR) << "Failed to initialize PSNRVideoFrameValidator.";
     return nullptr;
   }
-
   return video_frame_validator;
 }
 
@@ -392,8 +439,10 @@
     const GetModelFrameCB& get_model_frame_cb,
     std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor,
     ValidationMode validation_mode,
-    double tolerance)
-    : VideoFrameValidator(std::move(corrupt_frame_processor)),
+    double tolerance,
+    CropHelper crop_helper)
+    : VideoFrameValidator(std::move(corrupt_frame_processor),
+                          std::move(crop_helper)),
       get_model_frame_cb_(get_model_frame_cb),
       tolerance_(tolerance),
       validation_mode_(validation_mode) {}
@@ -406,6 +455,10 @@
   DCHECK_CALLED_ON_VALID_SEQUENCE(validator_thread_sequence_checker_);
   auto model_frame = get_model_frame_cb_.Run(frame_index);
   CHECK(model_frame);
+
+  if (ShouldCrop())
+    model_frame = CloneAndCropFrame(std::move(model_frame));
+
   double psnr = ComputePSNR(*frame, *model_frame);
   DVLOGF(4) << "frame_index: " << frame_index << ", psnr: " << psnr;
   psnr_[frame_index] = psnr;
@@ -425,6 +478,7 @@
     average += psnr.second;
   }
   average /= psnr_.size();
+  DVLOGF(4) << "Average PSNR: " << average;
   if (average < tolerance_) {
     LOG(ERROR) << "Average PSNR is too low: " << average;
     return false;
@@ -449,15 +503,15 @@
     const GetModelFrameCB& get_model_frame_cb,
     std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor,
     ValidationMode validation_mode,
-    double tolerance) {
+    double tolerance,
+    CropHelper crop_helper) {
   auto video_frame_validator = base::WrapUnique(new SSIMVideoFrameValidator(
       get_model_frame_cb, std::move(corrupt_frame_processor), validation_mode,
-      tolerance));
+      tolerance, std::move(crop_helper)));
   if (!video_frame_validator->Initialize()) {
     LOG(ERROR) << "Failed to initialize SSIMVideoFrameValidator.";
     return nullptr;
   }
-
   return video_frame_validator;
 }
 
@@ -465,8 +519,10 @@
     const GetModelFrameCB& get_model_frame_cb,
     std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor,
     ValidationMode validation_mode,
-    double tolerance)
-    : VideoFrameValidator(std::move(corrupt_frame_processor)),
+    double tolerance,
+    CropHelper crop_helper)
+    : VideoFrameValidator(std::move(corrupt_frame_processor),
+                          std::move(crop_helper)),
       get_model_frame_cb_(get_model_frame_cb),
       tolerance_(tolerance),
       validation_mode_(validation_mode) {}
@@ -478,8 +534,11 @@
                                   size_t frame_index) {
   DCHECK_CALLED_ON_VALID_SEQUENCE(validator_thread_sequence_checker_);
   auto model_frame = get_model_frame_cb_.Run(frame_index);
-
   CHECK(model_frame);
+
+  if (ShouldCrop())
+    model_frame = CloneAndCropFrame(std::move(model_frame));
+
   double ssim = ComputeSSIM(*frame, *model_frame);
   DVLOGF(4) << "frame_index: " << frame_index << ", ssim: " << ssim;
   ssim_[frame_index] = ssim;
@@ -499,6 +558,7 @@
     average += ssim.second;
   }
   average /= ssim_.size();
+  DVLOGF(4) << "Average SSIM: " << average;
   if (average < tolerance_) {
     LOG(ERROR) << "Average SSIM is too low: " << average;
     return false;
diff --git a/media/gpu/test/video_frame_validator.h b/media/gpu/test/video_frame_validator.h
index 9bd658a..2c504b5a 100644
--- a/media/gpu/test/video_frame_validator.h
+++ b/media/gpu/test/video_frame_validator.h
@@ -48,6 +48,9 @@
   using GetModelFrameCB =
       base::RepeatingCallback<scoped_refptr<const VideoFrame>(size_t)>;
 
+  using CropHelper =
+      base::RepeatingCallback<gfx::Rect(const VideoFrame& frame)>;
+
   VideoFrameValidator(const VideoFrameValidator&) = delete;
   VideoFrameValidator& operator=(const VideoFrameValidator&) = delete;
 
@@ -78,13 +81,22 @@
   };
 
   VideoFrameValidator(
-      std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor);
+      std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor,
+      CropHelper crop_helper);
 
   // Start the frame validation thread.
   bool Initialize();
 
   SEQUENCE_CHECKER(validator_thread_sequence_checker_);
 
+  // Returns true if the processed frame or model frame should be cropped with
+  // CloneAndCropFrame() or VideoFrame::WrapVideoFrame().
+  bool ShouldCrop() const { return static_cast<bool>(crop_helper_); }
+
+  // This can be used by VideoFrameValidators to support cropping.
+  scoped_refptr<VideoFrame> CloneAndCropFrame(
+      scoped_refptr<const VideoFrame> frame) const;
+
  private:
   void CleanUpOnValidatorThread();
 
@@ -108,6 +120,10 @@
   // forwarded to. This can be used to e.g. write corrupted frames to disk.
   std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor_;
 
+  // If |crop_helper_| is runnable, then ShouldCrop() will return true and
+  // CloneAndCropFrame() can be used.
+  const CropHelper crop_helper_;
+
   // The number of frames currently queued for validation.
   size_t num_frames_validating_ GUARDED_BY(frame_validator_lock_);
   // The results of invalid frame data.
@@ -122,6 +138,11 @@
   SEQUENCE_CHECKER(validator_sequence_checker_);
 };
 
+// Returns a cropping rectangle containing the bottom |row_height| rows of the
+// input frame. This can be used by VideoFrameValidator::CropHelper.
+gfx::Rect BottomRowCrop(int row_height, const VideoFrame& frame);
+constexpr int kDefaultBottomRowCropHeight = 2;
+
 // Validate by converting the frame to be validated to |validation_format| to
 // resolve pixel format differences on different platforms. Thereafter, it
 // compares md5 values of the mapped and converted buffer with golden md5
@@ -132,7 +153,8 @@
   static std::unique_ptr<MD5VideoFrameValidator> Create(
       const std::vector<std::string>& expected_frame_checksums,
       VideoPixelFormat validation_format = PIXEL_FORMAT_I420,
-      std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor = nullptr);
+      std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor = nullptr,
+      CropHelper crop_helper = CropHelper());
   ~MD5VideoFrameValidator() override;
 
  private:
@@ -141,7 +163,8 @@
   MD5VideoFrameValidator(
       const std::vector<std::string>& expected_frame_checksums,
       VideoPixelFormat validation_format,
-      std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor);
+      std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor,
+      CropHelper crop_helper);
   MD5VideoFrameValidator(const MD5VideoFrameValidator&) = delete;
   MD5VideoFrameValidator& operator=(const MD5VideoFrameValidator&) = delete;
 
@@ -167,7 +190,8 @@
   static std::unique_ptr<RawVideoFrameValidator> Create(
       const GetModelFrameCB& get_model_frame_cb,
       std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor = nullptr,
-      uint8_t tolerance = kDefaultTolerance);
+      uint8_t tolerance = kDefaultTolerance,
+      CropHelper crop_helper = CropHelper());
   ~RawVideoFrameValidator() override;
 
  private:
@@ -176,7 +200,8 @@
   RawVideoFrameValidator(
       const GetModelFrameCB& get_model_frame_cb,
       std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor,
-      uint8_t tolerance);
+      uint8_t tolerance,
+      CropHelper crop_helper);
 
   std::unique_ptr<MismatchedFrameInfo> Validate(
       scoped_refptr<const VideoFrame> frame,
@@ -186,8 +211,6 @@
   const uint8_t tolerance_;
 };
 
-// Validate by computing PSNR from the frame to be validated and the model frame
-// acquired by |get_model_frame_cb_|. If the PSNR value is equal to or more than
 // |tolerance_|, the validation on the frame passes.
 class PSNRVideoFrameValidator : public VideoFrameValidator {
  public:
@@ -197,7 +220,8 @@
       const GetModelFrameCB& get_model_frame_cb,
       std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor = nullptr,
       ValidationMode validation_mode = ValidationMode::kThreshold,
-      double tolerance = kDefaultTolerance);
+      double tolerance = kDefaultTolerance,
+      CropHelper crop_helper = CropHelper());
   const std::map<size_t, double>& GetPSNRValues() const { return psnr_; }
   ~PSNRVideoFrameValidator() override;
 
@@ -208,7 +232,8 @@
       const GetModelFrameCB& get_model_frame_cb,
       std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor,
       ValidationMode validation_mode,
-      double tolerance);
+      double tolerance,
+      CropHelper crop_helper);
 
   std::unique_ptr<MismatchedFrameInfo> Validate(
       scoped_refptr<const VideoFrame> frame,
@@ -217,6 +242,7 @@
   bool Passed() const override;
 
   const GetModelFrameCB get_model_frame_cb_;
+  const CropHelper crop_helper_;
   const double tolerance_;
   const ValidationMode validation_mode_;
   std::map<size_t, double> psnr_;
@@ -233,7 +259,8 @@
       const GetModelFrameCB& get_model_frame_cb,
       std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor = nullptr,
       ValidationMode validation_mode = ValidationMode::kThreshold,
-      double tolerance = kDefaultTolerance);
+      double tolerance = kDefaultTolerance,
+      CropHelper crop_helper = CropHelper());
   const std::map<size_t, double>& GetSSIMValues() const { return ssim_; }
   ~SSIMVideoFrameValidator() override;
 
@@ -244,7 +271,8 @@
       const GetModelFrameCB& get_model_frame_cb,
       std::unique_ptr<VideoFrameProcessor> corrupt_frame_processor,
       ValidationMode validation_mode,
-      double tolerance);
+      double tolerance,
+      CropHelper crop_helper);
 
   std::unique_ptr<MismatchedFrameInfo> Validate(
       scoped_refptr<const VideoFrame> frame,
diff --git a/media/gpu/v4l2/generic_v4l2_device.cc b/media/gpu/v4l2/generic_v4l2_device.cc
index fa9a95b..c3c57a0 100644
--- a/media/gpu/v4l2/generic_v4l2_device.cc
+++ b/media/gpu/v4l2/generic_v4l2_device.cc
@@ -316,9 +316,8 @@
 
   // TODO(b/220336463): plumb the right color space.
   auto image =
-      base::MakeRefCounted<gl::GLImageNativePixmap>(size, buffer_format);
-  bool ret = image->Initialize(std::move(pixmap));
-  DCHECK(ret);
+      gl::GLImageNativePixmap::Create(size, buffer_format, std::move(pixmap));
+  DCHECK(image);
   return image;
 }
 
diff --git a/media/gpu/vaapi/vaapi_picture_native_pixmap_egl.cc b/media/gpu/vaapi/vaapi_picture_native_pixmap_egl.cc
index f739173..4098b3a 100644
--- a/media/gpu/vaapi/vaapi_picture_native_pixmap_egl.cc
+++ b/media/gpu/vaapi/vaapi_picture_native_pixmap_egl.cc
@@ -86,10 +86,10 @@
     return VaapiStatus::Codes::kBadContext;
 
   // TODO(b/220336463): plumb the right color space.
-  auto image =
-      base::MakeRefCounted<gl::GLImageNativePixmap>(visible_size_, format);
+  auto image = gl::GLImageNativePixmap::CreateFromTexture(visible_size_, format,
+                                                          texture_id_);
   // Create an EGLImage from a gl texture
-  if (!image->InitializeFromTexture(texture_id_)) {
+  if (!image) {
     DLOG(ERROR) << "Failed to initialize eglimage from texture id: "
                 << texture_id_;
     return VaapiStatus::Codes::kFailedToInitializeImage;
diff --git a/media/gpu/vaapi/vaapi_picture_native_pixmap_ozone.cc b/media/gpu/vaapi/vaapi_picture_native_pixmap_ozone.cc
index 2abb6886..0b646a87f5 100644
--- a/media/gpu/vaapi/vaapi_picture_native_pixmap_ozone.cc
+++ b/media/gpu/vaapi/vaapi_picture_native_pixmap_ozone.cc
@@ -89,13 +89,14 @@
 
   // TODO(b/220336463): plumb the right color space.
   auto image =
-      base::MakeRefCounted<gl::GLImageNativePixmap>(visible_size_, format);
-  if (!image->Initialize(std::move(pixmap))) {
+      gl::GLImageNativePixmap::Create(visible_size_, format, std::move(pixmap));
+  if (!image) {
     LOG(ERROR) << "Failed to create GLImage";
     return VaapiStatus::Codes::kFailedToInitializeImage;
   }
 
-  gl_image_ = image;
+  gl_image_ = std::move(image);
+
   if (!gl_image_->BindTexImage(texture_target_)) {
     LOG(ERROR) << "Failed to bind texture to GLImage";
     return VaapiStatus::Codes::kFailedToBindTexture;
diff --git a/media/mojo/clients/mojo_android_overlay_unittest.cc b/media/mojo/clients/mojo_android_overlay_unittest.cc
index 7e1447a8..aa426a8 100644
--- a/media/mojo/clients/mojo_android_overlay_unittest.cc
+++ b/media/mojo/clients/mojo_android_overlay_unittest.cc
@@ -138,7 +138,7 @@
     surface_ = gl::ScopedJavaSurface(surface_texture_.get());
     surface_key_ = gpu::GpuSurfaceTracker::Get()->AddSurfaceForNativeWidget(
         gpu::GpuSurfaceTracker::SurfaceRecord(
-            gfx::kNullAcceleratedWidget, surface_.j_surface(),
+            surface_.CopyRetainOwnership(),
             false /* can_be_used_with_surface_control */));
 
     mock_provider_.client_->OnSurfaceReady(surface_key_);
diff --git a/mojo/public/tools/bindings/generators/mojom_ts_generator.py b/mojo/public/tools/bindings/generators/mojom_ts_generator.py
index 8ee12d63..94296195 100644
--- a/mojo/public/tools/bindings/generators/mojom_ts_generator.py
+++ b/mojo/public/tools/bindings/generators/mojom_ts_generator.py
@@ -285,8 +285,7 @@
       # names.
       return False
     return (mojom.IsIntegralKind(kind) or mojom.IsFloatKind(kind)
-            or mojom.IsDoubleKind(kind) or mojom.IsStringKind(kind)
-            or mojom.IsEnumKind(kind))
+            or mojom.IsDoubleKind(kind) or mojom.IsStringKind(kind))
 
   def _TypescriptType(self, kind, maybe_nullable=False):
     def recurse_nullable(kind):
@@ -302,10 +301,13 @@
         else:
           return "%s[]" % get_type_name(kind.kind)
 
-      if (mojom.IsMapKind(kind) and self._IsStringableKind(kind.key_kind)
-          and not mojom.IsNullableKind(kind.key_kind)):
-        return "{[key: %s]: %s}" % (get_type_name(
-            kind.key_kind), recurse_nullable(kind.value_kind))
+      if (mojom.IsMapKind(kind) and not mojom.IsNullableKind(kind.key_kind)):
+        if mojom.IsEnumKind(kind.key_kind):
+          return "{[key in %s]?: %s}" % (get_type_name(
+              kind.key_kind), recurse_nullable(kind.value_kind))
+        if self._IsStringableKind(kind.key_kind):
+          return "{[key: %s]: %s}" % (get_type_name(
+              kind.key_kind), recurse_nullable(kind.value_kind))
 
       if mojom.IsMapKind(kind):
         return "Map<%s, %s>" % (recurse_nullable(
diff --git a/sandbox/linux/bpf_dsl/bpf_dsl.cc b/sandbox/linux/bpf_dsl/bpf_dsl.cc
index 2173963..87f8323f 100644
--- a/sandbox/linux/bpf_dsl/bpf_dsl.cc
+++ b/sandbox/linux/bpf_dsl/bpf_dsl.cc
@@ -52,9 +52,7 @@
 class TrapResultExprImpl : public internal::ResultExprImpl {
  public:
   TrapResultExprImpl(TrapRegistry::TrapFnc func, const void* arg, bool safe)
-      : func_(func), arg_(reinterpret_cast<uintptr_t>(arg)), safe_(safe) {
-    DCHECK(func_);
-  }
+      : handler_(func, arg, safe) {}
 
   TrapResultExprImpl(const TrapResultExprImpl&) = delete;
   TrapResultExprImpl& operator=(const TrapResultExprImpl&) = delete;
@@ -62,17 +60,14 @@
   ~TrapResultExprImpl() override {}
 
   CodeGen::Node Compile(PolicyCompiler* pc) const override {
-    return pc->Trap(func_, reinterpret_cast<const void*>(arg_), safe_);
+    return pc->Trap(handler_);
   }
 
-  bool HasUnsafeTraps() const override { return safe_ == false; }
-
+  bool HasUnsafeTraps() const override { return !handler_.safe; }
   bool IsDeny() const override { return true; }
 
  private:
-  TrapRegistry::TrapFnc func_;
-  uintptr_t arg_;  // Usually a pointer, but may be a smuggled int.
-  bool safe_;
+  TrapRegistry::Handler handler_;
 };
 
 class IfThenResultExprImpl : public internal::ResultExprImpl {
diff --git a/sandbox/linux/bpf_dsl/policy_compiler.cc b/sandbox/linux/bpf_dsl/policy_compiler.cc
index d3c42644..b94ec6e 100644
--- a/sandbox/linux/bpf_dsl/policy_compiler.cc
+++ b/sandbox/linux/bpf_dsl/policy_compiler.cc
@@ -435,17 +435,15 @@
     // The performance penalty for this extra round-trip to user-space is not
     // actually that bad, as we only ever pay it for denied system calls; and a
     // typical program has very few of these.
-    return Trap(ReturnErrno, reinterpret_cast<void*>(ret & SECCOMP_RET_DATA),
-                true);
+    return Trap(
+        {ReturnErrno, reinterpret_cast<void*>(ret & SECCOMP_RET_DATA), true});
   }
 
   return gen_.MakeInstruction(BPF_RET + BPF_K, ret);
 }
 
-CodeGen::Node PolicyCompiler::Trap(TrapRegistry::TrapFnc fnc,
-                                   const void* aux,
-                                   bool safe) {
-  uint16_t trap_id = registry_->Add(fnc, aux, safe);
+CodeGen::Node PolicyCompiler::Trap(const TrapRegistry::Handler& handler) {
+  uint16_t trap_id = registry_->Add(handler);
   return gen_.MakeInstruction(BPF_RET + BPF_K, SECCOMP_RET_TRAP + trap_id);
 }
 
diff --git a/sandbox/linux/bpf_dsl/policy_compiler.h b/sandbox/linux/bpf_dsl/policy_compiler.h
index 5439c0a..d35862af 100644
--- a/sandbox/linux/bpf_dsl/policy_compiler.h
+++ b/sandbox/linux/bpf_dsl/policy_compiler.h
@@ -60,7 +60,7 @@
 
   // Trap returns a CodeGen::Node to indicate the system call should
   // instead invoke a trap handler.
-  CodeGen::Node Trap(TrapRegistry::TrapFnc fnc, const void* aux, bool safe);
+  CodeGen::Node Trap(const TrapRegistry::Handler& handler);
 
   // MaskedEqual returns a CodeGen::Node that represents a conditional branch.
   // Argument "argno" (1..6) will be bitwise-AND'd with "mask" and compared
diff --git a/sandbox/linux/bpf_dsl/test_trap_registry.cc b/sandbox/linux/bpf_dsl/test_trap_registry.cc
index bf546d3f..2b71fce 100644
--- a/sandbox/linux/bpf_dsl/test_trap_registry.cc
+++ b/sandbox/linux/bpf_dsl/test_trap_registry.cc
@@ -11,14 +11,16 @@
 namespace sandbox {
 namespace bpf_dsl {
 
-TestTrapRegistry::TestTrapRegistry() : map_() {}
-TestTrapRegistry::~TestTrapRegistry() {}
+TestTrapRegistry::TestTrapRegistry() = default;
 
-uint16_t TestTrapRegistry::Add(TrapFnc fnc, const void* aux, bool safe) {
-  EXPECT_TRUE(safe);
+TestTrapRegistry::~TestTrapRegistry() = default;
+
+uint16_t TestTrapRegistry::Add(const Handler& handler) {
+  EXPECT_TRUE(handler.safe);
 
   const uint16_t next_id = map_.size() + 1;
-  return map_.insert(std::make_pair(Key(fnc, aux), next_id)).first->second;
+  auto result = map_.insert({handler, next_id});
+  return result.first->second;  // Old value if pre-existing handler.
 }
 
 bool TestTrapRegistry::EnableUnsafeTraps() {
diff --git a/sandbox/linux/bpf_dsl/test_trap_registry.h b/sandbox/linux/bpf_dsl/test_trap_registry.h
index 5c080bd..e51b440fc 100644
--- a/sandbox/linux/bpf_dsl/test_trap_registry.h
+++ b/sandbox/linux/bpf_dsl/test_trap_registry.h
@@ -24,13 +24,11 @@
 
   virtual ~TestTrapRegistry();
 
-  uint16_t Add(TrapFnc fnc, const void* aux, bool safe) override;
+  uint16_t Add(const Handler& handler) override;
   bool EnableUnsafeTraps() override;
 
  private:
-  using Key = std::pair<TrapFnc, const void*>;
-
-  std::map<Key, uint16_t> map_;
+  std::map<Handler, uint16_t> map_;
 };
 
 }  // namespace bpf_dsl
diff --git a/sandbox/linux/bpf_dsl/test_trap_registry_unittest.cc b/sandbox/linux/bpf_dsl/test_trap_registry_unittest.cc
index eaf2657..76ec007 100644
--- a/sandbox/linux/bpf_dsl/test_trap_registry_unittest.cc
+++ b/sandbox/linux/bpf_dsl/test_trap_registry_unittest.cc
@@ -39,7 +39,7 @@
   for (int i = 0; i < 2; ++i) {
     for (size_t j = 0; j < std::size(funcs); ++j) {
       // Trap IDs start at 1.
-      EXPECT_EQ(j + 1, traps.Add(funcs[j].fnc, funcs[j].aux, true));
+      EXPECT_EQ(j + 1, traps.Add({funcs[j].fnc, funcs[j].aux, true}));
     }
   }
 }
diff --git a/sandbox/linux/bpf_dsl/trap_registry.h b/sandbox/linux/bpf_dsl/trap_registry.h
index 00bf79c9..99bd439 100644
--- a/sandbox/linux/bpf_dsl/trap_registry.h
+++ b/sandbox/linux/bpf_dsl/trap_registry.h
@@ -7,6 +7,7 @@
 
 #include <stdint.h>
 
+#include "base/check.h"
 #include "sandbox/linux/system_headers/linux_seccomp.h"
 #include "sandbox/sandbox_export.h"
 
@@ -33,6 +34,22 @@
   // http://pubs.opengroup.org/onlinepubs/009695399/functions/xsh_chap02_04.html
   typedef intptr_t (*TrapFnc)(const struct arch_seccomp_data& args, void* aux);
 
+  struct Handler {
+    Handler() = default;
+    Handler(TrapFnc f, const void* a, bool s)
+        : fnc(f), aux(reinterpret_cast<uintptr_t>(a)), safe(s) {
+      DCHECK(fnc);
+    }
+
+    bool operator<(const Handler& that) const {
+      return std::tie(fnc, aux, safe) < std::tie(that.fnc, that.aux, that.safe);
+    }
+
+    TrapFnc fnc = nullptr;
+    uintptr_t aux = 0;  // Usually a pointer, but may be a smuggled int.
+    bool safe = false;
+  };
+
   TrapRegistry(const TrapRegistry&) = delete;
   TrapRegistry& operator=(const TrapRegistry&) = delete;
 
@@ -40,7 +57,7 @@
   // non-zero trap ID that uniquely identifies the tuple for the life
   // time of the trap registry. If the same tuple is registered
   // multiple times, the same value will be returned each time.
-  virtual uint16_t Add(TrapFnc fnc, const void* aux, bool safe) = 0;
+  virtual uint16_t Add(const Handler& handler) = 0;
 
   // EnableUnsafeTraps tries to enable unsafe traps and returns
   // whether it was successful. This is a one-way operation.
diff --git a/sandbox/linux/seccomp-bpf/trap.cc b/sandbox/linux/seccomp-bpf/trap.cc
index f381216..7f262aa 100644
--- a/sandbox/linux/seccomp-bpf/trap.cc
+++ b/sandbox/linux/seccomp-bpf/trap.cc
@@ -227,7 +227,7 @@
                        SECCOMP_PARM6(ctx));
 #endif  // defined(__mips__)
   } else {
-    const TrapKey& trap = trap_array_[info->si_errno - 1];
+    const auto& trap = trap_array_[info->si_errno - 1];
     if (!trap.safe) {
       SetIsInSigHandler();
     }
@@ -260,12 +260,8 @@
   return;
 }
 
-bool Trap::TrapKey::operator<(const TrapKey& o) const {
-  return std::tie(fnc, aux, safe) < std::tie(o.fnc, o.aux, o.safe);
-}
-
-uint16_t Trap::Add(TrapFnc fnc, const void* aux, bool safe) {
-  if (!safe && !SandboxDebuggingAllowedByUser()) {
+uint16_t Trap::Add(const Handler& handler) {
+  if (!handler.safe && !SandboxDebuggingAllowedByUser()) {
     // Unless the user set the CHROME_SANDBOX_DEBUGGING environment variable,
     // we never return an ErrorCode that is marked as "unsafe". This also
     // means, the BPF compiler will never emit code that allow unsafe system
@@ -282,10 +278,6 @@
         "is enabled");
   }
 
-  // Each unique pair of TrapFnc and auxiliary data make up a distinct instance
-  // of a SECCOMP_RET_TRAP.
-  TrapKey key(fnc, aux, safe);
-
   // We return unique identifiers together with SECCOMP_RET_TRAP. This allows
   // us to associate trap with the appropriate handler. The kernel allows us
   // identifiers in the range from 0 to SECCOMP_RET_DATA (0xFFFF). We want to
@@ -295,7 +287,7 @@
   // calls that might be async-signal-unsafe.
   // In order to do so, we store all of our traps in a C-style trap_array_.
 
-  TrapIds::const_iterator iter = trap_ids_.find(key);
+  auto iter = trap_ids_.find(handler);
   if (iter != trap_ids_.end()) {
     // We have seen this pair before. Return the same id that we assigned
     // earlier.
@@ -328,8 +320,8 @@
   // events.
   if (trap_array_size_ >= trap_array_capacity_) {
     trap_array_capacity_ += kCapacityIncrement;
-    TrapKey* old_trap_array = trap_array_;
-    TrapKey* new_trap_array = new TrapKey[trap_array_capacity_];
+    auto* old_trap_array = trap_array_;
+    auto* new_trap_array = new TrapRegistry::Handler[trap_array_capacity_];
     std::copy_n(old_trap_array, trap_array_size_, new_trap_array);
 
     trap_array_ = new_trap_array;
@@ -341,8 +333,8 @@
   }
 
   uint16_t id = trap_array_size_ + 1;
-  trap_ids_[key] = id;
-  trap_array_[trap_array_size_] = key;
+  trap_ids_[handler] = id;
+  trap_array_[trap_array_size_] = handler;
   trap_array_size_++;
   return id;
 }
diff --git a/sandbox/linux/seccomp-bpf/trap.h b/sandbox/linux/seccomp-bpf/trap.h
index 67df58c..60a1ead 100644
--- a/sandbox/linux/seccomp-bpf/trap.h
+++ b/sandbox/linux/seccomp-bpf/trap.h
@@ -33,8 +33,7 @@
   Trap(const Trap&) = delete;
   Trap& operator=(const Trap&) = delete;
 
-  uint16_t Add(TrapFnc fnc, const void* aux, bool safe) override;
-
+  uint16_t Add(const Handler& handler) override;
   bool EnableUnsafeTraps() override;
 
   // Registry returns the trap registry used by Trap's SIGSYS handler,
@@ -46,18 +45,7 @@
   static bool SandboxDebuggingAllowedByUser();
 
  private:
-  struct TrapKey {
-    TrapKey() = default;
-    TrapKey(TrapFnc f, const void* a, bool s)
-        : fnc(f), aux(reinterpret_cast<uintptr_t>(a)), safe(s) {}
-
-    bool operator<(const TrapKey&) const;
-
-    TrapFnc fnc = nullptr;
-    uintptr_t aux = 0;  // Usually a pointer, but may be a smuggled int.
-    bool safe = false;
-  };
-  using TrapIds = std::map<TrapKey, uint16_t>;
+  using HandlerToIdMap = std::map<TrapRegistry::Handler, uint16_t>;
 
   // Our constructor is private. A shared global instance is created
   // automatically as needed.
@@ -73,18 +61,20 @@
   // dumps.
   void SigSys(int nr, LinuxSigInfo* info, ucontext_t* ctx)
       __attribute__((noinline));
+
   // We have a global singleton that handles all of our SIGSYS traps. This
   // variable must never be deallocated after it has been set up initially, as
   // there is no way to reset in-kernel BPF filters that generate SIGSYS
   // events.
   static Trap* global_trap_;
 
-  TrapIds trap_ids_;  // Maps from TrapKeys to numeric ids
-  // Array of TrapKeys indexed by ids.
+  HandlerToIdMap trap_ids_;  // Maps from Handlers to numeric ids
+
+  // Array of handlers indexed by ids.
   //
-  // This is not a raw_ptr as it is an owning pointer anyway, and is meant to be
-  // used between normal code and signal handlers.
-  RAW_PTR_EXCLUSION TrapKey* trap_array_ = nullptr;
+  // This is not a raw_ptr as it is an owning pointer anyway, and needs
+  // to be safe for signal handlers.
+  RAW_PTR_EXCLUSION TrapRegistry::Handler* trap_array_ = nullptr;
   size_t trap_array_size_ = 0;      // Currently used size of array
   size_t trap_array_capacity_ = 0;  // Currently allocated capacity of array
   bool has_unsafe_traps_ = false;   // Whether unsafe traps have been enabled
diff --git a/services/network/public/cpp/features.cc b/services/network/public/cpp/features.cc
index de04861b..5ad8138 100644
--- a/services/network/public/cpp/features.cc
+++ b/services/network/public/cpp/features.cc
@@ -325,7 +325,7 @@
 
 BASE_FEATURE(kPrefetchNoVarySearch,
              "PrefetchNoVarySearch",
-             base::FEATURE_DISABLED_BY_DEFAULT);
+             base::FEATURE_ENABLED_BY_DEFAULT);
 
 BASE_FEATURE(kPrerender2ContentSecurityPolicyExtensions,
              "Prerender2ContentSecurityPolicyExtensions",
diff --git a/services/network/public/cpp/parsed_headers.cc b/services/network/public/cpp/parsed_headers.cc
index f1f24c7..a8fa17e 100644
--- a/services/network/public/cpp/parsed_headers.cc
+++ b/services/network/public/cpp/parsed_headers.cc
@@ -101,6 +101,10 @@
     }
   }
 
+  // We're not checking that PrefetchNoVarySearch is enabled on the
+  // renderer side through the Origin Trial, as the network service
+  // doesn't know anything about blink.
+  // The code here only parses the No-Vary-Search header if it is present.
   if (base::FeatureList::IsEnabled(network::features::kPrefetchNoVarySearch))
     parsed_headers->no_vary_search = ParseNoVarySearch(*headers);
 
diff --git a/services/network/public/cpp/parsed_headers_unittest.cc b/services/network/public/cpp/parsed_headers_unittest.cc
index 9367be2..63d4e1d 100644
--- a/services/network/public/cpp/parsed_headers_unittest.cc
+++ b/services/network/public/cpp/parsed_headers_unittest.cc
@@ -32,7 +32,16 @@
 
 class NoVarySearchPrefetchDisabledTest
     : public ::testing::Test,
-      public ::testing::WithParamInterface<base::StringPiece> {};
+      public ::testing::WithParamInterface<base::StringPiece> {
+ public:
+  NoVarySearchPrefetchDisabledTest() {
+    scoped_feature_list_.InitAndDisableFeature(
+        network::features::kPrefetchNoVarySearch);
+  }
+
+ private:
+  base::test::ScopedFeatureList scoped_feature_list_;
+};
 
 TEST_P(NoVarySearchPrefetchDisabledTest, ParsingNVSReturnsDefaultURLVariance) {
   TestNVSDefaultURLVariance(GetParam());
diff --git a/services/network/web_transport.cc b/services/network/web_transport.cc
index a0e1ac53..2ccb558 100644
--- a/services/network/web_transport.cc
+++ b/services/network/web_transport.cc
@@ -217,7 +217,7 @@
 
   void Send() {
     MaySendFin();
-    while (outgoing_ && outgoing_->CanWrite()) {
+    while (readable_ && outgoing_ && outgoing_->CanWrite()) {
       const void* data = nullptr;
       uint32_t available = 0;
       MojoResult result = readable_->BeginReadData(
@@ -250,7 +250,7 @@
   }
 
   void MaySendFin() {
-    if (!outgoing_) {
+    if (!readable_ || !outgoing_) {
       return;
     }
     if (!has_seen_end_of_pipe_for_readable_ || !has_received_fin_from_client_) {
diff --git a/skia/config/SkUserConfig.h b/skia/config/SkUserConfig.h
index abc3086..50670b75 100644
--- a/skia/config/SkUserConfig.h
+++ b/skia/config/SkUserConfig.h
@@ -220,6 +220,9 @@
 // skia changes to implement new api is completed.
 #define SK_USE_LEGACY_VMA_MEMORY_QUERY
 
+// Temporary until web tests can be rebaselined (skbug.com/13752)
+#define SK_DISABLE_RASTER_PIPELINE_SAMPLING_FIXES
+
 ///////////////////////// Imported from BUILD.gn and skia_common.gypi
 
 /* In some places Skia can use static initializers for global initialization,
diff --git a/testing/buildbot/chromium.perf.json b/testing/buildbot/chromium.perf.json
index ad0ef524..33b2164 100644
--- a/testing/buildbot/chromium.perf.json
+++ b/testing/buildbot/chromium.perf.json
@@ -361,9 +361,8 @@
           "can_use_on_swarming_builders": true,
           "dimension_sets": [
             {
-              "device_os": "MASTER",
               "device_os_flavor": "google",
-              "device_type": "wembley",
+              "device_type": "wembley_2GB",
               "os": "Android",
               "pool": "chrome.tests.perf"
             }
diff --git a/testing/buildbot/client.v8.fyi.json b/testing/buildbot/client.v8.fyi.json
index 66122af05..38ffd154 100644
--- a/testing/buildbot/client.v8.fyi.json
+++ b/testing/buildbot/client.v8.fyi.json
@@ -514,6 +514,41 @@
       },
       {
         "args": [
+          "mediapipe",
+          "--show-stdout",
+          "--browser=release",
+          "--passthrough",
+          "-v",
+          "--stable-jobs",
+          "--extra-browser-args=--enable-logging=stderr --js-flags=--expose-gc --force_higher_performance_gpu --use-cmd-decoder=passthrough --use-gl=angle"
+        ],
+        "isolate_name": "telemetry_gpu_integration_test",
+        "merge": {
+          "args": [],
+          "script": "//testing/merge_scripts/standard_isolated_script_merge.py"
+        },
+        "name": "mediapipe_passthrough_tests",
+        "resultdb": {
+          "enable": true,
+          "has_native_resultdb_integration": true
+        },
+        "should_retry_with_patch": false,
+        "swarming": {
+          "can_use_on_swarming_builders": true,
+          "dimension_sets": [
+            {
+              "gpu": "10de:2184-440.100",
+              "os": "Ubuntu-18.04.5|Ubuntu-18.04.6",
+              "pool": "chromium.tests.gpu"
+            }
+          ],
+          "idempotent": false,
+          "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com"
+        },
+        "test_id_prefix": "ninja://chrome/test:telemetry_gpu_integration_test/"
+      },
+      {
+        "args": [
           "pixel",
           "--show-stdout",
           "--browser=release",
@@ -629,13 +664,96 @@
       },
       {
         "args": [
+          "pixel",
+          "--show-stdout",
+          "--browser=release",
+          "--passthrough",
+          "-v",
+          "--stable-jobs",
+          "--extra-browser-args=--enable-logging=stderr --js-flags=--expose-gc --use-vulkan=native --disable-vulkan-fallback-to-gl-for-testing --enable-features=Vulkan --use-gl=angle --use-angle=gl --use-cmd-decoder=passthrough",
+          "--dont-restore-color-profile-after-test",
+          "--test-machine-name",
+          "${buildername}",
+          "--git-revision=${got_cr_revision}"
+        ],
+        "isolate_name": "telemetry_gpu_integration_test",
+        "merge": {
+          "args": [],
+          "script": "//testing/merge_scripts/standard_isolated_script_merge.py"
+        },
+        "name": "vulkan_pixel_skia_gold_test",
+        "precommit_args": [
+          "--gerrit-issue=${patch_issue}",
+          "--gerrit-patchset=${patch_set}",
+          "--buildbucket-id=${buildbucket_build_id}"
+        ],
+        "resultdb": {
+          "enable": true,
+          "has_native_resultdb_integration": true
+        },
+        "should_retry_with_patch": false,
+        "swarming": {
+          "can_use_on_swarming_builders": true,
+          "dimension_sets": [
+            {
+              "gpu": "10de:2184-440.100",
+              "os": "Ubuntu-18.04.5|Ubuntu-18.04.6",
+              "pool": "chromium.tests.gpu"
+            }
+          ],
+          "idempotent": false,
+          "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com"
+        },
+        "test_id_prefix": "ninja://chrome/test:telemetry_gpu_integration_test/"
+      },
+      {
+        "args": [
+          "webgl2_conformance",
+          "--show-stdout",
+          "--browser=release",
+          "--passthrough",
+          "-v",
+          "--stable-jobs",
+          "--extra-browser-args=--enable-logging=stderr --js-flags=--expose-gc --use-gl=angle --use-angle=gl --use-cmd-decoder=passthrough --force_high_performance_gpu",
+          "--webgl-conformance-version=2.0.1",
+          "--read-abbreviated-json-results-from=../../content/test/data/gpu/webgl2_conformance_tests_output.json",
+          "--jobs=4"
+        ],
+        "isolate_name": "telemetry_gpu_integration_test",
+        "merge": {
+          "args": [],
+          "script": "//testing/merge_scripts/standard_isolated_script_merge.py"
+        },
+        "name": "webgl2_conformance_gl_passthrough_tests",
+        "resultdb": {
+          "enable": true,
+          "has_native_resultdb_integration": true
+        },
+        "should_retry_with_patch": false,
+        "swarming": {
+          "can_use_on_swarming_builders": true,
+          "dimension_sets": [
+            {
+              "gpu": "10de:2184-440.100",
+              "os": "Ubuntu-18.04.5|Ubuntu-18.04.6",
+              "pool": "chromium.tests.gpu"
+            }
+          ],
+          "idempotent": false,
+          "service_account": "chromium-tester@chops-service-accounts.iam.gserviceaccount.com",
+          "shards": 20
+        },
+        "test_id_prefix": "ninja://chrome/test:telemetry_gpu_integration_test/"
+      },
+      {
+        "args": [
           "webgl1_conformance",
           "--show-stdout",
           "--browser=release",
           "--passthrough",
           "-v",
           "--stable-jobs",
-          "--extra-browser-args=--enable-logging=stderr --js-flags=--expose-gc --force_high_performance_gpu",
+          "--extra-browser-args=--enable-logging=stderr --js-flags=--expose-gc --use-gl=angle --use-angle=gl --use-cmd-decoder=passthrough --force_high_performance_gpu",
           "--read-abbreviated-json-results-from=../../content/test/data/gpu/webgl_conformance_tests_output.json",
           "--jobs=4"
         ],
@@ -644,7 +762,7 @@
           "args": [],
           "script": "//testing/merge_scripts/standard_isolated_script_merge.py"
         },
-        "name": "webgl_conformance_tests",
+        "name": "webgl_conformance_gl_passthrough_tests",
         "resultdb": {
           "enable": true,
           "has_native_resultdb_integration": true
diff --git a/testing/buildbot/test_suite_exceptions.pyl b/testing/buildbot/test_suite_exceptions.pyl
index f982529..8a4ae8b 100644
--- a/testing/buildbot/test_suite_exceptions.pyl
+++ b/testing/buildbot/test_suite_exceptions.pyl
@@ -3658,6 +3658,17 @@
       'VR Linux',
     ],
   },
+  'vulkan_pixel_skia_gold_test': {
+    'replacements': {
+      # The V8 builders pass the V8 revision for ${got_revision}, so instead
+      # use ${got_cr_revision}, which is only set on the V8 bots.
+      'Linux V8 FYI Release (NVIDIA)': {
+        'args': {
+          '--git-revision': '${got_cr_revision}',
+        },
+      },
+    },
+  },
   'wayland_client_perftests': {
     'remove_from': [
       'linux-chromeos-dbg',  # https://crbug.com/859307
diff --git a/testing/buildbot/waterfalls.pyl b/testing/buildbot/waterfalls.pyl
index e10483b..eb0efd105 100644
--- a/testing/buildbot/waterfalls.pyl
+++ b/testing/buildbot/waterfalls.pyl
@@ -6318,7 +6318,7 @@
           'linux_nvidia_gtx_1660_stable',
         ],
         'test_suites': {
-          'gpu_telemetry_tests': 'gpu_v8_desktop_passthrough_telemetry_tests',
+          'gpu_telemetry_tests': 'gpu_fyi_linux_release_vulkan_telemetry_tests',
         },
       },
       'Linux V8 FYI Release - pointer compression (NVIDIA)': {
diff --git a/testing/variations/fieldtrial_testing_config.json b/testing/variations/fieldtrial_testing_config.json
index efb322c..40e21512 100644
--- a/testing/variations/fieldtrial_testing_config.json
+++ b/testing/variations/fieldtrial_testing_config.json
@@ -775,6 +775,33 @@
                     "enable_features": [
                         "AssistMultiWord"
                     ]
+                },
+                {
+                    "name": "EnabledWithGboard_20221213",
+                    "params": {
+                        "group": "gboard"
+                    },
+                    "enable_features": [
+                        "AssistMultiWord"
+                    ]
+                },
+                {
+                    "name": "EnabledWithGboardD_20221213",
+                    "params": {
+                        "group": "gboard_d"
+                    },
+                    "enable_features": [
+                        "AssistMultiWord"
+                    ]
+                },
+                {
+                    "name": "EnabledWithGboardE_20221213",
+                    "params": {
+                        "group": "gboard_e"
+                    },
+                    "enable_features": [
+                        "AssistMultiWord"
+                    ]
                 }
             ]
         }
@@ -904,6 +931,21 @@
             ]
         }
     ],
+    "AutocorrectByDefault": [
+        {
+            "platforms": [
+                "chromeos"
+            ],
+            "experiments": [
+                {
+                    "name": "Enabled_20221208",
+                    "enable_features": [
+                        "AutocorrectByDefault"
+                    ]
+                }
+            ]
+        }
+    ],
     "AutofillAcrossIframes": [
         {
             "platforms": [
@@ -3174,30 +3216,6 @@
             ]
         }
     ],
-    "ContextualPageActions": [
-        {
-            "platforms": [
-                "android"
-            ],
-            "experiments": [
-                {
-                    "name": "Enabled",
-                    "params": {
-                        "action_chip": "true",
-                        "action_chip_time_ms": "6000",
-                        "action_chip_with_different_color": "false",
-                        "enable_ui": "true",
-                        "reader_mode_session_rate_limiting": "true"
-                    },
-                    "enable_features": [
-                        "ContextualPageActionPriceTracking",
-                        "ContextualPageActionReaderMode",
-                        "ContextualPageActions"
-                    ]
-                }
-            ]
-        }
-    ],
     "CrOSBluetoothLLPrivacy": [
         {
             "platforms": [
@@ -5410,21 +5428,6 @@
             ]
         }
     ],
-    "HttpsOnlyMode": [
-        {
-            "platforms": [
-                "ios"
-            ],
-            "experiments": [
-                {
-                    "name": "Enabled",
-                    "enable_features": [
-                        "HttpsOnlyMode"
-                    ]
-                }
-            ]
-        }
-    ],
     "IOSAutofillBranding": [
         {
             "platforms": [
@@ -7391,7 +7394,7 @@
                 {
                     "name": "Enabled",
                     "params": {
-                        "omnibox_answer_color_reversal_countries": "zh-CN,zh-TW,ja-JP,ko-KR",
+                        "omnibox_answer_color_reversal_countries": "ja-JP,ko-KR,zh-CN,zh-TW",
                         "omnibox_answer_color_reversal_finance_only": "true"
                     },
                     "enable_features": [
diff --git a/third_party/blink/public/mojom/speculation_rules/speculation_rules.mojom b/third_party/blink/public/mojom/speculation_rules/speculation_rules.mojom
index 99d0cc2..5acb443 100644
--- a/third_party/blink/public/mojom/speculation_rules/speculation_rules.mojom
+++ b/third_party/blink/public/mojom/speculation_rules/speculation_rules.mojom
@@ -22,6 +22,8 @@
   // sent.
   UpdateSpeculationCandidates(
       array<SpeculationCandidate> candidates);
+  // Enables matching prefetches according to No-Vary-Search response headers.
+  EnableNoVarySearchSupport();
 };
 
 // The action that is proposed.
diff --git a/third_party/blink/renderer/core/fetch/request.cc b/third_party/blink/renderer/core/fetch/request.cc
index 8de39af4..ae9eb2e50 100644
--- a/third_party/blink/renderer/core/fetch/request.cc
+++ b/third_party/blink/renderer/core/fetch/request.cc
@@ -793,6 +793,14 @@
     input_request->BodyBuffer()->CloseAndLockAndDisturb();
   }
 
+  // Back/forward-cache is interested in use of the "Authorization" header.
+  if (r->getHeaders() &&
+      r->getHeaders()->has("Authorization", exception_state)) {
+    execution_context->GetScheduler()->RegisterStickyFeature(
+        SchedulingPolicy::Feature::kAuthorizationHeader,
+        {SchedulingPolicy::DisableBackForwardCache()});
+  }
+
   // "Return |r|."
   return r;
 }
diff --git a/third_party/blink/renderer/core/layout/ng/ng_container_fragment_builder.cc b/third_party/blink/renderer/core/layout/ng/ng_container_fragment_builder.cc
index 0955b9b..f38352a 100644
--- a/third_party/blink/renderer/core/layout/ng/ng_container_fragment_builder.cc
+++ b/third_party/blink/renderer/core/layout/ng/ng_container_fragment_builder.cc
@@ -512,9 +512,9 @@
         fixedpos_containing_block_offset = converter.ToLogical(
             multicol_info->fixedpos_containing_block.Offset(),
             fixedpos_containing_block_fragment->Size());
-        fixedpos_containing_block_rel_offset = converter.ToLogical(
+        fixedpos_containing_block_rel_offset = RelativeInsetToLogical(
             multicol_info->fixedpos_containing_block.RelativeOffset(),
-            fixedpos_containing_block_fragment->Size());
+            GetWritingDirection());
         fixedpos_containing_block_rel_offset += relative_offset;
         // We want the fixedpos containing block offset to be the offset from
         // the containing block to the top of the fragmentation context root,
@@ -612,9 +612,8 @@
     LogicalOffset containing_block_offset =
         converter.ToLogical(descendant.containing_block.Offset(),
                             containing_block_fragment->Size());
-    LogicalOffset containing_block_rel_offset =
-        converter.ToLogical(descendant.containing_block.RelativeOffset(),
-                            containing_block_fragment->Size());
+    LogicalOffset containing_block_rel_offset = RelativeInsetToLogical(
+        descendant.containing_block.RelativeOffset(), GetWritingDirection());
     containing_block_rel_offset += relative_offset;
     if (!fragment.IsFragmentainerBox())
       containing_block_offset += offset;
@@ -658,9 +657,9 @@
       fixedpos_containing_block_offset =
           converter.ToLogical(descendant.fixedpos_containing_block.Offset(),
                               fixedpos_containing_block_fragment->Size());
-      fixedpos_containing_block_rel_offset = converter.ToLogical(
+      fixedpos_containing_block_rel_offset = RelativeInsetToLogical(
           descendant.fixedpos_containing_block.RelativeOffset(),
-          fixedpos_containing_block_fragment->Size());
+          GetWritingDirection());
       fixedpos_containing_block_rel_offset += relative_offset;
       if (!fragment.IsFragmentainerBox())
         fixedpos_containing_block_offset += offset;
diff --git a/third_party/blink/renderer/core/layout/ng/ng_out_of_flow_layout_part.cc b/third_party/blink/renderer/core/layout/ng/ng_out_of_flow_layout_part.cc
index b1e7189..228df94 100644
--- a/third_party/blink/renderer/core/layout/ng/ng_out_of_flow_layout_part.cc
+++ b/third_party/blink/renderer/core/layout/ng/ng_out_of_flow_layout_part.cc
@@ -1016,9 +1016,9 @@
         fixedpos_containing_block_offset =
             converter.ToLogical(descendant.fixedpos_containing_block.Offset(),
                                 fixedpos_containing_block_fragment->Size());
-        fixedpos_containing_block_rel_offset = converter.ToLogical(
+        fixedpos_containing_block_rel_offset = RelativeInsetToLogical(
             descendant.fixedpos_containing_block.RelativeOffset(),
-            fixedpos_containing_block_fragment->Size());
+            writing_direction);
       }
 
       NGInlineContainer<LogicalOffset> inline_container(
diff --git a/third_party/blink/renderer/core/layout/ng/ng_out_of_flow_positioned_node.h b/third_party/blink/renderer/core/layout/ng/ng_out_of_flow_positioned_node.h
index 7295d86..f1ffccc5 100644
--- a/third_party/blink/renderer/core/layout/ng/ng_out_of_flow_positioned_node.h
+++ b/third_party/blink/renderer/core/layout/ng/ng_out_of_flow_positioned_node.h
@@ -380,6 +380,20 @@
   MulticolCollection multicols_with_pending_oofs;
 };
 
+inline PhysicalOffset RelativeInsetToPhysical(
+    LogicalOffset relative_inset,
+    WritingDirectionMode writing_direction) {
+  return relative_inset.ConvertToPhysical(writing_direction, PhysicalSize(),
+                                          PhysicalSize());
+}
+
+inline LogicalOffset RelativeInsetToLogical(
+    PhysicalOffset relative_inset,
+    WritingDirectionMode writing_direction) {
+  return relative_inset.ConvertToLogical(writing_direction, PhysicalSize(),
+                                         PhysicalSize());
+}
+
 }  // namespace blink
 
 WTF_ALLOW_CLEAR_UNUSED_SLOTS_WITH_MEM_FUNCTIONS(
diff --git a/third_party/blink/renderer/core/layout/ng/ng_physical_box_fragment.cc b/third_party/blink/renderer/core/layout/ng/ng_physical_box_fragment.cc
index cc03ca42..401c132 100644
--- a/third_party/blink/renderer/core/layout/ng/ng_physical_box_fragment.cc
+++ b/third_party/blink/renderer/core/layout/ng/ng_physical_box_fragment.cc
@@ -102,8 +102,8 @@
   return NGContainingBlock<PhysicalOffset>(
       containing_block.Offset().ConvertToPhysical(
           builder->Style().GetWritingDirection(), outer_size, inner_size),
-      containing_block.RelativeOffset().ConvertToPhysical(
-          builder->Style().GetWritingDirection(), outer_size, inner_size),
+      RelativeInsetToPhysical(containing_block.RelativeOffset(),
+                              builder->Style().GetWritingDirection()),
       containing_block.Fragment(), containing_block.IsInsideColumnSpanner(),
       containing_block.RequiresContentBeforeBreaking());
 }
diff --git a/third_party/blink/renderer/core/loader/threadable_loader.cc b/third_party/blink/renderer/core/loader/threadable_loader.cc
index b56f74b8..b0377b28 100644
--- a/third_party/blink/renderer/core/loader/threadable_loader.cc
+++ b/third_party/blink/renderer/core/loader/threadable_loader.cc
@@ -112,13 +112,6 @@
 }
 
 void ThreadableLoader::Start(ResourceRequest request) {
-  // Back/forward-cache is interested in use of the "Authorization" header.
-  if (request.HttpHeaderField("Authorization")) {
-    execution_context_->GetScheduler()->RegisterStickyFeature(
-        SchedulingPolicy::Feature::kAuthorizationHeader,
-        {SchedulingPolicy::DisableBackForwardCache()});
-  }
-
   const auto request_context = request.GetRequestContext();
   if (request.GetMode() == network::mojom::RequestMode::kNoCors) {
     SECURITY_CHECK(cors::IsNoCorsAllowedContext(request_context));
diff --git a/third_party/blink/renderer/core/paint/html_canvas_painter_test.cc b/third_party/blink/renderer/core/paint/html_canvas_painter_test.cc
index b3e7c1c4..ae25b305 100644
--- a/third_party/blink/renderer/core/paint/html_canvas_painter_test.cc
+++ b/third_party/blink/renderer/core/paint/html_canvas_painter_test.cc
@@ -75,8 +75,6 @@
   std::unique_ptr<Canvas2DLayerBridge> bridge = MakeCanvas2DLayerBridge(size);
   element->SetResourceProviderForTesting(nullptr, std::move(bridge), size);
   ASSERT_EQ(context, element->RenderingContext());
-  ASSERT_TRUE(context->IsComposited());
-  ASSERT_TRUE(element->IsAccelerated());
 
   // Force the page to paint.
   element->PreFinalizeFrame();
@@ -84,6 +82,9 @@
   element->PostFinalizeFrame();
   UpdateAllLifecyclePhasesForTest();
 
+  ASSERT_TRUE(context->IsComposited());
+  ASSERT_TRUE(element->IsAccelerated());
+
   // Fetch the layer associated with the <canvas>, and check that it was
   // correctly configured in the layer tree.
   const cc::Layer* layer = context->CcLayer();
diff --git a/third_party/blink/renderer/core/paint/paint_property_tree_builder.cc b/third_party/blink/renderer/core/paint/paint_property_tree_builder.cc
index 9336ba1..b797ed6 100644
--- a/third_party/blink/renderer/core/paint/paint_property_tree_builder.cc
+++ b/third_party/blink/renderer/core/paint/paint_property_tree_builder.cc
@@ -4240,6 +4240,10 @@
     const LayoutObject& object) {
   DCHECK(CanDoDeferredTransformNodeUpdate(object));
 
+  // GeometryMapper depends on paint properties. This is typically called from
+  // the PrePaintTreeWalk, but we may skip that for this direct update.
+  GeometryMapper::ClearCache();
+
   auto& box = To<LayoutBox>(object);
   PhysicalSize size = PhysicalSize(box.Size());
   FragmentData* fragment_data = &object.GetMutableForPainting().FirstFragment();
diff --git a/third_party/blink/renderer/core/paint/paint_property_tree_builder_test.cc b/third_party/blink/renderer/core/paint/paint_property_tree_builder_test.cc
index fb1a0e1f..3c67a49 100644
--- a/third_party/blink/renderer/core/paint/paint_property_tree_builder_test.cc
+++ b/third_party/blink/renderer/core/paint/paint_property_tree_builder_test.cc
@@ -87,12 +87,6 @@
       .PaintProperties();
 }
 
-const GeometryMapperTransformCache&
-PaintPropertyTreeBuilderTest::GetTransformCache(
-    const TransformPaintPropertyNode& transform) {
-  return transform.GetTransformCache();
-}
-
 void PaintPropertyTreeBuilderTest::SetUp() {
   EnableCompositing();
   RenderingTest::SetUp();
@@ -7383,30 +7377,4 @@
   EXPECT_FALSE(properties->Effect()->OutputClip());
 }
 
-TEST_P(PaintPropertyTreeBuilderTest, TransformChangesInvalidateGeometryMapper) {
-  SetBodyInnerHTML(R"HTML(
-    <style>#div { width:10px; height:10px; transform:translateX(9px); }</style>
-    <div id="div" style="transform: translateX(5px);"></div>
-  )HTML");
-
-  const auto* properties = PaintPropertiesForElement("div");
-  const auto& transform_cache = GetTransformCache(*properties->Transform());
-  EXPECT_TRUE(transform_cache.IsValid());
-
-  // Change the transform and ensure the geometry mapper cache is invalidated.
-  auto* div = GetDocument().getElementById("div");
-  div->removeAttribute(html_names::kStyleAttr);
-  UpdateAllLifecyclePhasesExceptPaint();
-  EXPECT_FALSE(transform_cache.IsValid());
-
-  UpdateAllLifecyclePhasesForTest();
-  EXPECT_TRUE(transform_cache.IsValid());
-
-  // Make a color change and ensure the geometry mapper cache is not
-  // invalidated.
-  div->setAttribute(html_names::kStyleAttr, "background: green;");
-  UpdateAllLifecyclePhasesExceptPaint();
-  EXPECT_TRUE(transform_cache.IsValid());
-}
-
 }  // namespace blink
diff --git a/third_party/blink/renderer/core/paint/paint_property_tree_builder_test.h b/third_party/blink/renderer/core/paint/paint_property_tree_builder_test.h
index 419bcbf..76e25ed2 100644
--- a/third_party/blink/renderer/core/paint/paint_property_tree_builder_test.h
+++ b/third_party/blink/renderer/core/paint/paint_property_tree_builder_test.h
@@ -13,7 +13,6 @@
 namespace blink {
 
 class ClipPaintPropertyNode;
-class GeometryMapperTransformCache;
 class ScrollPaintPropertyNode;
 class TransformPaintPropertyNode;
 struct PhysicalOffset;
@@ -42,9 +41,6 @@
 
   const ObjectPaintProperties* PaintPropertiesForElement(const char* name);
 
-  const GeometryMapperTransformCache& GetTransformCache(
-      const TransformPaintPropertyNode&);
-
   static unsigned NumFragments(const LayoutObject* obj) {
     unsigned count = 0;
     auto* fragment = &obj->FirstFragment();
diff --git a/third_party/blink/renderer/core/paint/pre_paint_tree_walk.cc b/third_party/blink/renderer/core/paint/pre_paint_tree_walk.cc
index 6829a72..9d6f47a 100644
--- a/third_party/blink/renderer/core/paint/pre_paint_tree_walk.cc
+++ b/third_party/blink/renderer/core/paint/pre_paint_tree_walk.cc
@@ -27,6 +27,7 @@
 #include "third_party/blink/renderer/core/paint/object_paint_invalidator.h"
 #include "third_party/blink/renderer/core/paint/paint_layer.h"
 #include "third_party/blink/renderer/core/paint/paint_property_tree_printer.h"
+#include "third_party/blink/renderer/platform/graphics/paint/geometry_mapper.h"
 #include "third_party/blink/renderer/platform/runtime_enabled_features.h"
 
 namespace blink {
@@ -55,10 +56,11 @@
 
   PrePaintTreeWalkContext context;
 
-#if DCHECK_IS_ON()
-  bool needed_tree_builder_context_update =
+  // GeometryMapper depends on paint properties.
+  bool needs_tree_builder_context_update =
       NeedsTreeBuilderContextUpdate(root_frame_view, context);
-#endif
+  if (needs_tree_builder_context_update)
+    GeometryMapper::ClearCache();
 
   VisualViewport& visual_viewport =
       root_frame_view.GetPage()->GetVisualViewport();
@@ -72,7 +74,7 @@
   paint_invalidator_.ProcessPendingDelayedPaintInvalidations();
 
 #if DCHECK_IS_ON()
-  if (needed_tree_builder_context_update && VLOG_IS_ON(1))
+  if (needs_tree_builder_context_update && VLOG_IS_ON(1))
     ShowAllPropertyTrees(root_frame_view);
 #endif
 
diff --git a/third_party/blink/renderer/core/speculation_rules/build.gni b/third_party/blink/renderer/core/speculation_rules/build.gni
index 784aa8c..5a40c52 100644
--- a/third_party/blink/renderer/core/speculation_rules/build.gni
+++ b/third_party/blink/renderer/core/speculation_rules/build.gni
@@ -18,6 +18,7 @@
 ]
 
 blink_core_tests_speculation_rules = [
+  "no_vary_search_origin_trial_test.cc",
   "speculation_rule_set_test.cc",
   "speculation_rules_header_test.cc",
   "speculation_rules_origin_trial_test.cc",
diff --git a/third_party/blink/renderer/core/speculation_rules/document_speculation_rules.cc b/third_party/blink/renderer/core/speculation_rules/document_speculation_rules.cc
index 53343d46..46920683 100644
--- a/third_party/blink/renderer/core/speculation_rules/document_speculation_rules.cc
+++ b/third_party/blink/renderer/core/speculation_rules/document_speculation_rules.cc
@@ -313,6 +313,7 @@
   auto* execution_context = GetSupplementable()->GetExecutionContext();
   if (!execution_context)
     return;
+
   has_pending_update_ = true;
   execution_context->GetAgent()->event_loop()->EnqueueMicrotask(
       WTF::BindOnce(&DocumentSpeculationRules::UpdateSpeculationCandidates,
@@ -389,6 +390,11 @@
   // Add candidates derived from document rule predicates.
   AddLinkBasedSpeculationCandidates(candidates);
 
+  if (!sent_is_part_of_no_vary_search_trial_ &&
+      RuntimeEnabledFeatures::NoVarySearchPrefetchEnabled(execution_context)) {
+    sent_is_part_of_no_vary_search_trial_ = true;
+    host->EnableNoVarySearchSupport();
+  }
   host->UpdateSpeculationCandidates(std::move(candidates));
 }
 
diff --git a/third_party/blink/renderer/core/speculation_rules/document_speculation_rules.h b/third_party/blink/renderer/core/speculation_rules/document_speculation_rules.h
index ebe2986..51836fb 100644
--- a/third_party/blink/renderer/core/speculation_rules/document_speculation_rules.h
+++ b/third_party/blink/renderer/core/speculation_rules/document_speculation_rules.h
@@ -106,6 +106,7 @@
 
   bool has_pending_update_ = false;
   bool initialized_ = false;
+  bool sent_is_part_of_no_vary_search_trial_ = false;
 };
 
 }  // namespace blink
diff --git a/third_party/blink/renderer/core/speculation_rules/no_vary_search_origin_trial_test.cc b/third_party/blink/renderer/core/speculation_rules/no_vary_search_origin_trial_test.cc
new file mode 100644
index 0000000..fe1b2f88
--- /dev/null
+++ b/third_party/blink/renderer/core/speculation_rules/no_vary_search_origin_trial_test.cc
@@ -0,0 +1,141 @@
+// Copyright 2021 The Chromium Authors
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#include <memory>
+
+#include "base/run_loop.h"
+#include "testing/gtest/include/gtest/gtest.h"
+#include "third_party/blink/public/common/browser_interface_broker_proxy.h"
+#include "third_party/blink/public/common/features.h"
+#include "third_party/blink/public/common/features_generated.h"
+#include "third_party/blink/public/common/origin_trials/scoped_test_origin_trial_policy.h"
+#include "third_party/blink/public/platform/web_url_response.h"
+#include "third_party/blink/public/web/web_navigation_params.h"
+#include "third_party/blink/renderer/core/frame/local_dom_window.h"
+#include "third_party/blink/renderer/core/frame/local_frame.h"
+#include "third_party/blink/renderer/core/frame/settings.h"
+#include "third_party/blink/renderer/core/html/html_head_element.h"
+#include "third_party/blink/renderer/core/html/html_script_element.h"
+#include "third_party/blink/renderer/core/speculation_rules/stub_speculation_host.h"
+#include "third_party/blink/renderer/core/testing/dummy_page_holder.h"
+#include "third_party/blink/renderer/platform/weborigin/kurl.h"
+#include "third_party/blink/renderer/platform/wtf/functional.h"
+#include "third_party/blink/renderer/platform/wtf/vector.h"
+
+namespace blink {
+namespace {
+
+void CommitTestNavigation(
+    LocalFrame& frame,
+    const KURL& url,
+    const Vector<std::pair<String, String>>& response_headers) {
+  auto navigation_params = std::make_unique<WebNavigationParams>();
+  navigation_params->url = url;
+  WebNavigationParams::FillStaticResponse(navigation_params.get(), "text/html",
+                                          "UTF-8", "<!DOCTYPE html>");
+  for (const auto& [header, value] : response_headers)
+    navigation_params->response.AddHttpHeaderField(header, value);
+  frame.Loader().CommitNavigation(std::move(navigation_params), nullptr);
+}
+
+HTMLScriptElement* InsertSpeculationRules(Document& document,
+                                          const String& speculation_script) {
+  HTMLScriptElement* script =
+      MakeGarbageCollected<HTMLScriptElement>(document, CreateElementFlags());
+  script->setAttribute(html_names::kTypeAttr, "SpEcUlAtIoNrUlEs");
+  script->setText(speculation_script);
+  document.head()->appendChild(script);
+  return script;
+}
+
+// Generated by:
+//  tools/origin_trials/generate_token.py --version 3 --expire-days 3650
+//  https://speculationrules.test NoVarySearchPrefetch
+// Token details:
+//  Version: 3
+//  Origin: https://speculationrules.test:443
+//  Is Subdomain: None
+//  Is Third Party: None
+//  Usage Restriction: None
+//  Feature: NoVarySearchPrefetch
+//  Expiry: 1985830923 (2032-12-05 03:42:03 UTC)
+//  Signature (Base64):
+//  fFyfaSvsR9K2Wqm/Nvo3KWQsdLUaEGHZj+La5IUXRK/LdCrvtdggLOtoQiEZwkL8rJJz3S+/Mfa6I/LOY0KOCA==
+[[maybe_unused]] constexpr char kNoVarySearchPrefetchToken[] =
+    "A3xcn2kr7EfStlqpvzb6NylkLHS1GhBh2Y/i2uSFF0Svy3Qq77XYICzraEIhGcJC/"
+    "KySc90vvzH2uiPyzmNCjggAAABoeyJvcmlnaW4iOiAiaHR0cHM6Ly9zcGVjdWxhdGlvbnJ1bGV"
+    "zLnRlc3Q6NDQzIiwgImZlYXR1cmUiOiAiTm9WYXJ5U2VhcmNoUHJlZmV0Y2giLCAiZXhwaXJ5I"
+    "jogMTk4NTgzMDkyM30=";
+
+TEST(PrefetchNoVarySearchOriginTrialTest, CanEnableFromToken) {
+  ScopedTestOriginTrialPolicy using_test_keys;
+  DummyPageHolder page_holder;
+  LocalFrame& frame = page_holder.GetFrame();
+
+  CommitTestNavigation(frame, KURL("https://speculationrules.test/"),
+                       {{"Origin-Trial", kNoVarySearchPrefetchToken}});
+
+  // This should have enabled the origin trial and all its dependent features.
+  EXPECT_TRUE(
+      RuntimeEnabledFeatures::NoVarySearchPrefetchEnabled(frame.DomWindow()));
+  EXPECT_TRUE(RuntimeEnabledFeatures::SpeculationRulesPrefetchProxyEnabled(
+      frame.DomWindow()));
+}
+
+TEST(PrefetchNoVarySearchOriginTrialTest, DoesNotEnableWithoutToken) {
+  ScopedTestOriginTrialPolicy using_test_keys;
+  DummyPageHolder page_holder;
+  LocalFrame& frame = page_holder.GetFrame();
+
+  // Do not send the Origin-Trial token.
+  CommitTestNavigation(frame, KURL("https://speculationrules.test/"), {});
+
+  // This should not have enabled the origin trial.
+  EXPECT_FALSE(
+      RuntimeEnabledFeatures::NoVarySearchPrefetchEnabled(frame.DomWindow()));
+}
+
+void NoVarySearchPrefetchEnabledTest(StubSpeculationHost& speculation_host) {
+  DummyPageHolder page_holder;
+  LocalFrame& frame = page_holder.GetFrame();
+  frame.GetSettings()->SetScriptEnabled(true);
+
+  auto& broker = frame.DomWindow()->GetBrowserInterfaceBroker();
+  broker.SetBinderForTesting(
+      mojom::blink::SpeculationHost::Name_,
+      WTF::BindRepeating(&StubSpeculationHost::BindUnsafe,
+                         WTF::Unretained(&speculation_host)));
+
+  base::RunLoop run_loop;
+  speculation_host.SetDoneClosure(run_loop.QuitClosure());
+
+  const String speculation_script =
+      R"({"prefetch": [
+           {"source": "list",
+            "urls": ["https://example.com/foo"],
+            "requires": ["anonymous-client-ip-when-cross-origin"]}
+         ]})";
+  InsertSpeculationRules(page_holder.GetDocument(), speculation_script);
+  run_loop.Run();
+
+  broker.SetBinderForTesting(mojom::blink::SpeculationHost::Name_, {});
+}
+
+TEST(PrefetchNoVarySearchOriginTrialTest,
+     EnabledNoVarySearchPrefetchInBrowser) {
+  ScopedNoVarySearchPrefetchForTest enable_no_vary_search_prefetch_{true};
+  StubSpeculationHost speculation_host;
+  NoVarySearchPrefetchEnabledTest(speculation_host);
+  EXPECT_TRUE(speculation_host.sent_no_vary_search_support_to_browser());
+}
+
+TEST(PrefetchNoVarySearchOriginTrialTest,
+     DoNotEnableNoVarySearchPrefetchInBrowser) {
+  StubSpeculationHost speculation_host;
+  NoVarySearchPrefetchEnabledTest(speculation_host);
+  EXPECT_FALSE(speculation_host.sent_no_vary_search_support_to_browser());
+}
+
+}  // namespace
+}  // namespace blink
diff --git a/third_party/blink/renderer/core/speculation_rules/stub_speculation_host.cc b/third_party/blink/renderer/core/speculation_rules/stub_speculation_host.cc
index 4d6bfdd..21fef772 100644
--- a/third_party/blink/renderer/core/speculation_rules/stub_speculation_host.cc
+++ b/third_party/blink/renderer/core/speculation_rules/stub_speculation_host.cc
@@ -32,4 +32,8 @@
     std::move(done_closure_).Run();
 }
 
+void StubSpeculationHost::EnableNoVarySearchSupport() {
+  sent_no_vary_search_support_to_browser_ = true;
+}
+
 }  // namespace blink
diff --git a/third_party/blink/renderer/core/speculation_rules/stub_speculation_host.h b/third_party/blink/renderer/core/speculation_rules/stub_speculation_host.h
index 41c5f4f2..7226199 100644
--- a/third_party/blink/renderer/core/speculation_rules/stub_speculation_host.h
+++ b/third_party/blink/renderer/core/speculation_rules/stub_speculation_host.h
@@ -37,13 +37,21 @@
   // mojom::blink::SpeculationHost.
   void UpdateSpeculationCandidates(Candidates candidates) override;
 
+  // mojom::blink::SpeculationHost.
+  void EnableNoVarySearchSupport() override;
+
   void OnConnectionLost();
 
   bool is_bound() const { return receiver_.is_bound(); }
 
+  bool sent_no_vary_search_support_to_browser() const {
+    return sent_no_vary_search_support_to_browser_;
+  }
+
  private:
   mojo::Receiver<SpeculationHost> receiver_{this};
   Vector<mojom::blink::SpeculationCandidatePtr> candidates_;
+  bool sent_no_vary_search_support_to_browser_ = false;
   base::OnceClosure done_closure_;
   base::RepeatingCallback<void(const Candidates&)> candidates_updated_callback_;
 };
diff --git a/third_party/blink/renderer/core/timing/performance.cc b/third_party/blink/renderer/core/timing/performance.cc
index 0f343820a..f998630 100644
--- a/third_party/blink/renderer/core/timing/performance.cc
+++ b/third_party/blink/renderer/core/timing/performance.cc
@@ -483,41 +483,49 @@
   if (!window) {
     return entries;
   }
-  LocalFrame* parent_frame = window->GetFrame();
-  if (!parent_frame) {
+  LocalFrame* root_frame = window->GetFrame();
+  if (!root_frame) {
     return entries;
   }
+  const SecurityOrigin* root_origin =
+      ExecutionContext::From(script_state)->GetSecurityOrigin();
 
   HeapDeque<Member<Frame>> queue;
-  queue.push_back(parent_frame);
+  queue.push_back(root_frame);
 
   while (!queue.empty()) {
     Frame* current_frame = queue.TakeFirst();
 
-    if (LocalFrame* local_frame = DynamicTo<LocalFrame>(current_frame);
-        local_frame && !local_frame->IsCrossOriginToNearestMainFrame()) {
+    if (LocalFrame* local_frame = DynamicTo<LocalFrame>(current_frame)) {
       // Get the Performance object from the current frame.
       LocalDOMWindow* current_window = local_frame->DomWindow();
       // As we verified that the frame this was called with is not detached when
       // entring this loop, we can assume that all its children are also not
       // detached, and hence have a window object.
       DCHECK(current_window);
-      WindowPerformance* window_performance =
-          DOMWindowPerformance::performance(*current_window);
 
-      // Get the performance entries based on entry_type input.
-      PerformanceEntryVector current_entries;
-      if (entry_type.IsNull()) {
-        current_entries = window_performance->GetEntriesForCurrentFrame();
-      } else {
-        current_entries =
-            window_performance->GetEntriesByTypeForCurrentFrame(entry_type);
+      // Validate that the child frame's origin is the same as the root
+      // frame.
+      const SecurityOrigin* current_origin =
+          current_window->GetSecurityOrigin();
+      if (root_origin->IsSameOriginWith(current_origin)) {
+        WindowPerformance* window_performance =
+            DOMWindowPerformance::performance(*current_window);
+
+        // Get the performance entries based on entry_type input.
+        PerformanceEntryVector current_entries;
+        if (entry_type.IsNull()) {
+          current_entries = window_performance->GetEntriesForCurrentFrame();
+        } else {
+          current_entries =
+              window_performance->GetEntriesByTypeForCurrentFrame(entry_type);
+        }
+
+        entries.AppendVector(current_entries);
       }
-
-      entries.AppendVector(current_entries);
     }
 
-    // Add both Local and Remote Frames to the queue.
+    // Add both Local and Remote Frame children to the queue.
     for (Frame* child = current_frame->FirstChild(); child;
          child = child->NextSibling()) {
       queue.push_back(child);
diff --git a/third_party/blink/renderer/modules/canvas/canvas2d/canvas_rendering_context_2d.cc b/third_party/blink/renderer/modules/canvas/canvas2d/canvas_rendering_context_2d.cc
index 020b4d9..e46a8fcb 100644
--- a/third_party/blink/renderer/modules/canvas/canvas2d/canvas_rendering_context_2d.cc
+++ b/third_party/blink/renderer/modules/canvas/canvas2d/canvas_rendering_context_2d.cc
@@ -189,7 +189,10 @@
 }
 
 bool CanvasRenderingContext2D::IsComposited() const {
-  return IsAccelerated();
+  if (Canvas2DLayerBridge* layer_bridge = canvas()->GetCanvas2DLayerBridge()) {
+    return layer_bridge->IsComposited();
+  }
+  return false;
 }
 
 void CanvasRenderingContext2D::Stop() {
diff --git a/third_party/blink/renderer/modules/webaudio/async_audio_decoder.cc b/third_party/blink/renderer/modules/webaudio/async_audio_decoder.cc
index 08b2f69a..661b0cea 100644
--- a/third_party/blink/renderer/modules/webaudio/async_audio_decoder.cc
+++ b/third_party/blink/renderer/modules/webaudio/async_audio_decoder.cc
@@ -32,7 +32,6 @@
 #include "third_party/blink/renderer/modules/webaudio/audio_buffer.h"
 #include "third_party/blink/renderer/modules/webaudio/base_audio_context.h"
 #include "third_party/blink/renderer/platform/audio/audio_bus.h"
-#include "third_party/blink/renderer/platform/audio/audio_file_reader.h"
 #include "third_party/blink/renderer/platform/bindings/cross_thread_copier.h"
 #include "third_party/blink/renderer/platform/bindings/exception_context.h"
 #include "third_party/blink/renderer/platform/bindings/exception_state.h"
@@ -79,7 +78,7 @@
     scoped_refptr<base::SingleThreadTaskRunner> task_runner,
     const ExceptionContext& exception_context) {
   DCHECK(!IsMainThread());
-  scoped_refptr<AudioBus> bus = CreateBusFromInMemoryAudioFile(
+  scoped_refptr<AudioBus> bus = AudioBus::CreateBusFromInMemoryAudioFile(
       audio_data->Data(), audio_data->ByteLength(), false, sample_rate);
 
   // Decoding is finished, but we need to do the callbacks on the main thread.
diff --git a/third_party/blink/renderer/modules/webaudio/audio_buffer.cc b/third_party/blink/renderer/modules/webaudio/audio_buffer.cc
index 5933e32..2e03280 100644
--- a/third_party/blink/renderer/modules/webaudio/audio_buffer.cc
+++ b/third_party/blink/renderer/modules/webaudio/audio_buffer.cc
@@ -33,7 +33,6 @@
 #include "third_party/blink/renderer/bindings/modules/v8/v8_audio_buffer_options.h"
 #include "third_party/blink/renderer/modules/webaudio/base_audio_context.h"
 #include "third_party/blink/renderer/platform/audio/audio_bus.h"
-#include "third_party/blink/renderer/platform/audio/audio_file_reader.h"
 #include "third_party/blink/renderer/platform/audio/audio_utilities.h"
 #include "third_party/blink/renderer/platform/bindings/exception_messages.h"
 #include "third_party/blink/renderer/platform/bindings/exception_state.h"
diff --git a/third_party/blink/renderer/platform/BUILD.gn b/third_party/blink/renderer/platform/BUILD.gn
index 9c01b81..7c8fc318 100644
--- a/third_party/blink/renderer/platform/BUILD.gn
+++ b/third_party/blink/renderer/platform/BUILD.gn
@@ -304,7 +304,6 @@
     "audio/audio_dsp_kernel.h",
     "audio/audio_dsp_kernel_processor.cc",
     "audio/audio_dsp_kernel_processor.h",
-    "audio/audio_file_reader.h",
     "audio/audio_io_callback.h",
     "audio/audio_processor.cc",
     "audio/audio_processor.h",
diff --git a/third_party/blink/renderer/platform/audio/audio_bus.cc b/third_party/blink/renderer/platform/audio/audio_bus.cc
index 9a4d923..e8a48f1 100644
--- a/third_party/blink/renderer/platform/audio/audio_bus.cc
+++ b/third_party/blink/renderer/platform/audio/audio_bus.cc
@@ -37,7 +37,6 @@
 #include "base/ranges/algorithm.h"
 #include "third_party/blink/public/platform/platform.h"
 #include "third_party/blink/public/platform/web_audio_bus.h"
-#include "third_party/blink/renderer/platform/audio/audio_file_reader.h"
 #include "third_party/blink/renderer/platform/audio/denormal_disabler.h"
 #include "third_party/blink/renderer/platform/audio/sinc_resampler.h"
 #include "third_party/blink/renderer/platform/audio/vector_math.h"
@@ -744,10 +743,11 @@
                                                 sample_rate);
 }
 
-scoped_refptr<AudioBus> CreateBusFromInMemoryAudioFile(const void* data,
-                                                       size_t data_size,
-                                                       bool mix_to_mono,
-                                                       float sample_rate) {
+scoped_refptr<AudioBus> AudioBus::CreateBusFromInMemoryAudioFile(
+    const void* data,
+    size_t data_size,
+    bool mix_to_mono,
+    float sample_rate) {
   scoped_refptr<AudioBus> audio_bus =
       DecodeAudioFileData(static_cast<const char*>(data), data_size);
   if (!audio_bus.get()) {
diff --git a/third_party/blink/renderer/platform/audio/audio_bus.h b/third_party/blink/renderer/platform/audio/audio_bus.h
index 3290de3..39c64c0e 100644
--- a/third_party/blink/renderer/platform/audio/audio_bus.h
+++ b/third_party/blink/renderer/platform/audio/audio_bus.h
@@ -70,6 +70,16 @@
                                         uint32_t length,
                                         bool allocate = true);
 
+  // Pass in 0.0 for sampleRate to use the file's sample-rate, otherwise a
+  // sample-rate conversion to the requested sampleRate will be made (if it
+  // doesn't already match the file's sample-rate).  The created buffer will
+  // have its sample-rate set correctly to the result.
+  static scoped_refptr<AudioBus> CreateBusFromInMemoryAudioFile(
+      const void* data,
+      size_t data_size,
+      bool mix_to_mono,
+      float sample_rate);
+
   AudioBus(const AudioBus&) = delete;
   AudioBus& operator=(const AudioBus&) = delete;
 
diff --git a/third_party/blink/renderer/platform/audio/audio_file_reader.h b/third_party/blink/renderer/platform/audio/audio_file_reader.h
deleted file mode 100644
index 8a499b79..0000000
--- a/third_party/blink/renderer/platform/audio/audio_file_reader.h
+++ /dev/null
@@ -1,64 +0,0 @@
-/*
- * Copyright (C) 2010 Google Inc. All rights reserved.
- *
- * Redistribution and use in source and binary forms, with or without
- * modification, are permitted provided that the following conditions
- * are met:
- *
- * 1.  Redistributions of source code must retain the above copyright
- *     notice, this list of conditions and the following disclaimer.
- * 2.  Redistributions in binary form must reproduce the above copyright
- *     notice, this list of conditions and the following disclaimer in the
- *     documentation and/or other materials provided with the distribution.
- * 3.  Neither the name of Apple Computer, Inc. ("Apple") nor the names of
- *     its contributors may be used to endorse or promote products derived
- *     from this software without specific prior written permission.
- *
- * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
- * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
- * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
- * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
- * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
- * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
- * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
- * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
- * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
- * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
- */
-
-#ifndef THIRD_PARTY_BLINK_RENDERER_PLATFORM_AUDIO_AUDIO_FILE_READER_H_
-#define THIRD_PARTY_BLINK_RENDERER_PLATFORM_AUDIO_AUDIO_FILE_READER_H_
-
-#include "base/memory/scoped_refptr.h"
-#include "third_party/blink/renderer/platform/platform_export.h"
-
-namespace blink {
-
-class AudioBus;
-
-// For both create functions:
-// Pass in 0.0 for sampleRate to use the file's sample-rate, otherwise a
-// sample-rate conversion to the requested sampleRate will be made (if it
-// doesn't already match the file's sample-rate).  The created buffer will have
-// its sample-rate set correctly to the result.
-
-PLATFORM_EXPORT scoped_refptr<AudioBus> CreateBusFromInMemoryAudioFile(
-    const void* data,
-    size_t data_size,
-    bool mix_to_mono,
-    float sample_rate);
-
-PLATFORM_EXPORT scoped_refptr<AudioBus> CreateBusFromAudioFile(
-    const char* file_path,
-    bool mix_to_mono,
-    float sample_rate);
-
-// May pass in 0.0 for sampleRate in which case it will use the AudioBus's
-// sampleRate
-PLATFORM_EXPORT void WriteBusToAudioFile(AudioBus* bus,
-                                         const char* file_path,
-                                         double file_sample_rate);
-
-}  // namespace blink
-
-#endif  // THIRD_PARTY_BLINK_RENDERER_PLATFORM_AUDIO_AUDIO_FILE_READER_H_
diff --git a/third_party/blink/renderer/platform/graphics/canvas_2d_layer_bridge.cc b/third_party/blink/renderer/platform/graphics/canvas_2d_layer_bridge.cc
index 262d8ac..ceeb6c8 100644
--- a/third_party/blink/renderer/platform/graphics/canvas_2d_layer_bridge.cc
+++ b/third_party/blink/renderer/platform/graphics/canvas_2d_layer_bridge.cc
@@ -152,6 +152,25 @@
   return ShouldAccelerate();
 }
 
+bool Canvas2DLayerBridge::IsComposited() const {
+  if (IsHibernating()) {
+    return false;
+  }
+
+  if (UNLIKELY(!resource_host_)) {
+    return false;
+  }
+
+  CanvasResourceProvider* resource_provider =
+      resource_host_->ResourceProvider();
+  if (UNLIKELY(!resource_provider)) {
+    return false;
+  }
+
+  return resource_provider->SupportsDirectCompositing() &&
+         !resource_host_->LowLatencyEnabled();
+}
+
 static void HibernateWrapper(base::WeakPtr<Canvas2DLayerBridge> bridge,
                              base::TimeTicks /*idleDeadline*/) {
   if (bridge) {
@@ -268,14 +287,13 @@
 
   if (resource_provider && resource_provider->IsValid()) {
 #if DCHECK_IS_ON()
-    // If resource provider is accelerated, a layer should already exist.
+    // If resource provider is composited, a layer should already exist.
     // unless this is a canvas in low latency mode.
     // If this DCHECK fails, it probably means that
     // CanvasRenderingContextHost::GetOrCreateCanvasResourceProvider() was
     // called on a 2D context before this function.
-    if (IsAccelerated()) {
-      DCHECK(!!layer_ ||
-             (resource_host_ && resource_host_->LowLatencyEnabled()));
+    if (IsComposited()) {
+      DCHECK(!!layer_);
     }
 #endif
     return resource_provider;
@@ -310,7 +328,7 @@
   // TODO crbug/1090081: Check possibility to move DidDraw inside Clear.
   DidDraw();
 
-  if (IsAccelerated() && !layer_) {
+  if (IsComposited() && !layer_) {
     layer_ = cc::TextureLayer::CreateForMailbox(this);
     layer_->SetIsDrawable(true);
     layer_->SetHitTestable(true);
@@ -320,6 +338,7 @@
                                cc::PaintFlags::FilterQuality::kNone);
     layer_->SetHDRConfiguration(resource_host_->GetHDRMode(),
                                 resource_host_->GetHDRMetadata());
+    layer_->SetFlipped(!resource_provider->IsOriginTopLeft());
   }
   // After the page becomes visible and successfully restored the canvas
   // resource provider, set |lose_context_in_background_| to false.
@@ -747,7 +766,7 @@
     constexpr unsigned kMaxCanvasAnimationBacklog = 2;
     if (frames_since_last_commit_ >=
         static_cast<int>(kMaxCanvasAnimationBacklog)) {
-      if (IsAccelerated() && !rate_limiter_) {
+      if (IsComposited() && !rate_limiter_) {
         rate_limiter_ = std::make_unique<SharedContextRateLimiter>(
             kMaxCanvasAnimationBacklog);
       }
@@ -759,8 +778,9 @@
 }
 
 void Canvas2DLayerBridge::DoPaintInvalidation(const gfx::Rect& dirty_rect) {
-  if (layer_ && raster_mode_ == RasterMode::kGPU)
+  if (layer_ && IsComposited()) {
     layer_->SetNeedsDisplayRect(dirty_rect);
+  }
 }
 
 scoped_refptr<StaticBitmapImage> Canvas2DLayerBridge::NewImageSnapshot() {
diff --git a/third_party/blink/renderer/platform/graphics/canvas_2d_layer_bridge.h b/third_party/blink/renderer/platform/graphics/canvas_2d_layer_bridge.h
index af3cb0fe..66a9272 100644
--- a/third_party/blink/renderer/platform/graphics/canvas_2d_layer_bridge.h
+++ b/third_party/blink/renderer/platform/graphics/canvas_2d_layer_bridge.h
@@ -95,6 +95,8 @@
   virtual void DidRestoreCanvasMatrixClipStack(cc::PaintCanvas*) {}
   virtual bool IsAccelerated() const;
 
+  bool IsComposited() const;
+
   // This may recreate CanvasResourceProvider
   cc::PaintCanvas* GetPaintCanvas();
   bool IsValid();
diff --git a/third_party/blink/renderer/platform/graphics/canvas_2d_layer_bridge_test.cc b/third_party/blink/renderer/platform/graphics/canvas_2d_layer_bridge_test.cc
index d620c77..3160461 100644
--- a/third_party/blink/renderer/platform/graphics/canvas_2d_layer_bridge_test.cc
+++ b/third_party/blink/renderer/platform/graphics/canvas_2d_layer_bridge_test.cc
@@ -1019,4 +1019,29 @@
   EXPECT_FALSE(bridge->HasRateLimiterForTesting());
 }
 
+TEST_F(Canvas2DLayerBridgeTest, SoftwareCanvasIsCompositedIfImageChromium) {
+  ScopedTestingPlatformSupport<GpuMemoryBufferTestPlatform> platform;
+  ScopedCanvas2dImageChromiumForTest canvas_2d_image_chromium(true);
+  const_cast<gpu::Capabilities&>(SharedGpuContext::ContextProviderWrapper()
+                                     ->ContextProvider()
+                                     ->GetCapabilities())
+      .gpu_memory_buffer_formats.Add(gfx::BufferFormat::BGRA_8888);
+  std::unique_ptr<Canvas2DLayerBridge> bridge =
+      MakeBridge(gfx::Size(300, 150), RasterMode::kCPU, kNonOpaque);
+  EXPECT_TRUE(bridge->IsValid());
+  DrawSomething(bridge.get());
+  EXPECT_FALSE(bridge->IsAccelerated());
+  EXPECT_TRUE(bridge->IsComposited());
+}
+
+TEST_F(Canvas2DLayerBridgeTest, SoftwareCanvasNotCompositedIfNotImageChromium) {
+  ScopedCanvas2dImageChromiumForTest canvas_2d_image_chromium(false);
+  std::unique_ptr<Canvas2DLayerBridge> bridge =
+      MakeBridge(gfx::Size(300, 150), RasterMode::kCPU, kNonOpaque);
+  EXPECT_TRUE(bridge->IsValid());
+  DrawSomething(bridge.get());
+  EXPECT_FALSE(bridge->IsAccelerated());
+  EXPECT_FALSE(bridge->IsComposited());
+}
+
 }  // namespace blink
diff --git a/third_party/blink/renderer/platform/graphics/canvas_resource.cc b/third_party/blink/renderer/platform/graphics/canvas_resource.cc
index 2989bd8..f0b976c3 100644
--- a/third_party/blink/renderer/platform/graphics/canvas_resource.cc
+++ b/third_party/blink/renderer/platform/graphics/canvas_resource.cc
@@ -130,7 +130,12 @@
     scoped_refptr<CanvasResource>&& resource,
     const gpu::SyncToken& sync_token,
     bool lost_resource) {
-  if (!resource)
+  // If there is a LastUnrefCallback, we need to abort because recycling the
+  // resource now will prevent the LastUnrefCallback from ever being called.
+  // In such cases, ReleaseFrameResources will be called again when
+  // CanvasResourceDispatcher destroys the corresponding FrameResource object,
+  // at which time this resource will be safely recycled.
+  if (!resource || resource->HasLastUnrefCallback())
     return;
 
   resource->WaitSyncToken(sync_token);
@@ -144,8 +149,9 @@
   if (lost_resource)
     resource->NotifyResourceLost();
   if (resource_provider && !lost_resource && resource->IsRecycleable() &&
-      resource->HasOneRef())
+      resource->HasOneRef()) {
     resource_provider->RecycleResource(std::move(resource));
+  }
 }
 
 bool CanvasResource::PrepareTransferableResource(
@@ -699,7 +705,10 @@
   auto surface = SkSurface::MakeRasterDirect(CreateSkImageInfo(),
                                              gpu_memory_buffer_->memory(0),
                                              gpu_memory_buffer_->stride(0));
-  surface->getCanvas()->drawImage(image, 0, 0);
+
+  SkPixmap pixmap;
+  image->peekPixels(&pixmap);
+  surface->writePixels(pixmap, 0, 0);
   auto* sii =
       ContextProviderWrapper()->ContextProvider()->SharedImageInterface();
   gpu_memory_buffer_->Unmap();
diff --git a/third_party/blink/renderer/platform/graphics/canvas_resource.h b/third_party/blink/renderer/platform/graphics/canvas_resource.h
index f80baf6..8a74aa78 100644
--- a/third_party/blink/renderer/platform/graphics/canvas_resource.h
+++ b/third_party/blink/renderer/platform/graphics/canvas_resource.h
@@ -83,6 +83,8 @@
     last_unref_callback_ = std::move(callback);
   }
 
+  bool HasLastUnrefCallback() { return !!last_unref_callback_; }
+
   // We perform a lazy copy on write if the canvas content needs to be updated
   // while its current resource is in use. In order to avoid re-allocating
   // resources, its preferable to reuse a resource if its no longer in use.
diff --git a/third_party/blink/renderer/platform/graphics/canvas_resource_provider_test.cc b/third_party/blink/renderer/platform/graphics/canvas_resource_provider_test.cc
index 4c4c7a25..86fb8ec 100644
--- a/third_party/blink/renderer/platform/graphics/canvas_resource_provider_test.cc
+++ b/third_party/blink/renderer/platform/graphics/canvas_resource_provider_test.cc
@@ -251,6 +251,44 @@
   EXPECT_EQ(original_mailbox, provider->Snapshot()->GetMailboxHolder().mailbox);
 }
 
+TEST_F(CanvasResourceProviderTest, NoRecycleIfLastRefCallback) {
+  const SkImageInfo kInfo = SkImageInfo::MakeN32Premul(10, 10);
+
+  const uint32_t shared_image_usage_flags =
+      gpu::SHARED_IMAGE_USAGE_DISPLAY_READ | gpu::SHARED_IMAGE_USAGE_SCANOUT;
+
+  auto provider = CanvasResourceProvider::CreateSharedImageProvider(
+      kInfo, cc::PaintFlags::FilterQuality::kMedium,
+      CanvasResourceProvider::ShouldInitialize::kCallClear,
+      context_provider_wrapper_, RasterMode::kGPU, true /*is_origin_top_left*/,
+      shared_image_usage_flags);
+
+  ASSERT_TRUE(provider->IsValid());
+
+  scoped_refptr<StaticBitmapImage> snapshot1 = provider->Snapshot();
+  ASSERT_TRUE(snapshot1);
+
+  // Set up a LastUnrefCallback that recycles the resource asynchronously,
+  // similarly to what OffscreenCanvasPlaceholder would do.
+  provider->ProduceCanvasResource()->SetLastUnrefCallback(
+      base::BindOnce([](scoped_refptr<CanvasResource> resource) {}));
+
+  // Resource updated after draw.
+  provider->Canvas()->clear(SkColors::kWhite);
+  provider->FlushCanvas();
+  scoped_refptr<StaticBitmapImage> snapshot2 = provider->Snapshot();
+  EXPECT_NE(snapshot2->GetMailboxHolder().mailbox,
+            snapshot1->GetMailboxHolder().mailbox);
+
+  auto snapshot1_mailbox = snapshot1->GetMailboxHolder().mailbox;
+  snapshot1.reset();  // resource not recycled due to LastUnrefCallback
+  provider->Canvas()->clear(SkColors::kBlack);
+  provider->FlushCanvas();
+  scoped_refptr<StaticBitmapImage> snapshot3 = provider->Snapshot();
+  // confirm resource is not recycled.
+  EXPECT_NE(snapshot3->GetMailboxHolder().mailbox, snapshot1_mailbox);
+}
+
 TEST_F(CanvasResourceProviderTest,
        CanvasResourceProviderSharedImageCopyOnWriteDisabled) {
   auto* fake_context = static_cast<FakeWebGraphicsContext3DProvider*>(
diff --git a/third_party/blink/renderer/platform/graphics/gradient.cc b/third_party/blink/renderer/platform/graphics/gradient.cc
index 315e111..50f2af1 100644
--- a/third_party/blink/renderer/platform/graphics/gradient.cc
+++ b/third_party/blink/renderer/platform/graphics/gradient.cc
@@ -104,7 +104,7 @@
   if (stops_.empty()) {
     // A gradient with no stops must be transparent black.
     pos.push_back(WebCoreDoubleToSkScalar(0));
-    colors.push_back(SK_ColorTRANSPARENT);
+    colors.push_back(SkColors::kTransparent);
   } else if (stops_.front().stop > 0) {
     // Copy the first stop to 0.0. The first stop position may have a slight
     // rounding error, but we don't care in this float comparison, since
@@ -112,20 +112,20 @@
     // with a stop at (0 + epsilon).
     pos.push_back(WebCoreDoubleToSkScalar(0));
     if (color_filter_) {
-      colors.push_back(color_filter_->filterColor(
-          stops_.front().color.ToSkColorDeprecated()));
+      colors.push_back(color_filter_->filterColor4f(
+          stops_.front().color.toSkColor4f(), nullptr, nullptr));
     } else {
-      colors.push_back(stops_.front().color.ToSkColorDeprecated());
+      colors.push_back(stops_.front().color.toSkColor4f());
     }
   }
 
   for (const auto& stop : stops_) {
     pos.push_back(WebCoreDoubleToSkScalar(stop.stop));
     if (color_filter_) {
-      colors.push_back(
-          color_filter_->filterColor(stop.color.ToSkColorDeprecated()));
+      colors.push_back(color_filter_->filterColor4f(stop.color.toSkColor4f(),
+                                                    nullptr, nullptr));
     } else {
-      colors.push_back(stop.color.ToSkColorDeprecated());
+      colors.push_back(stop.color.toSkColor4f());
     }
   }
 
@@ -232,10 +232,8 @@
 
   if (is_dark_mode_enabled_) {
     for (auto& color : colors) {
-      color = EnsureDarkModeFilter()
-                  .InvertColorIfNeeded(SkColor4f::FromColor(color),
-                                       DarkModeFilter::ElementRole::kBackground)
-                  .toSkColor();
+      color = EnsureDarkModeFilter().InvertColorIfNeeded(
+          color, DarkModeFilter::ElementRole::kBackground);
     }
   }
   // The matrix type is mutable and set lazily. Force it to be computed here to
@@ -298,23 +296,16 @@
                                   SkTileMode tile_mode,
                                   SkGradientShader::Interpolation interpolation,
                                   const SkMatrix& local_matrix,
-                                  SkColor fallback_color) const override {
+                                  SkColor4f fallback_color) const override {
     if (GetDegenerateHandling() == DegenerateHandling::kDisallow &&
         p0_ == p1_) {
       return PaintShader::MakeEmpty();
     }
 
     SkPoint pts[2] = {FloatPointToSkPoint(p0_), FloatPointToSkPoint(p1_)};
-    // TODO(crbug/1308932): Remove this helper vector colors4f and make all
-    // SkColor4f.
-    std::vector<SkColor4f> colors4f;
-    colors4f.reserve(colors.size());
-    for (auto& color : colors)
-      colors4f.push_back(SkColor4f::FromColor(color));
     return PaintShader::MakeLinearGradient(
-        pts, colors4f.data(), pos.data(), static_cast<int>(colors4f.size()),
-        tile_mode, interpolation, 0 /* flags */, &local_matrix,
-        SkColor4f::FromColor(fallback_color));
+        pts, colors.data(), pos.data(), static_cast<int>(colors.size()),
+        tile_mode, interpolation, 0 /* flags */, &local_matrix, fallback_color);
   }
 
  private:
@@ -348,7 +339,7 @@
                                   SkTileMode tile_mode,
                                   SkGradientShader::Interpolation interpolation,
                                   const SkMatrix& local_matrix,
-                                  SkColor fallback_color) const override {
+                                  SkColor4f fallback_color) const override {
     const SkMatrix* matrix = &local_matrix;
     absl::optional<SkMatrix> adjusted_local_matrix;
     if (aspect_ratio_ != 1) {
@@ -370,17 +361,10 @@
       return PaintShader::MakeEmpty();
     }
 
-    // TODO(crbug/1308932): Remove this helper vector colors4f and make all
-    // SkColor4f.
-    std::vector<SkColor4f> colors4f;
-    colors4f.reserve(colors.size());
-    for (auto& color : colors)
-      colors4f.push_back(SkColor4f::FromColor(color));
     return PaintShader::MakeTwoPointConicalGradient(
         FloatPointToSkPoint(p0_), radius0, FloatPointToSkPoint(p1_), radius1,
-        colors4f.data(), pos.data(), static_cast<int>(colors4f.size()),
-        tile_mode, interpolation, 0 /* flags */, matrix,
-        SkColor4f::FromColor(fallback_color));
+        colors.data(), pos.data(), static_cast<int>(colors.size()), tile_mode,
+        interpolation, 0 /* flags */, matrix, fallback_color);
   }
 
  private:
@@ -415,7 +399,7 @@
                                   SkTileMode tile_mode,
                                   SkGradientShader::Interpolation interpolation,
                                   const SkMatrix& local_matrix,
-                                  SkColor fallback_color) const override {
+                                  SkColor4f fallback_color) const override {
     if (GetDegenerateHandling() == DegenerateHandling::kDisallow &&
         start_angle_ == end_angle_) {
       return PaintShader::MakeEmpty();
@@ -432,17 +416,10 @@
       matrix = &*adjusted_local_matrix;
     }
 
-    // TODO(crbug/1308932): Remove this helper vector colors4f and make all
-    // SkColor4f.
-    std::vector<SkColor4f> colors4f;
-    colors4f.reserve(colors.size());
-    for (auto& color : colors)
-      colors4f.push_back(SkColor4f::FromColor(color));
     return PaintShader::MakeSweepGradient(
-        position_.x(), position_.y(), colors4f.data(), pos.data(),
-        static_cast<int>(colors4f.size()), tile_mode, start_angle_, end_angle_,
-        interpolation, 0 /* flags */, matrix,
-        SkColor4f::FromColor(fallback_color));
+        position_.x(), position_.y(), colors.data(), pos.data(),
+        static_cast<int>(colors.size()), tile_mode, start_angle_, end_angle_,
+        interpolation, 0 /* flags */, matrix, fallback_color);
   }
 
  private:
diff --git a/third_party/blink/renderer/platform/graphics/gradient.h b/third_party/blink/renderer/platform/graphics/gradient.h
index 7706f8c6..05e00d6 100644
--- a/third_party/blink/renderer/platform/graphics/gradient.h
+++ b/third_party/blink/renderer/platform/graphics/gradient.h
@@ -128,14 +128,14 @@
  protected:
   Gradient(Type, GradientSpreadMethod, ColorInterpolation, DegenerateHandling);
 
-  using ColorBuffer = Vector<SkColor, 8>;
+  using ColorBuffer = Vector<SkColor4f, 8>;
   using OffsetBuffer = Vector<SkScalar, 8>;
   virtual sk_sp<PaintShader> CreateShader(const ColorBuffer&,
                                           const OffsetBuffer&,
                                           SkTileMode,
                                           SkGradientShader::Interpolation,
                                           const SkMatrix&,
-                                          SkColor) const = 0;
+                                          SkColor4f) const = 0;
 
   DegenerateHandling GetDegenerateHandling() const {
     return degenerate_handling_;
diff --git a/third_party/blink/renderer/platform/graphics/paint/clip_paint_property_node.h b/third_party/blink/renderer/platform/graphics/paint/clip_paint_property_node.h
index 2dd3c43..e4f5dba 100644
--- a/third_party/blink/renderer/platform/graphics/paint/clip_paint_property_node.h
+++ b/third_party/blink/renderer/platform/graphics/paint/clip_paint_property_node.h
@@ -199,8 +199,14 @@
       : ClipPaintPropertyNodeOrAlias(parent), state_(std::move(state)) {}
 
   void AddChanged(PaintPropertyChangeType changed) {
+    // TODO(crbug.com/814815): This is a workaround of the bug. When the bug is
+    // fixed, change the following condition to
+    //   DCHECK(!clip_cache_ || !clip_cache_->IsValid());
     DCHECK_NE(PaintPropertyChangeType::kUnchanged, changed);
-    GeometryMapperClipCache::ClearCache();
+    if (clip_cache_ && clip_cache_->IsValid()) {
+      DLOG(WARNING) << "Clip tree changed without invalidating the cache.";
+      GeometryMapperClipCache::ClearCache();
+    }
     PaintPropertyNode::AddChanged(changed);
   }
 
diff --git a/third_party/blink/renderer/platform/graphics/paint/transform_paint_property_node.h b/third_party/blink/renderer/platform/graphics/paint/transform_paint_property_node.h
index a1040d2..30d6776 100644
--- a/third_party/blink/renderer/platform/graphics/paint/transform_paint_property_node.h
+++ b/third_party/blink/renderer/platform/graphics/paint/transform_paint_property_node.h
@@ -426,11 +426,15 @@
   }
 
   void AddChanged(PaintPropertyChangeType changed) {
+    // TODO(crbug.com/814815): This is a workaround of the bug. When the bug is
+    // fixed, change the following condition to
+    //   DCHECK(!transform_cache_ || !transform_cache_->IsValid());
     DCHECK_NE(PaintPropertyChangeType::kUnchanged, changed);
-
-    GeometryMapperTransformCache::ClearCache();
-    GeometryMapperClipCache::ClearCache();
-
+    if (transform_cache_ && transform_cache_->IsValid()) {
+      DLOG(WARNING) << "Transform tree changed without invalidating the cache.";
+      GeometryMapperTransformCache::ClearCache();
+      GeometryMapperClipCache::ClearCache();
+    }
     TransformPaintPropertyNodeOrAlias::AddChanged(changed);
   }
 
@@ -439,7 +443,6 @@
   friend class GeometryMapperTest;
   friend class GeometryMapperTransformCache;
   friend class GeometryMapperTransformCacheTest;
-  friend class PaintPropertyTreeBuilderTest;
 
   const GeometryMapperTransformCache& GetTransformCache() const {
     if (!transform_cache_)
diff --git a/third_party/blink/renderer/platform/network/http_parsers_test.cc b/third_party/blink/renderer/platform/network/http_parsers_test.cc
index 668c1a57d..ae7d707 100644
--- a/third_party/blink/renderer/platform/network/http_parsers_test.cc
+++ b/third_party/blink/renderer/platform/network/http_parsers_test.cc
@@ -860,7 +860,16 @@
 
 class NoVarySearchPrefetchDisabledTest
     : public ::testing::Test,
-      public ::testing::WithParamInterface<base::StringPiece> {};
+      public ::testing::WithParamInterface<base::StringPiece> {
+ public:
+  NoVarySearchPrefetchDisabledTest() {
+    scoped_feature_list_.InitAndDisableFeature(
+        network::features::kPrefetchNoVarySearch);
+  }
+
+ private:
+  base::test::ScopedFeatureList scoped_feature_list_;
+};
 
 TEST_P(NoVarySearchPrefetchDisabledTest, ParsingNVSReturnsDefaultURLVariance) {
   const auto parsed_headers =
diff --git a/third_party/blink/renderer/platform/runtime_enabled_features.json5 b/third_party/blink/renderer/platform/runtime_enabled_features.json5
index 9489e7b70..a437ceb 100644
--- a/third_party/blink/renderer/platform/runtime_enabled_features.json5
+++ b/third_party/blink/renderer/platform/runtime_enabled_features.json5
@@ -1888,6 +1888,16 @@
       origin_trial_feature_name: "NotificationTriggers",
       status: "experimental",
     },
+    // Origin trial to experiment with No-Vary-Search in Prefetch Cache
+    // See crbug.com/1378075. Depends on SpeculationRulesPrefetchProxy to be
+    // enabled.
+    // On the browser side, the trial needs features::kPrefetchUseContentRefactor
+    // enabled. If features::kPrefetchUseContentRefactor is disabled, then
+    // the code will behave as if PrefetchNoVarySearch is disabled.
+    {
+      name: "NoVarySearchPrefetch",
+      origin_trial_feature_name: "NoVarySearchPrefetch",
+    },
     {
       name: "OffMainThreadCSSPaint",
       status: "stable",
@@ -2617,7 +2627,7 @@
       status: "stable",
       base_feature: "SpeculationRulesPrefetchProxy",
       copied_from_base_feature_if: "overridden",
-      implied_by: ["SpeculationRulesPrefetchFuture"],
+      implied_by: ["SpeculationRulesPrefetchFuture", "NoVarySearchPrefetch"],
     },
     {
       "name": "SpeculationRulesPrefetchWithSubresources",
diff --git a/third_party/blink/web_tests/VirtualTestSuites b/third_party/blink/web_tests/VirtualTestSuites
index 86583ca..a3de662 100644
--- a/third_party/blink/web_tests/VirtualTestSuites
+++ b/third_party/blink/web_tests/VirtualTestSuites
@@ -614,6 +614,13 @@
     "args": ["--disable-site-isolation-trials"],
     "expires": "Jul 1, 2023"
   },
+  {
+    "prefix": "not-site-per-process-nonexclusive",
+    "platforms": ["Linux", "Mac", "Win"],
+    "bases": ["external/wpt/performance-timeline/tentative"],
+    "args": ["--disable-site-isolation-trials"],
+    "expires": "Jul 1, 2023"
+  },
 
   "----------------------- origin-keyed agent clusters --------------------",
   "Origin-keyed agent clusters web platform tests are for the feature at",
@@ -635,7 +642,6 @@
              "--reset-browsing-instance-between-tests"],
     "expires": "Jul 1, 2023"
   },
-
   {
     "prefix": "focusless-spat-nav",
     "platforms": ["Linux", "Mac", "Win"],
@@ -1544,13 +1550,26 @@
     ],
     "exclusive_tests": "ALL",
     "args": [
-      "--enable-blink-features=SpeculationRulesPrefetchProxy,SpeculationRulesFetchFromHeader,SpeculationRulesReferrerPolicyKey,SpeculationRulesDocumentRules",
-      "--enable-features=SpeculationRulesPrefetchProxy,PrefetchUseContentRefactor,PrefetchNoVarySearch",
+      "--enable-blink-features=SpeculationRulesPrefetchProxy,SpeculationRulesFetchFromHeader,SpeculationRulesReferrerPolicyKey,SpeculationRulesDocumentRules,NoVarySearchPrefetch",
+      "--enable-features=SpeculationRulesPrefetchProxy,PrefetchUseContentRefactor",
       "--bypass-prefetch-proxy-for-host=not-web-platform.test"
     ],
     "expires": "Jul 1, 2023"
   },
   {
+    "prefix": "prefetch-no-vary-search",
+    "platforms": ["Linux", "Mac", "Win"],
+    "bases": [
+      "external/wpt/speculation-rules/prefetch/no-vary-search"
+    ],
+    "exclusive_tests": "ALL",
+    "args": [
+      "--enable-blink-features=NoVarySearchPrefetch",
+      "--enable-features=PrefetchUseContentRefactor"
+    ],
+    "expires": "Jul 1, 2023"
+  },
+  {
     "prefix": "cookie-reject-domain-non-ascii",
     "platforms": ["Linux", "Mac", "Win"],
     "bases": [ "http/tests/inspector-protocol/issues/cookie-domain-non-ascii.js" ],
diff --git a/third_party/blink/web_tests/external/wpt/css/css-break/out-of-flow-in-multicolumn-100.html b/third_party/blink/web_tests/external/wpt/css/css-break/out-of-flow-in-multicolumn-100.html
new file mode 100644
index 0000000..1bf71e83
--- /dev/null
+++ b/third_party/blink/web_tests/external/wpt/css/css-break/out-of-flow-in-multicolumn-100.html
@@ -0,0 +1,13 @@
+<!DOCTYPE html>
+<link rel="author" title="Morten Stenshorne" href="mailto:mstensho@chromium.org">
+<link rel="help" href="https://bugs.chromium.org/p/chromium/issues/detail?id=1395581">
+<link rel="match" href="../reference/ref-filled-green-100px-square.xht">
+<p>Test passes if there is a filled green square and <strong>no red</strong>.</p>
+<div style="columns:2; column-fill:auto; height:200px;">
+  <div style="width:60px; height:60px; direction:rtl; border:20px solid green; background:red;">
+    <div style="position:relative;">
+      <div style="position:absolute; left:0; top:0; width:60px; height:30px; background:green;"></div>
+      <div style="position:absolute; right:0; top:30px; width:60px; height:30px; background:green;"></div>
+    </div>
+  </div>
+</div>
diff --git a/third_party/blink/web_tests/external/wpt/css/css-break/out-of-flow-in-multicolumn-101.html b/third_party/blink/web_tests/external/wpt/css/css-break/out-of-flow-in-multicolumn-101.html
new file mode 100644
index 0000000..ed7e1e23
--- /dev/null
+++ b/third_party/blink/web_tests/external/wpt/css/css-break/out-of-flow-in-multicolumn-101.html
@@ -0,0 +1,13 @@
+<!DOCTYPE html>
+<link rel="author" title="Morten Stenshorne" href="mailto:mstensho@chromium.org">
+<link rel="help" href="https://bugs.chromium.org/p/chromium/issues/detail?id=1395581">
+<link rel="match" href="../reference/ref-filled-green-100px-square.xht">
+<p>Test passes if there is a filled green square and <strong>no red</strong>.</p>
+<div style="columns:2; column-fill:auto; gap:0; width:100px; height:100px; background:red;">
+  <div style="width:30px; height:180px; direction:rtl; border:10px solid green;">
+    <div style="position:relative;">
+      <div style="position:absolute; left:0; top:0; width:30px; height:90px; background:green;"></div>
+      <div style="position:absolute; right:0; top:90px; width:30px; height:90px; background:green;"></div>
+    </div>
+  </div>
+</div>
diff --git a/third_party/blink/web_tests/external/wpt/css/css-break/out-of-flow-in-multicolumn-102.html b/third_party/blink/web_tests/external/wpt/css/css-break/out-of-flow-in-multicolumn-102.html
new file mode 100644
index 0000000..45cbc0b
--- /dev/null
+++ b/third_party/blink/web_tests/external/wpt/css/css-break/out-of-flow-in-multicolumn-102.html
@@ -0,0 +1,13 @@
+<!DOCTYPE html>
+<link rel="author" title="Morten Stenshorne" href="mailto:mstensho@chromium.org">
+<link rel="help" href="https://bugs.chromium.org/p/chromium/issues/detail?id=1395581">
+<link rel="match" href="../reference/ref-filled-green-100px-square.xht">
+<p>Test passes if there is a filled green square and <strong>no red</strong>.</p>
+<div style="position:absolute; margin-left:15px; margin-top:15px; width:30px; height:70px; background:green;"></div>
+<div style="columns:2; column-fill:auto; height:200px;">
+  <div style="width:70px; height:70px; direction:rtl; border:solid 15px green; background:red;">
+    <div style="position:relative;">
+      <div style="position:absolute; width:40px; height:70px; background:green;"></div>
+    </div>
+  </div>
+</div>
diff --git a/third_party/blink/web_tests/external/wpt/css/css-break/out-of-flow-in-multicolumn-103.html b/third_party/blink/web_tests/external/wpt/css/css-break/out-of-flow-in-multicolumn-103.html
new file mode 100644
index 0000000..b54cbcd
--- /dev/null
+++ b/third_party/blink/web_tests/external/wpt/css/css-break/out-of-flow-in-multicolumn-103.html
@@ -0,0 +1,14 @@
+<!DOCTYPE html>
+<link rel="author" title="Morten Stenshorne" href="mailto:mstensho@chromium.org">
+<link rel="help" href="https://bugs.chromium.org/p/chromium/issues/detail?id=1395581">
+<link rel="match" href="../reference/ref-filled-green-100px-square.xht">
+<p>Test passes if there is a filled green square and <strong>no red</strong>.</p>
+<div style="position:absolute; margin-left:10px; margin-top:10px; width:10px; height:90px; background:green;"></div>
+<div style="position:absolute; margin-left:60px; width:10px; height:90px; background:green;"></div>
+<div style="columns:2; column-fill:auto; gap:0; width:100px; height:100px; background:red;">
+  <div style="width:32px; height:180px; direction:rtl; border:solid 10px green; border-right-width:8px;">
+    <div style="position:relative;">
+      <div style="position:absolute; width:22px; height:180px; background:green;"></div>
+    </div>
+  </div>
+</div>
diff --git a/third_party/blink/web_tests/external/wpt/css/css-images/gradient/css-color-4-colors-default-to-oklab-gradient.html b/third_party/blink/web_tests/external/wpt/css/css-images/gradient/css-color-4-colors-default-to-oklab-gradient.html
index 5be18ac..63fdd6f 100644
--- a/third_party/blink/web_tests/external/wpt/css/css-images/gradient/css-color-4-colors-default-to-oklab-gradient.html
+++ b/third_party/blink/web_tests/external/wpt/css/css-images/gradient/css-color-4-colors-default-to-oklab-gradient.html
@@ -11,7 +11,7 @@
    .test {
        width: 480px;
        height: 50px;
-       background: linear-gradient(to right, rgb(255, 0, 0), color(srgb 0 255 0));
+       background: linear-gradient(to right, rgb(255, 0, 0), color(srgb 0 1 0));
    }
  </style>
  </head>
diff --git a/third_party/blink/web_tests/external/wpt/css/css-images/object-fit-contain-png-001c.html b/third_party/blink/web_tests/external/wpt/css/css-images/object-fit-contain-png-001c.html
index 4fb7f113..0f8195157 100644
--- a/third_party/blink/web_tests/external/wpt/css/css-images/object-fit-contain-png-001c.html
+++ b/third_party/blink/web_tests/external/wpt/css/css-images/object-fit-contain-png-001c.html
@@ -12,6 +12,7 @@
     <link rel="help" href="http://www.w3.org/TR/css3-images/#the-object-fit">
     <link rel="help" href="http://www.w3.org/TR/css3-images/#the-object-position">
     <link rel="match" href="object-fit-contain-png-001-ref.html">
+    <meta name=fuzzy content="maxDifference=0-20;totalPixels=0-2000">
     <style type="text/css">
       canvas {
         border: 1px dashed gray;
diff --git a/third_party/blink/web_tests/external/wpt/css/css-images/object-fit-contain-png-002c.html b/third_party/blink/web_tests/external/wpt/css/css-images/object-fit-contain-png-002c.html
index 738c015a..14834316 100644
--- a/third_party/blink/web_tests/external/wpt/css/css-images/object-fit-contain-png-002c.html
+++ b/third_party/blink/web_tests/external/wpt/css/css-images/object-fit-contain-png-002c.html
@@ -12,6 +12,7 @@
     <link rel="help" href="http://www.w3.org/TR/css3-images/#the-object-fit">
     <link rel="help" href="http://www.w3.org/TR/css3-images/#the-object-position">
     <link rel="match" href="object-fit-contain-png-002-ref.html">
+    <meta name=fuzzy content="maxDifference=0-20;totalPixels=0-2000">
     <style type="text/css">
       canvas {
         border: 1px dashed gray;
diff --git a/third_party/blink/web_tests/external/wpt/css/css-images/object-fit-fill-png-001c.html b/third_party/blink/web_tests/external/wpt/css/css-images/object-fit-fill-png-001c.html
index 36031175..0e2a388 100644
--- a/third_party/blink/web_tests/external/wpt/css/css-images/object-fit-fill-png-001c.html
+++ b/third_party/blink/web_tests/external/wpt/css/css-images/object-fit-fill-png-001c.html
@@ -12,6 +12,7 @@
     <link rel="help" href="http://www.w3.org/TR/css3-images/#the-object-fit">
     <link rel="help" href="http://www.w3.org/TR/css3-images/#the-object-position">
     <link rel="match" href="object-fit-fill-png-001-ref.html">
+    <meta name=fuzzy content="maxDifference=0-20;totalPixels=0-3200">
     <style type="text/css">
       canvas {
         border: 1px dashed gray;
diff --git a/third_party/blink/web_tests/external/wpt/css/css-images/object-fit-fill-png-002c.html b/third_party/blink/web_tests/external/wpt/css/css-images/object-fit-fill-png-002c.html
index a332c37..43bcced 100644
--- a/third_party/blink/web_tests/external/wpt/css/css-images/object-fit-fill-png-002c.html
+++ b/third_party/blink/web_tests/external/wpt/css/css-images/object-fit-fill-png-002c.html
@@ -12,6 +12,7 @@
     <link rel="help" href="http://www.w3.org/TR/css3-images/#the-object-fit">
     <link rel="help" href="http://www.w3.org/TR/css3-images/#the-object-position">
     <link rel="match" href="object-fit-fill-png-002-ref.html">
+    <meta name=fuzzy content="maxDifference=0-20;totalPixels=0-3200">
     <style type="text/css">
       canvas {
         border: 1px dashed gray;
diff --git a/third_party/blink/web_tests/external/wpt/performance-timeline/resources/parent-frame-with-child.sub.html b/third_party/blink/web_tests/external/wpt/performance-timeline/resources/parent-frame-with-cross-origin-child.sub.html
similarity index 67%
rename from third_party/blink/web_tests/external/wpt/performance-timeline/resources/parent-frame-with-child.sub.html
rename to third_party/blink/web_tests/external/wpt/performance-timeline/resources/parent-frame-with-cross-origin-child.sub.html
index 32dd4cb..4be0df87 100644
--- a/third_party/blink/web_tests/external/wpt/performance-timeline/resources/parent-frame-with-child.sub.html
+++ b/third_party/blink/web_tests/external/wpt/performance-timeline/resources/parent-frame-with-cross-origin-child.sub.html
@@ -2,6 +2,7 @@
 <head></head>
 <body></body>
 <script>
+    // Create child frame that is cross-origin with its parent.
     const childFrame = document.createElement('iframe')
     childFrame.src = "http://{{hosts[][]}}:{{ports[http][0]}}/performance-timeline/resources/child-frame.html"
     document.body.appendChild(childFrame)
@@ -9,6 +10,7 @@
     performance.mark("entry-name")
 
     childFrame.addEventListener('load', () => {
-        window.parent.postMessage("DONE", "*")
+        const entries = performance.getEntries(true)
+        window.parent.postMessage(entries.length, "*")
     })
 </script>
\ No newline at end of file
diff --git a/third_party/blink/web_tests/external/wpt/performance-timeline/resources/parent-frame-with-same-origin-child.html b/third_party/blink/web_tests/external/wpt/performance-timeline/resources/parent-frame-with-same-origin-child.html
new file mode 100644
index 0000000..c9248a4
--- /dev/null
+++ b/third_party/blink/web_tests/external/wpt/performance-timeline/resources/parent-frame-with-same-origin-child.html
@@ -0,0 +1,16 @@
+<!DOCTYPE html>
+<head></head>
+<body></body>
+<script>
+    // Create child frame that is same-origin with its parent.
+    const childFrame = document.createElement('iframe')
+    childFrame.src = "child-frame.html"
+    document.body.appendChild(childFrame)
+
+    performance.mark("entry-name")
+
+    childFrame.addEventListener('load', () => {
+        const entries = performance.getEntries(true)
+        window.parent.postMessage(entries.length, "*")
+    })
+</script>
\ No newline at end of file
diff --git a/third_party/blink/web_tests/external/wpt/performance-timeline/tentative/include-frames-from-child-cross-origin-grandchild.sub.html b/third_party/blink/web_tests/external/wpt/performance-timeline/tentative/include-frames-from-child-cross-origin-grandchild.sub.html
new file mode 100644
index 0000000..dcd716d
--- /dev/null
+++ b/third_party/blink/web_tests/external/wpt/performance-timeline/tentative/include-frames-from-child-cross-origin-grandchild.sub.html
@@ -0,0 +1,27 @@
+<!DOCTYPE html>
+<head>
+<script src="/resources/testharness.js"></script>
+<script src="/resources/testharnessreport.js"></script>
+</head>
+<body>
+</body>
+<script>
+promise_test(() => {
+    return new Promise(resolve => {
+        performance.clearResourceTimings()
+
+        // Create child iframe with an embedded frame that is same-origin with its parent but cross-origin with the current frame.
+        const crossOriginChildFrame = document.createElement('iframe')
+        crossOriginChildFrame.src = "http://{{hosts[][www]}}:{{ports[http][0]}}/performance-timeline/resources/parent-frame-with-same-origin-child.html"
+        document.body.appendChild(crossOriginChildFrame)
+
+        // Listen for postMessage() from child frame.
+        window.addEventListener("message", e => {
+            // 0 entries for parent, 4 for child, 2 for grandchild.
+            assert_equals(e.data, 6)
+
+            resolve()
+        })
+    })
+}, "GetEntries of a Cross-Origin child frame with one Cross-Origin grandchild frame")
+</script>
\ No newline at end of file
diff --git a/third_party/blink/web_tests/external/wpt/performance-timeline/tentative/include-frames-from-child-same-origin-grandchild.sub.html b/third_party/blink/web_tests/external/wpt/performance-timeline/tentative/include-frames-from-child-same-origin-grandchild.sub.html
new file mode 100644
index 0000000..991715f
--- /dev/null
+++ b/third_party/blink/web_tests/external/wpt/performance-timeline/tentative/include-frames-from-child-same-origin-grandchild.sub.html
@@ -0,0 +1,27 @@
+<!DOCTYPE html>
+<head>
+<script src="/resources/testharness.js"></script>
+<script src="/resources/testharnessreport.js"></script>
+</head>
+<body>
+</body>
+<script>
+promise_test(() => {
+    return new Promise(resolve => {
+        performance.clearResourceTimings()
+
+        // Create child iframe with an embedded frame that is cross-origin with its parent but same-origin with the current frame.
+        const crossOriginChildFrame = document.createElement('iframe')
+        crossOriginChildFrame.src = "http://{{hosts[][www]}}:{{ports[http][0]}}/performance-timeline/resources/parent-frame-with-cross-origin-child.sub.html"
+        document.body.appendChild(crossOriginChildFrame)
+
+        // Listen for postMessage() from child frame.
+        window.addEventListener("message", e => {
+            // 0 entries for parent, 4 for child, 0 for grandchild.
+            assert_equals(e.data, 4)
+
+            resolve()
+        })
+    })
+}, "GetEntries of a Cross-Origin child frame with one Same-Origin grandchild frame")
+</script>
\ No newline at end of file
diff --git a/third_party/blink/web_tests/external/wpt/performance-timeline/tentative/include-frames-one-local-child-one-local-grandchild.html b/third_party/blink/web_tests/external/wpt/performance-timeline/tentative/include-frames-one-local-child-one-local-grandchild.html
index ee2dadd..06a13f3 100644
--- a/third_party/blink/web_tests/external/wpt/performance-timeline/tentative/include-frames-one-local-child-one-local-grandchild.html
+++ b/third_party/blink/web_tests/external/wpt/performance-timeline/tentative/include-frames-one-local-child-one-local-grandchild.html
@@ -10,9 +10,9 @@
     return new Promise(resolve => {
         performance.clearResourceTimings()
 
-        // Create child iframe.
+        // Create child iframe with an embedded frame that is cross-origin with its parent, but same-origin with the current frame.
         const childFrame = document.createElement('iframe')
-        childFrame.src = "../resources/parent-frame-with-child.sub.html"
+        childFrame.src = "../resources/parent-frame-with-cross-origin-child.sub.html"
         document.body.appendChild(childFrame)
 
         // Listen for postMessage() from grandchild frame.
diff --git a/third_party/blink/web_tests/external/wpt/performance-timeline/tentative/include-frames-one-remote-child-one-local-grandchild.sub.html b/third_party/blink/web_tests/external/wpt/performance-timeline/tentative/include-frames-one-remote-child-one-local-grandchild.sub.html
index a9efe41..3248d601 100644
--- a/third_party/blink/web_tests/external/wpt/performance-timeline/tentative/include-frames-one-remote-child-one-local-grandchild.sub.html
+++ b/third_party/blink/web_tests/external/wpt/performance-timeline/tentative/include-frames-one-remote-child-one-local-grandchild.sub.html
@@ -10,9 +10,9 @@
     return new Promise(resolve => {
         performance.clearResourceTimings()
 
-        // Create child iframe.
+        // Create child iframe with an embedded frame that is cross-origin with its parent, but same-origin with the current frame.
         const remoteChildFrame = document.createElement('iframe')
-        remoteChildFrame.src = "http://{{hosts[][www]}}:{{ports[http][0]}}/performance-timeline/resources/parent-frame-with-child.sub.html"
+        remoteChildFrame.src = "http://{{hosts[][www]}}:{{ports[http][0]}}/performance-timeline/resources/parent-frame-with-cross-origin-child.sub.html"
         document.body.appendChild(remoteChildFrame)
 
         // Listen for postMessage() from grandchild frame.
diff --git a/third_party/blink/web_tests/external/wpt/speculation-rules/prefetch/no-vary-search/README.txt b/third_party/blink/web_tests/external/wpt/speculation-rules/prefetch/no-vary-search/README.txt
new file mode 100644
index 0000000..60ac226f
--- /dev/null
+++ b/third_party/blink/web_tests/external/wpt/speculation-rules/prefetch/no-vary-search/README.txt
@@ -0,0 +1 @@
+Web Platform Tests for No-Vary-Search support in prefetch cache.
diff --git a/third_party/blink/web_tests/flag-specific/disable-layout-ng/fast/gradients/generated-gradients-expected.png b/third_party/blink/web_tests/flag-specific/disable-layout-ng/fast/gradients/generated-gradients-expected.png
index a03c012..7f32aa6f 100644
--- a/third_party/blink/web_tests/flag-specific/disable-layout-ng/fast/gradients/generated-gradients-expected.png
+++ b/third_party/blink/web_tests/flag-specific/disable-layout-ng/fast/gradients/generated-gradients-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/flag-specific/disable-layout-ng/fast/gradients/simple-gradients-expected.png b/third_party/blink/web_tests/flag-specific/disable-layout-ng/fast/gradients/simple-gradients-expected.png
index 61a0d2f..16450ae6 100644
--- a/third_party/blink/web_tests/flag-specific/disable-layout-ng/fast/gradients/simple-gradients-expected.png
+++ b/third_party/blink/web_tests/flag-specific/disable-layout-ng/fast/gradients/simple-gradients-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/flag-specific/disable-layout-ng/svg/wicd/test-scalable-background-image2-expected.png b/third_party/blink/web_tests/flag-specific/disable-layout-ng/svg/wicd/test-scalable-background-image2-expected.png
index bb0f5b2..be7d1ba 100644
--- a/third_party/blink/web_tests/flag-specific/disable-layout-ng/svg/wicd/test-scalable-background-image2-expected.png
+++ b/third_party/blink/web_tests/flag-specific/disable-layout-ng/svg/wicd/test-scalable-background-image2-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/flag-specific/highdpi/svg/batik/text/textEffect-expected.png b/third_party/blink/web_tests/flag-specific/highdpi/svg/batik/text/textEffect-expected.png
index b4db6398..6ef1fe26 100644
--- a/third_party/blink/web_tests/flag-specific/highdpi/svg/batik/text/textEffect-expected.png
+++ b/third_party/blink/web_tests/flag-specific/highdpi/svg/batik/text/textEffect-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/flag-specific/highdpi/svg/wicd/test-scalable-background-image2-expected.png b/third_party/blink/web_tests/flag-specific/highdpi/svg/wicd/test-scalable-background-image2-expected.png
new file mode 100644
index 0000000..481f4f2
--- /dev/null
+++ b/third_party/blink/web_tests/flag-specific/highdpi/svg/wicd/test-scalable-background-image2-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/linux/fast/gradients/generated-gradients-expected.png b/third_party/blink/web_tests/platform/linux/fast/gradients/generated-gradients-expected.png
index e138595..1481503 100644
--- a/third_party/blink/web_tests/platform/linux/fast/gradients/generated-gradients-expected.png
+++ b/third_party/blink/web_tests/platform/linux/fast/gradients/generated-gradients-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/linux/fast/gradients/simple-gradients-expected.png b/third_party/blink/web_tests/platform/linux/fast/gradients/simple-gradients-expected.png
index d916621..99755b1 100644
--- a/third_party/blink/web_tests/platform/linux/fast/gradients/simple-gradients-expected.png
+++ b/third_party/blink/web_tests/platform/linux/fast/gradients/simple-gradients-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/linux/svg/W3C-SVG-1.1/pservers-grad-14-b-expected.png b/third_party/blink/web_tests/platform/linux/svg/W3C-SVG-1.1/pservers-grad-14-b-expected.png
index d76cc76..8226e918 100644
--- a/third_party/blink/web_tests/platform/linux/svg/W3C-SVG-1.1/pservers-grad-14-b-expected.png
+++ b/third_party/blink/web_tests/platform/linux/svg/W3C-SVG-1.1/pservers-grad-14-b-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/linux/svg/as-background-image/svg-as-background-3-expected.png b/third_party/blink/web_tests/platform/linux/svg/as-background-image/svg-as-background-3-expected.png
index df70ddad..b95cdbf 100644
--- a/third_party/blink/web_tests/platform/linux/svg/as-background-image/svg-as-background-3-expected.png
+++ b/third_party/blink/web_tests/platform/linux/svg/as-background-image/svg-as-background-3-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/linux/svg/wicd/test-scalable-background-image2-expected.png b/third_party/blink/web_tests/platform/linux/svg/wicd/test-scalable-background-image2-expected.png
index 481f4f2..5c1a25a8 100644
--- a/third_party/blink/web_tests/platform/linux/svg/wicd/test-scalable-background-image2-expected.png
+++ b/third_party/blink/web_tests/platform/linux/svg/wicd/test-scalable-background-image2-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac10.13/svg/wicd/test-scalable-background-image2-expected.png b/third_party/blink/web_tests/platform/mac-mac10.13/svg/wicd/test-scalable-background-image2-expected.png
index 10e1a16..3ab85fb 100644
--- a/third_party/blink/web_tests/platform/mac-mac10.13/svg/wicd/test-scalable-background-image2-expected.png
+++ b/third_party/blink/web_tests/platform/mac-mac10.13/svg/wicd/test-scalable-background-image2-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/canvas/canvas-incremental-repaint-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/canvas/canvas-incremental-repaint-expected.png
index 5f61902..1662057d 100644
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/canvas/canvas-incremental-repaint-expected.png
+++ b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/canvas/canvas-incremental-repaint-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color-scheme/color/color-picker-appearance-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color-scheme/color/color-picker-appearance-expected.png
deleted file mode 100644
index 5d03ab2a..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color-scheme/color/color-picker-appearance-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-click-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-click-expected.png
deleted file mode 100644
index 9597811..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-click-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-down-keyboard-navigation-from-top-left-corner-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-down-keyboard-navigation-from-top-left-corner-expected.png
deleted file mode 100644
index 2d91917..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-down-keyboard-navigation-from-top-left-corner-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-down-keyboard-navigation-from-top-right-corner-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-down-keyboard-navigation-from-top-right-corner-expected.png
deleted file mode 100644
index ef045b8..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-down-keyboard-navigation-from-top-right-corner-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-drag-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-drag-expected.png
deleted file mode 100644
index 9597811..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-drag-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-keyboard-navigation-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-keyboard-navigation-expected.png
deleted file mode 100644
index 68e5798..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-keyboard-navigation-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-left-keyboard-navigation-from-bottom-right-corner-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-left-keyboard-navigation-from-bottom-right-corner-expected.png
deleted file mode 100644
index c8b2b51..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-left-keyboard-navigation-from-bottom-right-corner-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-left-keyboard-navigation-from-top-right-corner-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-left-keyboard-navigation-from-top-right-corner-expected.png
deleted file mode 100644
index 7a299d1..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-left-keyboard-navigation-from-top-right-corner-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-right-keyboard-navigation-from-bottom-left-corner-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-right-keyboard-navigation-from-bottom-left-corner-expected.png
deleted file mode 100644
index 9176a3a1..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-right-keyboard-navigation-from-bottom-left-corner-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-right-keyboard-navigation-from-top-left-corner-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-right-keyboard-navigation-from-top-left-corner-expected.png
deleted file mode 100644
index 220dc263..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-right-keyboard-navigation-from-top-left-corner-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-touch-drag-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-touch-drag-expected.png
deleted file mode 100644
index 4090cebd..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-touch-drag-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-up-keyboard-navigation-from-bottom-left-corner-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-up-keyboard-navigation-from-bottom-left-corner-expected.png
deleted file mode 100644
index 66da85c..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-up-keyboard-navigation-from-bottom-left-corner-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-up-keyboard-navigation-from-bottom-right-corner-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-up-keyboard-navigation-from-bottom-right-corner-expected.png
deleted file mode 100644
index cb7b5b4..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-color-well-up-keyboard-navigation-from-bottom-right-corner-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-hex-format-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-hex-format-expected.png
deleted file mode 100644
index c8acdf4d..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-hex-format-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-hsl-format-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-hsl-format-expected.png
deleted file mode 100644
index 86aa0df..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-hsl-format-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-hue-slider-left-edge-zero-hue-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-hue-slider-left-edge-zero-hue-expected.png
deleted file mode 100644
index 6892e25b..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-hue-slider-left-edge-zero-hue-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-hue-slider-right-edge-zero-hue-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-hue-slider-right-edge-zero-hue-expected.png
deleted file mode 100644
index 450d929c..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-hue-slider-right-edge-zero-hue-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-imperfect-match-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-imperfect-match-expected.png
deleted file mode 100644
index ea6faa4..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-imperfect-match-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-manual-color-change-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-manual-color-change-expected.png
deleted file mode 100644
index 3d6f2cb..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-manual-color-change-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-set-value-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-set-value-expected.png
deleted file mode 100644
index 6027fc7..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-set-value-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-tap-hex-format-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-tap-hex-format-expected.png
deleted file mode 100644
index 36fe0c2..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-tap-hex-format-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-value-attribute-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-value-attribute-expected.png
deleted file mode 100644
index 92335fbd..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-appearance-value-attribute-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-top-left-selection-position-after-reopen-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-top-left-selection-position-after-reopen-expected.png
deleted file mode 100644
index 54909a8..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/forms/color/color-picker-top-left-selection-position-after-reopen-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/gradients/conic-gradient-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/gradients/conic-gradient-expected.png
index c5e3412..ba20a1ca 100644
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/gradients/conic-gradient-expected.png
+++ b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/gradients/conic-gradient-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/gradients/generated-gradients-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/gradients/generated-gradients-expected.png
new file mode 100644
index 0000000..448329e36
--- /dev/null
+++ b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/gradients/generated-gradients-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/gradients/unprefixed-linear-gradients-color-hints-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/gradients/unprefixed-linear-gradients-color-hints-expected.png
index ea05f2f..35559e0c 100644
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/gradients/unprefixed-linear-gradients-color-hints-expected.png
+++ b/third_party/blink/web_tests/platform/mac-mac11-arm64/fast/gradients/unprefixed-linear-gradients-color-hints-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/media/alpha-video-playback-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/media/alpha-video-playback-expected.png
deleted file mode 100644
index 7fc316c..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/media/alpha-video-playback-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/paint/invalidation/canvas-resize-no-full-invalidation-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/paint/invalidation/canvas-resize-no-full-invalidation-expected.png
deleted file mode 100644
index 2bfe7f7..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/paint/invalidation/canvas-resize-no-full-invalidation-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/svg/wicd/test-scalable-background-image2-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/svg/wicd/test-scalable-background-image2-expected.png
index 84df5d5..49428ee 100644
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/svg/wicd/test-scalable-background-image2-expected.png
+++ b/third_party/blink/web_tests/platform/mac-mac11-arm64/svg/wicd/test-scalable-background-image2-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/virtual/controls-refresh-hc/fast/forms/color-scheme/color/color-picker-appearance-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/virtual/controls-refresh-hc/fast/forms/color-scheme/color/color-picker-appearance-expected.png
deleted file mode 100644
index 3d15b57..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/virtual/controls-refresh-hc/fast/forms/color-scheme/color/color-picker-appearance-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/virtual/dark-color-scheme/fast/forms/color-scheme/color/color-picker-appearance-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/virtual/dark-color-scheme/fast/forms/color-scheme/color/color-picker-appearance-expected.png
deleted file mode 100644
index 8d0f8d8..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/virtual/dark-color-scheme/fast/forms/color-scheme/color/color-picker-appearance-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/virtual/dark-system-color-picker-appearance/fast/forms/color-scheme/color/color-picker-appearance-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/virtual/dark-system-color-picker-appearance/fast/forms/color-scheme/color/color-picker-appearance-expected.png
deleted file mode 100644
index c231417..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/virtual/dark-system-color-picker-appearance/fast/forms/color-scheme/color/color-picker-appearance-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/virtual/gpu/fast/canvas/canvas-incremental-repaint-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/virtual/gpu/fast/canvas/canvas-incremental-repaint-expected.png
new file mode 100644
index 0000000..5f61902
--- /dev/null
+++ b/third_party/blink/web_tests/platform/mac-mac11-arm64/virtual/gpu/fast/canvas/canvas-incremental-repaint-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11-arm64/virtual/system-color-picker-appearance/fast/forms/color-scheme/color/color-picker-appearance-expected.png b/third_party/blink/web_tests/platform/mac-mac11-arm64/virtual/system-color-picker-appearance/fast/forms/color-scheme/color/color-picker-appearance-expected.png
deleted file mode 100644
index 1344ecc..0000000
--- a/third_party/blink/web_tests/platform/mac-mac11-arm64/virtual/system-color-picker-appearance/fast/forms/color-scheme/color/color-picker-appearance-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11/fast/canvas/canvas-incremental-repaint-expected.png b/third_party/blink/web_tests/platform/mac-mac11/fast/canvas/canvas-incremental-repaint-expected.png
new file mode 100644
index 0000000..f0252fb
--- /dev/null
+++ b/third_party/blink/web_tests/platform/mac-mac11/fast/canvas/canvas-incremental-repaint-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11/svg/wicd/test-scalable-background-image2-expected.png b/third_party/blink/web_tests/platform/mac-mac11/svg/wicd/test-scalable-background-image2-expected.png
index 3b37f98ad..4d997c6 100644
--- a/third_party/blink/web_tests/platform/mac-mac11/svg/wicd/test-scalable-background-image2-expected.png
+++ b/third_party/blink/web_tests/platform/mac-mac11/svg/wicd/test-scalable-background-image2-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11/virtual/exotic-color-space/images/color-profile-munsell-adobe-to-srgb-expected.png b/third_party/blink/web_tests/platform/mac-mac11/virtual/exotic-color-space/images/color-profile-munsell-adobe-to-srgb-expected.png
index d9ba88e..66a699e 100644
--- a/third_party/blink/web_tests/platform/mac-mac11/virtual/exotic-color-space/images/color-profile-munsell-adobe-to-srgb-expected.png
+++ b/third_party/blink/web_tests/platform/mac-mac11/virtual/exotic-color-space/images/color-profile-munsell-adobe-to-srgb-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac11/virtual/exotic-color-space/images/color-profile-munsell-srgb-to-srgb-expected.png b/third_party/blink/web_tests/platform/mac-mac11/virtual/exotic-color-space/images/color-profile-munsell-srgb-to-srgb-expected.png
index ba341e1b..5b6ab5c 100644
--- a/third_party/blink/web_tests/platform/mac-mac11/virtual/exotic-color-space/images/color-profile-munsell-srgb-to-srgb-expected.png
+++ b/third_party/blink/web_tests/platform/mac-mac11/virtual/exotic-color-space/images/color-profile-munsell-srgb-to-srgb-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/canvas/canvas-incremental-repaint-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/canvas/canvas-incremental-repaint-expected.png
index 5f61902..1662057d 100644
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/canvas/canvas-incremental-repaint-expected.png
+++ b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/canvas/canvas-incremental-repaint-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color-scheme/color/color-picker-appearance-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color-scheme/color/color-picker-appearance-expected.png
deleted file mode 100644
index 5d03ab2a..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color-scheme/color/color-picker-appearance-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-click-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-click-expected.png
deleted file mode 100644
index 9597811..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-click-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-down-keyboard-navigation-from-top-left-corner-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-down-keyboard-navigation-from-top-left-corner-expected.png
deleted file mode 100644
index 2d91917..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-down-keyboard-navigation-from-top-left-corner-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-down-keyboard-navigation-from-top-right-corner-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-down-keyboard-navigation-from-top-right-corner-expected.png
deleted file mode 100644
index ef045b8..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-down-keyboard-navigation-from-top-right-corner-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-drag-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-drag-expected.png
deleted file mode 100644
index 9597811..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-drag-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-keyboard-navigation-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-keyboard-navigation-expected.png
deleted file mode 100644
index 68e5798..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-keyboard-navigation-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-left-keyboard-navigation-from-bottom-right-corner-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-left-keyboard-navigation-from-bottom-right-corner-expected.png
deleted file mode 100644
index c8b2b51..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-left-keyboard-navigation-from-bottom-right-corner-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-left-keyboard-navigation-from-top-right-corner-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-left-keyboard-navigation-from-top-right-corner-expected.png
deleted file mode 100644
index 7a299d1..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-left-keyboard-navigation-from-top-right-corner-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-right-keyboard-navigation-from-bottom-left-corner-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-right-keyboard-navigation-from-bottom-left-corner-expected.png
deleted file mode 100644
index 9176a3a1..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-right-keyboard-navigation-from-bottom-left-corner-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-right-keyboard-navigation-from-top-left-corner-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-right-keyboard-navigation-from-top-left-corner-expected.png
deleted file mode 100644
index 220dc263..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-right-keyboard-navigation-from-top-left-corner-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-touch-drag-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-touch-drag-expected.png
deleted file mode 100644
index 4090cebd..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-touch-drag-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-up-keyboard-navigation-from-bottom-left-corner-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-up-keyboard-navigation-from-bottom-left-corner-expected.png
deleted file mode 100644
index 66da85c..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-up-keyboard-navigation-from-bottom-left-corner-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-up-keyboard-navigation-from-bottom-right-corner-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-up-keyboard-navigation-from-bottom-right-corner-expected.png
deleted file mode 100644
index cb7b5b4..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-color-well-up-keyboard-navigation-from-bottom-right-corner-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-hex-format-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-hex-format-expected.png
deleted file mode 100644
index c8acdf4d..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-hex-format-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-hsl-format-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-hsl-format-expected.png
deleted file mode 100644
index 86aa0df..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-hsl-format-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-hue-slider-left-edge-zero-hue-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-hue-slider-left-edge-zero-hue-expected.png
deleted file mode 100644
index 6892e25b..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-hue-slider-left-edge-zero-hue-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-hue-slider-right-edge-zero-hue-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-hue-slider-right-edge-zero-hue-expected.png
deleted file mode 100644
index 450d929c..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-hue-slider-right-edge-zero-hue-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-imperfect-match-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-imperfect-match-expected.png
deleted file mode 100644
index ea6faa4..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-imperfect-match-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-manual-color-change-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-manual-color-change-expected.png
deleted file mode 100644
index 3d6f2cb..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-manual-color-change-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-set-value-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-set-value-expected.png
deleted file mode 100644
index 6027fc7..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-set-value-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-tap-hex-format-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-tap-hex-format-expected.png
deleted file mode 100644
index 36fe0c2..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-tap-hex-format-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-value-attribute-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-value-attribute-expected.png
deleted file mode 100644
index 92335fbd..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-appearance-value-attribute-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-top-left-selection-position-after-reopen-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-top-left-selection-position-after-reopen-expected.png
deleted file mode 100644
index 54909a8..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/forms/color/color-picker-top-left-selection-position-after-reopen-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/gradients/conic-gradient-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/gradients/conic-gradient-expected.png
index c5e3412..ba20a1ca 100644
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/gradients/conic-gradient-expected.png
+++ b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/gradients/conic-gradient-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/gradients/generated-gradients-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/gradients/generated-gradients-expected.png
new file mode 100644
index 0000000..448329e36
--- /dev/null
+++ b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/gradients/generated-gradients-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/gradients/unprefixed-linear-gradients-color-hints-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/gradients/unprefixed-linear-gradients-color-hints-expected.png
index ea05f2f..35559e0c 100644
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/gradients/unprefixed-linear-gradients-color-hints-expected.png
+++ b/third_party/blink/web_tests/platform/mac-mac12-arm64/fast/gradients/unprefixed-linear-gradients-color-hints-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/media/alpha-video-playback-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/media/alpha-video-playback-expected.png
deleted file mode 100644
index 7fc316c..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/media/alpha-video-playback-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/paint/invalidation/canvas-resize-no-full-invalidation-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/paint/invalidation/canvas-resize-no-full-invalidation-expected.png
deleted file mode 100644
index 2bfe7f7..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/paint/invalidation/canvas-resize-no-full-invalidation-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/svg/wicd/test-scalable-background-image2-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/svg/wicd/test-scalable-background-image2-expected.png
index cd18903b..6f110d15 100644
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/svg/wicd/test-scalable-background-image2-expected.png
+++ b/third_party/blink/web_tests/platform/mac-mac12-arm64/svg/wicd/test-scalable-background-image2-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/virtual/controls-refresh-hc/fast/forms/color-scheme/color/color-picker-appearance-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/virtual/controls-refresh-hc/fast/forms/color-scheme/color/color-picker-appearance-expected.png
deleted file mode 100644
index 3d15b57..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/virtual/controls-refresh-hc/fast/forms/color-scheme/color/color-picker-appearance-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/virtual/dark-color-scheme/fast/forms/color-scheme/color/color-picker-appearance-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/virtual/dark-color-scheme/fast/forms/color-scheme/color/color-picker-appearance-expected.png
deleted file mode 100644
index 8d0f8d8..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/virtual/dark-color-scheme/fast/forms/color-scheme/color/color-picker-appearance-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/virtual/dark-system-color-picker-appearance/fast/forms/color-scheme/color/color-picker-appearance-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/virtual/dark-system-color-picker-appearance/fast/forms/color-scheme/color/color-picker-appearance-expected.png
deleted file mode 100644
index c231417..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/virtual/dark-system-color-picker-appearance/fast/forms/color-scheme/color/color-picker-appearance-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/virtual/exotic-color-space/images/jpeg-yuv-progressive-canvas-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/virtual/exotic-color-space/images/jpeg-yuv-progressive-canvas-expected.png
deleted file mode 100644
index ffa89d0..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/virtual/exotic-color-space/images/jpeg-yuv-progressive-canvas-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/virtual/gpu/fast/canvas/canvas-incremental-repaint-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/virtual/gpu/fast/canvas/canvas-incremental-repaint-expected.png
new file mode 100644
index 0000000..5f61902
--- /dev/null
+++ b/third_party/blink/web_tests/platform/mac-mac12-arm64/virtual/gpu/fast/canvas/canvas-incremental-repaint-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac-mac12-arm64/virtual/system-color-picker-appearance/fast/forms/color-scheme/color/color-picker-appearance-expected.png b/third_party/blink/web_tests/platform/mac-mac12-arm64/virtual/system-color-picker-appearance/fast/forms/color-scheme/color/color-picker-appearance-expected.png
deleted file mode 100644
index 1344ecc..0000000
--- a/third_party/blink/web_tests/platform/mac-mac12-arm64/virtual/system-color-picker-appearance/fast/forms/color-scheme/color/color-picker-appearance-expected.png
+++ /dev/null
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/canvas/canvas-hidpi-blurry-expected.png b/third_party/blink/web_tests/platform/mac/fast/canvas/canvas-hidpi-blurry-expected.png
index 6a9b251..92e03005 100644
--- a/third_party/blink/web_tests/platform/mac/fast/canvas/canvas-hidpi-blurry-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/canvas/canvas-hidpi-blurry-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/canvas/canvas-incremental-repaint-expected.png b/third_party/blink/web_tests/platform/mac/fast/canvas/canvas-incremental-repaint-expected.png
index aef4be0..f0252fb 100644
--- a/third_party/blink/web_tests/platform/mac/fast/canvas/canvas-incremental-repaint-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/canvas/canvas-incremental-repaint-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/canvas/canvas-toDataURL-jpeg-maximum-quality-expected.png b/third_party/blink/web_tests/platform/mac/fast/canvas/canvas-toDataURL-jpeg-maximum-quality-expected.png
index ad7239b..28a0a35c 100644
--- a/third_party/blink/web_tests/platform/mac/fast/canvas/canvas-toDataURL-jpeg-maximum-quality-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/canvas/canvas-toDataURL-jpeg-maximum-quality-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/canvas/canvas-toDataURL-webp-expected.png b/third_party/blink/web_tests/platform/mac/fast/canvas/canvas-toDataURL-webp-expected.png
new file mode 100644
index 0000000..2159358a
--- /dev/null
+++ b/third_party/blink/web_tests/platform/mac/fast/canvas/canvas-toDataURL-webp-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/canvas/pixelated-expected.png b/third_party/blink/web_tests/platform/mac/fast/canvas/pixelated-expected.png
new file mode 100644
index 0000000..b75d52bc
--- /dev/null
+++ b/third_party/blink/web_tests/platform/mac/fast/canvas/pixelated-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color-scheme/color/color-picker-appearance-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color-scheme/color/color-picker-appearance-expected.png
index 577c247..9e2444f 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color-scheme/color/color-picker-appearance-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color-scheme/color/color-picker-appearance-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-click-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-click-expected.png
index 6770f89..5210665 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-click-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-click-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-down-keyboard-navigation-from-top-left-corner-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-down-keyboard-navigation-from-top-left-corner-expected.png
index 9af4d8d..73668a6 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-down-keyboard-navigation-from-top-left-corner-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-down-keyboard-navigation-from-top-left-corner-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-down-keyboard-navigation-from-top-right-corner-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-down-keyboard-navigation-from-top-right-corner-expected.png
index 0d159ed..13b9281 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-down-keyboard-navigation-from-top-right-corner-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-down-keyboard-navigation-from-top-right-corner-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-drag-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-drag-expected.png
index 6770f89..5210665 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-drag-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-drag-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-keyboard-navigation-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-keyboard-navigation-expected.png
index c824cba..3344e87 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-keyboard-navigation-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-keyboard-navigation-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-left-keyboard-navigation-from-bottom-right-corner-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-left-keyboard-navigation-from-bottom-right-corner-expected.png
index bf48c62d..c1d9b25b 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-left-keyboard-navigation-from-bottom-right-corner-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-left-keyboard-navigation-from-bottom-right-corner-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-left-keyboard-navigation-from-top-right-corner-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-left-keyboard-navigation-from-top-right-corner-expected.png
index 6744078..5bc891c 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-left-keyboard-navigation-from-top-right-corner-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-left-keyboard-navigation-from-top-right-corner-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-right-keyboard-navigation-from-bottom-left-corner-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-right-keyboard-navigation-from-bottom-left-corner-expected.png
index ba54ccd..867acb27 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-right-keyboard-navigation-from-bottom-left-corner-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-right-keyboard-navigation-from-bottom-left-corner-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-right-keyboard-navigation-from-top-left-corner-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-right-keyboard-navigation-from-top-left-corner-expected.png
index 16af243d..4e5b644 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-right-keyboard-navigation-from-top-left-corner-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-right-keyboard-navigation-from-top-left-corner-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-touch-drag-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-touch-drag-expected.png
index 02ae9d4..6e61b30 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-touch-drag-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-touch-drag-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-up-keyboard-navigation-from-bottom-left-corner-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-up-keyboard-navigation-from-bottom-left-corner-expected.png
index f863fcd..23fe32a 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-up-keyboard-navigation-from-bottom-left-corner-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-up-keyboard-navigation-from-bottom-left-corner-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-up-keyboard-navigation-from-bottom-right-corner-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-up-keyboard-navigation-from-bottom-right-corner-expected.png
index db826b4..f1e886ee 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-up-keyboard-navigation-from-bottom-right-corner-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-color-well-up-keyboard-navigation-from-bottom-right-corner-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hex-format-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hex-format-expected.png
index 2f59f7f..0368d30d 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hex-format-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hex-format-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hsl-format-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hsl-format-expected.png
index 574a796..c621520 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hsl-format-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hsl-format-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-accelerated-keyboard-navigation-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-accelerated-keyboard-navigation-expected.png
index 4de189f..dd4d416 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-accelerated-keyboard-navigation-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-accelerated-keyboard-navigation-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-click-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-click-expected.png
index 42f81d5a..aa13f9c 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-click-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-click-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-drag-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-drag-expected.png
index 42f81d5a..aa13f9c 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-drag-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-drag-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-keyboard-navigation-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-keyboard-navigation-expected.png
index 3189d77..4dd743b 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-keyboard-navigation-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-keyboard-navigation-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-left-edge-zero-hue-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-left-edge-zero-hue-expected.png
index f75fe963..0476678 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-left-edge-zero-hue-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-left-edge-zero-hue-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-right-edge-zero-hue-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-right-edge-zero-hue-expected.png
index a2bc636..2a1980a 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-right-edge-zero-hue-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-right-edge-zero-hue-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-touch-drag-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-touch-drag-expected.png
index 8d82275..a5c05a8 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-touch-drag-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-hue-slider-touch-drag-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-imperfect-match-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-imperfect-match-expected.png
index f65ba0d..a095fc9 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-imperfect-match-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-imperfect-match-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-manual-color-change-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-manual-color-change-expected.png
index dec5aeb..63d3411 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-manual-color-change-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-manual-color-change-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-set-value-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-set-value-expected.png
index 9e27acf..0c84723 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-set-value-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-set-value-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-tap-hex-format-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-tap-hex-format-expected.png
index 777f415b..114e9dc 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-tap-hex-format-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-tap-hex-format-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-value-attribute-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-value-attribute-expected.png
index f400434..ecf3bcc5 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-value-attribute-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-value-attribute-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-zoom125-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-zoom125-expected.png
index 5bb2acd0..dcce6db4 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-zoom125-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-zoom125-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-zoom200-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-zoom200-expected.png
index 900f5a7..fe6ecd6d 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-zoom200-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-appearance-zoom200-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-top-left-selection-position-after-reopen-expected.png b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-top-left-selection-position-after-reopen-expected.png
index a434e66..0ad74dc 100644
--- a/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-top-left-selection-position-after-reopen-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/forms/color/color-picker-top-left-selection-position-after-reopen-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/gradients/conic-gradient-expected.png b/third_party/blink/web_tests/platform/mac/fast/gradients/conic-gradient-expected.png
index 9f4d9b65..114f0cfb 100644
--- a/third_party/blink/web_tests/platform/mac/fast/gradients/conic-gradient-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/gradients/conic-gradient-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/gradients/generated-gradients-expected.png b/third_party/blink/web_tests/platform/mac/fast/gradients/generated-gradients-expected.png
index 448329e36..00855abf 100644
--- a/third_party/blink/web_tests/platform/mac/fast/gradients/generated-gradients-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/gradients/generated-gradients-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/gradients/simple-gradients-expected.png b/third_party/blink/web_tests/platform/mac/fast/gradients/simple-gradients-expected.png
index 1bda305..bc6d9a7 100644
--- a/third_party/blink/web_tests/platform/mac/fast/gradients/simple-gradients-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/gradients/simple-gradients-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/gradients/unprefixed-linear-gradients-color-hints-expected.png b/third_party/blink/web_tests/platform/mac/fast/gradients/unprefixed-linear-gradients-color-hints-expected.png
index 7d3b99d5..303e8178 100644
--- a/third_party/blink/web_tests/platform/mac/fast/gradients/unprefixed-linear-gradients-color-hints-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/gradients/unprefixed-linear-gradients-color-hints-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/fast/replaced/border-radius-clip-content-edge-expected.png b/third_party/blink/web_tests/platform/mac/fast/replaced/border-radius-clip-content-edge-expected.png
index 50044ee..ef8cc96 100644
--- a/third_party/blink/web_tests/platform/mac/fast/replaced/border-radius-clip-content-edge-expected.png
+++ b/third_party/blink/web_tests/platform/mac/fast/replaced/border-radius-clip-content-edge-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/inspector-protocol/overlay/overlay-with-emulation-scale-expected.txt b/third_party/blink/web_tests/platform/mac/inspector-protocol/overlay/overlay-with-emulation-scale-expected.txt
new file mode 100644
index 0000000..f546a917
--- /dev/null
+++ b/third_party/blink/web_tests/platform/mac/inspector-protocol/overlay/overlay-with-emulation-scale-expected.txt
@@ -0,0 +1,4 @@
+Verifies that overlay is correctly rendered with emulation scale > 1.
+The test passes if the image URL below is 300x600 image containing a 270x420 brown rectangle, without any green or red.
+
+
diff --git a/third_party/blink/web_tests/platform/mac/media/alpha-video-playback-expected.png b/third_party/blink/web_tests/platform/mac/media/alpha-video-playback-expected.png
index e2872f31..7fc316c 100644
--- a/third_party/blink/web_tests/platform/mac/media/alpha-video-playback-expected.png
+++ b/third_party/blink/web_tests/platform/mac/media/alpha-video-playback-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/paint/invalidation/canvas-putImageData-expected.txt b/third_party/blink/web_tests/platform/mac/paint/invalidation/canvas-putImageData-expected.txt
new file mode 100644
index 0000000..9051839
--- /dev/null
+++ b/third_party/blink/web_tests/platform/mac/paint/invalidation/canvas-putImageData-expected.txt
@@ -0,0 +1,16 @@
+{
+  "layers": [
+    {
+      "name": "Scrolling background of LayoutNGView #document",
+      "bounds": [800, 600],
+      "contentsOpaque": true,
+      "backgroundColor": "#FFFFFF"
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='canvas'",
+      "position": [8, 8],
+      "bounds": [100, 100]
+    }
+  ]
+}
+
diff --git a/third_party/blink/web_tests/platform/mac/paint/invalidation/canvas-resize-expected.txt b/third_party/blink/web_tests/platform/mac/paint/invalidation/canvas-resize-expected.txt
new file mode 100644
index 0000000..dce35520
--- /dev/null
+++ b/third_party/blink/web_tests/platform/mac/paint/invalidation/canvas-resize-expected.txt
@@ -0,0 +1,16 @@
+{
+  "layers": [
+    {
+      "name": "Scrolling background of LayoutNGView #document",
+      "bounds": [800, 600],
+      "contentsOpaque": true,
+      "backgroundColor": "#FFFFFF"
+    },
+    {
+      "name": "LayoutHTMLCanvas (positioned) CANVAS id='canvas'",
+      "position": [50, 50],
+      "bounds": [500, 500]
+    }
+  ]
+}
+
diff --git a/third_party/blink/web_tests/platform/mac/paint/invalidation/canvas-resize-no-full-invalidation-expected.png b/third_party/blink/web_tests/platform/mac/paint/invalidation/canvas-resize-no-full-invalidation-expected.png
index 4359ac3..2bfe7f7 100644
--- a/third_party/blink/web_tests/platform/mac/paint/invalidation/canvas-resize-no-full-invalidation-expected.png
+++ b/third_party/blink/web_tests/platform/mac/paint/invalidation/canvas-resize-no-full-invalidation-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/paint/invalidation/canvas-resize-no-full-invalidation-expected.txt b/third_party/blink/web_tests/platform/mac/paint/invalidation/canvas-resize-no-full-invalidation-expected.txt
new file mode 100644
index 0000000..dbfdd46
--- /dev/null
+++ b/third_party/blink/web_tests/platform/mac/paint/invalidation/canvas-resize-no-full-invalidation-expected.txt
@@ -0,0 +1,20 @@
+{
+  "layers": [
+    {
+      "name": "Scrolling background of LayoutNGView #document",
+      "bounds": [800, 600],
+      "contentsOpaque": true,
+      "backgroundColor": "#FFFFFF",
+      "invalidations": [
+        [50, 50, 600, 500]
+      ]
+    },
+    {
+      "name": "LayoutHTMLCanvas (positioned) CANVAS id='canvas'",
+      "position": [50, 50],
+      "bounds": [500, 500],
+      "backgroundColor": "#003300"
+    }
+  ]
+}
+
diff --git a/third_party/blink/web_tests/platform/mac/paint/invalidation/image/canvas-composite-repaint-by-all-imagesource-expected.txt b/third_party/blink/web_tests/platform/mac/paint/invalidation/image/canvas-composite-repaint-by-all-imagesource-expected.txt
index 5598dfc..f5ec0f3 100644
--- a/third_party/blink/web_tests/platform/mac/paint/invalidation/image/canvas-composite-repaint-by-all-imagesource-expected.txt
+++ b/third_party/blink/web_tests/platform/mac/paint/invalidation/image/canvas-composite-repaint-by-all-imagesource-expected.txt
@@ -1,56 +1,235 @@
 {
   "layers": [
     {
-      "name": "Scrolling background of LayoutView #document",
+      "name": "Scrolling background of LayoutNGView #document",
       "bounds": [785, 928],
       "contentsOpaque": true,
-      "backgroundColor": "#FFFFFF",
-      "invalidations": [
-        [569, 562, 133, 42],
-        [569, 512, 133, 42],
-        [569, 462, 133, 42],
-        [569, 412, 133, 42],
-        [569, 362, 133, 42],
-        [569, 312, 133, 42],
-        [569, 262, 133, 42],
-        [569, 212, 133, 42],
-        [569, 162, 133, 42],
-        [569, 112, 133, 42],
-        [569, 62, 133, 42],
-        [427, 562, 133, 42],
-        [427, 512, 133, 42],
-        [427, 462, 133, 42],
-        [427, 412, 133, 42],
-        [427, 362, 133, 42],
-        [427, 312, 133, 42],
-        [427, 262, 133, 42],
-        [427, 212, 133, 42],
-        [427, 162, 133, 42],
-        [427, 112, 133, 42],
-        [427, 62, 133, 42],
-        [285, 562, 133, 42],
-        [285, 512, 133, 42],
-        [285, 462, 133, 42],
-        [285, 412, 133, 42],
-        [285, 362, 133, 42],
-        [285, 312, 133, 42],
-        [285, 262, 133, 42],
-        [285, 212, 133, 42],
-        [285, 162, 133, 42],
-        [285, 112, 133, 42],
-        [285, 62, 133, 42],
-        [143, 562, 133, 42],
-        [143, 512, 133, 42],
-        [143, 462, 133, 42],
-        [143, 412, 133, 42],
-        [143, 362, 133, 42],
-        [143, 312, 133, 42],
-        [143, 262, 133, 42],
-        [143, 212, 133, 42],
-        [143, 162, 133, 42],
-        [143, 112, 133, 42],
-        [143, 62, 133, 42]
-      ]
+      "backgroundColor": "#FFFFFF"
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='source-oversolid color'",
+      "position": [145, 63],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='source-overimage'",
+      "position": [287, 63],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='source-overcanvas'",
+      "position": [429, 63],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='source-overvideo'",
+      "position": [571, 63],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='source-insolid color'",
+      "position": [145, 113],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='source-inimage'",
+      "position": [287, 113],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='source-incanvas'",
+      "position": [429, 113],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='source-invideo'",
+      "position": [571, 113],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='source-outsolid color'",
+      "position": [145, 163],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='source-outimage'",
+      "position": [287, 163],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='source-outcanvas'",
+      "position": [429, 163],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='source-outvideo'",
+      "position": [571, 163],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='source-atopsolid color'",
+      "position": [145, 213],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='source-atopimage'",
+      "position": [287, 213],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='source-atopcanvas'",
+      "position": [429, 213],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='source-atopvideo'",
+      "position": [571, 213],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='destination-oversolid color'",
+      "position": [145, 263],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='destination-overimage'",
+      "position": [287, 263],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='destination-overcanvas'",
+      "position": [429, 263],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='destination-overvideo'",
+      "position": [571, 263],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='destination-insolid color'",
+      "position": [145, 313],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='destination-inimage'",
+      "position": [287, 313],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='destination-incanvas'",
+      "position": [429, 313],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='destination-invideo'",
+      "position": [571, 313],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='destination-outsolid color'",
+      "position": [145, 363],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='destination-outimage'",
+      "position": [287, 363],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='destination-outcanvas'",
+      "position": [429, 363],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='destination-outvideo'",
+      "position": [571, 363],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='destination-atopsolid color'",
+      "position": [145, 413],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='destination-atopimage'",
+      "position": [287, 413],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='destination-atopcanvas'",
+      "position": [429, 413],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='destination-atopvideo'",
+      "position": [571, 413],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='lightersolid color'",
+      "position": [145, 463],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='lighterimage'",
+      "position": [287, 463],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='lightercanvas'",
+      "position": [429, 463],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='lightervideo'",
+      "position": [571, 463],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='copysolid color'",
+      "position": [145, 513],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='copyimage'",
+      "position": [287, 513],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='copycanvas'",
+      "position": [429, 513],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='copyvideo'",
+      "position": [571, 513],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='xorsolid color'",
+      "position": [145, 563],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='xorimage'",
+      "position": [287, 563],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='xorcanvas'",
+      "position": [429, 563],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='xorvideo'",
+      "position": [571, 563],
+      "bounds": [130, 40]
+    },
+    {
+      "name": "LayoutHTMLCanvas CANVAS id='source-canvas'",
+      "position": [16, 751],
+      "bounds": [150, 60]
     },
     {
       "name": "LayoutVideo VIDEO id='video'",
diff --git a/third_party/blink/web_tests/platform/mac/svg/W3C-SVG-1.1/pservers-grad-14-b-expected.png b/third_party/blink/web_tests/platform/mac/svg/W3C-SVG-1.1/pservers-grad-14-b-expected.png
index 82032ccc..d848959 100644
--- a/third_party/blink/web_tests/platform/mac/svg/W3C-SVG-1.1/pservers-grad-14-b-expected.png
+++ b/third_party/blink/web_tests/platform/mac/svg/W3C-SVG-1.1/pservers-grad-14-b-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/svg/as-background-image/svg-as-background-3-expected.png b/third_party/blink/web_tests/platform/mac/svg/as-background-image/svg-as-background-3-expected.png
index b662737a..67644339 100644
--- a/third_party/blink/web_tests/platform/mac/svg/as-background-image/svg-as-background-3-expected.png
+++ b/third_party/blink/web_tests/platform/mac/svg/as-background-image/svg-as-background-3-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/svg/wicd/test-scalable-background-image2-expected.png b/third_party/blink/web_tests/platform/mac/svg/wicd/test-scalable-background-image2-expected.png
index b52fb41..b30acc9 100644
--- a/third_party/blink/web_tests/platform/mac/svg/wicd/test-scalable-background-image2-expected.png
+++ b/third_party/blink/web_tests/platform/mac/svg/wicd/test-scalable-background-image2-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/virtual/controls-refresh-hc/fast/forms/color-scheme/color/color-picker-appearance-expected.png b/third_party/blink/web_tests/platform/mac/virtual/controls-refresh-hc/fast/forms/color-scheme/color/color-picker-appearance-expected.png
index e53a7e8..9b27031 100644
--- a/third_party/blink/web_tests/platform/mac/virtual/controls-refresh-hc/fast/forms/color-scheme/color/color-picker-appearance-expected.png
+++ b/third_party/blink/web_tests/platform/mac/virtual/controls-refresh-hc/fast/forms/color-scheme/color/color-picker-appearance-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/virtual/dark-color-scheme/fast/forms/color-scheme/color/color-picker-appearance-expected.png b/third_party/blink/web_tests/platform/mac/virtual/dark-color-scheme/fast/forms/color-scheme/color/color-picker-appearance-expected.png
index 4eb24ef..da2f7b8c 100644
--- a/third_party/blink/web_tests/platform/mac/virtual/dark-color-scheme/fast/forms/color-scheme/color/color-picker-appearance-expected.png
+++ b/third_party/blink/web_tests/platform/mac/virtual/dark-color-scheme/fast/forms/color-scheme/color/color-picker-appearance-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/virtual/dark-system-color-picker-appearance/fast/forms/color-scheme/color/color-picker-appearance-expected.png b/third_party/blink/web_tests/platform/mac/virtual/dark-system-color-picker-appearance/fast/forms/color-scheme/color/color-picker-appearance-expected.png
index fb5bce8..91c1688 100644
--- a/third_party/blink/web_tests/platform/mac/virtual/dark-system-color-picker-appearance/fast/forms/color-scheme/color/color-picker-appearance-expected.png
+++ b/third_party/blink/web_tests/platform/mac/virtual/dark-system-color-picker-appearance/fast/forms/color-scheme/color/color-picker-appearance-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/color-profile-image-canvas-svg-expected.png b/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/color-profile-image-canvas-svg-expected.png
index 93ddabe..ddfd164 100644
--- a/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/color-profile-image-canvas-svg-expected.png
+++ b/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/color-profile-image-canvas-svg-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/color-profile-munsell-adobe-to-srgb-expected.png b/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/color-profile-munsell-adobe-to-srgb-expected.png
index 183ab2f..13013cb5b 100644
--- a/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/color-profile-munsell-adobe-to-srgb-expected.png
+++ b/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/color-profile-munsell-adobe-to-srgb-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/color-profile-munsell-srgb-to-srgb-expected.png b/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/color-profile-munsell-srgb-to-srgb-expected.png
index 7606226..0d95b5e3 100644
--- a/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/color-profile-munsell-srgb-to-srgb-expected.png
+++ b/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/color-profile-munsell-srgb-to-srgb-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/jpeg-yuv-progressive-canvas-expected.png b/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/jpeg-yuv-progressive-canvas-expected.png
new file mode 100644
index 0000000..1aa13baa
--- /dev/null
+++ b/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/jpeg-yuv-progressive-canvas-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/optimize-contrast-canvas-expected.png b/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/optimize-contrast-canvas-expected.png
new file mode 100644
index 0000000..4b949f6
--- /dev/null
+++ b/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/optimize-contrast-canvas-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/paint-subrect-expected.png b/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/paint-subrect-expected.png
new file mode 100644
index 0000000..1fca07f0
--- /dev/null
+++ b/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/paint-subrect-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/paint-subrect-grid-expected.png b/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/paint-subrect-grid-expected.png
new file mode 100644
index 0000000..f55c833
--- /dev/null
+++ b/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/paint-subrect-grid-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/pixelated-canvas-expected.png b/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/pixelated-canvas-expected.png
new file mode 100644
index 0000000..1021936
--- /dev/null
+++ b/third_party/blink/web_tests/platform/mac/virtual/exotic-color-space/images/pixelated-canvas-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/virtual/media-foundation-for-clear-dcomp/media/alpha-video-playback-expected.png b/third_party/blink/web_tests/platform/mac/virtual/media-foundation-for-clear-dcomp/media/alpha-video-playback-expected.png
new file mode 100644
index 0000000..e2872f31
--- /dev/null
+++ b/third_party/blink/web_tests/platform/mac/virtual/media-foundation-for-clear-dcomp/media/alpha-video-playback-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/virtual/media-foundation-for-clear-frameserver/media/alpha-video-playback-expected.png b/third_party/blink/web_tests/platform/mac/virtual/media-foundation-for-clear-frameserver/media/alpha-video-playback-expected.png
new file mode 100644
index 0000000..e2872f31
--- /dev/null
+++ b/third_party/blink/web_tests/platform/mac/virtual/media-foundation-for-clear-frameserver/media/alpha-video-playback-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/virtual/no-alloc-direct-call/fast/canvas/canvas-hidpi-blurry-expected.png b/third_party/blink/web_tests/platform/mac/virtual/no-alloc-direct-call/fast/canvas/canvas-hidpi-blurry-expected.png
new file mode 100644
index 0000000..6a9b251
--- /dev/null
+++ b/third_party/blink/web_tests/platform/mac/virtual/no-alloc-direct-call/fast/canvas/canvas-hidpi-blurry-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/virtual/no-alloc-direct-call/fast/canvas/canvas-toDataURL-jpeg-maximum-quality-expected.png b/third_party/blink/web_tests/platform/mac/virtual/no-alloc-direct-call/fast/canvas/canvas-toDataURL-jpeg-maximum-quality-expected.png
new file mode 100644
index 0000000..ad7239b
--- /dev/null
+++ b/third_party/blink/web_tests/platform/mac/virtual/no-alloc-direct-call/fast/canvas/canvas-toDataURL-jpeg-maximum-quality-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/mac/virtual/system-color-picker-appearance/fast/forms/color-scheme/color/color-picker-appearance-expected.png b/third_party/blink/web_tests/platform/mac/virtual/system-color-picker-appearance/fast/forms/color-scheme/color/color-picker-appearance-expected.png
index fa3bcbc..c0ced16 100644
--- a/third_party/blink/web_tests/platform/mac/virtual/system-color-picker-appearance/fast/forms/color-scheme/color/color-picker-appearance-expected.png
+++ b/third_party/blink/web_tests/platform/mac/virtual/system-color-picker-appearance/fast/forms/color-scheme/color/color-picker-appearance-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/win/fast/gradients/conic-gradient-expected.png b/third_party/blink/web_tests/platform/win/fast/gradients/conic-gradient-expected.png
index cab834f..92291e33 100644
--- a/third_party/blink/web_tests/platform/win/fast/gradients/conic-gradient-expected.png
+++ b/third_party/blink/web_tests/platform/win/fast/gradients/conic-gradient-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/win/fast/gradients/generated-gradients-expected.png b/third_party/blink/web_tests/platform/win/fast/gradients/generated-gradients-expected.png
index 5478656..3b7b9f1 100644
--- a/third_party/blink/web_tests/platform/win/fast/gradients/generated-gradients-expected.png
+++ b/third_party/blink/web_tests/platform/win/fast/gradients/generated-gradients-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/win/fast/gradients/simple-gradients-expected.png b/third_party/blink/web_tests/platform/win/fast/gradients/simple-gradients-expected.png
index 24061be..4cf5a70 100644
--- a/third_party/blink/web_tests/platform/win/fast/gradients/simple-gradients-expected.png
+++ b/third_party/blink/web_tests/platform/win/fast/gradients/simple-gradients-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/win/fast/gradients/unprefixed-linear-gradients-color-hints-expected.png b/third_party/blink/web_tests/platform/win/fast/gradients/unprefixed-linear-gradients-color-hints-expected.png
index 3970797..b83e74d 100644
--- a/third_party/blink/web_tests/platform/win/fast/gradients/unprefixed-linear-gradients-color-hints-expected.png
+++ b/third_party/blink/web_tests/platform/win/fast/gradients/unprefixed-linear-gradients-color-hints-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/win/fast/gradients/unprefixed-repeating-gradient-color-hint-expected.png b/third_party/blink/web_tests/platform/win/fast/gradients/unprefixed-repeating-gradient-color-hint-expected.png
index b784d383..0ebb499 100644
--- a/third_party/blink/web_tests/platform/win/fast/gradients/unprefixed-repeating-gradient-color-hint-expected.png
+++ b/third_party/blink/web_tests/platform/win/fast/gradients/unprefixed-repeating-gradient-color-hint-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/win/svg/W3C-SVG-1.1/pservers-grad-14-b-expected.png b/third_party/blink/web_tests/platform/win/svg/W3C-SVG-1.1/pservers-grad-14-b-expected.png
index 922589a..90034c74 100644
--- a/third_party/blink/web_tests/platform/win/svg/W3C-SVG-1.1/pservers-grad-14-b-expected.png
+++ b/third_party/blink/web_tests/platform/win/svg/W3C-SVG-1.1/pservers-grad-14-b-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/win/svg/as-background-image/svg-as-background-3-expected.png b/third_party/blink/web_tests/platform/win/svg/as-background-image/svg-as-background-3-expected.png
index 524e955..cc57d74 100644
--- a/third_party/blink/web_tests/platform/win/svg/as-background-image/svg-as-background-3-expected.png
+++ b/third_party/blink/web_tests/platform/win/svg/as-background-image/svg-as-background-3-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/platform/win/svg/wicd/test-scalable-background-image2-expected.png b/third_party/blink/web_tests/platform/win/svg/wicd/test-scalable-background-image2-expected.png
index 4bf8691..46ad5fe 100644
--- a/third_party/blink/web_tests/platform/win/svg/wicd/test-scalable-background-image2-expected.png
+++ b/third_party/blink/web_tests/platform/win/svg/wicd/test-scalable-background-image2-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/svg/as-background-image/svg-as-background-with-relative-size-expected.png b/third_party/blink/web_tests/svg/as-background-image/svg-as-background-with-relative-size-expected.png
index c4d42e1..e2e753a 100644
--- a/third_party/blink/web_tests/svg/as-background-image/svg-as-background-with-relative-size-expected.png
+++ b/third_party/blink/web_tests/svg/as-background-image/svg-as-background-with-relative-size-expected.png
Binary files differ
diff --git a/third_party/blink/web_tests/virtual/not-site-per-process-nonexclusive/README.md b/third_party/blink/web_tests/virtual/not-site-per-process-nonexclusive/README.md
new file mode 100644
index 0000000..6481c20
--- /dev/null
+++ b/third_party/blink/web_tests/virtual/not-site-per-process-nonexclusive/README.md
@@ -0,0 +1,5 @@
+# virtual/not-site-per-process
+
+## Summary
+
+See virtual/not-site-per-process. This virtual test suite removes the `exclusive_tests: "ALL"` configuration, and specifies tests that should be run both with and without site isolation enabled.
\ No newline at end of file
diff --git a/third_party/blink/web_tests/virtual/prefetch-no-vary-search/README.md b/third_party/blink/web_tests/virtual/prefetch-no-vary-search/README.md
new file mode 100644
index 0000000..60ac226f
--- /dev/null
+++ b/third_party/blink/web_tests/virtual/prefetch-no-vary-search/README.md
@@ -0,0 +1 @@
+Web Platform Tests for No-Vary-Search support in prefetch cache.
diff --git a/third_party/nearby/README.chromium b/third_party/nearby/README.chromium
index 6e4f2f01..dda6a8b 100644
--- a/third_party/nearby/README.chromium
+++ b/third_party/nearby/README.chromium
@@ -1,7 +1,7 @@
 Name: Nearby Connections Library
 Short Name: Nearby
 URL: https://github.com/google/nearby
-Version: 73c1fba668a86dd003fa2ac6af86ab7d89c09908
+Version: b3f85342261db10bbeb2e5a07c7e54b616309b9e
 License: Apache 2.0
 License File: LICENSE
 Security Critical: yes
diff --git a/tools/licenses/licenses_test.py b/tools/licenses/licenses_test.py
index 4772582..1af0ebe 100755
--- a/tools/licenses/licenses_test.py
+++ b/tools/licenses/licenses_test.py
@@ -40,6 +40,16 @@
             'Name': 'lib3',
             'License File': os.path.join('third_party', 'lib3', 'LICENSE'),
         },
+        os.path.join('third_party', 'lib3-v1'): {
+            # Test SPDX license file dedup. (different name, same license file)
+            'Name': 'lib3-v1',
+            'License File': os.path.join('third_party', 'lib3', 'LICENSE'),
+        },
+        os.path.join('third_party', 'lib3-v2'): {
+            # Test SPDX id dedup. (same name, different license file)
+            'Name': 'lib3',
+            'License File': os.path.join('third_party', 'lib3-v2', 'LICENSE'),
+        },
     }
 
   def test_get_third_party_deps_from_gn_deps_output(self):
@@ -91,6 +101,8 @@
         'lib1 license text\n',
         'lib2 license text\n',
         'lib3 license text\n',
+        'lib3 license text\n',
+        'lib3-v2 license text\n',
     ]
 
     license_txt = licenses.GenerateLicenseFilePlainText(
@@ -113,6 +125,16 @@
         'lib3',
         '--------------------',
         'lib3 license text',
+        '',
+        '--------------------',
+        'lib3-v1',
+        '--------------------',
+        'lib3 license text',
+        '',
+        '--------------------',
+        'lib3-v2',
+        '--------------------',
+        'lib3-v2 license text',
     ]) + '\n'  # extra new line to account for join not adding one to the end
     self.assertEqual(license_txt, expected)
 
@@ -122,6 +144,7 @@
         'lib1\nlicense text\n',
         'lib2\nlicense text\n',
         'lib3\nlicense text\n',
+        'lib3-v2\nlicense text\n',
     ]
 
     license_txt = licenses.GenerateLicenseFileSpdx(
@@ -167,11 +190,21 @@
             "SPDXID": "SPDXRef-Package-lib3",
             "name": "lib3",
             "licenseConcluded": "LicenseRef-lib3"
+        },
+        {
+            "SPDXID": "SPDXRef-Package-lib3-v1",
+            "name": "lib3-v1",
+            "licenseConcluded": "LicenseRef-lib3"
+        },
+        {
+            "SPDXID": "SPDXRef-Package-lib3-1",
+            "name": "lib3",
+            "licenseConcluded": "LicenseRef-lib3-1"
         }
     ],
     "hasExtractedLicensingInfos": [
         {
-            "name": "Chromium License",
+            "name": "Chromium",
             "licenseId": "LicenseRef-Chromium",
             "extractedText": "root\\nlicense text\\n",
             "crossRefs": [
@@ -181,7 +214,7 @@
             ]
         },
         {
-            "name": "lib1 License",
+            "name": "lib1",
             "licenseId": "LicenseRef-lib1",
             "extractedText": "lib1\\nlicense text\\n",
             "crossRefs": [
@@ -191,7 +224,7 @@
             ]
         },
         {
-            "name": "lib2 License",
+            "name": "lib2",
             "licenseId": "LicenseRef-lib2",
             "extractedText": "lib2\\nlicense text\\n",
             "crossRefs": [
@@ -201,7 +234,7 @@
             ]
         },
         {
-            "name": "lib3 License",
+            "name": "lib3",
             "licenseId": "LicenseRef-lib3",
             "extractedText": "lib3\\nlicense text\\n",
             "crossRefs": [
@@ -209,6 +242,16 @@
                     "url": "http://google.com/third_party/lib3/LICENSE"
                 }
             ]
+        },
+        {
+            "name": "lib3",
+            "licenseId": "LicenseRef-lib3-1",
+            "extractedText": "lib3-v2\\nlicense text\\n",
+            "crossRefs": [
+                {
+                    "url": "http://google.com/third_party/lib3-v2/LICENSE"
+                }
+            ]
         }
     ],
     "relationships": [
@@ -226,6 +269,16 @@
             "spdxElementId": "SPDXRef-Package-Chromium",
             "relationshipType": "CONTAINS",
             "relatedSpdxElement": "SPDXRef-Package-lib3"
+        },
+        {
+            "spdxElementId": "SPDXRef-Package-Chromium",
+            "relationshipType": "CONTAINS",
+            "relatedSpdxElement": "SPDXRef-Package-lib3-v1"
+        },
+        {
+            "spdxElementId": "SPDXRef-Package-Chromium",
+            "relationshipType": "CONTAINS",
+            "relatedSpdxElement": "SPDXRef-Package-lib3-1"
         }
     ]
 }'''
diff --git a/tools/licenses/spdx_writer.py b/tools/licenses/spdx_writer.py
index 0ede233..c612d18 100644
--- a/tools/licenses/spdx_writer.py
+++ b/tools/licenses/spdx_writer.py
@@ -2,12 +2,13 @@
 # Use of this source code is governed by a BSD-style license that can be
 # found in the LICENSE file.
 
+import collections
 import dataclasses
 import json
 import os
 import pathlib
 import re
-from typing import Callable
+from typing import Callable, DefaultDict, Tuple
 
 
 class SpdxWriter:
@@ -22,7 +23,9 @@
                doc_namespace: str = None,
                read_file=lambda x: pathlib.Path(x).read_text(encoding='utf-8')):
     self.root_package = _Package(root_package_name, root_license)
-    self.packages = [self.root_package]
+    # Use dict to ensure no duplicate pkgs.
+    # In >=py3.7 dicts are ordered by insertion.
+    self.packages = {}
 
     self.root = root
     self.link_prefix = link_prefix
@@ -35,7 +38,7 @@
 
   def add_package(self, name: str, license_file: str):
     """Add a package to the SPDX output."""
-    self.packages.append(_Package(name, license_file))
+    self.packages[_Package(name, license_file)] = None
 
   def write_to_file(self, file_path: str):
     """Writes the content to a file."""
@@ -50,13 +53,10 @@
     for pkg in self.packages:
       writer.add_package(pkg)
 
-    for pkg in self.packages:
-      writer.add_license_file(pkg)
-
     return writer.write()
 
 
-@dataclasses.dataclass
+@dataclasses.dataclass(frozen=True)
 class _Package:
   """Stores needed data for a package to output SPDX."""
   name: str
@@ -118,37 +118,93 @@
         'relationships': [],
     }
 
+    # Used to dedup license files based on file path.
+    self.existing_license_files = {}  # 'file path': 'licenseId'
+    # Used to make sure that there are no duplicate ids.
+    self.existing_package_ids = collections.defaultdict(int)  # 'packageId': num
+    self.existing_license_ids = collections.defaultdict(int)  # 'licenseId': num
+
+    # Add the root package to make sure that its ID isn't taken.
+    self.add_package(root_package)
+
   def write(self) -> str:
     """Returns a JSON string for the current state of the writer."""
     return json.dumps(self.content, indent=4)
 
+  def _get_dedup_id(self, elem_id: str, id_dict: DefaultDict[str, int]) -> str:
+    """Returns a unique id given a dictionary with existing ids.
+
+    IDs are case sensitive, so this method ignores casing for uniqueness.
+
+    Args:
+      elem_id: the requested id to use for the element.
+      id_dict: dictionary holding already used ids.
+
+    Returns:
+      When the elem_id is already unique, return elem_id.
+      When the elem_id has been used, return elem_id + '-[next num]'.
+    """
+    suffix = id_dict[elem_id]
+    id_dict[elem_id] += 1
+    return f'{elem_id}-{suffix}' if suffix > 0 else elem_id
+
+  def _get_package_id(self, pkg: _Package) -> str:
+    """Makes sure that there are no pkg id duplicates."""
+    return self._get_dedup_id(pkg.package_spdx_id, self.existing_package_ids)
+
+  def _get_license_id(self, pkg: _Package) -> Tuple[str, bool]:
+    """Handles license deduplication.
+
+    If this pkg.file has already been seen, reuse that same id instead. If
+    there are two packages with the same name but different license files,
+    handle deduping the names.
+
+    Args:
+      pkg: The package to get a license id for.
+
+    Returns:
+      First return value is the id, second is whether the license needs to be
+        added to the SPDX doc (False if it already exists in the doc).
+    """
+    existing = self.existing_license_files.get(pkg.file)
+    if existing:
+      return existing, False
+
+    license_id = self._get_dedup_id(pkg.license_spdx_id,
+                                    self.existing_license_ids)
+    self.existing_license_files[pkg.file] = license_id
+    return license_id, True
+
   def add_package(self, pkg: _Package):
     """Writes a package to the file (package metadata)."""
+    pkg_id = self._get_package_id(pkg)
+    license_id, need_to_add_license = self._get_license_id(pkg)
+
     self.content['packages'].append({
-        'SPDXID': pkg.package_spdx_id,
+        'SPDXID': pkg_id,
         'name': pkg.name,
-        'licenseConcluded': pkg.license_spdx_id,
+        'licenseConcluded': license_id,
     })
 
     if pkg.package_spdx_id != self.root_package_id:
       self.content['relationships'].append({
-          'spdxElementId':
-          self.root_package_id,
-          'relationshipType':
-          'CONTAINS',
-          'relatedSpdxElement':
-          pkg.package_spdx_id,
+          'spdxElementId': self.root_package_id,
+          'relationshipType': 'CONTAINS',
+          'relatedSpdxElement': pkg_id,
       })
 
-  def add_license_file(self, pkg: _Package):
+    if need_to_add_license:
+      self._add_license_file(pkg, license_id)
+
+  def _add_license_file(self, pkg: _Package, license_id: str):
     """Writes a license to the file (raw license text)."""
     spdx_path = _get_spdx_path(self.root, pkg.file)
     url = f'{self.link_prefix}{spdx_path.replace(os.sep, "/")}'
     self.content['hasExtractedLicensingInfos'].append({
         'name':
-        f'{pkg.name} License',
+        f'{pkg.name}',
         'licenseId':
-        pkg.license_spdx_id,
+        license_id,
         'extractedText':
         self.read_file(pkg.file),
         'crossRefs': [{
diff --git a/tools/licenses/spdx_writer_test.py b/tools/licenses/spdx_writer_test.py
index 67746dd..15945c9d 100755
--- a/tools/licenses/spdx_writer_test.py
+++ b/tools/licenses/spdx_writer_test.py
@@ -4,6 +4,7 @@
 # found in the LICENSE file.
 """Unit tests for //tools/spdx_writer.py."""
 
+import collections
 import os
 import sys
 import unittest
@@ -13,6 +14,7 @@
 
 from spdx_writer import _get_spdx_path
 from spdx_writer import _Package
+from spdx_writer import _SPDXJSONWriter
 from test_utils import path_from_root
 
 
@@ -41,5 +43,38 @@
     self.assertEqual(self.p.license_spdx_id, 'LicenseRef-abc-def-ghi')
 
 
+class SPDXJSONWriterTest(unittest.TestCase):
+  def setUp(self):
+    super().setUp()
+
+    root_pkg = _Package('root', path_from_root('src', 'LICENSE'))
+    self.writer = _SPDXJSONWriter(path_from_root('src'), root_pkg, '', '',
+                                  '', lambda _: '')
+
+  def test_get_dedup_id(self):
+    id_dict = collections.defaultdict(int)
+    elem_id = 'abc'
+
+    id1 = self.writer._get_dedup_id(elem_id, id_dict)
+    self.assertEqual(id1, elem_id)
+
+    id2 = self.writer._get_dedup_id(elem_id, id_dict)
+    self.assertEqual(id2, elem_id + '-1')
+
+  def test_get_license_id(self):
+    license_path = path_from_root('src', 'p1', 'LICENSE')
+    p1 = _Package('p1', license_path)
+
+    p1_id, need_license = self.writer._get_license_id(p1)
+    self.assertEqual(p1_id, p1.license_spdx_id)
+    self.assertTrue(need_license)
+
+    # Try a new package with the same license path.
+    p2 = _Package('p2', license_path)
+    p2_id, need_license = self.writer._get_license_id(p2)
+    self.assertEqual(p2_id, p1.license_spdx_id)
+    self.assertFalse(need_license)
+
+
 if __name__ == '__main__':
   unittest.main()
diff --git a/tools/metrics/histograms/enums.xml b/tools/metrics/histograms/enums.xml
index 040a599..65dd903f 100644
--- a/tools/metrics/histograms/enums.xml
+++ b/tools/metrics/histograms/enums.xml
@@ -29430,6 +29430,18 @@
   <int value="9" label="WMV"/>
 </enum>
 
+<enum name="DownloadWarningAction">
+  <int value="0" label="SHOWN"/>
+  <int value="1" label="PROCEED"/>
+  <int value="2" label="DISCARD"/>
+  <int value="3" label="KEEP"/>
+  <int value="4" label="CLOSE"/>
+  <int value="5" label="CANCEL"/>
+  <int value="6" label="DISMISS"/>
+  <int value="7" label="BACK"/>
+  <int value="8" label="OPEN_SUBPAGE"/>
+</enum>
+
 <enum name="DownloadWarningSurface">
   <int value="1" label="BUBBLE_MAINPAGE"/>
   <int value="2" label="BUBBLE_SUBPAGE"/>
@@ -32315,6 +32327,7 @@
   <int value="1042" label="ShowCastSessionsStartedByOtherDevices"/>
   <int value="1043" label="CloudAPAuthEnabled"/>
   <int value="1044" label="UsbDetectorNotificationEnabled"/>
+  <int value="1045" label="LacrosSelection"/>
 </enum>
 
 <enum name="EnterprisePoliciesSources">
@@ -51322,6 +51335,7 @@
   <int value="9" label="Text node sibling (includes &lt;b&gt;, etc)"/>
   <int value="10" label="value attribute"/>
   <int value="11" label="&lt;label&gt; for attribute"/>
+  <int value="12" label="Succeeding DOM node overlaying the input"/>
 </enum>
 
 <enum name="InfoBarClosingStates">
diff --git a/tools/metrics/histograms/metadata/android/histograms.xml b/tools/metrics/histograms/metadata/android/histograms.xml
index dcc63339..fde8bad 100644
--- a/tools/metrics/histograms/metadata/android/histograms.xml
+++ b/tools/metrics/histograms/metadata/android/histograms.xml
@@ -1335,6 +1335,21 @@
   </summary>
 </histogram>
 
+<histogram
+    name="Android.FamilyLinkUser.LocalWebApprovalParentAuthenticationError"
+    units="Status error codes raised in GMS Core" expires_after="2023-03-26">
+  <owner>anthie@google.com</owner>
+  <owner>ljjlee@google.com</owner>
+  <owner>chrome-kids-eng@google.com</owner>
+  <owner>cros-families-eng@google.com</owner>
+  <summary>
+    The different GMS error status codes that a user may encounter during the
+    parent authentication step in the local web approval flow. Recorded on
+    Android only. See go/chrome-local-web-approval-parent-auth-error-codes
+    (Googlers only) for the error code definitions.
+  </summary>
+</histogram>
+
 <histogram base="true" name="Android.FeatureModules.AvailabilityStatus"
     enum="FeatureModuleAvailabilityStatus" expires_after="2023-10-16">
 <!-- Name completed by histogram_suffixes
diff --git a/tools/metrics/histograms/metadata/arc/histograms.xml b/tools/metrics/histograms/metadata/arc/histograms.xml
index 6907b1e..faab4f5 100644
--- a/tools/metrics/histograms/metadata/arc/histograms.xml
+++ b/tools/metrics/histograms/metadata/arc/histograms.xml
@@ -498,7 +498,7 @@
 </histogram>
 
 <histogram name="Arc.AppShortcutSearchResult.ShortcutStatus"
-    enum="ArcAppShortcutStatus" expires_after="2022-12-04">
+    enum="ArcAppShortcutStatus" expires_after="2023-06-10">
   <owner>batoon@google.com</owner>
   <owner>arc-core@google.com</owner>
   <summary>
@@ -509,7 +509,7 @@
 </histogram>
 
 <histogram name="Arc.AppShortcutsRequest.ShortcutStatus"
-    enum="ArcAppShortcutStatus" expires_after="2022-12-04">
+    enum="ArcAppShortcutStatus" expires_after="2023-06-10">
   <owner>batoon@google.com</owner>
   <owner>arc-core@google.com</owner>
   <summary>
diff --git a/tools/metrics/histograms/metadata/ash/histograms.xml b/tools/metrics/histograms/metadata/ash/histograms.xml
index 7f72314..1a8060f 100644
--- a/tools/metrics/histograms/metadata/ash/histograms.xml
+++ b/tools/metrics/histograms/metadata/ash/histograms.xml
@@ -346,6 +346,22 @@
   </summary>
 </histogram>
 
+<histogram name="Ash.ArcAppInitialAppsInstallDuration" units="ms"
+    expires_after="2023-12-01">
+  <owner>alemate@chromium.org</owner>
+  <owner>khmel@chromium.org</owner>
+  <owner>arc-performance@google.com</owner>
+  <owner>chromeos-perfmetrics-eng@google.com</owner>
+  <summary>
+    For each user after they opted-in into ARC++, records the duration of
+    initial apps installation. Duration is measured as an interval between the
+    opt-in and the moment when all default apps become either &quot;ready&quot;
+    or &quot;update error&quot;. This is only reported within the same opt-in
+    session, and is not reported if default apps were not installed within the
+    session.
+  </summary>
+</histogram>
+
 <histogram
     name="Ash.Assistant.AnimationSmoothness{AshAssistantAnimationSmoothness}"
     units="%" expires_after="2021-11-21">
@@ -3323,6 +3339,25 @@
   </summary>
 </histogram>
 
+<histogram name="Ash.NotifierFramework.Nudge.TimeToAction.{TimeRange}"
+    enum="NudgeCatalogName" expires_after="2023-02-10">
+  <owner>kradtke@google.com</owner>
+  <owner>cros-status-area-eng@google.com</owner>
+  <summary>
+    Tracks the time from when a specific educational nudge is shown to when it's
+    interacted with. Starts measuring time when the nudge is shown and records
+    the Nudge catalog name in one of the time buckets available if the user
+    performs the expected action that the nudge is informing about. This metric
+    can be compared with `Ash.NotifierFramework.Nudge.ShownCount` to infer the
+    nudge action wasn't taken within that session.
+  </summary>
+  <token key="TimeRange">
+    <variant name="Within1h"/>
+    <variant name="Within1m"/>
+    <variant name="WithinSession"/>
+  </token>
+</histogram>
+
 <histogram name="Ash.NotifierFramework.Toast.Dismissed.{TimeRange}"
     enum="ToastCatalogName" expires_after="2023-02-10">
   <owner>kradtke@google.com</owner>
diff --git a/tools/metrics/histograms/metadata/download/histograms.xml b/tools/metrics/histograms/metadata/download/histograms.xml
index 54d1cf78..04f8ca9 100644
--- a/tools/metrics/histograms/metadata/download/histograms.xml
+++ b/tools/metrics/histograms/metadata/download/histograms.xml
@@ -1447,6 +1447,19 @@
   <summary>Records events for local video thumbnail retrieval.</summary>
 </histogram>
 
+<histogram name="Download.WarningData.ActionAdded" enum="DownloadWarningAction"
+    expires_after="2023-06-07">
+  <owner>xinghuilu@chromium.org</owner>
+  <owner>chrome-counter-abuse-alerts@google.com</owner>
+  <summary>
+    Records the warning action that is successfully added in warning action
+    event. Logged each time an action is taken on the download warning. For the
+    SHOWN action, only logged for the first time. Not logged if the action is
+    not added because the first warning is missing or the events have exceeded
+    the max length.
+  </summary>
+</histogram>
+
 <histogram name="Download.WarningData.AddWarningActionEventOutcome"
     enum="DownloadAddWarningActionEventOutcome" expires_after="2023-05-22">
   <owner>xinghuilu@chromium.org</owner>
diff --git a/tools/metrics/histograms/metadata/gpu/histograms.xml b/tools/metrics/histograms/metadata/gpu/histograms.xml
index df5966a..7bcca37 100644
--- a/tools/metrics/histograms/metadata/gpu/histograms.xml
+++ b/tools/metrics/histograms/metadata/gpu/histograms.xml
@@ -1129,7 +1129,7 @@
 </histogram>
 
 <histogram name="GPU.PaintOpReader.DeserializationError"
-    enum="PaintOpDeserializationError" expires_after="2023-01-22">
+    enum="PaintOpDeserializationError" expires_after="2024-01-22">
   <owner>junov@chromium.org</owner>
   <owner>graphics-dev@chromium.org</owner>
   <summary>
diff --git a/tools/metrics/histograms/metadata/history/histograms.xml b/tools/metrics/histograms/metadata/history/histograms.xml
index ed3fac18c..7fff9a1 100644
--- a/tools/metrics/histograms/metadata/history/histograms.xml
+++ b/tools/metrics/histograms/metadata/history/histograms.xml
@@ -1353,6 +1353,27 @@
   </token>
 </histogram>
 
+<histogram name="History.Clusters.UpdateClusters.TimeBetweenTasks" units="ms"
+    expires_after="2023-10-01">
+  <owner>manukh@chromium.org</owner>
+  <owner>chrome-journeys@google.com</owner>
+  <component>UI&gt;Browser&gt;Journeys</component>
+  <summary>
+    The time elapsed between subsequent cluster update tasks
+    (HistoryClustersServiceTaskUpdateClusters). Depending on finch params, this
+    occurs either:
+
+    1) Never (e.g. if updating and persisting clusters is disabled).
+
+    2) OR N time after startup in addition to every M time; where N is typically
+    5-60 minutes, and M is typically 1-12 hours.
+
+    3) OR on keyword refresh request bounded to at most every M time, where M is
+    typically 1-12 hours. Keyword refresh requests occur when the omnibox input
+    looks like a search query and journeys in the omnibox is enabled.
+  </summary>
+</histogram>
+
 <histogram name="History.Clusters.WebUISessionDuration" units="ms"
     expires_after="2023-04-16">
   <owner>tommycli@chromium.org</owner>
diff --git a/tools/metrics/histograms/metadata/others/histograms.xml b/tools/metrics/histograms/metadata/others/histograms.xml
index fa42a9a..b59c6ed 100644
--- a/tools/metrics/histograms/metadata/others/histograms.xml
+++ b/tools/metrics/histograms/metadata/others/histograms.xml
@@ -7983,7 +7983,7 @@
 </histogram>
 
 <histogram base="true" name="LoadingPredictor.PreconnectLearningPrecision"
-    units="%" expires_after="2022-12-19">
+    units="%" expires_after="2023-12-12">
   <owner>spelchat@chromium.org</owner>
   <owner>chrome-brapp-loading@chromium.org</owner>
   <summary>
@@ -8018,7 +8018,7 @@
 </histogram>
 
 <histogram name="LoadingPredictor.PreresolveCount" units="hosts"
-    expires_after="2022-12-19">
+    expires_after="2023-12-12">
   <owner>spelchat@chromium.org</owner>
   <owner>chrome-brapp-loading@chromium.org</owner>
   <summary>
@@ -8029,7 +8029,7 @@
 </histogram>
 
 <histogram name="LoadingPredictor.PreresolveHitsPercentage" units="%"
-    expires_after="2022-12-19">
+    expires_after="2023-12-12">
   <owner>spelchat@chromium.org</owner>
   <owner>chrome-brapp-loading@chromium.org</owner>
   <summary>
diff --git a/tools/metrics/histograms/metadata/startup/histograms.xml b/tools/metrics/histograms/metadata/startup/histograms.xml
index 0fb01294..f4c76d3 100644
--- a/tools/metrics/histograms/metadata/startup/histograms.xml
+++ b/tools/metrics/histograms/metadata/startup/histograms.xml
@@ -219,7 +219,7 @@
 </histogram>
 
 <histogram name="Startup.Android.FeedsLoadingPlaceholderShown.Instant"
-    units="ms" expires_after="2023-01-15">
+    units="ms" expires_after="2023-06-30">
   <owner>hanxi@chromium.org</owner>
   <owner>wychen@chromium.org</owner>
   <summary>
diff --git a/tools/metrics/histograms/metadata/tab/histograms.xml b/tools/metrics/histograms/metadata/tab/histograms.xml
index c69d0ec..65db5f3 100644
--- a/tools/metrics/histograms/metadata/tab/histograms.xml
+++ b/tools/metrics/histograms/metadata/tab/histograms.xml
@@ -266,6 +266,16 @@
   </summary>
 </histogram>
 
+<histogram name="Tab.AgeAtDeletion" units="ms" expires_after="2023-12-13">
+  <owner>ewannpv@chromium.org</owner>
+  <owner>gambard@chromium.org</owner>
+  <owner>bling-team@google.com</owner>
+  <summary>
+    The age of a Tab when closed (Time between creation time on the current
+    device and closure time).
+  </summary>
+</histogram>
+
 <histogram name="Tab.AgeUponRestoreFromColdStart" units="minutes"
     expires_after="M88">
   <owner>dtrainor@chromium.org</owner>
diff --git a/tools/metrics/structured/structured.xml b/tools/metrics/structured/structured.xml
index 33930cc8..7c449c5 100644
--- a/tools/metrics/structured/structured.xml
+++ b/tools/metrics/structured/structured.xml
@@ -416,6 +416,13 @@
     Project used to record a sequence of events that are related to each other.
   </summary>
 
+  <event name="UserLogin">
+    <summary>
+      An event to signify a user is using the system.
+    </summary>
+
+  </event>
+
   <event name="Test1">
     <summary>
       Test event used for unit tests to ensure that the code-gen for the event
diff --git a/tools/perf/core/bot_platforms.py b/tools/perf/core/bot_platforms.py
index d83fe216..3a45dcbb 100644
--- a/tools/perf/core/bot_platforms.py
+++ b/tools/perf/core/bot_platforms.py
@@ -681,7 +681,7 @@
     'android-pixel4a_power-perf-pgo', 'Android QD4A.200102.001.A1',
     _ANDROID_PIXEL4A_POWER_BENCHMARK_CONFIGS, 12, 'android')
 ANDROID_GO_WEMBLEY = PerfPlatform('android-go-wembley-perf',
-                                  'Android M | MASTER',
+                                  'Android 9131443',
                                   _ANDROID_GO_WEMBLEY_BENCHMARK_CONFIGS, 2,
                                   'android')
 ANDROID_NEW_PIXEL = PerfPlatform('android-new-pixel-perf',
diff --git a/tools/perf/core/perf_data_generator.py b/tools/perf/core/perf_data_generator.py
index 888e428d..8bde4a9 100755
--- a/tools/perf/core/perf_data_generator.py
+++ b/tools/perf/core/perf_data_generator.py
@@ -904,8 +904,7 @@
         'dimension': {
             'pool': 'chrome.tests.perf',
             'os': 'Android',
-            'device_type': 'wembley',
-            'device_os': 'MASTER',
+            'device_type': 'wembley_2GB',
             'device_os_flavor': 'google',
         },
     },
diff --git a/tools/perf/core/perf_json_config_validator.py b/tools/perf/core/perf_json_config_validator.py
index 3a88d3e..907c3bd 100644
--- a/tools/perf/core/perf_json_config_validator.py
+++ b/tools/perf/core/perf_json_config_validator.py
@@ -68,7 +68,6 @@
         raise ValueError('Invalid perf pool %s in %s' % (v, builder_name))
       if k == 'os' and v == 'Android':
         if (not 'device_type' in dimension.keys() or
-            not 'device_os' in dimension.keys() or
             not 'device_os_flavor' in dimension.keys()):
           raise ValueError(
               'Invalid android dimensions %s in %s' % (v, builder_name))
diff --git a/tools/perf/core/perfetto_binary_roller/binary_deps.json b/tools/perf/core/perfetto_binary_roller/binary_deps.json
index 0d90eae..24101be 100644
--- a/tools/perf/core/perfetto_binary_roller/binary_deps.json
+++ b/tools/perf/core/perfetto_binary_roller/binary_deps.json
@@ -5,8 +5,8 @@
             "full_remote_path": "perfetto-luci-artifacts/v31.0/linux-arm64/trace_processor_shell"
         },
         "win": {
-            "hash": "ecbfa1398bd09e5a57bc5218aa9c7aec691cce25",
-            "full_remote_path": "chromium-telemetry/perfetto_binaries/trace_processor_shell/win/ae7f044c460c89dac35657f618549998d665d7eb/trace_processor_shell.exe"
+            "hash": "cc817f9c3998e1c2dea11752c61456413c98f1f6",
+            "full_remote_path": "chromium-telemetry/perfetto_binaries/trace_processor_shell/win/aa7e493dcf9bf77707735b98723b916e08561bbe/trace_processor_shell.exe"
         },
         "linux_arm": {
             "hash": "6373f26144aad58f230d11d6a91efda5a09c9873",
@@ -21,8 +21,8 @@
             "full_remote_path": "perfetto-luci-artifacts/v31.0/mac-arm64/trace_processor_shell"
         },
         "linux": {
-            "hash": "47d0feffdbcd8c0dfe7650dd0874b4c7a291153f",
-            "full_remote_path": "chromium-telemetry/perfetto_binaries/trace_processor_shell/linux/ae7f044c460c89dac35657f618549998d665d7eb/trace_processor_shell"
+            "hash": "375e45389f78aec067a6df88497229d65190aa06",
+            "full_remote_path": "chromium-telemetry/perfetto_binaries/trace_processor_shell/linux/aa7e493dcf9bf77707735b98723b916e08561bbe/trace_processor_shell"
         }
     },
     "power_profile.sql": {
diff --git a/tools/perf/core/shard_maps/android-go-wembley-perf_map.json b/tools/perf/core/shard_maps/android-go-wembley-perf_map.json
index 89006ec..b0494fa 100644
--- a/tools/perf/core/shard_maps/android-go-wembley-perf_map.json
+++ b/tools/perf/core/shard_maps/android-go-wembley-perf_map.json
@@ -3,23 +3,28 @@
         "benchmarks": {
             "speedometer2": {
                 "abridged": false
+            },
+            "startup.mobile": {
+                "end": 2,
+                "abridged": false
             }
         }
     },
     "1": {
         "benchmarks": {
             "startup.mobile": {
+                "begin": 2,
                 "abridged": false
             }
         }
     },
     "extra_infos": {
         "num_stories": 5,
-        "predicted_min_shard_time": 40,
+        "predicted_min_shard_time": 20,
         "predicted_min_shard_index": 1,
-        "predicted_max_shard_time": 140.0,
+        "predicted_max_shard_time": 30,
         "predicted_max_shard_index": 0,
-        "shard #0": 140.0,
-        "shard #1": 40
+        "shard #0": 30,
+        "shard #1": 20
     }
 }
\ No newline at end of file
diff --git a/tools/perf/core/shard_maps/timing_data/android-go-wembley-perf_timing.json b/tools/perf/core/shard_maps/timing_data/android-go-wembley-perf_timing.json
index 08f2d08..0637a08 100644
--- a/tools/perf/core/shard_maps/timing_data/android-go-wembley-perf_timing.json
+++ b/tools/perf/core/shard_maps/timing_data/android-go-wembley-perf_timing.json
@@ -1,6 +1 @@
-[
-    {
-        "duration": "140.0",
-        "name": "speedometer2/Speedometer2"
-    }
-]
\ No newline at end of file
+[]
\ No newline at end of file
diff --git a/tools/traffic_annotation/auditor/chromeos/safe_list.txt b/tools/traffic_annotation/auditor/chromeos/safe_list.txt
index 96fc26073..e79f113 100644
--- a/tools/traffic_annotation/auditor/chromeos/safe_list.txt
+++ b/tools/traffic_annotation/auditor/chromeos/safe_list.txt
@@ -11,7 +11,6 @@
 all,chromeos/ash/services/device_sync/cryptauth_client_impl.cc
 all,chrome/browser/supervised_user/kids_chrome_management/kids_chrome_management_client.cc
 all,chrome/browser/ash/net/network_diagnostics/network_diagnostics_util.cc
-all,chrome/browser/ash/net/network_diagnostics/http_request_manager.cc
 all,chrome/browser/ash/net/network_diagnostics/tls_prober.cc
 all,ash/quick_pair/repository/fast_pair/fast_pair_image_decoder.cc
 all,chrome/browser/search/background/ntp_background_service.cc
diff --git a/tools/traffic_annotation/summary/annotations.xml b/tools/traffic_annotation/summary/annotations.xml
index ca60eed2..f637e36 100644
--- a/tools/traffic_annotation/summary/annotations.xml
+++ b/tools/traffic_annotation/summary/annotations.xml
@@ -405,4 +405,5 @@
  <item id="projector_xhr_loader" added_in_milestone="110" content_hash_code="071c4ac5" os_list="chromeos" file_path="ash/webui/projector_app/projector_xhr_sender.cc" />
  <item id="bruschetta_installer_download" added_in_milestone="110" content_hash_code="01b953f4" os_list="chromeos" file_path="chrome/browser/ash/bruschetta/bruschetta_installer.cc" />
  <item id="quick_start_session_auth_requester" added_in_milestone="110" content_hash_code="08353e70" os_list="chromeos" file_path="chrome/browser/ash/login/oobe_quick_start/second_device_auth_broker.cc" />
+ <item id="network_diagnostics_routines" added_in_milestone="110" content_hash_code="0007b237" os_list="chromeos" file_path="chrome/browser/ash/net/network_diagnostics/http_request_manager.cc" />
 </annotations>
diff --git a/tools/traffic_annotation/summary/grouping.xml b/tools/traffic_annotation/summary/grouping.xml
index 62326cd..64aa0f0a 100644
--- a/tools/traffic_annotation/summary/grouping.xml
+++ b/tools/traffic_annotation/summary/grouping.xml
@@ -272,6 +272,7 @@
       <annotation id="projector_xhr_loader"/>
       <annotation id="bruschetta_installer_download"/>
       <annotation id="quick_start_session_auth_requester"/>
+      <annotation id="network_diagnostics_routines"/>
     </sender>
   </group>
   <group name="Admin Features" hidden="true">
diff --git a/ui/android/BUILD.gn b/ui/android/BUILD.gn
index 67289b87..8f6206c 100644
--- a/ui/android/BUILD.gn
+++ b/ui/android/BUILD.gn
@@ -131,6 +131,14 @@
 
 android_resources("ui_java_resources") {
   sources = [
+    "java/res/anim/menu_enter.xml",
+    "java/res/anim/menu_enter_from_bottom.xml",
+    "java/res/anim/menu_enter_from_bottom_left.xml",
+    "java/res/anim/menu_enter_from_top_left.xml",
+    "java/res/anim/menu_exit.xml",
+    "java/res/anim/menu_exit_from_bottom.xml",
+    "java/res/anim/menu_exit_from_bottom_left.xml",
+    "java/res/anim/menu_exit_from_top_left.xml",
     "java/res/color/blue_when_enabled_dark.xml",
     "java/res/color/blue_when_enabled_list.xml",
     "java/res/color/default_text_color_light_list.xml",
@@ -179,6 +187,7 @@
     "java/res/values-ldrtl/values.xml",
     "java/res/values-night/colors.xml",
     "java/res/values-night/dimens.xml",
+    "java/res/values-sw600dp/dimens.xml",
     "java/res/values-sw600dp/values.xml",
     "java/res/values-sw720dp-v17/values.xml",
     "java/res/values-v17/styles.xml",
@@ -193,6 +202,7 @@
     "java/res/values/semantic_colors_adaptive.xml",
     "java/res/values/semantic_colors_non_adaptive.xml",
     "java/res/values/strings.xml",
+    "java/res/values/styles.xml",
     "java/res/values/values.xml",
   ]
 
diff --git a/components/browser_ui/widget/android/java/res/anim/menu_enter.xml b/ui/android/java/res/anim/menu_enter.xml
similarity index 100%
rename from components/browser_ui/widget/android/java/res/anim/menu_enter.xml
rename to ui/android/java/res/anim/menu_enter.xml
diff --git a/components/browser_ui/widget/android/java/res/anim/menu_enter_from_bottom.xml b/ui/android/java/res/anim/menu_enter_from_bottom.xml
similarity index 100%
rename from components/browser_ui/widget/android/java/res/anim/menu_enter_from_bottom.xml
rename to ui/android/java/res/anim/menu_enter_from_bottom.xml
diff --git a/components/browser_ui/widget/android/java/res/anim/menu_enter_from_bottom_left.xml b/ui/android/java/res/anim/menu_enter_from_bottom_left.xml
similarity index 100%
rename from components/browser_ui/widget/android/java/res/anim/menu_enter_from_bottom_left.xml
rename to ui/android/java/res/anim/menu_enter_from_bottom_left.xml
diff --git a/components/browser_ui/widget/android/java/res/anim/menu_enter_from_top_left.xml b/ui/android/java/res/anim/menu_enter_from_top_left.xml
similarity index 100%
rename from components/browser_ui/widget/android/java/res/anim/menu_enter_from_top_left.xml
rename to ui/android/java/res/anim/menu_enter_from_top_left.xml
diff --git a/components/browser_ui/widget/android/java/res/anim/menu_exit.xml b/ui/android/java/res/anim/menu_exit.xml
similarity index 100%
rename from components/browser_ui/widget/android/java/res/anim/menu_exit.xml
rename to ui/android/java/res/anim/menu_exit.xml
diff --git a/components/browser_ui/widget/android/java/res/anim/menu_exit_from_bottom.xml b/ui/android/java/res/anim/menu_exit_from_bottom.xml
similarity index 100%
rename from components/browser_ui/widget/android/java/res/anim/menu_exit_from_bottom.xml
rename to ui/android/java/res/anim/menu_exit_from_bottom.xml
diff --git a/components/browser_ui/widget/android/java/res/anim/menu_exit_from_bottom_left.xml b/ui/android/java/res/anim/menu_exit_from_bottom_left.xml
similarity index 100%
rename from components/browser_ui/widget/android/java/res/anim/menu_exit_from_bottom_left.xml
rename to ui/android/java/res/anim/menu_exit_from_bottom_left.xml
diff --git a/components/browser_ui/widget/android/java/res/anim/menu_exit_from_top_left.xml b/ui/android/java/res/anim/menu_exit_from_top_left.xml
similarity index 100%
rename from components/browser_ui/widget/android/java/res/anim/menu_exit_from_top_left.xml
rename to ui/android/java/res/anim/menu_exit_from_top_left.xml
diff --git a/ui/android/java/res/values-ldrtl/values.xml b/ui/android/java/res/values-ldrtl/values.xml
index 913ceb9..6e02d750 100644
--- a/ui/android/java/res/values-ldrtl/values.xml
+++ b/ui/android/java/res/values-ldrtl/values.xml
@@ -9,4 +9,8 @@
     <!--  Expand more  -->
     <integer name="expand_more_horizontal_rotation_degree"
         tools:ignore="UnusedResources">90</integer>
+
+    <!-- Menu Animation Constants -->
+    <item type="fraction" format="fraction" name="menu_animation_pivot_x">5%</item>
+    <item type="fraction" format="fraction" name="menu_start_animation_pivot_x">95%</item>
 </resources>
diff --git a/ui/android/java/res/values-sw600dp/dimens.xml b/ui/android/java/res/values-sw600dp/dimens.xml
new file mode 100644
index 0000000..26cc88c
--- /dev/null
+++ b/ui/android/java/res/values-sw600dp/dimens.xml
@@ -0,0 +1,12 @@
+<?xml version="1.0" encoding="utf-8"?>
+<!--
+Copyright 2019 The Chromium Authors
+Use of this source code is governed by a BSD-style license that can be
+found in the LICENSE file.
+-->
+
+<resources>
+      <!-- Menu Dimensions -->
+      <!-- Necessary to align the menu icon with the actual button. -->
+      <dimen name="menu_negative_software_vertical_offset">6dp</dimen>
+</resources>
\ No newline at end of file
diff --git a/ui/android/java/res/values/dimens.xml b/ui/android/java/res/values/dimens.xml
index afa55dc..7746c7d 100644
--- a/ui/android/java/res/values/dimens.xml
+++ b/ui/android/java/res/values/dimens.xml
@@ -40,6 +40,9 @@
     <dimen name="menu_footer_chip_icon_size">16dp</dimen>
     <dimen name="menu_footer_chip_vertical_inset">8dp</dimen>
 
+    <!-- Custom menu-->
+    <dimen name="menu_negative_software_vertical_offset">0dp</dimen>
+
     <!-- Dropdown default measures -->
     <dimen name="dropdown_item_height">50dp</dimen>
     <dimen name="dropdown_item_divider_height">1px</dimen>
diff --git a/ui/android/java/res/values/styles.xml b/ui/android/java/res/values/styles.xml
new file mode 100644
index 0000000..37042d7
--- /dev/null
+++ b/ui/android/java/res/values/styles.xml
@@ -0,0 +1,26 @@
+<?xml version="1.0" encoding="utf-8"?>
+<!--
+Copyright 2022 The Chromium Authors
+Use of this source code is governed by a BSD-style license that can be
+found in the LICENSE file.
+-->
+
+<resources xmlns:tools="http://schemas.android.com/tools">
+    <!-- AnchoredPopupAnimationStyle -->
+    <style name="AnchoredPopupAnimEndTop">
+        <item name="android:windowEnterAnimation">@anim/menu_enter</item>
+        <item name="android:windowExitAnimation">@anim/menu_exit</item>
+    </style>
+    <style name="AnchoredPopupAnimEndBottom">
+        <item name="android:windowEnterAnimation">@anim/menu_enter_from_bottom</item>
+        <item name="android:windowExitAnimation">@anim/menu_exit_from_bottom</item>
+    </style>
+    <style name="AnchoredPopupAnimStartTop">
+        <item name="android:windowEnterAnimation">@anim/menu_enter_from_top_left</item>
+        <item name="android:windowExitAnimation">@anim/menu_exit_from_top_left</item>
+    </style>
+    <style name="AnchoredPopupAnimStartBottom">
+        <item name="android:windowEnterAnimation">@anim/menu_enter_from_bottom_left</item>
+        <item name="android:windowExitAnimation">@anim/menu_exit_from_bottom_left</item>
+    </style>
+</resources>
diff --git a/ui/android/java/res/values/values.xml b/ui/android/java/res/values/values.xml
index e86eec9..6de551e 100644
--- a/ui/android/java/res/values/values.xml
+++ b/ui/android/java/res/values/values.xml
@@ -16,4 +16,8 @@
     <!-- Drag shadow resize ratio -->
     <item name="drag_shadow_resize_ratio" format="float" type="dimen">0.6</item>
     <item name="drag_shadow_max_size_to_window_ratio" format="float" type="dimen">0.35</item>
+
+    <!-- Menu Animation Constants -->
+    <item type="fraction" format="fraction" name="menu_animation_pivot_x">95%</item>
+    <item type="fraction" format="fraction" name="menu_start_animation_pivot_x">5%</item>
 </resources>
diff --git a/ui/android/java/src/org/chromium/ui/widget/AnchoredPopupWindow.java b/ui/android/java/src/org/chromium/ui/widget/AnchoredPopupWindow.java
index a2d5e54..ff7a961 100644
--- a/ui/android/java/src/org/chromium/ui/widget/AnchoredPopupWindow.java
+++ b/ui/android/java/src/org/chromium/ui/widget/AnchoredPopupWindow.java
@@ -22,10 +22,12 @@
 import android.widget.PopupWindow.OnDismissListener;
 
 import androidx.annotation.IntDef;
+import androidx.annotation.StyleRes;
 import androidx.annotation.VisibleForTesting;
 
 import org.chromium.base.ObserverList;
 import org.chromium.base.metrics.RecordUserAction;
+import org.chromium.ui.R;
 
 import java.lang.annotation.Retention;
 import java.lang.annotation.RetentionPolicy;
@@ -186,6 +188,9 @@
     private boolean mUpdateOrientationOnChange;
     private boolean mSmartAnchorWithMaxWidth;
 
+    private @StyleRes int mAnimationStyleId;
+    private boolean mAnimateFromAnchor;
+
     /**
      * Constructs an {@link AnchoredPopupWindow} instance.
      * @param context  Context to draw resources from.
@@ -313,15 +318,27 @@
     }
 
     /**
-     * Sets the animation style for the popup.  This should be called before the popup is shown.
+     * Sets the animation style for the popup. This should be called before the popup is shown.
+     * Setting this style will take precedence over {@link #setAnimateFromAnchor(boolean)}.
      * @param animationStyleId The id of the animation style.
      */
     public void setAnimationStyle(int animationStyleId) {
+        mAnimationStyleId = animationStyleId;
         mPopupWindow.setAnimationStyle(animationStyleId);
     }
 
     /**
-     * If set to true, orientation will be updated everytime that the {@link OnRectChanged} is
+     * Set whether the popup should enter from / exit to the anchor point. This should be
+     * called before the popup is shown. If an animation style is specified by
+     * {@link #setAnimationStyle(int)}, this method will have no effect.
+     * @param animateFromAnchor Whether the popup should animator from anchor point.
+     */
+    public void setAnimateFromAnchor(boolean animateFromAnchor) {
+        mAnimateFromAnchor = animateFromAnchor;
+    }
+
+    /**
+     * If set to true, orientation will be updated every time that the {@link OnRectChanged} is
      * called.
      */
     public void setUpdateOrientationOnChange(boolean updateOrientationOnChange) {
@@ -728,7 +745,35 @@
         return value;
     }
 
-    private void showPopupWindow() {
+    /**
+     * Calculate the style Id to use when showing the popup window,
+     * when {@link #setAnimateFromAnchor(true)}.
+     * @param isPositionBelow Whether the popup is positioned below the anchor rect.
+     * @param isPositionToLeft Whether the popup is positioned below the anchor rect.
+     * @return The style resource Id to use for {@link PopupWindow#setAnimationStyle}
+     */
+    @VisibleForTesting
+    static @StyleRes int calculateAnimationStyle(
+            boolean isPositionBelow, boolean isPositionToLeft) {
+        if (isPositionToLeft) {
+            return isPositionBelow
+                    ? R.style.AnchoredPopupAnimEndTop // Left + below -> enter top right (end)
+                    : R.style.AnchoredPopupAnimEndBottom; // Left + above -> enter bottom right
+                                                          // (end)
+        }
+        return isPositionBelow
+                ? R.style.AnchoredPopupAnimStartTop // Right & below -> enter top left (start)
+                : R.style.AnchoredPopupAnimStartBottom; // Right & above -> enter bottom left
+                                                        // (start)
+    }
+
+    @VisibleForTesting
+    void showPopupWindow() {
+        if (mAnimateFromAnchor && mAnimationStyleId == 0) {
+            int animationStyle = calculateAnimationStyle(
+                    mPopupSpec.isPositionBelow, mPopupSpec.isPositionToLeft);
+            mPopupWindow.setAnimationStyle(animationStyle);
+        }
         try {
             mPopupWindow.showAtLocation(mRootView, Gravity.TOP | Gravity.START,
                     mPopupSpec.popupRect.left, mPopupSpec.popupRect.top);
diff --git a/ui/android/junit/src/org/chromium/ui/widget/AnchoredPopupWindowTest.java b/ui/android/junit/src/org/chromium/ui/widget/AnchoredPopupWindowTest.java
index 7ac05767..e038de1 100644
--- a/ui/android/junit/src/org/chromium/ui/widget/AnchoredPopupWindowTest.java
+++ b/ui/android/junit/src/org/chromium/ui/widget/AnchoredPopupWindowTest.java
@@ -7,11 +7,20 @@
 import static org.junit.Assert.assertEquals;
 import static org.junit.Assert.assertFalse;
 import static org.junit.Assert.assertTrue;
+import static org.mockito.ArgumentMatchers.any;
+import static org.mockito.ArgumentMatchers.anyInt;
+import static org.mockito.Mockito.doReturn;
+import static org.mockito.Mockito.mock;
+import static org.mockito.Mockito.times;
+import static org.mockito.Mockito.verify;
 
 import android.app.Activity;
 import android.graphics.Rect;
+import android.view.View;
 import android.widget.FrameLayout;
+import android.widget.PopupWindow;
 
+import org.junit.After;
 import org.junit.Assert;
 import org.junit.Before;
 import org.junit.Test;
@@ -21,6 +30,7 @@
 import org.robolectric.shadows.ShadowView;
 
 import org.chromium.base.test.BaseRobolectricTestRunner;
+import org.chromium.ui.R;
 import org.chromium.ui.widget.AnchoredPopupWindow.HorizontalOrientation;
 import org.chromium.ui.widget.AnchoredPopupWindow.PopupSpec;
 import org.chromium.ui.widget.AnchoredPopupWindow.VerticalOrientation;
@@ -53,6 +63,7 @@
     private boolean mSmartAnchorWithMaxWidth;
 
     private FrameLayout mContentView;
+    private Activity mActivity;
 
     @Before
     public void setUp() {
@@ -62,14 +73,21 @@
         mPopupHeight = 300;
         mWindowRect = new Rect(0, 0, mRootWidth, mRootHeight);
 
-        final Activity activity = Robolectric.buildActivity(Activity.class).get();
-        mContentView = new FrameLayout(activity);
+        mActivity = Robolectric.buildActivity(Activity.class).get();
+
+        mContentView = new FrameLayout(mActivity);
         mContentView.setMinimumWidth(mPopupWidth);
         mContentView.setMinimumHeight(mPopupHeight);
 
         setDefaultValueForAnchoredPopup();
     }
 
+    @After
+    public void tearDown() {
+        mActivity.finish();
+        UiWidgetFactory.setInstance(null);
+    }
+
     @Test
     public void testGetPopupPosition_BelowRight() {
         Rect anchorRect = new Rect(10, 10, 20, 20);
@@ -600,6 +618,71 @@
                 /*expectedPopupRect*/ new Rect(350, 800, 500, 1000));
     }
 
+    @Test
+    public void calculateAnimationStyleStartTop() {
+        assertEquals("Position below right -> animate from start top.",
+                R.style.AnchoredPopupAnimStartTop,
+                AnchoredPopupWindow.calculateAnimationStyle(/*isPositionBelow*/ true,
+                        /*isPositionToLeft*/ false));
+    }
+
+    @Test
+    public void calculateAnimationStyleStartBottom() {
+        assertEquals("Position above right -> animate from start bottom.",
+                R.style.AnchoredPopupAnimStartBottom,
+                AnchoredPopupWindow.calculateAnimationStyle(/*isPositionBelow*/ false,
+                        /*isPositionToLeft*/ false));
+    }
+
+    @Test
+    public void calculateAnimationStyleEndTop() {
+        assertEquals("Position below left -> animate from end top.",
+                R.style.AnchoredPopupAnimEndTop,
+                AnchoredPopupWindow.calculateAnimationStyle(/*isPositionBelow*/ true,
+                        /*isPositionToLeft*/ true));
+    }
+
+    @Test
+    public void calculateAnimationStyleEndBottom() {
+        assertEquals("Position above left -> animate from end bottom.",
+                R.style.AnchoredPopupAnimEndBottom,
+                AnchoredPopupWindow.calculateAnimationStyle(/*isPositionBelow*/ false,
+                        /*isPositionToLeft*/ true));
+    }
+
+    @Test
+    public void setAnimateFromAnchor() {
+        // Set up for test case, so we have a mock popup window.
+        UiWidgetFactory mockFactory = mock(UiWidgetFactory.class);
+        UiWidgetFactory.setInstance(mockFactory);
+
+        PopupWindow mockPopup = mock(PopupWindow.class);
+        doReturn(mockPopup).when(mockFactory).createPopupWindow(any());
+
+        AnchoredPopupWindow popupWindow = createAnchorPopupWindow();
+        popupWindow.setAnimateFromAnchor(true);
+        popupWindow.showPopupWindow();
+        verify(mockPopup).setAnimationStyle(anyInt());
+    }
+
+    @Test
+    public void setAnimationStyleNotOverrideByAnimateFromAnchor() {
+        // Set up for test case, so we have a mock popup window.
+        UiWidgetFactory mockFactory = mock(UiWidgetFactory.class);
+        UiWidgetFactory.setInstance(mockFactory);
+        PopupWindow mockPopup = mock(PopupWindow.class);
+        doReturn(mockPopup).when(mockFactory).createPopupWindow(any());
+
+        AnchoredPopupWindow popupWindow = createAnchorPopupWindow();
+        popupWindow.setAnimationStyle(R.style.Animation_AppCompat_Dialog);
+        verify(mockPopup).setAnimationStyle(R.style.Animation_AppCompat_Dialog);
+
+        popupWindow.setAnimateFromAnchor(true);
+        popupWindow.showPopupWindow();
+        // setAnimationStyle should only called once, since #setAnimateFromAnchor is no-op.
+        verify(mockPopup, times(1)).setAnimationStyle(anyInt());
+    }
+
     private void setDefaultValueForAnchoredPopup() {
         mPaddingX = 0;
         mPaddingY = 0;
@@ -632,4 +715,10 @@
                 String.format("PopupRect does not match expected Rect. Test case:<%s>", testCase),
                 expectedRect, popupRect);
     }
+
+    private AnchoredPopupWindow createAnchorPopupWindow() {
+        View rootView = mock(View.class);
+        RectProvider provider = new RectProvider(new Rect(0, 0, 0, 0));
+        return new AnchoredPopupWindow(mActivity, rootView, null, mContentView, provider);
+    }
 }
diff --git a/ui/base/clipboard/clipboard_mac.mm b/ui/base/clipboard/clipboard_mac.mm
index e554874..532fdfb 100644
--- a/ui/base/clipboard/clipboard_mac.mm
+++ b/ui/base/clipboard/clipboard_mac.mm
@@ -135,8 +135,8 @@
   // Safari only places RTF on the pasteboard, never HTML. We can convert RTF
   // to HTML, so the presence of either indicates success when looking for HTML.
   if (format == ClipboardFormatType::HtmlType()) {
-    return [types containsObject:NSHTMLPboardType] ||
-           [types containsObject:NSRTFPboardType];
+    return [types containsObject:NSPasteboardTypeHTML] ||
+           [types containsObject:NSPasteboardTypeRTF];
   }
   // Chrome can retrieve an image from the clipboard as either a bitmap or PNG.
   if (format == ClipboardFormatType::PngType() ||
@@ -266,14 +266,15 @@
 
   NSPasteboard* pb = GetPasteboard();
   NSArray* supportedTypes =
-      @[ NSHTMLPboardType, NSRTFPboardType, NSPasteboardTypeString ];
+      @[ NSPasteboardTypeHTML, NSPasteboardTypeRTF, NSPasteboardTypeString ];
   NSString* bestType = [pb availableTypeFromArray:supportedTypes];
   if (bestType) {
     NSString* contents;
-    if ([bestType isEqualToString:NSRTFPboardType])
+    if ([bestType isEqualToString:NSPasteboardTypeRTF]) {
       contents = ClipboardUtil::GetHTMLFromRTFOnPasteboard(pb);
-    else
+    } else {
       contents = [pb stringForType:bestType];
+    }
     *markup = base::SysNSStringToUTF16(contents);
   }
 
@@ -417,7 +418,7 @@
   NSString* html_fragment = base::SysUTF8ToNSString(html_fragment_str);
 
   // TODO(avi): url_data?
-  [GetPasteboard() setString:html_fragment forType:NSHTMLPboardType];
+  [GetPasteboard() setString:html_fragment forType:NSPasteboardTypeHTML];
 }
 
 void ClipboardMac::WriteSvg(const char* markup_data, size_t markup_len) {
diff --git a/ui/base/dragdrop/os_exchange_data_provider_mac.mm b/ui/base/dragdrop/os_exchange_data_provider_mac.mm
index 7f583d5..b5b03f0 100644
--- a/ui/base/dragdrop/os_exchange_data_provider_mac.mm
+++ b/ui/base/dragdrop/os_exchange_data_provider_mac.mm
@@ -364,10 +364,9 @@
 // static
 NSArray* OSExchangeDataProviderMac::SupportedPasteboardTypes() {
   return @[
-    kUTTypeChromiumWebCustomData, kUTTypeWebKitWebURLsWithTitles,
-    NSURLPboardType, NSFilenamesPboardType, kUTTypeChromiumInitiatedDrag,
-    NSStringPboardType, NSHTMLPboardType, NSRTFPboardType,
-    NSFilenamesPboardType, kUTTypeChromiumWebCustomData, NSPasteboardTypeString
+    kUTTypeChromiumInitiatedDrag, kUTTypeChromiumWebCustomData,
+    kUTTypeWebKitWebURLsWithTitles, NSFilenamesPboardType, NSPasteboardTypeHTML,
+    NSPasteboardTypeRTF, NSPasteboardTypeString, NSPasteboardTypeURL
   ];
 }
 
diff --git a/ui/compositor/test/in_process_context_factory.cc b/ui/compositor/test/in_process_context_factory.cc
index 36483ab..77501b9 100644
--- a/ui/compositor/test/in_process_context_factory.cc
+++ b/ui/compositor/test/in_process_context_factory.cc
@@ -55,10 +55,6 @@
 #include "ui/accelerated_widget_mac/ca_transaction_observer.h"
 #endif
 
-#if !defined(GPU_SURFACE_HANDLE_IS_ACCELERATED_WINDOW)
-#include "gpu/ipc/common/gpu_surface_tracker.h"
-#endif
-
 namespace ui {
 namespace {
 
@@ -318,10 +314,6 @@
   PerCompositorData* data = it->second.get();
   frame_sink_manager_->UnregisterBeginFrameSource(data->begin_frame_source());
   DCHECK(data);
-#if !defined(GPU_SURFACE_HANDLE_IS_ACCELERATED_WINDOW)
-  if (data->surface_handle())
-    gpu::GpuSurfaceTracker::Get()->RemoveSurface(data->surface_handle());
-#endif
   per_compositor_data_.erase(it);
 }
 
@@ -399,20 +391,6 @@
   } else {
 #if defined(GPU_SURFACE_HANDLE_IS_ACCELERATED_WINDOW)
     data->SetSurfaceHandle(widget);
-#else
-    gpu::GpuSurfaceTracker* tracker = gpu::GpuSurfaceTracker::Get();
-    data->SetSurfaceHandle(tracker->AddSurfaceForNativeWidget(
-        gpu::GpuSurfaceTracker::SurfaceRecord(
-            widget
-#if BUILDFLAG(IS_ANDROID)
-            // We have to provide a surface too, but we don't have one.  For
-            // now, we don't proide it, since nobody should ask anyway.
-            // If we ever provide a valid surface here, then GpuSurfaceTracker
-            // can be more strict about enforcing it.
-            ,
-            nullptr, false /* can_be_used_with_surface_control */
-#endif
-            )));
 #endif
   }
 
diff --git a/ui/gfx/buffer_format_util.h b/ui/gfx/buffer_format_util.h
index 01f17f55..35f4592f 100644
--- a/ui/gfx/buffer_format_util.h
+++ b/ui/gfx/buffer_format_util.h
@@ -29,6 +29,11 @@
 GFX_EXPORT size_t SubsamplingFactorForBufferFormat(BufferFormat format,
                                                    size_t plane);
 
+// Returns the alignment requirement to store a row of the given zero-indexed
+// |plane| of |format|.
+GFX_EXPORT size_t RowByteAlignmentForBufferFormat(BufferFormat format,
+                                                  size_t plane);
+
 // Returns the number of bytes used to store a row of the given zero-indexed
 // |plane| of |format|.
 GFX_EXPORT size_t RowSizeForBufferFormat(size_t width,
diff --git a/ui/gl/android/scoped_a_native_window.cc b/ui/gl/android/scoped_a_native_window.cc
index 8e93462..a73ea62 100644
--- a/ui/gl/android/scoped_a_native_window.cc
+++ b/ui/gl/android/scoped_a_native_window.cc
@@ -16,6 +16,9 @@
   return ScopedANativeWindow(a_native_window);
 }
 
+ScopedANativeWindow::ScopedANativeWindow() = default;
+ScopedANativeWindow::ScopedANativeWindow(std::nullptr_t) {}
+
 ScopedANativeWindow::ScopedANativeWindow(const ScopedJavaSurface& surface) {
   if (!surface.j_surface()) {
     return;
diff --git a/ui/gl/android/scoped_a_native_window.h b/ui/gl/android/scoped_a_native_window.h
index 339f384..a5e5785 100644
--- a/ui/gl/android/scoped_a_native_window.h
+++ b/ui/gl/android/scoped_a_native_window.h
@@ -5,6 +5,8 @@
 #ifndef UI_GL_ANDROID_SCOPED_A_NATIVE_WINDOW_H_
 #define UI_GL_ANDROID_SCOPED_A_NATIVE_WINDOW_H_
 
+#include <cstddef>
+
 #include "ui/gl/gl_export.h"
 
 struct ANativeWindow;
@@ -16,6 +18,8 @@
 class GL_EXPORT ScopedANativeWindow {
  public:
   static ScopedANativeWindow Wrap(ANativeWindow* a_native_window);
+  ScopedANativeWindow();
+  ScopedANativeWindow(std::nullptr_t);
   explicit ScopedANativeWindow(const ScopedJavaSurface& surface);
   ~ScopedANativeWindow();
 
diff --git a/ui/gl/android/scoped_java_surface.cc b/ui/gl/android/scoped_java_surface.cc
index 8ac2f58c..d3bf54a 100644
--- a/ui/gl/android/scoped_java_surface.cc
+++ b/ui/gl/android/scoped_java_surface.cc
@@ -14,14 +14,15 @@
 
 namespace gl {
 
-ScopedJavaSurface::ScopedJavaSurface() {
-}
+ScopedJavaSurface::ScopedJavaSurface() = default;
+ScopedJavaSurface::ScopedJavaSurface(std::nullptr_t) {}
 
 ScopedJavaSurface::ScopedJavaSurface(
-    const base::android::JavaRef<jobject>& surface) {
+    const base::android::JavaRef<jobject>& surface,
+    bool auto_release)
+    : auto_release_(auto_release), j_surface_(surface) {
   JNIEnv* env = base::android::AttachCurrentThread();
   DCHECK(env->IsInstanceOf(surface.obj(), android_view_Surface_clazz(env)));
-  j_surface_.Reset(surface);
 }
 
 ScopedJavaSurface::ScopedJavaSurface(const SurfaceTexture* surface_texture) {
@@ -45,6 +46,10 @@
   ReleaseSurfaceIfNeeded();
 }
 
+ScopedJavaSurface ScopedJavaSurface::CopyRetainOwnership() const {
+  return ScopedJavaSurface(j_surface_, /*auto_release=*/false);
+}
+
 void ScopedJavaSurface::ReleaseSurfaceIfNeeded() {
   if (auto_release_ && !j_surface_.is_null()) {
     JNIEnv* env = base::android::AttachCurrentThread();
@@ -56,7 +61,6 @@
   ReleaseSurfaceIfNeeded();
   j_surface_ = std::move(other.j_surface_);
   auto_release_ = other.auto_release_;
-  is_protected_ = other.is_protected_;
 }
 
 bool ScopedJavaSurface::IsEmpty() const {
@@ -68,13 +72,4 @@
   return !IsEmpty() && JNI_Surface::Java_Surface_isValidZ(env, j_surface_);
 }
 
-// static
-ScopedJavaSurface ScopedJavaSurface::AcquireExternalSurface(
-    const base::android::JavaRef<jobject>& surface) {
-  ScopedJavaSurface scoped_surface(surface);
-  scoped_surface.auto_release_ = false;
-  scoped_surface.is_protected_ = true;
-  return scoped_surface;
-}
-
 }  // namespace gl
diff --git a/ui/gl/android/scoped_java_surface.h b/ui/gl/android/scoped_java_surface.h
index b04b0a0..d11c73b5 100644
--- a/ui/gl/android/scoped_java_surface.h
+++ b/ui/gl/android/scoped_java_surface.h
@@ -6,6 +6,7 @@
 #define UI_GL_ANDROID_SCOPED_JAVA_SURFACE_H_
 
 #include <jni.h>
+#include <cstddef>
 
 #include "base/android/scoped_java_ref.h"
 #include "ui/gl/gl_export.h"
@@ -20,9 +21,11 @@
 class GL_EXPORT ScopedJavaSurface {
  public:
   ScopedJavaSurface();
+  ScopedJavaSurface(std::nullptr_t);
 
   // Wraps an existing Java Surface object in a ScopedJavaSurface.
-  explicit ScopedJavaSurface(const base::android::JavaRef<jobject>& surface);
+  ScopedJavaSurface(const base::android::JavaRef<jobject>& surface,
+                    bool auto_release);
 
   // Creates a Java Surface from a SurfaceTexture and wraps it in a
   // ScopedJavaSurface.
@@ -33,26 +36,21 @@
   ScopedJavaSurface(ScopedJavaSurface&& rvalue);
   ScopedJavaSurface& operator=(ScopedJavaSurface&& rhs);
 
-  // Creates a ScopedJavaSurface that is owned externally, i.e.,
-  // someone else is responsible to call Surface.release().
-  static ScopedJavaSurface AcquireExternalSurface(
-      const base::android::JavaRef<jobject>& surface);
-
   ScopedJavaSurface(const ScopedJavaSurface&) = delete;
   ScopedJavaSurface& operator=(const ScopedJavaSurface&) = delete;
 
   ~ScopedJavaSurface();
 
+  // Make a copy that does not retain ownership. Client is responsible for not
+  // using the copy after this is destroyed.
+  ScopedJavaSurface CopyRetainOwnership() const;
+
   // Checks whether the surface is an empty one.
   bool IsEmpty() const;
 
   // Checks whether this object references a valid surface.
   bool IsValid() const;
 
-  // Checks whether the surface is hardware protected so that no readback is
-  // possible.
-  bool is_protected() const { return is_protected_; }
-
   const base::android::JavaRef<jobject>& j_surface() const {
     return j_surface_;
   }
@@ -63,7 +61,6 @@
   void ReleaseSurfaceIfNeeded();
 
   bool auto_release_ = true;
-  bool is_protected_ = false;
 
   base::android::ScopedJavaGlobalRef<jobject> j_surface_;
 };
diff --git a/ui/gl/gl_image_egl.h b/ui/gl/gl_image_egl.h
index c5e6bcbc..4e4e1df 100644
--- a/ui/gl/gl_image_egl.h
+++ b/ui/gl/gl_image_egl.h
@@ -17,8 +17,6 @@
 // Abstract base class for EGL-based images.
 class GL_EXPORT GLImageEGL : public GLImage {
  public:
-  explicit GLImageEGL(const gfx::Size& size);
-
   GLImageEGL(const GLImageEGL&) = delete;
   GLImageEGL& operator=(const GLImageEGL&) = delete;
 
@@ -30,6 +28,7 @@
   void ReleaseTexImage(unsigned target) override {}
 
  protected:
+  explicit GLImageEGL(const gfx::Size& size);
   ~GLImageEGL() override;
 
   // Same semantic as specified for eglCreateImageKHR. There two main usages:
diff --git a/ui/gl/gl_image_native_pixmap.cc b/ui/gl/gl_image_native_pixmap.cc
index d492ad8f..68d0c7c 100644
--- a/ui/gl/gl_image_native_pixmap.cc
+++ b/ui/gl/gl_image_native_pixmap.cc
@@ -126,6 +126,39 @@
 
 }  // namespace
 
+scoped_refptr<GLImageNativePixmap> GLImageNativePixmap::Create(
+    const gfx::Size& size,
+    gfx::BufferFormat format,
+    scoped_refptr<gfx::NativePixmap> pixmap) {
+  return CreateForPlane(size, format, gfx::BufferPlane::DEFAULT,
+                        std::move(pixmap));
+}
+
+scoped_refptr<GLImageNativePixmap> GLImageNativePixmap::CreateForPlane(
+    const gfx::Size& size,
+    gfx::BufferFormat format,
+    gfx::BufferPlane plane,
+    scoped_refptr<gfx::NativePixmap> pixmap) {
+  auto image =
+      base::WrapRefCounted(new GLImageNativePixmap(size, format, plane));
+  if (!image->Initialize(std::move(pixmap))) {
+    return nullptr;
+  }
+  return image;
+}
+
+scoped_refptr<GLImageNativePixmap> GLImageNativePixmap::CreateFromTexture(
+    const gfx::Size& size,
+    gfx::BufferFormat format,
+    uint32_t texture_id) {
+  auto image = base::WrapRefCounted(
+      new GLImageNativePixmap(size, format, gfx::BufferPlane::DEFAULT));
+  if (!image->InitializeFromTexture(texture_id)) {
+    return nullptr;
+  }
+  return image;
+}
+
 GLImageNativePixmap::GLImageNativePixmap(const gfx::Size& size,
                                          gfx::BufferFormat format,
                                          gfx::BufferPlane plane)
@@ -238,7 +271,6 @@
                                 &attrs[0])) {
       return false;
     }
-    did_initialize_ = true;
   }
 
   pixmap_ = pixmap;
@@ -265,7 +297,6 @@
                               nullptr)) {
     return false;
   }
-  did_initialize_ = true;
   return true;
 }
 
@@ -361,12 +392,10 @@
 }
 
 bool GLImageNativePixmap::BindTexImage(unsigned target) {
-  DCHECK(did_initialize_);
   return GLImageEGL::BindTexImage(target);
 }
 
 bool GLImageNativePixmap::CopyTexImage(unsigned target) {
-  DCHECK(did_initialize_);
   if (egl_image_ != EGL_NO_IMAGE_KHR)
     return false;
 
diff --git a/ui/gl/gl_image_native_pixmap.h b/ui/gl/gl_image_native_pixmap.h
index 59bd313..c54afe9 100644
--- a/ui/gl/gl_image_native_pixmap.h
+++ b/ui/gl/gl_image_native_pixmap.h
@@ -17,14 +17,24 @@
 
 class GL_EXPORT GLImageNativePixmap : public gl::GLImageEGL {
  public:
-  GLImageNativePixmap(const gfx::Size& size,
-                      gfx::BufferFormat format,
-                      gfx::BufferPlane plane = gfx::BufferPlane::DEFAULT);
-
   // Create an EGLImage from a given NativePixmap.
-  bool Initialize(scoped_refptr<gfx::NativePixmap> pixmap);
+  static scoped_refptr<GLImageNativePixmap> Create(
+      const gfx::Size& size,
+      gfx::BufferFormat format,
+      scoped_refptr<gfx::NativePixmap> pixmap);
+
+  // Create an EGLImage from a given NativePixmap and plane.
+  static scoped_refptr<GLImageNativePixmap> CreateForPlane(
+      const gfx::Size& size,
+      gfx::BufferFormat format,
+      gfx::BufferPlane plane,
+      scoped_refptr<gfx::NativePixmap> pixmap);
   // Create an EGLImage from a given GL texture.
-  bool InitializeFromTexture(uint32_t texture_id);
+  static scoped_refptr<GLImageNativePixmap> CreateFromTexture(
+      const gfx::Size& size,
+      gfx::BufferFormat format,
+      uint32_t texture_id);
+
   // Export the wrapped EGLImage to dmabuf fds.
   gfx::NativePixmapHandle ExportHandle();
 
@@ -45,11 +55,18 @@
   ~GLImageNativePixmap() override;
 
  private:
+  GLImageNativePixmap(const gfx::Size& size,
+                      gfx::BufferFormat format,
+                      gfx::BufferPlane plane);
+  // Create an EGLImage from a given NativePixmap.
+  bool Initialize(scoped_refptr<gfx::NativePixmap> pixmap);
+  // Create an EGLImage from a given GL texture.
+  bool InitializeFromTexture(uint32_t texture_id);
+
   gfx::BufferFormat format_;
   scoped_refptr<gfx::NativePixmap> pixmap_;
   gfx::BufferPlane plane_;
   bool has_image_dma_buf_export_;
-  bool did_initialize_;
 };
 
 }  // namespace gl
diff --git a/ui/gl/gl_image_native_pixmap_unittest.cc b/ui/gl/gl_image_native_pixmap_unittest.cc
index dc65e52..aafe748 100644
--- a/ui/gl/gl_image_native_pixmap_unittest.cc
+++ b/ui/gl/gl_image_native_pixmap_unittest.cc
@@ -59,8 +59,9 @@
     glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, size.width(), size.height(), 0,
                  GL_RGBA, GL_UNSIGNED_BYTE, pixels.get());
 
-    auto image = base::MakeRefCounted<gl::GLImageNativePixmap>(size, format);
-    EXPECT_TRUE(image->InitializeFromTexture(texture_id));
+    auto image =
+        gl::GLImageNativePixmap::CreateFromTexture(size, format, texture_id);
+    EXPECT_TRUE(image);
 
     glDeleteTextures(1, &texture_id);
     return image;
diff --git a/ui/ozone/common/native_pixmap_egl_binding.cc b/ui/ozone/common/native_pixmap_egl_binding.cc
index efb2090..d4c21a6 100644
--- a/ui/ozone/common/native_pixmap_egl_binding.cc
+++ b/ui/ozone/common/native_pixmap_egl_binding.cc
@@ -22,15 +22,16 @@
     const gfx::ColorSpace& color_space,
     GLenum target,
     GLuint texture_id) {
-  auto gl_image = base::MakeRefCounted<gl::GLImageNativePixmap>(
-      plane_size, plane_format, plane);
-  if (color_space.IsValid())
-    gl_image->SetColorSpace(color_space);
-  if (!gl_image->Initialize(std::move(pixmap))) {
+  auto gl_image = gl::GLImageNativePixmap::CreateForPlane(
+      plane_size, plane_format, plane, std::move(pixmap));
+  if (!gl_image) {
     LOG(ERROR) << "Unable to initialize GL image from pixmap";
     return nullptr;
   }
 
+  if (color_space.IsValid())
+    gl_image->SetColorSpace(color_space);
+
   auto binding = std::make_unique<NativePixmapEGLBinding>();
   if (!binding->BindTexture(std::move(gl_image), target, texture_id)) {
     return nullptr;
diff --git a/ui/ozone/demo/skia/skia_surfaceless_gl_renderer.cc b/ui/ozone/demo/skia/skia_surfaceless_gl_renderer.cc
index 8e1b4430b..6258cd0 100644
--- a/ui/ozone/demo/skia/skia_surfaceless_gl_renderer.cc
+++ b/ui/ozone/demo/skia/skia_surfaceless_gl_renderer.cc
@@ -122,12 +122,11 @@
           ->GetSurfaceFactoryOzone()
           ->CreateNativePixmap(widget, nullptr, size, format,
                                gfx::BufferUsage::SCANOUT);
-  auto image = base::MakeRefCounted<gl::GLImageNativePixmap>(size, format);
-  if (!image->Initialize(std::move(pixmap))) {
+  image_ = gl::GLImageNativePixmap::Create(size, format, std::move(pixmap));
+  if (!image_) {
     LOG(ERROR) << "Failed to create GLImage";
     return false;
   }
-  image_ = image;
 
   glBindTexture(GL_TEXTURE_2D, gl_tex_);
   image_->BindTexImage(GL_TEXTURE_2D);
diff --git a/ui/ozone/demo/surfaceless_gl_renderer.cc b/ui/ozone/demo/surfaceless_gl_renderer.cc
index f7baedf..d564ef0 100644
--- a/ui/ozone/demo/surfaceless_gl_renderer.cc
+++ b/ui/ozone/demo/surfaceless_gl_renderer.cc
@@ -93,12 +93,11 @@
           ->GetSurfaceFactoryOzone()
           ->CreateNativePixmap(widget, nullptr, size, format,
                                gfx::BufferUsage::SCANOUT);
-  auto image = base::MakeRefCounted<gl::GLImageNativePixmap>(size, format);
-  if (!image->Initialize(std::move(pixmap))) {
+  image_ = gl::GLImageNativePixmap::Create(size, format, std::move(pixmap));
+  if (!image_) {
     LOG(ERROR) << "Failed to create GLImage";
     return false;
   }
-  image_ = image;
 
   glBindFramebufferEXT(GL_FRAMEBUFFER, gl_fb_);
   glBindTexture(GL_TEXTURE_2D, gl_tex_);
diff --git a/ui/ozone/gl/gl_image_ozone_native_pixmap_unittest.cc b/ui/ozone/gl/gl_image_ozone_native_pixmap_unittest.cc
index 061e5507..425ea8fd 100644
--- a/ui/ozone/gl/gl_image_ozone_native_pixmap_unittest.cc
+++ b/ui/ozone/gl/gl_image_ozone_native_pixmap_unittest.cc
@@ -60,8 +60,9 @@
       client_pixmap->Unmap();
     }
 
-    auto image = base::MakeRefCounted<gl::GLImageNativePixmap>(size, format);
-    EXPECT_TRUE(image->Initialize(pixmap.get()));
+    auto image =
+        gl::GLImageNativePixmap::Create(size, format, std::move(pixmap));
+    EXPECT_TRUE(image);
     return image;
   }
 
diff --git a/ui/ozone/platform/wayland/host/wayland_data_drag_controller_unittest.cc b/ui/ozone/platform/wayland/host/wayland_data_drag_controller_unittest.cc
index 9dfb392f..38934c5e 100644
--- a/ui/ozone/platform/wayland/host/wayland_data_drag_controller_unittest.cc
+++ b/ui/ozone/platform/wayland/host/wayland_data_drag_controller_unittest.cc
@@ -128,8 +128,10 @@
   void OnDragDrop(std::unique_ptr<OSExchangeData> data,
                   int modifiers) override {
     MockOnDragDrop();
-    on_drop_closure_.Run();
-    on_drop_closure_.Reset();
+    if (on_drop_closure_) {
+      on_drop_closure_.Run();
+      on_drop_closure_.Reset();
+    }
   }
 
   int OnDragMotion(const gfx::PointF& point,
@@ -501,9 +503,21 @@
   SendMotionEvent(top_left);
 }
 
+// Emulating an incoming DnD session, ensures that data drag controller
+// fetches all the data for the mime-types offered by the source client.
 TEST_P(WaylandDataDragControllerTest, DropSeveralMimeTypes) {
-  EXPECT_CALL(*drop_handler_, MockOnDragEnter()).Times(1);
   const uint32_t surface_id = window_->root_surface()->get_surface_id();
+
+  // As data for each offered mime-type is asynchronously read (eg: using
+  // wl_display.sync callbacks, etc), a nested run loop is used to ensure
+  // it is reliably done. Furthermore, WmDropHandler::OnDragEnter() is expected
+  // to be called only once the data is fully fetched, so it's used here as
+  // condition to quit the loop and verify the expectations, otherwise some
+  // flakiness may be observed, see https://crbug.com/1395127.
+  base::RunLoop loop;
+  EXPECT_CALL(*drop_handler_, MockOnDragEnter()).WillOnce([&loop]() {
+    loop.Quit();
+  });
   PostToServerAndWait([surface_id](wl::TestWaylandServerThread* server) {
     auto* data_offer =
         server->data_device_manager()->data_device()->CreateAndSendDataOffer();
@@ -522,15 +536,13 @@
         1002, surface->resource(), wl_fixed_from_int(entered_point.x()),
         wl_fixed_from_int(entered_point.y()), data_offer);
   });
+  loop.Run();
+  Mock::VerifyAndClearExpectations(drop_handler_.get());
 
   EXPECT_CALL(*drop_handler_, MockOnDragDrop()).Times(1);
-  base::RunLoop loop;
-  drop_handler_->SetOnDropClosure(loop.QuitClosure());
   PostToServerAndWait([](wl::TestWaylandServerThread* server) {
     server->data_device_manager()->data_device()->OnDrop();
   });
-
-  loop.Run();
   Mock::VerifyAndClearExpectations(drop_handler_.get());
 
   ASSERT_NE(drop_handler_->dropped_data(), nullptr);
diff --git a/ui/strings/ui_strings.grd b/ui/strings/ui_strings.grd
index bc48ab46..2b0181d 100644
--- a/ui/strings/ui_strings.grd
+++ b/ui/strings/ui_strings.grd
@@ -605,7 +605,7 @@
           Emoji &amp;&amp; Symbols
         </message>
       </if>
-      <if expr="chromeos_ash">
+      <if expr="is_chromeos">
         <message name="IDS_APP_SHOW_CLIPBOARD_HISTORY" desc="The context menu item to show the clipboard history menu.">
           Clipboard
         </message>
diff --git a/ui/views/BUILD.gn b/ui/views/BUILD.gn
index c92b020..e683ccc1 100644
--- a/ui/views/BUILD.gn
+++ b/ui/views/BUILD.gn
@@ -596,11 +596,11 @@
   }
 
   if (is_chromeos) {
-    sources += [ "controls/menu/menu_config_chromeos.cc" ]
-    if (!is_chromeos_lacros) {
-      public += [ "controls/views_text_services_context_menu_chromeos.h" ]
-      sources += [ "controls/views_text_services_context_menu_chromeos.cc" ]
-    }
+    public += [ "controls/views_text_services_context_menu_chromeos.h" ]
+    sources += [
+      "controls/menu/menu_config_chromeos.cc",
+      "controls/views_text_services_context_menu_chromeos.cc",
+    ]
   }
 
   if (is_mac) {
diff --git a/ui/views/cocoa/immersive_mode_controller_unittest.mm b/ui/views/cocoa/immersive_mode_controller_unittest.mm
index 51aa430a..9579af9 100644
--- a/ui/views/cocoa/immersive_mode_controller_unittest.mm
+++ b/ui/views/cocoa/immersive_mode_controller_unittest.mm
@@ -236,11 +236,16 @@
   [fullscreen_window.get().contentView addSubview:titlebar_container_view];
   [fullscreen_window orderBack:nil];
 
+  auto immersive_mode_controller = std::make_unique<ImmersiveModeController>(
+      browser(), overlay(), base::DoNothing());
+  base::WeakPtrFactory<ImmersiveModeController> weak_ptr_factory(
+      immersive_mode_controller.get());
+
   // Create a titlebar observer. This is the class under test.
   base::scoped_nsobject<ImmersiveModeTitlebarObserver> titlebar_observer(
       [[ImmersiveModeTitlebarObserver alloc]
-          initWithOverlayWindow:overlay()
-                    overlayView:overlay_view]);
+          initWithController:weak_ptr_factory.GetWeakPtr()
+                 overlayView:overlay_view]);
 
   // Observer the fake fake titlebar container view.
   [titlebar_container_view addObserver:titlebar_observer
diff --git a/ui/views/controls/views_text_services_context_menu_base.cc b/ui/views/controls/views_text_services_context_menu_base.cc
index 2ec8faa..57d33de 100644
--- a/ui/views/controls/views_text_services_context_menu_base.cc
+++ b/ui/views/controls/views_text_services_context_menu_base.cc
@@ -88,7 +88,7 @@
   return command_id == IDS_CONTENT_CONTEXT_EMOJI;
 }
 
-#if !BUILDFLAG(IS_MAC) && !BUILDFLAG(IS_CHROMEOS_ASH)
+#if !BUILDFLAG(IS_MAC) && !BUILDFLAG(IS_CHROMEOS)
 // static
 std::unique_ptr<ViewsTextServicesContextMenu>
 ViewsTextServicesContextMenu::Create(ui::SimpleMenuModel* menu,
diff --git a/ui/views/controls/views_text_services_context_menu_base.h b/ui/views/controls/views_text_services_context_menu_base.h
index 1bd3aa3..7bdea6d 100644
--- a/ui/views/controls/views_text_services_context_menu_base.h
+++ b/ui/views/controls/views_text_services_context_menu_base.h
@@ -35,7 +35,7 @@
   bool SupportsCommand(int command_id) const override;
 
  protected:
-#if BUILDFLAG(IS_MAC) || BUILDFLAG(IS_CHROMEOS_ASH)
+#if BUILDFLAG(IS_MAC) || BUILDFLAG(IS_CHROMEOS)
   Textfield* client() { return client_; }
   const Textfield* client() const { return client_; }
 #endif
diff --git a/ui/views/controls/views_text_services_context_menu_chromeos.cc b/ui/views/controls/views_text_services_context_menu_chromeos.cc
index 63de8cf..86c09714 100644
--- a/ui/views/controls/views_text_services_context_menu_chromeos.cc
+++ b/ui/views/controls/views_text_services_context_menu_chromeos.cc
@@ -34,10 +34,11 @@
 
   // In unit tests, `impl_factory` may not be set. Use
   // `ViewTextServicesContextMenuBase` in that case.
-  if (impl_factory)
+  if (impl_factory) {
     impl_ = impl_factory.Run(menu, client);
-  else
+  } else {
     impl_ = std::make_unique<ViewsTextServicesContextMenuBase>(menu, client);
+  }
 }
 
 ViewsTextServicesContextMenuChromeos::~ViewsTextServicesContextMenuChromeos() =
@@ -66,7 +67,7 @@
 
 bool ViewsTextServicesContextMenuChromeos::SupportsCommand(
     int command_id) const {
-  return impl_->IsCommandIdEnabled(command_id);
+  return impl_->SupportsCommand(command_id);
 }
 
 // static
diff --git a/ui/views/controls/views_text_services_context_menu_chromeos.h b/ui/views/controls/views_text_services_context_menu_chromeos.h
index 9cdd0de..f583a26 100644
--- a/ui/views/controls/views_text_services_context_menu_chromeos.h
+++ b/ui/views/controls/views_text_services_context_menu_chromeos.h
@@ -12,8 +12,9 @@
 
 namespace views {
 
-// This class is used to add and handle text service items in the text context
-// menu under the CrOS environment.
+// This class is used to add and handle text service items in ChromeOS native UI
+// textfield context menus. The implementation is specific to the platform (Ash
+// or Lacros) where the textfield lives.
 class VIEWS_EXPORT ViewsTextServicesContextMenuChromeos
     : public ViewsTextServicesContextMenu {
  public:
@@ -40,9 +41,9 @@
   bool SupportsCommand(int command_id) const override;
 
  private:
-  // CrOS functionality must be provided by the embedder, so requests are
-  // forwarded to this concrete object, whose construction can be controlled by
-  // `SetImplFactory()`.
+  // ChromeOS functionality is provided by a platform-specific implementation.
+  // Function calls are forwarded to this instance, whose construction is
+  // controlled by `SetImplFactory()`.
   std::unique_ptr<ViewsTextServicesContextMenu> impl_;
 };
 
diff --git a/ui/webui/resources/cr_components/history_clusters/url_visit.html b/ui/webui/resources/cr_components/history_clusters/url_visit.html
index d552dbd..86e5985 100644
--- a/ui/webui/resources/cr_components/history_clusters/url_visit.html
+++ b/ui/webui/resources/cr_components/history_clusters/url_visit.html
@@ -47,7 +47,7 @@
     margin-inline-end: var(--cluster-padding-horizontal);
     min-width: 0;
     outline: none;
-    padding-inline-end: 2px; /* So focus outline does not intersect text */
+    padding-inline: 2px; /* So focus outline does not intersect text */
   }
 
   :host(:hover) #link-container {
diff --git a/ui/webui/resources/js/metrics_reporter/BUILD.gn b/ui/webui/resources/js/metrics_reporter/BUILD.gn
index 6fcb00b..487a50b 100644
--- a/ui/webui/resources/js/metrics_reporter/BUILD.gn
+++ b/ui/webui/resources/js/metrics_reporter/BUILD.gn
@@ -10,6 +10,7 @@
   sources = [ "metrics_reporter.mojom" ]
   webui_module_path = "chrome://resources/js/metrics_reporter"
   public_deps = [ "//mojo/public/mojom/base" ]
+  use_typescript_sources = true
 }
 
 # Output folder used to hold ts_library() output.
@@ -20,24 +21,24 @@
   root_dir = target_gen_dir
   out_dir = preprocess_folder
   composite = true
-  tsconfig_base = "tsconfig_base.json"
   in_files = [
     "metrics_reporter.ts",
     "browser_proxy.ts",
-    "metrics_reporter.mojom-webui.js",
+    "metrics_reporter.mojom-webui.ts",
   ]
   definitions = [ "//tools/typescript/definitions/chrome_timeticks.d.ts" ]
   deps = [
     "//ui/webui/resources:library",
     "//ui/webui/resources/mojo:library",
   ]
-  extra_deps = [ ":copy_src_and_mojom" ]
+  extra_deps = [
+    ":copy_src",
+    ":mojo_bindings_ts__generator",
+  ]
 }
 
-copy("copy_src_and_mojom") {
-  deps = [ ":mojo_bindings_js__generator" ]
+copy("copy_src") {
   sources = [
-    "$root_gen_dir/mojom-webui/ui/webui/resources/js/metrics_reporter/metrics_reporter.mojom-webui.js",
     "browser_proxy.ts",
     "metrics_reporter.ts",
   ]
diff --git a/ui/webui/resources/js/metrics_reporter/tsconfig_base.json b/ui/webui/resources/js/metrics_reporter/tsconfig_base.json
deleted file mode 100644
index 5502828..0000000
--- a/ui/webui/resources/js/metrics_reporter/tsconfig_base.json
+++ /dev/null
@@ -1,6 +0,0 @@
-{
-  "extends": "../../../../../tools/typescript/tsconfig_base.json",
-  "compilerOptions": {
-    "allowJs": true
-  }
-}